检查mongoose验证错误的最佳方法

时间:2013-01-18 20:20:43

标签: node.js mongoose

我的usermodel

有两个validation函数
User.schema.path('email').validate(function(value, respond) {
  User.findOne({email: value}, function(err, user) {
    if(err) throw err;
    if(user) return respond(false);
    respond(true);
  });
}, 'EMAIL_EXISTS');

username

相同
User.schema.path('username').validate(function(value, respond) {
  User.findOne({username: value}, function(err, user) {
    if(err) throw err;
    if(user) return respond(false);
    respond(true);
  });
}, 'USERNAME_TAKEN');

他们以下列格式返回错误

{ message: 'Validation failed',
  name: 'ValidationError',
  errors: 
    { username: 
      { message: 'Validator "USERNAME_TAKEN" failed for path username',
        name: 'ValidatorError',
        path: 'username',
        type: 'USERNAME_TAKEN' } } }

email路径的错误类似。是否有更智能的方法来检查这些错误而不是以下?

if (err && err.errors && err.errors.username) { ... }

这有点难看。

8 个答案:

答案 0 :(得分:29)

从技术上讲,您必须首先检查错误名称,因为并非所有错误都以相同的方式处理。然后,根据错误名称,您必须检查特定属性,作为ValidationError附带的errors属性。

此外,您将字段名称放在错误类型中,这是多余的,最好使用相同的错误类型,因为在错误检查过程中,您还将获得字段名称。

所以你的代码可能是这样的:

User.schema.path('email').validate(function(value, respond) {
  User.findOne({email: value}, function(err, user) {
    if(err) throw err;
    if(user) return respond(false);
    respond(true);
  });
}, 'exists');

User.schema.path('username').validate(function(value, respond) {
  User.findOne({username: value}, function(err, user) {
    if(err) throw err;
    if(user) return respond(false);
    respond(true);
  });
}, 'exists');

然后,错误检查程序:

if (err) {
  switch (err.name) {
    case 'ValidationError':
      for (field in err.errors) {
        switch (err.errors[field].type) {
          case 'exists':
            ...
            break;
          case 'invalid':
            ...
            break;
          ...
        }
      }
      break;
    default:
      ...
  }
}

如果你想缩短它,你有各种选择。如果您只有一种类型的验证,您可以这样做:

if (err) {
  if (err.name == 'ValidationError') {
    for (field in err.errors) {
      ...
    }
  } else {
    // A general error (db, crypto, etc…)
    ...
  }
}

错误检查程序的最小表达式与您在帖子中写的内容类似:

if (err) {
  for (field in err.errors) {
    ...
  }
}

这样可行,因为如果没有定义错误,它将忽略for。但是你在这里忽略了所有其他错误类型。

我也认为这些错误布局有点乱,但不要指望在不久的将来这会改变。

答案 1 :(得分:7)

只需编写以下代码即可享受。

if (err) {
    console.log('Error Inserting New Data');
    if (err.name == 'ValidationError') {
        for (field in err.errors) {
            console.log(err.errors[field].message); 
        }
    }
}

答案 2 :(得分:3)

我使用AngularJS,因此ngRepeat会在网络表单上显示我的验证错误。我需要做的就是返回一组错误消息。

有时,Mongoose会抛出一个错误,这个错误不是验证错误,在这种情况下,err.errors对象将不会出现。我记录了执行错误。我仍然使用Web表单上的相同位置向用户显示执行错误。

var makeMongooseErrorMsgArray = function(err){
    var msgArray = [];
    if (err.errors) { // validation errors
        $.each(err.errors, function (key, val) {
            msgArray.push(val.message);
        });
    } else if (err.message){ // should be execution error without err.errors
        errLogr.log(err); // log execution errors
        msgArray.push(err.message);
    } else {
        msgArray.push('Unknown error');
    }
    return msgArray;
}

答案 3 :(得分:1)

我发现这很有用,可以显示数组中的所有错误。

例如,我提交了一个包含短密码和无效电子邮件的表单。

if (err && err.name === 'ValidationError') {
   err.toString().replace('ValidationError: ', '').split(',')
}

这导致了这个

[ 'Please provide a valid email address',
'The password should be at least 6 characters long' ]

如果您的错误消息中有逗号,,请尝试不使用.split(',')

不需要for循环。确保您的架构中有验证错误消息。对于上面的例子,我有

const validateEmail = email => {
  const re = /^\w+([\.-]?\w+)*@\w+([\.-]?\w+)*(\.\w{2,3})+$/;
  return re.test(email);
};

const Schema = mongoose.Schema;
const userSchema = new Schema({
   ...
   email: {
      type: String,
      trim: true,
      required: 'Email address is required',
      validate: [validateEmail, 'Please provide a valid email address'],
   },
   password: { type: String, set: encryptPassword, maxlength: [6, 'The password should be at least {MAXLENGTH} characters long'] },
   ...
});

答案 4 :(得分:1)

通过阅读所有这些答案,我认为最好创建效用函数并将其重复使用:

这是通过使用验证消息向客户端发送所需响应来处理ValidationError的函数,并可选择使用console.log在控制台中显示消息。

function handleValidationError(err, res, consoleLog = false){
  const messages = []
  for (let field in err.errors) {
    messages.push(err.errors[field].message)
    consoleLog && console.log(err.errors[field].message)
  }
  res.status(422).json({ messages })
}

然后在我们想要处理错误的控制器中,我们检查err.name是否为ValidationError,如果是,我们使用上面的效用函数。

user.save((err) => {
  if (err) {
    if (err.name === 'ValidationError') return handleValidationError(err, res) // here
    return res.status(500).json({ message: 'Error while creating new user' })
  }
  return res.status(201).json({ message: 'User created' })
})

然后客户端会在响应中得到验证错误:

curl\
-H 'Content-Type: application/json'\
-d '{"email": "foo", "password": "barbaz"}'\
http://localhost:3000/user/new

输出:

{"messages":["Email validation failure"]}

答案 5 :(得分:0)

为什么不使用validation方法as described in the API

objectToSave.validate(function(err) {          
  if (err) {
    // handle error
  }     
  else {
    // validation passed
  }
});

答案 6 :(得分:0)

这是我处理猫鼬验证错误的独特方法

一旦准备好,代码仍在进行中,我将对其进行更新,或者您可以贡献力量来扩展我的代码。

let message = "";
let title = "Validation Error";
let code = 400;
let requiredFields = [];
for (let field in err.errors) {
    let subMsg = ""
    if (err.errors[field].kind === "required") {
        requiredFields.push(field)
    }
}
if (requiredFields.length > 0) {
    message = "Following fields are required: " + requiredFields.join(", ");
} else {
    message = "Unknown";
}
res.status(code).json({
    status: code,
    message: message,
    title: title
});

答案 7 :(得分:0)

方法1

我对这种方法的最佳猜测是将广义验证器与基于承诺的回报结合使用, 希望对以后的工作有帮助

function validateDoc(model,fieldName) {
  return new Promise((resolve, reject) => {
    model.validate(err => {
      if (err) {
        return reject(err.errors[fieldName]);
      } else {
        return resolve(model);
      }
    });
  })
}

// Access error message in catch() or get the validated doc in then()

validateDoc(model,fieldName)
           .then((model) => console.log(model))
           .catch((message) => console.warn(message))

方法2

const userSchema = new Schema({
  name: {
    type:String,
    required:[true,'Name is required.'],
    validate:{
      validator:(name) => { name.length > 2 }
      message:'Name must be longer than two characters'
    }
  }
})
const User = mongoose.model('user',userSchema)

在架构内使用 validate:Object validator:function message:String

稍后,当您尝试保存记录时,它将在catch函数中显示验证错误对象

例如

const newUser = new User({ name: undefined });

newUser.save().catch( ( { errors } ) => { 

let errorBag = [];

Object.keys(errors).forEach((fieldName) => {

errorMessages.push({ [fieldName]:errors[fieldName].message })

});

 // All the errors with the **fieldName** and  **errorMessages** 

console.log(errorBag);  

   })