nginx0.1.0之event模块初始化源码分析(3)

2024-03-27 21:48

本文主要是介绍nginx0.1.0之event模块初始化源码分析(3),希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

前面已经分析了event初始化的整体流程和第一步create_conf,接下来看一下第二步ngx_conf_parse。这里不分析该函数的代码,该函数主要是遍历配置文件的内容,然后读取命令和参数。最后匹配nginx所有模块的配置,找到处理该指令的函数。我们首先看一下event模块中ngx_event_core_module模块的指令配置。

static ngx_command_t  ngx_event_core_commands[] = {{ ngx_string("connections"),NGX_EVENT_CONF|NGX_CONF_TAKE1,ngx_event_connections,0,0,NULL },{ ngx_string("use"),NGX_EVENT_CONF|NGX_CONF_TAKE1,ngx_event_use,0,0,NULL },{ ngx_string("multi_accept"),NGX_EVENT_CONF|NGX_CONF_TAKE1,ngx_conf_set_flag_slot,0,offsetof(ngx_event_conf_t, multi_accept),NULL },{ ngx_string("accept_mutex"),NGX_EVENT_CONF|NGX_CONF_TAKE1,ngx_conf_set_flag_slot,0,offsetof(ngx_event_conf_t, accept_mutex),&ngx_accept_mutex_post },{ ngx_string("accept_mutex_delay"),NGX_EVENT_CONF|NGX_CONF_TAKE1,ngx_conf_set_msec_slot,0,offsetof(ngx_event_conf_t, accept_mutex_delay),NULL },{ ngx_string("debug_connection"),NGX_EVENT_CONF|NGX_CONF_TAKE1,ngx_event_debug_connection,0,0,NULL },ngx_null_command
}

然后在看一下每个指令处理函数的代码。

connections指令

//把连接数记录在结构体中
static char *ngx_event_connections(ngx_conf_t *cf, ngx_command_t *cmd,void *conf)
{ngx_event_conf_t  *ecf = conf;ngx_str_t  *value;// 已经初始化过了if (ecf->connections != NGX_CONF_UNSET_UINT) {return "is duplicate" ;}// 从配置文件中解析出来的配置value = cf->args->elts;// 字符串转成整形ecf->connections = ngx_atoi(value[1].data, value[1].len);if (ecf->connections == (ngx_uint_t) NGX_ERROR) {ngx_conf_log_error(NGX_LOG_EMERG, cf, 0,"invalid number \"%s\"", value[1].data);return NGX_CONF_ERROR;}cf->cycle->connection_n = ecf->connections;return NGX_CONF_OK;
}

use指令

// 记录use和name的值
static char *ngx_event_use(ngx_conf_t *cf, ngx_command_t *cmd, void *conf)
{ngx_event_conf_t  *ecf = conf;ngx_int_t             m;ngx_str_t            *value;ngx_event_conf_t     *old_ecf;ngx_event_module_t   *module;if (ecf->use != NGX_CONF_UNSET_UINT) {return "is duplicate" ;}value = cf->args->elts;if (cf->cycle->old_cycle->conf_ctx) {old_ecf = ngx_event_get_conf(cf->cycle->old_cycle->conf_ctx,ngx_event_core_module);} else {old_ecf = NULL;}// 判断使用哪个事件驱动模块,比如use epoll,则使用epollfor (m = 0; ngx_modules[m]; m++) {if (ngx_modules[m]->type != NGX_EVENT_MODULE) {continue;}module = ngx_modules[m]->ctx;if (module->name->len == value[1].len) {if (ngx_strcmp(module->name->data, value[1].data) == 0) {// 存储事件模块的下标,而不是字符串ecf->use = ngx_modules[m]->ctx_index;ecf->name = module->name->data;if (ngx_process == NGX_PROCESS_SINGLE&& old_ecf&& old_ecf->use != ecf->use){ngx_conf_log_error(NGX_LOG_EMERG, cf, 0,"when the server runs without a master process ""the \"%s\" event type must be the same as ""in previous configuration - \"%s\" ""and it can not be changed on the fly, ""to change it you need to stop server ""and start it again",value[1].data, old_ecf->name);return NGX_CONF_ERROR;}return NGX_CONF_OK;}}}ngx_conf_log_error(NGX_LOG_EMERG, cf, 0,"invalid event type \"%s\"", value[1].data);return NGX_CONF_ERROR;
}

multi_accept指令

char *ngx_conf_set_flag_slot(ngx_conf_t *cf, ngx_command_t *cmd, void *conf)
{char  *p = conf;ngx_str_t        *value;ngx_flag_t       *fp;ngx_conf_post_t  *post;fp = (ngx_flag_t *) (p + cmd->offset);if (*fp != NGX_CONF_UNSET) {return "is duplicate";}value = cf->args->elts;// on代表开启if (ngx_strcasecmp(value[1].data, "on") == 0) {*fp = 1;} else if (ngx_strcasecmp(value[1].data, "off") == 0) {*fp = 0;} else {ngx_conf_log_error(NGX_LOG_EMERG, cf, 0,"invalid value \"%s\" in \"%s\" directive, ""it must be \"on\" or \"off\"",value[1].data, cmd->name.data);return NGX_CONF_ERROR;}// 设置完值后执行后置函数,一般是参数检查if (cmd->post) {post = cmd->post;return post->post_handler(cf, post, fp);}return NGX_CONF_OK;
}

accept_mutex指令(见上面的ngx_conf_set_flag_slot函数)

该指令配置了handle函数
ngx_conf_post_t  ngx_accept_mutex_post = { ngx_accept_mutex_check } 
static char *ngx_accept_mutex_check(ngx_conf_t *cf, void *post, void *data)
{
// 不支持该功能,重置字段
#if !(NGX_HAVE_ATOMIC_OPS)ngx_flag_t *fp = data;*fp = 0;ngx_conf_log_error(NGX_LOG_WARN, cf, 0,"\"accept_mutex\" is not supported, ignored");#endifreturn NGX_CONF_OK;
}

accept_mutex_delay指令

char *ngx_conf_set_msec_slot(ngx_conf_t *cf, ngx_command_t *cmd, void *conf)
{char  *p = conf;ngx_msec_t       *msp;ngx_str_t        *value;ngx_conf_post_t  *post;msp = (ngx_msec_t *) (p + cmd->offset);if (*msp != NGX_CONF_UNSET_MSEC) {return "is duplicate";}value = cf->args->elts;// 把字符串解析成时间*msp = ngx_parse_time(&value[1], 0);if (*msp == (ngx_msec_t) NGX_ERROR) {return "invalid value";}if (*msp == (ngx_msec_t) NGX_PARSE_LARGE_TIME) {return "value must be less than 597 hours";}if (cmd->post) {post = cmd->post;return post->post_handler(cf, post, msp);}return NGX_CONF_OK;
}

debug_connection指令

static char *ngx_event_debug_connection(ngx_conf_t *cf, ngx_command_t *cmd,void *conf)
{
// 配置后需要开启debug参数
#if (NGX_DEBUG)ngx_event_conf_t  *ecf = conf;in_addr_t       *addr;ngx_str_t       *value;struct hostent  *h;value = cf->args->elts;/* AF_INET only */if (!(addr = ngx_push_array(&ecf->debug_connection))) {return NGX_CONF_ERROR;}// 将ip转成长整型*addr = inet_addr((char *) value[1].data);// 转成功则返回if (*addr != INADDR_NONE) {return NGX_OK;}// 没转成功可能是一个主机字符串,获取该主机对应的ip信息h = gethostbyname((char *) value[1].data);if (h == NULL || h->h_addr_list[0] == NULL) {ngx_conf_log_error(NGX_LOG_EMERG, cf, 0,"host %s not found", value[1].data);return NGX_CONF_ERROR;
}*addr = *(in_addr_t *)(h->h_addr_list[0]);#elsengx_conf_log_error(NGX_LOG_WARN, cf, 0,"\"debug_connection\" is ignored, you need to rebuild ""nginx using --with-debug option to enable it");#endifreturn NGX_OK;
}

还有事件驱动模块的一些配置,比如epoll模块的epoll_events指令,几乎都是简单地对相应模块的配置结构体的某个字段进行赋值。

到这里,event模块关于配置文件解析的部分就结束了。主要就是对下图中最右边的那个数组里的结构体进行赋值。在这里插入图片描述

这篇关于nginx0.1.0之event模块初始化源码分析(3)的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/853493

相关文章

Nginx屏蔽服务器名称与版本信息方式(源码级修改)

《Nginx屏蔽服务器名称与版本信息方式(源码级修改)》本文详解如何通过源码修改Nginx1.25.4,移除Server响应头中的服务类型和版本信息,以增强安全性,需重新配置、编译、安装,升级时需重复... 目录一、背景与目的二、适用版本三、操作步骤修改源码文件四、后续操作提示五、注意事项六、总结一、背景与

Android实现图片浏览功能的示例详解(附带源码)

《Android实现图片浏览功能的示例详解(附带源码)》在许多应用中,都需要展示图片并支持用户进行浏览,本文主要为大家介绍了如何通过Android实现图片浏览功能,感兴趣的小伙伴可以跟随小编一起学习一... 目录一、项目背景详细介绍二、项目需求详细介绍三、相关技术详细介绍四、实现思路详细介绍五、完整实现代码

Python中logging模块用法示例总结

《Python中logging模块用法示例总结》在Python中logging模块是一个强大的日志记录工具,它允许用户将程序运行期间产生的日志信息输出到控制台或者写入到文件中,:本文主要介绍Pyt... 目录前言一. 基本使用1. 五种日志等级2.  设置报告等级3. 自定义格式4. C语言风格的格式化方法

Python 基于http.server模块实现简单http服务的代码举例

《Python基于http.server模块实现简单http服务的代码举例》Pythonhttp.server模块通过继承BaseHTTPRequestHandler处理HTTP请求,使用Threa... 目录测试环境代码实现相关介绍模块简介类及相关函数简介参考链接测试环境win11专业版python

Nginx添加内置模块过程

《Nginx添加内置模块过程》文章指导如何检查并添加Nginx的with-http_gzip_static模块:确认该模块未默认安装后,需下载同版本源码重新编译,备份替换原有二进制文件,最后重启服务验... 目录1、查看Nginx已编辑的模块2、Nginx官网查看内置模块3、停止Nginx服务4、Nginx

python urllib模块使用操作方法

《pythonurllib模块使用操作方法》Python提供了多个库用于处理URL,常用的有urllib、requests和urlparse(Python3中为urllib.parse),下面是这些... 目录URL 处理库urllib 模块requests 库urlparse 和 urljoin编码和解码

Android 缓存日志Logcat导出与分析最佳实践

《Android缓存日志Logcat导出与分析最佳实践》本文全面介绍AndroidLogcat缓存日志的导出与分析方法,涵盖按进程、缓冲区类型及日志级别过滤,自动化工具使用,常见问题解决方案和最佳实... 目录android 缓存日志(Logcat)导出与分析全攻略为什么要导出缓存日志?按需过滤导出1. 按

创建springBoot模块没有目录结构的解决方案

《创建springBoot模块没有目录结构的解决方案》2023版IntelliJIDEA创建模块时可能出现目录结构识别错误,导致文件显示异常,解决方法为选择模块后点击确认,重新校准项目结构设置,确保源... 目录创建spChina编程ringBoot模块没有目录结构解决方案总结创建springBoot模块没有目录

Linux中的HTTPS协议原理分析

《Linux中的HTTPS协议原理分析》文章解释了HTTPS的必要性:HTTP明文传输易被篡改和劫持,HTTPS通过非对称加密协商对称密钥、CA证书认证和混合加密机制,有效防范中间人攻击,保障通信安全... 目录一、什么是加密和解密?二、为什么需要加密?三、常见的加密方式3.1 对称加密3.2非对称加密四、

MySQL中读写分离方案对比分析与选型建议

《MySQL中读写分离方案对比分析与选型建议》MySQL读写分离是提升数据库可用性和性能的常见手段,本文将围绕现实生产环境中常见的几种读写分离模式进行系统对比,希望对大家有所帮助... 目录一、问题背景介绍二、多种解决方案对比2.1 原生mysql主从复制2.2 Proxy层中间件:ProxySQL2.3