亚洲激情专区-91九色丨porny丨老师-久久久久久久女国产乱让韩-国产精品午夜小视频观看

溫馨提示×

溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊×
其他方式登錄
點擊 登錄注冊 即表示同意《億速云用戶服務條款》

怎么在nginx中使用proxy_pass

發布時間:2021-03-26 17:42:57 來源:億速云 閱讀:218 作者:Leah 欄目:服務器

怎么在nginx中使用proxy_pass?相信很多沒有經驗的人對此束手無策,為此本文總結了問題出現的原因和解決方法,通過這篇文章希望你能解決這個問題。

一個線上的http服務,示例nginx關鍵配置如下:

server {
  listen 80;
  server_name ligang.gdemo.com;
  server_tokens off;

  keepalive_timeout 5;

  charset utf-8;

  include /home/ligang/devspace/gobox-demo/conf/http/general/gzip.conf;

  access_log logs/ligang.gdemo.com.log combinedio buffer=1k;
  error_log logs/ligang.gdemo.com.log.err;

  location / {
    include /home/ligang/devspace/gobox-demo/conf/http/general/http_proxy.conf;

    proxy_intercept_errors on;
    proxy_pass http://ligang.proxy.gdemo.com;
  }
}

這里可以看到,請求 ligang.gdemo.com 時,nginx把請求反向代理到 ligang.proxy.gdemo.com 去做處理。

ligang.proxy.gdemo.com 這個服務在線上部署并解析到了A、B、C這3個機房,現在我想調整解析,去掉C機房,僅留A、B兩個機房。

調整解析后,查看新的解析已經生效,但觀察C機房的請求量,發現和之前一樣,沒有任何變化。

于是我觀察C機房的nginx的log,發現請求來源還是 ligang.gdemo.com 的機器,域名解析調整后nginx那邊依舊使用之前的IP。

于是我將 ligang.gdemo.com 的機器上的nginx全部reload后,C機房的請求終于沒有了。

問題說明

上面的問題,說明在nginx的proxy_pass中如果使用了域名,那么nginx會把解析的結果緩存下來,貌似不會更新,因為上面的例子中,我調整解析后是幾乎是隔了一天去看C機房的log發現流量沒有任何變化的。

這樣的話,如果你配置一個反向代理服務器,如果上游調整了域名,而你又沒有得到通知,那么你的代理服務相當于不可用了。

從代碼中看下nginx是如何解析主機ip的

有點好奇nginx是如何解析主機ip的,所以追蹤下代碼:

proxy_pass指令定義的地方(http/modules/ngx_http_proxy_module.c):

{ ngx_string("proxy_pass"),
   NGX_HTTP_LOC_CONF|NGX_HTTP_LIF_CONF|NGX_HTTP_LMT_CONF|NGX_CONF_TAKE1,
   ngx_http_proxy_pass,    //處理方法
   NGX_HTTP_LOC_CONF_OFFSET,
   0,  
   NULL },

ngx_http_proxy_pass方法(http/modules/ngx_http_proxy_module.c):

static char *
ngx_http_proxy_pass(ngx_conf_t *cf, ngx_command_t *cmd, void *conf)
{
  ngx_http_proxy_loc_conf_t *plcf = conf;

  size_t           add;
  u_short           port;
  ngx_str_t         *value, *url;
  ngx_url_t          u;
  ngx_uint_t         n;
  ngx_http_core_loc_conf_t  *clcf;
  ngx_http_script_compile_t  sc;

 ......

  url = &value[1];

 ......

 ngx_memzero(&u, sizeof(ngx_url_t));

  u.url.len = url->len - add;
  u.url.data = url->data + add;
  u.default_port = port;
  u.uri_part = 1;
  u.no_resolve = 1;

 plcf->upstream.upstream = ngx_http_upstream_add(cf, &u, 0);
}

這里繼續追蹤ngx_http_upstream_add方法(http/ngx_http_upstream.c):

ngx_http_upstream_srv_conf_t *
ngx_http_upstream_add(ngx_conf_t *cf, ngx_url_t *u, ngx_uint_t flags)
{
  ngx_uint_t           i;
  ngx_http_upstream_server_t   *us;
  ngx_http_upstream_srv_conf_t  *uscf, **uscfp;
  ngx_http_upstream_main_conf_t *umcf;

  if (!(flags & NGX_HTTP_UPSTREAM_CREATE)) {

    if (ngx_parse_url(cf->pool, u) != NGX_OK) {
      if (u->err) {
        ngx_conf_log_error(NGX_LOG_EMERG, cf, 0,
                  "%s in upstream \"%V\"", u->err, &u->url);
      }

繼續追蹤ngx_parse_url方法(core/ngx_inet.c):

ngx_int_t
ngx_parse_url(ngx_pool_t *pool, ngx_url_t *u)
{
  u_char *p;           
  
  p = u->url.data;
  
  if (ngx_strncasecmp(p, (u_char *) "unix:", 5) == 0) {
    return ngx_parse_unix_domain_url(pool, u);
  }
    
  if (p[0] == '[') {
    return ngx_parse_inet6_url(pool, u);
  }               
      
  return ngx_parse_inet_url(pool, u);
}

然后是ngx_parse_inet_url方法(core/ngx_inet.c):

static ngx_int_t
ngx_parse_inet_url(ngx_pool_t *pool, ngx_url_t *u)
{
......

  if (ngx_inet_resolve_host(pool, u) != NGX_OK) {
    return NGX_ERROR;
  }

......
}

然后是ngx_inet_resolve_host方法(core/ngx_inet.c):

#if (NGX_HAVE_GETADDRINFO && NGX_HAVE_INET6)                    
  
ngx_int_t                                     
ngx_inet_resolve_host(ngx_pool_t *pool, ngx_url_t *u)               
{
......
  if (getaddrinfo((char *) host, NULL, &hints, &res) != 0) {
    u->err = "host not found";
    ngx_free(host);
    return NGX_ERROR;
  }
......
}

#else /* !NGX_HAVE_GETADDRINFO || !NGX_HAVE_INET6 */

ngx_int_t
ngx_inet_resolve_host(ngx_pool_t *pool, ngx_url_t *u)
{
......
    h = gethostbyname((char *) host);
......
}

思考下如何解決這個問題

最簡單的解決方法,我想到如下幾種:

執行 nginx reload

這種方法優缺點都很明顯:

優點:操作簡單。

缺點:屬于我們常說的后手,需要做好監控。

配置resolver

可以通過在nginx中配置resolver來動態更新解析,大致做法如下:

server {
    listen   80;
    server_name ligang.gdemo.com;

    resolver 8.8.8.8 valid=60s;
    resolver_timeout 3s;

    set $gproxy "ligang.proxy.gdemo.com";

    location / {
     proxy_pass http://$gproxy;
    }
  }

這個方法優缺點如下:

優點:解析地址每隔一段時間自動更新,無需人工做 nginx reload 。

缺點:需要指定DNS服務器地址,如果這個服務器掛了,或是地址變了,則需要修改nginx配置后reload。

看完上述內容,你們掌握怎么在nginx中使用proxy_pass的方法了嗎?如果還想學到更多技能或想了解更多相關內容,歡迎關注億速云行業資訊頻道,感謝各位的閱讀!

向AI問一下細節

免責聲明:本站發布的內容(圖片、視頻和文字)以原創、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。

AI

盐山县| 新泰市| 甘德县| 平遥县| 长春市| 京山县| 昭平县| 浠水县| 准格尔旗| 乌拉特前旗| 如东县| 荣成市| 永靖县| 华容县| 临海市| 宁明县| 萝北县| 花垣县| 桃园市| 朔州市| 柘荣县| 郴州市| 广州市| 罗江县| 岗巴县| 苗栗县| 镇宁| 金秀| 裕民县| 水城县| 鹤山市| 凤阳县| 平定县| 拜城县| 遵义县| 怀宁县| 高阳县| 宜宾县| 新建县| 宁陕县| 即墨市|