httpd使用VirtualHost指令進行虛擬主機的定義。支持三種虛擬主機:基於ip,基於埠和基於名稱。其中基於埠的虛擬主機在httpd的術語上(例如官方手冊)也屬於基於IP的。 當一個請求到達時,將首先匹配虛擬主機。匹配虛擬主機的規則為最佳匹配法。所謂最佳,是指通配的越少,匹配的優先順序越高。 ...
httpd使用VirtualHost指令進行虛擬主機的定義。支持三種虛擬主機:基於ip,基於埠和基於名稱。其中基於埠的虛擬主機在httpd的術語上(例如官方手冊)也屬於基於IP的。
當一個請求到達時,將首先匹配虛擬主機。匹配虛擬主機的規則為最佳匹配法。所謂最佳,是指通配的越少,匹配的優先順序越高。例如"192.168.100.14:80"的優先順序高於"*:80"。如果基於名稱的虛擬主機無法匹配上,則採用虛擬主機列表中的第一個虛擬主機作為響應主機。如果所有虛擬主機都無法匹配上,則採用從主配置段落中的主機,如果主配置段落中註釋了DocumentRoot,則返回對應的錯誤。
主配置段落的指令基本上都能使用在虛擬主機容器中。至於虛擬主機中必須配有什麼指令,這沒有規定,因為虛擬主機只是封裝一組指令而已,即使其中沒有任何指令,它也會從主配置段落中繼承。但是,既然要使用且已經使用了虛擬主機,按照常理來說,至少得提供不同的ServerName,DocumentRoot等指令以讓它們各自獨立。
最後需要說明的是,httpd的"-S"選項在調試虛擬主機配置選項時非常有用。
1 基於IP的虛擬主機
基於IP的虛擬主機是在不同的IP+PORT上提供不同的站點服務,最常見的是在不同埠上提供不同站點。
如果僅基於IP,即使用不同IP地址,那麼要求操作系統上有兩個或更多IP地址,可以提供多個網卡,或者通過網卡別名來實現。
如果基於埠,即使用不同埠,則使用相同IP或不同IP均可,但在httpd術語中,基於單個IP但不同埠的虛擬主機,也是基於IP的虛擬主機。
假設本機為192.168.100.14。
# 首先設置個虛擬網卡。
shell> ip a add 192.168.100.144 dev eth0 label eth0:0
# 添加基於IP地址的虛擬主機,DocumentRoot使用的相對路徑,基於ServerRoot
shell> vim /etc/apache/extra/vhosts.conf
<VirtualHost 192.168.100.14:80>
ServerName www.a.com
DocumentRoot htdocs/a.com
</VirtaulHost>
<VirtualHost 192.168.100.144:80>
ServerName www.b.com
DocumentRoot htdocs/b.com
</VirtaulHost>
在主配置文件中,將該虛擬主機配置文件vhosts.conf包含進去。
include /etc/apache/extra/vhosts.conf
再提供DocumentRoot和各自的index.html。
mkdir /usr/local/apache/htdocs/{a.com,b.com}
echo '<h1>a.com<h1>' >/usr/local/apache/htdocs/a.com/index.html
echo '<h1>b.com<h1>' >/usr/local/apache/htdocs/b.com/index.html
使用httpd -S
查看配置文件載入過程。
[root@xuexi httpd-2.4.27]# httpd -S -f /etc/apache/httpd.conf
VirtualHost configuration:
192.168.100.14:80 www.a.com (/etc/apache/extra/vhosts.conf:23)
192.168.100.144:80 www.b.com (/etc/apache/extra/vhosts.conf:28)
ServerRoot: "/usr/local/apache"
Main DocumentRoot: "/usr/local/apache/htdocs"
Main ErrorLog: "/usr/local/apache/logs/error_log"
Mutex proxy: using_defaults
Mutex default: dir="/usr/local/apache/logs/" mechanism=default
PidFile: "/usr/local/apache/logs/httpd.pid"
Define: DUMP_VHOSTS
Define: DUMP_RUN_CFG
User: name="daemon" id=2
Group: name="daemon" id=2
重啟httpd。
service httpd restart
測試。
2 基於埠的虛擬主機
基於埠的虛擬主機需要監聽兩個套接字。
首先在配置文件中使用Listen指令修改監聽套接字,這裡假設只基於埠,所以只需修改埠號即可。
listen 80
listen 8080
修改虛擬主機配置文件vhosts.conf文件如下:
shell> vim /etc/apache/extra/vhosts.conf
<VirtualHost 192.168.100.14:80>
ServerName www.a.com
DocumentRoot htdocs/a.com
</VirtaulHost>
<VirtualHost 192.168.100.14:8080>
ServerName www.b.com
DocumentRoot htdocs/b.com
</VirtaulHost>
重啟httpd。測試www.a.com和www.b.com能否顯示。
3 基於名稱的虛擬主機
請求報文中獲取資源時包含了兩部分資源定位的格式:TCP/IP協議和HTTP協議,雖然TCP/IP部分相同,但是HTTP協議的請求報文中指定了HOST,這就是基於功能變數名稱的虛擬主機能實現的原因。也因此,基於名稱的虛擬主機必須指定ServerName指令,否則它將會繼承操作系統的FQDN。
shell> vim /etc/apache/extra/vhosts.conf
<VirtualHost 192.168.100.14:80>
ServerName www.a.com
DocumentRoot htdocs/a.com
</VirtaulHost>
<VirtualHost 192.168.100.14:80>
ServerName www.b.com
DocumentRoot htdocs/b.com
</VirtaulHost>
註意,對於基於名稱的虛擬主機,當使用IP地址請求(例如瀏覽器中輸入的是IP地址),或者無法匹配到任何虛擬主機時,將採用第一個虛擬主機作為預設虛擬主機。
例如,當某個hosts文件中添加了"192.168.100.14 www.c.com"時,即使在配置文件中並沒有配置www.c.com的虛擬主機,但訪問時仍然會訪問虛擬主機列表的第一個。