重复注册发生时,Node express服务器崩溃

时间:2016-09-21 16:24:19

标签: javascript node.js mongodb mongoose

所以我有一个节点服务器启动并运行api端点,当通过邮递员输入重复的电子邮件时,该端点不断崩溃。如果发生非重复注册,则服务器将继续运行而不会出现打嗝。但是,如果提交了重复的电子邮件,我确实在邮递员身上看到了重复的电子邮件错误:

{
  "code": 11000,
  "index": 0,
  "errmsg": "E11000 duplicate key error collection: register.users index: email_1 dup key: { : \"atlanteavila@gmail.com\" }",
  "op": {
    "userRole": "isUser",
    "password": "yudfjadjslkafdaljj343",
    "zip": "94596",
    "state": "CA",
    "city": "Walnut Creek",
    "address2": "",
    "streetAddress": "1 fake address ",
    "phoneNumber": "(925) 555-0644",
    "email": "fake@email.com",
    "lastName": "Last",
    "firstName": "Person",
    "_id": "57e2b2e49f0de63201914616",
    "__v": 0
  }
}

这是api路线:

var express = require('express');
var app = express();
var router = express.Router();
var User = require('../models/user');

module.exports = function (app) {

  router.route('/users')

  .get( function (req, res) {
    res.json({ message: 'welcome!!' });
  })

  .post(function (req, res) {
    var rb = req.body;
    var user = new User();
    user.firstName = rb.firstName;
    user.lastName = rb.lastName;
    user.email = rb.email;
    user.phoneNumber = rb.phoneNumber;
    user.streetAddress = rb.streetAddress;
    user.address2 = rb.address2;
    user.city = rb.city;
    user.state = rb.state;
    user.zip = rb.zip
    user.password = rb.password;
    user.userRole = rb.userRole;


    user.save( function(err) {
      if (err) {
        res.send(err);
        next();
      };
      res.json({ message: 'UserCreated!' });
    });
  });

  app.use('/api', router);

}

和猫鼬模式:

var mongoose = require('mongoose');
var Schema = mongoose.Schema;

var UserSchema = new Schema({
  firstName: {
    type: String,
    required: true
  },
  lastName: {
    type: String,
    required: true
  },
  email: {
    type: String,
    required: true,
    index: true,
    unique: true
  },
  phoneNumber: {
    type: String, 
    required: true
  },
  streetAddress: {
    type: String, 
    required: true
  },
  address2: String,
  city: {
    type: String, 
    required: true
  },
  state: {
    type: String, 
    required: true
  },
  zip: {
    type: String, 
    required: true
  },
  password: {
    type: String, 
    required: true
  },
  userRole: {
    type: String,
    required: true
  }
});

module.exports = mongoose.model('User', UserSchema);

最后终端上的错误:

[nodemon] restarting due to changes...
[nodemon] starting `node server/server.js`
Up and running on port 3000
events.js:160
      throw er; // Unhandled 'error' event
      ^

Error: Can't set headers after they are sent.
    at ServerResponse.OutgoingMessage.setHeader (_http_outgoing.js:346:11)
    at ServerResponse.header (/Users/atlanteavila/Documents/myapps/register/node_modules/express/lib/response.js:719:10)
    at ServerResponse.send (/Users/atlanteavila/Documents/myapps/register/node_modules/express/lib/response.js:164:12)
    at ServerResponse.json (/Users/atlanteavila/Documents/myapps/register/node_modules/express/lib/response.js:250:15)
    at /Users/atlanteavila/Documents/myapps/register/server/routes/signUp.js:34:11
    at /Users/atlanteavila/Documents/myapps/register/node_modules/mongoose/lib/model.js:3336:16
    at /Users/atlanteavila/Documents/myapps/register/node_modules/mongoose/lib/document.js:1932:15
    at _combinedTickCallback (internal/process/next_tick.js:67:7)
    at process._tickCallback (internal/process/next_tick.js:98:9)
[nodemon] app crashed - waiting for file changes before starting...

提前感谢您帮助我们解决这个问题。谢谢!

3 个答案:

答案 0 :(得分:2)

当出现错误(抛出异常)时,您正在发送两个响应。您可以通过返回next()或在致电后返回来缓解此问题。

user.save( function(err) {
  if (err) {
    return res.send(err);
  }
  res.json({ message: 'UserCreated!' });
});

答案 1 :(得分:1)

  

您正在发送two个回复,这就是您收到此错误的原因

如果您要在next() save function发送回复,我认为您不需要使用fail or success回调。

尝试按此操作后返回您的回复

user.save( function(err) {
      if (err) {
       return res.send(err);
      };
      return res.json({ message: 'UserCreated!' });
    });

答案 2 :(得分:0)

您的架构中有唯一:true,并且您尝试插入重复记录时为true。它违反了架构验证规则。您应该对查询使用异常处理并解析这些错误。您可以使用try-catch块,如果您正在使用promise,则在每个查询中使用catch块。

发送有关重复保存记录的响应。将以下内容添加到路由器错误处理程序:

if (err instanceof Error && err.name === "MongoError" && err.driver) {
        if (err.code == 11000) { // unique index conflict
            return ['Resource document already exists.'];
        }
    }

并从错误处理程序返回它。