Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Params are parsed for all matching routes, not just the first route. #292

Open
pepkin88 opened this issue Aug 29, 2016 · 3 comments
Open

Comments

@pepkin88
Copy link

Considering the following program:

'use strict'
var koa = require('koa')
var Router = require('koa-router')
var app = koa()
var router = new Router()

function* checkIdParam(next) {
    var {id} = this.params
    console.log(this.path, this.params) // here it shows {id: 'new'} for the path '/new'
    if ('id' in this.params && !/^[1-9]\d*$/.test(id)) this.throw(404)
    yield* next
}

router.use(checkIdParam)
router.get('/new', function* () {
    this.body = 'new'
})
router.get('/:id/edit', function* () {
    this.body = 'edit'
})
router.get('/:id', function* () {
    this.redirect(`${this.path}/edit${this.search}`)
})

app.use(router.routes())
app.listen(8080)

when I go to http://localhost:8080/new the app matches the /new route and the /:id route. But it runs param setters far all the matching routes, instead of just the first one.
The result is, that although the path is /new (so it should match the /new route, since it is the first one), the checkIdParam middleware gets 'new' as an id param, which it shouldn't.

@GavinOsborn
Copy link

👍 I just ran into this myself when trying to implement some contextual middle-ware.
I would really like to see this fixed.

@jbielick
Copy link
Collaborator

This is actually caused by #231

here's a failing test case for 7.x

  it('only runs param functions for matched route', function (done) {

    var app = new Koa();
    var router = new Router();

    router
      .get('/new', function (ctx, next) {
        ctx.body = {ran: ''};
        ctx.body.ran += '/new';
        return next();
      })
      .get('/:id', function (ctx, next) {
        ctx.body.ran += '/:id';
        return next();
      });

    app.use(router.routes());

    app.use(function(ctx, next) {
      ctx.body.params = ctx.params;
      // console.log(ctx._matchedRoute) // == /:id
      return next();
    });

    request(http.createServer(app.callback()))
      .get('/new')
      .expect(200)
      .end(function (err, res) {
        expect(res.body.ran).to.eql('/new/:id');
        expect(res.body.params).not.to.have.property('id');
        done(err);
      });

  });

@jpravetz
Copy link

jpravetz commented Mar 30, 2017

I am seeing this same problem while working to add koa2 support to express-restify-mongoose. For that project I am testing with both koa-router and koa-better-router. The later of these is fully working at this point.

With REST, we have /customer/count and /customer/:id. koa-router v7.1.0 is using the last matching route, and middleware for both routes. It shouldn't be hard to fix this to take the first matching route and just the middleware for this route, and I'll try to fix in within the next couple of days.

However, I am wondering whether there is background (reasons) for the way it is now, and therefore a reason my PR wouldn't be accepted? I'd include the work along with my existing koa-router PR 334, which makes debugging these routes and spotting poorly assembled middleware chains a lot easier.

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Projects
None yet
Development

No branches or pull requests

4 participants