2016-07-15 19:34:24 +00:00
|
|
|
|
/**
|
|
|
|
|
* Module dependencies.
|
|
|
|
|
* @private
|
|
|
|
|
*/
|
|
|
|
|
|
2016-07-14 13:14:16 +00:00
|
|
|
|
import HTTP_METHODS from './methods';
|
|
|
|
|
import Settings from './settings';
|
2016-07-09 22:58:03 +00:00
|
|
|
|
import Router, {Route} from './router';
|
|
|
|
|
import Middleware from './middleware';
|
|
|
|
|
|
2016-07-15 19:34:24 +00:00
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* Application class.
|
|
|
|
|
*/
|
|
|
|
|
|
2016-07-09 22:58:03 +00:00
|
|
|
|
export default class Application {
|
2016-07-15 19:34:24 +00:00
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* Initialize the application.
|
|
|
|
|
*
|
|
|
|
|
* - setup default configuration
|
|
|
|
|
*
|
|
|
|
|
* @private
|
|
|
|
|
*/
|
|
|
|
|
|
2016-07-09 22:58:03 +00:00
|
|
|
|
constructor() {
|
|
|
|
|
this.routers = [];
|
2016-07-17 11:37:35 +00:00
|
|
|
|
this.isDOMLoaded = false;
|
|
|
|
|
this.isDOMReady = false;
|
2016-07-14 13:14:16 +00:00
|
|
|
|
this.settings = new Settings();
|
2016-07-09 22:58:03 +00:00
|
|
|
|
}
|
|
|
|
|
|
2016-07-15 19:34:24 +00:00
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* Assign `setting` to `val`, or return `setting`'s value.
|
|
|
|
|
*
|
|
|
|
|
* app.set('foo', 'bar');
|
|
|
|
|
* app.set('foo');
|
|
|
|
|
* // => "bar"
|
|
|
|
|
*
|
|
|
|
|
* @param {String} setting
|
|
|
|
|
* @param {*} [val]
|
|
|
|
|
* @return {app} for chaining
|
|
|
|
|
* @public
|
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
|
set(...args) {
|
|
|
|
|
// get behaviour
|
|
|
|
|
if (args.length === 1) {
|
|
|
|
|
const name = [args];
|
|
|
|
|
return this.settings.get(name);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// set behaviour
|
|
|
|
|
const [name, value] = args;
|
2016-07-14 13:14:16 +00:00
|
|
|
|
this.settings.set(name, value);
|
2016-07-15 19:34:24 +00:00
|
|
|
|
|
|
|
|
|
return this;
|
2016-07-09 22:58:03 +00:00
|
|
|
|
}
|
|
|
|
|
|
2016-07-15 19:34:24 +00:00
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* Listen for DOM initialization and history state changes.
|
|
|
|
|
*
|
|
|
|
|
* The callback function is called once the DOM has
|
|
|
|
|
* the `document.readyState` equals to 'interactive'.
|
|
|
|
|
*
|
|
|
|
|
* app.listen(()=> {
|
|
|
|
|
* console.log('App is listening requests');
|
|
|
|
|
* console.log('DOM is ready!');
|
|
|
|
|
* });
|
|
|
|
|
*
|
|
|
|
|
*
|
|
|
|
|
* @param {Function} callback
|
|
|
|
|
* @public
|
|
|
|
|
*/
|
|
|
|
|
|
2016-07-09 22:58:03 +00:00
|
|
|
|
listen(callback) {
|
2016-07-17 11:37:35 +00:00
|
|
|
|
window.onbeforeunload = () => {
|
|
|
|
|
this._callMiddlewareExited();
|
|
|
|
|
};
|
2016-07-09 22:58:03 +00:00
|
|
|
|
|
2016-07-17 11:37:35 +00:00
|
|
|
|
window.onpopstate = (event) => {
|
|
|
|
|
if (event.state) {
|
|
|
|
|
const {response, request} = event.state;
|
|
|
|
|
const currentRoutes = this._routes(request.uri, request.method);
|
2016-07-09 22:58:03 +00:00
|
|
|
|
|
|
|
|
|
this._callMiddlewareEntered(currentRoutes, request);
|
2016-07-17 11:37:35 +00:00
|
|
|
|
this._callMiddlewareUpdated(currentRoutes, request, response);
|
|
|
|
|
}
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
document.onreadystatechange = () => {
|
|
|
|
|
const request = {method: 'GET', uri: window.location.pathname + window.location.search};
|
|
|
|
|
const response = {status: 200, statusText: 'OK'};
|
|
|
|
|
const currentRoutes = this._routes();
|
|
|
|
|
// DOM state
|
|
|
|
|
if (document.readyState === 'loading' && !this.isDOMLoaded) {
|
|
|
|
|
this.isDOMLoaded = true;
|
|
|
|
|
this._callMiddlewareEntered(currentRoutes, request);
|
|
|
|
|
} else if (document.readyState === 'interactive' && !this.isDOMReady) {
|
|
|
|
|
if (!this.isDOMLoaded) {
|
|
|
|
|
this.isDOMLoaded = true;
|
2016-07-09 22:58:03 +00:00
|
|
|
|
this._callMiddlewareEntered(currentRoutes, request);
|
|
|
|
|
}
|
2016-07-17 11:37:35 +00:00
|
|
|
|
this.isDOMReady = true;
|
2016-07-09 22:58:03 +00:00
|
|
|
|
this._callMiddlewareUpdated(currentRoutes, request, response);
|
|
|
|
|
if (callback) {
|
|
|
|
|
callback(request, response);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
};
|
|
|
|
|
}
|
|
|
|
|
|
2016-07-15 19:34:24 +00:00
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* Returns a new `Router` instance for the _uri_.
|
|
|
|
|
* See the Router api docs for details.
|
|
|
|
|
*
|
|
|
|
|
* app.route('/');
|
|
|
|
|
* // => new Router instance
|
|
|
|
|
*
|
|
|
|
|
* @param {String} uri
|
|
|
|
|
* @return {Router} for chaining
|
|
|
|
|
*
|
|
|
|
|
* @public
|
|
|
|
|
*/
|
|
|
|
|
|
2016-07-09 22:58:03 +00:00
|
|
|
|
route(uri) {
|
|
|
|
|
const router = new Router(uri);
|
|
|
|
|
this.routers.push(router);
|
|
|
|
|
return router;
|
|
|
|
|
}
|
|
|
|
|
|
2016-07-15 19:34:24 +00:00
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* Use the given middleware function or object, with optional _uri_.
|
|
|
|
|
* Default _uri_ is "/".
|
|
|
|
|
*
|
|
|
|
|
* // middleware function will be applied on path "/"
|
|
|
|
|
* app.use((req, res, next) => {console.log('Hello')});
|
|
|
|
|
*
|
|
|
|
|
* // middleware object will be applied on path "/"
|
|
|
|
|
* app.use(new Middleware());
|
|
|
|
|
*
|
|
|
|
|
* @param {String} uri
|
|
|
|
|
* @param {Middleware|Function} middleware or fn
|
|
|
|
|
* @return {app} for chaining
|
|
|
|
|
*
|
|
|
|
|
* @public
|
|
|
|
|
*/
|
|
|
|
|
|
2016-07-09 22:58:03 +00:00
|
|
|
|
use(...args) {
|
|
|
|
|
if (args.length === 0) {
|
|
|
|
|
throw new TypeError('use method takes at least a middleware or a router');
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
let baseUri, middleware, router, which;
|
|
|
|
|
|
|
|
|
|
if (args.length === 1) {
|
|
|
|
|
[which,] = args;
|
|
|
|
|
} else {
|
|
|
|
|
[baseUri, which,] = args;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (!(which instanceof Middleware) && (typeof which !== 'function') && !(which instanceof Router)) {
|
|
|
|
|
throw new TypeError('use method takes at least a middleware or a router');
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (which instanceof Router) {
|
|
|
|
|
router = which;
|
|
|
|
|
router.baseUri = baseUri;
|
|
|
|
|
} else {
|
|
|
|
|
middleware = which;
|
|
|
|
|
router = new Router(baseUri);
|
2016-07-14 13:14:16 +00:00
|
|
|
|
for (const method of HTTP_METHODS) {
|
2016-07-09 22:58:03 +00:00
|
|
|
|
router[method.toLowerCase()](middleware);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
this.routers.push(router);
|
2016-07-15 19:34:24 +00:00
|
|
|
|
|
|
|
|
|
return this;
|
2016-07-09 22:58:03 +00:00
|
|
|
|
}
|
|
|
|
|
|
2016-07-15 19:34:24 +00:00
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* Gather routes from all routers filtered by _uri_ and HTTP _method_.
|
|
|
|
|
* See Router#routes() documentation for details.
|
|
|
|
|
*
|
|
|
|
|
* @private
|
|
|
|
|
*/
|
|
|
|
|
|
2016-07-17 11:37:35 +00:00
|
|
|
|
_routes(uri=window.location.pathname + window.location.search, method='GET') {
|
2016-07-09 22:58:03 +00:00
|
|
|
|
const currentRoutes = [];
|
|
|
|
|
for (const router of this.routers) {
|
|
|
|
|
const routes = router.routes(uri, method);
|
|
|
|
|
currentRoutes.push(...routes);
|
|
|
|
|
}
|
2016-07-17 11:37:35 +00:00
|
|
|
|
|
2016-07-09 22:58:03 +00:00
|
|
|
|
return currentRoutes;
|
|
|
|
|
}
|
|
|
|
|
|
2016-07-15 19:34:24 +00:00
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* Call `Middleware#entered` on _currentRoutes_.
|
|
|
|
|
* Invoked before sending ajax request or when DOM
|
|
|
|
|
* is loading (document.readyState === 'loading').
|
|
|
|
|
*
|
|
|
|
|
* @private
|
|
|
|
|
*/
|
|
|
|
|
|
2016-07-09 22:58:03 +00:00
|
|
|
|
_callMiddlewareEntered(currentRoutes, request) {
|
|
|
|
|
for (const route of currentRoutes) {
|
|
|
|
|
if (route.middleware.entered) {
|
|
|
|
|
route.middleware.entered(request);
|
|
|
|
|
}
|
|
|
|
|
if (route.middleware.next && !route.middleware.next()) {
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2016-07-15 19:34:24 +00:00
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* Call `Middleware#updated` or middleware function on _currentRoutes_.
|
|
|
|
|
* Invoked on ajax request responding or on DOM ready
|
|
|
|
|
* (document.readyState === 'interactive').
|
|
|
|
|
*
|
|
|
|
|
* @private
|
|
|
|
|
*/
|
|
|
|
|
|
2016-07-09 22:58:03 +00:00
|
|
|
|
_callMiddlewareUpdated(currentRoutes, request, response) {
|
|
|
|
|
for (const route of currentRoutes) {
|
|
|
|
|
route.visited = request;
|
|
|
|
|
// calls middleware updated method
|
|
|
|
|
if (route.middleware.updated) {
|
|
|
|
|
route.middleware.updated(request, response);
|
|
|
|
|
if (route.middleware.next && !route.middleware.next()) {
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
} else {
|
|
|
|
|
// calls middleware method
|
|
|
|
|
let breakMiddlewareLoop = true;
|
|
|
|
|
const next = () => {
|
|
|
|
|
breakMiddlewareLoop = false;
|
|
|
|
|
};
|
|
|
|
|
route.middleware(request, response, next);
|
|
|
|
|
if (breakMiddlewareLoop) {
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2016-07-15 19:34:24 +00:00
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* Call `Middleware#exited` on _currentRoutes_.
|
|
|
|
|
* Invoked before sending a new ajax request or before DOM unloading.
|
|
|
|
|
*
|
|
|
|
|
* @private
|
|
|
|
|
*/
|
|
|
|
|
|
2016-07-09 22:58:03 +00:00
|
|
|
|
_callMiddlewareExited() {
|
|
|
|
|
// calls middleware exited method
|
|
|
|
|
for (const router of this.routers) {
|
|
|
|
|
const routes = router.visited();
|
|
|
|
|
for (const route of routes) {
|
|
|
|
|
if (route.middleware.exited) {
|
|
|
|
|
route.middleware.exited(route.visited);
|
|
|
|
|
route.visited = null;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2016-07-15 19:34:24 +00:00
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* Call `Middleware#failed` or middleware function on _currentRoutes_.
|
|
|
|
|
* Invoked when ajax request fails.
|
|
|
|
|
*
|
|
|
|
|
* @private
|
|
|
|
|
*/
|
|
|
|
|
|
2016-07-09 22:58:03 +00:00
|
|
|
|
_callMiddlewareFailed(currentRoutes, request, response) {
|
|
|
|
|
for (const route of currentRoutes) {
|
|
|
|
|
// calls middleware failed method
|
|
|
|
|
if (route.middleware.failed) {
|
|
|
|
|
route.middleware.failed(request, response);
|
|
|
|
|
if (route.middleware.next && !route.middleware.next()) {
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
} else {
|
|
|
|
|
// calls middleware method
|
|
|
|
|
let breakMiddlewareLoop = true;
|
|
|
|
|
const next = () => {
|
|
|
|
|
breakMiddlewareLoop = false;
|
|
|
|
|
};
|
|
|
|
|
route.middleware(request, response, next);
|
|
|
|
|
if (breakMiddlewareLoop) {
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2016-07-15 19:34:24 +00:00
|
|
|
|
|
|
|
|
|
/**
|
2016-07-17 11:37:35 +00:00
|
|
|
|
* Make an ajax request. Manage History#pushState if history object set.
|
2016-07-15 19:34:24 +00:00
|
|
|
|
*
|
|
|
|
|
* @private
|
|
|
|
|
*/
|
|
|
|
|
|
2016-07-17 11:37:35 +00:00
|
|
|
|
_fetch({method, uri, headers, data, history}, resolve, reject) {
|
2016-07-14 13:14:16 +00:00
|
|
|
|
const httpMethodTransformer = this.get(`http ${method} transformer`);
|
|
|
|
|
if (httpMethodTransformer) {
|
|
|
|
|
uri = httpMethodTransformer.uri ? httpMethodTransformer.uri({uri, headers, data}) : uri;
|
|
|
|
|
headers = httpMethodTransformer.headers ? httpMethodTransformer.headers({uri, headers, data}) : headers;
|
|
|
|
|
data = httpMethodTransformer.data ? httpMethodTransformer.data({uri, headers, data}) : data;
|
|
|
|
|
}
|
|
|
|
|
|
2016-07-09 22:58:03 +00:00
|
|
|
|
// calls middleware exited method
|
|
|
|
|
this._callMiddlewareExited();
|
|
|
|
|
|
|
|
|
|
// gathers all routes impacted by the uri
|
|
|
|
|
const currentRoutes = this._routes(uri, method);
|
|
|
|
|
|
|
|
|
|
// calls middleware entered method
|
|
|
|
|
this._callMiddlewareEntered(currentRoutes, {method, uri, headers, data});
|
|
|
|
|
|
|
|
|
|
// invokes http request
|
2016-07-14 13:14:16 +00:00
|
|
|
|
this.settings.get('http requester').fetch({method, uri, headers, data},
|
2016-07-09 22:58:03 +00:00
|
|
|
|
(request, response) => {
|
2016-07-17 11:37:35 +00:00
|
|
|
|
if (history) {
|
|
|
|
|
if (history.state) {
|
|
|
|
|
response.historyState = history.state;
|
|
|
|
|
}
|
|
|
|
|
history.state = {request, response};
|
|
|
|
|
window.history.pushState(history.state, history.title, history.uri);
|
|
|
|
|
}
|
2016-07-09 22:58:03 +00:00
|
|
|
|
this._callMiddlewareUpdated(currentRoutes, request, response);
|
|
|
|
|
if (resolve) {
|
|
|
|
|
resolve(request, response);
|
|
|
|
|
}
|
|
|
|
|
},
|
|
|
|
|
(request, response) => {
|
|
|
|
|
this._callMiddlewareFailed(currentRoutes, request, response);
|
|
|
|
|
if (reject) {
|
|
|
|
|
reject(request, response);
|
|
|
|
|
}
|
|
|
|
|
});
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2016-07-14 13:14:16 +00:00
|
|
|
|
HTTP_METHODS.reduce((reqProto, method) => {
|
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 "/"
|
|
|
|
|
* app.get((req, res, next) => {console.log('Hello')});
|
|
|
|
|
*
|
|
|
|
|
* // middleware object will be applied on path "/" and
|
|
|
|
|
* app.get(new Middleware());
|
|
|
|
|
*
|
|
|
|
|
* // get a setting value
|
|
|
|
|
* app.set('foo', 'bar');
|
|
|
|
|
* app.get('foo');
|
|
|
|
|
* // => "bar"
|
|
|
|
|
*
|
|
|
|
|
* @param {String} uri or setting
|
|
|
|
|
* @param {Middleware|Function} middleware or fn
|
|
|
|
|
* @return {app} for chaining
|
|
|
|
|
* @public
|
|
|
|
|
*/
|
|
|
|
|
|
2016-07-14 13:14:16 +00:00
|
|
|
|
const middlewareMethodName = method.toLowerCase();
|
|
|
|
|
reqProto[middlewareMethodName] = function(...args) {
|
|
|
|
|
if (middlewareMethodName === 'get') {
|
|
|
|
|
if (args.length === 0) {
|
|
|
|
|
throw new TypeError(`${middlewareMethodName} method takes at least a string or a middleware`);
|
|
|
|
|
} else if (args.length === 1) {
|
|
|
|
|
const [name] = args;
|
|
|
|
|
if (typeof name === 'string') {
|
|
|
|
|
return this.settings.get(name);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
} else if (args.length === 0) {
|
|
|
|
|
throw new TypeError(`${middlewareMethodName} method takes at least a middleware`);
|
2016-07-09 22:58:03 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
let baseUri, middleware, which;
|
|
|
|
|
|
|
|
|
|
if (args.length === 1) {
|
|
|
|
|
[which,] = args;
|
|
|
|
|
} else {
|
|
|
|
|
[baseUri, which,] = args;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (!(which instanceof Middleware) && (typeof which !== 'function')) {
|
2016-07-14 13:14:16 +00:00
|
|
|
|
throw new TypeError(`${middlewareMethodName} method takes at least a middleware`);
|
2016-07-09 22:58:03 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
const router = new Router();
|
|
|
|
|
middleware = which;
|
2016-07-14 13:14:16 +00:00
|
|
|
|
router[middlewareMethodName](baseUri, middleware);
|
2016-07-09 22:58:03 +00:00
|
|
|
|
|
|
|
|
|
this.routers.push(router);
|
2016-07-15 19:34:24 +00:00
|
|
|
|
|
|
|
|
|
return this;
|
2016-07-09 22:58:03 +00:00
|
|
|
|
};
|
|
|
|
|
|
2016-07-15 19:34:24 +00:00
|
|
|
|
/**
|
|
|
|
|
* Ajax request (get, post, put, delete...).
|
|
|
|
|
*
|
|
|
|
|
* // HTTP GET method
|
|
|
|
|
* httpGet('/route1');
|
2016-07-17 11:37:35 +00:00
|
|
|
|
*
|
|
|
|
|
* // HTTP GET method
|
2016-07-15 19:34:24 +00:00
|
|
|
|
* httpGet({uri: '/route1', data: {'p1': 'val1'});
|
|
|
|
|
* // uri invoked => /route1?p1=val1
|
|
|
|
|
*
|
2016-07-17 11:37:35 +00:00
|
|
|
|
* // HTTP GET method with browser history management
|
|
|
|
|
* httpGet({uri: '/api/users', history: {state: {foo: "bar"}, title: 'users page', uri: '/view/users'});
|
|
|
|
|
*
|
|
|
|
|
* Samples above can be applied on other HTTP methods.
|
2016-07-15 19:34:24 +00:00
|
|
|
|
*
|
2016-07-17 11:37:35 +00:00
|
|
|
|
* @param {String|Object} uri or object containing uri, http headers, data, history
|
2016-07-15 19:34:24 +00:00
|
|
|
|
* @param {Function} success callback
|
|
|
|
|
* @param {Function} failure callback
|
|
|
|
|
* @public
|
|
|
|
|
*/
|
2016-07-09 22:58:03 +00:00
|
|
|
|
const httpMethodName = 'http'+method.charAt(0).toUpperCase() + method.slice(1).toLowerCase();
|
|
|
|
|
reqProto[httpMethodName] = function(request, resolve, reject) {
|
2016-07-17 11:37:35 +00:00
|
|
|
|
let {uri, headers, data, history} = request;
|
2016-07-09 22:58:03 +00:00
|
|
|
|
if (!uri) {
|
|
|
|
|
uri = request;
|
|
|
|
|
}
|
|
|
|
|
return this._fetch({
|
|
|
|
|
uri,
|
|
|
|
|
method,
|
|
|
|
|
headers,
|
2016-07-17 11:37:35 +00:00
|
|
|
|
data,
|
|
|
|
|
history
|
2016-07-09 22:58:03 +00:00
|
|
|
|
}, resolve, reject);
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
return reqProto;
|
2016-07-08 21:29:40 +00:00
|
|
|
|
}, Application.prototype);
|