设为首页 收藏本站
查看: 1226|回复: 0

[经验分享] nginx 处理请求原理和每个阶段的意思

[复制链接]

尚未签到

发表于 2016-12-27 10:08:22 | 显示全部楼层 |阅读模式
  
  适用
  Nginx通过fastcgi调用python, Perl和C++等写的CGI程序。PHP有PHP-FPM (FastCGI Process Manager),这个对PHP来说更好的FastCGI进程管理器,关于PHP-FPM的更多信息,请Google php-fpm+nginx。
  原理
  Nginx并不提供支持对外部程序的直接调用或者解析(所以缺少像apache里的mod_php这样的模块),所有的外部程序(包括PHP)必须通过fastcgi接口来调用,在Linux下接口是socket (文件socket或者Internet socket)。所以为了调用CGI程序,我们需要一个fastcgi的wrapper,这个wrapper绑定在某个固定socket上(比如端口或者文件socket),当nginx将CGI请求发送给这个socket的时候,wrapper接纳请求并fork一个新的线程,这个线程调用外部的程序或者解释器处理脚本并读取返回值,而wrapper再将返回的数据(网页或者图片等)通过fastcgi将数据通过那个固定的socket传递给nginx。示意图如下:

http://huichen.org/wp-content/uploads/2010/03/nginx-fastcgi.pngNingx的FastCGI原理

  所以,我们首先需要一个wrapper,这个wrapper需要完成的工作:

  • 通过调用fastcgi(库)的函数通过socket和ningx通信(读写socket是fastcgi内部实现的功能,对wrapper是非透明的)
  • 调度thread,进行fork和kill
  • 和application进行通信


  在编写nginx的http的模块的时候,需要在各个阶段对http请求做相应的处理,以达到不同的目的,
  比如请求发起的时候是否有访问权限、内容生成的时候进行过滤或者其它处理等等。
  如果在编译nginx模块内注册的处理阶段不正确会导致达不到想要的结果,比如你想处理内容的时候内容实际上这个时候是没有的,如此等等。
  在nginx内部定义了多个阶段的类型以满足不同的处理要求(ngx_http_core_module.h中,不同版本不一样):
  
typedef enum {
   NGX_HTTP_POST_READ_PHASE = 0,
  

   NGX_HTTP_SERVER_REWRITE_PHASE,

NGX_HTTP_FIND_CONFIG_PHASE,
NGX_HTTP_REWRITE_PHASE,
NGX_HTTP_POST_REWRITE_PHASE,

NGX_HTTP_PREACCESS_PHASE,

NGX_HTTP_ACCESS_PHASE,
NGX_HTTP_POST_ACCESS_PHASE,

NGX_HTTP_TRY_FILES_PHASE,
NGX_HTTP_CONTENT_PHASE,

NGX_HTTP_LOG_PHASE
} ngx_http_phases;
  
  各对应的意思分别为:
  NGX_HTTP_POST_READ_PHASE = 0 //读取请求阶段
NGX_HTTP_SERVER_REWRITE_PHASE //URI转换阶段
NGX_HTTP_FIND_CONFIG_PHASE //查找相应的配置来执行阶段
NGX_HTTP_REWRITE_PHASE //URI转换阶段(不太清楚此处)
NGX_HTTP_POST_REWRITE_PHASE //对转换后的URL结果进行处理的阶段
NGX_HTTP_PREACCESS_PHASE //权限检查准备阶段
NGX_HTTP_ACCESS_PHASE //权限检查阶段
NGX_HTTP_POST_ACCESS_PHASE //对权限检查结果进行处理阶段
NGX_HTTP_TRY_FILES_PHASE //处理配置中的try_files阶段
NGX_HTTP_CONTENT_PHASE //处理生成返回数据阶段(此处认为不太细,当然有filter也可以忽略)
NGX_HTTP_LOG_PHASE //记录日志处理阶段,具体说明应当是请求完成后,关闭请求时处理

  从这个配置中可以分析出来nginx在处理请求的整个流程,流程是从头执行到尾的,可见LOG是放在最后面执行,对于内容段的处理一般都是在 filter模块中去做,在NGX_HTTP_LOG_PHASE阶段注册的处理段也不能获取到返回的数据,返回数据在发送至客户端后就直接给释放了。
  因此,在各个阶段处理时应当清楚这个阶段的数据准备情况。http://www.nginx.com.cn/wp-includes/images/smilies/icon_smile.gif
  

Nginx配置指令的执行顺序(一)
  NGINX源码关于HTTP请求的概要解析:参考文章:Nginx的HTTP请求处理


  • typedefenum{
  • //0读取请求phase
  • NGX_HTTP_POST_READ_PHASE=0,
  • //1这个阶段主要是处理全局的(serverblock)的rewrite。
  • NGX_HTTP_SERVER_REWRITE_PHASE,
  • //2这个阶段主要是通过uri来查找对应的location,然后根据loc_conf设置r的相应变量
  • //e.g.根据location内配置的具体命令设置r->content_handler,到NGX_HTTP_CONTENT_PHASE调用
  • NGX_HTTP_FIND_CONFIG_PHASE,
  • //3这个主要处理location的rewrite
  • NGX_HTTP_REWRITE_PHASE,
  • //4postrewrite,这个主要是进行一些校验以及收尾工作,以便于交给后面的模块。
  • NGX_HTTP_POST_REWRITE_PHASE,
  • //5比如流控这种类型的access就放在这个phase,也就是说它主要是进行一些比较粗粒度的access。
  • NGX_HTTP_PREACCESS_PHASE,
  • //6这个比如存取控制,权限验证就放在这个phase,一般来说处理动作是交给下面的模块做的.这个主要是做一些细粒度的access
  • NGX_HTTP_ACCESS_PHASE,
  • //7一般来说当上面的access模块得到access_code之后就会由这个模块根据access_code来进行操作
  • NGX_HTTP_POST_ACCESS_PHASE,
  • //8try_file模块,也就是对应配置文件中的try_files指令,可接收多个路径作为参数,当前一个路径的资源无法找到,则自动查找下一个路径
  • NGX_HTTP_TRY_FILES_PHASE,
  • //9内容处理模块
  • NGX_HTTP_CONTENT_PHASE,
  • //10log模块
  • NGX_HTTP_LOG_PHASE
  • }ngx_http_phases;
  

  

  正常情况下,我们可以通过如下的方式来注册我们自己的处理模块:
  static ngx_int_t
ngx_http_xxx_init(ngx_conf_t *cf)
{
ngx_http_handler_pt *h;
ngx_http_core_main_conf_t *cmcf;

  cmcf = ngx_http_conf_get_module_main_conf(cf, ngx_http_core_module);
  h = ngx_array_push(&cmcf->phases[NGX_HTTP_CONTENT_PHASE].handlers);
if (h == NULL) {
return NGX_ERROR;
}

  *h = ngx_http_xxx_handler;
  return NGX_OK;
}

  且ngx_http_xxx_up_handler的返回值只能是如下几个:
  NGX_OK //处理成功,进入下一阶段
NGX_DECLINED //放弃处理
NGX_AGAIN || NGX_DONE //处理完成,返回该值会触发请求
NGX_ERROR || NGX_HTTP_.. //处理错误或者HTTP的其它状态值

  另外对于NGX_HTTP_CONTENT_PHASE阶段,实际上还有另外一种注册方式,Just like this:
  static char *
ngx_http_xxx_server(ngx_conf_t *cf, ngx_command_t *cmd, void *conf)
{
ngx_str_t *value;
ngx_url_t u;
ngx_http_core_loc_conf_t *clcf;

  clcf = ngx_http_conf_get_module_loc_conf(cf, ngx_http_core_module);
  clcf->handler = ngx_http_xxx_handler;
  if (clcf->name.data[clcf->name.len - 1] == ‘/’) {
clcf->auto_redirect = 1;
}

  return NGX_CONF_OK;
}

  不过这样子,你要做的东西就太多了,更多的情况下考虑下upstream结合或者对请求进行特殊处理,比如对于分布式存储的分发,需要对请求处理和文件系统关联时、比如请求的数据直接交给特殊的SERVER来拿内容时。呵呵.http://www.nginx.com.cn/wp-includes/images/smilies/icon_smile.gif

运维网声明 1、欢迎大家加入本站运维交流群:群②:261659950 群⑤:202807635 群⑦870801961 群⑧679858003
2、本站所有主题由该帖子作者发表,该帖子作者与运维网享有帖子相关版权
3、所有作品的著作权均归原作者享有,请您和我们一样尊重他人的著作权等合法权益。如果您对作品感到满意,请购买正版
4、禁止制作、复制、发布和传播具有反动、淫秽、色情、暴力、凶杀等内容的信息,一经发现立即删除。若您因此触犯法律,一切后果自负,我们对此不承担任何责任
5、所有资源均系网友上传或者通过网络收集,我们仅提供一个展示、介绍、观摩学习的平台,我们不对其内容的准确性、可靠性、正当性、安全性、合法性等负责,亦不承担任何法律责任
6、所有作品仅供您个人学习、研究或欣赏,不得用于商业或者其他用途,否则,一切后果均由您自己承担,我们对此不承担任何法律责任
7、如涉及侵犯版权等问题,请您及时通知我们,我们将立即采取措施予以解决
8、联系人Email:admin@iyunv.com 网址:www.yunweiku.com

所有资源均系网友上传或者通过网络收集,我们仅提供一个展示、介绍、观摩学习的平台,我们不对其承担任何法律责任,如涉及侵犯版权等问题,请您及时通知我们,我们将立即处理,联系人Email:kefu@iyunv.com,QQ:1061981298 本贴地址:https://www.iyunv.com/thread-320077-1-1.html 上篇帖子: Nginx+Apache搭建前后端web生产环境 下篇帖子: Nginx+PHP-FPM的域Socket配置方法
您需要登录后才可以回帖 登录 | 立即注册

本版积分规则

扫码加入运维网微信交流群X

扫码加入运维网微信交流群

扫描二维码加入运维网微信交流群,最新一手资源尽在官方微信交流群!快快加入我们吧...

扫描微信二维码查看详情

客服E-mail:kefu@iyunv.com 客服QQ:1061981298


QQ群⑦:运维网交流群⑦ QQ群⑧:运维网交流群⑧ k8s群:运维网kubernetes交流群


提醒:禁止发布任何违反国家法律、法规的言论与图片等内容;本站内容均来自个人观点与网络等信息,非本站认同之观点.


本站大部分资源是网友从网上搜集分享而来,其版权均归原作者及其网站所有,我们尊重他人的合法权益,如有内容侵犯您的合法权益,请及时与我们联系进行核实删除!



合作伙伴: 青云cloud

快速回复 返回顶部 返回列表