Skip to content

Commit c6a8fc8

Browse files
authoredJul 4, 2022
feat: add exclusive option (#129)
1 parent 3454a7d commit c6a8fc8

File tree

3 files changed

+54
-1
lines changed

3 files changed

+54
-1
lines changed
 

‎API.md

+1
Original file line numberDiff line numberDiff line change
@@ -34,6 +34,7 @@ Create a new router.
3434
| --- | --- | --- |
3535
| [opts] | <code>Object</code> | |
3636
| [opts.prefix] | <code>String</code> | prefix router paths |
37+
| [opts.exclusive] | <code>Boolean</code> | only run last matched route's controller when there are multiple matches |
3738

3839
**Example**
3940
Basic usage:

‎lib/router.js

+3-1
Original file line numberDiff line numberDiff line change
@@ -43,6 +43,7 @@ module.exports = Router;
4343
*
4444
* @alias module:koa-router
4545
* @param {Object=} opts
46+
* @param {Boolean=false} opts.exclusive only run last matched route's controller when there are multiple matches
4647
* @param {String=} opts.prefix prefix router paths
4748
* @constructor
4849
*/
@@ -60,6 +61,7 @@ function Router(opts = {}) {
6061
'POST',
6162
'DELETE'
6263
];
64+
this.exclusive = !!this.opts.exclusive;
6365

6466
this.params = {};
6567
this.stack = [];
@@ -364,7 +366,7 @@ Router.prototype.routes = Router.prototype.middleware = function () {
364366
ctx._matchedRouteName = mostSpecificLayer.name;
365367
}
366368

367-
layerChain = matchedLayers.reduce(function(memo, layer) {
369+
layerChain = (router.exclusive ? [mostSpecificLayer] : matchedLayers).reduce(function(memo, layer) {
368370
memo.push(function(ctx, next) {
369371
ctx.captures = layer.captures(path, ctx.captures);
370372
ctx.params = ctx.request.params = layer.params(path, ctx.captures, ctx.params);

‎test/lib/router.js

+50
Original file line numberDiff line numberDiff line change
@@ -251,6 +251,56 @@ describe('Router', function () {
251251
})
252252
});
253253

254+
it('runs multiple controllers when there are multiple matches', function (done) {
255+
const app = new Koa();
256+
const router = new Router();
257+
258+
router
259+
.get('users_single', '/users/:id(.*)', function (ctx, next) {
260+
ctx.body = { single: true };
261+
next();
262+
})
263+
.get('users_all', '/users/all', function (ctx, next) {
264+
ctx.body = Object.assign({}, ctx.body, { all: true });
265+
next();
266+
});
267+
268+
request(http.createServer(app.use(router.routes()).callback()))
269+
.get('/users/all')
270+
.expect(200)
271+
.end(function (err, res) {
272+
if (err) return done(err);
273+
expect(res.body).to.have.property('single', true);
274+
expect(res.body).to.have.property('all', true);
275+
done();
276+
})
277+
});
278+
279+
it('runs only the last match when the \'exclusive\' option is enabled', function (done) {
280+
const app = new Koa();
281+
const router = new Router({ exclusive: true });
282+
283+
router
284+
.get('users_single', '/users/:id(.*)', function (ctx, next) {
285+
ctx.body = { single: true };
286+
next();
287+
})
288+
.get('users_all', '/users/all', function (ctx, next) {
289+
ctx.body = Object.assign({}, ctx.body, { all: true });
290+
next();
291+
});
292+
293+
request(http.createServer(app.use(router.routes()).callback()))
294+
.get('/users/all')
295+
.expect(200)
296+
.end(function (err, res) {
297+
if (err) return done(err);
298+
expect(res.body).to.not.have.property('single');
299+
expect(res.body).to.have.property('all', true);
300+
done();
301+
})
302+
});
303+
254304
it('does not run subsequent middleware without calling next', function (done) {
255305
const app = new Koa();
256306
const router = new Router();

0 commit comments

Comments
 (0)
Please sign in to comment.