如何将从控制器返回的数据传递给Express'路由器?

时间:2018-03-19 17:36:03

标签: javascript node.js express

我试图将所有类型的数据返回到我的作者端点。如果传递给端点的url不包含查询参数,我希望路由器返回可用作者的完整列表。如果url包含firstName和lastName参数,我希望控制器找到匹配的作者,并将该数据传递回路由器。

目前,如果我发送了网址http://localhost:3001/authorshttp://localhost:3001/authors?firstName=tom&lastName=dooly,则会收到错误Error [ERR_HTTP_HEADERS_SENT]: Cannot set headers after they are sent to the client

有谁能告诉我为什么会这样,以及如何解决它?

主:

var express = require('express');
var path = require('path');
var favicon = require('serve-favicon');
var logger = require('morgan');
var cookieParser = require('cookie-parser');
var bodyParser = require('body-parser');
var mongoose = require('mongoose');

var app = express();
var dev_db_url = 'mongodb://localhost:27017/'
var mongoDB = process.env.MONGODB_URI || dev_db_url;

mongoose.connect(dev_db_url);

mongoose.Promise = global.Promise;
var db = mongoose.connection;
db.on('error', console.error.bind(console, 'MongoDB connection error:'));



// view engine setup
app.set('views', path.join(__dirname, 'views'));
app.set('view engine', 'jade');

// uncomment after placing your favicon in /public
//app.use(favicon(path.join(__dirname, 'public', 'favicon.ico')));
app.use(logger('dev'));
app.use(bodyParser.json());
app.use(bodyParser.urlencoded({ extended: false }));
app.use(cookieParser());
app.use(express.static(path.join(__dirname, 'public')));


var index = require('./routes/index');
var users = require('./routes/users');
var feedEntries = require('./routes/feedEntries');
var authors = require('./routes/authors');


app.use('/', index);
app.use('/users', users);
app.use('/feedEntries', feedEntries);
app.use('/authors', authors);


// catch 404 and forward to error handler
app.use(function(req, res, next) {
  var err = new Error('Not not Found');
  err.status = 404;
  next(err);
});


app.use(function(err, req, res, next) {

  res.locals.message = err.message;
  res.locals.error = req.app.get('env') === 'development' ? err : {};


  res.status(err.status || 500);
  res.render('error');
});

module.exports = app;

路线:

var express = require('express');
var router = express.Router();

var authorController = require('../controllers/authorController');

authorController.findAuthorsByFirstAndLastName);


router.get('/', function (req, res) {
    if(req.query.firstName||req.query.lastName) {
        res.send(authorController.findAuthorsByFirstAndLastName(req,res));
    }else{
        res.send(authorController.author_list(req,res));
    }
  });


module.exports = router;

控制器:

var Author = require('../models/author')
var async = require('async')


exports.author_list = function(req, res, next) {

    Author.find({},function(err, authors) {
        if (err){
            res.send(err);
        }
            return.json(authors);
    });

  };

  exports.findAuthorsByFirstAndLastName = function (req, res, next){
    var query = {}

    if(req.query.firstName||req.query.lastName) {

        query = {$or:[{firstName:{$regex: req.query.firstName, $options: 'i'}},
            {lastName:{$regex: req.query.lastName, $options: 'i'}}]}
    }

    else {
        return res.status(500).send({ error: 'Unable to parse data'});
    }

    var firstName =  req.body.firstName;
    var lastName = req.body.lastName;

    Author.find(query , function (err, authors) {
        if(err) {
            res.send(err);
        }
        res.json(authors);
     });
  };

1 个答案:

答案 0 :(得分:1)

当您的路线中有两个cannot set headers after they are sent时,您会获得res.[whatever]。所以你有res.send(functionCallThatAlsoDoesRes.Send)。导致错误的是什么。

如果您希望路由在请求和响应之间执行多个操作,则可以将这些操作写为单独的中间件。中间件始终采用参数reqresnext(这个函数说明要转到列表中的下一个中间件)。

所以,你可以写:

authorController.findAuthorsByFirstAndLastName = function(req, res, next) {
  if (!(req.query.firstName || req.query.lastName)) {
    res.locals.getFullAuthorList = true
    return next()
  } else {
    const query = /* whatever */
    Author.find(query, (err, authors) => {
      if (err) return next(err)
      res.locals.authors = authors
      next()
    })
  }
}

authorController.author_list = function(req, res, next) {
  if (!res.locals.getFullAuthorList) return next() // if we already have authors we don't need to do anything
  Author.find({}, (err, authors) => {
    if (err) return next(err)
    res.locals.authors = authors
    next()
  })
}

然后在你的路线中,你要说:

router.get('/', authorController.findAuthorsByFirstAndLastName, authorController.author_list, (req, res) => {
    res.json({ authors: res.locals.authors })
})

如果您之前没有看到res.locals,那么它只是响应对象上可用于附加内容的属性。它在整个请求/响应周期中持续存在,并为每个新请求清除。