1
0
mirror of https://github.com/fafhrd91/actix-web synced 2025-01-18 05:41:50 +01:00

do not use regset for route recognition

This commit is contained in:
Nikolay Kim 2018-02-21 14:31:22 -08:00
parent 5634e5794f
commit fd56e5dc82
4 changed files with 102 additions and 122 deletions

View File

@ -241,7 +241,7 @@ impl<S> Application<S> where S: 'static {
let mut resource = Resource::default(); let mut resource = Resource::default();
f(&mut resource); f(&mut resource);
let pattern = Pattern::new(resource.get_name(), path, "^/"); let pattern = Pattern::new(resource.get_name(), path);
if parts.resources.contains_key(&pattern) { if parts.resources.contains_key(&pattern) {
panic!("Resource {:?} is registered.", path); panic!("Resource {:?} is registered.", path);
} }
@ -305,7 +305,7 @@ impl<S> Application<S> where S: 'static {
panic!("External resource {:?} is registered.", name.as_ref()); panic!("External resource {:?} is registered.", name.as_ref());
} }
parts.external.insert( parts.external.insert(
String::from(name.as_ref()), Pattern::new(name.as_ref(), url.as_ref(), "^/")); String::from(name.as_ref()), Pattern::new(name.as_ref(), url.as_ref()));
} }
self self
} }

View File

@ -884,7 +884,7 @@ mod tests {
let mut resource = Resource::<()>::default(); let mut resource = Resource::<()>::default();
resource.name("index"); resource.name("index");
let mut map = HashMap::new(); let mut map = HashMap::new();
map.insert(Pattern::new("index", "/{key}/", "^/"), Some(resource)); map.insert(Pattern::new("index", "/{key}/"), Some(resource));
let (router, _) = Router::new("", ServerSettings::default(), map); let (router, _) = Router::new("", ServerSettings::default(), map);
assert!(router.recognize(&mut req).is_some()); assert!(router.recognize(&mut req).is_some());
@ -995,7 +995,7 @@ mod tests {
let mut resource = Resource::<()>::default(); let mut resource = Resource::<()>::default();
resource.name("index"); resource.name("index");
let mut map = HashMap::new(); let mut map = HashMap::new();
map.insert(Pattern::new("index", "/user/{name}.{ext}", "^/"), Some(resource)); map.insert(Pattern::new("index", "/user/{name}.{ext}"), Some(resource));
let (router, _) = Router::new("/", ServerSettings::default(), map); let (router, _) = Router::new("/", ServerSettings::default(), map);
assert!(router.has_route("/user/test.html")); assert!(router.has_route("/user/test.html"));
assert!(!router.has_route("/test/unknown")); assert!(!router.has_route("/test/unknown"));
@ -1020,7 +1020,7 @@ mod tests {
let mut resource = Resource::<()>::default(); let mut resource = Resource::<()>::default();
resource.name("index"); resource.name("index");
let mut map = HashMap::new(); let mut map = HashMap::new();
map.insert(Pattern::new("index", "/user/{name}.{ext}", "^/"), Some(resource)); map.insert(Pattern::new("index", "/user/{name}.{ext}"), Some(resource));
let (router, _) = Router::new("/prefix/", ServerSettings::default(), map); let (router, _) = Router::new("/prefix/", ServerSettings::default(), map);
assert!(router.has_route("/user/test.html")); assert!(router.has_route("/user/test.html"));
assert!(!router.has_route("/prefix/user/test.html")); assert!(!router.has_route("/prefix/user/test.html"));
@ -1037,7 +1037,7 @@ mod tests {
let mut resource = Resource::<()>::default(); let mut resource = Resource::<()>::default();
resource.name("index"); resource.name("index");
let mut map = HashMap::new(); let mut map = HashMap::new();
map.insert(Pattern::new("youtube", "https://youtube.com/watch/{video_id}", "^/"), None); map.insert(Pattern::new("youtube", "https://youtube.com/watch/{video_id}"), None);
let (router, _) = Router::new::<()>("", ServerSettings::default(), map); let (router, _) = Router::new::<()>("", ServerSettings::default(), map);
assert!(!router.has_route("https://youtube.com/watch/unknown")); assert!(!router.has_route("https://youtube.com/watch/unknown"));

View File

@ -14,8 +14,8 @@ use middleware::{Middleware, Started, Finished};
/// `Middleware` for logging request and response info to the terminal. /// `Middleware` for logging request and response info to the terminal.
/// `Logger` middleware uses standard log crate to log information. You should /// `Logger` middleware uses standard log crate to log information. You should
/// enable logger for *actix_web* package to see access log. /// enable logger for `actix_web` package to see access log.
/// ([env_logger](https://docs.rs/env_logger/*/env_logger/) or similar) /// ([`env_logger`](https://docs.rs/env_logger/*/env_logger/) or similar)
/// ///
/// ## Usage /// ## Usage
/// ///

View File

@ -3,9 +3,10 @@ use std::rc::Rc;
use std::hash::{Hash, Hasher}; use std::hash::{Hash, Hasher};
use std::collections::HashMap; use std::collections::HashMap;
use regex::{Regex, RegexSet, escape}; use regex::{Regex, escape};
use error::UrlGenerationError; use error::UrlGenerationError;
use param::Params;
use resource::Resource; use resource::Resource;
use httprequest::HttpRequest; use httprequest::HttpRequest;
use server::ServerSettings; use server::ServerSettings;
@ -17,11 +18,9 @@ pub struct Router(Rc<Inner>);
struct Inner { struct Inner {
prefix: String, prefix: String,
prefix_len: usize, prefix_len: usize,
regset: RegexSet,
named: HashMap<String, (Pattern, bool)>, named: HashMap<String, (Pattern, bool)>,
patterns: Vec<Pattern>, patterns: Vec<Pattern>,
srv: ServerSettings, srv: ServerSettings,
hasroutes: bool,
} }
impl Router { impl Router {
@ -34,7 +33,6 @@ impl Router {
let mut named = HashMap::new(); let mut named = HashMap::new();
let mut patterns = Vec::new(); let mut patterns = Vec::new();
let mut resources = Vec::new(); let mut resources = Vec::new();
let mut paths = Vec::new();
for (pattern, resource) in map { for (pattern, resource) in map {
if !pattern.name().is_empty() { if !pattern.name().is_empty() {
@ -43,7 +41,6 @@ impl Router {
} }
if let Some(resource) = resource { if let Some(resource) = resource {
paths.push(pattern.pattern().to_owned());
patterns.push(pattern); patterns.push(pattern);
resources.push(resource); resources.push(resource);
} }
@ -53,10 +50,8 @@ impl Router {
(Router(Rc::new( (Router(Rc::new(
Inner{ prefix: prefix, Inner{ prefix: prefix,
prefix_len: len, prefix_len: len,
regset: RegexSet::new(&paths).unwrap(),
named: named, named: named,
patterns: patterns, patterns: patterns,
hasroutes: !paths.is_empty(),
srv: settings })), resources) srv: settings })), resources)
} }
@ -74,28 +69,18 @@ impl Router {
/// Query for matched resource /// Query for matched resource
pub fn recognize<S>(&self, req: &mut HttpRequest<S>) -> Option<usize> { pub fn recognize<S>(&self, req: &mut HttpRequest<S>) -> Option<usize> {
if !self.0.hasroutes { return None } if self.0.prefix_len > req.path().len() {
let mut idx = None; return None
{
if self.0.prefix_len > req.path().len() {
return None
}
let path = &req.path()[self.0.prefix_len..];
if path.is_empty() {
if let Some(i) = self.0.regset.matches("/").into_iter().next() {
idx = Some(i);
}
} else if let Some(i) = self.0.regset.matches(path).into_iter().next() {
idx = Some(i);
}
} }
let path: &str = unsafe{mem::transmute(&req.path()[self.0.prefix_len..])};
let route_path = if path.is_empty() { "/" } else { path };
if let Some(idx) = idx { for (idx, pattern) in self.0.patterns.iter().enumerate() {
self.0.patterns[idx].update_match_info(req, self.0.prefix_len); if pattern.match_with_params(route_path, req.match_info_mut()) {
return Some(idx) return Some(idx)
} else { }
None
} }
None
} }
/// Check if application contains matching route. /// Check if application contains matching route.
@ -105,12 +90,12 @@ impl Router {
/// following path would be recognizable `/test/name` but `has_route()` call /// following path would be recognizable `/test/name` but `has_route()` call
/// would return `false`. /// would return `false`.
pub fn has_route(&self, path: &str) -> bool { pub fn has_route(&self, path: &str) -> bool {
if path.is_empty() { let path = if path.is_empty() { "/" } else { path };
if self.0.regset.matches("/").into_iter().next().is_some() {
for pattern in &self.0.patterns {
if pattern.is_match(path) {
return true return true
} }
} else if self.0.regset.matches(path).into_iter().next().is_some() {
return true
} }
false false
} }
@ -148,12 +133,17 @@ enum PatternElement {
Var(String), Var(String),
} }
#[derive(Clone)]
enum PatternType {
Static(String),
Dynamic(Regex, Vec<String>),
}
#[derive(Clone)] #[derive(Clone)]
pub struct Pattern { pub struct Pattern {
re: Regex, tp: PatternType,
name: String, name: String,
pattern: String, pattern: String,
names: Vec<String>,
elements: Vec<PatternElement>, elements: Vec<PatternElement>,
} }
@ -161,22 +151,26 @@ impl Pattern {
/// Parse path pattern and create new `Pattern` instance. /// Parse path pattern and create new `Pattern` instance.
/// ///
/// Panics if path pattern is wrong. /// Panics if path pattern is wrong.
pub fn new(name: &str, path: &str, starts: &str) -> Self { pub fn new(name: &str, path: &str) -> Self {
let (pattern, elements) = Pattern::parse(path, starts); let (pattern, elements, is_dynamic) = Pattern::parse(path);
let re = match Regex::new(&pattern) { let tp = if is_dynamic {
Ok(re) => re, let re = match Regex::new(&pattern) {
Err(err) => panic!("Wrong path pattern: \"{}\" {}", path, err) Ok(re) => re,
Err(err) => panic!("Wrong path pattern: \"{}\" {}", path, err)
};
let names = re.capture_names()
.filter_map(|name| name.map(|name| name.to_owned()))
.collect();
PatternType::Dynamic(re, names)
} else {
PatternType::Static(pattern.clone())
}; };
let names = re.capture_names()
.filter_map(|name| name.map(|name| name.to_owned()))
.collect();
Pattern { Pattern {
re: re, tp: tp,
name: name.into(), name: name.into(),
pattern: pattern, pattern: pattern,
names: names,
elements: elements, elements: elements,
} }
} }
@ -191,44 +185,33 @@ impl Pattern {
&self.pattern &self.pattern
} }
/// Extract pattern parameters from the text pub fn is_match(&self, path: &str) -> bool {
// This method unsafe internally, assumption that Pattern instance lives match self.tp {
// longer than `req` PatternType::Static(ref s) => s == path,
pub fn update_match_info<S>(&self, req: &mut HttpRequest<S>, prefix: usize) { PatternType::Dynamic(ref re, _) => re.is_match(path),
if !self.names.is_empty() {
let text: &str = unsafe{ mem::transmute(&req.path()[prefix..]) };
if let Some(captures) = self.re.captures(text) {
let mut idx = 0;
for capture in captures.iter() {
if let Some(ref m) = capture {
if idx != 0 {
req.match_info_mut().add(
self.names[idx-1].as_str(), m.as_str());
}
idx += 1;
}
}
};
} }
} }
/// Extract pattern parameters from the text pub fn match_with_params<'a>(&'a self, path: &'a str, params: &'a mut Params<'a>) -> bool {
pub fn get_match_info<'a>(&self, text: &'a str) -> HashMap<&str, &'a str> { match self.tp {
let mut info = HashMap::new(); PatternType::Static(ref s) => s == path,
if !self.names.is_empty() { PatternType::Dynamic(ref re, ref names) => {
if let Some(captures) = self.re.captures(text) { if let Some(captures) = re.captures(path) {
let mut idx = 0; let mut idx = 0;
for capture in captures.iter() { for capture in captures.iter() {
if let Some(ref m) = capture { if let Some(ref m) = capture {
if idx != 0 { if idx != 0 {
info.insert(self.names[idx-1].as_str(), m.as_str()); params.add(names[idx-1].as_str(), m.as_str());
}
idx += 1;
} }
idx += 1;
} }
true
} else {
false
} }
}; }
} }
info
} }
/// Build pattern path. /// Build pattern path.
@ -257,15 +240,16 @@ impl Pattern {
Ok(path) Ok(path)
} }
fn parse(pattern: &str, starts: &str) -> (String, Vec<PatternElement>) { fn parse(pattern: &str) -> (String, Vec<PatternElement>, bool) {
const DEFAULT_PATTERN: &str = "[^/]+"; const DEFAULT_PATTERN: &str = "[^/]+";
let mut re = String::from(starts); let mut re = String::from("/");
let mut el = String::new(); let mut el = String::new();
let mut in_param = false; let mut in_param = false;
let mut in_param_pattern = false; let mut in_param_pattern = false;
let mut param_name = String::new(); let mut param_name = String::new();
let mut param_pattern = String::from(DEFAULT_PATTERN); let mut param_pattern = String::from(DEFAULT_PATTERN);
let mut is_dynamic = false;
let mut elems = Vec::new(); let mut elems = Vec::new();
for (index, ch) in pattern.chars().enumerate() { for (index, ch) in pattern.chars().enumerate() {
@ -299,6 +283,7 @@ impl Pattern {
} }
} else if ch == '{' { } else if ch == '{' {
in_param = true; in_param = true;
is_dynamic = true;
elems.push(PatternElement::Str(el.clone())); elems.push(PatternElement::Str(el.clone()));
el.clear(); el.clear();
} else { } else {
@ -307,8 +292,11 @@ impl Pattern {
} }
} }
re.push('$'); if is_dynamic {
(re, elems) re.insert(0, '^');
re.push('$');
}
(re, elems, is_dynamic)
} }
} }
@ -329,21 +317,20 @@ impl Hash for Pattern {
#[cfg(test)] #[cfg(test)]
mod tests { mod tests {
use super::*; use super::*;
use regex::Regex;
use test::TestRequest; use test::TestRequest;
#[test] #[test]
fn test_recognizer() { fn test_recognizer() {
let mut routes = HashMap::new(); let mut routes = HashMap::new();
routes.insert(Pattern::new("", "/name", "^/"), Some(Resource::default())); routes.insert(Pattern::new("", "/name"), Some(Resource::default()));
routes.insert(Pattern::new("", "/name/{val}", "^/"), Some(Resource::default())); routes.insert(Pattern::new("", "/name/{val}"), Some(Resource::default()));
routes.insert(Pattern::new("", "/name/{val}/index.html", "^/"), routes.insert(Pattern::new("", "/name/{val}/index.html"),
Some(Resource::default())); Some(Resource::default()));
routes.insert(Pattern::new("", "/file/{file}.{ext}", "^/"), Some(Resource::default())); routes.insert(Pattern::new("", "/file/{file}.{ext}"), Some(Resource::default()));
routes.insert(Pattern::new("", "/v{val}/{val2}/index.html", "^/"), routes.insert(Pattern::new("", "/v{val}/{val2}/index.html"),
Some(Resource::default())); Some(Resource::default()));
routes.insert(Pattern::new("", "/v/{tail:.*}", "^/"), Some(Resource::default())); routes.insert(Pattern::new("", "/v/{tail:.*}"), Some(Resource::default()));
routes.insert(Pattern::new("", "{test}/index.html", "^/"), Some(Resource::default())); routes.insert(Pattern::new("", "{test}/index.html"), Some(Resource::default()));
let (rec, _) = Router::new::<()>("", ServerSettings::default(), routes); let (rec, _) = Router::new::<()>("", ServerSettings::default(), routes);
let mut req = TestRequest::with_uri("/name").finish(); let mut req = TestRequest::with_uri("/name").finish();
@ -381,8 +368,8 @@ mod tests {
#[test] #[test]
fn test_recognizer_with_prefix() { fn test_recognizer_with_prefix() {
let mut routes = HashMap::new(); let mut routes = HashMap::new();
routes.insert(Pattern::new("", "/name", "^/"), Some(Resource::default())); routes.insert(Pattern::new("", "/name"), Some(Resource::default()));
routes.insert(Pattern::new("", "/name/{val}", "^/"), Some(Resource::default())); routes.insert(Pattern::new("", "/name/{val}"), Some(Resource::default()));
let (rec, _) = Router::new::<()>("/test", ServerSettings::default(), routes); let (rec, _) = Router::new::<()>("/test", ServerSettings::default(), routes);
let mut req = TestRequest::with_uri("/name").finish(); let mut req = TestRequest::with_uri("/name").finish();
@ -398,8 +385,8 @@ mod tests {
// same patterns // same patterns
let mut routes = HashMap::new(); let mut routes = HashMap::new();
routes.insert(Pattern::new("", "/name", "^/"), Some(Resource::default())); routes.insert(Pattern::new("", "/name"), Some(Resource::default()));
routes.insert(Pattern::new("", "/name/{val}", "^/"), Some(Resource::default())); routes.insert(Pattern::new("", "/name/{val}"), Some(Resource::default()));
let (rec, _) = Router::new::<()>("/test2", ServerSettings::default(), routes); let (rec, _) = Router::new::<()>("/test2", ServerSettings::default(), routes);
let mut req = TestRequest::with_uri("/name").finish(); let mut req = TestRequest::with_uri("/name").finish();
@ -408,61 +395,54 @@ mod tests {
assert!(rec.recognize(&mut req).is_some()); assert!(rec.recognize(&mut req).is_some());
} }
fn assert_parse(pattern: &str, expected_re: &str) -> Regex {
let (re_str, _) = Pattern::parse(pattern, "^/");
assert_eq!(&*re_str, expected_re);
Regex::new(&re_str).unwrap()
}
#[test] #[test]
fn test_parse_static() { fn test_parse_static() {
let re = assert_parse("/", r"^/$"); let re = Pattern::new("test", "/");
assert!(re.is_match("/")); assert!(re.is_match("/"));
assert!(!re.is_match("/a")); assert!(!re.is_match("/a"));
let re = assert_parse("/name", r"^/name$"); let re = Pattern::new("test", "/name");
assert!(re.is_match("/name")); assert!(re.is_match("/name"));
assert!(!re.is_match("/name1")); assert!(!re.is_match("/name1"));
assert!(!re.is_match("/name/")); assert!(!re.is_match("/name/"));
assert!(!re.is_match("/name~")); assert!(!re.is_match("/name~"));
let re = assert_parse("/name/", r"^/name/$"); let re = Pattern::new("test", "/name/");
assert!(re.is_match("/name/")); assert!(re.is_match("/name/"));
assert!(!re.is_match("/name")); assert!(!re.is_match("/name"));
assert!(!re.is_match("/name/gs")); assert!(!re.is_match("/name/gs"));
let re = assert_parse("/user/profile", r"^/user/profile$"); let re = Pattern::new("test", "/user/profile");
assert!(re.is_match("/user/profile")); assert!(re.is_match("/user/profile"));
assert!(!re.is_match("/user/profile/profile")); assert!(!re.is_match("/user/profile/profile"));
} }
#[test] #[test]
fn test_parse_param() { fn test_parse_param() {
let re = assert_parse("/user/{id}", r"^/user/(?P<id>[^/]+)$"); let mut req = HttpRequest::default();
let re = Pattern::new("test", "/user/{id}");
assert!(re.is_match("/user/profile")); assert!(re.is_match("/user/profile"));
assert!(re.is_match("/user/2345")); assert!(re.is_match("/user/2345"));
assert!(!re.is_match("/user/2345/")); assert!(!re.is_match("/user/2345/"));
assert!(!re.is_match("/user/2345/sdg")); assert!(!re.is_match("/user/2345/sdg"));
let captures = re.captures("/user/profile").unwrap(); req.match_info_mut().clear();
assert_eq!(captures.get(1).unwrap().as_str(), "profile"); assert!(re.match_with_params("/user/profile", req.match_info_mut()));
assert_eq!(captures.name("id").unwrap().as_str(), "profile"); assert_eq!(req.match_info().get("id").unwrap(), "profile");
let captures = re.captures("/user/1245125").unwrap(); req.match_info_mut().clear();
assert_eq!(captures.get(1).unwrap().as_str(), "1245125"); assert!(re.match_with_params("/user/1245125", req.match_info_mut()));
assert_eq!(captures.name("id").unwrap().as_str(), "1245125"); assert_eq!(req.match_info().get("id").unwrap(), "1245125");
let re = assert_parse( let re = Pattern::new("test", "/v{version}/resource/{id}");
"/v{version}/resource/{id}",
r"^/v(?P<version>[^/]+)/resource/(?P<id>[^/]+)$",
);
assert!(re.is_match("/v1/resource/320120")); assert!(re.is_match("/v1/resource/320120"));
assert!(!re.is_match("/v/resource/1")); assert!(!re.is_match("/v/resource/1"));
assert!(!re.is_match("/resource")); assert!(!re.is_match("/resource"));
let captures = re.captures("/v151/resource/adahg32").unwrap(); req.match_info_mut().clear();
assert_eq!(captures.get(1).unwrap().as_str(), "151"); assert!(re.match_with_params("/v151/resource/adahg32", req.match_info_mut()));
assert_eq!(captures.name("version").unwrap().as_str(), "151"); assert_eq!(req.match_info().get("version").unwrap(), "151");
assert_eq!(captures.name("id").unwrap().as_str(), "adahg32"); assert_eq!(req.match_info().get("id").unwrap(), "adahg32");
} }
} }