git不同分支push之後通過hook自動部署到伺服器不同路徑

null1145發表於2017-08-22

我想做的是這麼一件事: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 即線上的程式碼。

然後我們分開步驟來說:

  1. 首先完成自動部署功能。
  2. 各個分支自動部署到不同路徑。
  3. 訪問不同的埠對映到不同的分支程式碼。
首先完成自動部署

這裡我用的是 碼雲 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 了。

相關文章