Nginx如何根據(jù)前綴路徑轉(zhuǎn)發(fā)到不同的Flask服務(wù)
開(kāi)端
想通過(guò)不同的前綴路徑經(jīng)過(guò)nginx轉(zhuǎn)發(fā)到不同的服務(wù)上,比如 /user/
轉(zhuǎn)發(fā)到用戶(hù)服務(wù),/other/
轉(zhuǎn)發(fā)到其他服務(wù)。
首先配置nginx的location根據(jù)前綴匹配。
server { listen 80; server_name localhost; location /user/ { proxy_pass http://127.0.0.1:5000; # 用戶(hù)服務(wù) } location /other/ { proxy_pass http://127.0.0.1:5001; # 其他服務(wù) } }
這樣訪(fǎng)問(wèn)http://127.0.0.1:5000/user/xxx/
就會(huì)轉(zhuǎn)發(fā)到用戶(hù)服務(wù),訪(fǎng)問(wèn)http://127.0.0.1:5000/other/xxx/
就會(huì)轉(zhuǎn)發(fā)到其他服務(wù)。
這樣就會(huì)有一個(gè)問(wèn)題,flask application并不知道有前綴,所以使用url_for構(gòu)造url的時(shí)候并不會(huì)自己添加前綴,要構(gòu)造出正確帶前綴的url就需要把前綴加入到WSGI環(huán)境中的SCRIPT_NAME去。
解決一(設(shè)置SCRIPT_NAME)
gunicorn文檔上:
可以把SCRIPT_NAME設(shè)置到環(huán)境變量中或者HTTP header中。
通過(guò)docker部署設(shè)置SCRIPT_NAME在環(huán)境變量中,可以在docker-compose.yml中加入
environment: - SCRIPT_NAME=/user/
或者把SCRIPT_NAME設(shè)置在header中可以在nginx配置中加上
proxy_set_header SCRIPT_NAME /user/;
gunicorn.wsgi處理請(qǐng)求的時(shí)候是這樣處理PATH_INFO和SCRIPT_NAME的:
解決二(設(shè)置頭部X-Forwarded-Prefix再用ProxyFix調(diào)整WSGI環(huán)境)
同樣的也可以ProxyFix中間件來(lái)調(diào)整WSGI環(huán)境,設(shè)置SCRIPT_NAME。
來(lái)自werkzeug ProxyFix文檔:
通過(guò)nginx設(shè)置頭部信息X-Forwarded-Prefix:
proxy_set_header X-Forwarded-Prefix /user/;
使用ProxyFix:
from werkzeug.middleware.proxy_fix import ProxyFix app = ProxyFix(app, x_prefix=1)
還需要把nginx的proxy_pass修改下:
server { listen 80; server_name example.com; location /user/ { proxy_pass http://127.0.0.1:5000/; # 用戶(hù)服務(wù) } location /other/ { proxy_pass http://127.0.0.1:5001/; # 其他服務(wù) } }
兩種解決的區(qū)別
nginx proxy_pass配置的區(qū)別
區(qū)別在于nginx的proxy_pass中結(jié)尾是否帶/。
如果proxy_pass不帶uri,就是不帶/,則請(qǐng)求會(huì)原封不動(dòng)的轉(zhuǎn)發(fā)給下一個(gè)服務(wù)。
如果proxy_pass帶uri,則匹配的uri部分將會(huì)被修改為該proxy_pass中的uri。
為什么需要這樣處理
以我的理解是這樣的,請(qǐng)求進(jìn)來(lái)通過(guò)gunicorn處理請(qǐng)求,gunicorn.wsgi中會(huì)根據(jù)SCRIPT_NAME來(lái)制定PATH_INFO,所以當(dāng)解決一帶著SCRIPT_NAME=/user/
,PATH_INFO=/user/xxx/
經(jīng)過(guò)處理后PATH_INFO會(huì)變成/xxx/
。
而解決二中當(dāng)gunicorn.wsgi處理請(qǐng)求時(shí)ProxyFix還沒(méi)對(duì)WSGI環(huán)境進(jìn)行處理,所以SCRIPT_NAME是為空的,PATH_INFO則會(huì)一直是帶著SCRIPT_NAME前綴為/user/xxx/
,是不能正確匹配到route的,所以把nginx proxy_pass改為uri形式使PATH_INFO能正確匹配。
總結(jié)
以上為個(gè)人經(jīng)驗(yàn),希望能給大家一個(gè)參考,也希望大家多多支持腳本之家。
相關(guān)文章
Nginx設(shè)置成服務(wù)并開(kāi)機(jī)自動(dòng)啟動(dòng)的配置
Nginx?是一個(gè)高性能的HTTP和反向代理web服務(wù)器,同時(shí)也提供了IMAP/POP3/SMTP服務(wù),接下來(lái)通過(guò)本文給大家介紹Nginx設(shè)置成服務(wù)并開(kāi)機(jī)自動(dòng)啟動(dòng)的配置,需要的朋友可以參考下2022-01-01Nginx訪(fǎng)問(wèn)靜態(tài)資源配置的實(shí)現(xiàn)步驟
Nginx 擅長(zhǎng)于底層服務(wù)器端資源的處理,例如靜態(tài)資源處理轉(zhuǎn)發(fā)、反向代理,負(fù)載均衡等,本文主要介紹了Nginx訪(fǎng)問(wèn)靜態(tài)資源配置的實(shí)現(xiàn)步驟,具有一定的參考價(jià)值,感興趣的可以了解一下2023-09-09負(fù)載均衡下的webshell上傳+nginx解析漏洞的過(guò)程
這篇文章主要介紹了負(fù)載均衡下的webshell上傳+nginx解析漏洞,首先介紹了負(fù)載均衡下webshell上傳的四大難點(diǎn)及環(huán)境搭建教程,感興趣的朋友跟隨小編一起看看吧2024-02-02Nginx+Tomcat搭建高性能負(fù)載均衡集群的實(shí)現(xiàn)方法
這篇文章主要介紹了Nginx+Tomcat搭建高性能負(fù)載均衡集群的實(shí)現(xiàn)方法,小編覺(jué)得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧2018-03-03針對(duì)OpenSSL安全漏洞調(diào)整Nginx服務(wù)器的方法
這篇文章主要介紹了針對(duì)OpenSSL漏洞調(diào)整Nginx服務(wù)器的方法,2014年爆出的SSL安全漏洞震驚了全世界,需要的朋友可以參考下2015-06-06簡(jiǎn)介使用Nginx Plus的在線(xiàn)活動(dòng)監(jiān)控功能的方法
這篇文章主要介紹了簡(jiǎn)介使用Nginx Plus的在線(xiàn)活動(dòng)監(jiān)控功能的方法,注意其目前暫時(shí)為收費(fèi)項(xiàng)目,需要的朋友可以參考下2015-06-06云服務(wù)器使用寶塔搭建Python環(huán)境,運(yùn)行django程序
本文詳細(xì)講解了在云服務(wù)器使用寶塔搭建Python環(huán)境,運(yùn)行django程序的方法。對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2021-12-12