nginx重寫規(guī)則
nginx rewrite 正則表達(dá)式匹配
大小寫匹配
~ 為區(qū)分大小寫匹配
~* 為不區(qū)分大小寫匹配
!和!*分別為區(qū)分大小寫不匹配及不區(qū)分大小寫不匹配
文件及目錄匹配
-f和!-f用來判斷是否存在文件
-d和!-d用來判斷是否存在目錄
-e和!-e用來判斷是否存在文件或目錄
-x和!-x用來判斷文件是否可執(zhí)行
flag標(biāo)記
last 相當(dāng)于Apache里的[L]標(biāo)記,表示完成rewrite
break 終止匹配, 不再匹配后面的規(guī)則。
redirect 返回302臨時(shí)重定向 地址欄會(huì)顯示跳轉(zhuǎn)后的地址。
permanent 返回301永久重定向 地址欄會(huì)顯示跳轉(zhuǎn)后的地址。
logcation的幾個(gè)使用實(shí)例:
1)location / { }:匹配任何查詢,因?yàn)樗姓?qǐng)求都以 / 開頭。但是正則表達(dá)式規(guī)則將被優(yōu)先和查詢匹配。
2)location =/ {}:僅僅匹配/
3)location ~* .(gif|jpg|jpeg)$
{
rewrite .(gif|jpg)$ /logo.png;
}:location不區(qū)分大小寫,匹配任何以gif,jpg,jpeg結(jié)尾的文件。
幾個(gè)實(shí)例:
多目錄轉(zhuǎn)成參數(shù)
要求:abc.domian.com/sort/2 => abc.domian.com/index.php?act=sort&name=abc&id=2
規(guī)則配置:
if ($host ~* (.*).domain.com) {
set $sub_name $1;
rewrite ^/sort/(\d+)/?$ /index.php?act=sort&cid=$sub_name&id=$1 last;
}
目錄對(duì)換
要求:/123456/xxxx -> /xxxx?id=123456
規(guī)則配置:
rewrite ^/(\d+)/(.+)/ /$2?id=$1 last;
再來一個(gè)針對(duì)瀏覽器優(yōu)化的自動(dòng)rewrite,這里rewrite后的目錄可以是存在的;
例如設(shè)定nginx在用戶使用ie的使用重定向到/nginx-ie目錄
規(guī)則如下:
if ($http_user_agent ~ MSIE) {
rewrite ^(.*)$ /nginx-ie/$1 break;
}
目錄自動(dòng)加“/” ,這個(gè)功能一般瀏覽器自動(dòng)完成
if (-d $request_filename){
rewrite /(.*)([/])$ http://$host/$1$2/ permanent;
}
以下這些可能就跟廣義的rewrite重寫無關(guān)了
禁止htaccess
location ~/.ht {
deny all;
}
禁止多個(gè)目錄
location ~ ^/(cron|templates)/ {
deny all; break;
}
禁止以/data開頭的文件,可以禁止/data/下多級(jí)目錄下.log.txt等請(qǐng)求
location ~ ^/data {
deny all;
}
禁止單個(gè)文件
location ~ /data/sql/data.sql {
deny all;
}
給favicon.ico和robots.txt設(shè)置過期時(shí)間; 這里為favicon.ico為99天,robots.txt為7天并不記錄404錯(cuò)誤日志
location ~(favicon.ico) {
log_not_found off;
expires 99d;
break;
}
location ~(robots.txt) {
log_not_found off;
expires 7d;
break;
}
設(shè)定某個(gè)文件的瀏覽器緩存過期時(shí)間;這里為600秒,并不記錄訪問日志
location ^~ /html/scripts/loadhead_1.js {
access_log off;
expires 600;
break;
}
Nginx還可以自定義某一類型的文件的保質(zhì)期時(shí)間,具體寫法看下文的代碼:
location ~* .(js|css|jpg|jpeg|gif|png|swf)$ {
if (-f $request_filename) {
expires 1h;
break;
}
}
//上段代碼就將js|css|jpg|jpeg|gif|png|swf這類文件的保質(zhì)期設(shè)置為一小時(shí)。
防盜鏈的設(shè)置:
防盜鏈:如果你的網(wǎng)站是個(gè)下載網(wǎng)站,下載步驟應(yīng)該是先經(jīng)過你的主頁找到下載地址,才能下載,為了防止某些網(wǎng)友直接訪問下載地址完全不通過主頁下載,我們就可以使用防盜鏈的方式,具體代碼如下:
location ~* .(gif|jpg|swf)$ {
valid_referers none blocked start.igrow.cn sta.igrow.cn;
if ($invalid_referer) {
rewrite ^/ http://$host/logo.png;
}
}
文件反盜鏈并設(shè)置過期時(shí)間--<盜鏈多次請(qǐng)求也會(huì)打開你的站點(diǎn)的圖片啊,所以設(shè)置下緩存時(shí)間,不會(huì)每次盜鏈都請(qǐng)求并下載這張圖片>
location ~* ^.+.(jpg|jpeg|gif|png|swf|rar|zip|css|js)$ {
valid_referers none blocked *.jjonline.cn *.jjonline.com.cn *.lanwei.org *.jjonline.org localhost 42.121.107.189;
if ($invalid_referer) {
rewrite ^/ http://img.jjonline.cn/forbid.gif;
return 417;
break;
}
access_log off;
break;
}
說明:
這里的return 417 為自定義的http狀態(tài)碼,默認(rèn)為403,方便通過nginx的log文件找出正確的盜鏈的請(qǐng)求地址
“rewrite ^/ http://img.jjonline.cn/forbid.gif;”顯示一張防盜鏈圖片
“access_log off;”不記錄訪問日志,減輕壓力
“expires 3d”所有文件3天的瀏覽器緩存
只充許固定ip訪問網(wǎng)站,并加上密碼;這個(gè)對(duì)有權(quán)限認(rèn)證的應(yīng)用比較在行
location \ {
allow 22.27.164.25; #允許的ipd
deny all;
auth_basic “KEY”; #認(rèn)證的一些設(shè)置
auth_basic_user_file htpasswd;
}
說明:location的應(yīng)用也有各種變化,這里的寫法就針對(duì)了根目錄了。
文件和目錄不存在的時(shí)重定向
if (!-e $request_filename) {
proxy_pass http://127.0.0.1; #這里是跳轉(zhuǎn)到代理ip,這個(gè)代理ip上有一個(gè)監(jiān)聽的web服務(wù)器
rewrite ^/ http://www.jjonline.cn/none.html; #跳轉(zhuǎn)到這個(gè)網(wǎng)頁去
return 404; #直接返回404碼,然后會(huì)尋找root指定的404.html文件
}
域名跳轉(zhuǎn)
server {
listen 80;
server_name jump.jjonline.cn ;#需要跳轉(zhuǎn)的多級(jí)域名
index index.html index.htm index.php; #入口索引文件的名字
root /var/www/public_html/; #這個(gè)站點(diǎn)的根目錄
rewrite ^/ http://www.jjonline.cn/;
rewrite到這個(gè)地址,功能表現(xiàn):在瀏覽器上輸入jump.jjonline.cn并回車,不會(huì)有任何提示直接變成www.jjonline.cn
access_log off;
}
多域名轉(zhuǎn)向
server {
listen 80;
server_name www.jjonline.cn www.jjonline.org;
index index.html index.htm index.php;
root /var/www/public_html/;
if ($host ~ “jjonline.org”) {
rewrite ^(.*) http://www.jjonline.cn$1 permanent;
}
}
三級(jí)域名跳轉(zhuǎn)
if ($http_host ~* “^(.*).i.jjonline.cn$”) {
rewrite ^(.*) http://demo.jjonline.cn$1;
break;
}
域名鏡向
server {
listen 80;
server_name mirror.jjonline.cn;
index index.html index.htm index.php;
root /var/www/public_html;
rewrite ^/(.*) http://www.jjonline.cn/$1 last;
access_log off;
}
某個(gè)子目錄作鏡向,這里的示例是demo子目錄
location ^~ /demo {
rewrite ^.+ http://demo.jjonline.cn/ last;
break;
}
以下在附帶本博客的rewrite寫法,emlog系統(tǒng)的rewrite
location ~ {
if (!-e $request_filename) {
rewrite ^/(.+)$ /index.php last;
}
}
今天搞https接入的時(shí)候,nginx這邊進(jìn)行https的認(rèn)證加解密功能,所以后端的nginx和apache都是不需要進(jìn)行什么變化的,業(yè)務(wù) 也是如此,但是有一個(gè)業(yè)務(wù)稍微有點(diǎn)不同,其需要根據(jù)http接入和https接入的不同來做吐出不同的東東,由于nginx這一層接入來做的ssl相關(guān)的 東東,到了后端的apache這里已經(jīng)是http的了(這樣做的目的是為了省事和性能考慮)。這是cgi并不知道轉(zhuǎn)發(fā)過來的是http的請(qǐng)求還是 https的請(qǐng)求。
于是就需要在前端接入nginx這里做一些調(diào)整,使得到后端的時(shí)候能夠判斷是否是https的。
想到的一個(gè)簡(jiǎn)單的方法就是通過rewrite規(guī)則,將請(qǐng)求的參數(shù)后面增加一個(gè)特殊的參數(shù)用于表示這是https的請(qǐng)求。
于是配置轉(zhuǎn)發(fā)的規(guī)則,在location里面配置了怎么也不行,于是只能在server里面通過if判斷來進(jìn)行配置了。
if ( $request_uri ~* ^// ) {
rewrite ^(.+)$ $1?https=1 last;
}
問題得到了解決,到后端apache的請(qǐng)求都會(huì)帶上一個(gè)特殊的***https的請(qǐng)求參數(shù)啦,從而方便了后端的程序來進(jìn)行判斷。
呵呵,當(dāng)然也可以通過配置其它的虛擬主機(jī)的方式來實(shí)現(xiàn),不過這樣的話,就要拷貝一份程序的代碼,為https寫一份差不多相同的代碼,實(shí)在麻煩。
在使用nginx的rewrite的時(shí)候,開始加上了$query_string 發(fā)現(xiàn)到了后端的參數(shù)是double了的,查閱了一下發(fā)現(xiàn):
rewrite
語法: rewrite regex replacement flag
默認(rèn): none
作用域: server, location, if
This directive changes URI in accordance with the regular expression and the replacement string. Directives are carried out in order of appearance in the configuration file.
這個(gè)指令根據(jù)表達(dá)式來更改URI,或者修改字符串。指令根據(jù)配置文件中的順序來執(zhí)行。
Be aware that the rewrite regex only matches the relative path instead of the absolute URL. If you want to match the hostname, you should use an if condition, like so:
注意重寫表達(dá)式只對(duì)相對(duì)路徑有效。如果你想配對(duì)主機(jī)名,你應(yīng)該使用if語句。
呵呵,rewrite只是會(huì)改寫路徑部分的東東,不會(huì)改動(dòng)用戶的輸入?yún)?shù),因此這里的if規(guī)則里面,你無需關(guān)心用戶在瀏覽器里輸入的參數(shù),rewrite后會(huì)自動(dòng)添加的 ,因此,我們只是加上了一個(gè)?號(hào)和后面我們想要的一個(gè)小小的參數(shù)***https=1就可以了。
nginx的rewrite規(guī)則參考:
~ 為區(qū)分大小寫匹配
~* 為不區(qū)分大小寫匹配
!和!*分別為區(qū)分大小寫不匹配及不區(qū)分大小寫不匹
-f和!-f用來判斷是否存在文件
-d和!-d用來判斷是否存在目錄
-e和!-e用來判斷是否存在文件或目錄
-x和!-x用來判斷文件是否可執(zhí)行
last 相當(dāng)于Apache里的[L]標(biāo)記,表示完成rewrite,呵呵這應(yīng)該是最常用的
break 終止匹配, 不再匹配后面的規(guī)則
redirect 返回302臨時(shí)重定向 地址欄會(huì)顯示跳轉(zhuǎn)后的地址
permanent 返回301永久重定向 地址欄會(huì)顯示跳轉(zhuǎn)后的地址
$args
$content_length
$content_type
$document_root
$document_uri
$host
$http_user_agent
$http_cookie
$limit_rate
$request_body_file
$request_method
$remote_addr
$remote_port
$remote_user
$request_filename
$request_uri
$query_string
$scheme
$server_protocol
$server_addr
$server_name
$server_port
$uri
結(jié)合QeePHP的例子
if (!-d $request_filename) {
rewrite ^/([a-z-A-Z]+)/([a-z-A-Z]+)/?(.*)$ /index.php?namespace=user&controller=$1&action=$2&$3 last;
rewrite ^/([a-z-A-Z]+)/?$ /index.php?namespace=user&controller=$1 last;
break;
多目錄轉(zhuǎn)成參數(shù)
abc.domian.com/sort/2 => abc.domian.com/index.php?act=sort&name=abc&id=2
if ($host ~* (.*).domain.com) {
set $sub_name $1;
rewrite ^/sort/(\d+)/?$ /index.php?act=sort&cid=$sub_name&id=$1 last;
}
目錄對(duì)換
/123456/xxxx -> /xxxx?id=123456
rewrite ^/(\d+)/(.+)/ /$2?id=$1 last;
例如下面設(shè)定nginx在用戶使用ie的使用重定向到/nginx-ie目錄下:
if ($http_user_agent ~ MSIE) {
rewrite ^(.*)$ /nginx-ie/$1 break;
}
目錄自動(dòng)加“/”
if (-d $request_filename){
rewrite /(.*)([/])$ http://$host/$1$2/ permanent;
}
禁止htaccess
location ~/.ht {
deny all;
}
禁止多個(gè)目錄
location ~ ^/(cron|templates)/ {
deny all;
break;
}
禁止以/data開頭的文件
可以禁止/data/下多級(jí)目錄下.log.txt等請(qǐng)求;
location ~ ^/data {
deny all;
}
禁止單個(gè)目錄
不能禁止.log.txt能請(qǐng)求
location /searchword/cron/ {
deny all;
}
禁止單個(gè)文件
location ~ /data/sql/data.sql {
deny all;
}
給favicon.ico和robots.txt設(shè)置過期時(shí)間;
這里為favicon.ico為99天,robots.txt為7天并不記錄404錯(cuò)誤日志
location ~(favicon.ico) {
log_not_found off;
expires 99d;
break;
}
location ~(robots.txt) {
log_not_found off;
expires 7d;
break;
}
設(shè)定某個(gè)文件的過期時(shí)間;這里為600秒,并不記錄訪問日志
location ^~ /html/scripts/loadhead_1.js {
access_log off;
root /opt/lampp/htdocs/web;
expires 600;
break;
}
文件反盜鏈并設(shè)置過期時(shí)間
這里的return 412 為自定義的http狀態(tài)碼,默認(rèn)為403,方便找出正確的盜鏈的請(qǐng)求
“rewrite ^/ http://leech.c1gstudio.com/leech.gif;”顯示一張防盜鏈圖片
“access_log off;”不記錄訪問日志,減輕壓力
“expires 3d”所有文件3天的瀏覽器緩存
location ~* ^.+.(jpg|jpeg|gif|png|swf|rar|zip|css|js)$ {
valid_referers none blocked *.c1gstudio.com *.c1gstudio.net localhost 208.97.167.194;
if ($invalid_referer) {
rewrite ^/ http://leech.c1gstudio.com/leech.gif;
return 412;
break;
}
access_log off;
root /opt/lampp/htdocs/web;
expires 3d;
break;
}
只充許固定ip訪問網(wǎng)站,并加上密碼
root /opt/htdocs/www;
allow 208.97.167.194;
allow 222.33.1.2;
allow 231.152.49.4;
deny all;
auth_basic “C1G_ADMIN”;
auth_basic_user_file htpasswd;
將多級(jí)目錄下的文件轉(zhuǎn)成一個(gè)文件,增強(qiáng)seo效果
/job-123-456-789.html 指向/job/123/456/789.html
rewrite ^/job-([0-9]+)-([0-9]+)-([0-9]+).html$ /job/$1/$2/jobshow_$3.html last;
將根目錄下某個(gè)文件夾指向2級(jí)目錄
如/shanghai job/ 指向 /area/shanghai /
如果你將last改成permanent,那么瀏覽器地址欄顯是/location/shanghai/
rewrite ^/([0-9a-z]+)job/(.*)$ /area/$1/$2 last;
上面例子有個(gè)問題是訪問/shanghai 時(shí)將不會(huì)匹配
rewrite ^/([0-9a-z]+)job$ /area/$1/ last;
rewrite ^/([0-9a-z]+)job/(.*)$ /area/$1/$2 last;
這樣/shanghai 也可以訪問了,但頁面中的相對(duì)鏈接無法使用,
如./list_1.html真實(shí)地址是/area/shanghia/list_1.html會(huì)變成/list_1.html,導(dǎo)至無法訪問。
那我加上自動(dòng)跳轉(zhuǎn)也是不行咯
(-d $request_filename)它有個(gè)條件是必需為真實(shí)目錄,而我的rewrite不是的,所以沒有效果
if (-d $request_filename){
rewrite /(.*)([/])$ http://$host/$1$2/ permanent;
}
知道原因后就好辦了,讓我手動(dòng)跳轉(zhuǎn)吧
rewrite ^/([0-9a-z]+)job$ /$1job/ permanent;
rewrite ^/([0-9a-z]+)job/(.*)$ /area/$1/$2 last;
文件和目錄不存在的時(shí)候重定向:
if (!-e $request_filename) {
proxy_pass http://127.0.0.1;
}
域名跳轉(zhuǎn)
server
{
listen 80;
server_name jump.c1gstudio.com;
index index.html index.htm index.php;
root /opt/lampp/htdocs/www;
rewrite ^/ http://www.c1gstudio.com/;
access_log off;
}
多域名轉(zhuǎn)向
server_name www.c1gstudio.com www.c1gstudio.net;
index index.html index.htm index.php;
root /opt/lampp/htdocs;
if ($host ~ “c1gstudio.net”) {
rewrite ^(.*) http://www.c1gstudio.com$1 permanent;
}
三級(jí)域名跳轉(zhuǎn)
if ($http_host ~* “^(.).i.c1gstudio.com$”) {
rewrite ^(.) http://top.yingjiesheng.com$1;
break;
}
域名鏡向
server
{
listen 80;
server_name mirror.c1gstudio.com;
index index.html index.htm index.php;
root /opt/lampp/htdocs/www;
rewrite ^/(.*) http://www.c1gstudio.com/$1 last;
access_log off;
}
某個(gè)子目錄作鏡向
location ^~ /zhaopinhui {
rewrite ^.+ http://zph.c1gstudio.com/ last;
break;
}
discuz ucenter home (uchome) rewrite
rewrite ^/(space|network)-(.+).html$ /$1.php?rewrite=$2 last;
rewrite ^/(space|network).html$ /$1.php last;
rewrite ^/([0-9]+)$ /space.php?uid=$1 last;
discuz 7 rewrite
rewrite ^(.)/archiver/((fid|tid)-[\w-]+.html)$ $1/archiver/index.php?$2 last;
rewrite ^(.)/forum-([0-9]+)-([0-9]+).html$ $1/forumdisplay.php?fid=$2&page=$3 last;
rewrite ^(.)/thread-([0-9]+)-([0-9]+)-([0-9]+).html$ $1/viewthread.php?tid=$2&extra=page%3D$4&page=$3 last;
rewrite ^(.)/profile-(username|uid)-(.+).html$ $1/viewpro.php?$2=$3 last;
rewrite ^(.)/space-(username|uid)-(.+).html$ $1/space.php?$2=$3 last;
rewrite ^(.)/tag-(.+).html$ $1/tag.php?name=$2 last;
給discuz某版塊單獨(dú)配置域名
server_name bbs.c1gstudio.com news.c1gstudio.com;
location = / {
if ($http_host ~ news.c1gstudio.com$) {
rewrite ^.+ http://news.c1gstudio.com/forum-831-1.html last;
break;
}
}
discuz ucenter 頭像 rewrite 優(yōu)化
location ^~ /ucenter {
location ~ .*.php?$
{
fastcgi_pass unix:/tmp/php-cgi.sock;
fastcgi_pass 127.0.0.1:9000;
fastcgi_index index.php;
include fcgi.conf;
}
location /ucenter/data/avatar {
log_not_found off;
access_log off;
location ~ /(.)_big.jpg$ {
error_page 404 /ucenter/images/noavatar_big.gif;
}
location ~ /(.)_middle.jpg$ {
error_page 404 /ucenter/images/noavatar_middle.gif;
}
location ~ /(.*)_small.jpg$ {
error_page 404 /ucenter/images/noavatar_small.gif;
}
expires 300;
break;
}
}
jspace rewrite
location ~ .*.php?$
{
fastcgi_pass unix:/tmp/php-cgi.sock;
fastcgi_pass 127.0.0.1:9000;
fastcgi_index index.php;
include fcgi.conf;
}
location ~* ^/index.php/
{
rewrite ^/index.php/(.*) /index.php?$1 break;
fastcgi_pass 127.0.0.1:9000;
fastcgi_index index.php;
include fcgi.conf;
}