2016-11-22 03:31:09 +00:00
|
|
|
part of angel_route.src.router;
|
2016-10-11 18:53:32 +00:00
|
|
|
|
2016-10-12 17:58:32 +00:00
|
|
|
String _matcherify(String path, {bool expand: true}) {
|
2016-10-19 22:04:06 +00:00
|
|
|
var p = path.replaceAll(new RegExp(r'/\*$'), "*").replaceAll('/', r'\/');
|
2016-10-12 17:58:32 +00:00
|
|
|
|
|
|
|
if (expand) {
|
|
|
|
var match = _param.firstMatch(p);
|
|
|
|
|
|
|
|
while (match != null) {
|
|
|
|
if (match.group(3) == null)
|
|
|
|
p = p.replaceAll(match[0], '([^\/]+)');
|
|
|
|
else
|
|
|
|
p = p.replaceAll(match[0], '(${match[3]})');
|
|
|
|
match = _param.firstMatch(p);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
p = p.replaceAll(new RegExp('\\*'), '.*');
|
|
|
|
|
|
|
|
p = '^$p\$';
|
|
|
|
return p;
|
|
|
|
}
|
|
|
|
|
|
|
|
String _pathify(String path) {
|
|
|
|
var p = path.replaceAll(_straySlashes, '');
|
|
|
|
|
|
|
|
Map<String, String> replace = {};
|
|
|
|
|
|
|
|
for (Match match in _param.allMatches(p)) {
|
|
|
|
if (match[3] != null) replace[match[0]] = ':${match[1]}';
|
|
|
|
}
|
|
|
|
|
|
|
|
replace.forEach((k, v) {
|
|
|
|
p = p.replaceAll(k, v);
|
|
|
|
});
|
|
|
|
|
|
|
|
return p;
|
|
|
|
}
|
|
|
|
|
2016-10-19 22:04:06 +00:00
|
|
|
/// Represents a virtual location within an application.
|
2016-10-11 18:53:32 +00:00
|
|
|
class Route {
|
|
|
|
final List<Route> _children = [];
|
|
|
|
final List _handlers = [];
|
2016-11-22 03:31:09 +00:00
|
|
|
RegExp _head;
|
2016-10-11 18:53:32 +00:00
|
|
|
RegExp _matcher;
|
2016-10-14 03:07:34 +00:00
|
|
|
String _method;
|
2016-10-12 17:58:32 +00:00
|
|
|
String _name;
|
2016-10-11 18:53:32 +00:00
|
|
|
Route _parent;
|
2016-10-15 00:45:22 +00:00
|
|
|
RegExp _parentResolver;
|
2016-10-11 18:53:32 +00:00
|
|
|
String _path;
|
2016-10-12 17:58:32 +00:00
|
|
|
String _pathified;
|
|
|
|
RegExp _resolver;
|
2016-10-19 22:04:06 +00:00
|
|
|
RegExp _stub;
|
|
|
|
|
|
|
|
/// Set to `true` to print verbose debug output when interacting with this route.
|
|
|
|
bool debug;
|
|
|
|
|
|
|
|
/// Contains any child routes attached to this one.
|
2016-10-11 18:53:32 +00:00
|
|
|
List<Route> get children => new List.unmodifiable(_children);
|
2016-10-19 22:04:06 +00:00
|
|
|
|
|
|
|
/// A `List` of arbitrary objects chosen to respond to this request.
|
2016-10-11 18:53:32 +00:00
|
|
|
List get handlers => new List.unmodifiable(_handlers);
|
2016-10-19 22:04:06 +00:00
|
|
|
|
|
|
|
/// A `RegExp` that matches requests to this route.
|
2016-10-11 18:53:32 +00:00
|
|
|
RegExp get matcher => _matcher;
|
2016-10-19 22:04:06 +00:00
|
|
|
|
|
|
|
/// The HTTP method this route is designated for.
|
2016-10-14 03:07:34 +00:00
|
|
|
String get method => _method;
|
2016-10-19 22:04:06 +00:00
|
|
|
|
|
|
|
/// The name of this route, if any.
|
2016-10-12 17:58:32 +00:00
|
|
|
String get name => _name;
|
2016-10-19 22:04:06 +00:00
|
|
|
|
|
|
|
/// The hierarchical parent of this route.
|
2016-10-11 18:53:32 +00:00
|
|
|
Route get parent => _parent;
|
2016-10-19 22:04:06 +00:00
|
|
|
|
|
|
|
/// The virtual path on which this route is mounted.
|
2016-10-11 18:53:32 +00:00
|
|
|
String get path => _path;
|
2016-10-19 22:04:06 +00:00
|
|
|
|
|
|
|
/// Arbitrary state attached to this route.
|
2016-10-12 17:58:32 +00:00
|
|
|
final Extensible state = new Extensible();
|
|
|
|
|
2016-10-19 22:04:06 +00:00
|
|
|
/// The [Route] at the top of the hierarchy this route is found in.
|
2016-10-12 17:58:32 +00:00
|
|
|
Route get absoluteParent {
|
|
|
|
Route result = this;
|
|
|
|
|
|
|
|
while (result.parent != null) result = result.parent;
|
|
|
|
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
2016-11-22 03:31:09 +00:00
|
|
|
/// Returns the [Route] instances that will respond to requests
|
|
|
|
/// to the index of this instance's path.
|
|
|
|
///
|
|
|
|
/// May return `this`.
|
|
|
|
Iterable<Route> get allIndices {
|
|
|
|
return children.where((r) => r.path.replaceAll(path, '').isEmpty);
|
|
|
|
}
|
|
|
|
|
2016-10-12 17:58:32 +00:00
|
|
|
/// Backtracks up the hierarchy, and builds
|
|
|
|
/// a sequential list of all handlers from both
|
|
|
|
/// this route, and every found parent route.
|
|
|
|
///
|
|
|
|
/// The resulting list puts handlers from routes
|
|
|
|
/// higher in the tree at lower indices. Thus,
|
|
|
|
/// this can be used in a routing-enabled application
|
|
|
|
/// to evaluate multiple middleware on a single route,
|
|
|
|
/// and apply them to all children.
|
|
|
|
List get handlerSequence {
|
|
|
|
final result = [];
|
|
|
|
var r = this;
|
|
|
|
|
|
|
|
while (r != null) {
|
|
|
|
result.insertAll(0, r.handlers);
|
|
|
|
r = r.parent;
|
|
|
|
}
|
|
|
|
|
|
|
|
return result;
|
|
|
|
}
|
2016-10-11 18:53:32 +00:00
|
|
|
|
2016-10-21 03:13:13 +00:00
|
|
|
/// Returns the [Route] instance that will respond to requests
|
|
|
|
/// to the index of this instance's path.
|
|
|
|
///
|
|
|
|
/// May return `this`.
|
|
|
|
Route get indexRoute {
|
|
|
|
return children.firstWhere((r) => r.path.replaceAll(path, '').isEmpty,
|
|
|
|
orElse: () => this);
|
|
|
|
}
|
|
|
|
|
2016-10-19 22:04:06 +00:00
|
|
|
void _printDebug(msg) {
|
|
|
|
if (debug) print(msg);
|
|
|
|
}
|
|
|
|
|
2016-11-22 03:31:09 +00:00
|
|
|
Route._base();
|
|
|
|
|
2016-10-11 18:53:32 +00:00
|
|
|
Route(Pattern path,
|
|
|
|
{Iterable<Route> children: const [],
|
2016-10-19 22:04:06 +00:00
|
|
|
this.debug: false,
|
2016-10-11 18:53:32 +00:00
|
|
|
Iterable handlers: const [],
|
2016-10-14 03:07:34 +00:00
|
|
|
method: "GET",
|
2016-10-12 17:58:32 +00:00
|
|
|
String name: null}) {
|
2016-10-11 18:53:32 +00:00
|
|
|
if (children != null) _children.addAll(children);
|
|
|
|
if (handlers != null) _handlers.addAll(handlers);
|
2016-10-14 03:07:34 +00:00
|
|
|
_method = method;
|
2016-10-12 17:58:32 +00:00
|
|
|
_name = name;
|
2016-10-11 18:53:32 +00:00
|
|
|
|
|
|
|
if (path is RegExp) {
|
|
|
|
_matcher = path;
|
|
|
|
_path = path.pattern;
|
|
|
|
} else {
|
2016-10-12 17:58:32 +00:00
|
|
|
_matcher = new RegExp(
|
|
|
|
_matcherify(path.toString().replaceAll(_straySlashes, '')));
|
|
|
|
_path = _pathified = _pathify(path.toString());
|
|
|
|
_resolver = new RegExp(_matcherify(
|
|
|
|
path.toString().replaceAll(_straySlashes, ''),
|
|
|
|
expand: false));
|
2016-10-11 18:53:32 +00:00
|
|
|
}
|
2016-10-15 00:45:22 +00:00
|
|
|
|
|
|
|
_parentResolver = new RegExp(_matcher.pattern.replaceAll(_rgxEnd, ''));
|
2016-10-11 18:53:32 +00:00
|
|
|
}
|
|
|
|
|
2016-10-14 03:07:34 +00:00
|
|
|
/// Splits a route path into a list of segments, and then
|
|
|
|
/// builds a hierarchy of off that.
|
|
|
|
///
|
|
|
|
/// This should generally be used instead of the original
|
|
|
|
/// Route constructor.
|
|
|
|
///
|
|
|
|
/// All children and handlers, as well as the method, will be
|
|
|
|
/// assigned to the last child route created.
|
|
|
|
///
|
|
|
|
/// The final child route is returned.
|
|
|
|
factory Route.build(Pattern path,
|
|
|
|
{Iterable<Route> children: const [],
|
2016-10-21 03:13:13 +00:00
|
|
|
bool debug: false,
|
2016-10-14 03:07:34 +00:00
|
|
|
Iterable handlers: const [],
|
|
|
|
method: "GET",
|
|
|
|
String name: null}) {
|
|
|
|
Route result;
|
|
|
|
|
2016-10-23 00:52:28 +00:00
|
|
|
if (path is RegExp) {
|
2016-11-23 09:13:42 +00:00
|
|
|
result = new Route(path, debug: debug, handlers: handlers, method: method, name: name);
|
2016-10-23 00:52:28 +00:00
|
|
|
} else {
|
|
|
|
final segments = path
|
|
|
|
.toString()
|
|
|
|
.split('/')
|
|
|
|
.where((str) => str.isNotEmpty)
|
|
|
|
.toList(growable: false);
|
2016-10-15 00:45:22 +00:00
|
|
|
|
2016-10-23 00:52:28 +00:00
|
|
|
if (segments.isEmpty) {
|
|
|
|
return new Route('/',
|
|
|
|
children: children,
|
|
|
|
debug: debug,
|
|
|
|
handlers: handlers,
|
|
|
|
method: method,
|
|
|
|
name: name);
|
|
|
|
}
|
2016-10-19 22:04:06 +00:00
|
|
|
|
2016-11-22 03:31:09 +00:00
|
|
|
var head = '';
|
|
|
|
|
2016-10-23 00:52:28 +00:00
|
|
|
for (int i = 0; i < segments.length; i++) {
|
|
|
|
final segment = segments[i];
|
2016-11-22 03:31:09 +00:00
|
|
|
head = (head + '/$segment').replaceAll(_straySlashes, '');
|
2016-10-23 00:52:28 +00:00
|
|
|
|
|
|
|
if (i == segments.length - 1) {
|
|
|
|
if (result == null) {
|
|
|
|
result = new Route(segment, debug: debug);
|
|
|
|
} else {
|
|
|
|
result = result.child(segment, debug: debug);
|
|
|
|
}
|
2016-10-19 22:04:06 +00:00
|
|
|
} else {
|
2016-10-23 00:52:28 +00:00
|
|
|
if (result == null) {
|
|
|
|
result = new Route(segment, debug: debug, method: "*");
|
|
|
|
} else {
|
|
|
|
result = result.child(segment, debug: debug, method: "*");
|
|
|
|
}
|
2016-10-19 22:04:06 +00:00
|
|
|
}
|
2016-11-22 03:31:09 +00:00
|
|
|
|
|
|
|
result._head = new RegExp(_matcherify(head).replaceAll(_rgxEnd, ''));
|
2016-10-19 22:04:06 +00:00
|
|
|
}
|
2016-10-14 03:07:34 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
result._children.addAll(children);
|
|
|
|
result._handlers.addAll(handlers);
|
|
|
|
result._method = method;
|
|
|
|
result._name = name;
|
|
|
|
|
2016-10-21 03:13:13 +00:00
|
|
|
return result..debug = debug;
|
2016-10-14 03:07:34 +00:00
|
|
|
}
|
|
|
|
|
2016-10-19 22:04:06 +00:00
|
|
|
/// Combines the paths and matchers of two [Route] instances, and creates a new instance.
|
2016-10-21 03:13:13 +00:00
|
|
|
factory Route.join(Route parent, Route child, {bool debug: false}) {
|
2016-10-12 17:58:32 +00:00
|
|
|
final String path1 = parent.path
|
|
|
|
.replaceAll(_rgxStart, '')
|
|
|
|
.replaceAll(_rgxEnd, '')
|
|
|
|
.replaceAll(_straySlashes, '');
|
|
|
|
final String path2 = child.path
|
|
|
|
.replaceAll(_rgxStart, '')
|
|
|
|
.replaceAll(_rgxEnd, '')
|
|
|
|
.replaceAll(_straySlashes, '');
|
|
|
|
final String pattern1 = parent.matcher.pattern
|
|
|
|
.replaceAll(_rgxEnd, '')
|
|
|
|
.replaceAll(_rgxStraySlashes, '');
|
|
|
|
final String pattern2 = child.matcher.pattern
|
|
|
|
.replaceAll(_rgxStart, '')
|
|
|
|
.replaceAll(_rgxStraySlashes, '');
|
2016-10-11 18:53:32 +00:00
|
|
|
|
2016-10-12 17:58:32 +00:00
|
|
|
final route = new Route('$path1/$path2',
|
2016-10-11 18:53:32 +00:00
|
|
|
children: child.children,
|
|
|
|
handlers: child.handlers,
|
|
|
|
method: child.method,
|
|
|
|
name: child.name);
|
|
|
|
|
2016-10-12 17:58:32 +00:00
|
|
|
String separator = (pattern1.isEmpty || pattern1 == '^') ? '' : '\\/';
|
|
|
|
|
2016-10-14 03:07:34 +00:00
|
|
|
parent._children.add(route
|
2016-10-12 17:58:32 +00:00
|
|
|
.._matcher = new RegExp('$pattern1$separator$pattern2')
|
2016-11-22 03:31:09 +00:00
|
|
|
.._head = new RegExp(_matcherify('$path1/$path2'.replaceAll(_straySlashes, '')).replaceAll(_rgxEnd, ''))
|
2016-10-19 22:04:06 +00:00
|
|
|
.._parent = parent
|
|
|
|
.._stub = child.matcher);
|
2016-10-21 03:13:13 +00:00
|
|
|
|
2016-10-19 22:04:06 +00:00
|
|
|
parent._printDebug(
|
2016-11-22 03:31:09 +00:00
|
|
|
"Joined '/$path1' and '/$path2', created head: ${route._head.pattern} and stub: ${route._stub.pattern}");
|
2016-10-14 03:07:34 +00:00
|
|
|
|
2016-10-21 03:13:13 +00:00
|
|
|
return route..debug = parent.debug || child.debug || debug;
|
|
|
|
}
|
|
|
|
|
2016-10-19 22:04:06 +00:00
|
|
|
/// Calls [addChild] on all given routes.
|
2016-10-12 17:58:32 +00:00
|
|
|
List<Route> addAll(Iterable<Route> routes, {bool join: true}) {
|
|
|
|
return routes.map((route) => addChild(route, join: join)).toList();
|
2016-10-11 18:53:32 +00:00
|
|
|
}
|
|
|
|
|
2016-10-19 22:04:06 +00:00
|
|
|
/// Adds the given route as a hierarchical child of this one.
|
2016-10-11 18:53:32 +00:00
|
|
|
Route addChild(Route route, {bool join: true}) {
|
2016-10-21 03:13:13 +00:00
|
|
|
Route created;
|
|
|
|
|
|
|
|
if (join) {
|
|
|
|
created = new Route.join(this, route);
|
|
|
|
} else {
|
|
|
|
_children.add(created = route.._parent = this);
|
|
|
|
}
|
|
|
|
|
2016-10-20 09:21:59 +00:00
|
|
|
return created..debug = debug;
|
2016-10-11 18:53:32 +00:00
|
|
|
}
|
|
|
|
|
2016-10-12 17:58:32 +00:00
|
|
|
/// Assigns a name to this route.
|
|
|
|
Route as(String name) => this.._name = name;
|
|
|
|
|
2016-10-19 22:04:06 +00:00
|
|
|
/// Creates a hierarchical child of this route with the given path.
|
2016-10-11 18:53:32 +00:00
|
|
|
Route child(Pattern path,
|
|
|
|
{Iterable<Route> children: const [],
|
2016-10-21 03:13:13 +00:00
|
|
|
bool debug: false,
|
2016-10-11 18:53:32 +00:00
|
|
|
Iterable handlers: const [],
|
|
|
|
String method: "GET",
|
|
|
|
String name: null}) {
|
2016-10-14 03:07:34 +00:00
|
|
|
final route = new Route.build(path,
|
2016-10-21 03:13:13 +00:00
|
|
|
children: children,
|
|
|
|
debug: debug,
|
|
|
|
handlers: handlers,
|
|
|
|
method: method,
|
|
|
|
name: name);
|
2016-10-11 18:53:32 +00:00
|
|
|
return addChild(route);
|
|
|
|
}
|
|
|
|
|
2016-10-12 17:58:32 +00:00
|
|
|
/// Generates a URI to this route with the given parameters.
|
|
|
|
String makeUri([Map<String, dynamic> params]) {
|
2016-11-23 09:13:42 +00:00
|
|
|
String result = _pathify(path);
|
2016-10-12 17:58:32 +00:00
|
|
|
if (params != null) {
|
|
|
|
for (String key in (params.keys)) {
|
|
|
|
result = result.replaceAll(
|
|
|
|
new RegExp(":$key" + r"\??"), params[key].toString());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return result.replaceAll("*", "");
|
|
|
|
}
|
|
|
|
|
2016-10-19 22:04:06 +00:00
|
|
|
/// Attempts to match a path against this route.
|
2016-10-12 17:58:32 +00:00
|
|
|
Match match(String path) =>
|
|
|
|
matcher.firstMatch(path.replaceAll(_straySlashes, ''));
|
|
|
|
|
|
|
|
/// Extracts route parameters from a given path.
|
|
|
|
Map parseParameters(String requestPath) {
|
|
|
|
Map result = {};
|
|
|
|
|
2016-10-13 20:50:27 +00:00
|
|
|
Iterable<String> values =
|
|
|
|
_parseParameters(requestPath.replaceAll(_straySlashes, ''));
|
2016-11-23 09:13:42 +00:00
|
|
|
|
|
|
|
_printDebug('Searched request path $requestPath and found these values: $values');
|
|
|
|
|
|
|
|
final pathString = _pathify(path).replaceAll(new RegExp('\/'), r'\/');
|
2016-10-13 20:50:27 +00:00
|
|
|
Iterable<Match> matches =
|
2016-11-23 09:13:42 +00:00
|
|
|
_param.allMatches(pathString);
|
|
|
|
_printDebug('All param names parsed in $pathString: ${matches.map((m) => m.group(0))}');
|
|
|
|
|
2016-10-15 00:45:22 +00:00
|
|
|
for (int i = 0; i < matches.length && i < values.length; i++) {
|
2016-10-12 17:58:32 +00:00
|
|
|
Match match = matches.elementAt(i);
|
|
|
|
String paramName = match.group(1);
|
|
|
|
String value = values.elementAt(i);
|
|
|
|
num numValue = num.parse(value, (_) => double.NAN);
|
|
|
|
if (!numValue.isNaN)
|
|
|
|
result[paramName] = numValue;
|
|
|
|
else
|
|
|
|
result[paramName] = value;
|
|
|
|
}
|
|
|
|
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
_parseParameters(String requestPath) sync* {
|
|
|
|
Match routeMatch = matcher.firstMatch(requestPath);
|
2016-10-15 00:45:22 +00:00
|
|
|
|
|
|
|
if (routeMatch != null)
|
|
|
|
for (int i = 1; i <= routeMatch.groupCount; i++)
|
|
|
|
yield routeMatch.group(i);
|
2016-10-12 17:58:32 +00:00
|
|
|
}
|
|
|
|
|
2016-10-19 22:04:06 +00:00
|
|
|
/// Finds the first route available within this hierarchy that can respond to the given path.
|
|
|
|
///
|
|
|
|
/// Can be used to navigate a route hierarchy like a file system.
|
2016-10-15 00:45:22 +00:00
|
|
|
Route resolve(String path, {bool filter(Route route), String fullPath}) {
|
2016-10-23 00:52:28 +00:00
|
|
|
_printDebug(
|
|
|
|
'Path to resolve: "/${path.replaceAll(_straySlashes, '')}", our matcher: ${matcher.pattern}');
|
2016-10-21 03:13:13 +00:00
|
|
|
bool _filter(route) {
|
|
|
|
if (filter == null) {
|
|
|
|
_printDebug('No filter provided, returning true for $route');
|
|
|
|
return true;
|
|
|
|
} else {
|
|
|
|
_printDebug('Running filter on $route');
|
|
|
|
final result = filter(route);
|
|
|
|
_printDebug('Filter result: $result');
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-10-15 00:45:22 +00:00
|
|
|
final _fullPath = fullPath ?? path;
|
2016-10-11 18:53:32 +00:00
|
|
|
|
2016-10-21 03:13:13 +00:00
|
|
|
if ((path.isEmpty || path == '.') && _filter(indexRoute)) {
|
2016-10-19 22:04:06 +00:00
|
|
|
// Try to find index
|
2016-10-21 03:13:13 +00:00
|
|
|
_printDebug('Empty path, resolving with indexRoute: $indexRoute');
|
|
|
|
return indexRoute;
|
2016-10-19 22:04:06 +00:00
|
|
|
} else if (path == '/') {
|
|
|
|
return absoluteParent.resolve('');
|
2016-10-12 17:58:32 +00:00
|
|
|
} else if (path.replaceAll(_straySlashes, '').isEmpty) {
|
|
|
|
for (Route route in children) {
|
|
|
|
final stub = route.path.replaceAll(this.path, '');
|
|
|
|
|
2016-10-21 03:13:13 +00:00
|
|
|
if ((stub == '/' || stub.isEmpty) && _filter(route))
|
|
|
|
return route.resolve('');
|
2016-10-12 17:58:32 +00:00
|
|
|
}
|
|
|
|
|
2016-10-21 03:13:13 +00:00
|
|
|
if (_filter(indexRoute)) {
|
|
|
|
_printDebug(
|
|
|
|
'Path "/$path" is technically empty, sending to indexRoute: $indexRoute');
|
|
|
|
return indexRoute;
|
|
|
|
} else
|
2016-10-12 17:58:32 +00:00
|
|
|
return null;
|
|
|
|
} else if (path == '..') {
|
|
|
|
if (parent != null)
|
|
|
|
return parent;
|
|
|
|
else
|
|
|
|
throw new RoutingException.orphan();
|
|
|
|
} else if (path.startsWith('/') &&
|
|
|
|
path.length > 1 &&
|
|
|
|
path[1] != '/' &&
|
|
|
|
absoluteParent != null) {
|
2016-10-15 00:45:22 +00:00
|
|
|
return absoluteParent.resolve(path.substring(1),
|
|
|
|
filter: _filter, fullPath: _fullPath);
|
2016-10-13 20:50:27 +00:00
|
|
|
} else if (matcher.hasMatch(path.replaceAll(_straySlashes, '')) ||
|
|
|
|
_resolver.hasMatch(path.replaceAll(_straySlashes, ''))) {
|
2016-10-21 03:13:13 +00:00
|
|
|
_printDebug(
|
|
|
|
'Path "/$path" matched our matcher, sending to indexRoute: $indexRoute');
|
|
|
|
return indexRoute;
|
2016-10-11 18:53:32 +00:00
|
|
|
} else {
|
2016-10-13 20:50:27 +00:00
|
|
|
final segments = path.split('/').where((str) => str.isNotEmpty).toList();
|
2016-10-19 22:04:06 +00:00
|
|
|
_printDebug('Segments: $segments on "/${this.path}"');
|
|
|
|
|
|
|
|
if (segments.isEmpty) {
|
2016-10-21 03:13:13 +00:00
|
|
|
_printDebug('Empty segments, sending to indexRoute: $indexRoute');
|
|
|
|
return indexRoute;
|
2016-10-19 22:04:06 +00:00
|
|
|
}
|
2016-10-11 18:53:32 +00:00
|
|
|
|
2016-10-12 17:58:32 +00:00
|
|
|
if (segments[0] == '..') {
|
|
|
|
if (parent != null)
|
2016-10-15 00:45:22 +00:00
|
|
|
return parent.resolve(segments.skip(1).join('/'),
|
|
|
|
filter: _filter, fullPath: _fullPath);
|
2016-10-12 17:58:32 +00:00
|
|
|
else
|
|
|
|
throw new RoutingException.orphan();
|
2016-10-13 20:50:27 +00:00
|
|
|
} else if (segments[0] == '.') {
|
2016-10-15 00:45:22 +00:00
|
|
|
return resolve(segments.skip(1).join('/'),
|
|
|
|
filter: _filter, fullPath: _fullPath);
|
2016-10-12 17:58:32 +00:00
|
|
|
}
|
|
|
|
|
2016-10-11 18:53:32 +00:00
|
|
|
for (Route route in children) {
|
|
|
|
final subPath = '${this.path}/${segments[0]}';
|
2016-10-19 22:04:06 +00:00
|
|
|
_printDebug(
|
2016-10-21 03:13:13 +00:00
|
|
|
'seg0: ${segments[0]}, stub: ${route._stub?.pattern}, path: $path, route.path: ${route.path}, route.matcher: ${route.matcher.pattern}, this.matcher: ${matcher.pattern}');
|
2016-10-11 18:53:32 +00:00
|
|
|
|
2016-10-12 17:58:32 +00:00
|
|
|
if (route.match(subPath) != null ||
|
|
|
|
route._resolver.firstMatch(subPath) != null) {
|
|
|
|
if (segments.length == 1 && _filter(route))
|
2016-10-21 03:13:13 +00:00
|
|
|
return route.resolve('');
|
2016-10-11 18:53:32 +00:00
|
|
|
else {
|
2016-10-15 00:45:22 +00:00
|
|
|
return route.resolve(segments.skip(1).join('/'),
|
|
|
|
filter: _filter,
|
|
|
|
fullPath: this.path.replaceAll(_straySlashes, '') +
|
|
|
|
'/' +
|
|
|
|
_fullPath.replaceAll(_straySlashes, ''));
|
|
|
|
}
|
2016-10-19 22:04:06 +00:00
|
|
|
} else if (route._stub != null && route._stub.hasMatch(segments[0])) {
|
2016-10-21 03:13:13 +00:00
|
|
|
if (segments.length == 1) {
|
|
|
|
_printDebug('Stub perhaps matches');
|
|
|
|
return route.resolve('');
|
|
|
|
} else {
|
|
|
|
_printDebug(
|
|
|
|
'Maybe stub matches. Sending remaining segments to $route');
|
|
|
|
return route.resolve(segments.skip(1).join('/'));
|
|
|
|
}
|
2016-10-15 00:45:22 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Try to match "subdirectory"
|
|
|
|
for (Route route in children) {
|
2016-10-19 22:04:06 +00:00
|
|
|
_printDebug(
|
2016-10-15 00:45:22 +00:00
|
|
|
'Trying to match subdir for $path; child ${route.path} on ${this.path}');
|
|
|
|
final match = route._parentResolver.firstMatch(path);
|
|
|
|
|
|
|
|
if (match != null) {
|
|
|
|
final subPath =
|
|
|
|
path.replaceFirst(match[0], '').replaceAll(_straySlashes, '');
|
2016-10-19 22:04:06 +00:00
|
|
|
_printDebug("Subdir path: $subPath");
|
2016-10-15 00:45:22 +00:00
|
|
|
|
|
|
|
for (Route child in route.children) {
|
|
|
|
final testPath = child.path
|
|
|
|
.replaceFirst(route.path, '')
|
|
|
|
.replaceAll(_straySlashes, '');
|
|
|
|
|
|
|
|
if (subPath == testPath &&
|
|
|
|
(child.match(_fullPath) != null ||
|
|
|
|
child._resolver.firstMatch(_fullPath) != null) &&
|
|
|
|
_filter(child)) {
|
2016-10-21 03:13:13 +00:00
|
|
|
return child.resolve('');
|
2016-10-15 00:45:22 +00:00
|
|
|
}
|
2016-10-11 18:53:32 +00:00
|
|
|
}
|
|
|
|
}
|
2016-10-21 03:13:13 +00:00
|
|
|
}
|
2016-10-11 18:53:32 +00:00
|
|
|
|
2016-10-12 17:58:32 +00:00
|
|
|
// Try to match the whole route, if nothing else works
|
|
|
|
for (Route route in children) {
|
2016-10-19 22:04:06 +00:00
|
|
|
_printDebug(
|
2016-10-15 00:45:22 +00:00
|
|
|
'Trying to match full $_fullPath for ${route.path} on ${this.path}');
|
|
|
|
if ((route.match(_fullPath) != null ||
|
|
|
|
route._resolver.firstMatch(_fullPath) != null) &&
|
2016-10-23 00:52:28 +00:00
|
|
|
_filter(route)) {
|
|
|
|
_printDebug('Matched full path!');
|
2016-10-21 03:13:13 +00:00
|
|
|
return route.resolve('');
|
2016-10-23 00:52:28 +00:00
|
|
|
} else if ((route.match('/$_fullPath') != null ||
|
2016-10-15 00:45:22 +00:00
|
|
|
route._resolver.firstMatch('/$_fullPath') != null) &&
|
2016-10-23 00:52:28 +00:00
|
|
|
_filter(route)) {
|
|
|
|
_printDebug('Matched full path (with a leading slash!)');
|
|
|
|
return route.resolve('');
|
|
|
|
}
|
2016-10-21 03:13:13 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Lastly, check to see if we have an index route to resolve with
|
|
|
|
if (indexRoute != this) {
|
|
|
|
_printDebug('Forwarding "/$path" to indexRoute');
|
|
|
|
return indexRoute.resolve(path);
|
2016-10-12 17:58:32 +00:00
|
|
|
}
|
|
|
|
|
2016-10-11 18:53:32 +00:00
|
|
|
return null;
|
|
|
|
}
|
|
|
|
}
|
2016-10-15 00:45:22 +00:00
|
|
|
|
|
|
|
@override
|
|
|
|
String toString() => "$method '$path' => ${handlers.length} handler(s)";
|
2016-10-11 18:53:32 +00:00
|
|
|
}
|