Apache 重寫規則的常見應用 (rewrite)

本文出自:http://www.linuxforum.net 作者:吳阿亭 Jephe wu (2001-09-05 08:00:00) 
一:目的

本文旨在提供如何用Apache重寫規則來解決一些常見的URL重寫方法的問題,通過常見的 
實例給用戶一些使用重寫規則的基本方法和線索。

二:爲什麼需要用重寫規則? 
一個網站,如果是長期需要放在internet上提供服務,必定會有不斷地更新和維護,如臨 
時轉移到其它服務器進行維護,重新組織目錄結構,變換URL甚至改變到新的域名等等, 
而爲了讓客戶不會因此受到任何影響,最好的方法就是使用Apache Rewrite Rule(重寫 
規則)。

三: 重寫規則的作用範圍 
1) 可以使用在Apache主配置文件httpd.conf中 
2) 可以使用在httpd.conf裏定義的虛擬主機配置中 
3) 可以使用在基本目錄的跨越配置文件.htaccess中

四:重寫規則的應用條件 
只有當用戶的WEB請求最終被導向到某臺WEB服務器的Apache後臺,則這臺WEB服務器接受 
進來的請求,根據配置文件該請求是主配置還是虛擬主機,再根據用戶在瀏覽器中請求的 
URI來配對重寫規則並且根據實際的請求路徑配對.htaccess中的重寫規則。最後把請求 
的內容傳回給用戶,該響應可能有兩種:

1) 對瀏覽器請求內容的外部重定向(Redirect)到另一個URL。 
讓瀏覽器再次以新的URI發出請求(R=301或者R=302,臨時的或是永久的重定向) 
如:一個網站有正規的URL和別名URL,對別名URL進行重定向到正規URL,或者網站改換 
成了新的域名 
則把舊的域名重定向到新的域名(Redirect)

2) 也可能是由Apache內部子請求代理產生新的內容送回給客戶[P,L] 
這是Apache內部根據重寫後的URI內部通過代理模塊請求內容並送回內容給客戶,而客戶 
端瀏覽器並 
不知道,瀏覽器中的URI不會被重寫。但實際內容被Apache根據重寫規則後的URI得到。 
如:在公司防火牆上運行的Apache啓動這種代理重寫規則,代理對內部網段上的WEB服務 
器的請求。

五:重寫規則怎樣工作? 
我們假定在編譯Apache時已經把mod_rewrite編譯成模塊,確信你的httpd.conf中有 
LoadModule rewrite_module libexec/mod_rewrite.so 
並且在Addmodule中有 
Addmodule mod_rewrite.c 
則可以使用重寫規則。 
當外部請求來到Apache,Apache調用重寫規則中的定義來重寫由用戶瀏覽器指定請求的 
URI,最後被重寫的URI如果是重定向,則送由瀏覽器作再一次請求;如果是代理則把重寫 
後的URI交給代理模塊請求最終的內容(Content),最後把內容送回給瀏覽器。

六: 何時使用.htaccess中的重寫規則定義? 
假如你對你的的網站內容所在的服務器沒有管理員權限,或者你的網站放在ISP的服務器 
上託管等等條件下,你無法改寫主配置文件,然而你可以對你的WEB站點內容所在的目錄 
有寫權限,則你可以設置自己的.htaccess 
文件達到同樣的目的。但你需要確定主配置文件中對你的網站所在的目錄定義了下面的內 
容:

Options Indexes FollowSymLinks 
AllowOverride all

否則你的.htaccess不會工作。

七: 應用舉例 
假定Apache被編譯安裝在主機192.168.1.56的/usr/local/apache/ 目錄下面,我們編 
譯進了重寫和代理模塊。

1) 隱藏Apache下的某個目錄,使得對該目錄的任何請求都重定向到另一個文件。

a> httpd.conf的實現方法

我們放下面的部分到/usr/local/apache/conf/httpd.conf


options Indexes followsymlinks 
allowoverride all 
rewriteengine on 
rewritebase / 
rewriterule ^(.*)$ index.html.en [R=301]


注:rewriteengine on 爲重寫引擎開關,如果設爲off,則任何重寫規則定義將不被應 
用,該開關的另一好處就是如果爲了臨時拿掉重寫規則,則改爲off再重啓動Apache即 
可,不必將下面一條條的重寫規則註釋掉。 
rewritebase / 的作用是如果在下面的rewriterule定義中被重寫後的部分(此處爲文件 
名index.html.en)前面沒有/,則是相對目錄,相對於這個rewritebase後面的定義也就 
是/usr/local/apache/htdocs/index.html.en,否則,如果此處沒有rewritebase /這 
一項,則被重寫成 
http://192.168.1.56/usr/local/apache/htdocs/manual/index.html.en ,顯然是 
不正確的。

不過這裏我們也可以不用rewritebase / , 而改爲 
rewriteengine on 
rewriterule ^(.*)$ /index.html.en [R=301] 
或者 
rewriteengine on 
rewriterule ^(.*)$ http://192.168.1.56/index.html.en [R=301]

b> .htaccess的實現方法

我們先放下面的部分到httpd.conf


options Indexes followsymlinks 
allowoverride all


然後放下面的部分到/usr/local/apache/htdocs/manual/.htaccess中 
rewriteengine on 
rewritebase / 
rewriterule ^(.*)$ index.html.en [R=301]

注:對文件.htaccess所作的任何改動不需要重啓動Apache.

問:要是把這個manual目錄重定向到用戶jephe的自己的主目錄呢? 
用下面的.htaccess方案。 
rewriteengine on 
rewritebase /~jephe/ 
rewriterule ^(.*)$ $1 [R=301]

則對manual目錄下任何文件的請求被重定向到~jephe目錄下相同文件的請求。

2) 轉換www.username.domain.com的對於username的主頁請求爲 
www.domain.com/username

對於HTTP/1.1的請求包括一個Host: HTTP頭,我們能用下面的規則集重寫 
http://www.username.domain.com/anypath 到 /home/username/anypath

Rewriteengine on 
rewritecond %{HTTP_HOST} ^www/.[^.]+/.host/.com$ 
rewriterule ^(.+) %{HTTP_HOST}$1 [C] 
rewriterule ^www/.([^.]+)/.host/.com(.*) /home/$1$2

注: 
rewritecond 條件重寫規則,當滿足後面定義的條件後纔會應用下面的重寫規則, 
rewritecond有各種變量 
,請查閱相關文檔。

3) 防火牆上的重寫規則代理內部網段上服務器的請求。

NameVirtualhost 1.2.3.4


servername www.domain.com 
rewriteengine on 
proxyrequest on 
rewriterule ^/(.*)$ http://192.168.1.3/$1 [P,L]


注:當外部瀏覽器請求www.domain.com時被解析到IP地址1.2.3.4 ,Apache 交出 
mod_rewrite處理轉換成 
http://192.168.1.3/$1後再交由代理模塊mod_proxy得到內容後傳送回用戶的瀏覽器。


4) 基本預先設定的轉換MAP表進行重寫 rewritemap

轉換www.domain.com/{countrycode}/anypath 到Map表中規定的URI,上面是虛擬主機 
中的定義

rewritelog /usr/local/apache/logs/rewrite.log 
rewriteloglevel 9

rewriteengine on 
proxyrequest on 
rewritemap sitemap txt:/usr/local/apache/conf/rewrite.map 
rewriterule ^/([^/]+)+/(.*)$ http://%{REMOTE_HOST}::$1 [C] 
rewriterule (.*)::([a-z]+)$ ${sitemap:$2|http://h.i.j.k/} [R=301,L]

文件/usr/local/apache/conf/rewrite.map的內容如下:

sg http://a.b.c.d/ 
sh http://e.f.g.h/

注: 當用戶請求http://www.domain.com/sg/anypath時被重寫爲 
http://a.b.c.d/anypath . 
當需要調試時請用rewritelog and rewriteloglevel 9聯合,9爲最大即得到最多的調試 
信息 
最小爲1,最小的調試信息,默認爲0,沒有調試信息。 
sitemap的語法是${sitemap: LookupKey | Defaultvalue} ,有些書上把$寫成了%是錯 
誤的。

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章