feat: 改用gitbeaker操作gitlab
All checks were successful
CI Monitor MIflow / build-image (push) Successful in 1m1s

This commit is contained in:
zhaoyingbo 2024-08-08 12:42:03 +00:00
parent f5ee6f8555
commit 8d6e8a2844
18 changed files with 82 additions and 603 deletions

View File

@ -11,10 +11,12 @@
"devcontainers",
"eamodio",
"esbenp",
"gitbeaker",
"Gruntfuggly",
"micr",
"mioffice",
"oxlint",
"sonarqube",
"tseslint",
"wlpbbgiky",
"wujiali",

BIN
bun.lockb

Binary file not shown.

View File

@ -1,14 +1,15 @@
import { ExpandedPipelineSchema, PipelineSchema } from "@gitbeaker/rest"
import moment from "moment"
import pLimit from "p-limit"
import db from "../../db"
import service from "../../service"
import { DB } from "../../types/db"
import { Gitlab } from "../../types/gitlab"
/**
* pipeline列表
* @param {DB.Project} project -
* @returns {Promise<(Gitlab.PipelineDetail & { created_at: string })[]>} - pipeline列表
* @returns {Promise<(ExpandedPipelineSchema & { created_at: string })[]>} - pipeline列表
*/
const getFullPipelineList = async (project: DB.Project) => {
// 先获取最新的pipelineID
@ -18,14 +19,15 @@ const getFullPipelineList = async (project: DB.Project) => {
latestOne?.created_at || "2024-01-01T00:00:00.000+08:00"
)
// 获取pipeline列表并保存
const fullPipelineList: Gitlab.Pipeline[] = []
const fullPipelineList: PipelineSchema[] = []
let page = 1
let hasBeforeLatestTime = false
while (!hasBeforeLatestTime) {
const pipelines = await service.gitlab.pipeline.getList(
project.project_id,
page++
)
const pipelines = await service.gitlab.Pipelines.all(project.project_id, {
page: page++,
perPage: 100,
scope: "finished",
}).catch(() => [])
// 如果当前页没有数据,则直接跳出
if (pipelines.length === 0) break
pipelines.forEach((pipeline) => {
@ -38,25 +40,30 @@ const getFullPipelineList = async (project: DB.Project) => {
}
})
}
const limit = pLimit(10)
const fullPipelineDetailList = await Promise.all(
fullPipelineList.map(({ project_id, id, created_at }) =>
service.gitlab.pipeline.getDetail(project_id, id, created_at)
limit(() =>
service.gitlab.Pipelines.show(project_id, id)
.catch(() => null)
.then((res) => (res ? { ...res, created_at } : null))
)
)
)
return fullPipelineDetailList.filter((v) => v) as (Gitlab.PipelineDetail & {
return fullPipelineDetailList.filter((v) => v) as (ExpandedPipelineSchema & {
created_at: string
})[]
}
/**
* pipeline列表到数据库
* @param {(Gitlab.PipelineDetail & { created_at: string })[][]} fullPipelineList - pipeline列表
* @param {(ExpandedPipelineSchema & { created_at: string })[][]} fullPipelineList - pipeline列表
* @param {Record<string, string>} fullUserMap -
* @param {Record<string, string>} fullProjectMap -
* @returns {Promise<void>}
*/
const insertFullPipelineList = async (
fullPipelineList: (Gitlab.PipelineDetail & { created_at: string })[][],
fullPipelineList: (ExpandedPipelineSchema & { created_at: string })[][],
fullUserMap: Record<string, string>,
fullProjectMap: Record<string, string>
) => {

View File

@ -1,3 +1,5 @@
import type { MergeRequestSchema } from "@gitbeaker/rest"
import db from "../../db"
import service from "../../service"
import netTool from "../../service/netTool"
@ -54,10 +56,9 @@ const getNextAction = async (
}
// 在流水线为`running`时且该stage全部的job有非结束状态时即`created`、`pending`、`running`、`manual`、`scheduled`时,添加监控
if (pipeline.object_attributes.status === "running") {
const jobs = await service.gitlab.pipeline.getJobs(
pipeline.project.id,
pipeline.object_attributes.id
)
const jobs = await service.gitlab.Jobs.all(pipeline.project.id, {
pipelineId: pipeline.object_attributes.id,
}).catch(() => [])
if (
jobs.some(
(job) =>
@ -75,16 +76,16 @@ const getNextAction = async (
/**
*
* @param {Gitlab.PipelineEvent} pipeline - GitLab 线
* @returns {Promise<Gitlab.MergeRequest | null>} - null
* @returns {Promise<MergeRequestSchema | null>} - null
*/
const getMergeRequest = async (
pipeline: Gitlab.PipelineEvent
): Promise<Gitlab.MergeRequest | null> => {
): Promise<MergeRequestSchema | null> => {
if (!checkIsMergeCommit(pipeline)) return null
const res = await service.gitlab.commit.getMr(
const res = await service.gitlab.Commits.allMergeRequests(
pipeline.project.id,
pipeline.object_attributes.sha
)
).catch(() => [])
if (res.length === 0) return null
return res.find((mr) => mr.merge_commit_sha === pipeline.commit.id) || null
}
@ -92,12 +93,12 @@ const getMergeRequest = async (
/**
*
* @param {Gitlab.PipelineEvent} pipeline - GitLab 线
* @param {Gitlab.MergeRequest | null} mergeRequest - null
* @param {MergeRequestSchema | null} mergeRequest - null
* @returns {{ participant: string, receiver: string[] }} -
*/
const getUserInfo = (
pipeline: Gitlab.PipelineEvent,
mergeRequest: Gitlab.MergeRequest | null
mergeRequest: MergeRequestSchema | null
): { participant: string; receiver: string[] } => {
let participant = pipeline.user.name
const receiver = [pipeline.user.username]

View File

@ -8,13 +8,15 @@ import { DB } from "../../types/db"
* @returns {Promise<DB.Project>} -
*/
const fillProj = async (project: DB.Project): Promise<DB.Project> => {
const projDetail = await service.gitlab.project.getDetail(project.project_id)
const projDetail = await service.gitlab.Projects.show(
project.project_id
).catch(() => null)
if (!projDetail) {
return project
}
const useFulParams: Partial<DB.Project> = {
...project,
avatar_url: projDetail.avatar_url,
avatar_url: projDetail.avatar_url || "",
description: projDetail.description,
name: projDetail.name,
path_with_namespace: projDetail.path_with_namespace,

View File

@ -1,13 +1,15 @@
import { ExpandedPipelineSchema } from "@gitbeaker/rest"
import db from "../../db"
import { Gitlab } from "../../types/gitlab"
/**
*
* @param {Gitlab.PipelineDetail[][]} fullPipelineList - pipeline列表
* @param {ExpandedPipelineSchema[][]} fullPipelineList - pipeline列表
* @returns {Promise<Record<string, string>>} -
*/
const getFullUserMap = async (
fullPipelineList: Gitlab.PipelineDetail[][]
fullPipelineList: ExpandedPipelineSchema[][]
): Promise<Record<string, string>> => {
const userList: Gitlab.User[] = []
fullPipelineList.forEach((fullPipeline) => {

View File

@ -35,6 +35,7 @@
"typescript": "^5.0.0"
},
"dependencies": {
"@gitbeaker/rest": "^40.1.2",
"lodash": "^4.17.21",
"moment": "^2.30.1",
"node-schedule": "^2.1.1",

View File

@ -15,10 +15,9 @@ const doMonitor = async (monitor: DB.Monitor): Promise<void> => {
const { project_id, pipeline_id, api_key, stage, receiver, variable } =
monitor
// 获取Job列表
const jobList = await service.gitlab.pipeline.getJobs(
Number(project_id),
Number(pipeline_id)
)
const jobList = await service.gitlab.Jobs.all(project_id, {
pipelineId: Number(pipeline_id),
}).catch(() => [])
// 是否所有Stage关联的Job都成功了
const isAllSuccess = jobList.every(
(job) => job.stage === stage && job.status === "success"
@ -28,10 +27,10 @@ const doMonitor = async (monitor: DB.Monitor): Promise<void> => {
// 先删除监控
await db.monitor.del(monitor.id)
// 获取最新的执行时长
const pipelineDetail = await service.gitlab.pipeline.getDetail(
Number(project_id),
const pipelineDetail = await service.gitlab.Pipelines.show(
project_id,
Number(pipeline_id)
)
).catch(() => null)
if (pipelineDetail) {
variable["duration"] = sec2minStr(pipelineDetail.duration)
}

View File

@ -1,6 +1,7 @@
import { ProjectSchema } from "@gitbeaker/rest"
import pLimit from "p-limit"
import service from "../../service"
import { BadgeSetParams } from "../../service/gitlab/badge"
import { Gitlab } from "../../types/gitlab"
const projectList = [
// "cloud-ml/cloudml-maas",
@ -24,16 +25,23 @@ const projectList = [
"cloud-ml/lark_auth",
]
interface BadgeSetParams {
groupId: number
linkUrl: string
imageUrl: string
name: string
}
const getNewProjectBadge = async (
projectDetail: Gitlab.ProjDetail
projectDetail: ProjectSchema
): Promise<BadgeSetParams[]> => {
// 项目路径 cloud-ml/cloudml-dev
const projectPath = projectDetail.path_with_namespace
// 根据项目路径获取sonarqubeId 类似于 cloud-ml/cloudml-dev -> cloud-ml:cloudml-dev
const sonarqubeId = projectPath.replace("/", ":")
// 获取项目的badges
const badges: Gitlab.Badge[] = await service.gitlab.badge.get(
projectDetail.id
const badges = await service.gitlab.ProjectBadges.all(projectDetail.id).catch(
() => []
)
// 对badges进行补全可能只有 name: "sonarqube_coverage" 的情况,把剩下的补全
const badgeNames = badges.map((badge) => badge.name)
@ -47,20 +55,18 @@ const getNewProjectBadge = async (
]
const diff = [...badgeNameList].filter((x) => !badgeNameSet.has(x))
const newBadges: BadgeSetParams[] = diff.map((name) => {
const link_url = encodeURI(
const linkUrl = encodeURI(
`https://sonarqube.mioffice.cn/dashboard?id=${sonarqubeId}`
)
const metric = name.replace("sonarqube_", "")
const image_url =
const imageUrl =
name !== "sonarqube_quality_gate"
? `https://sonarqube.mioffice.cn/api/badges/measure?key=${sonarqubeId}&metric=${metric}`
: `https://sonarqube.mioffice.cn/api/badges/gate?key=${sonarqubeId}`
return {
id: projectDetail.id,
link_url,
image_url,
rendered_image_url: image_url,
rendered_link_url: link_url,
groupId: projectDetail.id,
linkUrl,
imageUrl,
name,
}
})
@ -68,21 +74,25 @@ const getNewProjectBadge = async (
}
const addNewProjectBadge = async (badgeSetParamsList: BadgeSetParams[]) => {
const chunkSize = 5 // 每次并发请求的数量
for (let i = 0; i < badgeSetParamsList.length; i += chunkSize) {
const chunk = badgeSetParamsList.slice(i, i + chunkSize)
const res = await Promise.all(
chunk.map((badgeSetParams) => service.gitlab.badge.add(badgeSetParams))
const limit = pLimit(5)
await Promise.all(
badgeSetParamsList.map((b) =>
limit(() =>
service.gitlab.ProjectBadges.add(b.groupId, b.linkUrl, b.imageUrl, {
name: b.name,
}).catch(() => null)
)
)
console.log(res)
}
)
}
const main = async () => {
const errorList: string[] = []
const badgeAddParamsList = await Promise.all(
projectList.map(async (projectName) => {
const projectDetail = await service.gitlab.project.getDetail(projectName)
const projectDetail = await service.gitlab.Projects.show(
projectName
).catch(() => null)
if (!projectDetail) {
errorList.push(projectName)
return []

View File

@ -1,113 +0,0 @@
import service from "../../service"
import { BadgeSetParams } from "../../service/gitlab/badge"
import { Gitlab } from "../../types/gitlab"
const projectList = [
"miai-fe/fe/ai-admin-fe",
"miai-fe/fe/ai-operation-fe",
"miai-fe/fe/ai-dripback-fe",
"miai-fe/fe/ai-open-admin-fe",
"miai-fe/fe/ai-model-deployment-fe",
"miai-fe/fe/ai-workbench",
"miai-fe/fe/ai-review-fe",
"miai-fe/fe/cms",
"miai-fe/fe/xiaoai-open-Platform",
"miai-fe/fe/ai-robot-fe",
"miai-fe/fe/ai-scene-review-fe",
"miai-fe/fe/houyi",
"miai-fe/fe/kms",
"miai-fe/fe/ai-productive-fe",
"miai-fe/fe/ai-sliding-label-fe",
"miai-fe/fe/ai-label-control-fe",
"miai-fe/fe/asr-fe",
"miai-fe/fe/feedback-platform-fe",
"miai-fe/fe/statisfaction-evaluation-platform-fe",
"miai-fe/fe/ai-quality-score-fe",
"miai-fe/fe/wakeup",
"miai-fe/fe/ai-badcase-label-fe",
"miai-fe/fe/func-manager",
"miai-fe/service/ado-server",
"miai-fe/fe/ado-fe",
"miai-fe/pwa/ai-rn-domain",
"miai-fe/pwa/ai-ak-fe",
"miai-fe/fe/ai-user-info-fe",
"miai-fe/fe-infra/xiaoai-jssdk",
"miai-fe/fe-infra/ai-ui",
"miai-fe/fe/ai-class-sheducle-fe",
"miai-fe/service/ai-schedule-service",
"miai-fe/fe/ai-schedule-devtool-v3",
"miai-fe/ai-scenes-fe",
"miai-fe/activity/om-music-tone",
"miai-fe/activity/ug-guide-test",
"miai-fe/activity/om-mood-blindbox",
"miai-fe/activity/om-love-poem",
"miai-fe/fe/ai-introduce",
"miai-fe/fe/ai-phonecall-guider-fe",
"miai-fe/activity/om-spring-couplets",
"miai-fe/activity/ug-play-assistant",
"miai-fe/fe/ai-iot-introduce-fe",
"miai-fe/fe/ai-rookie-play-fe",
"miai-fe/fe/ai-custom-tts-fe",
"miai-fe/fe-infra/split-rpk-sdk",
"miai-fe/fe/ai-full-animation-template-fe",
"miai-fe/fe/ai-schedule-manager-fe",
"miai-fe/fe/ai-voiceprint-fe",
"miai-fe/fe/ai-shortcut-fe",
]
const getProjectId = async (projectName: string) => {
const res = await service.gitlab.project.getDetail(projectName)
return res?.id
}
const getNewProjectBadge = async (
projectId: number
): Promise<BadgeSetParams[]> => {
const badges: Gitlab.Badge[] = await service.gitlab.badge.get(projectId)
const replacePath = (value: string) =>
value.replace(
"https://sonarqube.mioffice.cn",
"http://scan.sonarqube.xiaomi.srv"
)
return badges.map((badge: Gitlab.Badge) => ({
id: projectId,
badge_id: badge.id,
link_url: replacePath(badge.link_url),
image_url: badge.image_url,
rendered_image_url: badge.rendered_image_url,
rendered_link_url: replacePath(badge.rendered_link_url),
}))
}
const setNewProjectBadge = async (badgeSetParamsList: BadgeSetParams[]) => {
const chunkSize = 5 // 每次并发请求的数量
for (let i = 0; i < badgeSetParamsList.length; i += chunkSize) {
const chunk = badgeSetParamsList.slice(i, i + chunkSize)
const res = await Promise.all(
chunk.map((badgeSetParams) => service.gitlab.badge.set(badgeSetParams))
)
console.log(res)
}
}
const main = async () => {
const errorList: string[] = []
const badgeSetParamsList = await Promise.all(
projectList.map(async (projectName) => {
const projectId = await getProjectId(projectName)
if (!projectId) {
errorList.push(projectName)
return []
}
return await getNewProjectBadge(projectId)
})
)
await setNewProjectBadge(badgeSetParamsList.flat())
console.log("errorList", errorList)
}
main()

View File

@ -1,60 +0,0 @@
import { Gitlab } from "../../types/gitlab"
import netTool from "../netTool"
import { GITLAB_AUTH_HEADER, GITLAB_BASE_URL, gitlabReqWarp } from "./tools"
/**
* GitLab
*/
export type BadgeSetParams = Omit<Gitlab.Badge, "kind" | "name"> & {
badge_id?: number
}
/**
* GitLab
* @param {number} project_id - ID
* @returns {Promise<Gitlab.Badge[]>} GitLab
*/
const get = async (project_id: number): Promise<Gitlab.Badge[]> => {
const URL = `${GITLAB_BASE_URL}/projects/${project_id}/badges`
return gitlabReqWarp<Gitlab.Badge[]>(
() => netTool.get(URL, {}, GITLAB_AUTH_HEADER),
[]
)
}
/**
* GitLab
* @param {BadgeSetParams} badge -
* @returns {Promise<Gitlab.Badge | null>}
*/
const set = async (badge: BadgeSetParams): Promise<Gitlab.Badge | null> => {
const URL = `${GITLAB_BASE_URL}/projects/${badge.id}/badges/${badge.badge_id}`
return gitlabReqWarp<Gitlab.Badge>(
() => netTool.put(URL, badge, {}, GITLAB_AUTH_HEADER),
null
)
}
/**
* GitLab
* @param {BadgeSetParams} badge -
* @returns {Promise<Gitlab.Badge | null>}
*/
const add = async (badge: BadgeSetParams): Promise<Gitlab.Badge | null> => {
const URL = `${GITLAB_BASE_URL}/projects/${badge.id}/badges`
return gitlabReqWarp<Gitlab.Badge>(
() => netTool.post(URL, badge, {}, GITLAB_AUTH_HEADER),
null
)
}
/**
*
*/
const badge = {
get,
set,
add,
}
export default badge

View File

@ -1,26 +0,0 @@
import { Gitlab } from "../../types/gitlab"
import netTool from "../netTool"
import { GITLAB_AUTH_HEADER, GITLAB_BASE_URL, gitlabReqWarp } from "./tools"
/**
*
* @param {number} project_id - ID
* @param {string} sha - SHA
* @returns {Promise<Gitlab.MergeRequest[]>} promise
*/
const getMr = async (
project_id: number,
sha: string
): Promise<Gitlab.MergeRequest[]> => {
const URL = `${GITLAB_BASE_URL}/projects/${project_id}/repository/commits/${sha}/merge_requests`
return gitlabReqWarp<Gitlab.MergeRequest[]>(
() => netTool.get(URL, {}, GITLAB_AUTH_HEADER),
[]
)
}
const commit = {
getMr,
}
export default commit

View File

@ -1,13 +1,8 @@
import badge from "./badge"
import commit from "./commit"
import pipeline from "./pipeline"
import project from "./project"
import { Gitlab } from "@gitbeaker/rest"
const gitlab = {
project,
badge,
commit,
pipeline,
}
const gitlab = new Gitlab({
token: "Zd1UASPcMwVox5tNS6ep",
host: "https://git.n.xiaomi.com",
})
export default gitlab

View File

@ -1,66 +0,0 @@
import { Gitlab } from "../../types/gitlab"
import netTool from "../netTool"
import { GITLAB_AUTH_HEADER, GITLAB_BASE_URL, gitlabReqWarp } from "./tools"
/**
* GitLab流水线的详细信息
* @param {number} project_id - ID
* @param {number} pipeline_id - 线ID
* @param {string} created_at - 线
*/
const getDetail = async (
project_id: number,
pipeline_id: number,
created_at?: string
) => {
const URL = `${GITLAB_BASE_URL}/projects/${project_id}/pipelines/${pipeline_id}`
const res = await gitlabReqWarp<Gitlab.PipelineDetail>(
() => netTool.get(URL, {}, GITLAB_AUTH_HEADER),
null
)
if (res === null) return null
return { ...res, created_at }
}
/**
* GitLab流水线列表
* @param {number} project_id - ID
* @param {number} [page=1] - 1
* @returns {Promise<Gitlab.Pipeline[]>} 线promise
*/
const getList = async (
project_id: number,
page = 1
): Promise<Gitlab.Pipeline[]> => {
const URL = `${GITLAB_BASE_URL}/projects/${project_id}/pipelines`
const params = { scope: "finished", per_page: 100, page }
return gitlabReqWarp<Gitlab.Pipeline[]>(
() => netTool.get(URL, params, GITLAB_AUTH_HEADER),
[]
)
}
/**
* GitLab流水线的任务列表
* @param {number} project_id - ID
* @param {number} pipeline_id - 线ID
* @returns {Promise<Gitlab.Job[]>} promise
*/
const getJobs = async (
project_id: number,
pipeline_id: number
): Promise<Gitlab.Job[]> => {
const URL = `${GITLAB_BASE_URL}/projects/${project_id}/pipelines/${pipeline_id}/jobs`
return gitlabReqWarp<Gitlab.Job[]>(
() => netTool.get(URL, {}, GITLAB_AUTH_HEADER),
[]
)
}
const pipeline = {
getDetail,
getList,
getJobs,
}
export default pipeline

View File

@ -1,26 +0,0 @@
import { Gitlab } from "../../types/gitlab"
import netTool from "../netTool"
import { GITLAB_AUTH_HEADER, GITLAB_BASE_URL, gitlabReqWarp } from "./tools"
/**
* GitLab
* @param {number | string} project_id - ID URL-encoded
* @returns {Promise<Gitlab.ProjDetail | null>} promise
*/
const getDetail = async (
project_id: number | string
): Promise<Gitlab.ProjDetail | null> => {
if (typeof project_id === "string")
project_id = encodeURIComponent(project_id)
const URL = `${GITLAB_BASE_URL}/projects/${project_id}`
return gitlabReqWarp<Gitlab.ProjDetail>(
() => netTool.get(URL, {}, GITLAB_AUTH_HEADER),
null
)
}
const project = {
getDetail,
}
export default project

View File

@ -1,34 +0,0 @@
import { Gitlab } from "../../types/gitlab"
/**
* GitLab
* @template T
* @param {() => Promise<T>} func -
* @param {any} default_value -
* @returns {Promise<T>} promise
*/
export const gitlabReqWarp = async <T = any>(
func: () => Promise<T>,
default_value: any
): Promise<T> => {
try {
let response = {} as T & Gitlab.Error
response = (await func()) as T & Gitlab.Error
if (response.message === "404 Project Not Found") return default_value
return response
} catch {
return default_value
}
}
/**
* GitLab API URL
* @type {string}
*/
export const GITLAB_BASE_URL = "https://git.n.xiaomi.com/api/v4"
/**
* GitLab API
* @type {object}
*/
export const GITLAB_AUTH_HEADER = { "PRIVATE-TOKEN": "Zd1UASPcMwVox5tNS6ep" }

View File

@ -2,8 +2,8 @@ import gitlab from "./gitlab"
import message from "./message"
const service = {
gitlab,
message,
gitlab,
}
export default service

View File

@ -1,11 +1,4 @@
export namespace Gitlab {
/* 错误 */
export interface Error {
/**
*
*/
message: string
}
/* 用户 */
export interface User {
/**
@ -34,164 +27,6 @@ export namespace Gitlab {
web_url: string
}
/* 项目详情 */
export interface ProjDetail {
/**
* ID
*/
id: number
/**
*
*/
description: string
/**
*
*/
name: string
/**
*
*/
path_with_namespace: string
/**
* URL
*/
web_url: string
/**
* URL
*/
avatar_url?: any
/**
*
*/
message?: string
}
/* 管道详情 */
export interface PipelineDetail {
/**
* ID
*/
id: number
/**
* ID
*/
project_id: number
/**
*
*/
ref: string
/**
*
*/
status: string
/**
* URL
*/
web_url: string
/**
*
*/
user: User
/**
*
*/
started_at: string
/**
*
*/
finished_at: string
/**
*
*/
duration: number
/**
*
*/
queued_duration: number
/**
*
*/
message?: string
}
/* 管道 */
export interface Pipeline {
/**
* ID
*/
id: number
/**
* ID
*/
project_id: number
/**
* SHA值
*/
sha: string
/**
*
*/
ref: string
/**
*
*/
status: string
/**
*
*/
source: string
/**
*
*/
created_at: string
/**
*
*/
updated_at: string
/**
* URL
*/
web_url: string
}
/* 任务 */
export interface Job {
status: string
stage: string
}
/* 徽章 */
export interface Badge {
/**
* ID
*/
id: number
/**
*
*/
name: string
/**
* URL
*/
link_url: string
/**
* URL
*/
image_url: string
/**
* URL
*/
rendered_link_url: string
/**
* URL
*/
rendered_image_url: string
/**
*
*/
kind: "project" | "group"
}
/* 管道事件 */
export interface PipelineEvent {
/**
@ -297,54 +132,4 @@ export namespace Gitlab {
finished_at: string | null
}[]
}
/* 合并请求 */
export interface MergeRequest {
/**
*
*/
title: string
/**
*
*/
target_branch: string
/**
*
*/
source_branch: string
/**
*
*/
author: {
/**
*
*/
username: string
/**
*
*/
name: string
}
/**
* SHA值
*/
sha: string
/**
* SHA值
*/
merge_commit_sha: string
/**
*
*/
references: {
/**
*
*/
full: string
}
/**
* web URL
*/
web_url: string
}
}