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

fcipmucu  于 2022-12-29  发布在  Node.js
关注(0)|答案(9)|浏览(197)

我的用户模型有两个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) { ... }

这是一种丑陋。

z4bn682m

z4bn682m1#

从技术上讲,您必须首先检查错误名称,因为并非所有错误的处理方式都相同。然后,您必须根据错误名称检查特定属性,如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) {
    case err instanceof mongoose.Error.ValidationError:
      for (field in err.errors) {
        switch (err.errors[field].type) {
          case 'exists':
            ...
            break;
          case 'invalid':
            ...
            break;
          ...
        }
      }
      break;
    default:
      ...
  }
}

如果要缩短验证时间,您有多种选择。如果只有一种验证类型,则可以按如下方式进行:

if (err) {
  if (err instanceof mongoose.Error.ValidationError) {
    for (field in err.errors) {
      ...
    }
  } else {
    // A general error (db, crypto, etc…)
    ...
  }
}

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

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

这是可行的,因为如果errors没有定义,它将忽略for,但是这里忽略了所有其他的错误类型。
我还认为这些错误布局有点混乱,但不要指望在不久的将来会有所改变。

nnt7mjpx

nnt7mjpx2#

只需编写下面的代码即可。

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

tez616oj3#

我使用AngularJS,所以ngRepeat在web表单上显示我的验证错误,我需要做的就是返回一个错误消息数组。
有时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;
}
ezykj2lf

ezykj2lf4#

通过阅读所有这些答案,我认为最好创建效用函数并像这样重用它:
此函数通过向客户端发送所需响应和验证消息来处理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"]}
w46czmvw

w46czmvw5#

我发现这很有用,它显示了数组中的所有错误。
例如,我提交了一个短密码和无效的电子邮件形式.

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'] },
   ...
});
nmpmafwu

nmpmafwu6#

为什么不使用API中描述的validation方法呢?

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

hmmo2u0o7#

下面是我处理mongoose验证错误的独特方法
该代码仍在进行中,一旦它准备好了,我会更新它,或者你可以帮助扩展我的代码。

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
});
nfeuvbwi

nfeuvbwi8#

方法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:Objectvalidator:function消息:String一起使用
稍后,当你试图保存记录时,它将在catch函数中保存验证错误对象
例如

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

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

let errorBag = [];

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

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

});

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

console.log(errorBag);  

   })
qyuhtwio

qyuhtwio9#

注意:违反约束会在保存时从MongoDB返回E11000错误,而不是Mongoose验证错误。

try {
  question.options.push(req.body.optiontext);
  await question.save();
  res.json(question);
} catch (err) {
  res.json({ message: "cannot add more than 4 options to a polling question" });
}

相关问题