Nginx配置proxy_pass后返回404的問題及解決
項目場景
需求
開發(fā)一個平臺系統(tǒng),前端需要調用多個來自不同服務器的接口,為了滿足該需求,需要通過Nginx去轉發(fā)代理不同的接口地址,防止跨域,實現(xiàn)多接口的調用。
在一次生產涉及多次轉發(fā)的配置中, 需求是下面的圖:

問題描述
問題
在配置好了 proxy_pass 之后,請求 https://smartaitest.com/aitools 直接返回 404,沒有什么其他的異常。
但是我們直接請求后端 http://ai-ttxt.com/ 是正常響應的。
看日志請求也是轉發(fā)到了 http://ai-ttxt.com/。但是轉發(fā)后的請求響應就是404.

在配置nginx接口轉發(fā)代理的過程中發(fā)現(xiàn)兩種不同狀況
- 當轉發(fā)的地址為 ip 時可正常訪問。
- 當轉發(fā)的地址為 域名 時報錯為404。
問題原因
我們的默認的 Nginx的 proxy_set_header 配置是
proxy_set_header Host $host;
當我們是這個的設置的時候,當?shù)谝粚?Nginx(Nginx1)代理后,我們請求的域名是 www.djx.com(假設域名) ,從這個請求的 header 獲取到的 host 的值是 www.djx.com, 我們通過 配置將 host 的值設置為轉發(fā) 的Host 值,但是請求的域名 , 也就是 header 里面的是 host 字段 , 請求的域名和 header 里面的 Host 的不一致導致的。

總結一下 出現(xiàn)兩種情況的原因:
- 當轉發(fā)的地址為 ip 時可正常訪問。
- 當轉發(fā)的地址為 域名 時報錯為404。
1.當使用 proxy_pass 將請求轉發(fā)到 IP 地址時,Nginx 會將請求的 Host 頭信息保持不變地傳遞給后端服務器。
2.但是當 proxy_pass 設置為域名時,默認情況下,Nginx 會將請求的 Host 頭信息設置為當前請求的域名。
這就意味著,如果你使用 proxy_pass 設置為域名時,Nginx 會將請求的 Host頭信息設置為當前請求的域名,而不是你指定的域名。這可能會導致后端服務器無法正確識別請求的來源,從而返回 404 錯誤。
最終造成代理轉發(fā)訪問404的問題出現(xiàn)
解決方案
修改Nginx的 proxy_set_header 配置
proxy_set_header Host $proxy_host;
修改前的nginx.conf 配置
server {
listen 99;
server_name _;
client_max_body_size 100m;
access_log /var/log/nginx/access-front.log main;
error_log /var/log/nginx/error-front.log notice;
location /aitools {
alias /opt/html/dist;
index index.html index.htm;
try_files $uri $uri/ /aitools/index.html; #aitools為路由 而不是目錄
}
location /aitools/vapi/ {
proxy_pass http://ai-ttxt.com/; # 替換為實際的后端服務器地址
proxy_set_header Host $host;
proxy_set_header X-Real-IP $remote_addr;
proxy_set_header X-Forwarded-For $proxy_add_x_forwarded_for;
proxy_set_header X-Forwarded-Proto $scheme;
}
}
修改后的配置
server {
listen 99;
server_name _;
client_max_body_size 100m;
access_log /var/log/nginx/access-front.log main;
error_log /var/log/nginx/error-front.log notice;
location /aitools {
alias /opt/html/dist;
index index.html index.htm;
try_files $uri $uri/ /aitools/index.html; #aitools為路由 而不是目錄
}
location /aitools/vapi/ {
proxy_pass http://ai-ttxt.com/; # 替換為實際的后端服務器地址
proxy_set_header Host $proxy_host; #(需要修改的地方)
proxy_set_header X-Real-IP $remote_addr;
proxy_set_header X-Forwarded-For $proxy_add_x_forwarded_for;
proxy_set_header X-Forwarded-Proto $scheme;
}
}
注意:轉發(fā)的接口為 域名沒有其他路徑時,proxy_pass http://ai-ttxt.com/ 的接口地址后最好要將末尾的 /代理進去
proxy_set_header Host $host; 和 proxy_set_header Host $proxy_host; 的區(qū)別
proxy_set_header Host $host; 和 proxy_set_header Host $proxy_host; 在 Nginx 配置中用于設置代理請求的 Host 頭信息:
- $host 變量: $host 變量表示客戶端發(fā)送請求時的 Host 頭信息,即請求的目標域名。它是 Nginx 內置的變量,表示當前請求的主機名。
- $proxy_host 變量: $proxy_host 變量表示 Nginx 代理請求時使用的目標服務器的主機名或 IP 地址。它是 Nginx 內置的變量,表示當前請求中被代理的服務器的主機名或 IP 地址。
因此,區(qū)別主要在于這兩個變量表示的含義:
- proxy_set_header Host $host; 將請求的 Host 頭信息設置為客戶端發(fā)送請求時的目標域名。這通常用于將請求的 Host 頭信息傳遞給后端服務器,以確保后端服務器能夠正確處理請求。
- proxy_set_header Host $proxy_host; 將請求的 Host 頭信息設置為 Nginx 代理請求時使用的目標服務器的主機名或 IP 地址。這通常用于在代理請求時設置一個自定義的 Host 頭信息,而不是直接使用客戶端發(fā)送的 Host 頭信息。
總結
以上為個人經驗,希望能給大家一個參考,也希望大家多多支持腳本之家。
相關文章
深入分析nginx+php-fpm服務HTTP狀態(tài)碼502
這篇文章主要介紹了深入分析nginx+php-fpm服務HTTP狀態(tài)碼502,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友們下面隨著小編來一起學習學習吧2019-07-07
CentOS 7.3.1611編譯安裝Nginx1.10.3+MySQL5.7.16+PHP7.1.2
這篇文章主要介紹了CentOS 7.3.1611編譯安裝Nginx1.10.3+MySQL5.7.16+PHP7.1.2,需要的朋友可以參考下2018-01-01
nginx服務器access日志中大量400 bad request錯誤的解決方法
這篇文章主要介紹了nginx服務器access日志中大量400 bad request錯誤的解決方法,本文結論是空主機頭導致的大量400錯誤日志,關閉默認主機的日志記錄就可以解決問題,需要的朋友可以參考下2015-01-01
詳解nginx 的 default_server 定義及匹配規(guī)則
這篇文章主要介紹了詳解nginx 的 default_server 定義及匹配規(guī)則,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友們下面隨著小編來一起學習學習吧2020-08-08

