diff --git a/.gitignore b/.gitignore index e5c56d1..e758945 100644 --- a/.gitignore +++ b/.gitignore @@ -46,3 +46,4 @@ pipeline2.yml /pipeline3.yml /pkg/service/arrange_test.go pkg/service/bnb_test.go +/hamster-test-private-key.pem diff --git a/Dockerfile b/Dockerfile index e5d49bb..8005322 100644 --- a/Dockerfile +++ b/Dockerfile @@ -1,7 +1,6 @@ -FROM docker.io/hamstershare/debian_docker_cli:20231010 -RUN npm install -g truffle +FROM docker.io/hamstershare/debian_docker_cli:20240308 -COPY ./aline-test /usr/local/bin/aline-test +COPY ./aline-test /usr/local/bin/aline-test ENV PORT=8080 ENV GRPC_PORT=50001 @@ -9,6 +8,7 @@ ENV DB_USER=root ENV DB_PASSWORD=123456 ENV DB_HOST=127.0.0.1 ENV DB_PORT=3306 +ENV REDIS_HOST=redis ENV DB_NAME=aline EXPOSE ${PORT} EXPOSE ${GRPC_PORT} diff --git a/Dockerfile_base b/Dockerfile_base index 7838149..79967cf 100644 --- a/Dockerfile_base +++ b/Dockerfile_base @@ -33,3 +33,5 @@ RUN "$HOME/.cargo/bin/rustup" default stable && \ "$HOME/.cargo/bin/rustup" update nightly && \ "$HOME/.cargo/bin/rustup" target add wasm32-unknown-unknown --toolchain nightly + +RUN npm install -g truffle diff --git a/cmd/daemon.go b/cmd/daemon.go index 9ab3f59..26fe012 100644 --- a/cmd/daemon.go +++ b/cmd/daemon.go @@ -9,6 +9,7 @@ import ( "github.com/hamster-shared/hamster-develop/pkg/application" "github.com/hamster-shared/hamster-develop/pkg/controller" "github.com/hamster-shared/hamster-develop/pkg/service" + "github.com/redis/go-redis/v9" "github.com/spf13/cobra" "gorm.io/driver/mysql" "gorm.io/gorm" @@ -58,7 +59,28 @@ to quickly create a Cobra application.`, if err != nil { return } + application.SetBean[*gorm.DB]("db", db) + redisHost := os.Getenv("REDIS_HOST") + if redisHost == "" { + redisHost = "127.0.0.1" + } + redisPort := os.Getenv("REDIS_PORT") + if redisPort == "" { + redisPort = "6379" + } + + fmt.Println(redisHost) + redisAddr := fmt.Sprintf("%s:%s", redisHost, redisPort) + + fmt.Println("redis addr : ", redisAddr) + + rdb := redis.NewClient(&redis.Options{ + Addr: redisAddr, + Password: "", // no password set + DB: 0, // use default DB + }) + application.SetBean[*redis.Client]("rdb", rdb) application.SetBean[engine.Engine]("engine", Engine) workflowService := service.NewWorkflowService() application.SetBean[*service.WorkflowService]("workflowService", workflowService) @@ -83,7 +105,7 @@ to quickly create a Cobra application.`, icpService := service.NewIcpService(icNetwork) application.SetBean[*service.IcpService]("icpService", icpService) templateService.Init(db) - projectService.Init(db) + projectService.Init(db, rdb) application.SetBean[service.IProjectService]("projectService", projectService) arrangeService := service.NewArrangeService() application.SetBean[*service.ArrangeService]("arrangeService", arrangeService) diff --git a/deploy-master.yml b/deploy-master.yml index 8beb3a1..09a4809 100644 --- a/deploy-master.yml +++ b/deploy-master.yml @@ -144,6 +144,14 @@ spec: key: APPS_RW_CLIENT_SECRETS - name: GITHUB_APP_RW_PEM value: "/home/ubuntu/rwGithubApp/hamster-rw-private-key.pem" + - name: REDIS_HOST + value: redis + - name: REDIS_PORT + value: "6379" + - name: GITHUB_BRANCH_WEBHOOK_NAME + value: hamster_test_webhook + - name: GITHUB_BRANCH_WEBHOOK_URL + value: https://develop.hamster.newtouch.com/api/v2/github/webhook ports: - containerPort: 8080 volumeMounts: diff --git a/go.mod b/go.mod index bad95d9..a0f5aac 100644 --- a/go.mod +++ b/go.mod @@ -19,6 +19,7 @@ require ( github.com/jinzhu/copier v0.3.5 github.com/mohaijiang/agent-go v0.3.1 github.com/pkg/errors v0.9.1 + github.com/redis/go-redis/v9 v9.5.1 github.com/samber/lo v1.36.0 github.com/sirupsen/logrus v1.9.0 github.com/spf13/cobra v1.6.1 @@ -45,11 +46,13 @@ require ( github.com/aviate-labs/secp256k1 v0.0.0-5e6736a // indirect github.com/benbjohnson/clock v1.3.0 // indirect github.com/btcsuite/btcd/btcec/v2 v2.2.0 // indirect + github.com/cespare/xxhash/v2 v2.2.0 // indirect github.com/cloudflare/circl v1.3.3 // indirect github.com/crackcomm/go-gitignore v0.0.0-20170627025303-887ab5e44cc3 // indirect github.com/davecgh/go-spew v1.1.1 // indirect github.com/deckarep/golang-set/v2 v2.1.0 // indirect github.com/decred/dcrd/dcrec/secp256k1/v4 v4.1.0 // indirect + github.com/dgryski/go-rendezvous v0.0.0-20200823014737-9f7001d12a5f // indirect github.com/di-wu/parser v0.3.0 // indirect github.com/emicklei/go-restful/v3 v3.10.2 // indirect github.com/evanphx/json-patch v5.6.0+incompatible // indirect diff --git a/pkg/controller/gin.go b/pkg/controller/gin.go index 6316aa3..90c22ba 100644 --- a/pkg/controller/gin.go +++ b/pkg/controller/gin.go @@ -62,6 +62,7 @@ func (h *HttpServer) StartHttpServer() { api.POST("/projects/code", h.handlerServer.createProjectByCodeV2) api.GET("/projects/:id", h.handlerServer.projectDetail) api.PUT("/projects/:id", h.handlerServer.updateProject) + api.PUT("/projects/:id/branch", h.handlerServer.setProjectRepositoryBranch) // 查询项目分支信息 api.DELETE("projects/:id", h.handlerServer.deleteProject) api.POST("/projects/check-name", h.handlerServer.checkName) api.GET("/user", h.handlerServer.getUseInfo) diff --git a/pkg/controller/log_handler.go b/pkg/controller/log_handler.go index d784383..62cb852 100644 --- a/pkg/controller/log_handler.go +++ b/pkg/controller/log_handler.go @@ -169,7 +169,7 @@ func (h *HandlerServer) getDeployFrontendLog(gin *gin.Context) { //userAny, _ := gin.Get("user") //user, _ := userAny.(db2.User) - project, err := h.projectService.GetProject(projectIdStr) + project, err := h.projectService.GetProject(projectIdStr, 0) if err != nil { log.Println("get project failed", err.Error()) Fail(err.Error(), gin) diff --git a/pkg/controller/login_handler.go b/pkg/controller/login_handler.go index 0787fe4..f629e5e 100644 --- a/pkg/controller/login_handler.go +++ b/pkg/controller/login_handler.go @@ -1,9 +1,12 @@ package controller import ( + "context" + "encoding/json" "fmt" "github.com/gin-gonic/gin" "github.com/golang-jwt/jwt/v4" + "github.com/google/go-github/v48/github" "github.com/hamster-shared/aline-engine/logger" "github.com/hamster-shared/hamster-develop/pkg/application" "github.com/hamster-shared/hamster-develop/pkg/consts" @@ -11,6 +14,7 @@ import ( "github.com/hamster-shared/hamster-develop/pkg/parameter" "github.com/hamster-shared/hamster-develop/pkg/service" "github.com/hamster-shared/hamster-develop/pkg/utils" + "github.com/redis/go-redis/v9" "gorm.io/gorm" "log" "net/http" @@ -132,13 +136,25 @@ func (h *HandlerServer) githubWebHook(gin *gin.Context) { func (h *HandlerServer) githubWebHookV2(gin *gin.Context) { event := gin.GetHeader("X-GitHub-Event") githubService := application.GetBean[*service.GithubService]("githubService") - var githubInstall parameter.GithubWebHookInstall - err := gin.BindJSON(&githubInstall) + rdb := application.GetBean[*redis.Client]("rdb") + + bytes, err := gin.GetRawData() + + logger.Info("event name: ", event) + logger.Info("github webhook: ", string(bytes)) + + //err := gin.BindJSON(&githubInstall) if err != nil { Fail(err.Error(), gin) return } if event == "installation" { + var githubInstall parameter.GithubWebHookInstall + err = json.Unmarshal(bytes, &githubInstall) + if err != nil { + Fail(err.Error(), gin) + return + } if githubInstall.Action == "created" { githubService.HandleAppsInstall(githubInstall, consts.SAVE_INSTALL, "GITHUB_APP_ID", "GITHUB_APP_PEM") err = githubService.HandlerInstallData(githubInstall.Installation.GetID(), githubInstall.Installation.GetAppID(), consts.INSTALLATION_CREATED) @@ -160,6 +176,12 @@ func (h *HandlerServer) githubWebHookV2(gin *gin.Context) { } } if event == "installation_repositories" { + var githubInstall parameter.GithubWebHookInstall + err = json.Unmarshal(bytes, &githubInstall) + if err != nil { + Fail(err.Error(), gin) + return + } githubService.UpdateRepositorySelection(githubInstall.Installation.GetID(), githubInstall.Installation.GetRepositorySelection()) if githubInstall.Action == "added" { //err = githubService.HandlerInstallData(githubInstall.Installation.GetID(), consts.REPO_ADDED) @@ -179,6 +201,33 @@ func (h *HandlerServer) githubWebHookV2(gin *gin.Context) { } } } + if event == "create" { + var createEvent github.CreateEvent + err = json.Unmarshal(bytes, &createEvent) + if err != nil { + Fail(err.Error(), gin) + return + } + if createEvent.GetRefType() == "branch" { + key := fmt.Sprintf("PROJECT_BRANCH:https://github.com/%s.git", createEvent.GetRepo().GetFullName()) + ctx, _ := context.WithTimeout(context.Background(), time.Second*10) + _, err = rdb.RPush(ctx, key, createEvent.GetRef()).Result() + } + } + if event == "delete" { + var deleteEvent github.DeleteEvent + err = json.Unmarshal(bytes, &deleteEvent) + if err != nil { + Fail(err.Error(), gin) + return + } + if deleteEvent.GetRefType() == "branch" { + key := fmt.Sprintf("PROJECT_BRANCH:https://github.com/%s.git", deleteEvent.GetRepo().GetFullName()) + ctx, _ := context.WithTimeout(context.Background(), time.Second*10) + _, err = rdb.LRem(ctx, key, 0, deleteEvent.GetRef()).Result() + } + } + } func (h *HandlerServer) githubWebHookRw(gin *gin.Context) { @@ -384,6 +433,7 @@ func (h *HandlerServer) JwtAuthorize() gin.HandlerFunc { gin.Abort() return } + githubToken = user.Token gin.Set("user", user) gin.Set("userId", user.Id) diff --git a/pkg/controller/project_handler.go b/pkg/controller/project_handler.go index 9d10701..84f4e25 100644 --- a/pkg/controller/project_handler.go +++ b/pkg/controller/project_handler.go @@ -1,13 +1,16 @@ package controller import ( + "context" "embed" + "errors" "fmt" "io/ioutil" "log" "net/http" "strconv" "strings" + "time" "github.com/gin-gonic/gin" "github.com/hamster-shared/aline-engine/logger" @@ -108,6 +111,7 @@ func (h *HandlerServer) importProject(g *gin.Context) { return } token := tokenData.GetToken() + fmt.Println("import token: ", token) // parsing url owner, name, err := service.ParsingGitHubURL(importData.CloneURL) if err != nil { @@ -123,6 +127,9 @@ func (h *HandlerServer) importProject(g *gin.Context) { return } + ctx, _ := context.WithTimeout(context.Background(), time.Second*10) + githubService.CreateRepoBranchWebhook(ctx, token, owner, name) + data := vo.CreateProjectParam{ Name: importData.Name, Type: importData.Type, @@ -181,7 +188,7 @@ func (h *HandlerServer) importProject(g *gin.Context) { return } // get project(check detail, build detail) - project, err := h.projectService.GetProject(id.String()) + project, err := h.projectService.GetProject(id.String(), 0) if err != nil { Fail(err.Error(), g) return @@ -309,7 +316,7 @@ func (h *HandlerServer) createProject(g *gin.Context) { Fail(err.Error(), g) return } - project, err := h.projectService.GetProject(id.String()) + project, err := h.projectService.GetProject(id.String(), 0) if err != nil { logger.Error(err) Fail(err.Error(), g) @@ -468,7 +475,7 @@ func (h *HandlerServer) createProjectByCodeV2(gin *gin.Context) { Fail(err.Error(), gin) return } - project, err := h.projectService.GetProject(id.String()) + project, err := h.projectService.GetProject(id.String(), 0) if err != nil { Fail(err.Error(), gin) return @@ -659,7 +666,7 @@ func (h *HandlerServer) createProjectV2(g *gin.Context) { Fail(err.Error(), g) return } - project, err := h.projectService.GetProject(id.String()) + project, err := h.projectService.GetProject(id.String(), 0) if err != nil { logger.Error(err) Fail(err.Error(), g) @@ -744,9 +751,41 @@ func (h *HandlerServer) createProjectV2(g *gin.Context) { Success(id, g) } +func getUserFromGin(gin *gin.Context) (*db2.User, error) { + loginType, exit := gin.Get("loginType") + if !exit { + return nil, errors.New("unauthorized") + } + var userAny any + if loginType == consts.GitHub { + userAny, exit = gin.Get("user") + if !exit { + return nil, errors.New("unauthorized") + } + user, _ := userAny.(db2.User) + return &user, nil + } + if loginType == consts.Metamask { + userAny, exit = gin.Get("githubUser") + if !exit { + return nil, errors.New("unauthorized") + } + } + user, _ := userAny.(db2.User) + return &user, nil +} + func (h *HandlerServer) projectDetail(gin *gin.Context) { id := gin.Param("id") - data, err := h.projectService.GetProject(id) + + user, err := getUserFromGin(gin) + if err != nil { + Failed(http.StatusUnauthorized, "access not authorized", gin) + return + } + userId := int(user.Id) + + data, err := h.projectService.GetProject(id, userId) if err != nil { Fail(err.Error(), gin) return @@ -1070,7 +1109,7 @@ func (h *HandlerServer) queryAptosParams(g *gin.Context) { } // 先查询到此项目的 github 仓库信息 - data, err := h.projectService.GetProject(projectID) + data, err := h.projectService.GetProject(projectID, 0) if err != nil { Fail(err.Error(), g) return @@ -1312,23 +1351,11 @@ func (h *HandlerServer) updateProject(gin *gin.Context) { } user, _ := userAny.(db2.User) updateData.UserId = int(user.Id) - project, err := h.projectService.GetProject(id) + project, err := h.projectService.GetProject(id, 0) if err != nil { Fail(err.Error(), gin) return } - //githubService := application.GetBean[*service.GithubService]("githubService") - //repo, res, err := githubService.UpdateRepo(token, user.Username, project.Name, updateData.Name) - //if err != nil { - // if res != nil { - // if res.StatusCode == http.StatusUnauthorized || res.StatusCode == http.StatusForbidden { - // Failed(http.StatusUnauthorized, "access not authorized", gin) - // return - // } - // } - // Fail(err.Error(), gin) - // return - //} updateData.RepositoryUrl = project.RepositoryUrl err = h.projectService.UpdateProject(id, updateData) if err != nil { @@ -1426,7 +1453,7 @@ func (h *HandlerServer) createProjectByCode(gin *gin.Context) { Fail(err.Error(), gin) return } - project, err := h.projectService.GetProject(id.String()) + project, err := h.projectService.GetProject(id.String(), 0) if err != nil { Fail(err.Error(), gin) return @@ -1514,7 +1541,7 @@ func (h *HandlerServer) workflowSetting(gin *gin.Context) { Fail(err.Error(), gin) return } - project, err := h.projectService.GetProject(id) + project, err := h.projectService.GetProject(id, 0) if err != nil { Fail(err.Error(), gin) return @@ -1690,3 +1717,22 @@ func (h *HandlerServer) getChainNetworkByName(gin *gin.Context) { } Success(list, gin) } + +func (h *HandlerServer) setProjectRepositoryBranch(gin *gin.Context) { + id := gin.Param("id") + userAny, _ := gin.Get("user") + user, _ := userAny.(db2.User) + var updateProjectBranch parameter.UpdateProjectBranch + err := gin.BindJSON(&updateProjectBranch) + if err != nil { + Fail(err.Error(), gin) + return + } + err = h.projectService.UpdateProjectBranch(id, int64(user.Id), updateProjectBranch.Branch) + if err != nil { + Fail(err.Error(), gin) + return + } + + Success(nil, gin) +} diff --git a/pkg/controller/workflow_handler.go b/pkg/controller/workflow_handler.go index c99b764..4428c0d 100644 --- a/pkg/controller/workflow_handler.go +++ b/pkg/controller/workflow_handler.go @@ -176,7 +176,7 @@ func (h *HandlerServer) contractFileContent(gin *gin.Context) { } userAny, _ := gin.Get("user") user, _ := userAny.(db2.User) - data, err := h.projectService.GetProject(idStr) + data, err := h.projectService.GetProject(idStr, 0) if err != nil { Fail(err.Error(), gin) return diff --git a/pkg/db/backend.go b/pkg/db/backend.go index 296e720..25e9575 100644 --- a/pkg/db/backend.go +++ b/pkg/db/backend.go @@ -21,6 +21,8 @@ type BackendPackage struct { Status consts.DeployStatus `json:"status"` // see #consts. Branch string `json:"branch"` CodeInfo string `json:"codeInfo"` + CommitId string `json:"commitId"` + CommitInfo string `json:"commitInfo"` } type BackendDeploy struct { @@ -39,4 +41,7 @@ type BackendDeploy struct { Status consts.DeployStatus `json:"status"` // 1: deploying, 2: success , 3: fail AbiInfo string `json:"abiInfo"` Name string `json:"name"` + Branch string `json:"branch"` + CommitId string `json:"commitId"` + CommitInfo string `json:"commitInfo"` } diff --git a/pkg/db/contract.go b/pkg/db/contract.go index 3946960..682846a 100644 --- a/pkg/db/contract.go +++ b/pkg/db/contract.go @@ -24,6 +24,8 @@ type Contract struct { Type uint `json:"type"` // see #consts.ProjectFrameType Status uint `json:"status"` // 1: deploying, 2: success , 3: fail Branch string `json:"branch"` + CommitId string `json:"commitId"` + CommitInfo string `json:"commitInfo"` CodeInfo string `json:"codeInfo"` } @@ -41,4 +43,7 @@ type ContractDeploy struct { DeployTxHash string `json:"deployTxHash"` Status uint `json:"status"` // 1: deploying, 2: success , 3: fail AbiInfo string `json:"abiInfo"` + Branch string `json:"branch"` + CommitId string `json:"commitId"` + CommitInfo string `json:"commitInfo"` } diff --git a/pkg/db/frontend_deploy.go b/pkg/db/frontend_deploy.go index 1fc0d69..ebfbdcc 100644 --- a/pkg/db/frontend_deploy.go +++ b/pkg/db/frontend_deploy.go @@ -17,6 +17,8 @@ type FrontendDeploy struct { Image string `json:"image"` Version string `json:"version"` Branch string `json:"branch"` + CommitId string `json:"commitId"` + CommitInfo string `json:"commitInfo"` Domain string `json:"domain"` DeployInfo string `json:"deployInfo"` DeployTime sql.NullTime `json:"deployTime"` diff --git a/pkg/db/frontend_package.go b/pkg/db/frontend_package.go index 18362cd..eba1d7b 100644 --- a/pkg/db/frontend_package.go +++ b/pkg/db/frontend_package.go @@ -13,6 +13,8 @@ type FrontendPackage struct { Name string `json:"name"` Version string `json:"version"` Branch string `json:"branch"` + CommitId string `json:"commitId"` + CommitInfo string `json:"commitInfo"` Domain string `json:"domain"` BuildTime time.Time `json:"buildTime"` PackageIdentity string `json:"packageIdentity"` diff --git a/pkg/db/migration/v41_repository_branch_support.sql b/pkg/db/migration/v41_repository_branch_support.sql new file mode 100644 index 0000000..aa38b7e --- /dev/null +++ b/pkg/db/migration/v41_repository_branch_support.sql @@ -0,0 +1,50 @@ +alter table t_backend_package + add commit_id varchar(50) null comment '代码提交id'; + +alter table t_backend_package + add commit_info varchar(100) null comment '代码提交信息'; + +alter table t_frontend_package + add commit_id varchar(50) null comment '代码提交id'; + +alter table t_frontend_package + add commit_info varchar(100) null comment '代码提交信息'; + +alter table t_workflow_detail + add branch varchar(50) null comment '分支信息'; + +alter table t_workflow_detail + add commit_id varchar(50) null comment '代码提交id'; + +alter table t_workflow_detail + add commit_info varchar(100) null comment '代码提交信息'; + +alter table t_contract + add commit_id varchar(50) null comment '代码提交id'; +alter table t_contract + add commit_info varchar(100) null comment '代码提交信息'; + +alter table t_frontend_deploy + add commit_id varchar(50) null comment '代码提交id'; + +alter table t_frontend_deploy + add commit_info varchar(100) null comment '代码提交信息'; + + +alter table t_contract_deploy + add branch varchar(50) null comment '分支信息'; + +alter table t_contract_deploy + add commit_id varchar(50) null comment '代码提交id'; + +alter table t_contract_deploy + add commit_info varchar(100) null comment '代码提交信息'; + +alter table t_backend_deploy + add branch varchar(50) null comment '分支信息'; + +alter table t_backend_deploy + add commit_id varchar(50) null comment '代码提交id'; + +alter table t_backend_deploy + add commit_info varchar(100) null comment '代码提交信息'; diff --git a/pkg/db/report.go b/pkg/db/report.go index 5c7e75c..b718902 100644 --- a/pkg/db/report.go +++ b/pkg/db/report.go @@ -21,4 +21,7 @@ type Report struct { Issues int `json:"issues"` MetaScanOverview string `json:"metaScanOverview"` CreateTime time.Time `gorm:"column:create_time;default:current_timestamp" json:"createTime"` + Branch string `json:"branch"` + CommitId string `json:"commitId"` + CommitInfo string `json:"commitInfo"` } diff --git a/pkg/db/workflow.go b/pkg/db/workflow.go index c5ede28..410a4e1 100644 --- a/pkg/db/workflow.go +++ b/pkg/db/workflow.go @@ -34,6 +34,9 @@ type WorkflowDetail struct { Status uint StartTime time.Time Duration int64 + Branch string + CommitId string + CommitInfo string CreateTime time.Time `gorm:"column:create_time;default:current_timestamp" json:"create_time"` UpdateTime time.Time `json:"update_time"` DeleteTime gorm.DeletedAt `gorm:"index;column:delete_time;" json:"delete_time"` diff --git a/pkg/parameter/project_param.go b/pkg/parameter/project_param.go index 912c83c..00b5540 100644 --- a/pkg/parameter/project_param.go +++ b/pkg/parameter/project_param.go @@ -46,3 +46,7 @@ type K8sDeployParam struct { ServicePort int32 `json:"servicePort"` ServiceTargetPort int32 `json:"serviceTargetPort"` } + +type UpdateProjectBranch struct { + Branch string `json:"branch"` +} diff --git a/pkg/service/contract.go b/pkg/service/contract.go index 1ff7340..ed6e89c 100644 --- a/pkg/service/contract.go +++ b/pkg/service/contract.go @@ -209,7 +209,7 @@ func (c *ContractService) SaveDeploy(deployParam parameter.ContractDeployParam) } projectService := application.GetBean[*ProjectService]("projectService") - project, err := projectService.GetProject(projectId.String()) + project, err := projectService.GetProject(projectId.String(), 0) _ = copier.Copy(&entity, &deployParam) entity.DeployTime = time.Now() entity.ProjectId = projectId @@ -275,7 +275,7 @@ func (c *ContractService) QueryContracts(projectId string, query, version, netwo return c.QueryContractsForICP(projectId, query, version, network, page, size) } - sql := fmt.Sprintf("select id, project_id,workflow_id,workflow_detail_id,name,version,group_concat( DISTINCT `network` SEPARATOR ',' ) as network,build_time,abi_info,byte_code,create_time from t_contract where project_id = ? ") + sql := fmt.Sprintf("select id, project_id,workflow_id,workflow_detail_id,name,version,group_concat( DISTINCT `network` SEPARATOR ',' ) as network,build_time,abi_info,byte_code,create_time,branch,commit_id,commit_info from t_contract where project_id = ? ") if query != "" && version != "" && network != "" { sql = sql + "and name like CONCAT('%',?,'%') and version = ? and network like CONCAT('%',?,'%') group by id order by create_time desc" c.db.Raw(sql, projectId, query, version, network).Scan(&contracts) @@ -331,7 +331,7 @@ func (c *ContractService) QueryContracts(projectId string, query, version, netwo func (c *ContractService) QueryContractsForICP(projectId string, query, version, network string, page int, size int) (vo.Page[vo.ContractArtifactsVo], error) { var backendPackages []db2.BackendPackage var afterData []db2.BackendPackage - sql := fmt.Sprintf("select id, project_id,workflow_id,workflow_detail_id,name,version,group_concat( DISTINCT `network` SEPARATOR ',' ) as network,build_time,abi_info,create_time from t_backend_package where project_id = ? ") + sql := fmt.Sprintf("select id, project_id,workflow_id,workflow_detail_id,name,version,group_concat( DISTINCT `network` SEPARATOR ',' ) as network,build_time,abi_info,create_time,branch,commit_id,commit_info from t_backend_package where project_id = ? ") if query != "" && version != "" && network != "" { sql = sql + "and name like CONCAT('%',?,'%') and version = ? and network like CONCAT('%',?,'%') group by id order by create_time desc" c.db.Raw(sql, projectId, query, version, network).Scan(&backendPackages) @@ -532,18 +532,18 @@ func (c *ContractService) GetCodeInfoByVersion(projectId, version string) (vo.Co } if project.FrameType == consts.InternetComputer { - res := c.db.Model(db2.BackendPackage{}).Select("version", "branch", "code_info").Where("project_id = ? and version = ?", projectId, version).Order("create_time desc").Limit(1).First(&contractVersionAndCodeInfoVo) + res := c.db.Model(db2.BackendPackage{}).Select("version", "branch", "commit_id", "commit_info").Where("project_id = ? and version = ?", projectId, version).Order("create_time desc").Limit(1).First(&contractVersionAndCodeInfoVo) if res.Error != nil { return contractVersionAndCodeInfoVo, res.Error } } else { - res := c.db.Model(db2.Contract{}).Select("version", "branch", "code_info").Where("project_id = ? and version = ?", projectId, version).Order("create_time desc").Limit(1).First(&contractVersionAndCodeInfoVo) + res := c.db.Model(db2.Contract{}).Select("version", "branch", "commit_id", "commit_info").Where("project_id = ? and version = ?", projectId, version).Order("create_time desc").Limit(1).First(&contractVersionAndCodeInfoVo) if res.Error != nil { return contractVersionAndCodeInfoVo, res.Error } } contractVersionAndCodeInfoVo.Type = int(project.Type) - contractVersionAndCodeInfoVo.Url = project.RepositoryUrl + "/" + contractVersionAndCodeInfoVo.Branch + contractVersionAndCodeInfoVo.Url = project.RepositoryUrl return contractVersionAndCodeInfoVo, nil } @@ -645,8 +645,10 @@ func (c *ContractService) GetContractDeployInfo(id int) (vo.ContractDeployVo, er } copier.Copy(&result, &contractDeploy) result.ContractName = contract.Name - result.Url = project.RepositoryUrl + "/" + contract.Branch - result.CodeInfo = contract.CodeInfo + result.Url = project.RepositoryUrl + result.Branch = contract.Branch + result.CommitId = contract.CommitId + result.CommitInfo = contract.CommitInfo return result, err } diff --git a/pkg/service/github.go b/pkg/service/github.go index 3cf40e6..7d01ed2 100644 --- a/pkg/service/github.go +++ b/pkg/service/github.go @@ -14,6 +14,7 @@ import ( "github.com/hamster-shared/hamster-develop/pkg/utils" "github.com/hamster-shared/hamster-develop/pkg/vo" "github.com/pkg/errors" + "github.com/samber/lo" "github.com/wujiangweiphp/go-curl" "golang.org/x/oauth2" "gorm.io/gorm" @@ -877,7 +878,7 @@ func (g *GithubService) HandleAppsInstallRw(appInstallData parameter.GithubWebHo return nil }) if err != nil { - logger.Errorf("save install user failed,user is: ", user.GetLogin()) + //logger.Errorf("save install user failed,user is: ", user.GetLogin()) continue } } @@ -1011,3 +1012,73 @@ func (g *GithubService) QueryRepos(installationId int64, page, size int, query s repoPage.PageSize = size return repoPage, nil } + +func (g *GithubService) ListRepositoryBranch(ctx context.Context, token string, owner, repoName string) ([]string, error) { + ts := oauth2.StaticTokenSource( + &oauth2.Token{AccessToken: token}, + ) + tc := oauth2.NewClient(ctx, ts) + client := github.NewClient(tc) + branches, _, err := client.Repositories.ListBranches(ctx, owner, repoName, &github.BranchListOptions{}) + if err != nil { + return nil, err + } + return lo.Map(branches, func(item *github.Branch, index int) string { + return item.GetName() + }), err +} + +func (g *GithubService) CreateRepoBranchWebhook(ctx context.Context, token string, owner, repo string) { + + webhookName := os.Getenv("GITHUB_BRANCH_WEBHOOK_NAME") + if webhookName == "" { + webhookName = "hamster_webhook" + } + + webhookUrl := os.Getenv("GITHUB_BRANCH_WEBHOOK_URL") + + ts := oauth2.StaticTokenSource( + &oauth2.Token{AccessToken: token}, + ) + tc := oauth2.NewClient(ctx, ts) + client := github.NewClient(tc) + + hooks, _, err := client.Repositories.ListHooks(ctx, owner, repo, &github.ListOptions{}) + if err != nil { + logger.Error("webhook query fail: ", err.Error()) + return + } + + for _, hook := range hooks { + if hook.Config["url"] == webhookUrl { + if utils.ContainsString(hook.Events, "create") && utils.ContainsString(hook.Events, "delete") { + return + } else { + hook.Events = append(hook.Events, "create") + hook.Events = append(hook.Events, "delete") + _, _, err := client.Repositories.EditHook(ctx, owner, repo, hook.GetID(), hook) + if err != nil { + logger.Error("edit hook fail: ", err.Error()) + return + } + } + } + } + + // 创建 webhook 配置 + hook := &github.Hook{ + Name: github.String(webhookName), + Events: []string{"create", "delete"}, + Config: map[string]interface{}{ + "url": webhookUrl, + "content_type": "json", + }, + Active: github.Bool(true), + } + + // 添加 webhook + _, _, err = client.Repositories.CreateHook(ctx, owner, repo, hook) + if err != nil { + logger.Error("Error creating webhook:", err.Error()) + } +} diff --git a/pkg/service/github_test.go b/pkg/service/github_test.go index 611bc90..0f3618e 100644 --- a/pkg/service/github_test.go +++ b/pkg/service/github_test.go @@ -5,8 +5,10 @@ import ( "fmt" "github.com/bradleyfalzon/ghinstallation/v2" "github.com/google/go-github/v48/github" + "github.com/hamster-shared/aline-engine/logger" "github.com/hamster-shared/hamster-develop/pkg/consts" "github.com/hamster-shared/hamster-develop/pkg/utils" + "github.com/sirupsen/logrus" "github.com/stretchr/testify/assert" "golang.org/x/oauth2" "io" @@ -197,3 +199,45 @@ func TestGetRepo(t *testing.T) { fmt.Println(fmt.Sprintf("SHA信息 %s, 提交时间是 %s, Message是 %s \n", sha, time, message)) } } + +func TestCreateWebhook(t *testing.T) { + logger.Init().ToStdoutAndFile().SetLevel(logrus.DebugLevel) + + token := "ghs_grsdp6pxRyGBU9Kk0DjP6xKulQFV0X19O19r" + // 创建一个 GitHub 客户端 + ctx := context.Background() + ts := oauth2.StaticTokenSource( + &oauth2.Token{AccessToken: token}, + ) + tc := oauth2.NewClient(ctx, ts) + client := github.NewClient(tc) + + // 设置要添加 webhook 的仓库信息 + owner := "mohaijiang" + repo := "my-vue-branch-test" + + // 创建 webhook 配置 + hook := &github.Hook{ + Name: github.String("hamster"), + Events: []string{"create", "delete"}, + Config: map[string]interface{}{ + "url": "https://develop.hamster.newtouch.com/api/v2/github/branch-webhook", + "content_type": "json", + }, + Active: github.Bool(true), + } + + // 添加 webhook + _, _, err := client.Repositories.CreateHook(ctx, owner, repo, hook) + if err != nil { + log.Fatal("Error creating webhook:", err) + } + + hooks, _, err := client.Repositories.ListHooks(ctx, owner, repo, &github.ListOptions{}) + if err != nil { + return + } + fmt.Println(hooks) + + log.Println("Webhook created successfully") +} diff --git a/pkg/service/project.go b/pkg/service/project.go index b714d14..f5b39f5 100644 --- a/pkg/service/project.go +++ b/pkg/service/project.go @@ -1,6 +1,7 @@ package service import ( + "context" "encoding/json" "errors" "fmt" @@ -13,6 +14,7 @@ import ( "github.com/hamster-shared/hamster-develop/pkg/vo" uuid "github.com/iris-contrib/go.uuid" "github.com/jinzhu/copier" + "github.com/redis/go-redis/v9" "gorm.io/gorm" "log" "os/exec" @@ -25,7 +27,7 @@ type IProjectService interface { GetProjects(userId int, keyword string, page, size, projectType int) (*vo.ProjectPage, error) HandleProjectsByUserId(user db2.User, page, size int, token, filter string) (vo.RepoListPage, error) CreateProject(createData vo.CreateProjectParam) (uuid.UUID, error) - GetProject(id string) (*vo.ProjectDetailVo, error) + GetProject(id string, userId int) (*vo.ProjectDetailVo, error) UpdateProject(id string, updateData vo.UpdateProjectParam) error DeleteProject(id string) error UpdateProjectParams(id string, updateData vo.UpdateProjectParams) error @@ -35,18 +37,21 @@ type IProjectService interface { ParsingEVMFrame(repoContents []*github.RepositoryContent) (consts.EVMFrameType, error) GetChainNetworkList() ([]db2.ChainNetwork, error) GetChainNetworkByName(name string) (db2.ChainNetwork, error) + UpdateProjectBranch(id string, userId int64, branch string) error } type ProjectService struct { - db *gorm.DB + db *gorm.DB + rdb *redis.Client } func NewProjectService() *ProjectService { return &ProjectService{} } -func (p *ProjectService) Init(db *gorm.DB) { +func (p *ProjectService) Init(db *gorm.DB, rdb *redis.Client) { p.db = db + p.rdb = rdb } func (p *ProjectService) GetProjects(userId int, keyword string, page, size, projectType int) (*vo.ProjectPage, error) { @@ -72,11 +77,11 @@ func (p *ProjectService) GetProjects(userId int, keyword string, page, size, pro var workflowBuildData db2.WorkflowDetail var workflowCheckData db2.WorkflowDetail _ = copier.Copy(&data, &project) - err := p.db.Model(db2.WorkflowDetail{}).Where("project_id = ? and type = ?", project.Id, consts.Check).Order("start_time DESC").Limit(1).Find(&workflowCheckData).Error + err := p.db.Model(db2.WorkflowDetail{}).Where("project_id = ? and type = ? and branch = ?", project.Id, consts.Check, project.Branch).Order("start_time DESC").Limit(1).Find(&workflowCheckData).Error if err == nil { _ = copier.Copy(&recentCheck, workflowCheckData) } - err = p.db.Model(db2.WorkflowDetail{}).Where("project_id = ? and type = ?", project.Id, consts.Build).Order("start_time DESC").Limit(1).Find(&workflowBuildData).Error + err = p.db.Model(db2.WorkflowDetail{}).Where("project_id = ? and type = ? and branch = ?", project.Id, consts.Build, project.Branch).Order("start_time DESC").Limit(1).Find(&workflowBuildData).Error if err == nil { _ = copier.Copy(&recentBuild, &workflowBuildData) if projectType == int(consts.CONTRACT) { @@ -87,6 +92,15 @@ func (p *ProjectService) GetProjects(userId int, keyword string, page, size, pro } } } + + // branches + branches, err := p.getProjectBranches(data.RepositoryUrl, userId) + if err != nil { + data.AllBranch = []string{data.Branch} + } else { + data.AllBranch = branches + } + //recentDeploy var workflowDeployData db2.WorkflowDetail if projectType == int(consts.CONTRACT) { @@ -113,7 +127,7 @@ func (p *ProjectService) GetProjects(userId int, keyword string, page, size, pro } else { var packageDeploy vo.PackageDeployVo var deployData db2.FrontendDeploy - err = p.db.Model(db2.WorkflowDetail{}).Where("project_id = ? and type = ?", project.Id, consts.Deploy).Order("create_time DESC").Limit(1).Find(&workflowDeployData).Error + err = p.db.Model(db2.WorkflowDetail{}).Where("project_id = ? and type = ? and branch = ?", project.Id, consts.Deploy, project.Branch).Order("create_time DESC").Limit(1).Find(&workflowDeployData).Error if err == nil { copier.Copy(&packageDeploy, workflowDeployData) err = p.db.Model(db2.FrontendDeploy{}).Where("project_id = ? and workflow_detail_id = ? ", project.Id, workflowDeployData.Id).Order("deploy_time DESC").Limit(1).Find(&deployData).Error @@ -126,6 +140,7 @@ func (p *ProjectService) GetProjects(userId int, keyword string, page, size, pro } data.RecentBuild = recentBuild data.RecentCheck = recentCheck + projectList = append(projectList, data) } } @@ -159,7 +174,7 @@ func (p *ProjectService) CreateProject(createData vo.CreateProjectParam) (uuid.U return project.Id, errors.New(fmt.Sprintf("application:%s already exists", createData.Name)) } -func (p *ProjectService) GetProject(id string) (*vo.ProjectDetailVo, error) { +func (p *ProjectService) GetProject(id string, userId int) (*vo.ProjectDetailVo, error) { var data db2.Project var detail vo.ProjectDetailVo result := p.db.Where("id = ? ", id).First(&data) @@ -225,6 +240,17 @@ func (p *ProjectService) GetProject(id string) (*vo.ProjectDetailVo, error) { } detail.RecentBuild = recentBuild detail.RecentCheck = recentCheck + + if userId != 0 { + // branches + branches, err := p.getProjectBranches(data.RepositoryUrl, userId) + if err != nil { + detail.AllBranch = []string{data.Branch} + } else { + detail.AllBranch = branches + } + } + return &detail, nil } @@ -459,3 +485,73 @@ func parsingPackageJson(fileContent *github.RepositoryContent, name, userName, t } return 0, fmt.Errorf("canot ensure the frontend frame type") } + +func (p *ProjectService) UpdateProjectBranch(id string, userId int64, branch string) error { + project, err := p.GetProjectById(id) + if err != nil { + return err + } + + if project.UserId != userId { + return errors.New("permission error") + } + + return p.db.Model(&db2.Project{}).Where("id", id).Update("branch", branch).Error +} + +func (p *ProjectService) getProjectBranches(repositoryUrl string, userId int) ([]string, error) { + + fmt.Println("getProjectBranches") + fmt.Println("userId: ", userId) + key := fmt.Sprintf("PROJECT_BRANCH:%s", repositoryUrl) + + ctx := context.Background() + exists, err := p.rdb.Exists(ctx, key).Result() + + fmt.Println("exists:", exists) + fmt.Println("exists err : ", err) + + if exists == 0 { + fmt.Println("api query github repo branch") + // branches + githubService := application.GetBean[*GithubService]("githubService") + ctx, _ := context.WithTimeout(context.Background(), time.Second*20) + owner, repo, err := ParsingGitHubURL(repositoryUrl) + + var gitAppInstall db2.GitAppInstall + err = p.db.Model(&db2.GitAppInstall{}).Where("user_id", userId).Where("name", owner).First(&gitAppInstall).Error + + tokenData, err := githubService.GetToken(gitAppInstall.InstallId) + if err != nil { + return nil, err + } + token := tokenData.GetToken() + + if err != nil { + return nil, err + } else { + branches, err2 := githubService.ListRepositoryBranch(ctx, token, owner, repo) + if err2 != nil { + fmt.Println(err2) + return nil, err2 + } + + fmt.Println("query branch result: ", branches) + + for _, branch := range branches { + _, err = p.rdb.RPush(ctx, key, branch).Result() + } + + return branches, err + } + + } else { + fmt.Println("redis query github repo branch") + branches, err := p.rdb.LRange(ctx, key, 0, -1).Result() + if err != nil { + return nil, err + } else { + return branches, nil + } + } +} diff --git a/pkg/service/user.go b/pkg/service/user.go index 90c0dc0..e423318 100644 --- a/pkg/service/user.go +++ b/pkg/service/user.go @@ -8,6 +8,7 @@ import ( type IUserService interface { GetUserByToken(token string) (db2.User, error) + GetGithubInstallId(userId uint) (int64, error) } type UserService struct { @@ -42,6 +43,15 @@ func (u *UserService) GetUserById(id int64) (db2.User, error) { return user, nil } +func (u *UserService) GetGithubInstallId(userId uint) (int64, error) { + var gitAppInstall db2.GitAppInstall + err := u.db.Model(db2.GitAppInstall{}).Where("user_id = ?", userId).First(&gitAppInstall).Error + if err != nil { + return 0, err + } + return gitAppInstall.InstallId, nil +} + func (u *UserService) GetUserCount() (int64, error) { var count int64 if err := u.db.Model(&db2.User{}).Count(&count).Error; err != nil { @@ -82,3 +92,7 @@ func (u *UserService) GetUserWalletById(id int) (db2.UserWallet, error) { func (u *UserService) UpdateUserWallet(userWallet db2.UserWallet) error { return u.db.Save(&userWallet).Error } + +func (u *UserService) SaveUserToken(id uint, token string) error { + return u.db.Model(db2.User{}).Where("id = ?", id).Update("token", token).Error +} diff --git a/pkg/service/workflow.go b/pkg/service/workflow.go index 3b4b58b..6a05c95 100644 --- a/pkg/service/workflow.go +++ b/pkg/service/workflow.go @@ -391,6 +391,7 @@ func (w *WorkflowService) ExecProjectWorkflow(project db.Project, user vo.UserAu //Status: uint(detail.Status), Status: 1, StartTime: detail.StartTime, + Branch: project.Branch, CreateTime: time.Now(), UpdateTime: time.Now(), } diff --git a/pkg/service/workflow_sync.go b/pkg/service/workflow_sync.go index 3831f6c..15cfd5a 100644 --- a/pkg/service/workflow_sync.go +++ b/pkg/service/workflow_sync.go @@ -71,6 +71,9 @@ func (w *WorkflowService) SyncStatus(message model.StatusChangeMessage) { workflowDetail.CodeBranch = codeInfo.Branch workflowDetail.CodeInfo = fmt.Sprintf("%s | commit on %s | %s", codeInfo.CommitId, codeInfo.CommitDate, codeInfo.CommitMessage) workflowDetail.Duration = jobDetail.Duration + workflowDetail.CommitInfo = codeInfo.CommitMessage + workflowDetail.CommitId = codeInfo.CommitId + workflowDetail.Branch = codeInfo.Branch if workflowDetail.Status != uint(message.Status) { // 如果 detail 的状态和 message 的状态不一致,可能是因为 detail 是从文件读取的,读取时还没有保存最新的状态,以 message 的状态为准 @@ -143,7 +146,9 @@ func (w *WorkflowService) syncFrontendBuild(detail *model.JobDetail, workflowDet WorkflowDetailId: workflowDetail.Id, Name: projectName, Version: fmt.Sprintf("%d", workflowDetail.ExecNumber), - Branch: workflowDetail.CodeInfo, + Branch: workflowDetail.Branch, + CommitId: workflowDetail.CommitId, + CommitInfo: workflowDetail.CommitInfo, BuildTime: workflowDetail.CreateTime, CreateTime: time.Now(), } @@ -204,6 +209,9 @@ func (w *WorkflowService) syncFrontendDeploy(detail *model.JobDetail, workflowDe packageDeploy.DeployTime = sql.NullTime{Time: time.Now(), Valid: true} packageDeploy.Name = data.Name packageDeploy.Branch = data.Branch + packageDeploy.CommitId = data.CommitId + packageDeploy.CommitInfo = data.CommitInfo + packageDeploy.CreateTime = time.Now() packageDeploy.Image = image err = w.db.Save(&packageDeploy).Error @@ -320,6 +328,9 @@ func (w *WorkflowService) SyncReport(message model.StatusChangeMessage, workflow Issues: int(datum.Total), ToolType: consts.CheckToolTypeMap[datum.Tool], MetaScanOverview: datum.ResultOverview, + Branch: workflowDetail.Branch, + CommitId: workflowDetail.CommitId, + CommitInfo: workflowDetail.CommitInfo, } w.db.Create(&report) } @@ -367,6 +378,9 @@ func (w *WorkflowService) SyncReport(message model.StatusChangeMessage, workflow CreateTime: time.Now(), Issues: contractCheckResult.Total, ToolType: consts.CheckToolTypeMap[contractCheckResult.Tool], + Branch: workflowDetail.Branch, + CommitId: workflowDetail.CommitId, + CommitInfo: workflowDetail.CommitInfo, } reportList = append(reportList, report) } @@ -385,6 +399,9 @@ func (w *WorkflowService) SyncReport(message model.StatusChangeMessage, workflow ReportFile: string(report.Content), CreateTime: time.Now(), ToolType: 5, + Branch: workflowDetail.Branch, + CommitId: workflowDetail.CommitId, + CommitInfo: workflowDetail.CommitInfo, } reportList = append(reportList, report) } @@ -432,8 +449,10 @@ func (w *WorkflowService) syncContractStarknet(projectId uuid.UUID, workflowId u CreateTime: time.Now(), Type: uint(consts.StarkWare), Status: consts.STATUS_SUCCESS, - Branch: workflowDetail.CodeBranch, CodeInfo: workflowDetail.CodeInfo, + Branch: workflowDetail.Branch, + CommitId: workflowDetail.CommitId, + CommitInfo: workflowDetail.CommitInfo, } return w.saveContractToDatabase(&contract) @@ -463,6 +482,8 @@ func (w *WorkflowService) syncContractAptos(projectId uuid.UUID, workflowId uint Status: consts.STATUS_SUCCESS, Branch: workflowDetail.CodeBranch, CodeInfo: workflowDetail.CodeInfo, + CommitId: workflowDetail.CommitId, + CommitInfo: workflowDetail.CommitInfo, } err = w.saveContractToDatabase(&contract) if err != nil { @@ -532,6 +553,8 @@ func (w *WorkflowService) syncContractSui(projectId uuid.UUID, workflowId uint, Status: consts.STATUS_SUCCESS, Branch: workflowDetail.CodeBranch, CodeInfo: workflowDetail.CodeInfo, + CommitId: workflowDetail.CommitId, + CommitInfo: workflowDetail.CommitInfo, } // logger.Tracef("aptos contract: %+v", contract) @@ -610,6 +633,9 @@ func (w *WorkflowService) syncContractSolana(projectId uuid.UUID, workflowId uin CreateTime: time.Now(), Type: uint(consts.Solana), Status: consts.STATUS_SUCCESS, + Branch: workflowDetail.Branch, + CommitId: workflowDetail.CommitId, + CommitInfo: workflowDetail.CommitInfo, } return w.saveContractToDatabase(&contract) @@ -640,6 +666,8 @@ func (w *WorkflowService) syncContractEvm(projectId uuid.UUID, workflowId uint, Status: consts.STATUS_SUCCESS, Branch: workflowDetail.CodeBranch, CodeInfo: workflowDetail.CodeInfo, + CommitId: workflowDetail.CommitId, + CommitInfo: workflowDetail.CommitInfo, } return w.saveContractToDatabase(&contract) } @@ -717,6 +745,8 @@ func (w *WorkflowService) syncInternetComputerBuild(projectId uuid.UUID, workflo Status: consts.DEPLOY_STATUS_SUCCESS, Branch: workflowDetail.CodeBranch, CodeInfo: workflowDetail.CodeInfo, + CommitId: workflowDetail.CommitId, + CommitInfo: workflowDetail.CommitInfo, } err := w.db.Save(&backendPackage).Error if err != nil { diff --git a/pkg/utils/github_client.go b/pkg/utils/github_client.go index 5d71d3a..8d54521 100644 --- a/pkg/utils/github_client.go +++ b/pkg/utils/github_client.go @@ -2,8 +2,14 @@ package utils import ( "context" + "errors" + "github.com/bradleyfalzon/ghinstallation/v2" "github.com/google/go-github/v48/github" + "github.com/hamster-shared/aline-engine/logger" "golang.org/x/oauth2" + "net/http" + "os" + "strconv" ) func NewGithubClient(ctx context.Context, token string) *github.Client { @@ -11,6 +17,34 @@ func NewGithubClient(ctx context.Context, token string) *github.Client { &oauth2.Token{AccessToken: token}, ) tc := oauth2.NewClient(ctx, ts) - return github.NewClient(tc) } + +func NewGithubClientWithPrivateKey() (*github.Client, error) { + appIdString, exist := os.LookupEnv("GITHUB_APP_ID") + if !exist { + logger.Errorf("please contact the administrator to configure 'GITHUB_APP_ID'") + return nil, errors.New("please contact the administrator to configure 'GITHUB_APP_ID'") + } + appId, err := strconv.Atoi(appIdString) + if err != nil { + logger.Errorf("app id format failed:%s", err) + return nil, err + } + appPemPath, exist := os.LookupEnv("GITHUB_APP_PEM") + if !exist { + logger.Errorf("please contact the administrator to configure 'GITHUB_APP_PEM'") + return nil, errors.New("please contact the administrator to configure 'GITHUB_APP_PEM'") + } + atr, err := ghinstallation.NewAppsTransportKeyFromFile(http.DefaultTransport, int64(appId), appPemPath) + if err != nil { + logger.Errorf("get github client by private key failed:%s", err) + return nil, err + } + client := github.NewClient(&http.Client{Transport: atr}) + return client, nil +} + +func NewGithubClientWithEmpty() *github.Client { + return github.NewClient(nil) +} diff --git a/pkg/utils/strings.go b/pkg/utils/strings.go index b5c9e06..796e5b5 100644 --- a/pkg/utils/strings.go +++ b/pkg/utils/strings.go @@ -26,3 +26,13 @@ func RemoveDuplicatesAndJoin(input string, split string) string { // 将切片重新合并为字符串,以逗号分隔 return strings.Join(result, split) } + +// 判断字符串数组是否包含特定字符串 +func ContainsString(arr []string, target string) bool { + for _, str := range arr { + if str == target { + return true + } + } + return false +} diff --git a/pkg/vo/contract_vo.go b/pkg/vo/contract_vo.go index aa9007b..f467d92 100644 --- a/pkg/vo/contract_vo.go +++ b/pkg/vo/contract_vo.go @@ -42,11 +42,13 @@ type ContractVo struct { } type ContractVersionAndCodeInfoVo struct { - Version string `json:"version"` - Type int `json:"type"` - Branch string `json:"branch"` - CodeInfo string `json:"codeInfo"` - Url string `json:"url"` + Version string `json:"version"` + Type int `json:"type"` + Branch string `json:"branch"` + CommitId string `json:"commitId"` + CommitInfo string `json:"commitInfo"` + CodeInfo string `json:"codeInfo"` + Url string `json:"url"` } type ContractDeployVo struct { @@ -66,6 +68,9 @@ type ContractDeployVo struct { AbiInfo string `json:"abiInfo"` Url string `json:"url"` CodeInfo string `json:"codeInfo"` + Branch string `json:"branch"` + CommitId string `json:"commitId"` + CommitInfo string `json:"commitInfo"` } type ContractArtifactsVo struct { @@ -84,6 +89,8 @@ type ContractArtifactsVo struct { Type uint `json:"type"` // see #consts.ProjectFrameType Status uint `json:"status"` // 1: deploying, 2: success , 3: fail Branch string `json:"branch"` + CommitId string `json:"commitId"` + CommitInfo string `json:"commitInfo"` CodeInfo string `json:"codeInfo"` LastContractDeployId uint `json:"lastContractDeployId"` } diff --git a/pkg/vo/project_vo.go b/pkg/vo/project_vo.go index c32d98b..02434b0 100644 --- a/pkg/vo/project_vo.go +++ b/pkg/vo/project_vo.go @@ -29,6 +29,7 @@ type ProjectListVo struct { RecentCheck RecentCheckVo `json:"recentCheck"` RecentBuild RecentBuildVo `json:"recentBuild"` RecentDeploy interface{} `json:"recentDeploy"` + AllBranch []string `json:"allBranch"` } type ProjectDetailVo struct { @@ -46,6 +47,7 @@ type ProjectDetailVo struct { RecentBuild RecentBuildVo `json:"recentBuild"` RecentDeploy interface{} `json:"recentDeploy"` EvmTemplateType uint `json:"evmTemplateType"` + AllBranch []string `json:"allBranch"` } type RecentCheckVo struct { diff --git a/pkg/vo/workflow_vo.go b/pkg/vo/workflow_vo.go index 261a052..7d0fba4 100644 --- a/pkg/vo/workflow_vo.go +++ b/pkg/vo/workflow_vo.go @@ -28,6 +28,8 @@ type WorkflowVo struct { Duration int64 `json:"duration"` Engine string `json:"engine"` // workflow,arrange_execute Version string `json:"version"` + Branch string `json:"branch"` + CommitId string `json:"commitId"` } type WorkflowDetailVo struct {