如何使用可选的快速查询参数编写Mongoose查询?

Lud*_*sen 1 rest mongoose mongodb node.js express

我正在Express 4之上构建一个REST API.对于一个路径,GET /api/users我想要相应地向可选的 queryParams 查询我的用户集合.

因此,对API的调用可能如下所示 host:port/api/users?city=Berlin

在这种情况下name,省略并且不影响查询,而city必须匹配.

我设法通过以下hack来做到这一点.还有比这更好的方法吗?

Users.statics.customFilter = function(qP){
    return this.find({
        $and: [
            {$or: [{undefined: {$eq: qP.city}}, {'city': qP.city}]},
            {$or: [{undefined: {$eq: qP.name}}, {'name': qP.name}]}
        ]
    });

mongoose.model('User', Users);
Run Code Online (Sandbox Code Playgroud)

我正在从像这样的猫鼬模式中调用这个静态函数...

const User = mongoose.model('User');
app.get('/api/users', (req, res) => {
        User.customFilter(req.query)
        .exec((err, results) => {
            if (err) return next(err);
            res.json(results);
        });
    });
Run Code Online (Sandbox Code Playgroud)

rob*_*lep 5

在我看来,从本质上讲,User.find(req.query)应该做你想做的事:

/api/users?city=Berlin           ? User.find({ city : 'Berlin' })
/api/users?name=John             ? User.find({ name : 'John' })
/api/users?name=John&city=Berlin ? User.find({ city : 'Berlin', name : 'John' })
Run Code Online (Sandbox Code Playgroud)

当然,如果没有传递参数,你必须决定会发生什么(在上面的例子中,它将成为一个匹配所有用户的查询,这可能不是你想要的).

此外,您应该过滤,req.query以便它只包含您的架构中定义的字段.

  • 如果您需要的不仅仅是简单的相等性检查,您无论如何都必须处理传入的数据,或者允许使用更复杂的查询格式([`mongo-rql`](https://www.npmjs.com/ package/mongo-rql)、[`monquery`](https://www.npmjs.com/package/monquery)、[`mongo-querystring`](https://www.npmjs.com/package/mongo-请求参数))。 (2认同)