user.schema.js
var Schema = require('mongoose').Schema;
var uniqueValidator = require('mongoose-unique-validator');
var _ = require('lodash');
var userSchema = new Schema({
local: {
username: String, // should exist when local exists
role: String,
hashedPassword: { type: String, select: false }
},
facebook: {
id: String,
token: { type: String, select: false }
},
twitter: {
id: String,
token: { type: String, select: false }
},
google: {
id: String,
token: { type: String, select: false }
}
});
userSchema.path('local').validate(function(local) {
var empty = _.isEmpty(local);
if (empty) {
return true;
}
else if (!empty && local.username) {
return true;
}
else if (!empty && !local.username) {
return false;
}
}, 'Local auth requires a username.');
module.exports = userSchema;
当username
不为空时,我试图验证local
是否存在。 IE浏览器。使用本地身份验证时,username
应该存在。
// should validate
user = {
local: {
username: 'foo';
hashedPassword: 'sfsdfs'
}
};
// shouldn't validate
user = {
local: {
hashedPassword: 'sdfsdfs'
}
};
// should validate (because local isn't being used)
user = {
local: {},
facebook {
...
}
};
我收到此错误:
/Users/azerner/code/mean-starter/server/api/users/user.schema.js:51
userSchema.path('local').validate(function(local) {
^
TypeError: Cannot read property 'validate' of undefined
您似乎无法获得path
个对象。我了解到here Schemas有一个paths
属性。当我console.log(userSchema.paths)
时:
{ 'local.username':
{ enumValues: [],
regExp: null,
path: 'local.username',
instance: 'String',
validators: [],
setters: [],
getters: [],
options: { type: [Function: String] },
_index: null },
'local.role':
{ enumValues: [],
regExp: null,
path: 'local.role',
instance: 'String',
validators: [],
setters: [],
getters: [],
options: { type: [Function: String] },
_index: null },
'local.hashedPassword':
{ enumValues: [],
regExp: null,
path: 'local.hashedPassword',
instance: 'String',
validators: [],
setters: [],
getters: [],
options: { type: [Function: String], select: false },
_index: null,
selected: false },
'facebook.id':
{ enumValues: [],
regExp: null,
path: 'facebook.id',
instance: 'String',
validators: [],
setters: [],
getters: [],
options: { type: [Function: String] },
_index: null },
'facebook.token':
{ enumValues: [],
regExp: null,
path: 'facebook.token',
instance: 'String',
validators: [],
setters: [],
getters: [],
options: { type: [Function: String], select: false },
_index: null,
selected: false },
'twitter.id':
{ enumValues: [],
regExp: null,
path: 'twitter.id',
instance: 'String',
validators: [],
setters: [],
getters: [],
options: { type: [Function: String] },
_index: null },
'twitter.token':
{ enumValues: [],
regExp: null,
path: 'twitter.token',
instance: 'String',
validators: [],
setters: [],
getters: [],
options: { type: [Function: String], select: false },
_index: null,
selected: false },
'google.id':
{ enumValues: [],
regExp: null,
path: 'google.id',
instance: 'String',
validators: [],
setters: [],
getters: [],
options: { type: [Function: String] },
_index: null },
'google.token':
{ enumValues: [],
regExp: null,
path: 'google.token',
instance: 'String',
validators: [],
setters: [],
getters: [],
options: { type: [Function: String], select: false },
_index: null,
selected: false },
_id:
{ path: '_id',
instance: 'ObjectID',
validators: [],
setters: [ [Function: resetId] ],
getters: [],
options: { type: [Object], auto: true },
_index: null,
defaultValue: [Function: defaultId] } }
所以似乎存在local.username
和facebook.token
之类的路径,而不是"顶级"像local
和facebook
这样的路径。
如果我尝试验证local.username
路径,它就不会像我想要的那样工作。
userSchema.path('local.username').validate(function(username) {
return !!username
}, 'Local auth requires a username.');
仅在local.username
存在时才会应用验证。我想验证它是否存在。因此,当它不存在时,验证不会被应用,因此它被认为是有效的并被保存。
我也尝试了以下方法,但结果与local.username
方法相同(当用户名不存在时,验证不会被点击,并且标记为有效)
var Schema = require('mongoose').Schema;
var uniqueValidator = require('mongoose-unique-validator');
var _ = require('lodash');
var userSchema = new Schema({
local: {
username: {
type: String,
validate: [validateUsernameRequired, 'Local auth requires a username.']
},
role: String,
hashedPassword: { type: String, select: false }
},
facebook: {
id: String,
token: { type: String, select: false }
},
twitter: {
id: String,
token: { type: String, select: false }
},
google: {
id: String,
token: { type: String, select: false }
}
});
function validateUsernameRequired(username) {
return !!username;
}
module.exports = userSchema;
答案 0 :(得分:1)
Adam,为什么不尝试预先验证的钩子,它会有条件地将错误传递给下一个函数。我认为这将为您提供您所寻求的灵活性。如果它不起作用,请告诉我。
例如
schema.pre('validate', function(next) {
if(/*your error case */){ next('validation error text') }
else { next() }
})
这会导致猫鼬向试图保存文档的人发回ValidationError
。
答案 1 :(得分:0)
看起来您正在尝试创建自定义验证。不确定您是否实现了所需的一切。它看起来像这样:
// make sure every value is equal to "something"
function validator (val) {
return val == 'something';
}
new Schema({ name: { type: String, validate: validator }});
// with a custom error message
var custom = [validator, 'Uh oh, {PATH} does not equal "something".']
new Schema({ name: { type: String, validate: custom }});
// adding many validators at a time
var many = [
{ validator: validator, msg: 'uh oh' }
, { validator: anotherValidator, msg: 'failed' }
]
new Schema({ name: { type: String, validate: many }});
// or utilizing SchemaType methods directly:
var schema = new Schema({ name: 'string' });
schema.path('name').validate(validator, 'validation of `{PATH}` failed with
value `{VALUE}`');