侦听单个 Express 应用程序的 HTTP 和 HTTPS

IT技术 javascript node.js https express
2021-03-15 19:23:49

我可以使用相同的路由和相同的中间件创建一个同时侦听 HTTP 和 HTTPS 的 Express 服务器吗?

目前,我在 HTTP 上使用 Express 执行此操作,使用stunnel隧道将 HTTPS 传输到 Express,但我更喜欢纯 Node 解决方案。

我可以用这段代码来做,但使用handle标记为私有方法:

var express = require( 'express' )
    , https = require("https")
    , fs = require( 'fs' );

var app = express.createServer();
// init routes and middlewares
app.listen( 80 );

var privateKey = fs.readFileSync( 'privatekey.pem' ).toString();
var certificate = fs.readFileSync( 'certificate.pem' ).toString();
var options = {key: privateKey, cert: certificate};
https.createServer( options, function(req,res)
{
    app.handle( req, res );
} ).listen( 443 );
6个回答

为了使您的应用程序监听双方httphttps有关港口80443分别,做到以下几点

创建一个快递应用:

var express = require('express');
var app = express();

返回的应用程序express()是一个 JavaScript 函数。它可以作为回调传递给 Node 的 HTTP 服务器来处理请求。这使得使用相同的代码库轻松提供应用程序的 HTTP 和 HTTPS 版本。

你可以这样做:

var express = require('express');
var https = require('https');
var http = require('http');
var fs = require('fs');
var app = express();

var options = {
  key: fs.readFileSync('/path/to/key.pem'),
  cert: fs.readFileSync('/path/to/cert.pem'),
  ca: fs.readFileSync('/path/to/ca.pem')
};

http.createServer(app).listen(80);
https.createServer(options, app).listen(443);

有关完整的详细信息,请参阅文档

作为对这个问题的可能更新,您可能想在此处查看 express 3 的更改。更改文档说:

的返回值express()是一个 JavaScript 函数,封装了使 Express 应用程序打勾的所有内容。这意味着您可以通过将应用程序传递给节点的http.createServer()来轻松设置应用程序的 HTTP 和 HTTPS 版本https.createServer()

在 Express 3 中,express.createServer()现在是express()

这是 express 3 的完整示例:

var fs = require('fs')
    , https = require('https')
    , http = require('http')
    , express = require('express')
    , keys_dir = 'keys/'
    , server_options = {
        key  : fs.readFileSync(keys_dir + 'privatekey.pem'),
        ca   : fs.readFileSync(keys_dir + 'certauthority.pem'),
        cert : fs.readFileSync(keys_dir + 'certificate.pem')
      }
    , app = express();
app.configure(function(){
  app.use(express.cookieParser());
  app.use(express.bodyParser());
  app.use(express.methodOverride());
  app.use(express.session( { secret: '' } ));
  app.use(app.router);
});
app.configure('development',function(){
  app.use(express.static(__dirname + '/public'));
  app.use(express.errorHandler({dumpExceptions: true, showStack:true}));
  app.set('view options', { pretty: true });
});
app.get('/', function(req, res){
  res.send('Hello World!');
});
https.createServer(server_options,app).listen(7000);
http.createServer(app).listen(8000);
最好使用提问者使用的相同语言回答问题。我冒昧地将 coffeescript 翻译成 vanilla JS。
2021-05-06 19:23:49

您可以通过以下方式共享实现:

var register = function (app) {
    // config middleware
    app.configure({

    });

    // config routes
    app.get(...);
};

var http = express.createServer();
register(http);
http.listen(80);

var https = express.createServer({ key: /* https properties */ });
register(https);
https.listen(443);
此解决方案现已弃用。有更新的解决方案吗?
2021-04-21 19:23:49
@CMN - 请参阅下面的 cmd 解决方案。它适用于当前版本的 Node (v4.4.2) 和 Express (4.13.4)
2021-05-11 19:23:49

您可以在同一端口使用 express 和 https。

这对我有用。

const express=require('express');
const app=express();
const cors=require('cors');
const path=require("path");
const routes=require('./routes/api');
const routerComplain=require('./routes/api');
const routerstores=require('./routes/api');
const routerstock=require('./routes/api');
const routerreport=require('./routes/api');
const routeritem=require('./routes/api');
const bodyParser=require('body-parser');
const routerRegister=require('./routes/api');
const mongoose=require('mongoose');
var server = require('http').Server(app);
var io = require('socket.io')(server);
require("dotenv").config();

mongoose.connect('mongodb://@@@@@@@@@@@@@@@@@',{ useNewUrlParser: true },(err)=>{
    if(!err){
        console.log('db connected')
    }else{
        console.log('error in db')
    }
});

mongoose.Promise = global.Promise;
app.use(express.static('public'));
app.use(bodyParser.json());
app.use(cors({credentials: true, origin:'http://localhost:3000'}));
app.use(express.static(path.join(__dirname, "client", "build")))

app.use('/reg',routes);
app.use('/complain',routerComplain);
app.use('/register',routerRegister);
app.use('/stores',routerstores);
app.use('/reports',routerreport);
app.use('/stock',routerstock);
app.use('/items',routeritem);

app.get("*", (req, res) => {
    res.sendFile(path.join(__dirname, "client", "build", "index.html"));
});

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

const port = process.env.port||4000;

server.listen(port,function(){
    console.log('now listening for request');
});

如果您想使用传统的两个端口,上述解决方案之一可能有效,但是使用 httpolyglot,您可以使用相同的中间件轻松地在同一端口上使用 http 和 https。

https://github.com/mscdex/httpolyglot

这是一些对我有用的骨架代码:

var express = require('express');
var fs = require('fs');
var httpolyglot = require('httpolyglot');
var app = express();

const options = {
    key: fs.readFileSync("/etc/ssl/certs/key"),
    cert: fs.readFileSync("/etc/ssl/certs/cer.cer")
};

httpolyglot.createServer(options, app).listen(port);

如果你想要 http -> https 转发,你可以在 createServer() 调用之前添加这个中间件函数:

app.use(function(req, res, next) {
    if (!req.secure ) {
            res.redirect (301, 'https://' + req.hostname + ':port' + req.originalUrl);
    }
    next();
});

这可以在自定义端口上设置

整洁......将它们放在同一个端口上有好处吗?
2021-05-11 19:23:49