我想做的是這麼一件事:git 每建立一個新分支,push之後自動部署的伺服器的不同路徑,然後根據不同埠訪問不同分支的程式碼,以方便進行該分支的測試。
並且我正在學習 thinkphp 5 呢,正好測試一下。
舉個例子來說就是:當我新建了一個分支為 8081_opqnext_test,然後完成了某個功能。git add、git commit、當 git push 之後,伺服器會自動拉取 8081_opqnext_test 分支上的程式碼到 /data/tp/8081 這個目錄裡,然後 訪問 beta.tp.opqnext.com:8081 就會直接訪問到該分支的程式碼。同理,當我建立看一個 8082_user_list 的分支。完成了使用者列表的功能。git push 之後伺服器會自動建立目錄 /data/tp/8082 並且將程式碼到 pull 到該目錄。當你訪問 beta.tp.opqnext.com:8082 的時候,會訪問到該分支的程式碼便於測試。那 master 的程式碼每次合併之後 都會自動部署到 /data/tp.opqnext.com/ 訪問 tp.opqnext.com 即線上的程式碼。
然後我們分開步驟來說:
- 首先完成自動部署功能。
- 各個分支自動部署到不同路徑。
- 訪問不同的埠對映到不同的分支程式碼。
首先完成自動部署
這裡我用的是 碼雲 git@oschina 來做的。
首先需要配置 ssh公鑰,看這裡:
特別提醒的是:你的 nginx 是用哪個使用者跑的,那麼在生產公鑰之前先要切換到該使用者下。比如我 nginx 是 ngxuser 使用者,假如你在 root 下生產公鑰並且配置到碼雲,你的hook是沒有許可權 clone 和 pull 程式碼的/(ㄒoㄒ)/~~
你可以按如下命令來生成sshkey:
sudo ngxuser # 你的nginx使用者
ssh-keygen -t rsa -C "xxxxx@xxxxx.com"
# Generating public/private rsa key pair...
# 三次回車即可生成 ssh key
檢視你的 public key,並把他新增到碼雲(Gitee.com)
cat ~/.ssh/id_rsa.pub
# ssh-rsa AAAAB3NzaC1yc2EAAAADAQABAAABAQC6eNtGpNGwstc....
新增後,在終端(Terminal)中輸入
ssh -T git@git.oschina.net
若返回
Welcome to Git@OSC, yourname!
則說明新增成功,然後去 碼雲後臺 新增你的 ssh key
然後去配置 webhook 檢視教程 配置完成 webhook 之後,你每次 push 之後就會觸發這個鉤子。最簡單的,你寫一個 auto_deploy_code.php 放到網站根目錄。內容是:
<?php
echo `<pre>`;
system("git pull", $var);
不用接收 webhook 給你發過來的資料,你只管 pull ,假如你只有一個分支的話。
但是我要接收呢,既然有了資料,當然要接收。
我建立了一個新的分支,這個分支名稱可以規定好,比如:8081_opqnext_user (埠-開發者-開發功能)這裡怎麼約定,那你後面的指令碼就怎麼寫唄。
自動部署的指令碼:
<?php
/**
* GIT webHook 自動部署
* @author 郭貳小姐
* @time: 2017/7/13
*/
$hook = str_replace(`"`,`"`,$_POST[`hook`]);
$res = json_decode($hook,true);
$token = `你配置的token`;
if ($res[`password`] != $token) {
error_log(`auto_git_pull [`.date(`Y-m-d H:i:s`).`] token is error]`);
} else {
$dir = `/data/tp/`;
$branch = explode(`/`,$res[`push_data`][`ref`])[2];
$port = explode(`_`,$branch)[0];
$dir = `/data/tp/`.$port;
if(isset($res[`push_data`][`commits`]) && !empty($res[`push_data`][`commits`])){
if($branch == `master`){
$cmd = "cd /data/tp.opqnext.com; git pull origin master";
shell_exec($cmd);
error_log(`auto_git_pull `.$res[`push_data`][`user_name`].` `.$branch.` msg:`.$res[`push_data`][`commits`][0][`message`].` =>`.date(`H:i:s`));
} else {
if(!is_dir($dir)){
mkdir($dir,0777,true);
$cmd = "cd $dir; git clone -b $branch git@git.oschina.net:opqnext/thinkphp.git $dir";
} else {
$cmd = "cd $dir; git pull origin $branch";
}
shell_exec($cmd);
error_log(`auto_git_pull `.$res[`push_data`][`user_name`].` `.$branch.` msg:`.$res[`push_data`][`commits`][0][`message`].` =>`.date(`H:i:s`));
}
} else {
// 刪除分支
$cmd = "cd /data/tp; rm -rf $port";
shell_exec($cmd);
error_log(`auto_git_del_branch `.$res[`push_data`][`user_name`].` `.$branch.` =>`.date(`H:i:s`));
}
}
首先檢測 token 是否正確,獲取分支名稱,擷取埠號,填寫埠完整路徑。
如果 commit 資訊有內容,說明判斷分支是否為 master 如果是就拉線上程式碼,如果是分支 commit 檢測埠路徑是否存在,存在即 pull 否則 clone 沒了。
既然各個分支的目錄都已經建立好了,程式碼也可以自動部署了。那再配置 nginx 就可以了,我本來以為可以寫一個類似於正則匹配,匹配到哪個埠就去訪問對應的路徑,但是貌似不行,所以只能寫多個 server 監聽不同的埠,配置相應的路徑。
比如 8081,8082,8083,8084…
每個開發者三五個埠,不錯了。大概就是這樣一個理論。我發現我寫程式碼從來都是道理基本通,但是不能用…如果你要用的話,再修改修改就 beautiful 了。