Nginx配置proxy_pass后返回404的問(wèn)題及解決
項(xiàng)目場(chǎng)景
需求
開(kāi)發(fā)一個(gè)平臺(tái)系統(tǒng),前端需要調(diào)用多個(gè)來(lái)自不同服務(wù)器的接口,為了滿足該需求,需要通過(guò)Nginx去轉(zhuǎn)發(fā)代理不同的接口地址,防止跨域,實(shí)現(xiàn)多接口的調(diào)用。
在一次生產(chǎn)涉及多次轉(zhuǎn)發(fā)的配置中, 需求是下面的圖:
問(wèn)題描述
問(wèn)題
在配置好了 proxy_pass 之后,請(qǐng)求 https://smartaitest.com/aitools 直接返回 404,沒(méi)有什么其他的異常。
但是我們直接請(qǐng)求后端 http://ai-ttxt.com/ 是正常響應(yīng)的。
看日志請(qǐng)求也是轉(zhuǎn)發(fā)到了 http://ai-ttxt.com/。但是轉(zhuǎn)發(fā)后的請(qǐng)求響應(yīng)就是404.
在配置nginx接口轉(zhuǎn)發(fā)代理的過(guò)程中發(fā)現(xiàn)兩種不同狀況
- 當(dāng)轉(zhuǎn)發(fā)的地址為 ip 時(shí)可正常訪問(wèn)。
- 當(dāng)轉(zhuǎn)發(fā)的地址為 域名 時(shí)報(bào)錯(cuò)為404。
問(wèn)題原因
我們的默認(rèn)的 Nginx的 proxy_set_header 配置是
proxy_set_header Host $host;
當(dāng)我們是這個(gè)的設(shè)置的時(shí)候,當(dāng)?shù)谝粚?Nginx(Nginx1)代理后,我們請(qǐng)求的域名是 www.djx.com(假設(shè)域名) ,從這個(gè)請(qǐng)求的 header 獲取到的 host 的值是 www.djx.com, 我們通過(guò) 配置將 host 的值設(shè)置為轉(zhuǎn)發(fā) 的Host 值,但是請(qǐng)求的域名 , 也就是 header 里面的是 host 字段 , 請(qǐng)求的域名和 header 里面的 Host 的不一致導(dǎo)致的。
總結(jié)一下 出現(xiàn)兩種情況的原因:
- 當(dāng)轉(zhuǎn)發(fā)的地址為 ip 時(shí)可正常訪問(wèn)。
- 當(dāng)轉(zhuǎn)發(fā)的地址為 域名 時(shí)報(bào)錯(cuò)為404。
1.當(dāng)使用 proxy_pass 將請(qǐng)求轉(zhuǎn)發(fā)到 IP 地址時(shí),Nginx 會(huì)將請(qǐng)求的 Host 頭信息保持不變地傳遞給后端服務(wù)器。
2.但是當(dāng) proxy_pass 設(shè)置為域名時(shí),默認(rèn)情況下,Nginx 會(huì)將請(qǐng)求的 Host 頭信息設(shè)置為當(dāng)前請(qǐng)求的域名。
這就意味著,如果你使用 proxy_pass 設(shè)置為域名時(shí),Nginx 會(huì)將請(qǐng)求的 Host頭信息設(shè)置為當(dāng)前請(qǐng)求的域名,而不是你指定的域名。這可能會(huì)導(dǎo)致后端服務(wù)器無(wú)法正確識(shí)別請(qǐng)求的來(lái)源,從而返回 404 錯(cuò)誤。
最終造成代理轉(zhuǎn)發(fā)訪問(wèn)404的問(wèn)題出現(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/; # 替換為實(shí)際的后端服務(wù)器地址 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/; # 替換為實(shí)際的后端服務(wù)器地址 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; } }
注意:轉(zhuǎn)發(fā)的接口為 域名沒(méi)有其他路徑時(shí),proxy_pass http://ai-ttxt.com/ 的接口地址后最好要將末尾的 /代理進(jìn)去
proxy_set_header Host $host; 和 proxy_set_header Host $proxy_host; 的區(qū)別
proxy_set_header Host $host; 和 proxy_set_header Host $proxy_host; 在 Nginx 配置中用于設(shè)置代理請(qǐng)求的 Host 頭信息:
- $host 變量: $host 變量表示客戶端發(fā)送請(qǐng)求時(shí)的 Host 頭信息,即請(qǐng)求的目標(biāo)域名。它是 Nginx 內(nèi)置的變量,表示當(dāng)前請(qǐng)求的主機(jī)名。
- $proxy_host 變量: $proxy_host 變量表示 Nginx 代理請(qǐng)求時(shí)使用的目標(biāo)服務(wù)器的主機(jī)名或 IP 地址。它是 Nginx 內(nèi)置的變量,表示當(dāng)前請(qǐng)求中被代理的服務(wù)器的主機(jī)名或 IP 地址。
因此,區(qū)別主要在于這兩個(gè)變量表示的含義:
- proxy_set_header Host $host; 將請(qǐng)求的 Host 頭信息設(shè)置為客戶端發(fā)送請(qǐng)求時(shí)的目標(biāo)域名。這通常用于將請(qǐng)求的 Host 頭信息傳遞給后端服務(wù)器,以確保后端服務(wù)器能夠正確處理請(qǐng)求。
- proxy_set_header Host $proxy_host; 將請(qǐng)求的 Host 頭信息設(shè)置為 Nginx 代理請(qǐng)求時(shí)使用的目標(biāo)服務(wù)器的主機(jī)名或 IP 地址。這通常用于在代理請(qǐng)求時(shí)設(shè)置一個(gè)自定義的 Host 頭信息,而不是直接使用客戶端發(fā)送的 Host 頭信息。
總結(jié)
以上為個(gè)人經(jīng)驗(yàn),希望能給大家一個(gè)參考,也希望大家多多支持腳本之家。
相關(guān)文章
nginx http響應(yīng)限速的具體實(shí)現(xiàn)
本文主要介紹了nginx http響應(yīng)限速的具體實(shí)現(xiàn),可以使用limite_rate和limit_rate_after來(lái)限制HTTP響應(yīng)的速度,具有一定的參考價(jià)值,感興趣的可以了解一下2024-05-05深入分析nginx+php-fpm服務(wù)HTTP狀態(tài)碼502
這篇文章主要介紹了深入分析nginx+php-fpm服務(wù)HTTP狀態(tài)碼502,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2019-07-07基于Nginx+lua實(shí)現(xiàn)簡(jiǎn)單的XSS攻擊攔截
WAF即web應(yīng)用防火墻,Nginx是一個(gè)主流的代理服務(wù),除了本身的Nginx日志,作為用戶肯定也支持對(duì)請(qǐng)求信息進(jìn)行操作,?很多都是通過(guò)在代理服務(wù)器上掛載規(guī)則特征,實(shí)現(xiàn)軟件層面的軟WAF進(jìn)行WEB防護(hù),本文主要給大家介紹了Nginx+Lua實(shí)現(xiàn)一個(gè)簡(jiǎn)單的XSS攻擊攔截,需要的朋友可以參考下2024-01-01nginx簡(jiǎn)單配置多個(gè)server的方法
這篇文章主要介紹了nginx簡(jiǎn)單配置多個(gè)server的方法,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2020-11-11CentOS 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-01Nginx根據(jù)不同瀏覽器語(yǔ)言配置頁(yè)面跳轉(zhuǎn)的方法
這篇文章主要介紹了Nginx根據(jù)不同瀏覽器語(yǔ)言配置頁(yè)面跳轉(zhuǎn)的方法,包括一個(gè)簡(jiǎn)體繁體的基本判斷方法及實(shí)際根據(jù)中英文跳轉(zhuǎn)的例子,需要的朋友可以參考下2016-04-04nginx服務(wù)器access日志中大量400 bad request錯(cuò)誤的解決方法
這篇文章主要介紹了nginx服務(wù)器access日志中大量400 bad request錯(cuò)誤的解決方法,本文結(jié)論是空主機(jī)頭導(dǎo)致的大量400錯(cuò)誤日志,關(guān)閉默認(rèn)主機(jī)的日志記錄就可以解決問(wèn)題,需要的朋友可以參考下2015-01-01詳解nginx 的 default_server 定義及匹配規(guī)則
這篇文章主要介紹了詳解nginx 的 default_server 定義及匹配規(guī)則,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2020-08-08