1 | ;
|
2 |
|
3 | // Small Express router that runs request handlers written as ES6 generators
|
4 | // using Node co. On top of that the router does a few useful things, including
|
5 | // some logging and error handling using a Node domain.
|
6 |
|
7 | var _ = require('underscore');
|
8 | var express = require('express');
|
9 | var domain = require('domain');
|
10 | var url = require('url');
|
11 | var yieldable = require('abacus-yieldable');
|
12 | var transform = require('abacus-transform');
|
13 |
|
14 | var toArray = _.toArray;
|
15 | var map = _.map;
|
16 | var extend = _.extend;
|
17 | var clone = _.clone;
|
18 |
|
19 | // Setup debug log
|
20 | var debug = require('abacus-debug')('abacus-router');
|
21 |
|
22 | // Convert a middleware function which can be a regular Express middleware or a
|
23 | // generator to a regular Express middleware function.
|
24 | var callbackify = function callbackify(m, trusted) {
|
25 |
|
26 | // If the callback is a regular function just use it as-is, otherwise it's
|
27 | // a generator and we need to wrap it using the co module
|
28 | var mfunc = yieldable.functioncb(m);
|
29 |
|
30 | // Return a middleware function. Middleware functions can be of the form
|
31 | // function(req, res, next) or function(err, req, res, next) so we need to
|
32 | // support both forms
|
33 | return function () {
|
34 | var next = arguments[arguments.length - 1];
|
35 | var res = arguments[1];
|
36 | var params = toArray(arguments).slice(0, arguments.length - 1);
|
37 |
|
38 | // Pass errors down the middleware stack, if the middleware is
|
39 | // un-trusted then we mark the error with bailout flag to trigger our
|
40 | // server bailout logic
|
41 | var error = function error(err, type) {
|
42 | debug('Route error - %s - %o', type, err);
|
43 | if (!trusted && !err.status && !err.statusCode) err.bailout = true;
|
44 | next(err);
|
45 | };
|
46 |
|
47 | // Call the middleware function
|
48 | try {
|
49 | mfunc.apply(undefined, params.concat([function (err, value) {
|
50 | if (err) error(err, 'generator error');else if (value) {
|
51 | // Store the returned value in the response, it'll be sent
|
52 | // by one of our Express middleware later down the
|
53 | // middleware stack
|
54 | res.value = value;
|
55 | next();
|
56 | }
|
57 | }]));
|
58 | } catch (exc) {
|
59 | error(exc, 'exception');
|
60 | }
|
61 | };
|
62 | };
|
63 |
|
64 | // Return an implementation of the router.use(path, middleware) function that supports
|
65 | // middleware implemented as generators in addition to regular callbacks
|
66 | var use = function use(original, trusted) {
|
67 | return function (path, m) {
|
68 | return typeof path === 'function' ? original.call(this, callbackify(path, trusted)) : original.call(this, path, callbackify(m, trusted));
|
69 | };
|
70 | };
|
71 |
|
72 | // Return an implementation of the router.route() function that supports middleware implemented
|
73 | // as generators in addition to regular callbacks
|
74 | var route = function route(original, trusted) {
|
75 | return function () {
|
76 | // Get the route
|
77 | var r = original.apply(this, arguments);
|
78 |
|
79 | // Monkey patch its HTTP methods
|
80 | map(['get', 'head', 'post', 'put', 'patch', 'delete', 'options', 'all', 'use'], function (method) {
|
81 | var f = r[method];
|
82 | r[method] = function () {
|
83 | var middleware = map(toArray(arguments), function (m) {
|
84 | return callbackify(m, trusted);
|
85 | });
|
86 | return f.apply(this, middleware);
|
87 | };
|
88 | });
|
89 | return r;
|
90 | };
|
91 | };
|
92 |
|
93 | // Return an Express middleware that uses a Node domain to run the middleware stack and
|
94 | // handle any errors not caught in async callbacks
|
95 | var catchall = function catchall(trusted) {
|
96 | return function (req, res, next) {
|
97 | var d = domain.create();
|
98 | d.on('error', function (err) {
|
99 | debug('Route domain error %o', err);
|
100 |
|
101 | // Pass the error down the middleware stack, if the router runs
|
102 | // un-trusted middleware then mark it with a bailout flag to
|
103 | // trigger our server bailout logic
|
104 | // Warning: mutating variable err
|
105 | if (!trusted && !err.status && !err.statusCode) err.bailout = true;
|
106 | next(err);
|
107 | });
|
108 |
|
109 | // Because req and res were created before this domain existed,
|
110 | // we need to explicitly add them. See the explanation of implicit
|
111 | // vs explicit binding in the Node domain docs.
|
112 | d.add(req);
|
113 | d.add(res);
|
114 |
|
115 | // Run the middleware stack in our new domain
|
116 | d.run(next);
|
117 | };
|
118 | };
|
119 |
|
120 | // Return an Express router middleware that works with generators
|
121 | var router = function router(trusted) {
|
122 | var r = express.Router();
|
123 |
|
124 | // Catch all errors down the middleware stack using a Node domain
|
125 | r.use(catchall(trusted));
|
126 |
|
127 | // Monkey patch the router function with our implementation of the use
|
128 | // and route functions
|
129 | r.use = use(r.use, trusted);
|
130 | r.route = route(r.route, trusted);
|
131 |
|
132 | return r;
|
133 | };
|
134 |
|
135 | // Return an express middleware that runs a batch of requests through the
|
136 | // given router
|
137 | var batch = function batch(routes) {
|
138 | return function (req, res, next) {
|
139 | if (req.method !== 'POST' || req.path !== '/batch') return next();
|
140 | debug('Handling batch request %o', req.body);
|
141 |
|
142 | // Run the batch of requests found in the body through the router
|
143 | transform.map(req.body, function (r, i, reqs, rcb) {
|
144 | // Setup an Express request representing the batched request
|
145 | var rreq = extend(clone(req), { method: r.method, url: url.resolve(req.url, r.uri), body: r.body });
|
146 |
|
147 | // Setup an Express response object that will capture the response
|
148 | var rres = extend(clone(res), {
|
149 | status: function status(s) {
|
150 | rres.statusCode = s;
|
151 | return rres;
|
152 | },
|
153 | send: function send(b) {
|
154 | rres.body = b;
|
155 | return rres.end();
|
156 | },
|
157 | json: function json(b) {
|
158 | rres.body = b;
|
159 | return rres.end();
|
160 | },
|
161 | end: function end() {
|
162 | rcb(undefined, { statusCode: rres.statusCode, body: rres.body });
|
163 | return rres;
|
164 | }
|
165 | });
|
166 |
|
167 | // Call the given router
|
168 | routes(rreq, rres, function () {
|
169 | if (rres.value) return rcb(undefined, rres.value);
|
170 | next();
|
171 | });
|
172 | }, function (err, bres) {
|
173 | if (err) return res.status(500).end();
|
174 |
|
175 | // Return the batch of results
|
176 | res.send(bres);
|
177 | });
|
178 | };
|
179 | };
|
180 |
|
181 | // Export our public functions
|
182 | module.exports = router;
|
183 | module.exports.batch = batch;
|
184 | //# sourceMappingURL=data:application/json;base64,{"version":3,"sources":["../src/index.js"],"names":[],"mappings":"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","file":"index.js","sourcesContent":["'use strict';\n\n// Small Express router that runs request handlers written as ES6 generators\n// using Node co. On top of that the router does a few useful things, including\n// some logging and error handling using a Node domain.\n\nconst _ = require('underscore');\nconst express = require('express');\nconst domain = require('domain');\nconst url = require('url');\nconst yieldable = require('abacus-yieldable');\nconst transform = require('abacus-transform');\n\nconst toArray = _.toArray;\nconst map = _.map;\nconst extend = _.extend;\nconst clone = _.clone;\n\n// Setup debug log\nconst debug = require('abacus-debug')('abacus-router');\n\n// Convert a middleware function which can be a regular Express middleware or a\n// generator to a regular Express middleware function.\nconst callbackify = (m, trusted) => {\n\n    // If the callback is a regular function just use it as-is, otherwise it's\n    // a generator and we need to wrap it using the co module\n    const mfunc = yieldable.functioncb(m);\n\n    // Return a middleware function. Middleware functions can be of the form\n    // function(req, res, next) or function(err, req, res, next) so we need to\n    // support both forms\n    return function() {\n        const next = arguments[arguments.length - 1];\n        const res = arguments[1];\n        const params = toArray(arguments).slice(0, arguments.length - 1);\n\n        // Pass errors down the middleware stack, if the middleware is\n        // un-trusted then we mark the error with bailout flag to trigger our\n        // server bailout logic\n        const error = (err, type) => {\n            debug('Route error - %s - %o', type, err);\n            if(!trusted && !err.status && !err.statusCode)\n                err.bailout = true;\n            next(err);\n        };\n\n        // Call the middleware function\n        try {\n            mfunc.apply(undefined, params.concat([(err, value) => {\n                if(err) error(err, 'generator error');\n\n                else if(value) {\n                    // Store the returned value in the response, it'll be sent\n                    // by one of our Express middleware later down the\n                    // middleware stack\n                    res.value = value;\n                    next();\n                }\n            }]));\n        }\n        catch(exc) {\n            error(exc, 'exception');\n        }\n    };\n};\n\n// Return an implementation of the router.use(path, middleware) function that supports\n// middleware implemented as generators in addition to regular callbacks\nconst use = (original, trusted) => {\n    return function(path, m) {\n        return typeof path === 'function' ? original.call(this, callbackify(path, trusted)) : original.call(this, path, callbackify(m, trusted));\n    };\n};\n\n// Return an implementation of the router.route() function that supports middleware implemented\n// as generators in addition to regular callbacks\nconst route = (original, trusted) => {\n    return function() {\n        // Get the route\n        const r = original.apply(this, arguments);\n\n        // Monkey patch its HTTP methods\n        map(['get', 'head', 'post', 'put', 'patch', 'delete', 'options', 'all', 'use'], (method) => {\n            const f = r[method];\n            r[method] = function() {\n                const middleware = map(toArray(arguments), function(m) {\n                    return callbackify(m, trusted);\n                });\n                return f.apply(this, middleware);\n            };\n        });\n        return r;\n    };\n};\n\n// Return an Express middleware that uses a Node domain to run the middleware stack and\n// handle any errors not caught in async callbacks\nconst catchall = (trusted) => {\n    return (req, res, next) => {\n        const d = domain.create();\n        d.on('error', (err) => {\n            debug('Route domain error %o', err);\n\n            // Pass the error down the middleware stack, if the router runs\n            // un-trusted middleware then mark it with a bailout flag to\n            // trigger our server bailout logic\n            // Warning: mutating variable err\n            if(!trusted && !err.status && !err.statusCode)\n                err.bailout = true;\n            next(err);\n        });\n\n        // Because req and res were created before this domain existed,\n        // we need to explicitly add them.  See the explanation of implicit\n        // vs explicit binding in the Node domain docs.\n        d.add(req);\n        d.add(res);\n\n        // Run the middleware stack in our new domain\n        d.run(next);\n    };\n};\n\n// Return an Express router middleware that works with generators\nconst router = (trusted) => {\n    const r = express.Router();\n\n    // Catch all errors down the middleware stack using a Node domain\n    r.use(catchall(trusted));\n\n    // Monkey patch the router function with our implementation of the use\n    // and route functions\n    r.use = use(r.use, trusted);\n    r.route = route(r.route, trusted);\n\n    return r;\n};\n\n// Return an express middleware that runs a batch of requests through the\n// given router\nconst batch = (routes) => {\n    return (req, res, next) => {\n        if(req.method !== 'POST' || req.path !== '/batch')\n            return next();\n        debug('Handling batch request %o', req.body);\n\n        // Run the batch of requests found in the body through the router\n        transform.map(req.body, (r, i, reqs, rcb) => {\n            // Setup an Express request representing the batched request\n            const rreq = extend(clone(req), { method: r.method, url: url.resolve(req.url, r.uri), body: r.body });\n\n            // Setup an Express response object that will capture the response\n            const rres = extend(clone(res), {\n                status: (s) => {\n                    rres.statusCode = s;\n                    return rres;\n                },\n                send: (b) => {\n                    rres.body = b;\n                    return rres.end();\n                },\n                json: (b) => {\n                    rres.body = b;\n                    return rres.end();\n                },\n                end: () => {\n                    rcb(undefined, { statusCode: rres.statusCode, body: rres.body });\n                    return rres;\n                }\n            });\n\n            // Call the given router\n            routes(rreq, rres, () => {\n                if(rres.value)\n                    return rcb(undefined, rres.value);\n                next();\n            });\n\n        }, (err, bres) => {\n            if(err) return res.status(500).end();\n\n            // Return the batch of results\n            res.send(bres);\n        });\n    };\n};\n\n// Export our public functions\nmodule.exports = router;\nmodule.exports.batch = batch;\n\n"]} |
\ | No newline at end of file |