2016-07-15 19:34:24 +00:00
|
|
|
/**
|
|
|
|
* Module dependencies.
|
|
|
|
* @private
|
|
|
|
*/
|
|
|
|
|
2016-07-14 13:14:16 +00:00
|
|
|
import HTTP_METHODS from './methods';
|
2016-06-26 10:10:37 +00:00
|
|
|
import Middleware from './middleware';
|
|
|
|
|
2016-07-15 19:34:24 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Route object.
|
|
|
|
* @private
|
|
|
|
*/
|
|
|
|
|
2016-06-26 10:10:37 +00:00
|
|
|
class Route {
|
2016-07-15 19:34:24 +00:00
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Initialize the route.
|
|
|
|
*
|
|
|
|
* @private
|
|
|
|
*/
|
|
|
|
|
2016-06-26 10:10:37 +00:00
|
|
|
constructor(router, uriPart, method, middleware) {
|
|
|
|
this.router = router;
|
|
|
|
this.uriPart = uriPart;
|
|
|
|
this.method = method;
|
|
|
|
this.middleware = middleware;
|
2016-06-30 21:35:20 +00:00
|
|
|
this.visited = false;
|
2016-06-26 10:10:37 +00:00
|
|
|
}
|
|
|
|
|
2016-07-15 19:34:24 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Return route's uri.
|
|
|
|
*
|
|
|
|
* @private
|
|
|
|
*/
|
|
|
|
|
2016-06-26 10:10:37 +00:00
|
|
|
get uri() {
|
2016-07-09 22:58:03 +00:00
|
|
|
if (!this.uriPart && !this.method) {
|
|
|
|
return undefined;
|
|
|
|
}
|
|
|
|
|
2016-06-26 10:10:37 +00:00
|
|
|
if (this.uriPart instanceof RegExp) {
|
|
|
|
return this.uriPart;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (this.router.baseUri instanceof RegExp) {
|
|
|
|
return this.router.baseUri;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (this.router.baseUri && this.uriPart) {
|
|
|
|
return (this.router.baseUri.trim() + this.uriPart.trim()).replace(/\/{2,}/, '/');
|
|
|
|
}
|
|
|
|
|
|
|
|
if (this.router.baseUri) {
|
|
|
|
return this.router.baseUri.trim();
|
|
|
|
}
|
|
|
|
|
|
|
|
return this.uriPart;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-07-15 19:34:24 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Router object.
|
|
|
|
* @public
|
|
|
|
*/
|
|
|
|
|
2016-06-26 10:10:37 +00:00
|
|
|
export default class Router {
|
2016-07-15 19:34:24 +00:00
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Initialize the router.
|
|
|
|
*
|
|
|
|
* @private
|
|
|
|
*/
|
|
|
|
|
2016-07-09 00:58:41 +00:00
|
|
|
constructor(uri) {
|
|
|
|
if (uri) {
|
|
|
|
this._baseUri = uri;
|
2016-06-26 10:10:37 +00:00
|
|
|
}
|
2016-06-30 21:35:20 +00:00
|
|
|
this._routes = [];
|
2016-06-26 10:10:37 +00:00
|
|
|
}
|
|
|
|
|
2016-07-15 19:34:24 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Do some checks and set _baseUri.
|
|
|
|
*
|
|
|
|
* @private
|
|
|
|
*/
|
|
|
|
|
2016-07-08 21:29:40 +00:00
|
|
|
set baseUri(uri) {
|
2016-07-09 00:58:41 +00:00
|
|
|
if (!uri) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2016-07-08 21:29:40 +00:00
|
|
|
if (!this._baseUri) {
|
|
|
|
this._baseUri = uri;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (this._baseUri instanceof RegExp) {
|
|
|
|
throw new TypeError(`the router already contains a regexp uri ${this._baseUri.toString()} It cannot be mixed with ${uri.toString()}`);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (uri instanceof RegExp) {
|
|
|
|
throw new TypeError(`the router already contains an uri ${this._baseUri.toString()} It cannot be mixed with regexp ${uri.toString()}`);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-07-15 19:34:24 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Return router's _baseUri.
|
|
|
|
*
|
|
|
|
* @private
|
|
|
|
*/
|
|
|
|
|
2016-07-08 21:29:40 +00:00
|
|
|
get baseUri() {
|
|
|
|
return this._baseUri;
|
|
|
|
}
|
|
|
|
|
2016-07-15 19:34:24 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Add a route to the router.
|
|
|
|
*
|
|
|
|
* @private
|
|
|
|
*/
|
|
|
|
|
2016-06-26 10:10:37 +00:00
|
|
|
_add(route) {
|
2016-06-30 21:35:20 +00:00
|
|
|
this._routes.push(route);
|
2016-06-26 10:10:37 +00:00
|
|
|
return this;
|
|
|
|
}
|
|
|
|
|
2016-07-15 19:34:24 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Gather routes from routers filtered by _uri_ and HTTP _method_.
|
|
|
|
*
|
|
|
|
* @private
|
|
|
|
*/
|
|
|
|
|
2016-06-30 21:35:20 +00:00
|
|
|
routes(uri, method) {
|
|
|
|
return this._routes.filter((route) => {
|
2016-07-09 22:58:03 +00:00
|
|
|
if (!route.uri && !route.method) {
|
|
|
|
return true;
|
|
|
|
}
|
2016-06-26 10:10:37 +00:00
|
|
|
if (route.method !== method) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!route.uri) {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2016-07-02 13:45:54 +00:00
|
|
|
let uriToCheck = uri;
|
|
|
|
|
|
|
|
//remove query string from uri to test
|
|
|
|
const questionMarkIndex = uriToCheck.indexOf('?');
|
|
|
|
if (questionMarkIndex >= 0) {
|
|
|
|
uriToCheck = uriToCheck.slice(0, questionMarkIndex);
|
|
|
|
}
|
|
|
|
|
|
|
|
//remove anchor from uri to test
|
|
|
|
const hashIndex = uriToCheck.indexOf('#');
|
|
|
|
if (hashIndex >= 0) {
|
|
|
|
uriToCheck = uriToCheck.slice(0, hashIndex);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (route.uri instanceof RegExp) {
|
|
|
|
return uriToCheck.match(route.uri);
|
|
|
|
}
|
|
|
|
|
|
|
|
return route.uri === uriToCheck;
|
2016-06-26 10:10:37 +00:00
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2016-07-15 19:34:24 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Gather visited routes from routers.
|
|
|
|
*
|
|
|
|
* @private
|
|
|
|
*/
|
|
|
|
|
2016-06-30 21:35:20 +00:00
|
|
|
visited() {
|
|
|
|
return this._routes.filter((route) => {
|
|
|
|
return route.visited;
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2016-07-15 19:34:24 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Use the given middleware function or object on this router.
|
|
|
|
*
|
|
|
|
* // middleware function
|
|
|
|
* router.use((req, res, next) => {console.log('Hello')});
|
|
|
|
*
|
|
|
|
* // middleware object
|
|
|
|
* router.use(new Middleware());
|
|
|
|
*
|
|
|
|
* @param {Middleware|Function} middleware or fn
|
|
|
|
* @return {Router} for chaining
|
|
|
|
*
|
|
|
|
* @public
|
|
|
|
*/
|
|
|
|
|
2016-07-09 22:58:03 +00:00
|
|
|
use(middleware) {
|
|
|
|
if (!(middleware instanceof Middleware) && (typeof middleware !== 'function') ) {
|
|
|
|
throw new TypeError('use method takes at least a middleware');
|
|
|
|
}
|
|
|
|
|
|
|
|
this._add(new Route(this, undefined, undefined, middleware));
|
|
|
|
|
|
|
|
return this;
|
|
|
|
}
|
|
|
|
|
2016-07-15 19:34:24 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Use the given middleware function or object on this router for
|
|
|
|
* all HTTP methods.
|
|
|
|
*
|
|
|
|
* // middleware function
|
|
|
|
* router.all((req, res, next) => {console.log('Hello')});
|
|
|
|
*
|
|
|
|
* // middleware object
|
|
|
|
* router.all(new Middleware());
|
|
|
|
*
|
|
|
|
* @param {Middleware|Function} middleware or fn
|
|
|
|
* @return {Router} for chaining
|
|
|
|
*
|
|
|
|
* @public
|
|
|
|
*/
|
|
|
|
|
2016-06-26 10:10:37 +00:00
|
|
|
all(...args) {
|
|
|
|
if (args.length === 0) {
|
2016-07-09 22:58:03 +00:00
|
|
|
throw new TypeError('use all method takes at least a middleware');
|
2016-06-26 10:10:37 +00:00
|
|
|
}
|
|
|
|
let middleware;
|
|
|
|
|
|
|
|
if (args.length === 1) {
|
|
|
|
[middleware,] = args;
|
|
|
|
} else {
|
|
|
|
[, middleware,] = args;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!(middleware instanceof Middleware) && (typeof middleware !== 'function') ) {
|
2016-07-09 22:58:03 +00:00
|
|
|
throw new TypeError('use all method takes at least a middleware');
|
2016-06-26 10:10:37 +00:00
|
|
|
}
|
|
|
|
|
2016-07-14 13:14:16 +00:00
|
|
|
for (const method of HTTP_METHODS) {
|
2016-06-26 10:10:37 +00:00
|
|
|
this[method.toLowerCase()](...args);
|
|
|
|
}
|
|
|
|
return this;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-07-14 13:14:16 +00:00
|
|
|
for (const method of HTTP_METHODS) {
|
2016-07-15 19:34:24 +00:00
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Use the given middleware function or object, with optional _uri_ on
|
|
|
|
* HTTP methods: get, post, put, delete...
|
|
|
|
* Default _uri_ is "/".
|
|
|
|
*
|
|
|
|
* // middleware function will be applied on path "/"
|
|
|
|
* router.get((req, res, next) => {console.log('Hello')});
|
|
|
|
*
|
|
|
|
* // middleware object will be applied on path "/" and
|
|
|
|
* router.get(new Middleware());
|
|
|
|
*
|
|
|
|
* // middleware function will be applied on path "/user"
|
|
|
|
* router.post('/user', (req, res, next) => {console.log('Hello')});
|
|
|
|
*
|
|
|
|
* // middleware object will be applied on path "/user" and
|
|
|
|
* router.post('/user', new Middleware());
|
|
|
|
*
|
|
|
|
* @param {String} uri or setting
|
|
|
|
* @param {Middleware|Function} middleware or fn
|
|
|
|
* @return {Router} for chaining
|
|
|
|
* @public
|
|
|
|
*/
|
|
|
|
|
2016-06-26 10:10:37 +00:00
|
|
|
const methodName = method.toLowerCase();
|
|
|
|
Router.prototype[methodName] = function(...args) {
|
|
|
|
if (args.length === 0) {
|
2016-07-02 13:45:54 +00:00
|
|
|
throw new TypeError(`use ${methodName} method takes at least a middleware`);
|
2016-06-26 10:10:37 +00:00
|
|
|
}
|
|
|
|
let uri, middleware;
|
|
|
|
|
|
|
|
if (args.length === 1) {
|
|
|
|
[middleware,] = args;
|
|
|
|
} else {
|
|
|
|
[uri, middleware,] = args;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!(middleware instanceof Middleware) && (typeof middleware !== 'function') ) {
|
2016-07-02 13:45:54 +00:00
|
|
|
throw new TypeError(`use ${methodName} method takes at least a middleware`);
|
|
|
|
}
|
|
|
|
|
2016-07-08 21:29:40 +00:00
|
|
|
if (uri && this._baseUri && this._baseUri instanceof RegExp) {
|
2016-07-09 22:58:03 +00:00
|
|
|
throw new TypeError('router contains a regexp cannot mix with route uri/regexp');
|
2016-06-26 10:10:37 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
this._add(new Route(this, uri, method, middleware));
|
|
|
|
|
|
|
|
return this;
|
2016-07-09 22:58:03 +00:00
|
|
|
};
|
2016-06-26 10:10:37 +00:00
|
|
|
}
|