亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb

首頁 > 網站 > Nginx > 正文

全面了解Nginx中的HTTP協議相關模塊配置

2024-08-30 12:28:11
字體:
來源:轉載
供稿:網友

要理解 HTTP 模塊配置解析的過程,首先需要對 nginx 的配置文件結構做一個了解

nginx 的配置文件是用樹狀結構組織的,每個 NGX_CORE_MODULE 作為根統領著其下的所有配置項

而如下圖所示,HTTP 模塊的配置被分成了 main、server、location 三層

Nginx,HTTP,配置,Nginx配置

整個 nginx 配置解析的過程其實就是這棵樹的深度遍歷過程

而遍歷 HTTP 子樹的函數就是下面要介紹的 ngx_http_block

 

配置文件解析 -- http 配置塊
當我們需要使用 http 模塊的時候,我們需要在配置文件中加入 http 配置塊:

http { // http 配置塊 {{{  include    mime.types;  default_type application/octet-stream;  #log_format main '$remote_addr - $remote_user [$time_local] "$request" '  #         '$status $body_bytes_sent "$http_referer" '  #         '"$http_user_agent" "$http_x_forwarded_for"';  #access_log logs/access.log main;  sendfile    on;  #tcp_nopush   on;  #keepalive_timeout 0;  keepalive_timeout 65;  #gzip on;  server {    listen    8001;    server_name localhost;    #autoindex  on;    #charset koi8-r;    #access_log logs/host.access.log main;    location / {      root  /var/www/;      index index.html index.htm index.php;

在 http 配置塊中,我們配置了 http 連接相關的信息,HTTP 框架也正是從這里啟動的.

在 nginx 初始化的過程中,執行了 ngx_init_cycle 函數,其中進行了配置文件解析,調用了 ngx_conf_parse 函數

 

配置文件解析

函數 ngx_conf_handler 根據配置項的 command 調用了對應的 set 回調函數

// static ngx_int_t ngx_conf_handler(ngx_conf_t *cf, ngx_int_t last)// 配置項解析 {{{static ngx_int_tngx_conf_handler(ngx_conf_t *cf, ngx_int_t last){  char      *rv;  void      *conf, **confp;  ngx_uint_t   i, found;  ngx_str_t   *name;  ngx_command_t *cmd;  name = cf->args->elts;  found = 0;  for (i = 0; ngx_modules[i]; i++) {    cmd = ngx_modules[i]->commands;    if (cmd == NULL) {      continue;    }    for ( /* void */ ; cmd->name.len; cmd++) {      if (name->len != cmd->name.len) {        continue;      }      if (ngx_strcmp(name->data, cmd->name.data) != 0) {        continue;      }

閱讀各模塊的 ngx_command_t 命令配置結構,可以找到:

// static ngx_command_t ngx_http_commands// http 模塊命令結構 {{{static ngx_command_t ngx_http_commands[] = {  { ngx_string("http"),   NGX_MAIN_CONF|NGX_CONF_BLOCK|NGX_CONF_NOARGS,   ngx_http_block,   0,   0,   NULL },   ngx_null_command}; // }}}

 

http 配置塊解析 -- ngx_http_block

Nginx,HTTP,配置,Nginx配置

在解析到 http 配置塊時,執行了對應的 set 回調函數 ngx_http_block

static char *ngx_http_block(ngx_conf_t *cf, ngx_command_t *cmd, void *conf){  char            *rv;  ngx_uint_t          mi, m, s;  ngx_conf_t          pcf;  ngx_http_module_t      *module;  ngx_http_conf_ctx_t     *ctx;  ngx_http_core_loc_conf_t  *clcf;  ngx_http_core_srv_conf_t  **cscfp;  ngx_http_core_main_conf_t  *cmcf;  /* the main http context */  ctx = ngx_pcalloc(cf->pool, sizeof(ngx_http_conf_ctx_t));  if (ctx == NULL) {    return NGX_CONF_ERROR;  } // 創建 http 配置結構,保存所有 http 模塊的配置信息  *(ngx_http_conf_ctx_t **) conf = ctx;  /* count the number of the http modules and set up their indices */  ngx_http_max_module = 0;  for (m = 0; ngx_modules[m]; m++) {    if (ngx_modules[m]->type != NGX_HTTP_MODULE) {      continue;    } // 重新設定 HTTP 模塊序號    ngx_modules[m]->ctx_index = ngx_http_max_module++;  }  /* the http main_conf context, it is the same in the all http contexts */  ctx->main_conf = ngx_pcalloc(cf->pool,                 sizeof(void *) * ngx_http_max_module);  if (ctx->main_conf == NULL) {    return NGX_CONF_ERROR;  }  /*   * the http null srv_conf context, it is used to merge   * the server{}s' srv_conf's   */  ctx->srv_conf = ngx_pcalloc(cf->pool, sizeof(void *) * ngx_http_max_module);  if (ctx->srv_conf == NULL) {    return NGX_CONF_ERROR;  }  /*   * the http null loc_conf context, it is used to merge   * the server{}s' loc_conf's   */  ctx->loc_conf = ngx_pcalloc(cf->pool, sizeof(void *) * ngx_http_max_module);  if (ctx->loc_conf == NULL) {    return NGX_CONF_ERROR;  }  /*   * create the main_conf's, the null srv_conf's, and the null loc_conf's   * of the all http modules   */  for (m = 0; ngx_modules[m]; m++) {    if (ngx_modules[m]->type != NGX_HTTP_MODULE) {      continue;    }    module = ngx_modules[m]->ctx;    mi = ngx_modules[m]->ctx_index;    if (module->create_main_conf) {  // 調用每個模塊的 create_main_conf 回調函數  // 創建自己的 main_conf  //  // ngx_http_core_module  ngx_http_core_create_main_conf  // ngx_http_log_module  ngx_http_log_create_main_conf  // ngx_http_upstream_module  ngx_http_upstream_create_main_conf  // ngx_http_map_module  ngx_http_map_create_conf  // ngx_http_ssi_filter_module ngx_http_ssi_create_main_conf  // ngx_http_charset_filter_module ngx_http_charset_create_main_conf      ctx->main_conf[mi] = module->create_main_conf(cf);      if (ctx->main_conf[mi] == NULL) {        return NGX_CONF_ERROR;      }    }    if (module->create_srv_conf) {  // 調用每個模塊的 create_srv_conf 回調函數  // 創建自己的 srv_conf  //  // ngx_http_core_module   ngx_http_core_create_srv_conf  // ngx_http_ssl_module   ngx_http_ssl_create_srv_conf  // ngx_http_upstream_hash_module ngx_http_upstream_hash_create_conf  // ngx_http_upstream_least_conn_module ngx_http_upstream_least_conn_create_conf  // ngx_http_upstream_keepalive_module ngx_http_upstream_keepalive_create_conf      ctx->srv_conf[mi] = module->create_srv_conf(cf);      if (ctx->srv_conf[mi] == NULL) {        return NGX_CONF_ERROR;      }    }    if (module->create_loc_conf) {  // 調用每個模塊的 create_loc_conf 回調函數  // 創建自己的 loc_conf  //  // ngx_http_core_module  ngx_http_core_create_loc_conf  // ngx_http_log_module  ngx_http_log_create_loc_conf  // ngx_http_gzip_static_module ngx_http_gzip_static_create_conf  // ngx_http_autoindex_module ngx_http_autoindex_create_loc_conf  // ngx_http_index_module  ngx_http_index_create_loc_conf  // ngx_http_auth_basic_module ngx_http_auth_basic_create_loc_conf  // ngx_http_access_module  ngx_http_access_create_loc_conf  // ngx_http_limit_conn_module ngx_http_limit_conn_create_conf  // ngx_http_limit_req_module ngx_http_limit_req_create_conf  // ngx_http_referer_module  ngx_http_referer_create_conf  // ngx_http_rewrite_module  ngx_http_rewrite_create_loc_conf  // ngx_http_proxy_module  ngx_http_proxy_create_loc_conf  // ngx_http_fastcgi_module  ngx_http_fastcgi_create_loc_conf  // ngx_http_uwsgi_module  ngx_http_uwsgi_create_loc_conf  // ngx_http_scgi_module  ngx_http_scgi_create_loc_conf  // ngx_http_memcached_module ngx_http_memcached_create_loc_conf  // ngx_http_browser_module  ngx_http_browser_create_conf  // ngx_http_gzip_filter_module ngx_http_gzip_create_conf  // ngx_http_ssi_filter_module ngx_http_ssi_create_loc_conf  // ngx_http_charset_filter_module ngx_http_charset_create_loc_conf  // ngx_http_userid_filter_module ngx_http_userid_create_conf  // ngx_http_headers_filter_module ngx_http_headers_create_conf  // ngx_http_copy_filter_module ngx_http_copy_filter_create_conf      ctx->loc_conf[mi] = module->create_loc_conf(cf);      if (ctx->loc_conf[mi] == NULL) {        return NGX_CONF_ERROR;      }    }  }  pcf = *cf;  cf->ctx = ctx;  for (m = 0; ngx_modules[m]; m++) {    if (ngx_modules[m]->type != NGX_HTTP_MODULE) {      continue;    }    module = ngx_modules[m]->ctx;    if (module->preconfiguration) {  // 調用每個模塊的 preconfiguration 回調函數  //  // ngx_http_core_module  ngx_http_core_preconfiguration  // ngx_http_upstream_module  ngx_http_upstream_add_variables  // ngx_http_ssl_module  ngx_http_ssl_add_variables  // ngx_http_proxy_module  ngx_http_proxy_add_variables  // ngx_http_fastcgi_module  ngx_http_fastcgi_add_variables  // ngx_http_browser_module  ngx_http_browser_add_variable  // ngx_http_stub_status_module ngx_http_stub_status_add_variables  // ngx_http_gzip_filter_module ngx_http_gzip_add_variables  // ngx_http_ssi_filter_module ngx_http_ssi_preconfiguration  // ngx_http_userid_filter_module ngx_http_userid_add_variables      if (module->preconfiguration(cf) != NGX_OK) {        return NGX_CONF_ERROR;      }    }  }  /* parse inside the http{} block */  cf->module_type = NGX_HTTP_MODULE;  cf->cmd_type = NGX_HTTP_MAIN_CONF;  rv = ngx_conf_parse(cf, NULL);  if (rv != NGX_CONF_OK) {    goto failed;  }  /*   * init http{} main_conf's, merge the server{}s' srv_conf's   * and its location{}s' loc_conf's   */  cmcf = ctx->main_conf[ngx_http_core_module.ctx_index];  cscfp = cmcf->servers.elts;  for (m = 0; ngx_modules[m]; m++) {    if (ngx_modules[m]->type != NGX_HTTP_MODULE) {      continue;    }    module = ngx_modules[m]->ctx;    mi = ngx_modules[m]->ctx_index;    /* init http{} main_conf's */    if (module->init_main_conf) {  // 調用每個模塊的 init_main_conf 回調函數  // 初始化自己的 main_conf  //  // ngx_http_core_module  ngx_http_core_init_main_conf  // ngx_http_upstream_module ngx_http_upstream_init_main_conf  // ngx_http_ssi_filter_module ngx_http_ssi_init_main_conf      rv = module->init_main_conf(cf, ctx->main_conf[mi]);      if (rv != NGX_CONF_OK) {        goto failed;      }    } // 合并同名配置項    rv = ngx_http_merge_servers(cf, cmcf, module, mi);    if (rv != NGX_CONF_OK) {      goto failed;    }  }  /* create location trees */  for (s = 0; s < cmcf->servers.nelts; s++) {    clcf = cscfp[s]->ctx->loc_conf[ngx_http_core_module.ctx_index];    if (ngx_http_init_locations(cf, cscfp[s], clcf) != NGX_OK) {      return NGX_CONF_ERROR;    }    if (ngx_http_init_static_location_trees(cf, clcf) != NGX_OK) {      return NGX_CONF_ERROR;    }  }  if (ngx_http_init_phases(cf, cmcf) != NGX_OK) {    return NGX_CONF_ERROR;  }  if (ngx_http_init_headers_in_hash(cf, cmcf) != NGX_OK) {    return NGX_CONF_ERROR;  }  for (m = 0; ngx_modules[m]; m++) {    if (ngx_modules[m]->type != NGX_HTTP_MODULE) {      continue;    }    module = ngx_modules[m]->ctx;    if (module->postconfiguration) {  // 調用每個 HTTP 模塊的 postconfiguration 回調函數  //  // ngx_http_log_module   ngx_http_log_init  // ngx_http_static_module  ngx_http_static_init  // ngx_http_gzip_static_module  ngx_http_gzip_static_init  // ngx_http_autoindex_module  ngx_http_autoindex_init  // ngx_http_index_module  ngx_http_index_init  // ngx_http_auth_basic_module  ngx_http_auth_basic_init  // ngx_http_access_module  ngx_http_access_init  // ngx_http_limit_conn_module  ngx_http_limit_conn_init  // ngx_http_limit_req_module  ngx_http_limit_req_init  // ngx_http_rewrite_module  ngx_http_rewrite_init  // ngx_http_ssl_module   ngx_http_ssl_init  // ngx_http_write_filter_module  ngx_http_write_filter_init  // ngx_http_header_filter_module ngx_http_header_filter_init  // ngx_http_chunked_filter_module ngx_http_chunked_filter_init  // ngx_http_range_body_filter_module ngx_http_range_body_filter_init  // ngx_http_gzip_filter_module  ngx_http_gzip_filter_init  // ngx_http_postpone_filter_module ngx_http_postpone_filter_init  // ngx_http_ssi_filter_module  ngx_http_ssi_filter_init  // ngx_http_charset_filter_module ngx_http_charset_postconfiguration  // ngx_http_userid_filter_module ngx_http_userid_init  // ngx_http_headers_filter_module ngx_http_headers_filter_init  // ngx_http_copy_filter_module  ngx_http_copy_filter_init  // ngx_http_range_body_filter_module ngx_http_range_body_filter_init  // ngx_http_not_modified_filter_module ngx_http_not_modified_filter_init      if (module->postconfiguration(cf) != NGX_OK) {        return NGX_CONF_ERROR;      }    }  } // 初始化所有的變量 // 不僅包括HTTP core模塊的變量 // 也包括其他的HTTP模塊導出的變量,以及配置文件中使用 set 命令設置的變量 // 這里的初始化包括初始化hash表,以及初始化數組索引  if (ngx_http_variables_init_vars(cf) != NGX_OK) {    return NGX_CONF_ERROR;  }  /*   * http{}'s cf->ctx was needed while the configuration merging   * and in postconfiguration process   */  *cf = pcf; // 初始化 phase_engine 結構  if (ngx_http_init_phase_handlers(cf, cmcf) != NGX_OK) {    return NGX_CONF_ERROR;  }  /* optimize the lists of ports, addresses and server names */ // 創建 http 連接,并設置為監聽狀態  if (ngx_http_optimize_servers(cf, cmcf, cmcf->ports) != NGX_OK) {    return NGX_CONF_ERROR;  }  return NGX_CONF_OK;failed:  *cf = pcf;  return rv;} // }}}

這個函數中,為所有的 http 模塊都分配并創建了配置結構,同時,調用了每個模塊相應的初始化回調。

最后,調用 ngx_http_optimize_servers 創建了 http 連接,加入 cycle 的監聽數組,并設為監聽狀態。

nginx 配置文件對 http 模塊的配置分為三層:main、sever、location,因此,http 模塊上下文 ngx_http_module_t 中定義了以下六個回調函數,用來創建和保存配置信息:

  • create_main_conf
  • init_main_conf
  • create_srv_conf
  • merge_srv_conf
  • create_loc_conf
  • merge_loc_conf

在 ngx_http_block 中,循環調用了所有 NGX_HTTP_MODULE 的這六個回調函數,創建相關的配置結構。

 

server、location 配置解析 -- ngx_http_core_server、ngx_http_core_location
在調用所有 HTTP 模塊的 create_main_conf、create_srv_conf、create_loc_conf 后,所有需要配置結構的模塊都完成了配置結構的創建,于是在調用所有模塊的 preconfiguration 回調函數后,配置解析工作正式展開

通過調用 ngx_conf_parse 函數,開始了 http 配置塊的解析,并通過解析到的命令調用相應的函數

在首個 NGX_HTTP_MODULE ngx_http_core_module 的 ngx_command_t 域中包含了大量的配置指令,它們都是在http{}塊中出現的,其中包括兩個重要的指令:

{ ngx_string("listen"), NGX_HTTP_SRV_CONF|NGX_CONF_1MORE, ngx_http_core_listen, NGX_HTTP_SRV_CONF_OFFSET, 0, NULL },{ ngx_string("server"), NGX_HTTP_MAIN_CONF|NGX_CONF_BLOCK|NGX_CONF_NOARGS, ngx_http_core_server, 0, 0, NULL },{ ngx_string("location"), NGX_HTTP_SRV_CONF|NGX_HTTP_LOC_CONF|NGX_CONF_BLOCK|NGX_CONF_TAKE12, ngx_http_core_location, NGX_HTTP_SRV_CONF_OFFSET, 0, NULL },

這里配置了 listen、server 與 location 塊的解析函數 ngx_http_core_listen、ngx_http_core_server 和 ngx_http_core_location.


server 塊解析 -- ngx_http_core_server

// static char *// ngx_http_core_server(ngx_conf_t *cf, ngx_command_t *cmd, void *dummy)// http server 塊解析 {{{static char *ngx_http_core_server(ngx_conf_t *cf, ngx_command_t *cmd, void *dummy){  char            *rv;  void            *mconf;  ngx_uint_t          i;  ngx_conf_t          pcf;  ngx_http_module_t      *module;  struct sockaddr_in     *sin;  ngx_http_conf_ctx_t     *ctx, *http_ctx;  ngx_http_listen_opt_t    lsopt;  ngx_http_core_srv_conf_t  *cscf, **cscfp;  ngx_http_core_main_conf_t  *cmcf;  ctx = ngx_pcalloc(cf->pool, sizeof(ngx_http_conf_ctx_t));  if (ctx == NULL) {    return NGX_CONF_ERROR;  }  http_ctx = cf->ctx;  ctx->main_conf = http_ctx->main_conf;  /* the server{}'s srv_conf */ // 為所有 HTTP 模塊分配空間存儲 srv_conf  ctx->srv_conf = ngx_pcalloc(cf->pool, sizeof(void *) * ngx_http_max_module);  if (ctx->srv_conf == NULL) {    return NGX_CONF_ERROR;  }  /* the server{}'s loc_conf */ // 為所有 HTTP 模塊分配空間存儲 loc_conf  ctx->loc_conf = ngx_pcalloc(cf->pool, sizeof(void *) * ngx_http_max_module);  if (ctx->loc_conf == NULL) {    return NGX_CONF_ERROR;  } // 循環調用每個模塊的 create_srv_conf 與 create_loc_conf 回調,創建配置結構  for (i = 0; ngx_modules[i]; i++) {    if (ngx_modules[i]->type != NGX_HTTP_MODULE) {      continue;    }    module = ngx_modules[i]->ctx;    if (module->create_srv_conf) {      mconf = module->create_srv_conf(cf);      if (mconf == NULL) {        return NGX_CONF_ERROR;      }      ctx->srv_conf[ngx_modules[i]->ctx_index] = mconf;    }    if (module->create_loc_conf) {      mconf = module->create_loc_conf(cf);      if (mconf == NULL) {        return NGX_CONF_ERROR;      }      ctx->loc_conf[ngx_modules[i]->ctx_index] = mconf;    }  }  /* the server configuration context */  cscf = ctx->srv_conf[ngx_http_core_module.ctx_index];  cscf->ctx = ctx;  cmcf = ctx->main_conf[ngx_http_core_module.ctx_index];  cscfp = ngx_array_push(&cmcf->servers);  if (cscfp == NULL) {    return NGX_CONF_ERROR;  }  *cscfp = cscf;  /* parse inside server{} */  pcf = *cf;  cf->ctx = ctx;  cf->cmd_type = NGX_HTTP_SRV_CONF; // 解析 server 塊配置  rv = ngx_conf_parse(cf, NULL);  *cf = pcf; // 如果沒有監聽任何端口,則監聽默認的 80 或 8000端口  if (rv == NGX_CONF_OK && !cscf->listen) {    ngx_memzero(&lsopt, sizeof(ngx_http_listen_opt_t));    sin = &lsopt.u.sockaddr_in;    sin->sin_family = AF_INET;#if (NGX_WIN32)    sin->sin_port = htons(80);#else    sin->sin_port = htons((getuid() == 0) ? 80 : 8000);#endif    sin->sin_addr.s_addr = INADDR_ANY;    lsopt.socklen = sizeof(struct sockaddr_in);    lsopt.backlog = NGX_LISTEN_BACKLOG;    lsopt.rcvbuf = -1;    lsopt.sndbuf = -1;#if (NGX_HAVE_SETFIB)    lsopt.setfib = -1;#endif#if (NGX_HAVE_TCP_FASTOPEN)    lsopt.fastopen = -1;#endif    lsopt.wildcard = 1;    (void) ngx_sock_ntop(&lsopt.u.sockaddr, lsopt.socklen, lsopt.addr,               NGX_SOCKADDR_STRLEN, 1);    if (ngx_http_add_listen(cf, cscf, &lsopt) != NGX_OK) {      return NGX_CONF_ERROR;    }  }  return rv;} // }}}

 

location 塊解析 -- ngx_http_core_location

// static char *// ngx_http_core_location(ngx_conf_t *cf, ngx_command_t *cmd, void *dummy)// location 塊配置解析 {{{static char *ngx_http_core_location(ngx_conf_t *cf, ngx_command_t *cmd, void *dummy){  char           *rv;  u_char          *mod;  size_t           len;  ngx_str_t         *value, *name;  ngx_uint_t         i;  ngx_conf_t         save;  ngx_http_module_t     *module;  ngx_http_conf_ctx_t    *ctx, *pctx;  ngx_http_core_loc_conf_t *clcf, *pclcf;  ctx = ngx_pcalloc(cf->pool, sizeof(ngx_http_conf_ctx_t));  if (ctx == NULL) {    return NGX_CONF_ERROR;  }  pctx = cf->ctx;  ctx->main_conf = pctx->main_conf;  ctx->srv_conf = pctx->srv_conf;  ctx->loc_conf = ngx_pcalloc(cf->pool, sizeof(void *) * ngx_http_max_module);  if (ctx->loc_conf == NULL) {    return NGX_CONF_ERROR;  }  for (i = 0; ngx_modules[i]; i++) {    if (ngx_modules[i]->type != NGX_HTTP_MODULE) {      continue;    }    module = ngx_modules[i]->ctx;    if (module->create_loc_conf) {      ctx->loc_conf[ngx_modules[i]->ctx_index] =                          module->create_loc_conf(cf);      if (ctx->loc_conf[ngx_modules[i]->ctx_index] == NULL) {         return NGX_CONF_ERROR;      }    }  }  clcf = ctx->loc_conf[ngx_http_core_module.ctx_index];  clcf->loc_conf = ctx->loc_conf;  value = cf->args->elts;  if (cf->args->nelts == 3) {    len = value[1].len;    mod = value[1].data;    name = &value[2];    if (len == 1 && mod[0] == '=') {      clcf->name = *name;      clcf->exact_match = 1;    } else if (len == 2 && mod[0] == '^' && mod[1] == '~') {      clcf->name = *name;      clcf->noregex = 1;    } else if (len == 1 && mod[0] == '~') {      if (ngx_http_core_regex_location(cf, clcf, name, 0) != NGX_OK) {        return NGX_CONF_ERROR;      }    } else if (len == 2 && mod[0] == '~' && mod[1] == '*') {      if (ngx_http_core_regex_location(cf, clcf, name, 1) != NGX_OK) {        return NGX_CONF_ERROR;      }    } else {      ngx_conf_log_error(NGX_LOG_EMERG, cf, 0,                "invalid location modifier /"%V/"", &value[1]);      return NGX_CONF_ERROR;    }  } else {    name = &value[1];    if (name->data[0] == '=') {      clcf->name.len = name->len - 1;      clcf->name.data = name->data + 1;      clcf->exact_match = 1;    } else if (name->data[0] == '^' && name->data[1] == '~') {      clcf->name.len = name->len - 2;      clcf->name.data = name->data + 2;      clcf->noregex = 1;    } else if (name->data[0] == '~') {      name->len--;      name->data++;      if (name->data[0] == '*') {        name->len--;        name->data++;        if (ngx_http_core_regex_location(cf, clcf, name, 1) != NGX_OK) {          return NGX_CONF_ERROR;        }      } else {        if (ngx_http_core_regex_location(cf, clcf, name, 0) != NGX_OK) {          return NGX_CONF_ERROR;        }      }    } else {      clcf->name = *name;      if (name->data[0] == '@') {        clcf->named = 1;      }    }  }  pclcf = pctx->loc_conf[ngx_http_core_module.ctx_index];  if (pclcf->name.len) {    /* nested location */#if 0    clcf->prev_location = pclcf;#endif    if (pclcf->exact_match) {      ngx_conf_log_error(NGX_LOG_EMERG, cf, 0,                "location /"%V/" cannot be inside "                "the exact location /"%V/"",                &clcf->name, &pclcf->name);      return NGX_CONF_ERROR;    }    if (pclcf->named) {      ngx_conf_log_error(NGX_LOG_EMERG, cf, 0,                "location /"%V/" cannot be inside "                "the named location /"%V/"",                &clcf->name, &pclcf->name);      return NGX_CONF_ERROR;    }    if (clcf->named) {      ngx_conf_log_error(NGX_LOG_EMERG, cf, 0,                "named location /"%V/" can be "                "on the server level only",                &clcf->name);      return NGX_CONF_ERROR;    }    len = pclcf->name.len;#if (NGX_PCRE)    if (clcf->regex == NULL      && ngx_filename_cmp(clcf->name.data, pclcf->name.data, len) != 0)#else    if (ngx_filename_cmp(clcf->name.data, pclcf->name.data, len) != 0)#endif    {      ngx_conf_log_error(NGX_LOG_EMERG, cf, 0,                "location /"%V/" is outside location /"%V/"",                &clcf->name, &pclcf->name);      return NGX_CONF_ERROR;    }  } // 將 location 配置加入到 locations 配置鏈表中  if (ngx_http_add_location(cf, &pclcf->locations, clcf) != NGX_OK) {    return NGX_CONF_ERROR;  }  save = *cf;  cf->ctx = ctx;  cf->cmd_type = NGX_HTTP_LOC_CONF;  rv = ngx_conf_parse(cf, NULL);  *cf = save;  return rv;} // }}}

與 server 塊解析函數 ngx_http_core_server 類似,他創建了所有模塊的 loc_conf,為了防止內外層具有相同指令,在配置賦值完成后,會通過 merge 函數合并到一起。

然而,與 server 塊不同,location 塊在 location 后面會通過路徑或正則表達式指定 location 配置的應用 uri,因此,在 ngx_http_core_location 函數中調用 PCRE 進行了 location 命令的解析。

解析完成后調用 ngx_http_add_location 將解析結果加入到 locations 鏈表中。

// ngx_int_t ngx_http_add_location(ngx_conf_t *cf, ngx_queue_t **locations,//   ngx_http_core_loc_conf_t *clcf)// 將 location 配置加入到 locations 配置鏈表中 {{{ngx_int_tngx_http_add_location(ngx_conf_t *cf, ngx_queue_t **locations,  ngx_http_core_loc_conf_t *clcf){  ngx_http_location_queue_t *lq;  if (*locations == NULL) {    *locations = ngx_palloc(cf->temp_pool,                sizeof(ngx_http_location_queue_t));    if (*locations == NULL) {      return NGX_ERROR;    }    ngx_queue_init(*locations);  }  lq = ngx_palloc(cf->temp_pool, sizeof(ngx_http_location_queue_t));  if (lq == NULL) {    return NGX_ERROR;  }  if (clcf->exact_match#if (NGX_PCRE)    || clcf->regex#endif    || clcf->named || clcf->noname)  {    lq->exact = clcf;    lq->inclusive = NULL;  } else {    lq->exact = NULL;    lq->inclusive = clcf;  }  lq->name = &clcf->name;  lq->file_name = cf->conf_file->file.name.data;  lq->line = cf->conf_file->line;  ngx_queue_init(&lq->list);  ngx_queue_insert_tail(*locations, &lq->queue);  return NGX_OK;} // }}}

配置解析全部完成后的配置結構。


發表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發表
亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb
国产欧美精品在线播放| 亚洲精品福利在线| 欧美高清视频免费观看| 精品成人av一区| 欧美午夜视频一区二区| 久久久久久有精品国产| 亚洲免费成人av电影| 国产精品高潮呻吟视频| 狠狠躁天天躁日日躁欧美| 欧美成人性色生活仑片| 日韩欧美在线免费| 视频在线观看一区二区| 国产视频在线一区二区| 久久999免费视频| 欧美日韩精品在线视频| 黑人精品xxx一区一二区| 成人免费视频在线观看超级碰| 成人免费网视频| 欧美成人午夜免费视在线看片| 国产精品极品美女在线观看免费| 国产亚洲福利一区| 日韩精品中文字幕在线播放| 在线观看欧美日韩国产| 国产精品草莓在线免费观看| 国产日本欧美一区| 久久亚洲精品视频| 美日韩精品免费观看视频| 亚洲欧洲xxxx| 日韩精品中文字幕在线| 欧美xxxx14xxxxx性爽| 国产精品jvid在线观看蜜臀| 欧美性xxxxxxxxx| 日韩电影中文 亚洲精品乱码| 亚洲免费福利视频| 亚洲第一区第一页| 国模吧一区二区| 亚洲精品一二区| 色婷婷久久一区二区| 国产精品视频999| 91在线网站视频| 国产精品99免视看9| 欧美日韩aaaa| 国产日本欧美视频| 欧美大片欧美激情性色a∨久久| 浅井舞香一区二区| 亚洲精品美女视频| 亚洲精品aⅴ中文字幕乱码| 国产日韩亚洲欧美| 久久国产精品网站| 国产91亚洲精品| 亚洲伊人第一页| 色噜噜久久综合伊人一本| 欧美最顶级丰满的aⅴ艳星| 亚洲精品电影网站| 亚洲久久久久久久久久久| 成人一区二区电影| 亚洲综合日韩在线| 欧美精品九九久久| 国产精自产拍久久久久久蜜| 亚洲性日韩精品一区二区| 日韩精品视频在线| 亚洲精品国精品久久99热| 国产精品久久综合av爱欲tv| 成人精品网站在线观看| 成人黄色中文字幕| 欧美极品美女视频网站在线观看免费| 久久国产精彩视频| 亚洲高清色综合| 国产极品jizzhd欧美| 亚洲综合一区二区不卡| 久久久久久久久久av| 亚洲视频在线视频| 精品无码久久久久久国产| 91国语精品自产拍在线观看性色| 亚洲欧美日韩一区二区三区在线| 尤物九九久久国产精品的分类| 亚洲精品福利免费在线观看| 欧美视频中文字幕在线| 亚洲视频axxx| 一区二区三区回区在观看免费视频| 91日韩在线播放| 国产精品九九久久久久久久| 国产日韩中文字幕在线| 97视频在线免费观看| 亚洲社区在线观看| yw.139尤物在线精品视频| 8050国产精品久久久久久| 在线性视频日韩欧美| 欧美精品videofree1080p| 午夜精品国产精品大乳美女| 爽爽爽爽爽爽爽成人免费观看| 中日韩美女免费视频网站在线观看| 668精品在线视频| 日韩亚洲精品视频| 欧美限制级电影在线观看| 欧美成人免费全部观看天天性色| 亚洲日本欧美中文幕| 欧美限制级电影在线观看| 亚洲美女动态图120秒| 久久免费观看视频| 欧美精品激情在线| 91精品国产高清久久久久久久久| 国内揄拍国内精品少妇国语| 亚洲精品免费在线视频| 日韩av片免费在线观看| 亚洲成人国产精品| 亚洲97在线观看| 成人精品网站在线观看| 精品久久久999| 国产一区二区三区欧美| 亚洲天堂av综合网| 久久视频免费在线播放| 欧美成人全部免费| 亚洲人午夜精品免费| 亚洲中国色老太| 欧美日韩视频免费播放| 91国内免费在线视频| 国产精品久久久久久久久免费看| 亚洲成人激情在线| 欧美性20hd另类| 日韩精品黄色网| 欧美日韩国产影院| 欧美大尺度激情区在线播放| www.欧美免费| 91免费高清视频| 久久久国产91| 在线免费看av不卡| 国产99久久久欧美黑人| 国产精品对白刺激| www.亚洲男人天堂| 亚洲视频日韩精品| 精品久久久91| 国模极品一区二区三区| 91sa在线看| 91sa在线看| 日韩精品在线免费观看| 久久91亚洲精品中文字幕| 青青久久aⅴ北条麻妃| 欧美专区在线观看| 亚洲国产女人aaa毛片在线| 欧美午夜激情视频| 久久久久日韩精品久久久男男| 性欧美办公室18xxxxhd| 亚洲国产精品一区二区三区| 78m国产成人精品视频| 中文国产成人精品| 欧美日韩在线看| 亚洲色图校园春色| 欧美午夜久久久| 91精品国产综合久久香蕉| 亚洲精品成a人在线观看| 国产成人亚洲综合91精品| 美女视频黄免费的亚洲男人天堂| 在线观看免费高清视频97| 欧美野外猛男的大粗鳮| 91手机视频在线观看| 中文字幕九色91在线| 中文字幕综合一区| 国产日韩精品在线观看| 欧洲中文字幕国产精品| 亚洲欧美激情精品一区二区| 久久夜精品va视频免费观看| 日韩精品免费看| 欧美大全免费观看电视剧大泉洋|