今天有一位同學(xué)問到 Nginx 的站點(diǎn)多路徑匹配的問題?
1.www.domain.com/a 需要返回 /var/www/domain.com/a/index.html
2.www.domain.com/b 需要返回 /var/www/domain.com/b/index.html
如何配置 Nginx 使之生效?
解決這個(gè)問題,第一的反映是直接使用 Nginx 的 location 指令來解決,不過在給出答案之前,我們先來了解一下 Nginx location 指令的基礎(chǔ)。
Nginx 區(qū)塊配置概念
在 Nginx 的配置文件中,通常會(huì)用兩個(gè)常用的區(qū)塊(Block)來進(jìn)行設(shè)置:
1.Server 區(qū)塊
2.Localtion 區(qū)塊
這里的區(qū)塊是指 Block,你甚至可以理解為后面的那一對(duì){}之間的配置內(nèi)容。
Sever 區(qū)塊主要是真的主機(jī)的配置,比如配置主機(jī)的域名,IP,端口等內(nèi)容。當(dāng)然,在一個(gè) Nginx 的配置文件里面,我們是可以指定多個(gè) Sever 區(qū)塊的配置的。
而 Location 區(qū)塊則是在 Sever 區(qū)塊里面,細(xì)分到針對(duì)不同的路徑和請(qǐng)求而進(jìn)行的配置。因?yàn)橐粋€(gè)站點(diǎn)中的 URI 通常會(huì)非常多,所以在 Location 區(qū)塊設(shè)置這部分,你也是可以寫多個(gè) Location 的配置的。
下面來看看 Location 配置的基本語法先:
location optional_modifier location_match {
# 這個(gè) {} 里面的配置內(nèi)容就是一個(gè)區(qū)塊 Block
}
上面的 optional_modifier 配置項(xiàng)是可以使用正則表達(dá)式的。常用的幾種如下:
- 留空。對(duì),留空也是一種設(shè)置方式。在留空的情況下,配置表示請(qǐng)求路徑由 location_match 開始。
- = ,等于號(hào)還是非常容易理解的:就是請(qǐng)求路徑正好等于后面的 location_match 的值;跟第一項(xiàng)留空還是有區(qū)別的。
- ~,飄號(hào)(注意是英文輸入的飄號(hào))表示大小寫敏感的正則匹配。
- ~*表示大小寫不敏感的正則匹配。
- ^~ 表示這里不希望有正則匹配發(fā)生。
Nginx 處理 Location 區(qū)塊的順序
上面了解了 location 指令基本的概念和常用配置。我們?cè)賮砜纯?Location 生效的順序!這個(gè)也很重要:
每一個(gè)請(qǐng)求進(jìn)來 Nginx 之后,Nginx 就會(huì)選擇一個(gè) Location 的最佳匹配項(xiàng)進(jìn)行響應(yīng),處理的具體流程是逐一跟 location 的配置進(jìn)行比對(duì),這個(gè)步驟可以分為以下幾步:
先進(jìn)行前綴式的匹配(也就是 location 的 optional_modifier 為空的配置)。
- Nginx 其次會(huì)根據(jù) URI 尋找完全匹配的 location 配置(也就是 location 的 optional_modifier 為 = 的配置).
- 如果還是沒有匹配到,那就先匹配 ^~ 配置,如果找到一個(gè)配置的話,則會(huì)停止尋找過程,直接返回響應(yīng)內(nèi)容。
- 如果還是沒有找到匹配項(xiàng)的話,則會(huì)先進(jìn)行大小寫敏感的正則匹配,然后再是大小不寫敏感的正則匹配。
Nginx Location 配置的一些例子:
多說無益,看了那么多理論,沒有具體的例子支撐也是白搭,所以我們來看一下具體的配置例子:
location = / {
# = 等號(hào)配置符,只匹配 / 這個(gè)路由
}
location /data {
# 留空配置,會(huì)匹配有 /data 開始的路由,后續(xù)有匹配會(huì)往下匹配。
}
location ^~ /img/ {
# 注意 ^~ 配置,這里匹配到 /img/ 開始的話,直接就返回了。
}
location ~* .(png|gif|ico|jpg|jpeg)$ {
# 匹配以 png, gif, ico, jpg or jpeg 結(jié)尾的請(qǐng)求;這個(gè)通常用來設(shè)置圖片的請(qǐng)求響應(yīng)。
}
非常實(shí)用的兩個(gè)例子:
1.簡單的圖片防盜鏈
location ~ .(png|gif|jpe?g)$ {
valid_referers none blocked yourwebsite.com *.yourwebsite.com;
# 注意上面寫上你的域名就好
if ($invalid_referer) {
return 403;
}
}
2.針對(duì)一些可寫入的路徑,禁止 php 或者 js 的腳步執(zhí)行
location ~* /(media|images|cache|tmp|logs)/.*.(php|jsp|pl|py|asp|cgi|sh)$ {
return 403;
}
問題的答案
最后,我們?cè)倏磫栴}的答案,可以是類似這個(gè)樣子的:
location /a {
root /var/www/domain.com/a;
}
location /b {
root /var/www/domain.com/b;
}