記錄一次 Nginx 配置 proxy_pass 後 返回404問題

1、 Nginx 配置 proxy_pass 後 返回404問題 故障解決和定位

1.一、 問題

在一次生產涉及屢次轉發的配置中, 需求是下面的圖: image
在配置好了 proxy_pass 以後,請求 www.djx.com 直接返回 404,沒有什麼其餘的異常。 可是咱們直接請求後端 www.baidu.com 是正常響應的。這就很怪異的。 看日誌請求也是轉發到了 www.baidu.com 的。可是請求響應就是404.html

1.二、 尋找問題緣由

咱們的默認的 Nginx的 proxy_set_header 配置是node

proxy_set_header Host $host;
  • 服務端: 192.168.2.189
  • 服務端1:192.168.1.180 Nginx1
  • 服務端2:192.168.1.90 Nginx2

image

當咱們是這個的設置的時候,當第一層 Nginx(Nginx1)代理後,咱們請求的域名是 www.djx.com ,從這個請求的 header 獲取到的 host 的值是 www.djx.com, 咱們經過 配置nginx

proxy_set_header Host $host;

將 host 的值設置爲轉發 的Host 值,可是請求的域名是 www.baidu.com , 也就是 header 裏面的是 host 字段是 www.djx.com , 請求的域名和 header 裏面的 Host 的不一致致使的。後端

image

1.三、 proxy_set_header 官方信息

官方文檔服務器

默認設置爲tcp

proxy_set_header Host $proxy_host;
proxy_set_header Connection close;

1.四、 解決辦法

Host 的值設置爲 $proxy_host, $proxy_host 的值詳解見下面擴展。url

proxy_set_header Host $proxy_host;

2、擴展 經常使用的配置

1. $proxy_host

proxy_set_header Host        $proxy_host; 
# 默認配置
# 顧名思義,請求頭設置的爲代理後的域名。

示例1:
當咱們配置了 upstream, 那麼$proxy_host 的值就是 upstream 的名字代理

upstream open-hz8443{
server 10.60.6.184:8000 max_fails=1 fail_timeout=3s weight=10;
}

那麼這裏 $proxy_host 的值就是 open-hz8443。日誌

示例2:
當咱們沒有配置 upstream, 那麼 $proxy_host 的值就是 proxy_pass 後面的地址ip和端口. 10.60.6.184:8000. 若是是 proxy_pass http://www.djx.com:8000; 那麼 $proxy_host 的值就是 www.djx.com:8000.code

location ^~ /wss/v1
{
    proxy_pass  http://10.60.6.184:8000;
    proxy_set_header   Host    $proxy_host;
    proxy_set_header Connection "upgrade";
    proxy_set_header Upgrade $http_upgrade;
    tcp_nodelay on; 
    
}

2. $host

proxy_set_header Host       $host;
# 當字段不在請求頭中就沒法傳遞了,在這種狀況下,可經過設置Host變量,將需傳遞值賦給Host變量

當請求 Header 裏 Host 無值的時候,直接拿 server_name 的值進行填充。

當請求 Header 裏 Host 的值的時候,就直接拿 請求 Header 裏面的 Host 的值。

3. $host:$proxy_port

proxy_set_header Host       $host:$proxy_port;

# 服務器名稱和端口一塊兒經過代理服務器傳遞,相對上一項,多了一個 $proxy_port,這個 $proxy_port 是proxy_pass 裏面的那個端口,若是沒有端口,像80 和 443 的話。也是會使用 80 /443 填充, 

示例: 
proxy_pass http://www.baidu.com;

$host:$proxy_port = 百度ip:80

4. $http_host

proxy_set_header Host       $http_host; 
# 一個不會變化的「Host」頭請求字段可經過以下方式被傳遞:

當請求 Header 裏 Host 無值的時候,直接拿 server_name 的值進行填充。並加上端口。若是是 80/443 則不加。 其實就是去 請求url 裏面的值。 http://server:port/v1

當請求 Header 裏 Host 的值的時候,就直接拿 請求 Header 裏面的 Host 的值。

示例

  • 服務端: 192.168.2.189
  • 服務端1:192.168.1.180 Nginx1
  • 服務端2:192.168.1.90 Nginx2

image

基礎配置
192.168.1.180 Nginx1

server{
    listen 80;
    server_name www.djx.com;
    
    location / {
        proxy_pass http://www.baidu.com/;
    }
    
}

192.168.1.190 Nginx2

server{
    listen 80;
    server_name www.baidu.com;
    
    location / {
        proxy_pass http://192.168.1.80:8080/;
    }
    
}

** 使用基礎配置 **
也就是

proxy_set_header Host        $proxy_host;

那麼到 http://192.168.1.80:8080/ header 的值爲 www.baidu.com.

** 使用 $host **
也就是

proxy_set_header Host        $host;

參考文章:https://cloud.tencent.com/developer/article/1557504

相關文章
相關標籤/搜索