style: 优化项目结构
All checks were successful
CI Monitor CI/CD / build-image (push) Successful in 29s
CI Monitor CI/CD / deploy (push) Successful in 34s

This commit is contained in:
zhaoyingbo 2024-07-24 10:35:50 +00:00
parent eca7e7cb41
commit 3f220f7943
29 changed files with 780 additions and 375 deletions

View File

@ -1,33 +1,24 @@
{
"name": "ci_monitor",
"image": "git.yingbo.im:333/zhaoyingbo/dev:bun",
"remoteUser": "bun",
"containerUser": "bun",
"image": "mcr.microsoft.com/devcontainers/base:ubuntu-24.04",
"customizations": {
"vscode": {
"settings": {
"files.autoSave": "afterDelay",
"editor.guides.bracketPairs": true,
"editor.defaultFormatter": "esbenp.prettier-vscode",
"editor.formatOnSave": true,
"editor.codeActionsOnSave": {
"source.fixAll.stylelint": true
}
"editor.formatOnSave": true
},
"extensions": [
"eamodio.gitlens",
"Gruntfuggly.todo-tree",
"dbaeumer.vscode-eslint",
"esbenp.prettier-vscode",
"eamodio.gitlens",
"unifiedjs.vscode-mdx",
"oderwat.indent-rainbow",
"jock.svg",
"ChakrounAnas.turbo-console-log",
"Gruntfuggly.todo-tree",
"MS-CEINTL.vscode-language-pack-zh-hans",
"stylelint.vscode-stylelint",
"GitHub.copilot",
"streetsidesoftware.code-spell-checker"
"streetsidesoftware.code-spell-checker",
"MS-CEINTL.vscode-language-pack-zh-hans"
]
}
}
},
"onCreateCommand": "curl -fsSL https://bun.sh/install | bash"
}

View File

@ -1,13 +1,13 @@
import db from "../../db";
import { PipelineRecordModel } from "../../db/pipeline";
import { ProjectRecordModel } from "../../db/project";
import service from "../../service";
import moment from "moment";
import { Gitlab } from "../../types/gitlab";
import { DB } from "../../types/db";
/**
* pipeline列表
*/
const getFullPipelineList = async (project: ProjectRecordModel) => {
const getFullPipelineList = async (project: DB.Project) => {
// 先获取最新的pipelineID
const latestOne = await db.pipeline.getLatestOne(project.id);
// 获取本次数据获取的截止时间如果没有则获取从20240101到现在所有pipeline信息
@ -15,11 +15,11 @@ const getFullPipelineList = async (project: ProjectRecordModel) => {
latestOne?.created_at || "2024-01-01T00:00:00.000+08:00"
);
// 获取pipeline列表并保存
const fullPipelineList: GitlabPipeline[] = [];
const fullPipelineList: Gitlab.Pipeline[] = [];
let page = 1;
let hasBeforeLatestTime = false;
while (!hasBeforeLatestTime) {
const pipelines = await service.gitlab.fetchPipelines(
const pipelines = await service.gitlab.pipeline.getList(
project.project_id,
page++
);
@ -37,20 +37,20 @@ const getFullPipelineList = async (project: ProjectRecordModel) => {
}
const fullPipelineDetailList = await Promise.all(
fullPipelineList.map(({ project_id, id, created_at }) =>
service.gitlab.fetchPipelineDetails(project_id, id, created_at)
service.gitlab.pipeline.getDetail(project_id, id, created_at)
)
);
return fullPipelineDetailList.filter(
(v) => v
) as GitlabPipelineDetailWithCreateAt[];
return fullPipelineDetailList.filter((v) => v) as (Gitlab.PipelineDetail & {
created_at: string;
})[];
};
const insertFullPipelineList = async (
fullPipelineList: GitlabPipelineDetailWithCreateAt[][],
fullPipelineList: (Gitlab.PipelineDetail & { created_at: string })[][],
fullUserMap: Record<string, string>,
fullProjectMap: Record<string, string>
) => {
const dbPipelineList: Partial<PipelineRecordModel> = [];
const dbPipelineList: Partial<DB.Pipeline> = [];
fullPipelineList.forEach((pipelineList) => {
pipelineList.forEach((pipeline) => {
@ -73,9 +73,7 @@ const insertFullPipelineList = async (
});
});
await Promise.all(
dbPipelineList.map((v: Partial<PipelineRecordModel>) =>
db.pipeline.create(v)
)
dbPipelineList.map((v: Partial<DB.Pipeline>) => db.pipeline.create(v))
);
};

View File

@ -1,18 +1,16 @@
import db from "../../db";
import { ProjectRecordModel } from "../../db/project";
import service from "../../service";
import { DB } from "../../types/db";
/**
*
*/
const fillProj = async (project: ProjectRecordModel) => {
const projDetail = await service.gitlab.fetchProjectDetails(
project.project_id
);
const fillProj = async (project: DB.Project) => {
const projDetail = await service.gitlab.project.getDetail(project.project_id);
if (!projDetail) {
return project;
}
const useFulParams: Partial<ProjectRecordModel> = {
const useFulParams: Partial<DB.Project> = {
...project,
avatar_url: projDetail.avatar_url,
description: projDetail.description,
@ -40,7 +38,7 @@ const getFullProjList = async () => {
return filledFullProjList;
};
const getFullProjectMap = (fullProjList: ProjectRecordModel[]) => {
const getFullProjectMap = (fullProjList: DB.Project[]) => {
const fullProjectMap: Record<string, string> = {};
fullProjList.forEach((item) => {
fullProjectMap[item.project_id] = item.id;

View File

@ -121,7 +121,7 @@ const getRobotMsg = async () =>
* @param chat_id
*/
const sendCIReportByChatId = async (chat_id: string) => {
await service.sendMessage.byChatId(chat_id, await getRobotMsg());
await service.message.byChatId(chat_id, await getRobotMsg());
};
/**
@ -129,7 +129,7 @@ const sendCIReportByChatId = async (chat_id: string) => {
* @returns
*/
const sendCIReportByCron = async () =>
await service.sendMessage.byGroupId("52usf3w8l6z4vs1", await getRobotMsg());
await service.message.byGroupId("52usf3w8l6z4vs1", await getRobotMsg());
const manageRobot = {
sendCIReportByChatId,

View File

@ -1,7 +1,8 @@
import db from "../../db";
import { Gitlab } from "../../types/gitlab";
const getFullUserMap = async (fullPipelineList: GitlabPipelineDetail[][]) => {
const userList: GitlabUser[] = [];
const getFullUserMap = async (fullPipelineList: Gitlab.PipelineDetail[][]) => {
const userList: Gitlab.User[] = [];
fullPipelineList.forEach((fullPipeline) => {
fullPipeline.forEach((item) => {
if (item.user && !userList.find((v) => v.id === item.user?.id)) {

View File

@ -1,33 +1,24 @@
import { RecordModel } from "pocketbase";
import pbClient from "../pbClient";
import { managePb404 } from "../../utils/pbTools";
import { DB } from "../../types/db";
export interface PipelineRecordModel extends RecordModel {
project_id: string;
user_id: string;
pipeline_id: number;
ref: string;
status: string;
web_url: string;
// 2024-03-06 02:53:59.509Z
created_at: string;
started_at: string;
finished_at: string;
duration: number;
queued_duration: number;
}
/**
* ID
* @param id ID
* @returns promise
*/
const getOne = (id: string) =>
managePb404<PipelineRecordModel>(
managePb404<DB.Pipeline>(
async () => await pbClient.collection("pipeline").getOne(id)
);
/**
*
* @param project_id id
*
* @param project_id ID
* @returns promise
*/
const getLatestOne = (project_id: string) => {
return managePb404<PipelineRecordModel>(
return managePb404<DB.Pipeline>(
async () =>
await pbClient
.collection("pipeline")
@ -37,12 +28,18 @@ const getLatestOne = (project_id: string) => {
);
};
const create = async (data: Partial<PipelineRecordModel>) =>
await pbClient.collection("pipeline").create<PipelineRecordModel>(data);
/**
*
* @param data
* @returns promise
*/
const create = async (data: Partial<DB.Pipeline>) =>
await pbClient.collection("pipeline").create<DB.Pipeline>(data);
const pipeline = {
create,
getOne,
getLatestOne,
};
export default pipeline;

View File

@ -1,28 +1,36 @@
import { RecordModel } from "pocketbase";
import { managePb404 } from "../../utils/pbTools";
import pbClient from "../pbClient";
import { DB } from "../../types/db";
export interface ProjectRecordModel extends RecordModel {
project_id: number;
description: string;
name: string;
path_with_namespace: string;
web_url: string;
avatar_url: string;
has_new_cicd: boolean;
}
/**
* ID
* @param id - ID
* @returns promise
*/
const getOne = (id: string) =>
managePb404<ProjectRecordModel>(
managePb404<DB.Project>(
async () => await pbClient.collection("project").getOne(id)
);
/**
*
* @returns promise
*/
const getFullList = async () =>
await pbClient.collection("project").getFullList<ProjectRecordModel>();
await pbClient.collection("project").getFullList<DB.Project>();
const update = async (id: string, data: Partial<ProjectRecordModel>) =>
await pbClient.collection("project").update<ProjectRecordModel>(id, data);
/**
* 使
* @param id - ID
* @param data -
* @returns promise
*/
const update = async (id: string, data: Partial<DB.Project>) =>
await pbClient.collection("project").update<DB.Project>(id, data);
/**
*
*/
const project = {
getFullList,
getOne,

View File

@ -1,22 +1,24 @@
import { RecordModel } from "pocketbase";
import { managePb404 } from "../../utils/pbTools";
import pbClient from "../pbClient";
import { DB } from "../../types/db";
export interface UserRecordModel extends RecordModel {
user_id: number;
username: string;
name: string;
avatar_url: string;
web_url: string;
}
/**
* ID从数据库检索单个用户
* @param id ID
* @returns promise404
*/
const getOne = (id: string) =>
managePb404<UserRecordModel>(
managePb404<DB.User>(
async () => await pbClient.collection("user").getOne(id)
);
/**
* ID从数据库检索单个用户
* @param user_id ID
* @returns promise404
*/
const getOneByUserId = (user_id: number) => {
return managePb404<UserRecordModel>(
return managePb404<DB.User>(
async () =>
await pbClient
.collection("user")
@ -26,16 +28,32 @@ const getOneByUserId = (user_id: number) => {
);
};
const create = async (data: Partial<UserRecordModel>) =>
await pbClient.collection("user").create<UserRecordModel>(data);
/**
*
* @param data
* @returns promise
*/
const create = async (data: Partial<DB.User>) =>
await pbClient.collection("user").create<DB.User>(data);
const upsert = async (data: Partial<UserRecordModel>) => {
/**
*
* ID的用户已存在
* ID的用户不存在
* @param data
* @returns promise
* IDnull
*/
const upsert = async (data: Partial<DB.User>) => {
if (!data.user_id) return null;
const userInfo = await getOneByUserId(data.user_id);
if (userInfo) return userInfo;
return await create(data);
};
/**
*
*/
const user = {
create,
upsert,

View File

@ -1,25 +1,14 @@
import { RecordModel } from "pocketbase";
import { managePb404 } from "../../utils/pbTools";
import pbClient from "../pbClient";
import { DB } from "../../types/db";
export interface StatisticsPerWeekRecordModel extends RecordModel {
week: string;
total_count: number;
failed_count: number;
success_count: number;
success_rate: number;
duration: number;
}
export interface StatisticsPerProjRecordModel extends RecordModel {
week: string;
name: string;
duration: number;
ref: string;
}
/**
*
* @param week -
* @returns promise
*/
const getFullStatisticsByWeek = (week: string) => {
return managePb404<StatisticsPerWeekRecordModel>(
return managePb404<DB.StatisticsPerWeek>(
async () =>
await pbClient
.collection("statisticsPerWeek")
@ -27,8 +16,13 @@ const getFullStatisticsByWeek = (week: string) => {
);
};
/**
*
* @param week -
* @returns promise
*/
const getProjStatisticsByWeek = (week: string) => {
return managePb404<StatisticsPerProjRecordModel[]>(
return managePb404<DB.StatisticsPerProj[]>(
async () =>
await pbClient
.collection("statisticsPerProj")

View File

@ -1,4 +1,4 @@
import manageRobot from "./controllers/manageRobot";
import { manageCIMonitorReq } from "./routes/ci";
import initSchedule from "./schedule";
import netTool from "./service/netTool";
@ -7,18 +7,19 @@ initSchedule();
const server = Bun.serve({
async fetch(req) {
const url = new URL(req.url);
if (url.pathname === "/ci") {
const chat_id = url.searchParams.get("chat_id");
if (!chat_id) return netTool.badRequest("chat_id is required!");
manageRobot.sendCIReportByChatId(chat_id);
return Response.json({
code: 0,
msg: "success",
data: "reporting...",
});
try {
const url = new URL(req.url);
// 根路由
if (url.pathname === "/") return netTool.ok("hello, glade to see you!");
// CI 监控
if (url.pathname === "/ci") return manageCIMonitorReq(req);
// 其他
return netTool.ok("hello, glade to see you!");
} catch (error: any) {
// 错误处理
console.error("🚀 ~ serve ~ error", error);
return netTool.serverError(error.message || "server error");
}
return netTool.ok();
},
port: 3000,
});

22
routes/ci/index.ts Normal file
View File

@ -0,0 +1,22 @@
import manageRobot from "../../controllers/manageRobot";
import netTool from "../../service/netTool";
/**
* CI监视器的请求
* @param req -
* @returns
*/
export const manageCIMonitorReq = (req: Request) => {
const url = new URL(req.url);
if (url.pathname === "/ci") {
const chat_id = url.searchParams.get("chat_id");
if (!chat_id) return netTool.badRequest("chat_id is required!");
manageRobot.sendCIReportByChatId(chat_id);
return Response.json({
code: 0,
msg: "success",
data: "reporting...",
});
}
return netTool.ok();
};

View File

@ -1,4 +1,6 @@
import service from "../service";
import service from "../../service";
import { BadgeSetParams } from "../../service/gitlab/badge";
import { Gitlab } from "../../types/gitlab";
const projectList = [
// "cloud-ml/cloudml-maas",
@ -23,14 +25,14 @@ const projectList = [
];
const getNewProjectBadge = async (
projectDetail: GitlabProjDetail
): Promise<GitlabBadgeSetParams[]> => {
projectDetail: Gitlab.ProjDetail
): 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: GitlabBadge[] = await service.gitlab.fetchProjectBadges(
const badges: Gitlab.Badge[] = await service.gitlab.badge.get(
projectDetail.id
);
// 对badges进行补全可能只有 name: "sonarqube_coverage" 的情况,把剩下的补全
@ -44,7 +46,7 @@ const getNewProjectBadge = async (
"sonarqube_quality_gate",
];
const diff = [...badgeNameList].filter((x) => !badgeNameSet.has(x));
const newBadges: GitlabBadgeSetParams[] = diff.map((name) => {
const newBadges: BadgeSetParams[] = diff.map((name) => {
const link_url = encodeURI(
`https://sonarqube.mioffice.cn/dashboard?id=${sonarqubeId}`
);
@ -65,16 +67,12 @@ const getNewProjectBadge = async (
return newBadges;
};
const addNewProjectBadge = async (
badgeSetParamsList: GitlabBadgeSetParams[]
) => {
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.addProjectBadge(badgeSetParams)
)
chunk.map((badgeSetParams) => service.gitlab.badge.add(badgeSetParams))
);
console.log(res);
}
@ -84,9 +82,7 @@ const main = async () => {
const errorList: string[] = [];
const badgeAddParamsList = await Promise.all(
projectList.map(async (projectName) => {
const projectDetail = await service.gitlab.fetchProjectDetails(
projectName
);
const projectDetail = await service.gitlab.project.getDetail(projectName);
if (!projectDetail) {
errorList.push(projectName);
return [];

View File

@ -1,4 +1,6 @@
import service from "../service";
import service from "../../service";
import { BadgeSetParams } from "../../service/gitlab/badge";
import { Gitlab } from "../../types/gitlab";
const projectList = [
"miai-fe/fe/ai-admin-fe",
@ -54,16 +56,14 @@ const projectList = [
];
const getProjectId = async (projectName: string) => {
const res = await service.gitlab.fetchProjectDetails(projectName);
const res = await service.gitlab.project.getDetail(projectName);
return res?.id;
};
const getNewProjectBadge = async (
projectId: number
): Promise<GitlabBadgeSetParams[]> => {
const badges: GitlabBadge[] = await service.gitlab.fetchProjectBadges(
projectId
);
): Promise<BadgeSetParams[]> => {
const badges: Gitlab.Badge[] = await service.gitlab.badge.get(projectId);
const replacePath = (value: string) =>
value.replace(
@ -71,7 +71,7 @@ const getNewProjectBadge = async (
"http://scan.sonarqube.xiaomi.srv"
);
return badges.map((badge: GitlabBadge) => ({
return badges.map((badge: Gitlab.Badge) => ({
id: projectId,
badge_id: badge.id,
link_url: replacePath(badge.link_url),
@ -81,16 +81,12 @@ const getNewProjectBadge = async (
}));
};
const setNewProjectBadge = async (
badgeSetParamsList: GitlabBadgeSetParams[]
) => {
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.setProjectBadge(badgeSetParams)
)
chunk.map((badgeSetParams) => service.gitlab.badge.set(badgeSetParams))
);
console.log(res);
}

View File

@ -0,0 +1,83 @@
{
"object_kind": "pipeline",
"object_attributes": {
"id": 8779191,
"ref": "main",
"tag": false,
"sha": "66ac20f0f6a839f29753b17dcdf12deb91148fed",
"before_sha": "39a21d2cb3a0d39d2fb0df81a244ebeae8989ce4",
"source": "push",
"status": "pending",
"detailed_status": "pending",
"stages": [
"print"
],
"created_at": "2024-07-23 09:28:44 +0800",
"finished_at": null,
"duration": null,
"queued_duration": null,
"variables": []
},
"merge_request": null,
"user": {
"id": 30441,
"name": "伍嘉丽",
"username": "wujiali5",
"avatar_url": "https://git.n.xiaomi.com/uploads/-/system/user/avatar/30441/avatar.png",
"email": "[REDACTED]"
},
"project": {
"id": 145623,
"name": "test",
"description": "",
"web_url": "https://git.n.xiaomi.com/cloudml-visuals/fe/test",
"avatar_url": null,
"git_ssh_url": "git@git.n.xiaomi.com:cloudml-visuals/fe/test.git",
"git_http_url": "https://git.n.xiaomi.com/cloudml-visuals/fe/test.git",
"namespace": "fe",
"visibility_level": 0,
"path_with_namespace": "cloudml-visuals/fe/test",
"default_branch": "main",
"ci_config_path": ""
},
"commit": {
"id": "66ac20f0f6a839f29753b17dcdf12deb91148fed",
"message": "Merge branch 'ci' into 'main'\n\nchore: Update .gitlab-ci.yml to print \"Hello, world~\"\n\nSee merge request cloudml-visuals/fe/test!2",
"title": "Merge branch 'ci' into 'main'",
"timestamp": "2024-07-23T09:28:43+08:00",
"url": "https://git.n.xiaomi.com/cloudml-visuals/fe/test/-/commit/66ac20f0f6a839f29753b17dcdf12deb91148fed",
"author": {
"name": "伍嘉丽",
"email": "wujiali5@xiaomi.com"
}
},
"builds": [
{
"id": 25375438,
"stage": "print",
"name": "print",
"status": "pending",
"created_at": "2024-07-23 09:28:44 +0800",
"started_at": null,
"finished_at": null,
"duration": null,
"queued_duration": 3.506639987,
"when": "on_success",
"manual": false,
"allow_failure": false,
"user": {
"id": 30441,
"name": "伍嘉丽",
"username": "wujiali5",
"avatar_url": "https://git.n.xiaomi.com/uploads/-/system/user/avatar/30441/avatar.png",
"email": "[REDACTED]"
},
"runner": null,
"artifacts_file": {
"filename": null,
"size": null
},
"environment": null
}
]
}

View File

@ -0,0 +1,92 @@
{
"object_kind": "pipeline",
"object_attributes": {
"id": 8778929,
"ref": "ci",
"tag": false,
"sha": "fc0fdf57c3b662b296b89dc2a289798d130d1be1",
"before_sha": "0000000000000000000000000000000000000000",
"source": "push",
"status": "running",
"detailed_status": "running",
"stages": [
"print"
],
"created_at": "2024-07-23 08:57:14 +0800",
"finished_at": null,
"duration": null,
"queued_duration": 3,
"variables": []
},
"merge_request": null,
"user": {
"id": 10011,
"name": "赵英博",
"username": "zhaoyingbo",
"avatar_url": "https://git.n.xiaomi.com/uploads/-/system/user/avatar/10011/avatar.png",
"email": "zhaoyingbo@live.cn"
},
"project": {
"id": 145623,
"name": "test",
"description": "",
"web_url": "https://git.n.xiaomi.com/cloudml-visuals/fe/test",
"avatar_url": null,
"git_ssh_url": "git@git.n.xiaomi.com:cloudml-visuals/fe/test.git",
"git_http_url": "https://git.n.xiaomi.com/cloudml-visuals/fe/test.git",
"namespace": "fe",
"visibility_level": 0,
"path_with_namespace": "cloudml-visuals/fe/test",
"default_branch": "main",
"ci_config_path": ""
},
"commit": {
"id": "fc0fdf57c3b662b296b89dc2a289798d130d1be1",
"message": "chore: Add .gitlab-ci.yml for printing \"Hello, world!\"\n",
"title": "chore: Add .gitlab-ci.yml for printing \"Hello, world!\"",
"timestamp": "2024-07-23T08:57:13+08:00",
"url": "https://git.n.xiaomi.com/cloudml-visuals/fe/test/-/commit/fc0fdf57c3b662b296b89dc2a289798d130d1be1",
"author": {
"name": "zhaoyingbo",
"email": "zhaoyingbo@xiaomi.com"
}
},
"builds": [
{
"id": 25374785,
"stage": "print",
"name": "print",
"status": "running",
"created_at": "2024-07-23 08:57:14 +0800",
"started_at": "2024-07-23 08:57:18 +0800",
"finished_at": null,
"duration": 2.977211601,
"queued_duration": 3.550911,
"when": "on_success",
"manual": false,
"allow_failure": false,
"user": {
"id": 10011,
"name": "赵英博",
"username": "zhaoyingbo",
"avatar_url": "https://git.n.xiaomi.com/uploads/-/system/user/avatar/10011/avatar.png",
"email": "zhaoyingbo@live.cn"
},
"runner": {
"id": 9134,
"description": "cloudml-fe-bj",
"runner_type": "group_type",
"active": true,
"is_shared": false,
"tags": [
"cloudml-fe-bj"
]
},
"artifacts_file": {
"filename": null,
"size": null
},
"environment": null
}
]
}

View File

@ -0,0 +1,92 @@
{
"object_kind": "pipeline",
"object_attributes": {
"id": 8778929,
"ref": "ci",
"tag": false,
"sha": "fc0fdf57c3b662b296b89dc2a289798d130d1be1",
"before_sha": "0000000000000000000000000000000000000000",
"source": "push",
"status": "success",
"detailed_status": "passed",
"stages": [
"print"
],
"created_at": "2024-07-23 08:57:14 +0800",
"finished_at": "2024-07-23 08:57:24 +0800",
"duration": 5,
"queued_duration": 3,
"variables": []
},
"merge_request": null,
"user": {
"id": 10011,
"name": "赵英博",
"username": "zhaoyingbo",
"avatar_url": "https://git.n.xiaomi.com/uploads/-/system/user/avatar/10011/avatar.png",
"email": "zhaoyingbo@live.cn"
},
"project": {
"id": 145623,
"name": "test",
"description": "",
"web_url": "https://git.n.xiaomi.com/cloudml-visuals/fe/test",
"avatar_url": null,
"git_ssh_url": "git@git.n.xiaomi.com:cloudml-visuals/fe/test.git",
"git_http_url": "https://git.n.xiaomi.com/cloudml-visuals/fe/test.git",
"namespace": "fe",
"visibility_level": 0,
"path_with_namespace": "cloudml-visuals/fe/test",
"default_branch": "main",
"ci_config_path": ""
},
"commit": {
"id": "fc0fdf57c3b662b296b89dc2a289798d130d1be1",
"message": "chore: Add .gitlab-ci.yml for printing \"Hello, world!\"\n",
"title": "chore: Add .gitlab-ci.yml for printing \"Hello, world!\"",
"timestamp": "2024-07-23T08:57:13+08:00",
"url": "https://git.n.xiaomi.com/cloudml-visuals/fe/test/-/commit/fc0fdf57c3b662b296b89dc2a289798d130d1be1",
"author": {
"name": "zhaoyingbo",
"email": "zhaoyingbo@xiaomi.com"
}
},
"builds": [
{
"id": 25374785,
"stage": "print",
"name": "print",
"status": "success",
"created_at": "2024-07-23 08:57:14 +0800",
"started_at": "2024-07-23 08:57:18 +0800",
"finished_at": "2024-07-23 08:57:24 +0800",
"duration": 5.914004,
"queued_duration": 3.550911,
"when": "on_success",
"manual": false,
"allow_failure": false,
"user": {
"id": 10011,
"name": "赵英博",
"username": "zhaoyingbo",
"avatar_url": "https://git.n.xiaomi.com/uploads/-/system/user/avatar/10011/avatar.png",
"email": "zhaoyingbo@live.cn"
},
"runner": {
"id": 9134,
"description": "cloudml-fe-bj",
"runner_type": "group_type",
"active": true,
"is_shared": false,
"tags": [
"cloudml-fe-bj"
]
},
"artifacts_file": {
"filename": null,
"size": null
},
"environment": null
}
]
}

View File

@ -1,116 +0,0 @@
import netTool from "./netTool";
const AUTH_HEADER = { "PRIVATE-TOKEN": "Zd1UASPcMwVox5tNS6ep" };
const BASE_URL = "https://git.n.xiaomi.com/api/v4";
const gitlabReqWarp = async <T = any>(
func: Function,
default_value: any
): Promise<T> => {
try {
let response = {} as T & GitlabError;
response = (await func()) as T & GitlabError;
if (response.message === "404 Project Not Found") return default_value;
return response;
} catch {
return default_value;
}
};
/**
*
* @param id
* @returns
*/
const fetchProjectDetails = async (id: number | string) => {
if (typeof id === "string") id = encodeURIComponent(id);
const URL = `${BASE_URL}/projects/${id}`;
return gitlabReqWarp<GitlabProjDetail>(
() => netTool.get(URL, {}, AUTH_HEADER),
null
);
};
/**
* 线
* @param project_id
* @param page
* @returns
*/
const fetchPipelines = async (project_id: number, page = 1) => {
const URL = `${BASE_URL}/projects/${project_id}/pipelines`;
const params = { scope: "finished", per_page: 100, page };
return gitlabReqWarp<GitlabPipeline[]>(
() => netTool.get(URL, params, AUTH_HEADER),
[]
);
};
/**
* 线
* @param project_id
* @param pipeline_id
* @param created_at
* @returns
*/
const fetchPipelineDetails = async (
project_id: number,
pipeline_id: number,
created_at: string
) => {
const URL = `${BASE_URL}/projects/${project_id}/pipelines/${pipeline_id}`;
const res = await gitlabReqWarp<GitlabPipelineDetail>(
() => netTool.get(URL, {}, AUTH_HEADER),
null
);
if (res === null) return null;
return { ...res, created_at };
};
/**
*
* @param project_id
*/
const fetchProjectBadges = async (project_id: number) => {
const URL = `${BASE_URL}/projects/${project_id}/badges`;
return gitlabReqWarp<GitlabBadge[]>(
() => netTool.get(URL, {}, AUTH_HEADER),
[]
);
};
/**
*
* @param badge
*/
const setProjectBadge = async (badge: GitlabBadgeSetParams) => {
const URL = `${BASE_URL}/projects/${badge.id}/badges/${badge.badge_id}`;
return gitlabReqWarp<GitlabBadge>(
() => netTool.put(URL, badge, AUTH_HEADER),
null
);
};
/**
*
* @param badge
*/
const addProjectBadge = async (badge: GitlabBadgeSetParams) => {
const URL = `${BASE_URL}/projects/${badge.id}/badges`;
return gitlabReqWarp<GitlabBadge>(
() => netTool.post(URL, badge, {}, AUTH_HEADER),
null
);
};
const gitlab = {
fetchPipelines,
setProjectBadge,
addProjectBadge,
fetchProjectBadges,
fetchProjectDetails,
fetchPipelineDetails,
};
export default gitlab;

60
service/gitlab/badge.ts Normal file
View File

@ -0,0 +1,60 @@
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 project_id ID
* @returns GitLab
*/
const get = async (project_id: number) => {
const URL = `${GITLAB_BASE_URL}/projects/${project_id}/badges`;
return gitlabReqWarp<Gitlab.Badge[]>(
() => netTool.get(URL, {}, GITLAB_AUTH_HEADER),
[]
);
};
/**
* GitLab
* @param badge
* @returns
*/
const set = async (badge: BadgeSetParams) => {
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 badge
* @returns
*/
const add = async (badge: BadgeSetParams) => {
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;

22
service/gitlab/commit.ts Normal file
View File

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

14
service/gitlab/index.ts Normal file
View File

@ -0,0 +1,14 @@
import badge from "./badge";
import commit from "./commit";
import pipeline from "./pipeline";
import project from "./project";
const gitlab = {
project,
badge,
commit,
pipeline,
};
export default gitlab;

View File

@ -0,0 +1,46 @@
import { Gitlab } from "../../types/gitlab";
import netTool from "../netTool";
import { GITLAB_AUTH_HEADER, GITLAB_BASE_URL, gitlabReqWarp } from "./tools";
/**
* GitLab流水线的详细信息
* @param project_id - ID
* @param pipeline_id - 线ID
* @param created_at - 线
* @returns 线promise
*/
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 project_id - ID
* @param page - 1
* @returns 线promise
*/
const getList = async (project_id: number, page = 1) => {
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),
[]
);
};
const pipeline = {
getDetail,
getList,
};
export default pipeline;

24
service/gitlab/project.ts Normal file
View File

@ -0,0 +1,24 @@
import { Gitlab } from "../../types/gitlab";
import netTool from "../netTool";
import { GITLAB_AUTH_HEADER, GITLAB_BASE_URL, gitlabReqWarp } from "./tools";
/**
* GitLab
* @param project_id - ID URL-encoded
* @returns promise
*/
const getDetail = async (project_id: number | string) => {
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;

19
service/gitlab/tools.ts Normal file
View File

@ -0,0 +1,19 @@
import { Gitlab } from "../../types/gitlab";
export const gitlabReqWarp = async <T = any>(
func: Function,
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;
}
};
export const GITLAB_BASE_URL = "https://git.n.xiaomi.com/api/v4";
export const GITLAB_AUTH_HEADER = { "PRIVATE-TOKEN": "Zd1UASPcMwVox5tNS6ep" };

View File

@ -1,9 +1,9 @@
import sendMessage from "./sendMessage";
import gitlab from "./gitlab";
import message from "./message";
const service = {
gitlab,
sendMessage,
message,
};
export default service;

44
service/message/index.ts Normal file
View File

@ -0,0 +1,44 @@
import netTool from "../netTool";
const API_KEY = "1dfz4wlpbbgiky0";
const URL = "https://egg.imoaix.cn/message";
const message = async (body: any) => {
try {
const res = await netTool.post(URL, {
api_key: API_KEY,
...body,
});
return res;
} catch {
return null;
}
};
message.byGroupId = async (group_id: string, content: string) => {
return message({
group_id,
msg_type: "interactive",
content,
});
};
message.byChatId = async (chat_id: string, content: string) => {
return message({
receive_id: chat_id,
receive_id_type: "chat_id",
msg_type: "interactive",
content,
});
};
message.byUserId = async (user_id: string, content: string) => {
return message({
receive_id: user_id,
receive_id_type: "user_id",
msg_type: "interactive",
content,
});
};
export default message;

View File

@ -1,35 +0,0 @@
import netTool from "./netTool";
const API_KEY = "1dfz4wlpbbgiky0";
const sendMessage = async (body: any) => {
const URL = "https://egg.imoaix.cn/message";
try {
const res = await netTool.post(URL, {
api_key: API_KEY,
...body,
});
return res;
} catch {
return null;
}
};
sendMessage.byGroupId = async (group_id: string, content: string) => {
return sendMessage({
group_id,
msg_type: "interactive",
content,
});
};
sendMessage.byChatId = async (chat_id: string, content: string) => {
return sendMessage({
receive_id: chat_id,
receive_id_type: "chat_id",
msg_type: "interactive",
content,
});
};
export default sendMessage;

71
service/typings.d.ts vendored
View File

@ -1,71 +0,0 @@
interface GitlabProjDetail {
id: number;
description: string;
name: string;
path_with_namespace: string;
web_url: string;
avatar_url?: any;
message?: string;
}
interface GitlabPipeline {
id: number;
project_id: number;
sha: string;
ref: string;
status: string;
source: string;
created_at: string;
updated_at: string;
web_url: string;
}
interface GitlabError {
message: string;
}
interface GitlabUser {
id: number;
username: string;
name: string;
state: string;
avatar_url: string;
web_url: string;
}
interface GitlabPipelineDetail {
id: number;
project_id: number;
ref: string;
status: string;
web_url: "https://git.n.xiaomi.com/miai-fe/fe/ai-scene-review-fe/-/pipelines/7646046";
user: GitlabUser;
started_at: string;
finished_at: string;
duration: number;
queued_duration: number;
message?: string;
}
interface GitlabPipelineDetailWithCreateAt extends GitlabPipelineDetail {
created_at: string;
}
interface GitlabBadge {
id: number;
name: string;
link_url: string;
image_url: string;
rendered_link_url: string;
rendered_image_url: string;
kind: "project" | "group";
}
interface GitlabBadgeSetParams {
id: number;
badge_id?: number;
link_url: string;
image_url: string;
rendered_link_url: string;
rendered_image_url: string;
}

52
types/db.ts Normal file
View File

@ -0,0 +1,52 @@
import { RecordModel } from "pocketbase";
export namespace DB {
export interface Pipeline extends RecordModel {
project_id: string;
user_id: string;
pipeline_id: number;
ref: string;
status: string;
web_url: string;
// 2024-03-06 02:53:59.509Z
created_at: string;
started_at: string;
finished_at: string;
duration: number;
queued_duration: number;
}
export interface Project extends RecordModel {
project_id: number;
description: string;
name: string;
path_with_namespace: string;
web_url: string;
avatar_url: string;
has_new_cicd: boolean;
}
export interface User extends RecordModel {
user_id: number;
username: string;
name: string;
avatar_url: string;
web_url: string;
}
export interface StatisticsPerWeek extends RecordModel {
week: string;
total_count: number;
failed_count: number;
success_count: number;
success_rate: number;
duration: number;
}
export interface StatisticsPerProj extends RecordModel {
week: string;
name: string;
duration: number;
ref: string;
}
}

59
types/gitlab.ts Normal file
View File

@ -0,0 +1,59 @@
export namespace Gitlab {
export interface Error {
message: string;
}
export interface User {
id: number;
username: string;
name: string;
state: string;
avatar_url: string;
web_url: string;
}
export interface ProjDetail {
id: number;
description: string;
name: string;
path_with_namespace: string;
web_url: string;
avatar_url?: any;
message?: string;
}
export interface PipelineDetail {
id: number;
project_id: number;
ref: string;
status: string;
web_url: "https://git.n.xiaomi.com/miai-fe/fe/ai-scene-review-fe/-/pipelines/7646046";
user: User;
started_at: string;
finished_at: string;
duration: number;
queued_duration: number;
message?: string;
}
export interface Pipeline {
id: number;
project_id: number;
sha: string;
ref: string;
status: string;
source: string;
created_at: string;
updated_at: string;
web_url: string;
}
export interface Badge {
id: number;
name: string;
link_url: string;
image_url: string;
rendered_link_url: string;
rendered_image_url: string;
kind: "project" | "group";
}
}