我目前正在将我的Next.js应用(版本13.4.19)部署到一个使用cPanel和LiteSpeed的共享主机提供商。不幸的是,我在部署过程中遇到了一个错误,我可以使用一些帮助来解决它。
我看到的错误消息是:
Error: http.Server.listen() was called more than once which is not allowed.
at Server.customListen [as listen] (/usr/local/lsws/fcgi-bin/lsnode.js:68:15)
at ipcPort (/home/[..webbroot..]/node_modules/next/dist/server/lib/server-ipc.js:73:17)
at new Promise (<anonymous>)
at createIpcServer (/home/[..webbroot..]/node_modules/next/dist/server/lib/server-ipc.js:72:27)
at initialize (/home/[..webbroot..]/node_modules/next/dist/server/lib/router-server.js:79:82)
at async Server.<anonymous> (/home/[..webbroot..]/node_modules/next/dist/server/lib/start-server.js:192:36)
因为我在cPanel主机上,我不能使用next start
来运行应用程序。相反,我需要使用独立构建,它会生成一个server.js
文件,可以将其设置为cPanel的入口脚本。
经过研究,我发现LiteSpeed有一个名为/usr/local/lsws/fcgi-bin/lsnode.js
的引导节点脚本。我怀疑这个lsnode.js脚本和Next.js可能不完全兼容。
我的问题:
1.有人在将Next.js应用部署到cPanel + LiteSpeed时遇到过类似的问题吗?
1.你对如何解决这个问题并使Next.js和LiteSpeed在此设置中无缝协作有什么见解或想法吗?
提前感谢您的帮助!
我在《乘客》杂志上找到了这份文件。https://www.phusionpassenger.com/library/indepth/nodejs/reverse_port_binding.html但我还没有设法让这个解决方案在Next.js库中工作。
我在下面嵌入lsnode.js脚本的源代码:
/*
* Copyright 2002-2018 Lite Speed Technologies Inc, All Rights Reserved.
* LITE SPEED PROPRIETARY/CONFIDENTIAL.
*/
var EventEmitter = require('events').EventEmitter;
var os = require('os');
var fs = require('fs');
var http = require('http');
var util = require('util');
var net = require('net');
var socketObject = { fd: 0 };
module.isApplicationLoader = true;
global.LsNode = new EventEmitter();
startApplication();
function startApplication() {
var appRoot = process.env.LSNODE_ROOT || process.cwd();
var startupFile = process.env.LSNODE_STARTUP_FILE || 'app.js';
LsNode.listenDone = false;
if (process.env.LSNODE_ROOT != undefined) {
try {
process.chdir(process.env.LSNODE_ROOT);
} catch (err) {
console.error("Error setting directory to: " +
process.env.LSNODE_ROOT + ": " + err);
}
}
if (!startupFile.startsWith('/')) {
startupFile = appRoot + '/' + startupFile;
}
process.title = 'lsnode:' + appRoot;
var consoleLog = process.env.LSNODE_CONSOLE_LOG || '/dev/null';
fs.closeSync(1);
try {
fs.openSync(consoleLog, "w+");
} catch(e) {
fs.openSync('/dev/null', "w+");
}
http.Server.prototype.realListen = http.Server.prototype.listen;
http.Server.prototype.listen = customListen;
http.Server.prototype.address = lsnode_address;
var app = require(startupFile);
if (!LsNode.listenDone) {
if (typeof app.listen === "function")
app.listen(3000);
}
}
function lsnode_address() {
return process.env.LSNODE_SOCKET;
}
function customListen(port) {
function onListenError(error) {
server.emit('error', error);
}
// The replacement for the listen call!
var server = this;
if (LsNode.listenDone) {
throw new Error("http.Server.listen() was called more than once " +
"which is not allowed.");
}
LsNode.listenDone = true;
var listeners = server.listeners('request');
var i;
server.removeAllListeners('request');
server.on('request', function(req) {
req.connection.__defineGetter__('remoteAddress', function() {
return '127.0.0.1';
});
req.connection.__defineGetter__('remotePort', function() {
return port;
});
req.connection.__defineGetter__('addressType', function() {
return 4;
});
});
for (i = 0; i < listeners.length; i++) {
server.on('request', listeners[i]);
}
var callback;
if (arguments.length > 1 && typeof(arguments[arguments.length - 1]) == 'function') {
callback = arguments[arguments.length - 1];
}
server.once('error', onListenError);
server.realListen(socketObject, function() {
server.removeListener('error', onListenError);
if (callback) {
server.once('listening', callback);
}
server.emit('listening');
});
return server;
}
1条答案
按热度按时间wnavrhmk1#
我在运行一个使用Fastify的API时也遇到了这个问题。默认情况下,Fastify会监听
localhost
,而LightSpeed似乎打算使用127.0.0.1
我修改了我的代码,使其类似于下面的代码,现在它可以像预期的那样工作了:
所以在你的例子中,我不确定Next.js在后台使用了什么,因为它是SSR,但是试着改变主机。另一个注意事项是,Express 4也可以开箱即用,所以如果有一个选项可以更改http引擎/库,那也可能是一个选项。