1use std::{
2 borrow::{Borrow, Cow},
3 collections::HashMap,
4 hash::{BuildHasher, Hash, Hasher},
5 mem,
6};
7
8use tracing::error;
9
10use crate::{
11 path::PathItem,
12 regex_set::{escape, Regex, RegexSet},
13 IntoPatterns, Patterns, Resource, ResourcePath,
14};
15
16const MAX_DYNAMIC_SEGMENTS: usize = 16;
17
18const REGEX_FLAGS: &str = "(?s-m)";
22
23#[derive(Clone, Debug)]
212pub struct ResourceDef {
213 id: u16,
214
215 name: Option<String>,
217
218 patterns: Patterns,
220
221 is_prefix: bool,
222
223 pat_type: PatternType,
225
226 segments: Vec<PatternSegment>,
228}
229
230#[derive(Debug, Clone, PartialEq)]
231enum PatternSegment {
232 Const(String),
234
235 Var(String),
237}
238
239#[derive(Debug, Clone)]
240#[allow(clippy::large_enum_variant)]
241enum PatternType {
242 Static(String),
244
245 Dynamic(Regex, Vec<&'static str>),
247
248 DynamicSet(RegexSet, Vec<(Regex, Vec<&'static str>)>),
250}
251
252impl ResourceDef {
253 pub fn new<T: IntoPatterns>(paths: T) -> Self {
277 Self::construct(paths, false)
278 }
279
280 pub fn prefix<T: IntoPatterns>(paths: T) -> Self {
304 ResourceDef::construct(paths, true)
305 }
306
307 pub fn root_prefix(path: &str) -> Self {
328 ResourceDef::prefix(insert_slash(path).into_owned())
329 }
330
331 pub fn id(&self) -> u16 {
345 self.id
346 }
347
348 pub fn set_id(&mut self, id: u16) {
358 self.id = id;
359 }
360
361 pub fn name(&self) -> Option<&str> {
372 self.name.as_deref()
373 }
374
375 pub fn set_name(&mut self, name: impl Into<String>) {
388 let name = name.into();
389
390 assert!(!name.is_empty(), "resource name should not be empty");
391
392 self.name = Some(name)
393 }
394
395 pub fn is_prefix(&self) -> bool {
404 self.is_prefix
405 }
406
407 pub fn pattern(&self) -> Option<&str> {
422 match &self.patterns {
423 Patterns::Single(pattern) => Some(pattern.as_str()),
424 Patterns::List(patterns) => patterns.first().map(AsRef::as_ref),
425 }
426 }
427
428 pub fn pattern_iter(&self) -> impl Iterator<Item = &str> {
444 struct PatternIter<'a> {
445 patterns: &'a Patterns,
446 list_idx: usize,
447 done: bool,
448 }
449
450 impl<'a> Iterator for PatternIter<'a> {
451 type Item = &'a str;
452
453 fn next(&mut self) -> Option<Self::Item> {
454 match &self.patterns {
455 Patterns::Single(pattern) => {
456 if self.done {
457 return None;
458 }
459
460 self.done = true;
461 Some(pattern.as_str())
462 }
463 Patterns::List(patterns) if patterns.is_empty() => None,
464 Patterns::List(patterns) => match patterns.get(self.list_idx) {
465 Some(pattern) => {
466 self.list_idx += 1;
467 Some(pattern.as_str())
468 }
469 None => {
470 self.done = true;
472 None
473 }
474 },
475 }
476 }
477
478 fn size_hint(&self) -> (usize, Option<usize>) {
479 match &self.patterns {
480 Patterns::Single(_) => (1, Some(1)),
481 Patterns::List(patterns) => (patterns.len(), Some(patterns.len())),
482 }
483 }
484 }
485
486 PatternIter {
487 patterns: &self.patterns,
488 list_idx: 0,
489 done: false,
490 }
491 }
492
493 pub fn join(&self, other: &ResourceDef) -> ResourceDef {
504 let patterns = self
505 .pattern_iter()
506 .flat_map(move |this| other.pattern_iter().map(move |other| (this, other)))
507 .map(|(this, other)| {
508 let mut pattern = String::with_capacity(this.len() + other.len());
509 pattern.push_str(this);
510 pattern.push_str(other);
511 pattern
512 })
513 .collect::<Vec<_>>();
514
515 match patterns.len() {
516 1 => ResourceDef::construct(&patterns[0], other.is_prefix()),
517 _ => ResourceDef::construct(patterns, other.is_prefix()),
518 }
519 }
520
521 #[inline]
555 pub fn is_match(&self, path: &str) -> bool {
556 match &self.pat_type {
561 PatternType::Static(pattern) => self.static_match(pattern, path).is_some(),
562 PatternType::Dynamic(re, _) => re.is_match(path),
563 PatternType::DynamicSet(re, _) => re.is_match(path),
564 }
565 }
566
567 pub fn find_match(&self, path: &str) -> Option<usize> {
603 match &self.pat_type {
604 PatternType::Static(pattern) => self.static_match(pattern, path),
605
606 PatternType::Dynamic(re, _) => Some(re.captures(path)?[1].len()),
607
608 PatternType::DynamicSet(re, params) => {
609 let idx = re.first_match_idx(path)?;
610 let (ref pattern, _) = params[idx];
611 Some(pattern.captures(path)?[1].len())
612 }
613 }
614 }
615
616 pub fn capture_match_info<R: Resource>(&self, resource: &mut R) -> bool {
637 self.capture_match_info_fn(resource, |_| true)
638 }
639
640 pub fn capture_match_info_fn<R, F>(&self, resource: &mut R, check_fn: F) -> bool
678 where
679 R: Resource,
680 F: FnOnce(&R) -> bool,
681 {
682 let mut segments = <[PathItem; MAX_DYNAMIC_SEGMENTS]>::default();
683 let path = resource.resource_path();
684 let path_str = path.unprocessed();
685
686 let (matched_len, matched_vars) = match &self.pat_type {
687 PatternType::Static(pattern) => match self.static_match(pattern, path_str) {
688 Some(len) => (len, None),
689 None => return false,
690 },
691
692 PatternType::Dynamic(re, names) => {
693 let captures = match re.captures(path.unprocessed()) {
694 Some(captures) => captures,
695 _ => return false,
696 };
697
698 for (no, name) in names.iter().enumerate() {
699 if let Some(m) = captures.name(name) {
700 segments[no] = PathItem::Segment(m.start() as u16, m.end() as u16);
701 } else {
702 error!("Dynamic path match but not all segments found: {}", name);
703 return false;
704 }
705 }
706
707 (captures[1].len(), Some(names))
708 }
709
710 PatternType::DynamicSet(re, params) => {
711 let path = path.unprocessed();
712 let (pattern, names) = match re.first_match_idx(path) {
713 Some(idx) => ¶ms[idx],
714 _ => return false,
715 };
716
717 let captures = match pattern.captures(path.path()) {
718 Some(captures) => captures,
719 _ => return false,
720 };
721
722 for (no, name) in names.iter().enumerate() {
723 if let Some(m) = captures.name(name) {
724 segments[no] = PathItem::Segment(m.start() as u16, m.end() as u16);
725 } else {
726 error!("Dynamic path match but not all segments found: {}", name);
727 return false;
728 }
729 }
730
731 (captures[1].len(), Some(names))
732 }
733 };
734
735 if !check_fn(resource) {
736 return false;
737 }
738
739 let path = resource.resource_path();
741
742 if let Some(vars) = matched_vars {
743 for i in 0..vars.len() {
744 path.add(vars[i], mem::take(&mut segments[i]));
745 }
746 }
747
748 path.skip(matched_len as u16);
749
750 true
751 }
752
753 fn build_resource_path<F, I>(&self, path: &mut String, mut vars: F) -> bool
755 where
756 F: FnMut(&str) -> Option<I>,
757 I: AsRef<str>,
758 {
759 for segment in &self.segments {
760 match segment {
761 PatternSegment::Const(val) => path.push_str(val),
762 PatternSegment::Var(name) => match vars(name) {
763 Some(val) => path.push_str(val.as_ref()),
764 _ => return false,
765 },
766 }
767 }
768
769 true
770 }
771
772 pub fn resource_path_from_iter<I>(&self, path: &mut String, values: I) -> bool
789 where
790 I: IntoIterator,
791 I::Item: AsRef<str>,
792 {
793 let mut iter = values.into_iter();
794 self.build_resource_path(path, |_| iter.next())
795 }
796
797 pub fn resource_path_from_map<K, V, S>(
819 &self,
820 path: &mut String,
821 values: &HashMap<K, V, S>,
822 ) -> bool
823 where
824 K: Borrow<str> + Eq + Hash,
825 V: AsRef<str>,
826 S: BuildHasher,
827 {
828 self.build_resource_path(path, |name| values.get(name))
829 }
830
831 fn static_match(&self, pattern: &str, path: &str) -> Option<usize> {
833 let rem = path.strip_prefix(pattern)?;
834
835 match self.is_prefix {
836 false if rem.is_empty() => Some(pattern.len()),
838
839 true if rem.is_empty() || rem.starts_with('/') => Some(pattern.len()),
841
842 _ => None,
844 }
845 }
846
847 fn construct<T: IntoPatterns>(paths: T, is_prefix: bool) -> Self {
848 let patterns = paths.patterns();
849
850 let (pat_type, segments) = match &patterns {
851 Patterns::Single(pattern) => ResourceDef::parse(pattern, is_prefix, false),
852
853 Patterns::List(patterns) if patterns.is_empty() => (
856 PatternType::DynamicSet(RegexSet::empty(), Vec::new()),
857 Vec::new(),
858 ),
859
860 Patterns::List(patterns) => {
861 let mut re_set = Vec::with_capacity(patterns.len());
862 let mut pattern_data = Vec::new();
863 let mut segments = None;
864
865 for pattern in patterns {
866 match ResourceDef::parse(pattern, is_prefix, true) {
867 (PatternType::Dynamic(re, names), segs) => {
868 re_set.push(re.as_str().to_owned());
869 pattern_data.push((re, names));
870 segments.get_or_insert(segs);
871 }
872 _ => unreachable!(),
873 }
874 }
875
876 let pattern_re_set = RegexSet::new(re_set);
877 let segments = segments.unwrap_or_default();
878
879 (
880 PatternType::DynamicSet(pattern_re_set, pattern_data),
881 segments,
882 )
883 }
884 };
885
886 ResourceDef {
887 id: 0,
888 name: None,
889 patterns,
890 is_prefix,
891 pat_type,
892 segments,
893 }
894 }
895
896 fn parse_param(pattern: &str) -> (PatternSegment, String, &str, bool) {
907 const DEFAULT_PATTERN: &str = "[^/]+";
908 const DEFAULT_PATTERN_TAIL: &str = ".*";
909
910 let mut params_nesting = 0usize;
911 let close_idx = pattern
912 .find(|c| match c {
913 '{' => {
914 params_nesting += 1;
915 false
916 }
917 '}' => {
918 params_nesting -= 1;
919 params_nesting == 0
920 }
921 _ => false,
922 })
923 .unwrap_or_else(|| {
924 panic!(
925 r#"pattern "{}" contains malformed dynamic segment"#,
926 pattern
927 )
928 });
929
930 let (mut param, mut unprocessed) = pattern.split_at(close_idx + 1);
931
932 param = ¶m[1..param.len() - 1];
934
935 let tail = unprocessed == "*";
936
937 let (name, pattern) = match param.find(':') {
938 Some(idx) => {
939 assert!(!tail, "custom regex is not supported for tail match");
940
941 let (name, pattern) = param.split_at(idx);
942 (name, &pattern[1..])
943 }
944 None => (
945 param,
946 if tail {
947 unprocessed = &unprocessed[1..];
948 DEFAULT_PATTERN_TAIL
949 } else {
950 DEFAULT_PATTERN
951 },
952 ),
953 };
954
955 let segment = PatternSegment::Var(name.to_string());
956 let regex = format!(r"(?P<{}>{})", &name, &pattern);
957
958 (segment, regex, unprocessed, tail)
959 }
960
961 fn parse(
972 pattern: &str,
973 is_prefix: bool,
974 force_dynamic: bool,
975 ) -> (PatternType, Vec<PatternSegment>) {
976 if !force_dynamic && pattern.find('{').is_none() && !pattern.ends_with('*') {
977 return (
979 PatternType::Static(pattern.to_owned()),
980 vec![PatternSegment::Const(pattern.to_owned())],
981 );
982 }
983
984 let mut unprocessed = pattern;
985 let mut segments = Vec::new();
986 let mut re = format!("{}^", REGEX_FLAGS);
987 let mut dyn_segment_count = 0;
988 let mut has_tail_segment = false;
989
990 while let Some(idx) = unprocessed.find('{') {
991 let (prefix, rem) = unprocessed.split_at(idx);
992
993 segments.push(PatternSegment::Const(prefix.to_owned()));
994 re.push_str(&escape(prefix));
995
996 let (param_pattern, re_part, rem, tail) = Self::parse_param(rem);
997
998 if tail {
999 has_tail_segment = true;
1000 }
1001
1002 segments.push(param_pattern);
1003 re.push_str(&re_part);
1004
1005 unprocessed = rem;
1006 dyn_segment_count += 1;
1007 }
1008
1009 if is_prefix && has_tail_segment {
1010 #[cfg(not(test))]
1013 tracing::warn!(
1014 "Prefix resources should not have tail segments. \
1015 Use `ResourceDef::new` constructor. \
1016 This may become a panic in the future."
1017 );
1018
1019 #[cfg(test)]
1021 panic!("prefix resource definitions should not have tail segments");
1022 }
1023
1024 #[allow(clippy::literal_string_with_formatting_args)]
1025 if unprocessed.ends_with('*') {
1026 #[cfg(not(test))]
1029 tracing::warn!(
1030 "Tail segments must have names. \
1031 Consider `.../{{tail}}*`. \
1032 This may become a panic in the future."
1033 );
1034
1035 #[cfg(test)]
1037 panic!("tail segments must have names");
1038 } else if !has_tail_segment && !unprocessed.is_empty() {
1039 segments.push(PatternSegment::Const(unprocessed.to_owned()));
1042 re.push_str(&escape(unprocessed));
1043 }
1044
1045 assert!(
1046 dyn_segment_count <= MAX_DYNAMIC_SEGMENTS,
1047 "Only {} dynamic segments are allowed, provided: {}",
1048 MAX_DYNAMIC_SEGMENTS,
1049 dyn_segment_count
1050 );
1051
1052 let mut re = format!("({})", re);
1054
1055 if !has_tail_segment {
1057 if is_prefix {
1058 re.push_str(r"(/|$)");
1059 } else {
1060 re.push('$');
1061 }
1062 }
1063
1064 let re = match Regex::new(&re) {
1065 Ok(re) => re,
1066 Err(err) => panic!("Wrong path pattern: \"{}\" {}", pattern, err),
1067 };
1068
1069 let names = re
1074 .capture_names()
1075 .filter_map(|name| name.map(|name| Box::leak(Box::new(name.to_owned())).as_str()))
1076 .collect();
1077
1078 (PatternType::Dynamic(re, names), segments)
1079 }
1080}
1081
1082impl Eq for ResourceDef {}
1083
1084impl PartialEq for ResourceDef {
1085 fn eq(&self, other: &ResourceDef) -> bool {
1086 self.patterns == other.patterns && self.is_prefix == other.is_prefix
1087 }
1088}
1089
1090impl Hash for ResourceDef {
1091 fn hash<H: Hasher>(&self, state: &mut H) {
1092 self.patterns.hash(state);
1093 }
1094}
1095
1096impl<'a> From<&'a str> for ResourceDef {
1097 fn from(path: &'a str) -> ResourceDef {
1098 ResourceDef::new(path)
1099 }
1100}
1101
1102impl From<String> for ResourceDef {
1103 fn from(path: String) -> ResourceDef {
1104 ResourceDef::new(path)
1105 }
1106}
1107
1108pub(crate) fn insert_slash(path: &str) -> Cow<'_, str> {
1109 if !path.is_empty() && !path.starts_with('/') {
1110 let mut new_path = String::with_capacity(path.len() + 1);
1111 new_path.push('/');
1112 new_path.push_str(path);
1113 Cow::Owned(new_path)
1114 } else {
1115 Cow::Borrowed(path)
1116 }
1117}
1118
1119#[cfg(test)]
1120mod tests {
1121 use super::*;
1122 use crate::Path;
1123
1124 #[test]
1125 fn equivalence() {
1126 assert_eq!(
1127 ResourceDef::root_prefix("/root"),
1128 ResourceDef::prefix("/root")
1129 );
1130 assert_eq!(
1131 ResourceDef::root_prefix("root"),
1132 ResourceDef::prefix("/root")
1133 );
1134 assert_eq!(
1135 ResourceDef::root_prefix("/{id}"),
1136 ResourceDef::prefix("/{id}")
1137 );
1138 assert_eq!(
1139 ResourceDef::root_prefix("{id}"),
1140 ResourceDef::prefix("/{id}")
1141 );
1142
1143 assert_eq!(ResourceDef::new("/"), ResourceDef::new(["/"]));
1144 assert_eq!(ResourceDef::new("/"), ResourceDef::new(vec!["/"]));
1145
1146 assert_ne!(ResourceDef::new(""), ResourceDef::prefix(""));
1147 assert_ne!(ResourceDef::new("/"), ResourceDef::prefix("/"));
1148 assert_ne!(ResourceDef::new("/{id}"), ResourceDef::prefix("/{id}"));
1149 }
1150
1151 #[test]
1152 fn parse_static() {
1153 let re = ResourceDef::new("");
1154
1155 assert!(!re.is_prefix());
1156
1157 assert!(re.is_match(""));
1158 assert!(!re.is_match("/"));
1159 assert_eq!(re.find_match(""), Some(0));
1160 assert_eq!(re.find_match("/"), None);
1161
1162 let re = ResourceDef::new("/");
1163 assert!(re.is_match("/"));
1164 assert!(!re.is_match(""));
1165 assert!(!re.is_match("/foo"));
1166
1167 let re = ResourceDef::new("/name");
1168 assert!(re.is_match("/name"));
1169 assert!(!re.is_match("/name1"));
1170 assert!(!re.is_match("/name/"));
1171 assert!(!re.is_match("/name~"));
1172
1173 let mut path = Path::new("/name");
1174 assert!(re.capture_match_info(&mut path));
1175 assert_eq!(path.unprocessed(), "");
1176
1177 assert_eq!(re.find_match("/name"), Some(5));
1178 assert_eq!(re.find_match("/name1"), None);
1179 assert_eq!(re.find_match("/name/"), None);
1180 assert_eq!(re.find_match("/name~"), None);
1181
1182 let re = ResourceDef::new("/name/");
1183 assert!(re.is_match("/name/"));
1184 assert!(!re.is_match("/name"));
1185 assert!(!re.is_match("/name/gs"));
1186
1187 let re = ResourceDef::new("/user/profile");
1188 assert!(re.is_match("/user/profile"));
1189 assert!(!re.is_match("/user/profile/profile"));
1190
1191 let mut path = Path::new("/user/profile");
1192 assert!(re.capture_match_info(&mut path));
1193 assert_eq!(path.unprocessed(), "");
1194 }
1195
1196 #[test]
1197 fn parse_param() {
1198 let re = ResourceDef::new("/user/{id}");
1199 assert!(re.is_match("/user/profile"));
1200 assert!(re.is_match("/user/2345"));
1201 assert!(!re.is_match("/user/2345/"));
1202 assert!(!re.is_match("/user/2345/sdg"));
1203
1204 let mut path = Path::new("/user/profile");
1205 assert!(re.capture_match_info(&mut path));
1206 assert_eq!(path.get("id").unwrap(), "profile");
1207 assert_eq!(path.unprocessed(), "");
1208
1209 let mut path = Path::new("/user/1245125");
1210 assert!(re.capture_match_info(&mut path));
1211 assert_eq!(path.get("id").unwrap(), "1245125");
1212 assert_eq!(path.unprocessed(), "");
1213
1214 let re = ResourceDef::new("/v{version}/resource/{id}");
1215 assert!(re.is_match("/v1/resource/320120"));
1216 assert!(!re.is_match("/v/resource/1"));
1217 assert!(!re.is_match("/resource"));
1218
1219 let mut path = Path::new("/v151/resource/adage32");
1220 assert!(re.capture_match_info(&mut path));
1221 assert_eq!(path.get("version").unwrap(), "151");
1222 assert_eq!(path.get("id").unwrap(), "adage32");
1223 assert_eq!(path.unprocessed(), "");
1224
1225 let re = ResourceDef::new("/{id:[[:digit:]]{6}}");
1226 assert!(re.is_match("/012345"));
1227 assert!(!re.is_match("/012"));
1228 assert!(!re.is_match("/01234567"));
1229 assert!(!re.is_match("/XXXXXX"));
1230
1231 let mut path = Path::new("/012345");
1232 assert!(re.capture_match_info(&mut path));
1233 assert_eq!(path.get("id").unwrap(), "012345");
1234 assert_eq!(path.unprocessed(), "");
1235 }
1236
1237 #[allow(clippy::cognitive_complexity)]
1238 #[test]
1239 fn dynamic_set() {
1240 let re = ResourceDef::new(vec![
1241 "/user/{id}",
1242 "/v{version}/resource/{id}",
1243 "/{id:[[:digit:]]{6}}",
1244 "/static",
1245 ]);
1246 assert!(re.is_match("/user/profile"));
1247 assert!(re.is_match("/user/2345"));
1248 assert!(!re.is_match("/user/2345/"));
1249 assert!(!re.is_match("/user/2345/sdg"));
1250
1251 let mut path = Path::new("/user/profile");
1252 assert!(re.capture_match_info(&mut path));
1253 assert_eq!(path.get("id").unwrap(), "profile");
1254 assert_eq!(path.unprocessed(), "");
1255
1256 let mut path = Path::new("/user/1245125");
1257 assert!(re.capture_match_info(&mut path));
1258 assert_eq!(path.get("id").unwrap(), "1245125");
1259 assert_eq!(path.unprocessed(), "");
1260
1261 assert!(re.is_match("/v1/resource/320120"));
1262 assert!(!re.is_match("/v/resource/1"));
1263 assert!(!re.is_match("/resource"));
1264
1265 let mut path = Path::new("/v151/resource/adage32");
1266 assert!(re.capture_match_info(&mut path));
1267 assert_eq!(path.get("version").unwrap(), "151");
1268 assert_eq!(path.get("id").unwrap(), "adage32");
1269
1270 assert!(re.is_match("/012345"));
1271 assert!(!re.is_match("/012"));
1272 assert!(!re.is_match("/01234567"));
1273 assert!(!re.is_match("/XXXXXX"));
1274
1275 assert!(re.is_match("/static"));
1276 assert!(!re.is_match("/a/static"));
1277 assert!(!re.is_match("/static/a"));
1278
1279 let mut path = Path::new("/012345");
1280 assert!(re.capture_match_info(&mut path));
1281 assert_eq!(path.get("id").unwrap(), "012345");
1282
1283 let re = ResourceDef::new([
1284 "/user/{id}",
1285 "/v{version}/resource/{id}",
1286 "/{id:[[:digit:]]{6}}",
1287 ]);
1288 assert!(re.is_match("/user/profile"));
1289 assert!(re.is_match("/user/2345"));
1290 assert!(!re.is_match("/user/2345/"));
1291 assert!(!re.is_match("/user/2345/sdg"));
1292
1293 let re = ResourceDef::new([
1294 "/user/{id}".to_string(),
1295 "/v{version}/resource/{id}".to_string(),
1296 "/{id:[[:digit:]]{6}}".to_string(),
1297 ]);
1298 assert!(re.is_match("/user/profile"));
1299 assert!(re.is_match("/user/2345"));
1300 assert!(!re.is_match("/user/2345/"));
1301 assert!(!re.is_match("/user/2345/sdg"));
1302 }
1303
1304 #[test]
1305 fn dynamic_set_prefix() {
1306 let re = ResourceDef::prefix(vec!["/u/{id}", "/{id:[[:digit:]]{3}}"]);
1307
1308 assert_eq!(re.find_match("/u/abc"), Some(6));
1309 assert_eq!(re.find_match("/u/abc/123"), Some(6));
1310 assert_eq!(re.find_match("/s/user/profile"), None);
1311
1312 assert_eq!(re.find_match("/123"), Some(4));
1313 assert_eq!(re.find_match("/123/456"), Some(4));
1314 assert_eq!(re.find_match("/12345"), None);
1315
1316 let mut path = Path::new("/151/res");
1317 assert!(re.capture_match_info(&mut path));
1318 assert_eq!(path.get("id").unwrap(), "151");
1319 assert_eq!(path.unprocessed(), "/res");
1320 }
1321
1322 #[test]
1323 fn parse_tail() {
1324 let re = ResourceDef::new("/user/-{id}*");
1325
1326 let mut path = Path::new("/user/-profile");
1327 assert!(re.capture_match_info(&mut path));
1328 assert_eq!(path.get("id").unwrap(), "profile");
1329
1330 let mut path = Path::new("/user/-2345");
1331 assert!(re.capture_match_info(&mut path));
1332 assert_eq!(path.get("id").unwrap(), "2345");
1333
1334 let mut path = Path::new("/user/-2345/");
1335 assert!(re.capture_match_info(&mut path));
1336 assert_eq!(path.get("id").unwrap(), "2345/");
1337
1338 let mut path = Path::new("/user/-2345/sdg");
1339 assert!(re.capture_match_info(&mut path));
1340 assert_eq!(path.get("id").unwrap(), "2345/sdg");
1341 }
1342
1343 #[test]
1344 fn static_tail() {
1345 let re = ResourceDef::new("/user{tail}*");
1346 assert!(re.is_match("/users"));
1347 assert!(re.is_match("/user-foo"));
1348 assert!(re.is_match("/user/profile"));
1349 assert!(re.is_match("/user/2345"));
1350 assert!(re.is_match("/user/2345/"));
1351 assert!(re.is_match("/user/2345/sdg"));
1352 assert!(!re.is_match("/foo/profile"));
1353
1354 let re = ResourceDef::new("/user/{tail}*");
1355 assert!(re.is_match("/user/profile"));
1356 assert!(re.is_match("/user/2345"));
1357 assert!(re.is_match("/user/2345/"));
1358 assert!(re.is_match("/user/2345/sdg"));
1359 assert!(!re.is_match("/foo/profile"));
1360 }
1361
1362 #[test]
1363 fn dynamic_tail() {
1364 let re = ResourceDef::new("/user/{id}/{tail}*");
1365 assert!(!re.is_match("/user/2345"));
1366 let mut path = Path::new("/user/2345/sdg");
1367 assert!(re.capture_match_info(&mut path));
1368 assert_eq!(path.get("id").unwrap(), "2345");
1369 assert_eq!(path.get("tail").unwrap(), "sdg");
1370 assert_eq!(path.unprocessed(), "");
1371 }
1372
1373 #[allow(clippy::literal_string_with_formatting_args)]
1374 #[test]
1375 fn newline_patterns_and_paths() {
1376 let re = ResourceDef::new("/user/a\nb");
1377 assert!(re.is_match("/user/a\nb"));
1378 assert!(!re.is_match("/user/a\nb/profile"));
1379
1380 let re = ResourceDef::new("/a{x}b/test/a{y}b");
1381 let mut path = Path::new("/a\nb/test/a\nb");
1382 assert!(re.capture_match_info(&mut path));
1383 assert_eq!(path.get("x").unwrap(), "\n");
1384 assert_eq!(path.get("y").unwrap(), "\n");
1385
1386 let re = ResourceDef::new("/user/{tail}*");
1387 assert!(re.is_match("/user/a\nb/"));
1388
1389 let re = ResourceDef::new("/user/{id}*");
1390 let mut path = Path::new("/user/a\nb/a\nb");
1391 assert!(re.capture_match_info(&mut path));
1392 assert_eq!(path.get("id").unwrap(), "a\nb/a\nb");
1393
1394 let re = ResourceDef::new("/user/{id:.*}");
1395 let mut path = Path::new("/user/a\nb/a\nb");
1396 assert!(re.capture_match_info(&mut path));
1397 assert_eq!(path.get("id").unwrap(), "a\nb/a\nb");
1398 }
1399
1400 #[cfg(feature = "http")]
1401 #[test]
1402 fn parse_urlencoded_param() {
1403 let re = ResourceDef::new("/user/{id}/test");
1404
1405 let mut path = Path::new("/user/2345/test");
1406 assert!(re.capture_match_info(&mut path));
1407 assert_eq!(path.get("id").unwrap(), "2345");
1408
1409 let mut path = Path::new("/user/qwe%25/test");
1410 assert!(re.capture_match_info(&mut path));
1411 assert_eq!(path.get("id").unwrap(), "qwe%25");
1412
1413 let uri = http::Uri::try_from("/user/qwe%25/test").unwrap();
1414 let mut path = Path::new(uri);
1415 assert!(re.capture_match_info(&mut path));
1416 assert_eq!(path.get("id").unwrap(), "qwe%25");
1417 }
1418
1419 #[test]
1420 fn prefix_static() {
1421 let re = ResourceDef::prefix("/name");
1422
1423 assert!(re.is_prefix());
1424
1425 assert!(re.is_match("/name"));
1426 assert!(re.is_match("/name/"));
1427 assert!(re.is_match("/name/test/test"));
1428 assert!(!re.is_match("/name1"));
1429 assert!(!re.is_match("/name~"));
1430
1431 let mut path = Path::new("/name");
1432 assert!(re.capture_match_info(&mut path));
1433 assert_eq!(path.unprocessed(), "");
1434
1435 let mut path = Path::new("/name/test");
1436 assert!(re.capture_match_info(&mut path));
1437 assert_eq!(path.unprocessed(), "/test");
1438
1439 assert_eq!(re.find_match("/name"), Some(5));
1440 assert_eq!(re.find_match("/name/"), Some(5));
1441 assert_eq!(re.find_match("/name/test/test"), Some(5));
1442 assert_eq!(re.find_match("/name1"), None);
1443 assert_eq!(re.find_match("/name~"), None);
1444
1445 let re = ResourceDef::prefix("/name/");
1446 assert!(re.is_match("/name/"));
1447 assert!(re.is_match("/name//gs"));
1448 assert!(!re.is_match("/name/gs"));
1449 assert!(!re.is_match("/name"));
1450
1451 let mut path = Path::new("/name/gs");
1452 assert!(!re.capture_match_info(&mut path));
1453
1454 let mut path = Path::new("/name//gs");
1455 assert!(re.capture_match_info(&mut path));
1456 assert_eq!(path.unprocessed(), "/gs");
1457
1458 let re = ResourceDef::root_prefix("name/");
1459 assert!(re.is_match("/name/"));
1460 assert!(re.is_match("/name//gs"));
1461 assert!(!re.is_match("/name/gs"));
1462 assert!(!re.is_match("/name"));
1463
1464 let mut path = Path::new("/name/gs");
1465 assert!(!re.capture_match_info(&mut path));
1466 }
1467
1468 #[test]
1469 fn prefix_dynamic() {
1470 let re = ResourceDef::prefix("/{name}");
1471
1472 assert!(re.is_prefix());
1473
1474 assert!(re.is_match("/name/"));
1475 assert!(re.is_match("/name/gs"));
1476 assert!(re.is_match("/name"));
1477
1478 assert_eq!(re.find_match("/name/"), Some(5));
1479 assert_eq!(re.find_match("/name/gs"), Some(5));
1480 assert_eq!(re.find_match("/name"), Some(5));
1481 assert_eq!(re.find_match(""), None);
1482
1483 let mut path = Path::new("/test2/");
1484 assert!(re.capture_match_info(&mut path));
1485 assert_eq!(&path["name"], "test2");
1486 assert_eq!(&path[0], "test2");
1487 assert_eq!(path.unprocessed(), "/");
1488
1489 let mut path = Path::new("/test2/subpath1/subpath2/index.html");
1490 assert!(re.capture_match_info(&mut path));
1491 assert_eq!(&path["name"], "test2");
1492 assert_eq!(&path[0], "test2");
1493 assert_eq!(path.unprocessed(), "/subpath1/subpath2/index.html");
1494
1495 let resource = ResourceDef::prefix("/user");
1496 assert!(resource.find_match("/foo").is_none());
1498 }
1499
1500 #[test]
1501 fn prefix_empty() {
1502 let re = ResourceDef::prefix("");
1503
1504 assert!(re.is_prefix());
1505
1506 assert!(re.is_match(""));
1507 assert!(re.is_match("/"));
1508 assert!(re.is_match("/name/test/test"));
1509 }
1510
1511 #[test]
1512 fn build_path_list() {
1513 let mut s = String::new();
1514 let resource = ResourceDef::new("/user/{item1}/test");
1515 assert!(resource.resource_path_from_iter(&mut s, &mut ["user1"].iter()));
1516 assert_eq!(s, "/user/user1/test");
1517
1518 let mut s = String::new();
1519 let resource = ResourceDef::new("/user/{item1}/{item2}/test");
1520 assert!(resource.resource_path_from_iter(&mut s, &mut ["item", "item2"].iter()));
1521 assert_eq!(s, "/user/item/item2/test");
1522
1523 let mut s = String::new();
1524 let resource = ResourceDef::new("/user/{item1}/{item2}");
1525 assert!(resource.resource_path_from_iter(&mut s, &mut ["item", "item2"].iter()));
1526 assert_eq!(s, "/user/item/item2");
1527
1528 let mut s = String::new();
1529 let resource = ResourceDef::new("/user/{item1}/{item2}/");
1530 assert!(resource.resource_path_from_iter(&mut s, &mut ["item", "item2"].iter()));
1531 assert_eq!(s, "/user/item/item2/");
1532
1533 let mut s = String::new();
1534 assert!(!resource.resource_path_from_iter(&mut s, &mut ["item"].iter()));
1535
1536 let mut s = String::new();
1537 assert!(resource.resource_path_from_iter(&mut s, &mut ["item", "item2"].iter()));
1538 assert_eq!(s, "/user/item/item2/");
1539 assert!(!resource.resource_path_from_iter(&mut s, &mut ["item"].iter()));
1540
1541 let mut s = String::new();
1542
1543 assert!(resource.resource_path_from_iter(
1544 &mut s,
1545 #[allow(clippy::useless_vec)]
1546 &mut vec!["item", "item2"].iter()
1547 ));
1548 assert_eq!(s, "/user/item/item2/");
1549 }
1550
1551 #[test]
1552 fn multi_pattern_build_path() {
1553 let resource = ResourceDef::new(["/user/{id}", "/profile/{id}"]);
1554 let mut s = String::new();
1555 assert!(resource.resource_path_from_iter(&mut s, &mut ["123"].iter()));
1556 assert_eq!(s, "/user/123");
1557 }
1558
1559 #[test]
1560 fn multi_pattern_capture_segment_values() {
1561 let resource = ResourceDef::new(["/user/{id}", "/profile/{id}"]);
1562
1563 let mut path = Path::new("/user/123");
1564 assert!(resource.capture_match_info(&mut path));
1565 assert!(path.get("id").is_some());
1566
1567 let mut path = Path::new("/profile/123");
1568 assert!(resource.capture_match_info(&mut path));
1569 assert!(path.get("id").is_some());
1570
1571 let resource = ResourceDef::new(["/user/{id}", "/profile/{uid}"]);
1572
1573 let mut path = Path::new("/user/123");
1574 assert!(resource.capture_match_info(&mut path));
1575 assert!(path.get("id").is_some());
1576 assert!(path.get("uid").is_none());
1577
1578 let mut path = Path::new("/profile/123");
1579 assert!(resource.capture_match_info(&mut path));
1580 assert!(path.get("id").is_none());
1581 assert!(path.get("uid").is_some());
1582 }
1583
1584 #[test]
1585 fn dynamic_prefix_proper_segmentation() {
1586 let resource = ResourceDef::prefix(r"/id/{id:\d{3}}");
1587
1588 assert!(resource.is_match("/id/123"));
1589 assert!(resource.is_match("/id/123/foo"));
1590 assert!(!resource.is_match("/id/1234"));
1591 assert!(!resource.is_match("/id/123a"));
1592
1593 assert_eq!(resource.find_match("/id/123"), Some(7));
1594 assert_eq!(resource.find_match("/id/123/foo"), Some(7));
1595 assert_eq!(resource.find_match("/id/1234"), None);
1596 assert_eq!(resource.find_match("/id/123a"), None);
1597 }
1598
1599 #[test]
1600 fn build_path_map() {
1601 let resource = ResourceDef::new("/user/{item1}/{item2}/");
1602
1603 let mut map = HashMap::new();
1604 map.insert("item1", "item");
1605
1606 let mut s = String::new();
1607 assert!(!resource.resource_path_from_map(&mut s, &map));
1608
1609 map.insert("item2", "item2");
1610
1611 let mut s = String::new();
1612 assert!(resource.resource_path_from_map(&mut s, &map));
1613 assert_eq!(s, "/user/item/item2/");
1614 }
1615
1616 #[test]
1617 fn build_path_tail() {
1618 let resource = ResourceDef::new("/user/{item1}*");
1619
1620 let mut s = String::new();
1621 assert!(!resource.resource_path_from_iter(&mut s, &mut [""; 0].iter()));
1622
1623 let mut s = String::new();
1624 assert!(resource.resource_path_from_iter(&mut s, &mut ["user1"].iter()));
1625 assert_eq!(s, "/user/user1");
1626
1627 let mut s = String::new();
1628 let mut map = HashMap::new();
1629 map.insert("item1", "item");
1630 assert!(resource.resource_path_from_map(&mut s, &map));
1631 assert_eq!(s, "/user/item");
1632 }
1633
1634 #[test]
1635 fn prefix_trailing_slash() {
1636 let re = ResourceDef::prefix("/abc/");
1640 assert_eq!(re.find_match("/abc/def"), None);
1641 assert_eq!(re.find_match("/abc//def"), Some(5));
1642
1643 let re = ResourceDef::prefix("/{id}/");
1644 assert_eq!(re.find_match("/abc/def"), None);
1645 assert_eq!(re.find_match("/abc//def"), Some(5));
1646 }
1647
1648 #[test]
1649 fn join() {
1650 fn seq_find_match(re1: &ResourceDef, re2: &ResourceDef, path: &str) -> Option<usize> {
1653 let len1 = re1.find_match(path)?;
1654 let len2 = re2.find_match(&path[len1..])?;
1655 Some(len1 + len2)
1656 }
1657
1658 macro_rules! join_test {
1659 ($pat1:expr, $pat2:expr => $($test:expr),+) => {{
1660 let pat1 = $pat1;
1661 let pat2 = $pat2;
1662 $({
1663 let _path = $test;
1664 let (re1, re2) = (ResourceDef::prefix(pat1), ResourceDef::new(pat2));
1665 let _seq = seq_find_match(&re1, &re2, _path);
1666 let _join = re1.join(&re2).find_match(_path);
1667 assert_eq!(
1668 _seq, _join,
1669 "patterns: prefix {:?}, {:?}; mismatch on \"{}\"; seq={:?}; join={:?}",
1670 pat1, pat2, _path, _seq, _join
1671 );
1672 assert!(!re1.join(&re2).is_prefix());
1673
1674 let (re1, re2) = (ResourceDef::prefix(pat1), ResourceDef::prefix(pat2));
1675 let _seq = seq_find_match(&re1, &re2, _path);
1676 let _join = re1.join(&re2).find_match(_path);
1677 assert_eq!(
1678 _seq, _join,
1679 "patterns: prefix {:?}, prefix {:?}; mismatch on \"{}\"; seq={:?}; join={:?}",
1680 pat1, pat2, _path, _seq, _join
1681 );
1682 assert!(re1.join(&re2).is_prefix());
1683 })+
1684 }}
1685 }
1686
1687 join_test!("", "" => "", "/hello", "/");
1688 join_test!("/user", "" => "", "/user", "/user/123", "/user11", "user", "user/123");
1689 join_test!("", "/user" => "", "/user", "foo", "/user11", "user", "user/123");
1690 join_test!("/user", "/xx" => "", "", "/", "/user", "/xx", "/userxx", "/user/xx");
1691
1692 join_test!(["/ver/{v}", "/v{v}"], ["/req/{req}", "/{req}"] => "/v1/abc",
1693 "/ver/1/abc", "/v1/req/abc", "/ver/1/req/abc", "/v1/abc/def",
1694 "/ver1/req/abc/def", "", "/", "/v1/");
1695 }
1696
1697 #[test]
1698 fn match_methods_agree() {
1699 macro_rules! match_methods_agree {
1700 ($pat:expr => $($test:expr),+) => {{
1701 match_methods_agree!(finish $pat, ResourceDef::new($pat), $($test),+);
1702 }};
1703 (prefix $pat:expr => $($test:expr),+) => {{
1704 match_methods_agree!(finish $pat, ResourceDef::prefix($pat), $($test),+);
1705 }};
1706 (finish $pat:expr, $re:expr, $($test:expr),+) => {{
1707 let re = $re;
1708 $({
1709 let _is = re.is_match($test);
1710 let _find = re.find_match($test).is_some();
1711 assert_eq!(
1712 _is, _find,
1713 "pattern: {:?}; mismatch on \"{}\"; is={}; find={}",
1714 $pat, $test, _is, _find
1715 );
1716 })+
1717 }}
1718 }
1719
1720 match_methods_agree!("" => "", "/", "/foo");
1721 match_methods_agree!("/" => "", "/", "/foo");
1722 match_methods_agree!("/user" => "user", "/user", "/users", "/user/123", "/foo");
1723 match_methods_agree!("/v{v}" => "v", "/v", "/v1", "/v222", "/foo");
1724 match_methods_agree!(["/v{v}", "/version/{v}"] => "/v", "/v1", "/version", "/version/1", "/foo");
1725
1726 match_methods_agree!("/path{tail}*" => "/path", "/path1", "/path/123");
1727 match_methods_agree!("/path/{tail}*" => "/path", "/path1", "/path/123");
1728
1729 match_methods_agree!(prefix "" => "", "/", "/foo");
1730 match_methods_agree!(prefix "/user" => "user", "/user", "/users", "/user/123", "/foo");
1731 match_methods_agree!(prefix r"/id/{id:\d{3}}" => "/id/123", "/id/1234");
1732 match_methods_agree!(["/v{v}", "/ver/{v}"] => "", "s/v", "/v1", "/v1/xx", "/ver/i3/5", "/ver/1");
1733 }
1734
1735 #[test]
1736 #[should_panic]
1737 fn duplicate_segment_name() {
1738 ResourceDef::new("/user/{id}/post/{id}");
1739 }
1740
1741 #[test]
1742 #[should_panic]
1743 fn invalid_dynamic_segment_delimiter() {
1744 ResourceDef::new("/user/{username");
1745 }
1746
1747 #[test]
1748 #[should_panic]
1749 fn invalid_dynamic_segment_name() {
1750 ResourceDef::new("/user/{}");
1751 }
1752
1753 #[test]
1754 #[should_panic]
1755 fn invalid_too_many_dynamic_segments() {
1756 ResourceDef::new("/{a}/{b}/{c}/{d}/{e}/{f}/{g}/{h}/{i}/{j}/{k}/{l}/{m}/{n}/{o}/{p}");
1758
1759 ResourceDef::new("/{a}/{b}/{c}/{d}/{e}/{f}/{g}/{h}/{i}/{j}/{k}/{l}/{m}/{n}/{o}/{p}/{q}");
1761 }
1762
1763 #[test]
1764 #[should_panic]
1765 fn invalid_custom_regex_for_tail() {
1766 ResourceDef::new(r"/{tail:\d+}*");
1767 }
1768
1769 #[test]
1770 #[should_panic]
1771 fn invalid_unnamed_tail_segment() {
1772 ResourceDef::new("/*");
1773 }
1774
1775 #[test]
1776 #[should_panic]
1777 fn prefix_plus_tail_match_disallowed() {
1778 ResourceDef::prefix("/user/{id}*");
1779 }
1780}