Express.js、bin/www 设置和 Socket.io

Posted

技术标签:

【中文标题】Express.js、bin/www 设置和 Socket.io【英文标题】:Express.js, bin/www settings and Socket.io 【发布时间】:2016-02-14 16:58:22 【问题描述】:

问题已解决。

请看下面我的回答。


问题:

我正在尝试让 socketio 和 express 在同一个端口上工作,这是我得到的错误。

events.js:141
      throw er; // Unhandled 'error' event
      ^
Error: listen EACCES 0.0.0.0:80
    at Object.exports._errnoException (util.js:874:11)
    at exports._exceptionWithHostPort (util.js:897:20)
    at Server._listen2 (net.js:1221:19)
    at listen (net.js:1270:10)
    at Server.listen (net.js:1366:5)
    at Object.<anonymous> (/Users/app.js:22:8)
    at Module._compile (module.js:435:26)
    at Object.Module._extensions..js (module.js:442:10)
    at Module.load (module.js:356:32)
    at Function.Module._load (module.js:311:12)

我可以使用 sudo npm start 启动我的服务器,但现在我有两台服务器正在运行,localhost:3000(express)和 localhost:80(express 和 socket.io)

如何在不破坏任何内容的情况下禁用 :3000 斌/www?

我正在使用 Express 4。

这是我的看法

html
    head
        title Socket.io
        script(src="/socket.io/socket.io.js")
    body
        script(src="javascripts/ol.js")

ol.js(客户端)

var socket = io.connect('http://localhost');
socket.on('news', function (data) 
    console.log(data);
    socket.emit('my other event', my: 'data');
);

这是我的 app.js

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 http = require('http');
var router = express.Router();

var app = express();

var server = http.Server(app);
var io = require('socket.io')(server);
server.listen(80);

io.on('connection', function (socket) 
  socket.emit('news',  hello: 'world' );
  socket.on('my other event', function (data) 
    console.log(data);
  );
);

io.on('error', function () 
    console.log("errr");
);


// view engine setup
app.set('views', path.join(__dirname, 'views'));
app.set('view engine', 'jade');
app.set('view options', layout: false);
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')));

app.get('/testit', function (req, res) 
    res.send("it works!");
);

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


// error handlers

// development error handler
// will print stacktrace
if (app.get('env') === 'development') 
    app.use(function (err, req, res, next) 
        res.status(err.status || 500);
        res.render('error', 
            message: err.message,
            error: err
        );
    );


// production error handler
// no stacktraces leaked to user
app.use(function (err, req, res, next) 
    res.status(err.status || 500);
    res.render('error', 
        message: err.message,
        error: 
    );
);

module.exports = app;

当我对 server.listen 使用 3000 而不是 80 并且如果我使用 node app.js 而不是 npm start 来运行我的应用程序时,我看不到任何问题并且一切正常,但是 socketio 会覆盖 Express.js 的设置那样吗?

我根本没有碰过 bin/www 文件。就在下面。

#!/usr/bin/env node

/**
 * Module dependencies.
 */

var app = require('../app');
var debug = require('debug')('future:server');
var http = require('http');

/**
 * Get port from environment and store in Express.
 */

var port = normalizePort(process.env.PORT || '3000');
app.set('port', port);

/**
 * Create HTTP server.
 */

var server = http.createServer(app);



/**
 * Listen on provided port, on all network interfaces.
 */

server.listen(port);
server.on('error', onError);
server.on('listening', onListening);

/**
 * Normalize a port into a number, string, or false.
 */

function normalizePort(val) 
  var port = parseInt(val, 10);

  if (isNaN(port)) 
    // named pipe
    return val;
  

  if (port >= 0) 
    // port number
    return port;
  

  return false;


/**
 * Event listener for HTTP server "error" event.
 */

function onError(error) 
  if (error.syscall !== 'listen') 
    throw error;
  

  var bind = typeof port === 'string'
    ? 'Pipe ' + port
    : 'Port ' + port;

  // handle specific listen errors with friendly messages
  switch (error.code) 
    case 'EACCES':
      console.error(bind + ' requires elevated privileges');
      process.exit(1);
      break;
    case 'EADDRINUSE':
      console.error(bind + ' is already in use');
      process.exit(1);
      break;
    default:
      throw error;
  


/**
 * Event listener for HTTP server "listening" event.
 */

function onListening() 
  var addr = server.address();
  var bind = typeof addr === 'string'
    ? 'pipe ' + addr
    : 'port ' + addr.port;
  debug('Listening on ' + bind);

【问题讨论】:

如果您要为自己的问题提供答案,*** 上的程序是您发布自己问题的答案。您将问题保留为问题(没有答案)。然后,您发布一个包含您找到的答案的答案。一段时间后,您甚至可以接受自己的答案,向社区展示它对您有用。这个地方不同于在线论坛编辑他们的问题以表明他们有解决方案。 你是对的。我添加了一个答案。谢谢。 【参考方案1】:
    在您的主目录中创建一个名为 io.js 的文件(这是我的选择,不过您可以将它放在其他位置)

io.js

var io = require('socket.io')(); // yes, no server arg here; it's not required
// attach stuff to io
module.exports = io;
    打开 bin/www

var app = require('../app');之后定义你的io变量

var io = require('../io');

io.attach(server); 放在server.listen(port) 之后

    打开 app.js

var app = express(); 之后定义您的 io 变量和 io.on 部分

var io = require('./io');

io.on('connection', function (socket) 
    socket.emit('news', hello: 'world');
    socket.on('my other event', function (data) 
        console.log(data);
    );
);

io.on('error', function () 
    console.log("errr");
);

最后,我们的观点。不用担心 socket.io.js 的存在,它就在那里。您不必手动下载或包含其他外部文件。

html
    head
        title Socket.io
        script(src="/socket.io/socket.io.js")
    body
        script(src="javascripts/ol.js")

ol.js(测试)

var socket = io.connect('http://localhost:3000');
socket.on('news', function (data) 
    console.log(data);
    socket.emit('my other event', 
        my: 'data'
    );
);

【讨论】:

以上是关于Express.js、bin/www 设置和 Socket.io的主要内容,如果未能解决你的问题,请参考以下文章

Express.js 虚拟主机子域设置

express js 中的 Socket.io:Router.use 需要中间件功能但未定义

NodeJS,Express 与 Socket.io 交互

为啥我的 Express.js 后端的 CORS 设置不起作用?

Express.js:为 webpack 中捆绑的静态资产设置缓存控制标头

将 Express.js 3 与数据库模块一起使用,在哪里初始化数据库客户端?