Hapijs和Joi:使用在线验证查询参数:'forbidden'

时间:2014-10-06 10:49:30

标签: javascript node.js hapijs joi

我的目标是为帖子创建索引路线。用户应该能够指定一些查询参数(即标签,类型),但不能指定其他参数。澄清:

这没关系:

/posts
/posts?tags=food
/posts?type=regular&tags=stackoverflow

这不行:

/posts?title=Hello

这是hapi pack config:

servers: [
        {
            host: 'localhost',
            port: 3000,
            options: {
                labels: ["api"],
                validation: {
                    abortEarly: false,
                    presence: 'forbidden'
                }
            }
        }
    ],

请注意presence: forbidden选项。

这是路线配置:

handler: function (request, reply) {
    Post.find(request.query, function (err, posts) {
        if(err) {
            console.log(err);
        }

        reply(posts);
    });
},
validate: {
    query: {
        type: Joi.string().optional(),
        tags: Joi.string().optional()
    }

}

我的想法是验证应该允许typetags参数的任何子集(包括空查询)。但是,在做出任何允许的请求后,我收到以下错误:

{
    "statusCode": 400,
    "error": "Bad Request",
    "message": "value is not allowed",
    "validation": {
        "source": "query",
        "keys": [
            "value"
        ]
    }
}

为什么?当然没有名为value的密钥。如何使验证按照我想要的方式运行?

1 个答案:

答案 0 :(得分:8)

如果您定义了非类型架构对象,Joi会将其内部转换为object()类型。所以这个架构:

var schema = {
    type: Joi.string().optional(),
    tags: Joi.string().optional()
};

变为:

var schema = Joi.object().keys({
    type: Joi.string().optional(),
    tags: Joi.string().optional()
});

因为您在服务器设置中将presence设置为forbidden,所以它将应用于对象类型,因此架构变为:

var schema = Joi.object().forbidden().keys({
    type: Joi.string().optional(),
    tags: Joi.string().optional()
});

如您所见,它将主要对象标记为禁止,不允许除undefined之外的任何值:

var Joi = require('joi');

var schema = Joi.object().forbidden().keys({
    type: Joi.string().optional(),
    tags: Joi.string().optional()
});

var value = {};

Joi.validate(value, schema, { presence: 'forbidden' }, function (err, value) {

    console.log(err);
});

输出:

{ [ValidationError: value is not allowed]
  name: 'ValidationError',
  details: 
   [ { message: 'value is not allowed',
       path: 'value',
       type: 'any.unknown' } ],
  _object: {},
  annotate: [Function] }

所以你需要做的是标记主要对象requiredoptional以覆盖forbidden

validate: {
    query: Joi.object().required().keys({
        type: Joi.string().optional(),
        tags: Joi.string().optional()
    })
}