update: 完善系统

This commit is contained in:
zhaoyingbo 2022-07-09 18:41:28 +08:00
parent c5a5e45c97
commit 7979662086
4 changed files with 240 additions and 56 deletions

1
devDocker.sh Normal file
View File

@ -0,0 +1 @@
docker run -it --rm --name $(basename `pwd`) -v `pwd`:`pwd` -w `pwd` -p 3001:3001 -v ~/.ssh:/root/.ssh golang:1.18.2 /bin/bash

215
main.go
View File

@ -12,6 +12,7 @@ import (
"strings"
"sync"
"time"
"regexp"
)
const (
@ -60,12 +61,136 @@ func init() {
}()
}
func getBranch(str string) (string) {
r, _ := regexp.Compile("refs/heads/(.*)")
matchArr := r.FindStringSubmatch(str)
if len(matchArr) > 0 {
return matchArr[len(matchArr)-1]
}
return ""
}
func runCommand(script string, projectPath string, isFile bool) (error) {
log.Printf("Running script: %s\n", script)
var cmd *exec.Cmd
if (isFile) {
cmd = exec.Command(script)
} else {
cmd = exec.Command("/bin/bash", "-c", script)
}
cmd.Stdout = logFile
cmd.Stderr = logFile
cmd.Dir = projectPath
err := cmd.Run()
return err
}
func manageProj(projectPath string, payload GogsPayload) (error) {
// 如果没有这个文件夹就现clone下来
if _, err := os.Stat(projectPath); os.IsNotExist(err) {
log.Printf("%s is not exist, start clone", projectPath)
// 创建文件夹
os.Mkdir(projectPath, os.ModePerm)
// 给文件夹权限
os.Chmod(projectPath, os.ModePerm)
// clone 到指定文件夹
script := "git clone " + payload.Repository.SSHURL + " " + projectPath
runErr := runCommand(script, projectPath, false)
if runErr != nil {
return runErr
}
}
// 切换到指定的分支
branch := getBranch(payload.Ref)
script := "git checkout " + branch
runErr := runCommand(script, projectPath, false)
if runErr != nil {
return runErr
}
// 拉取代码
script = "git pull"
runErr = runCommand(script, projectPath, false)
if runErr != nil {
return runErr
}
// 切换到指定的commit
script = "git reset --hard " + payload.After
runErr = runCommand(script, projectPath, false)
if runErr != nil {
return runErr
}
return nil
}
func manageDeploy(scriptPath string, projectPath string) (error) {
// 如果用户是Root就给部署文件添加执行权限
if os.Getenv("HOME") == "/root" {
log.Printf("Cur user is root, chmod u=rwx to %s", scriptPath)
script := "chmod u=rwx " + scriptPath
runErr := runCommand(script, projectPath, false)
if runErr != nil {
return runErr
}
}
// 开始执行脚本
runErr := runCommand(scriptPath, projectPath, true)
return runErr
}
func checkNeedDeploy(payload GogsPayload) (Config, bool) {
// 我们这个系统简单点只要本次提交包含了部署信号就直接部署最后一个commit
config := Config{
Repo: "",
Path: home,
Script: "deploy.py",
Signal: "{D}",
Branch: "",
}
// 找到指定的项目,获取用户配置
for _, item := range configs {
if item.Repo == payload.Repository.FullName {
config.Repo = item.Repo
if item.Path != "" {
config.Path = item.Path
}
if item.Script != "" {
config.Script = item.Script
}
if item.Signal != "" {
config.Signal = item.Signal
}
if item.Branch != "" {
config.Branch = item.Branch
}
break
}
}
// 检查是否需要执行部署脚本,优先分支,然后是推送信息
// 分支匹配
if config.Branch != "" && config.Branch == getBranch(payload.Ref) {
return config, true
}
// 推送信息匹配
for _, commit := range payload.Commits {
if strings.Contains(commit.Message, config.Signal) {
return config, true
}
}
return config, false
}
func main() {
addr := flag.String("a", ":3001", "Address to listen on")
flag.Parse()
http.HandleFunc(path, func(w http.ResponseWriter, r *http.Request) {
// 解析Payload
// 获取请求体
reqBody, err := ioutil.ReadAll(r.Body)
if err != nil {
log.Printf("Error reading request body: %v\n", err)
@ -73,6 +198,7 @@ func main() {
return
}
// 解析payload
var payload GogsPayload
err = json.Unmarshal(reqBody, &payload)
if err != nil {
@ -81,65 +207,42 @@ func main() {
return
}
log.Printf("Received: [Name] %s [Sender] %s", payload.Repository.FullName, payload.Sender.FullName)
// 打印代码库名称以及推送者
log.Printf("Received: [Repository] %s [Sender] %s", payload.Repository.FullName, payload.Sender.Username)
// 寻找包含Signal的commit
for _, commit := range payload.Commits {
// 寻找配置
c := Config{
Path: home,
Script: "deploy.py",
Signal: "{D}",
}
for _, config := range configs {
if config.Repo == payload.Repository.FullName {
c.Repo = config.Repo
if config.Path != "" {
c.Path = config.Path
}
if config.Script != "" {
c.Script = config.Script
}
if config.Signal != "" {
c.Signal = config.Signal
}
break
}
}
// 检查是否需要部署
deployConfig, needDeploy := checkNeedDeploy(payload)
// 包含部署信号
if strings.Contains(commit.Message, c.Signal) {
log.Printf("Commit: [SHA] %s [Message] %s", commit.ID, commit.Message)
log.Println("Ready to deploy")
if c.Repo == "" {
log.Println("Using default config")
}
projectPath := fmt.Sprintf(c.Path+"/%s/", payload.Repository.Name)
scriptPath := projectPath + c.Script
log.Printf("Running script: %s\n", scriptPath)
// 开始执行脚本
cmd := exec.Command(scriptPath)
cmd.Stdout = logFile
cmd.Stderr = logFile
cmd.Dir = projectPath
err = cmd.Run()
if err != nil {
log.Printf("Error running deploy script: %v\n", err)
w.Write([]byte(fmt.Sprintf("Deploy script error: %s", err)))
} else {
log.Print("Deploy finished\n\n")
w.Write([]byte("Deploy finished"))
}
return
}
// 不需要部署就直接返回
if (!needDeploy) {
log.Print("No deploy commit found\n\n")
w.Write([]byte("No deploy commit found. Skip."))
return
}
log.Print("No deploy commit found\n\n")
w.Write([]byte("No deploy commit found. Skip."))
// 准备部署,生成项目地址和脚本地址
projectPath := fmt.Sprintf(deployConfig.Path+"/%s/", payload.Repository.Name)
scriptPath := projectPath + deployConfig.Script
// 处理项目clone并切换到指定的分支和commit
err = manageProj(projectPath, payload)
if err != nil {
log.Printf("Error running manageProj script: %v\n", err)
w.Write([]byte(fmt.Sprintf("manageProj script error: %s", err)))
return
}
// 给脚本添加权限并执行脚本
err = manageDeploy(scriptPath, projectPath)
if err != nil {
log.Printf("Error running deploy script: %v\n", err)
w.Write([]byte(fmt.Sprintf("Deploy script error: %s", err)))
} else {
log.Print("Deploy finished\n\n")
w.Write([]byte("Deploy finished"))
}
return
})
log.Println("Webhook service started at " + *addr)

View File

@ -9,6 +9,8 @@ type Config struct {
Script string `json:"script"`
// 信号commit messgae里包含信号则执行脚本
Signal string `json:"signal"`
// 分支 指定分支执行脚本
Branch string `json:"branch"`
}
type GogsPayload struct {

78
path.dio Normal file
View File

@ -0,0 +1,78 @@
<mxfile host="65bd71144e">
<diagram id="gu7OVYjIMTTFq4SVhP3i" name="第 1 页">
<mxGraphModel dx="1406" dy="958" grid="1" gridSize="10" guides="1" tooltips="1" connect="1" arrows="1" fold="1" page="1" pageScale="1" pageWidth="827" pageHeight="1169" math="0" shadow="0">
<root>
<mxCell id="0"/>
<mxCell id="1" parent="0"/>
<mxCell id="4" value="" style="edgeStyle=none;html=1;" edge="1" parent="1" source="2">
<mxGeometry relative="1" as="geometry">
<mxPoint x="240" y="190" as="targetPoint"/>
</mxGeometry>
</mxCell>
<mxCell id="2" value="收到请求" style="rounded=0;whiteSpace=wrap;html=1;" vertex="1" parent="1">
<mxGeometry x="40" y="160" width="120" height="60" as="geometry"/>
</mxCell>
<mxCell id="6" value="" style="edgeStyle=none;html=1;" edge="1" parent="1">
<mxGeometry relative="1" as="geometry">
<mxPoint x="360" y="190" as="sourcePoint"/>
<mxPoint x="440" y="190" as="targetPoint"/>
</mxGeometry>
</mxCell>
<mxCell id="8" value="" style="edgeStyle=none;html=1;" edge="1" parent="1" target="7">
<mxGeometry relative="1" as="geometry">
<mxPoint x="300" y="220" as="sourcePoint"/>
</mxGeometry>
</mxCell>
<mxCell id="7" value="返回空" style="whiteSpace=wrap;html=1;rounded=0;" vertex="1" parent="1">
<mxGeometry x="240" y="300" width="120" height="60" as="geometry"/>
</mxCell>
<mxCell id="9" value="不需要" style="text;html=1;align=center;verticalAlign=middle;resizable=0;points=[];autosize=1;strokeColor=none;fillColor=none;" vertex="1" parent="1">
<mxGeometry x="255" y="250" width="50" height="20" as="geometry"/>
</mxCell>
<mxCell id="10" value="需要" style="text;html=1;align=center;verticalAlign=middle;resizable=0;points=[];autosize=1;strokeColor=none;fillColor=none;" vertex="1" parent="1">
<mxGeometry x="380" y="170" width="40" height="20" as="geometry"/>
</mxCell>
<mxCell id="11" value="是否需要部署" style="strokeWidth=2;html=1;shape=mxgraph.flowchart.decision;whiteSpace=wrap;" vertex="1" parent="1">
<mxGeometry x="240" y="140" width="120" height="100" as="geometry"/>
</mxCell>
<mxCell id="16" value="" style="edgeStyle=none;html=1;" edge="1" parent="1" source="12" target="15">
<mxGeometry relative="1" as="geometry"/>
</mxCell>
<mxCell id="18" value="" style="edgeStyle=none;html=1;" edge="1" parent="1" source="12" target="17">
<mxGeometry relative="1" as="geometry"/>
</mxCell>
<mxCell id="12" value="文件夹是否存在" style="strokeWidth=2;html=1;shape=mxgraph.flowchart.decision;whiteSpace=wrap;" vertex="1" parent="1">
<mxGeometry x="440" y="140" width="100" height="100" as="geometry"/>
</mxCell>
<mxCell id="20" value="" style="edgeStyle=none;html=1;" edge="1" parent="1" source="15" target="19">
<mxGeometry relative="1" as="geometry">
<Array as="points">
<mxPoint x="670" y="370"/>
</Array>
</mxGeometry>
</mxCell>
<mxCell id="15" value="创建文件夹" style="whiteSpace=wrap;html=1;strokeWidth=2;" vertex="1" parent="1">
<mxGeometry x="430" y="340" width="120" height="60" as="geometry"/>
</mxCell>
<mxCell id="22" value="" style="edgeStyle=none;html=1;" edge="1" parent="1" source="17" target="21">
<mxGeometry relative="1" as="geometry"/>
</mxCell>
<mxCell id="17" value="给脚本附权限" style="whiteSpace=wrap;html=1;strokeWidth=2;" vertex="1" parent="1">
<mxGeometry x="610" y="160" width="120" height="60" as="geometry"/>
</mxCell>
<mxCell id="19" value="" style="shape=waypoint;sketch=0;size=6;pointerEvents=1;points=[];fillColor=default;resizable=0;rotatable=0;perimeter=centerPerimeter;snapToPoint=1;strokeWidth=2;" vertex="1" parent="1">
<mxGeometry x="650" y="200" width="40" height="40" as="geometry"/>
</mxCell>
<mxCell id="24" value="" style="edgeStyle=none;html=1;" edge="1" parent="1" source="21" target="23">
<mxGeometry relative="1" as="geometry"/>
</mxCell>
<mxCell id="21" value="执行脚本" style="whiteSpace=wrap;html=1;strokeWidth=2;" vertex="1" parent="1">
<mxGeometry x="810" y="160" width="120" height="60" as="geometry"/>
</mxCell>
<mxCell id="23" value="返回" style="whiteSpace=wrap;html=1;strokeWidth=2;" vertex="1" parent="1">
<mxGeometry x="1010" y="160" width="120" height="60" as="geometry"/>
</mxCell>
</root>
</mxGraphModel>
</diagram>
</mxfile>