即使没有其他进程阻止该端口,Node.js应用也无法在端口80上运行


94

我正在安装了Node.js的Amazon EC2上运行Debian的实例。如果我运行下面的代码:

http = require('http');

http.createServer(function (request, response){
  response.writeHead(200, {'Content-Type':'text/plain'});
  response.end('Hello World\n');
}).listen(80);
console.log("Running server at port 80");

我得到下面的输出,它告诉我还有另一个进程正在侦听端口80:

Running server at port 80

events.js:72
        throw er; // Unhandled 'error' event
              ^
Error: listen EACCES
    at errnoException (net.js:901:11)
    at Server._listen2 (net.js:1020:19)
    at listen (net.js:1061:10)
    at Server.listen (net.js:1127:5)
    at Object.<anonymous> (/home/admin/nodetests/nodetest.js:6:4)
    at Module._compile (module.js:456:26)
    at Object.Module._extensions..js (module.js:474:10)
    at Module.load (module.js:356:32)
    at Function.Module._load (module.js:312:12)
    at Function.Module.runMain (module.js:497:10)

现在,当我检查是否有一个进程(以root用户身份,以防万物被隐藏)使用以下命令监听端口80时:

netstat -tupln

我得到以下输出,它告诉我在端口80上没有监听:

Active Internet connections (only servers)
Proto Recv-Q Send-Q Local Address           Foreign Address         State       PID/Program name
tcp        0      0 0.0.0.0:22              0.0.0.0:*               LISTEN      1667/sshd       
tcp6       0      0 :::22                   :::*                    LISTEN      1667/sshd

我应该注意,如果这有所作为,那么debian会将80端口作为入站规则打开。

我的问题是:我在做什么错?为什么无法识别监听端口80的进程?为什么在Debian中将其阻止?我应该采取什么步骤才能使代码正确运行?

Answers:


197

错误代码EACCES表示您没有适当的权限在该端口上运行应用程序。在Linux系统上,低于1024的任何端口都需要root用户访问权限。


5
因此,如果您有权使用sudo(只要对任何初学者都适用),那么sudo节点myapp.js就会执行此操作。
AlexMA 2013年

21
@AlexMA,但是以根用户身份运行服务器是一个很大的问题
Patrick Evans

1
那如何在80端口上运行节点呢?您是否应该...不使用代理?
AlexMA 2013年

9
@PatrickEvans我想最好的做法是不同的端口上运行,只是成立一个端口转发规则这里提到:stackoverflow.com/questions/16573668/...
AlexMA

73

您可以使用以下命令将端口80重定向到应用程序的端口(> 1024),而不是在端口80上运行:

iptables -t nat -I PREROUTING -p tcp --dport 80 -j REDIRECT --to-port 3000

如果您的应用程序在端口3000上运行,这将起作用。


1
做过这个。发现您必须是root用户才能在全新的debian安装中运行iptables,否则$ PATH不会指向它。
Brian Yeh 2013年

是的,这可能是最简单的方法。我在使用Google Cloud Compute时,碰到Port 80时出现了问题。这很棒。谢谢。
安迪

这应该是公认的解决方案。以sudo身份运行Web服务器很危险,如果应用程序中存在任何漏洞,则有可能授予攻击者root访问权限。同样,如果应用程序要创建任何文件,其他用户将无法访问它们,从而使您使用sudo更多文件。
jesusiniesta

不知道为什么,但是在Ubuntu 14.04上这对我不起作用。我现在通过ssh使用端口转发,这非常简单。我在下面发布了答案
panepeter


6

请注意,如果已apache运行,则可以在虚拟主机上创建反向代理。如果您的节点在端口上运行8080

<VirtualHost 127.0.0.1:80>
        ServerName myLocalServer

        ProxyPass        /  http://localhost:8080/
        ProxyPassReverse /  http://localhost:8080/
</VirtualHost>

当然,将服务器添加到/etc/hosts

127.0.0.1    myLocalServer

您将需要启用相关的apache模块:

sudo a2enmod proxy_html
sudo a2enmod proxy_http
sudo a2enmod proxy_connect
sudo a2enmod proxy_ajp
sudo service apache2 restart

...现在您可以连接到了http://myLocalServer


3

对于那些在开发环境中寻求快速简便解决方案的人来说,通过ssh进行端口转发可能是一个不错的选择:

ssh -L 80:localhost:3000 yourusername@localhost -N

这会将本地主机上的端口80转发到本地主机上的端口3000。

它需要以root(特权端口)身份运行。要取消它,只需在终端中按ctrl-c。(您可以添加-f标志以使该命令在后台运行,但随后需要再次找到它以将其杀死)。

此解决方案需要您在本地运行ssh服务器它可以快速完成,但是如果您在共享网络上,请牢记安全隐患。您可能希望至少应用某种级别的附加安全性(禁用密码和root登录)。

我个人只在本地计算机上使用过此功能。我不确定如果在生产环境中运行它会如何影响请求的处理速度,也许有人会想到。无论如何,您需要确保此命令始终保持运行,这会带来更多的麻烦。对于生产环境,我建议使用像nginx这样反向代理


2

六氰化物的答案是正确的。但是有什么解决方案可以做到这一点吗?

答案是肯定的。

怎么样?

您可以使用reverse proxy例如nginx reverse proxy在端口上运行80并将代理传递到ip:port该节点使用它的目的地。

您可以使用docker container它进行设置,使生活更加轻松。这是Nginx官方版本 docker hub,您可以将其拉出。

使用reverse proxyGoogle还能获得更多好处。


0

我有同样的错误,我尝试使用sudo运行我的应用程序,它对我有用。

没有须藤

mansi@mansi:~/NodePractice$ node myFirst.js 
events.js:141
      throw er; // Unhandled 'error' event
      ^

Error: listen EACCES 0.0.0.0:80
    at Object.exports._errnoException (util.js:870:11)
    at exports._exceptionWithHostPort (util.js:893:20)
    at Server._listen2 (net.js:1224:19)
    at listen (net.js:1273:10)
    at Server.listen (net.js:1369:5)
    at Object.<anonymous> (/home/mansi/NodePractice/myFirst.js:6:4)
    at Module._compile (module.js:410:26)
    at Object.Module._extensions..js (module.js:417:10)
    at Module.load (module.js:344:32)
    at Function.Module._load (module.js:301:12)

并与须藤

mansi@mansi:~/NodePractice$ sudo node myFirst.js 
^C

...这正是您不想做的。它产生了安全问题。
bvdb

-1

使用PORT 80需要一些特殊的权限。使用应用的sudo运行前语句可以解决我的问题。例如,如果您使用npm运行您的应用程序,则可以键入sudo npm start


1
此答案没有任何内容可以回答他人已经提供的内容。如果您要添加其他信息,请仅回答。
Brian Yeh

-2

错误代码 EACCES表示您没有适当的权限在该端口上运行应用程序。在Linux系统上,低于1024的任何端口都需要root用户访问权限。

sudo完美运行程序。运行sudo su运行程序之前命令。


参见六氰化物答案下的评论。谢谢。
Brian Yeh
By using our site, you acknowledge that you have read and understand our Cookie Policy and Privacy Policy.
Licensed under cc by-sa 3.0 with attribution required.