|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
'use strict'; |
|
|
|
|
|
|
|
|
|
|
|
|
|
var Route = require('./route'); |
|
var Layer = require('./layer'); |
|
var methods = require('methods'); |
|
var mixin = require('utils-merge'); |
|
var debug = require('debug')('express:router'); |
|
var deprecate = require('depd')('express'); |
|
var flatten = require('array-flatten'); |
|
var parseUrl = require('parseurl'); |
|
var setPrototypeOf = require('setprototypeof') |
|
|
|
|
|
|
|
|
|
|
|
|
|
var objectRegExp = /^\[object (\S+)\]$/; |
|
var slice = Array.prototype.slice; |
|
var toString = Object.prototype.toString; |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
var proto = module.exports = function(options) { |
|
var opts = options || {}; |
|
|
|
function router(req, res, next) { |
|
router.handle(req, res, next); |
|
} |
|
|
|
|
|
setPrototypeOf(router, proto) |
|
|
|
router.params = {}; |
|
router._params = []; |
|
router.caseSensitive = opts.caseSensitive; |
|
router.mergeParams = opts.mergeParams; |
|
router.strict = opts.strict; |
|
router.stack = []; |
|
|
|
return router; |
|
}; |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
proto.param = function param(name, fn) { |
|
|
|
if (typeof name === 'function') { |
|
deprecate('router.param(fn): Refactor to use path params'); |
|
this._params.push(name); |
|
return; |
|
} |
|
|
|
|
|
var params = this._params; |
|
var len = params.length; |
|
var ret; |
|
|
|
if (name[0] === ':') { |
|
deprecate('router.param(' + JSON.stringify(name) + ', fn): Use router.param(' + JSON.stringify(name.slice(1)) + ', fn) instead') |
|
name = name.slice(1) |
|
} |
|
|
|
for (var i = 0; i < len; ++i) { |
|
if (ret = params[i](name, fn)) { |
|
fn = ret; |
|
} |
|
} |
|
|
|
|
|
|
|
if ('function' !== typeof fn) { |
|
throw new Error('invalid param() call for ' + name + ', got ' + fn); |
|
} |
|
|
|
(this.params[name] = this.params[name] || []).push(fn); |
|
return this; |
|
}; |
|
|
|
|
|
|
|
|
|
|
|
|
|
proto.handle = function handle(req, res, out) { |
|
var self = this; |
|
|
|
debug('dispatching %s %s', req.method, req.url); |
|
|
|
var idx = 0; |
|
var protohost = getProtohost(req.url) || '' |
|
var removed = ''; |
|
var slashAdded = false; |
|
var sync = 0 |
|
var paramcalled = {}; |
|
|
|
|
|
|
|
var options = []; |
|
|
|
|
|
var stack = self.stack; |
|
|
|
|
|
var parentParams = req.params; |
|
var parentUrl = req.baseUrl || ''; |
|
var done = restore(out, req, 'baseUrl', 'next', 'params'); |
|
|
|
|
|
req.next = next; |
|
|
|
|
|
if (req.method === 'OPTIONS') { |
|
done = wrap(done, function(old, err) { |
|
if (err || options.length === 0) return old(err); |
|
sendOptionsResponse(res, options, old); |
|
}); |
|
} |
|
|
|
|
|
req.baseUrl = parentUrl; |
|
req.originalUrl = req.originalUrl || req.url; |
|
|
|
next(); |
|
|
|
function next(err) { |
|
var layerError = err === 'route' |
|
? null |
|
: err; |
|
|
|
|
|
if (slashAdded) { |
|
req.url = req.url.slice(1) |
|
slashAdded = false; |
|
} |
|
|
|
|
|
if (removed.length !== 0) { |
|
req.baseUrl = parentUrl; |
|
req.url = protohost + removed + req.url.slice(protohost.length) |
|
removed = ''; |
|
} |
|
|
|
|
|
if (layerError === 'router') { |
|
setImmediate(done, null) |
|
return |
|
} |
|
|
|
|
|
if (idx >= stack.length) { |
|
setImmediate(done, layerError); |
|
return; |
|
} |
|
|
|
|
|
if (++sync > 100) { |
|
return setImmediate(next, err) |
|
} |
|
|
|
|
|
var path = getPathname(req); |
|
|
|
if (path == null) { |
|
return done(layerError); |
|
} |
|
|
|
|
|
var layer; |
|
var match; |
|
var route; |
|
|
|
while (match !== true && idx < stack.length) { |
|
layer = stack[idx++]; |
|
match = matchLayer(layer, path); |
|
route = layer.route; |
|
|
|
if (typeof match !== 'boolean') { |
|
|
|
layerError = layerError || match; |
|
} |
|
|
|
if (match !== true) { |
|
continue; |
|
} |
|
|
|
if (!route) { |
|
|
|
continue; |
|
} |
|
|
|
if (layerError) { |
|
|
|
match = false; |
|
continue; |
|
} |
|
|
|
var method = req.method; |
|
var has_method = route._handles_method(method); |
|
|
|
|
|
if (!has_method && method === 'OPTIONS') { |
|
appendMethods(options, route._options()); |
|
} |
|
|
|
|
|
if (!has_method && method !== 'HEAD') { |
|
match = false; |
|
} |
|
} |
|
|
|
|
|
if (match !== true) { |
|
return done(layerError); |
|
} |
|
|
|
|
|
if (route) { |
|
req.route = route; |
|
} |
|
|
|
|
|
req.params = self.mergeParams |
|
? mergeParams(layer.params, parentParams) |
|
: layer.params; |
|
var layerPath = layer.path; |
|
|
|
|
|
self.process_params(layer, paramcalled, req, res, function (err) { |
|
if (err) { |
|
next(layerError || err) |
|
} else if (route) { |
|
layer.handle_request(req, res, next) |
|
} else { |
|
trim_prefix(layer, layerError, layerPath, path) |
|
} |
|
|
|
sync = 0 |
|
}); |
|
} |
|
|
|
function trim_prefix(layer, layerError, layerPath, path) { |
|
if (layerPath.length !== 0) { |
|
|
|
if (layerPath !== path.slice(0, layerPath.length)) { |
|
next(layerError) |
|
return |
|
} |
|
|
|
|
|
var c = path[layerPath.length] |
|
if (c && c !== '/' && c !== '.') return next(layerError) |
|
|
|
|
|
|
|
debug('trim prefix (%s) from url %s', layerPath, req.url); |
|
removed = layerPath; |
|
req.url = protohost + req.url.slice(protohost.length + removed.length) |
|
|
|
|
|
if (!protohost && req.url[0] !== '/') { |
|
req.url = '/' + req.url; |
|
slashAdded = true; |
|
} |
|
|
|
|
|
req.baseUrl = parentUrl + (removed[removed.length - 1] === '/' |
|
? removed.substring(0, removed.length - 1) |
|
: removed); |
|
} |
|
|
|
debug('%s %s : %s', layer.name, layerPath, req.originalUrl); |
|
|
|
if (layerError) { |
|
layer.handle_error(layerError, req, res, next); |
|
} else { |
|
layer.handle_request(req, res, next); |
|
} |
|
} |
|
}; |
|
|
|
|
|
|
|
|
|
|
|
|
|
proto.process_params = function process_params(layer, called, req, res, done) { |
|
var params = this.params; |
|
|
|
|
|
var keys = layer.keys; |
|
|
|
|
|
if (!keys || keys.length === 0) { |
|
return done(); |
|
} |
|
|
|
var i = 0; |
|
var name; |
|
var paramIndex = 0; |
|
var key; |
|
var paramVal; |
|
var paramCallbacks; |
|
var paramCalled; |
|
|
|
|
|
|
|
function param(err) { |
|
if (err) { |
|
return done(err); |
|
} |
|
|
|
if (i >= keys.length ) { |
|
return done(); |
|
} |
|
|
|
paramIndex = 0; |
|
key = keys[i++]; |
|
name = key.name; |
|
paramVal = req.params[name]; |
|
paramCallbacks = params[name]; |
|
paramCalled = called[name]; |
|
|
|
if (paramVal === undefined || !paramCallbacks) { |
|
return param(); |
|
} |
|
|
|
|
|
if (paramCalled && (paramCalled.match === paramVal |
|
|| (paramCalled.error && paramCalled.error !== 'route'))) { |
|
|
|
req.params[name] = paramCalled.value; |
|
|
|
|
|
return param(paramCalled.error); |
|
} |
|
|
|
called[name] = paramCalled = { |
|
error: null, |
|
match: paramVal, |
|
value: paramVal |
|
}; |
|
|
|
paramCallback(); |
|
} |
|
|
|
|
|
function paramCallback(err) { |
|
var fn = paramCallbacks[paramIndex++]; |
|
|
|
|
|
paramCalled.value = req.params[key.name]; |
|
|
|
if (err) { |
|
|
|
paramCalled.error = err; |
|
param(err); |
|
return; |
|
} |
|
|
|
if (!fn) return param(); |
|
|
|
try { |
|
fn(req, res, paramCallback, paramVal, key.name); |
|
} catch (e) { |
|
paramCallback(e); |
|
} |
|
} |
|
|
|
param(); |
|
}; |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
proto.use = function use(fn) { |
|
var offset = 0; |
|
var path = '/'; |
|
|
|
|
|
|
|
if (typeof fn !== 'function') { |
|
var arg = fn; |
|
|
|
while (Array.isArray(arg) && arg.length !== 0) { |
|
arg = arg[0]; |
|
} |
|
|
|
|
|
if (typeof arg !== 'function') { |
|
offset = 1; |
|
path = fn; |
|
} |
|
} |
|
|
|
var callbacks = flatten(slice.call(arguments, offset)); |
|
|
|
if (callbacks.length === 0) { |
|
throw new TypeError('Router.use() requires a middleware function') |
|
} |
|
|
|
for (var i = 0; i < callbacks.length; i++) { |
|
var fn = callbacks[i]; |
|
|
|
if (typeof fn !== 'function') { |
|
throw new TypeError('Router.use() requires a middleware function but got a ' + gettype(fn)) |
|
} |
|
|
|
|
|
debug('use %o %s', path, fn.name || '<anonymous>') |
|
|
|
var layer = new Layer(path, { |
|
sensitive: this.caseSensitive, |
|
strict: false, |
|
end: false |
|
}, fn); |
|
|
|
layer.route = undefined; |
|
|
|
this.stack.push(layer); |
|
} |
|
|
|
return this; |
|
}; |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
proto.route = function route(path) { |
|
var route = new Route(path); |
|
|
|
var layer = new Layer(path, { |
|
sensitive: this.caseSensitive, |
|
strict: this.strict, |
|
end: true |
|
}, route.dispatch.bind(route)); |
|
|
|
layer.route = route; |
|
|
|
this.stack.push(layer); |
|
return route; |
|
}; |
|
|
|
|
|
methods.concat('all').forEach(function(method){ |
|
proto[method] = function(path){ |
|
var route = this.route(path) |
|
route[method].apply(route, slice.call(arguments, 1)); |
|
return this; |
|
}; |
|
}); |
|
|
|
|
|
function appendMethods(list, addition) { |
|
for (var i = 0; i < addition.length; i++) { |
|
var method = addition[i]; |
|
if (list.indexOf(method) === -1) { |
|
list.push(method); |
|
} |
|
} |
|
} |
|
|
|
|
|
function getPathname(req) { |
|
try { |
|
return parseUrl(req).pathname; |
|
} catch (err) { |
|
return undefined; |
|
} |
|
} |
|
|
|
|
|
function getProtohost(url) { |
|
if (typeof url !== 'string' || url.length === 0 || url[0] === '/') { |
|
return undefined |
|
} |
|
|
|
var searchIndex = url.indexOf('?') |
|
var pathLength = searchIndex !== -1 |
|
? searchIndex |
|
: url.length |
|
var fqdnIndex = url.slice(0, pathLength).indexOf('://') |
|
|
|
return fqdnIndex !== -1 |
|
? url.substring(0, url.indexOf('/', 3 + fqdnIndex)) |
|
: undefined |
|
} |
|
|
|
|
|
function gettype(obj) { |
|
var type = typeof obj; |
|
|
|
if (type !== 'object') { |
|
return type; |
|
} |
|
|
|
|
|
return toString.call(obj) |
|
.replace(objectRegExp, '$1'); |
|
} |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
function matchLayer(layer, path) { |
|
try { |
|
return layer.match(path); |
|
} catch (err) { |
|
return err; |
|
} |
|
} |
|
|
|
|
|
function mergeParams(params, parent) { |
|
if (typeof parent !== 'object' || !parent) { |
|
return params; |
|
} |
|
|
|
|
|
var obj = mixin({}, parent); |
|
|
|
|
|
if (!(0 in params) || !(0 in parent)) { |
|
return mixin(obj, params); |
|
} |
|
|
|
var i = 0; |
|
var o = 0; |
|
|
|
|
|
while (i in params) { |
|
i++; |
|
} |
|
|
|
while (o in parent) { |
|
o++; |
|
} |
|
|
|
|
|
for (i--; i >= 0; i--) { |
|
params[i + o] = params[i]; |
|
|
|
|
|
if (i < o) { |
|
delete params[i]; |
|
} |
|
} |
|
|
|
return mixin(obj, params); |
|
} |
|
|
|
|
|
function restore(fn, obj) { |
|
var props = new Array(arguments.length - 2); |
|
var vals = new Array(arguments.length - 2); |
|
|
|
for (var i = 0; i < props.length; i++) { |
|
props[i] = arguments[i + 2]; |
|
vals[i] = obj[props[i]]; |
|
} |
|
|
|
return function () { |
|
|
|
for (var i = 0; i < props.length; i++) { |
|
obj[props[i]] = vals[i]; |
|
} |
|
|
|
return fn.apply(this, arguments); |
|
}; |
|
} |
|
|
|
|
|
function sendOptionsResponse(res, options, next) { |
|
try { |
|
var body = options.join(','); |
|
res.set('Allow', body); |
|
res.send(body); |
|
} catch (err) { |
|
next(err); |
|
} |
|
} |
|
|
|
|
|
function wrap(old, fn) { |
|
return function proxy() { |
|
var args = new Array(arguments.length + 1); |
|
|
|
args[0] = old; |
|
for (var i = 0, len = arguments.length; i < len; i++) { |
|
args[i + 1] = arguments[i]; |
|
} |
|
|
|
fn.apply(this, args); |
|
}; |
|
} |
|
|