Apache 上 NodeJS Express 的 HTTP2 设置引发错误

Posted

技术标签:

【中文标题】Apache 上 NodeJS Express 的 HTTP2 设置引发错误【英文标题】:HTTP2 setup for NodeJS Express on Apache throws Error 【发布时间】:2021-01-29 16:44:13 【问题描述】:

我在为通过 Apache 服务器提供的 NodeJS v12.19.0/Express 应用程序设置 HTTP2 时遇到问题。

我启用了 mod http2 & proxy_http2 并安装了 npm http2

bin/www 如下所示:

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

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

var server = http.createServer(app);
server.listen(port);
server.on('error', onError);
server.on('listening', onListening);

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

if (port >= 0) 
    return port;

return false;


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


var bind = typeof port === 'string'
    ? 'Pipe ' + port
    : 'Port ' + port;
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;



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

Apache 站点配置:

<VirtualHost *:443>

    Protocols h2 http/1.1

    ServerName domain.com
    ProxyRequests On
    #ProxyPass / http://localhost:3000/
    #ProxyPassReverse / http://localhost:3000/

    ProxyPass / h2://localhost:3000/
    ProxyPassReverse / http://localhost:3000/

    SSLCertificateFile /etc/letsencrypt/live/domain.com/fullchain.pem
    SSLCertificateKeyFile /etc/letsencrypt/live/domain.com/privkey.pem
    Include /etc/letsencrypt/options-ssl-apache.conf
</VirtualHost>

没有线条

ProxyPass / h2://localhost:3000/
ProxyPassReverse / http://localhost:3000/

我明白了:

代理错误

代理服务器收到来自上游服务器的无效响应。 代理服务器无法处理请求

原因:从远程服务器读取错误

Apache/2.4.41 (Ubuntu) 服务器位于 domain.com 端口 443

如果我包含这些行,我会得到:

内部服务器错误

服务器遇到内部错误或配置错误,并且 无法完成您的请求。

请联系 [no address given] 的服务器管理员以 告知他们此错误发生的时间,以及您采取的措施 在此错误之前执行。

有关此错误的更多信息可能在服务器错误中可用 日志。 domain.com 端口 443 上的 Apache/2.4.41 (Ubuntu) 服务器

感谢您的帮助..

【问题讨论】:

【参考方案1】:

您正在尝试通过 HTTP/2 通过 HTTPS (h2) 从 Apache 连接到 Node,但从您的 Node 代码看来,您正在使用 TLS 证书,所以我猜它只是一个 HTTP /2 通过 HTTP 服务器 (h2c)。

所以替换这些行:

ProxyPass / h2://localhost:3000/
ProxyPassReverse / http://localhost:3000/

有了这个:

ProxyPass / h2c://localhost:3000/
ProxyPassReverse / http://localhost:3000/

它应该通过 HTTP/2 连接。

但是我会质疑它的价值。 Mod_proxy_http2 是 still marked as experimental 和 the value of backend HTTP/2 connections is very much in question。如果您希望使用 HTTP/2 在 Node 中进行试验,那么您不应该直接连接而不是通过 Apache 连接吗?

【讨论】:

仍然遇到 502 和 503 错误。我也开始认为 express 并不真正支持 http2 模块。我发现的大多数示例都是针对已弃用的 spdy。我通过 Apache 连接,因为无论如何我已经为我的 Nextcloud 运行了它。我没有太多的节点部署经验,我认为它有点像烧瓶,如果我可以从完整的 http 服务器功能中受益。

以上是关于Apache 上 NodeJS Express 的 HTTP2 设置引发错误的主要内容,如果未能解决你的问题,请参考以下文章

express + nodeJS上的404错误,在正常路线上

在heroku上部署Nodejs+Express+React+Webpack应用

NodeJS Express - 同一端口上的两个 NodeJS 实例(vhost)

CORS“允许凭据”Nodejs/Express

Heroku 无法在 nodejs 的 express-router 上拥有从“@”开始的路由

Nodejs 502 Bad Gateway 在 Elastic Beanstalk AWS 上部署 Express