Nginx源码分析之开篇

开发 前端
Nginx源码构思精巧,每每阅读颇有收获,写此文与各码农分享。阅读从main开始,流程非常清晰。

Nginx源码构思精巧,每每阅读颇有收获,写此文与各码农分享。

阅读从main开始,流程非常清晰。

  1. /* 系统错误初始化,将构建ngx_sys_errlist */ 
  2.  
  3. ngx_strerror_init();  
  4. /* 选项处理 */ 
  5. ngx_get_options(argc, argv);  
  6.  
  7. /* 时间初始化, 当前时间:ngx_current_msec */ 
  8. ngx_time_init();  
  9.  
  10. /* 日志初始化 */ 
  11. log = ngx_log_init(ngx_prefix);  
  12.  
  13. /* 选项处理 */ 
  14. ngx_save_argv(..., argc, argv);  
  15. ngx_process_options(...);  
  16.  
  17. /* 操作系统初始化处理 */ 
  18. ngx_os_init(log);  
  19.  
  20. /* 模块点名, ngx_modules代表所有模块,是个数组 */ 
  21. ngx_max_module = 0;  
  22. for (i = 0; ngx_modules[i]; i++) {  
  23.     ngx_modules[i]->index = ngx_max_module++;  
  24. }  
  25.  
  26. /*  
  27.  * 系统初始化,这里将发生配置文件解析,模块上下文注册钩子调用,模块初始化  
  28.  * module     : 模块  
  29.  * commands  : 模块指令集,负责解析配置文件的选项,一个指令对应一个配置选项  
  30.  * conf            : 模块配置结构体,指令解析后的值就是存储在这个里面,每个模块都有自已的一个conf  
  31.  * ctx              : 模块上下文,有四种,core, event, http, mail,有注册钩子功能。比如 create conf, init conf  
  32.  */ 
  33. cycle = ngx_init_cycle(&init_cycle);  
  34.  
  35. /* 创建进程id文件 */ 
  36. ngx_create_pidfile(&ccf->pid, cycle->log);  
  37.  
  38. /*  
  39.  * 进程处理  
  40.  * 主进程(master)产生多个工作进程(worker)  
  41.  * 这里将做各模块进程初始化,监听,接受,请求处理,还有信号等  
  42.  */ 
  43. ngx_master_process_cycle(cycle) {  
  44.     ngx_start_worker_processes(cycle, ccf->worker_processes,  
  45.                                NGX_PROCESS_RESPAWN) {  
  46.         for ( ... ) {  
  47.             ngx_spawn_process(cycle, ngx_worker_process_cycle, NULL,  
  48.                           "worker process", type);  
  49.         }  
  50.     }  
  51. }  
  52.  
  53.  
  54. goto: 继续  
  55. /* 生产进程都在这个函数里处理 */ 
  56. ngx_worker_process_cycle {  
  57.     /* 各模块进程注册钩子调用 */ 
  58.     ngx_worker_process_init();  
  59.      
  60.     for ( ;; ) {  
  61.         ngx_process_events_and_timers(cycle);  
  62.     }  
  63. }  
  64.      
  65.  
  66. goto: 继续  
  67. ngx_process_events_and_timers(cycle) {  
  68.     /*  
  69.      * 定时器,用红黑树实现,这里找出某个event(事件)的超时时间  
  70.      * 每个事件在红黑树里的key的值为:ngx_currnet_msec + 超时时间(默认60s)  
  71.      * timer的值为 -1 (如果没有事件),或 ngx_current_msec - key  
  72.      */ 
  73.     timer = ngx_event_find_timer();  
  74.  
  75.     /* epoll 机制,这里将做 epoll_wait(..., timer); */ 
  76.     ngx_process_events(cycle, timer, flags);  
  77.      
  78.     /* 先处理所有可能超时的事件,如果超时,将event的timedout设为1,并且马上event->handler(ev); */ 
  79.     ngx_event_expire_timers();  
  80.  
  81.     /* 事件存队列方式,开始遍历,调用 */ 
  82.     ngx_event_process_posted(cycle, &ngx_posted_events);  
  83. }  
  84.  
  85.  
  86. /*  
  87.  * 几个重要的结构体  
  88.  * ngx_listening_t       : 监听套接字的结构体,比如地址,端口等  
  89.  * ngx_connection_t   : 每个socket将对应一个connection,意为连接,里面存着 fd,read(读事件),  
  90.  *                               write(写事件) 等。  
  91.  * ngx_event_t    : 事件结构体,有个重要的函数指针handler,fd事件被驱动时,将调用这个函数。  
  92.  *                               它有几个重要成员  
  93.  *                                     timer_set  : 每个event在epoll_wait前,要先进入定时器红黑树,这个标记就是  
  94.  *                                                       标记是否在定时器里, 超时处理用的.  
  95.  *                                     active       : 当ngx_add_event里(添加或更新事件) 进入epoll时,会置为1.  
  96.  *                                     ready       : 进入事件队列里,将置为1,只有为1,它对应的socket fd才可以读  
  97.  *                                     timedout   : 此事件对应的socket fd将视为超时  
  98.  */ 
  99.  
  100.  
  101. /*  
  102.  * http处理  
  103.  * 当 listen fd 有连接过来时,它将调用函数 ngx_http_init_connection  
  104.  * 当 accept fd 有传送东东时,它将调用函数 ngx_http_init_request,所以的处理都将从这函数开始  
  105.  */ 
  106. ngx_http_init_request {  
  107.     ngx_http_process_request_line {  
  108.         /* 读请求头 */ 
  109.         ngx_http_read_request_header(...);  
  110.  
  111.         /* 解析请求行 */ 
  112.         ngx_http_parse_request_line(...);  
  113.  
  114.         /* 处理请求头部信息 */ 
  115.         ngx_http_process_request_headers(...) {  
  116.             for ( ;; ) {  
  117.                 /* 解析每一行 */ 
  118.                 ngx_http_parse_header_line(...);  
  119.             }  
  120.  
  121.             /* 解析之后对所有行的处理 */ 
  122.             ngx_http_process_request_header(...);  
  123.  
  124.             /* 真正开始处理请求 */ 
  125.             ngx_http_process_request(r) {  
  126.                 ngx_http_handler(r) {  
  127.                     /* 非常巧妙的设计处理即将开始 */ 
  128.                     ngx_http_core_run_phases(r);  
  129.                 }  
  130.             }  
  131.         }  
  132.  
  133.     }  
  134. }  
  135.  
  136.  
  137. /*  
  138.  * 精巧的设计函数:责任链模式  
  139.  * http的每个请求可以分为好几个阶段  
  140.  *        规则重写(rewrite)  
  141.  *        处理配置(不同url有不同的配置)  
  142.  *        权限访问处理  
  143.  *        核心内容处理(是走fastcgi,还是直接输出或从缓存获取等)  
  144.  *        日志处理  
  145.  *  
  146.  * 每个阶段都可以由好几个模块处理,这些模块组成一个链,  
  147.  * 这是设计模式里的一种,责任链模式  
  148.  */ 
  149.  ngx_http_core_run_phases(r) {  
  150.     while (ph[r->phase_handler].checker) {  
  151.  
  152.         rc = ph[r->phase_handler].checker(r, &ph[r->phase_handler]);  
  153.  
  154.         if (rc == NGX_OK) {  
  155.             return;  
  156.         }  
  157.     }  
  158.  }  

 原文链接:http://my.oschina.net/fqing/blog/79145

责任编辑:林师授 来源: OSCHINA
相关推荐

2021-09-22 07:57:23

Vue3 插件Vue应用

2011-08-16 09:34:34

Nginx

2011-05-26 10:05:48

MongoDB

2023-02-26 08:42:10

源码demouseEffect

2011-05-26 16:18:51

Mongodb

2021-03-23 09:17:58

SpringMVCHttpServletJavaEE

2024-06-13 07:55:19

2021-07-06 09:29:38

Cobar源码AST

2020-07-28 08:54:39

内核通信Netlink

2012-09-06 10:07:26

jQuery

2021-09-05 07:35:58

lifecycleAndroid组件原理

2009-07-08 13:22:30

JDK源码分析Set

2017-01-12 14:52:03

JVMFinalRefere源码

2022-01-06 07:06:52

KubernetesResourceAPI

2022-05-30 07:36:54

vmstoragevmselect

2022-08-27 08:02:09

SQL函数语法

2021-10-19 11:22:08

SentinelGo源码

2017-02-09 15:46:09

数据分析互联网

2019-09-09 06:30:06

Springboot程序员开发

2023-03-17 07:53:20

K8sAPIServerKubernetes
点赞
收藏

51CTO技术栈公众号