nginx配置url重寫

url重寫是指經過配置conf文件,以讓網站的url中達到某種狀態時則定向/跳轉到某個規則,好比常見的僞靜態、301重定向、瀏覽器定向等php

rewrite

語法

在配置文件的server塊中寫,如:html

server {
    rewrite 規則 定向路徑 重寫類型;
}
  • 規則:能夠是字符串或者正則來表示想匹配的目標url
  • 定向路徑:表示匹配到規則後要定向的路徑,若是規則裏有正則,則可使用$index來表示正則裏的捕獲分組
  • 重寫類型:
    • last :至關於Apache裏德(L)標記,表示完成rewrite,瀏覽器地址欄URL地址不變
    • break;本條規則匹配完成後,終止匹配,再也不匹配後面的規則,瀏覽器地址欄URL地址不變
    • redirect:返回302臨時重定向,瀏覽器地址會顯示跳轉後的URL地址
    • permanent:返回301永久重定向,瀏覽器地址欄會顯示跳轉後的URL地址

簡單例子

server {
    # 訪問 /last.html 的時候,頁面內容重寫到 /index.html 中
    rewrite /last.html /index.html last;

    # 訪問 /break.html 的時候,頁面內容重寫到 /index.html 中,並中止後續的匹配
    rewrite /break.html /index.html break;

    # 訪問 /redirect.html 的時候,頁面直接302定向到 /index.html中
    rewrite /redirect.html /index.html redirect;

    # 訪問 /permanent.html 的時候,頁面直接301定向到 /index.html中
    rewrite /permanent.html /index.html permanent;

    # 把 /html/*.html => /post/*.html ,301定向
    rewrite ^/html/(.+?).html$ /post/$1.html permanent;

    # 把 /search/key => /search.html?keyword=key
    rewrite ^/search\/([^\/]+?)(\/|$) /search.html?keyword=$1 permanent;
}

last和break的區別

由於301和302不能簡單的只返回狀態碼,還必須有重定向的URL,這就是return指令沒法返回301,302的緣由了。這裏 last 和 break 區別有點難以理解:linux

  • last通常寫在server和if中,而break通常使用在location中
  • last不終止重寫後的url匹配,即新的url會再從server走一遍匹配流程,而break終止重寫後的匹配
  • break和last都能組織繼續執行後面的rewrite指令

location裏一旦返回break則直接生效並中止後續的匹配locationnginx

server {
    location / {
        rewrite /last/ /q.html last;
        rewrite /break/ /q.html break;
    }

    location = /q.html {
        return 400;
    }
}
  • 訪問/last/時重寫到/q.html,而後使用新的uri再匹配,正好匹配到locatoin = /q.html而後返回了400
  • 訪問/break時重寫到/q.html,因爲返回了break,則直接中止了

if判斷

只是上面的簡單重寫不少時候知足不了需求,好比須要判斷當文件不存在時、當路徑包含xx時等條件,則須要用到if正則表達式

語法

if (表達式) {
}
  • 當表達式只是一個變量時,若是值爲空或任何以0開頭的字符串都會當作false
  • 直接比較變量和內容時,使用=或!=
  • ~正則表達式匹配,~*不區分大小寫的匹配,!~區分大小寫的不匹配

一些內置的條件判斷:瀏覽器

  • -f和!-f用來判斷是否存在文件
  • -d和!-d用來判斷是否存在目錄
  • -e和!-e用來判斷是否存在文件或目錄
  • -x和!-x用來判斷文件是否可執行

內置的全局變量

$args :這個變量等於請求行中的參數,同$query_string $content_length : 請求頭中的Content-length字段。 $content_type : 請求頭中的Content-Type字段。 $document_root : 當前請求在root指令中指定的值。 $host : 請求主機頭字段,不然爲服務器名稱。 $http_user_agent : 客戶端agent信息 $http_cookie : 客戶端cookie信息 $limit_rate : 這個變量能夠限制鏈接速率。 $request_method : 客戶端請求的動做,一般爲GET或POST。 $remote_addr : 客戶端的IP地址。 $remote_port : 客戶端的端口。 $remote_user : 已經通過Auth Basic Module驗證的用戶名。 $request_filename : 當前請求的文件路徑,由root或alias指令與URI請求生成。 $scheme : HTTP方法(如http,https)。 $server_protocol : 請求使用的協議,一般是HTTP/1.0或HTTP/1.1。 $server_addr : 服務器地址,在完成一次系統調用後能夠肯定這個值。 $server_name : 服務器名稱。 $server_port : 請求到達服務器的端口號。 $request_uri : 包含請求參數的原始URI,不包含主機名,如:」/foo/bar.php?arg=baz」。 $uri : 不帶請求參數的當前URI,$uri不包含主機名,如」/foo/bar.html」。 $document_uri : 與$uri相同。 

如:服務器

訪問連接是:http://localhost:88/test1/test2/test.php 網站路徑是:/var/www/html $host:localhost $server_port:88 $request_uri:http://localhost:88/test1/test2/test.php $document_uri:/test1/test2/test.php $document_root:/var/www/html $request_filename:/var/www/html/test1/test2/test.php 

例子

# 若是文件不存在則返回400
if (!-f $request_filename) {
    return 400;
}

# 若是host不是xuexb.com,則301到xuexb.com中
if ( $host != "xuexb.com" ){
    rewrite ^/(.*)$ https://xuexb.com/$1 permanent;
}

# 若是請求類型不是POST則返回405
if ($request_method = POST) {
    return 405;
}

# 若是參數中有 a=1 則301到指定域名
if ($args ~ a=1) {
    rewrite ^ http://example.com/ permanent;
}

在某種場景下可結合location規則來使用,如:cookie

# 訪問 /test.html 時
location = /test.html {
    # 默認值爲xiaowu
    set $name xiaowu;

    # 若是參數中有 name=xx 則使用該值
    if ($args ~* name=(\w+?)(&|$)) {
        set $name $1;
    }

    # 301
    rewrite ^ /$name.html permanent;
}

上面表示:post

  • /test.html => /xiaowu.html
  • /test.html?name=ok => /ok.html?name=ok

location

語法

server塊中使用,如:網站

server {
    location 表達式 {
    }
}

location表達式類型

  • 若是直接寫一個路徑,則匹配該路徑下的
  • ~ 表示執行一個正則匹配,區分大小寫
  • ~* 表示執行一個正則匹配,不區分大小寫
  • ^~ 表示普通字符匹配。使用前綴匹配。若是匹配成功,則再也不匹配其餘location。
  • = 進行普通字符精確匹配。也就是徹底匹配。

優先級

  1. 等號類型(=)的優先級最高。一旦匹配成功,則再也不查找其餘匹配項。
  2. ^~類型表達式。一旦匹配成功,則再也不查找其餘匹配項。
  3. 正則表達式類型(~ ~*)的優先級次之。若是有多個location的正則能匹配的話,則使用正則表達式最長的那個。
  4. 常規字符串匹配類型。按前綴匹配。

例子 - 假地址掩飾真地址

server { # 用 xxoo_admin 來掩飾 admin location / { # 使用break拿一旦匹配成功則忽略後續location rewrite /xxoo_admin /admin break; } # 訪問真實地址直接報沒權限 location /admin { return 403; } } 

參數連接

參考 Nginx location在配置中的優先級

轉自連接:https://xuexb.com/html/nginx-url-rewrite.html

相關文章
相關標籤/搜索