工作中經(jīng)常會使用Nginx 配置反向代理,代理末尾的斜杠配置組合大體上可以分為8種,所以總是忘記 location 和 proxy_pass 最后面的斜杠會對代理結(jié)果有怎樣的影響。今天我們就來試試找個統(tǒng)一的規(guī)則。
server {
listen 8000;
server_name localhost;
#配置反向代理
location /test/ {
proxy_pass http://127.0.0.1:9000/api/;
}
}
我們簡單配置個反向代理的 conf,監(jiān)聽端口為8000,代理端口為 9000,下面分別羅列出 location 和proxy_pass 斜杠的8類組合,以及代理的結(jié)果。
第一種斜杠配置
#配置1
location /test/ {
proxy_pass http://127.0.0.1:9000/api/;
}
模擬請求
curl http://localhost:8000/test/user/query
由于我沒有開啟端口為9000的服務(wù),nginx會返回502,所以來看下nginx的 error.log日志,截取一部分如下:
request: "GET /test/user/query HTTP/1.1", upstream: "http://127.0.0.1:9000/api/user/query"
其中,upstream 后面就是代理的結(jié)果,對比模擬請求,可以發(fā)現(xiàn)除了 ip:port 被替換,還少了個 "test",我們大膽推測一下,nginx 監(jiān)聽到模擬請求后,判斷其和 "/test/" 匹配,截取了 "
http://localhost:8000/test/" ,剩余的 "user/query" 拼接到 proxy_pass 的內(nèi)容后面,就得到了 "
http://127.0.0.1:9000/api/user/query"
第二種斜杠配置
#配置2
location /test {
proxy_pass http://127.0.0.1:9000/api;
}
模擬請求
curl http://localhost:8000/test/user/query
error.log日志截取如下:
request: "GET /test/user/query HTTP/1.1", upstream: "http://127.0.0.1:9000/api/user/query"
這里截取了 "
http://localhost:8000/test" ,剩余 "/user/query" 拼接到 proxy_pass 的內(nèi)容后面,得到 "http://127.0.0.1:9000/api/user/query",符合推測。
第三種斜杠配置
#配置3
location /test {
proxy_pass http://127.0.0.1:9000/api/;
}
模擬請求
curl http://localhost:8000/test/user/query
error.log日志截取如下:
request: "GET /test/user/query HTTP/1.1", upstream: "http://127.0.0.1:9000/api//user/query"
這里截取了 "
http://localhost:8000/test" ,剩余 "/user/query" 拼接到 proxy_pass 的內(nèi)容后面,得到 "
http://127.0.0.1:9000/api//user/query",同樣符合推測。
第四種斜杠配置
#配置4
location /test/ {
proxy_pass http://127.0.0.1:9000/api;
}
模擬請求
curl http://localhost:8000/test/user/query
error.log日志截取如下:
request: "GET /test/user/query HTTP/1.1", upstream: "http://127.0.0.1:9000/apiuser/query"
這里截取了 "
http://localhost:8000/test/" ,剩余 "user/query" 拼接到 proxy_pass 的內(nèi)容后面,得到 "http://127.0.0.1:9000/apiuser/query",同樣符合推測。
第五種斜杠配置
#配置5
location /test/ {
proxy_pass http://127.0.0.1:9000/;
}
模擬請求
curl http://localhost:8000/test/user/query
error.log日志截取如下:
request: "GET /test/user/query HTTP/1.1", upstream: "http://127.0.0.1:9000/user/query"
這里截取了 "
http://localhost:8000/test/" ,剩余 "user/query" 拼接到 proxy_pass 的內(nèi)容后面,得到 "http://127.0.0.1:9000/user/query",同樣符合推測。
第六種斜杠配置
#配置6
location /test {
proxy_pass http://127.0.0.1:9000/;
}
模擬請求
curl http://localhost:8000/test/user/query
error.log日志截取如下:
request: "GET /test/user/query HTTP/1.1", upstream: "http://127.0.0.1:9000//user/query"
這里截取了 "
http://localhost:8000/test" ,剩余 "/user/query" 拼接到 proxy_pass 的內(nèi)容后面,得到 "http://127.0.0.1:9000//user/query",符合推測。
第七種斜杠配置
#配置7
location /test {
proxy_pass http://127.0.0.1:9000;
}
模擬請求
curl http://localhost:8000/test/user/query
error.log日志截取如下:
request: "GET /test/user/query HTTP/1.1", upstream: "http://127.0.0.1:9000/test/user/query"
這里截取 "
http://localhost:8000/test" ,剩余 "/user/query" 拼接到 proxy_pass 的內(nèi)容后面,得到 "http://127.0.0.1:9000/user/query",很遺憾不符合推測,但是比實際結(jié)果剛好少了 location 后面的內(nèi)容 "/test"。
第八種斜杠配置
#配置8
Location /test/ {
proxy_pass http://127.0.0.1:9000;
}
模擬請求
curl http://localhost:8000/test/user/query
error.log日志截取如下:
request: "GET /test/user/query HTTP/1.1", upstream: "http://127.0.0.1:9000/test/user/query"
這里截取 "
http://localhost:8000/test/" ,剩余 "user/query" 拼接到 proxy_pass 的內(nèi)容后面,得到 "http://127.0.0.1:9000user/query",同樣的也是比實際結(jié)果剛好少了location后面的內(nèi)容 "/test/"。
結(jié)果分析
經(jīng)過分析所有情況,我們可以得到以下結(jié)論:
當(dāng)proxy_pass 后面的內(nèi)容不是以端口號結(jié)尾,即端口號后面有"/**"時,其實就是將請求的地址,截取掉 location 后面的內(nèi)容,將剩余內(nèi)容拼接到proxy_pass代理地址上;
當(dāng)proxy_pass 后面的內(nèi)容以端口號結(jié)尾,即端口號后面什么都沒有,可以視為默認(rèn)拼接了 location 的內(nèi)容。那么配置7,8 就等同如下配置,再套用上面的結(jié)論就可以等到正確的結(jié)果。
#配置7
location /test {
proxy_pass http://127.0.0.1:9000/test;
}
#配置8
Location /test/ {
proxy_pass http://127.0.0.1:9000/test/;
}
不知道 nginx 的源碼關(guān)于這里的邏輯是怎樣處理的,但是有上面的結(jié)論以后再配置反向代理應(yīng)該不會因為末尾的斜杠疑惑了。