【Nginx】监听端口的管理

本文解析了Nginx如何通过配置项实现端口监听,包括如何将端口与虚拟主机关联,以及Nginx内部如何管理和存储这些配置。
监听端口属于server虚拟主机,由server{}块内的listen配置项决定。也就是说,在server{}块配置项内定义了该虚拟主机所要监听的端口。

在处理配置文件http块内main级别的配置项时,每个HTTP模块都会调用create_main_conf、create_srv_conf、create_loc_conf三个方法建立三个结构体,用来分别存储http块、server块、location块内的配置项。ngx_http_core_module是HTTP模块,所以它会调用ngx_http_module_t接口内的ngx_http_core_create_main_conf方法创建存储main级别配置项的结构体,函数如下:
static void *
ngx_http_core_create_main_conf(ngx_conf_t *cf)
{
    ngx_http_core_main_conf_t  *cmcf;
 
    ....
 
    return cmcf;
}


从代码可以看出创建了一个ngx_http_core_main_conf_t结构体用于存储配置项。此结构体定义如下:
typedef struct {
    ....
    ngx_array_t               *ports;           /* 保存http块配置项内监听的所有端口 */
    ....
} ngx_http_core_main_conf_t;


其中的ports成员就保存着所有需要监听的端口。每一个端口用结构体ngx_http_conf_port_t表示:
typedef struct {
    ngx_int_t                  family;
    in_port_t                  port;
    ngx_array_t                addrs;     /* array of ngx_http_conf_addr_t */
} ngx_http_conf_port_t;


port成员就保存要监听的端口,而addrs成员则保存有一系列的IP地址,每个地址用一个ngx_http_conf_addr_t表示,每一个IP地址都与一个端口绑定,例如:
  • 127.0.0.1:8000
  • 173.39.160.51:8000
这样,如果一台机器有多个IP,就能够同时监听这些IP的端口了。如果配置项直接就是listen 80,那么相当于默认监听该端口下的所有地址,即*.80。

下面再来看看ngx_http_conf_addr_t的定义:
typedef struct {
    ngx_http_listen_opt_t      opt;
 
    ngx_hash_t                 hash;
    ngx_hash_wildcard_t       *wc_head;
    ngx_hash_wildcard_t       *wc_tail;
 
#if (NGX_PCRE)
    ngx_uint_t                 nregex;
    ngx_http_server_name_t    *regex;
#endif
 
    /* the default server configuration for this address:port */
    ngx_http_core_srv_conf_t  *default_server;
    ngx_array_t                servers;  /* array of ngx_http_core_srv_conf_t */
} ngx_http_conf_addr_t;


这里关注servers数组,它把监听的端口与server{}虚拟主机关联起来了。什么意思呢?假如ngx_http_conf_addr_t对应的端口为8080,且servers成员包含虚拟主机A和虚拟主机B,那么在这两个块配置项中就存在listen 8080这个配置项。也就是说,每个监听地址ngx_http_conf_addr_t中的servers数组中关联着监听地址对应的server{}虚拟主机。

总的关系图如下:


从上图的关系中可以得出如下事实:
  • server A虚拟主机监听了两个端口,分别是*.80和127.0.0.1:8000
  • server B虚拟主机监听了三个端口,分别是*.8080、*.80、173.39.160.51:8000

参考:
《深入理解Nginx》 P367-P369.
### 配置 Nginx 监听 80 端口 为了使 Nginx 正确监听 80 端口,需确保配置文件中的设置无误并解决可能存在的权限问题。 #### 检查现有配置文件冲突 当在 `/etc/nginx/nginx.conf` 中配置监听 80 端口时,如果有其他地方也指定了相同的端口号,默认情况下可能会造成冲突。因此建议先检查是否有不必要的 `include` 或者重复定义的 server 块,并适当注释掉这些可能导致干扰的部分[^1]。 #### 修改主配置文件 编辑位于 `/etc/nginx/nginx.conf` 的主要配置文件,在 http 块内添加一个新的 server 定义来指定监听 80 端口: ```nginx http { ... server { listen 80; server_name localhost; location / { root html; index index.html index.htm; } error_page 500 502 503 504 /50x.html; location = /50x.html { root html; } # 更多location配置... } } ``` #### 处理权限不足的问题 有时即使正确设置了监听选项,普通用户启动服务仍会遇到 "Permission denied" 错误提示。这是因为 Linux 系统不允许非特权账户绑定低于 1024 的端口(如 80)。为此可以采取两种方法之一解决问题: - **更改运行身份**: 将 Nginx 设置成以 root 用户的身份启动,但这通常不是推荐的安全实践; - **调整 SELinux/AppArmor 规则** : 如果启用了安全模块,则需要相应修改策略允许 Nginx 绑定低编号端口;或者 - **使用 capabilities 工具授予特定能力给 nginx 进程**, 如下所示命令可赋予必要的权限而不必完全作为超级管理员执行: ```bash sudo setcap 'cap_net_bind_service=+ep' $(which nginx) ``` 完成上述操作后重启 Nginx 服务验证变更效果。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值