作為世界上最“好”的語言,在web里占據(jù)著大概80%的份額,中小公司基本都說 lnmp 架構(gòu)。當(dāng)一個倉庫開發(fā)人員大于1,20人的時候,每個人可能開發(fā)不同的模塊和功能,用代碼版本控制工具比如 git 開不同的分支,流程大概是先在本地搭一套完整的環(huán)境,開發(fā)好部署在測試環(huán)境,自測或者測試人員測試好之后部署在預(yù)發(fā)布環(huán)境,預(yù)發(fā)布基本和線上環(huán)境一樣,然后給產(chǎn)品驗收,驗收完成后再發(fā)布上線。
由于是并行開發(fā),肯定存在好幾個功能同時驗收或者測試的情況,這個時候預(yù)發(fā)環(huán)境到底部署誰的代碼呢?切換到A的分支,B就不能驗收了。所以希望存在一個多人開發(fā)環(huán)境,每個人的開發(fā)流程互不影響。
PHP運行原理
首先我們來分析下 PHP 的運行原理,看看 PHP 的語言特點。當(dāng)我們從瀏覽器發(fā)起一個請求,我們的web服務(wù)器(Nginx、Apache等)監(jiān)聽了80或者443端口,我們來看一個最簡單的 Nginx 的 vhost 配置:
server {
listen 80;
server_name test.com;
root /data/gateway/html;
index index.php;
location ~ \.php$ {
fastcgi_pass 127.0.0.1:9001; #unix:/Users/run/php-fcgi.sock;
fastcgi_index index.php;
fastcgi_param SCRIPT_FILENAME $document_root$fastcgi_script_name;
include fastcgi_params;
}
}
Nginx 監(jiān)聽80端口,當(dāng)匹配到用戶訪問的域名是test.com后使用對應(yīng)的 vhost 配置。在服務(wù)器里PHP-FPM起一個服務(wù),監(jiān)聽一個端口(比如9001)或者一個unix socket,Nginx通過fastcgi_pass配置,將請求傳遞給 PHP-FPM 來解析PHP代碼,PHP解析器每次從index.php開始解析,一路處理下去、做一系列的邏輯處理、查詢數(shù)據(jù)庫或者緩存等操作,返回一個 HTML 或者其他結(jié)果給 Nginx,Nginx 再返回給瀏覽器。流程如下圖:
CGI:是 Nginx 與 PHP_FPM 之間數(shù)據(jù)交換的一種協(xié)議。
FastCGI:同 CGI,是一種通信協(xié)議,但比 CGI 在效率上做了一些優(yōu)化。
PHP-CGI:是 PHP 對 Nginx 提供的 CGI 協(xié)議的接口程序。
PHP-FPM:是 PHP 對 Nginx 提供的 FastCGI 協(xié)議的接口程序,額外還提供了相對智能一些任務(wù)管理。
多人開發(fā)環(huán)境
從 PHP 原理我們可以看到,PHP其實只是一個解釋型的腳本語言,每次請求都要從index.php解析一次,那我們是不是可以在服務(wù)器根據(jù)不同開發(fā)者的名字,命名很多個文件夾,在各自文件夾里,clone 好代碼倉庫,切換到自己的分支。再讓 Nginx 處理每個人目錄下的index就可以了。比如直接訪問http://wulv.test.com/,在 Nginx 獲取到 wulv,把 root 設(shè)置到 wulv 這個目錄,這樣就訪問到 wulv 這個目錄下的代碼了??梢宰?Nginx 這樣設(shè)置:
set $who www;
if ($http_who != "") {
set $who $http_who;
}
root /data/gateway/$who/html;
我們可以讓 URL 里攜帶用戶的目錄,在 Nginx 截取下來,可以在一下幾個地方攜帶:
host: http://wulv.test.com
path: http://www.test.com/wulv
query: http://www.test.com?http_who=wulv
這樣大體上可以實現(xiàn)需求了,但還是有點問題,比如頁面里有些鏈接是寫死的,沒有使用相對路徑,你一點擊就又跑 www.test.com 去了,或者有些第三方應(yīng)用比如 OAuth 等需要校驗域名,你和線上域名不一致根本無法登陸。所以需要其他方式來實現(xiàn),比如:
http request header
cookie
我們可以使用Modify Headers這個瀏覽器插件,修改http request 頭信息,設(shè)置一個參數(shù) http_who 為 wulv,然后在 Nginx 獲取。
拓展
如果有條件的話,其實還可以做一個網(wǎng)關(guān)服務(wù)器,做一個配置頁面,在配置頁面里配置一下需要訪問的目錄,下次訪問,網(wǎng)關(guān)就直接幫你設(shè)置http header,代理到對應(yīng)服務(wù)器。這樣連瀏覽器插件都不需要裝了,對運營和產(chǎn)品設(shè)計更加友好。