1. 程式人生 > >如何設定nginx重定向

如何設定nginx重定向

原文: https://skyfi.github.io/2018/09/21/%E5%A6%82%E4%BD%95%E8%AE%BE%E7%BD%AEnginx%E9%87%8D%E5%AE%9A%E5%90%91/

nginx 是一個靈活且高效的網路伺服器,如果想要在nginx伺服器中重定,你可以從下面挑選一個適合的方式。

簡單且快速的 return

這是一個非常簡單的設定方式,只需要個return語句就可以了

1
return 301 https://example.com$request_uri;

你需要把這段程式碼放到nginx配置檔案的server程式碼塊中,301是永久重定向,你也可以設定成302

做一個臨時重定向(不建議)。

一個完整的例子:

123456
server {    listen 80;    listen [::]:80;    hostname example.com www.example.com;    return 301 https://example.com$request_uri;}

正則表示式 rewrite

如果return不能滿足你的複雜業務需求,你可以考慮下正則匹配重定向:

1
rewrite ^/foo/(bar)/(.*)$ https://$server_name/$1/$2 permanent;

同樣這也是需要在server

程式碼塊中,其中permanent301永久跳轉,若需要302可修改為redirect

一個完整的例子:

1234567
server {    listen 80;    listen [::]:80;    hostname example.com www.example.com;    root /var/www/example.com/public;    rewrite ^/foo/(bar)/(.*)$ $scheme://$server_name/$1/$2 permanent;}

又如:

1234567
server {    listen       80;    server_name   www.fangyongle.com  fangyongle.cn;    if ($host != 'www.fangyongle.com' ) {         rewrite ^/(.*)$ https://www.fangyongle.com/$1 permanent;     } }

再如:

1234567
# 根據檔案型別設定過期時間location ~* \.(js|css|jpg|jpeg|gif|png|swf)$ {    if (-f $request_filename) {       expires    1h;       break;    }}

使用Maps

如果你有一堆需要重定向的連線對映,你可以考慮在一個地方定義它,然後再通過if來手動判斷重定向。

首先定義重定向連結對映redirect-map.conf

12345
map $request_uri $redirect_uri {    /about.html          /about-us;    /customers.html      /our-customers;    /products.html       /our-products;}

然後在server程式碼塊使用:

1234567
include redirect-map.conf;server {    […]    if ( $redirect_uri ) {        return 301 $redirect_uri;    }}

對映也可以有一些語法:

123456789101112
map $request_uri $redirect_uri {    /about.html          /about-us;    /customers.html      /our-customers;    /products.html       /our-products;    # Match any url that ends in products.html or producs.htm    ~products\.html?$    /our-products;    # case-insensitive version of the above    ~*products\.html?$   /our-products;    # A named capture that maps    # e.g. product-1234.html into /products/item-1234/overview    ~product-(?<sku>\d+)\.html   /products/item-$sku/overview;}

一些實用的重定向例子

http 重定向為 https

1
return 301 https://$host$request_uri;

統一規範域名

1234
server_name example.com www.example.com example.net www.example.net _;if ( $host != $server_name ) {    return 301 $scheme://$server_name$request_uri;}

含 www 和 不含 www 之間的重定向

1234
# non-www to wwwif ( $host !~ ^www\. ) {    return 301 $scheme://www.$host$request_uri;}
1234
# www to non-wwwif ( $host ~ ^www\.(?<domain>.+)$ ) {    return 301 $scheme://$domain$request_uri;}

附錄

重定向中常用全域性變數

12345
$scheme		       // HTTP方法(如http,https),如:http$host			   // 請求主機頭欄位,否則為伺服器名稱,如:blog.fangyongle.com$server_name	   // 伺服器名稱,如:blog.fangyongle.com$request_uri	   // 包含請求引數的原始URI,不包含主機名,如:/2018/81.html?a=1&b=2$request_filename  // 當前請求的檔案的路徑名,由root或alias和URI request組合而成,如:/2013/81.html

nginx 部分常用全域性變數

123456789101112131415161718192021222324252627282930313233
$remote_addr		//獲取客戶端ip$binary_remote_addr	//客戶端ip(二進位制)$remote_port		//客戶端port,如:50472$remote_user		//已經經過Auth Basic Module驗證的使用者名稱$host			//請求主機頭欄位,否則為伺服器名稱,如:blog.fangyongle.com$request		//使用者請求資訊,如:GET ?a=1&b=2 HTTP/1.1$request_filename	//當前請求的檔案的路徑名,由root或alias和URI request組合而成,如:/2013/81.html$status			//請求的響應狀態碼,如:200$body_bytes_sent        // 響應時送出的body位元組數數量。即使連線中斷,這個資料也是精確的,如:40$content_length	       // 等於請求行的“Content_Length”的值$content_type	       // 等於請求行的“Content_Type”的值$http_referer	       // 引用地址$http_user_agent      // 客戶端agent資訊,如:Mozilla/5.0 (Windows NT 5.1) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/29.0.1547.76 Safari/537.36$args		     //與$query_string相同 等於當中URL的引數(GET),如a=1&b=2$document_uri	     //與$uri相同  這個變數指當前的請求URI,不包括任何引數(見$args) 如:/2018/81.html$document_root	     //針對當前請求的根路徑設定值$hostname	     //如:centos53.localdomain$http_cookie	    //客戶端cookie資訊$cookie_COOKIE	    //cookie COOKIE變數的值$is_args	//如果有$args引數,這個變數等於”?”,否則等於”",空值,如?$limit_rate	//這個變數可以限制連線速率,0表示不限速$query_string	    // 與$args相同 等於當中URL的引數(GET),如a=1&b=2$request_body	   // 記錄POST過來的資料資訊$request_body_file	//客戶端請求主體資訊的臨時檔名$request_method	      //客戶端請求的動作,通常為GET或POST,如:GET$request_uri	      //包含請求引數的原始URI,不包含主機名,如:/2018/81.html?a=1&b=2$scheme		       //HTTP方法(如http,https),如:http$uri			//這個變數指當前的請求URI,不包括任何引數(見$args) 如:/2018/81.html$request_completion	//如果請求結束,設定為OK. 當請求未結束或如果該請求不是請求鏈串的最後一個時,為空(Empty),如:OK$server_protocol	//請求使用的協議,通常是HTTP/1.0或HTTP/1.1,如:HTTP/1.1$server_addr		//伺服器IP地址,在完成一次系統呼叫後可以確定這個值$server_name		//伺服器名稱,如:blog.fangyongle.com$server_port		//請求到達伺服器的埠號,如:80

Rewrite正則相關指令詳解

nginx的rewrite相當於apache的rewriterule(大多數情況下可以把原有apache的rewrite規則加上引號就可以直接使用),它可以用在server,locationIF條件判斷塊中,命令格式如下:

1
rewrite <regex> <replacement> <flag>

正則表示式匹配

  • ~為區分大小寫匹配
  • ~*為不區分大小寫匹配
  • !~!~*分別為區分大小寫不匹配及不區分大小寫不匹配

檔案及目錄匹配判斷

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

flag標記

  • last - 基本上都用這個Flag。
  • break - 中止rewirte,不在繼續匹配
  • redirect - 返回臨時重定向的HTTP狀態302
  • permanent - 返回永久重定向的HTTP狀態301

使用lastbreak實現URI重寫,瀏覽器位址列不變。而且兩者有細微差別:

  • 使用alias指令必須用last標記
  • 使用proxy_pass指令時,需要使用break標記
  • last標記在本條rewrite規則執行完畢後,會對其所在server{......}標籤重新發起請求,而break標記則在本條規則匹配完成後,終止匹配。