Issue Deploying Next.js app to cPanel + LiteSpeed - Error:http.Server.listen()多次

qgelzfjb  于 2023-10-18  发布在  其他
关注(0)|答案(1)|浏览(67)

我目前正在将我的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;
}
wnavrhmk

wnavrhmk1#

我在运行一个使用Fastify的API时也遇到了这个问题。默认情况下,Fastify会监听localhost,而LightSpeed似乎打算使用127.0.0.1
我修改了我的代码,使其类似于下面的代码,现在它可以像预期的那样工作了:

const Fastify = require('fastify');

const app = Fastify();

app.get('/', (req, res) => {
  res.send({ hello: 'world' });
});

app.listen({ 
    port: process.env.PORT || 4201,
    host: '127.0.0.1'
});

所以在你的例子中,我不确定Next.js在后台使用了什么,因为它是SSR,但是试着改变主机。另一个注意事项是,Express 4也可以开箱即用,所以如果有一个选项可以更改http引擎/库,那也可能是一个选项。

相关问题