chore: 更新lint-staged和commitlint配置
This commit is contained in:
parent
a4555ac862
commit
18a95387ee
1
.husky/commit-msg
Normal file
1
.husky/commit-msg
Normal file
@ -0,0 +1 @@
|
|||||||
|
npx --no -- commitlint --edit $1
|
1
.husky/pre-commit
Normal file
1
.husky/pre-commit
Normal file
@ -0,0 +1 @@
|
|||||||
|
lint-staged
|
1
commitlint.config.js
Normal file
1
commitlint.config.js
Normal file
@ -0,0 +1 @@
|
|||||||
|
export default { extends: ["@commitlint/config-conventional"] }
|
@ -1,63 +1,63 @@
|
|||||||
import moment from "moment";
|
import moment from "moment"
|
||||||
|
|
||||||
import db from "../../db";
|
import db from "../../db"
|
||||||
import service from "../../service";
|
import service from "../../service"
|
||||||
import { DB } from "../../types/db";
|
import { DB } from "../../types/db"
|
||||||
import { Gitlab } from "../../types/gitlab";
|
import { Gitlab } from "../../types/gitlab"
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* 获取全部的pipeline列表
|
* 获取全部的pipeline列表
|
||||||
*/
|
*/
|
||||||
const getFullPipelineList = async (project: DB.Project) => {
|
const getFullPipelineList = async (project: DB.Project) => {
|
||||||
// 先获取最新的pipelineID
|
// 先获取最新的pipelineID
|
||||||
const latestOne = await db.pipeline.getLatestOne(project.id);
|
const latestOne = await db.pipeline.getLatestOne(project.id)
|
||||||
// 获取本次数据获取的截止时间,如果没有,则获取从20240101到现在所有pipeline信息
|
// 获取本次数据获取的截止时间,如果没有,则获取从20240101到现在所有pipeline信息
|
||||||
const latestTime = moment(
|
const latestTime = moment(
|
||||||
latestOne?.created_at || "2024-01-01T00:00:00.000+08:00"
|
latestOne?.created_at || "2024-01-01T00:00:00.000+08:00"
|
||||||
);
|
)
|
||||||
// 获取pipeline列表并保存
|
// 获取pipeline列表并保存
|
||||||
const fullPipelineList: Gitlab.Pipeline[] = [];
|
const fullPipelineList: Gitlab.Pipeline[] = []
|
||||||
let page = 1;
|
let page = 1
|
||||||
let hasBeforeLatestTime = false;
|
let hasBeforeLatestTime = false
|
||||||
while (!hasBeforeLatestTime) {
|
while (!hasBeforeLatestTime) {
|
||||||
const pipelines = await service.gitlab.pipeline.getList(
|
const pipelines = await service.gitlab.pipeline.getList(
|
||||||
project.project_id,
|
project.project_id,
|
||||||
page++
|
page++
|
||||||
);
|
)
|
||||||
// 如果当前页没有数据,则直接跳出
|
// 如果当前页没有数据,则直接跳出
|
||||||
if (pipelines.length === 0) break;
|
if (pipelines.length === 0) break
|
||||||
pipelines.forEach((pipeline) => {
|
pipelines.forEach((pipeline) => {
|
||||||
// 如果已经有了比最新的pipeline还要早的pipeline,则跳出
|
// 如果已经有了比最新的pipeline还要早的pipeline,则跳出
|
||||||
if (hasBeforeLatestTime) return;
|
if (hasBeforeLatestTime) return
|
||||||
if (moment(pipeline.created_at).isSameOrBefore(latestTime)) {
|
if (moment(pipeline.created_at).isSameOrBefore(latestTime)) {
|
||||||
hasBeforeLatestTime = true;
|
hasBeforeLatestTime = true
|
||||||
} else {
|
} else {
|
||||||
fullPipelineList.push(pipeline);
|
fullPipelineList.push(pipeline)
|
||||||
}
|
}
|
||||||
});
|
})
|
||||||
}
|
}
|
||||||
const fullPipelineDetailList = await Promise.all(
|
const fullPipelineDetailList = await Promise.all(
|
||||||
fullPipelineList.map(({ project_id, id, created_at }) =>
|
fullPipelineList.map(({ project_id, id, created_at }) =>
|
||||||
service.gitlab.pipeline.getDetail(project_id, id, created_at)
|
service.gitlab.pipeline.getDetail(project_id, id, created_at)
|
||||||
)
|
)
|
||||||
);
|
)
|
||||||
return fullPipelineDetailList.filter((v) => v) as (Gitlab.PipelineDetail & {
|
return fullPipelineDetailList.filter((v) => v) as (Gitlab.PipelineDetail & {
|
||||||
created_at: string;
|
created_at: string
|
||||||
})[];
|
})[]
|
||||||
};
|
}
|
||||||
|
|
||||||
const insertFullPipelineList = async (
|
const insertFullPipelineList = async (
|
||||||
fullPipelineList: (Gitlab.PipelineDetail & { created_at: string })[][],
|
fullPipelineList: (Gitlab.PipelineDetail & { created_at: string })[][],
|
||||||
fullUserMap: Record<string, string>,
|
fullUserMap: Record<string, string>,
|
||||||
fullProjectMap: Record<string, string>
|
fullProjectMap: Record<string, string>
|
||||||
) => {
|
) => {
|
||||||
const dbPipelineList: Partial<DB.Pipeline> = [];
|
const dbPipelineList: Partial<DB.Pipeline> = []
|
||||||
|
|
||||||
fullPipelineList.forEach((pipelineList) => {
|
fullPipelineList.forEach((pipelineList) => {
|
||||||
pipelineList.forEach((pipeline) => {
|
pipelineList.forEach((pipeline) => {
|
||||||
// 如果没有时间信息,则跳过
|
// 如果没有时间信息,则跳过
|
||||||
if (!pipeline.created_at || !pipeline.started_at || !pipeline.finished_at)
|
if (!pipeline.created_at || !pipeline.started_at || !pipeline.finished_at)
|
||||||
return;
|
return
|
||||||
dbPipelineList.push({
|
dbPipelineList.push({
|
||||||
project_id: fullProjectMap[pipeline.project_id],
|
project_id: fullProjectMap[pipeline.project_id],
|
||||||
user_id: fullUserMap[pipeline.user.id],
|
user_id: fullUserMap[pipeline.user.id],
|
||||||
@ -70,17 +70,17 @@ const insertFullPipelineList = async (
|
|||||||
finished_at: pipeline.finished_at,
|
finished_at: pipeline.finished_at,
|
||||||
duration: pipeline.duration,
|
duration: pipeline.duration,
|
||||||
queued_duration: pipeline.queued_duration,
|
queued_duration: pipeline.queued_duration,
|
||||||
});
|
})
|
||||||
});
|
})
|
||||||
});
|
})
|
||||||
await Promise.all(
|
await Promise.all(
|
||||||
dbPipelineList.map((v: Partial<DB.Pipeline>) => db.pipeline.create(v))
|
dbPipelineList.map((v: Partial<DB.Pipeline>) => db.pipeline.create(v))
|
||||||
);
|
)
|
||||||
};
|
}
|
||||||
|
|
||||||
const managePipeline = {
|
const managePipeline = {
|
||||||
getFullPipelineList,
|
getFullPipelineList,
|
||||||
insertFullPipelineList,
|
insertFullPipelineList,
|
||||||
};
|
}
|
||||||
|
|
||||||
export default managePipeline;
|
export default managePipeline
|
||||||
|
@ -1,14 +1,14 @@
|
|||||||
import db from "../../db";
|
import db from "../../db"
|
||||||
import service from "../../service";
|
import service from "../../service"
|
||||||
import { DB } from "../../types/db";
|
import { DB } from "../../types/db"
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* 填充项目信息
|
* 填充项目信息
|
||||||
*/
|
*/
|
||||||
const fillProj = async (project: DB.Project) => {
|
const fillProj = async (project: DB.Project) => {
|
||||||
const projDetail = await service.gitlab.project.getDetail(project.project_id);
|
const projDetail = await service.gitlab.project.getDetail(project.project_id)
|
||||||
if (!projDetail) {
|
if (!projDetail) {
|
||||||
return project;
|
return project
|
||||||
}
|
}
|
||||||
const useFulParams: Partial<DB.Project> = {
|
const useFulParams: Partial<DB.Project> = {
|
||||||
...project,
|
...project,
|
||||||
@ -17,38 +17,38 @@ const fillProj = async (project: DB.Project) => {
|
|||||||
name: projDetail.name,
|
name: projDetail.name,
|
||||||
path_with_namespace: projDetail.path_with_namespace,
|
path_with_namespace: projDetail.path_with_namespace,
|
||||||
web_url: projDetail.web_url,
|
web_url: projDetail.web_url,
|
||||||
};
|
}
|
||||||
return await db.project.update(project.id, useFulParams);
|
return await db.project.update(project.id, useFulParams)
|
||||||
};
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* 获取到当前所有的项目列表
|
* 获取到当前所有的项目列表
|
||||||
* 并把信息不全的项目送给fillProj填充内容
|
* 并把信息不全的项目送给fillProj填充内容
|
||||||
*/
|
*/
|
||||||
const getFullProjList = async () => {
|
const getFullProjList = async () => {
|
||||||
const fullList = await db.project.getFullList();
|
const fullList = await db.project.getFullList()
|
||||||
// 把信息不全的项目送过去填充
|
// 把信息不全的项目送过去填充
|
||||||
const filledProjList = await Promise.all(
|
const filledProjList = await Promise.all(
|
||||||
fullList.filter((v) => !v.name).map((item) => fillProj(item))
|
fullList.filter((v) => !v.name).map((item) => fillProj(item))
|
||||||
);
|
)
|
||||||
// 合并成完整数据
|
// 合并成完整数据
|
||||||
const filledFullProjList = fullList
|
const filledFullProjList = fullList
|
||||||
.filter((v) => v.name)
|
.filter((v) => v.name)
|
||||||
.concat(filledProjList);
|
.concat(filledProjList)
|
||||||
return filledFullProjList;
|
return filledFullProjList
|
||||||
};
|
}
|
||||||
|
|
||||||
const getFullProjectMap = (fullProjList: DB.Project[]) => {
|
const getFullProjectMap = (fullProjList: DB.Project[]) => {
|
||||||
const fullProjectMap: Record<string, string> = {};
|
const fullProjectMap: Record<string, string> = {}
|
||||||
fullProjList.forEach((item) => {
|
fullProjList.forEach((item) => {
|
||||||
fullProjectMap[item.project_id] = item.id;
|
fullProjectMap[item.project_id] = item.id
|
||||||
});
|
})
|
||||||
return fullProjectMap;
|
return fullProjectMap
|
||||||
};
|
}
|
||||||
|
|
||||||
const manageProject = {
|
const manageProject = {
|
||||||
getFullProjList,
|
getFullProjList,
|
||||||
getFullProjectMap,
|
getFullProjectMap,
|
||||||
};
|
}
|
||||||
|
|
||||||
export default manageProject;
|
export default manageProject
|
||||||
|
@ -1,37 +1,34 @@
|
|||||||
import db from "../../db";
|
import db from "../../db"
|
||||||
import service from "../../service";
|
import service from "../../service"
|
||||||
import { calculateWeeklyRate } from "../../utils/robotTools";
|
import { calculateWeeklyRate } from "../../utils/robotTools"
|
||||||
import {
|
import { getPrevWeekWithYear, getWeekTimeWithYear } from "../../utils/timeTools"
|
||||||
getPrevWeekWithYear,
|
|
||||||
getWeekTimeWithYear,
|
|
||||||
} from "../../utils/timeTools";
|
|
||||||
|
|
||||||
const getNewCicdStatus = async () => {
|
const getNewCicdStatus = async () => {
|
||||||
const fullProjList = await db.project.getFullList();
|
const fullProjList = await db.project.getFullList()
|
||||||
const has_new_cicd_count = String(
|
const has_new_cicd_count = String(
|
||||||
fullProjList.filter((item) => {
|
fullProjList.filter((item) => {
|
||||||
return item.has_new_cicd === true;
|
return item.has_new_cicd === true
|
||||||
}).length
|
}).length
|
||||||
);
|
)
|
||||||
const without_new_cicd_count = String(
|
const without_new_cicd_count = String(
|
||||||
fullProjList.filter((item) => {
|
fullProjList.filter((item) => {
|
||||||
return item.has_new_cicd === false;
|
return item.has_new_cicd === false
|
||||||
}).length
|
}).length
|
||||||
);
|
)
|
||||||
return {
|
return {
|
||||||
has_new_cicd_count,
|
has_new_cicd_count,
|
||||||
without_new_cicd_count,
|
without_new_cicd_count,
|
||||||
};
|
}
|
||||||
};
|
}
|
||||||
|
|
||||||
const getStatisticsInfo = async () => {
|
const getStatisticsInfo = async () => {
|
||||||
const curWeekInfo = await db.view.getFullStatisticsByWeek(
|
const curWeekInfo = await db.view.getFullStatisticsByWeek(
|
||||||
getWeekTimeWithYear()
|
getWeekTimeWithYear()
|
||||||
);
|
)
|
||||||
const prevWeekInfo = await db.view.getFullStatisticsByWeek(
|
const prevWeekInfo = await db.view.getFullStatisticsByWeek(
|
||||||
getPrevWeekWithYear()
|
getPrevWeekWithYear()
|
||||||
);
|
)
|
||||||
if (!curWeekInfo || !prevWeekInfo) return {};
|
if (!curWeekInfo || !prevWeekInfo) return {}
|
||||||
return {
|
return {
|
||||||
total_count: String(curWeekInfo?.total_count ?? 0),
|
total_count: String(curWeekInfo?.total_count ?? 0),
|
||||||
weekly_count_rate: calculateWeeklyRate(
|
weekly_count_rate: calculateWeeklyRate(
|
||||||
@ -48,37 +45,37 @@ const getStatisticsInfo = async () => {
|
|||||||
curWeekInfo?.success_rate,
|
curWeekInfo?.success_rate,
|
||||||
prevWeekInfo?.success_rate
|
prevWeekInfo?.success_rate
|
||||||
).text,
|
).text,
|
||||||
};
|
}
|
||||||
};
|
}
|
||||||
|
|
||||||
const getProjDiffInfo = async () => {
|
const getProjDiffInfo = async () => {
|
||||||
const curWeekInfo =
|
const curWeekInfo =
|
||||||
(await db.view.getProjStatisticsByWeek(getWeekTimeWithYear())) || [];
|
(await db.view.getProjStatisticsByWeek(getWeekTimeWithYear())) || []
|
||||||
const prevWeekInfo =
|
const prevWeekInfo =
|
||||||
(await db.view.getProjStatisticsByWeek(getPrevWeekWithYear())) || [];
|
(await db.view.getProjStatisticsByWeek(getPrevWeekWithYear())) || []
|
||||||
|
|
||||||
const group: {
|
const group: {
|
||||||
project_name: string;
|
project_name: string
|
||||||
project_ref: string;
|
project_ref: string
|
||||||
project_duration: string;
|
project_duration: string
|
||||||
project_duration_rate: string;
|
project_duration_rate: string
|
||||||
percentage: string;
|
percentage: string
|
||||||
}[] = [];
|
}[] = []
|
||||||
|
|
||||||
curWeekInfo.forEach((curWeekProjInfo) => {
|
curWeekInfo.forEach((curWeekProjInfo) => {
|
||||||
const prevWeekProjInfo = prevWeekInfo.find(
|
const prevWeekProjInfo = prevWeekInfo.find(
|
||||||
(info) =>
|
(info) =>
|
||||||
info.ref === curWeekProjInfo.ref && info.name === curWeekProjInfo.name
|
info.ref === curWeekProjInfo.ref && info.name === curWeekProjInfo.name
|
||||||
);
|
)
|
||||||
if (!prevWeekProjInfo) return;
|
if (!prevWeekProjInfo) return
|
||||||
|
|
||||||
const { text: project_duration_rate, percentage } = calculateWeeklyRate(
|
const { text: project_duration_rate, percentage } = calculateWeeklyRate(
|
||||||
curWeekProjInfo.duration,
|
curWeekProjInfo.duration,
|
||||||
prevWeekProjInfo.duration,
|
prevWeekProjInfo.duration,
|
||||||
false
|
false
|
||||||
);
|
)
|
||||||
|
|
||||||
if (percentage === "0") return;
|
if (percentage === "0") return
|
||||||
|
|
||||||
group.push({
|
group.push({
|
||||||
project_name: curWeekProjInfo.name,
|
project_name: curWeekProjInfo.name,
|
||||||
@ -86,15 +83,15 @@ const getProjDiffInfo = async () => {
|
|||||||
project_duration: String(curWeekProjInfo.duration),
|
project_duration: String(curWeekProjInfo.duration),
|
||||||
project_duration_rate,
|
project_duration_rate,
|
||||||
percentage,
|
percentage,
|
||||||
});
|
})
|
||||||
});
|
})
|
||||||
|
|
||||||
// 排序
|
// 排序
|
||||||
group.sort((a, b) => Number(b.percentage) - Number(a.percentage));
|
group.sort((a, b) => Number(b.percentage) - Number(a.percentage))
|
||||||
|
|
||||||
// 取前五个
|
// 取前五个
|
||||||
return group.slice(0, 5);
|
return group.slice(0, 5)
|
||||||
};
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* 获取机器人消息
|
* 获取机器人消息
|
||||||
@ -114,26 +111,26 @@ const getRobotMsg = async () =>
|
|||||||
group_table: await getProjDiffInfo(),
|
group_table: await getProjDiffInfo(),
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
});
|
})
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* 通过ChatID发送CI报告
|
* 通过ChatID发送CI报告
|
||||||
* @param chat_id
|
* @param chat_id
|
||||||
*/
|
*/
|
||||||
const sendCIReportByChatId = async (chat_id: string) => {
|
const sendCIReportByChatId = async (chat_id: string) => {
|
||||||
await service.message.byChatId(chat_id, await getRobotMsg());
|
await service.message.byChatId(chat_id, await getRobotMsg())
|
||||||
};
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* 通过定时任务发送CI报告
|
* 通过定时任务发送CI报告
|
||||||
* @returns
|
* @returns
|
||||||
*/
|
*/
|
||||||
const sendCIReportByCron = async () =>
|
const sendCIReportByCron = async () =>
|
||||||
await service.message.byGroupId("52usf3w8l6z4vs1", await getRobotMsg());
|
await service.message.byGroupId("52usf3w8l6z4vs1", await getRobotMsg())
|
||||||
|
|
||||||
const manageRobot = {
|
const manageRobot = {
|
||||||
sendCIReportByChatId,
|
sendCIReportByChatId,
|
||||||
sendCIReportByCron,
|
sendCIReportByCron,
|
||||||
};
|
}
|
||||||
|
|
||||||
export default manageRobot;
|
export default manageRobot
|
||||||
|
@ -1,15 +1,15 @@
|
|||||||
import db from "../../db";
|
import db from "../../db"
|
||||||
import { Gitlab } from "../../types/gitlab";
|
import { Gitlab } from "../../types/gitlab"
|
||||||
|
|
||||||
const getFullUserMap = async (fullPipelineList: Gitlab.PipelineDetail[][]) => {
|
const getFullUserMap = async (fullPipelineList: Gitlab.PipelineDetail[][]) => {
|
||||||
const userList: Gitlab.User[] = [];
|
const userList: Gitlab.User[] = []
|
||||||
fullPipelineList.forEach((fullPipeline) => {
|
fullPipelineList.forEach((fullPipeline) => {
|
||||||
fullPipeline.forEach((item) => {
|
fullPipeline.forEach((item) => {
|
||||||
if (item.user && !userList.find((v) => v.id === item.user?.id)) {
|
if (item.user && !userList.find((v) => v.id === item.user?.id)) {
|
||||||
userList.push(item.user);
|
userList.push(item.user)
|
||||||
}
|
}
|
||||||
});
|
})
|
||||||
});
|
})
|
||||||
|
|
||||||
const dbUserInfo = await Promise.all(
|
const dbUserInfo = await Promise.all(
|
||||||
userList
|
userList
|
||||||
@ -23,17 +23,17 @@ const getFullUserMap = async (fullPipelineList: Gitlab.PipelineDetail[][]) => {
|
|||||||
web_url: user.web_url,
|
web_url: user.web_url,
|
||||||
})
|
})
|
||||||
)
|
)
|
||||||
);
|
)
|
||||||
const userMap: Record<string, string> = {};
|
const userMap: Record<string, string> = {}
|
||||||
dbUserInfo.forEach((item) => {
|
dbUserInfo.forEach((item) => {
|
||||||
if (!item) return;
|
if (!item) return
|
||||||
userMap[item.user_id] = item.id;
|
userMap[item.user_id] = item.id
|
||||||
});
|
})
|
||||||
return userMap;
|
return userMap
|
||||||
};
|
}
|
||||||
|
|
||||||
const manageUser = {
|
const manageUser = {
|
||||||
getFullUserMap,
|
getFullUserMap,
|
||||||
};
|
}
|
||||||
|
|
||||||
export default manageUser;
|
export default manageUser
|
||||||
|
12
db/index.ts
12
db/index.ts
@ -1,13 +1,13 @@
|
|||||||
import pipeline from "./pipeline";
|
import pipeline from "./pipeline"
|
||||||
import project from "./project";
|
import project from "./project"
|
||||||
import user from "./user";
|
import user from "./user"
|
||||||
import view from "./view";
|
import view from "./view"
|
||||||
|
|
||||||
const db = {
|
const db = {
|
||||||
project,
|
project,
|
||||||
pipeline,
|
pipeline,
|
||||||
user,
|
user,
|
||||||
view,
|
view,
|
||||||
};
|
}
|
||||||
|
|
||||||
export default db;
|
export default db
|
||||||
|
@ -1,7 +1,7 @@
|
|||||||
import PocketBase from "pocketbase";
|
import PocketBase from "pocketbase"
|
||||||
|
|
||||||
const pbClient = new PocketBase("https://ci-pb.xiaomiwh.cn");
|
const pbClient = new PocketBase("https://ci-pb.xiaomiwh.cn")
|
||||||
|
|
||||||
pbClient.autoCancellation(false);
|
pbClient.autoCancellation(false)
|
||||||
|
|
||||||
export default pbClient;
|
export default pbClient
|
||||||
|
@ -1,6 +1,6 @@
|
|||||||
import { DB } from "../../types/db";
|
import { DB } from "../../types/db"
|
||||||
import { managePb404 } from "../../utils/pbTools";
|
import { managePb404 } from "../../utils/pbTools"
|
||||||
import pbClient from "../pbClient";
|
import pbClient from "../pbClient"
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* 通过其 ID 检索一个管道。
|
* 通过其 ID 检索一个管道。
|
||||||
@ -10,7 +10,7 @@ import pbClient from "../pbClient";
|
|||||||
const getOne = (id: string) =>
|
const getOne = (id: string) =>
|
||||||
managePb404<DB.Pipeline>(
|
managePb404<DB.Pipeline>(
|
||||||
async () => await pbClient.collection("pipeline").getOne(id)
|
async () => await pbClient.collection("pipeline").getOne(id)
|
||||||
);
|
)
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* 检索项目的最新管道。
|
* 检索项目的最新管道。
|
||||||
@ -25,8 +25,8 @@ const getLatestOne = (project_id: string) => {
|
|||||||
.getFirstListItem(`project_id="${project_id}"`, {
|
.getFirstListItem(`project_id="${project_id}"`, {
|
||||||
sort: "-created_at",
|
sort: "-created_at",
|
||||||
})
|
})
|
||||||
);
|
)
|
||||||
};
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* 创建一个新的管道。
|
* 创建一个新的管道。
|
||||||
@ -34,12 +34,12 @@ const getLatestOne = (project_id: string) => {
|
|||||||
* @returns 一个解析为创建的管道对象的 promise。
|
* @returns 一个解析为创建的管道对象的 promise。
|
||||||
*/
|
*/
|
||||||
const create = async (data: Partial<DB.Pipeline>) =>
|
const create = async (data: Partial<DB.Pipeline>) =>
|
||||||
await pbClient.collection("pipeline").create<DB.Pipeline>(data);
|
await pbClient.collection("pipeline").create<DB.Pipeline>(data)
|
||||||
|
|
||||||
const pipeline = {
|
const pipeline = {
|
||||||
create,
|
create,
|
||||||
getOne,
|
getOne,
|
||||||
getLatestOne,
|
getLatestOne,
|
||||||
};
|
}
|
||||||
|
|
||||||
export default pipeline;
|
export default pipeline
|
||||||
|
@ -1,6 +1,6 @@
|
|||||||
import { DB } from "../../types/db";
|
import { DB } from "../../types/db"
|
||||||
import { managePb404 } from "../../utils/pbTools";
|
import { managePb404 } from "../../utils/pbTools"
|
||||||
import pbClient from "../pbClient";
|
import pbClient from "../pbClient"
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* 通过其 ID 检索单个项目。
|
* 通过其 ID 检索单个项目。
|
||||||
@ -10,14 +10,14 @@ import pbClient from "../pbClient";
|
|||||||
const getOne = (id: string) =>
|
const getOne = (id: string) =>
|
||||||
managePb404<DB.Project>(
|
managePb404<DB.Project>(
|
||||||
async () => await pbClient.collection("project").getOne(id)
|
async () => await pbClient.collection("project").getOne(id)
|
||||||
);
|
)
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* 检索项目的完整列表。
|
* 检索项目的完整列表。
|
||||||
* @returns 一个解析为项目对象数组的 promise。
|
* @returns 一个解析为项目对象数组的 promise。
|
||||||
*/
|
*/
|
||||||
const getFullList = async () =>
|
const getFullList = async () =>
|
||||||
await pbClient.collection("project").getFullList<DB.Project>();
|
await pbClient.collection("project").getFullList<DB.Project>()
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* 使用新数据更新项目。
|
* 使用新数据更新项目。
|
||||||
@ -26,7 +26,7 @@ const getFullList = async () =>
|
|||||||
* @returns 一个解析为更新后的项目对象的 promise。
|
* @returns 一个解析为更新后的项目对象的 promise。
|
||||||
*/
|
*/
|
||||||
const update = async (id: string, data: Partial<DB.Project>) =>
|
const update = async (id: string, data: Partial<DB.Project>) =>
|
||||||
await pbClient.collection("project").update<DB.Project>(id, data);
|
await pbClient.collection("project").update<DB.Project>(id, data)
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* 用于管理项目的函数集合。
|
* 用于管理项目的函数集合。
|
||||||
@ -35,6 +35,6 @@ const project = {
|
|||||||
getFullList,
|
getFullList,
|
||||||
getOne,
|
getOne,
|
||||||
update,
|
update,
|
||||||
};
|
}
|
||||||
|
|
||||||
export default project;
|
export default project
|
||||||
|
@ -1,6 +1,6 @@
|
|||||||
import { DB } from "../../types/db";
|
import { DB } from "../../types/db"
|
||||||
import { managePb404 } from "../../utils/pbTools";
|
import { managePb404 } from "../../utils/pbTools"
|
||||||
import pbClient from "../pbClient";
|
import pbClient from "../pbClient"
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* 根据提供的ID从数据库检索单个用户。
|
* 根据提供的ID从数据库检索单个用户。
|
||||||
@ -8,9 +8,7 @@ import pbClient from "../pbClient";
|
|||||||
* @returns 如果找到,返回解析为用户对象的promise;如果未找到,抛出404错误。
|
* @returns 如果找到,返回解析为用户对象的promise;如果未找到,抛出404错误。
|
||||||
*/
|
*/
|
||||||
const getOne = (id: string) =>
|
const getOne = (id: string) =>
|
||||||
managePb404<DB.User>(
|
managePb404<DB.User>(async () => await pbClient.collection("user").getOne(id))
|
||||||
async () => await pbClient.collection("user").getOne(id)
|
|
||||||
);
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* 根据提供的用户ID从数据库检索单个用户。
|
* 根据提供的用户ID从数据库检索单个用户。
|
||||||
@ -25,8 +23,8 @@ const getOneByUserId = (user_id: number) => {
|
|||||||
.getFirstListItem(`user_id="${user_id}"`, {
|
.getFirstListItem(`user_id="${user_id}"`, {
|
||||||
sort: "-created",
|
sort: "-created",
|
||||||
})
|
})
|
||||||
);
|
)
|
||||||
};
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* 在数据库中创建一个新用户。
|
* 在数据库中创建一个新用户。
|
||||||
@ -34,7 +32,7 @@ const getOneByUserId = (user_id: number) => {
|
|||||||
* @returns 返回解析为已创建用户对象的promise。
|
* @returns 返回解析为已创建用户对象的promise。
|
||||||
*/
|
*/
|
||||||
const create = async (data: Partial<DB.User>) =>
|
const create = async (data: Partial<DB.User>) =>
|
||||||
await pbClient.collection("user").create<DB.User>(data);
|
await pbClient.collection("user").create<DB.User>(data)
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* 在数据库中插入或更新一个用户。
|
* 在数据库中插入或更新一个用户。
|
||||||
@ -45,11 +43,11 @@ const create = async (data: Partial<DB.User>) =>
|
|||||||
* 如果数据不包含用户ID,则返回null。
|
* 如果数据不包含用户ID,则返回null。
|
||||||
*/
|
*/
|
||||||
const upsert = async (data: Partial<DB.User>) => {
|
const upsert = async (data: Partial<DB.User>) => {
|
||||||
if (!data.user_id) return null;
|
if (!data.user_id) return null
|
||||||
const userInfo = await getOneByUserId(data.user_id);
|
const userInfo = await getOneByUserId(data.user_id)
|
||||||
if (userInfo) return userInfo;
|
if (userInfo) return userInfo
|
||||||
return await create(data);
|
return await create(data)
|
||||||
};
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* 用户模块提供了与数据库中的用户集合交互的函数。
|
* 用户模块提供了与数据库中的用户集合交互的函数。
|
||||||
@ -59,6 +57,6 @@ const user = {
|
|||||||
upsert,
|
upsert,
|
||||||
getOne,
|
getOne,
|
||||||
getOneByUserId,
|
getOneByUserId,
|
||||||
};
|
}
|
||||||
|
|
||||||
export default user;
|
export default user
|
||||||
|
@ -1,6 +1,6 @@
|
|||||||
import { DB } from "../../types/db";
|
import { DB } from "../../types/db"
|
||||||
import { managePb404 } from "../../utils/pbTools";
|
import { managePb404 } from "../../utils/pbTools"
|
||||||
import pbClient from "../pbClient";
|
import pbClient from "../pbClient"
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* 根据给定的周来检索完整的统计信息。
|
* 根据给定的周来检索完整的统计信息。
|
||||||
@ -13,8 +13,8 @@ const getFullStatisticsByWeek = (week: string) => {
|
|||||||
await pbClient
|
await pbClient
|
||||||
.collection("statisticsPerWeek")
|
.collection("statisticsPerWeek")
|
||||||
.getFirstListItem(`week="${week}"`)
|
.getFirstListItem(`week="${week}"`)
|
||||||
);
|
)
|
||||||
};
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* 根据给定的周来检索项目统计信息。
|
* 根据给定的周来检索项目统计信息。
|
||||||
@ -27,12 +27,12 @@ const getProjStatisticsByWeek = (week: string) => {
|
|||||||
await pbClient
|
await pbClient
|
||||||
.collection("statisticsPerProj")
|
.collection("statisticsPerProj")
|
||||||
.getFullList({ filter: `week="${week}"` })
|
.getFullList({ filter: `week="${week}"` })
|
||||||
);
|
)
|
||||||
};
|
}
|
||||||
|
|
||||||
const view = {
|
const view = {
|
||||||
getFullStatisticsByWeek,
|
getFullStatisticsByWeek,
|
||||||
getProjStatisticsByWeek,
|
getProjStatisticsByWeek,
|
||||||
};
|
}
|
||||||
|
|
||||||
export default view;
|
export default view
|
||||||
|
@ -1,7 +1,7 @@
|
|||||||
import pluginJs from "@eslint/js";
|
import pluginJs from "@eslint/js"
|
||||||
import simpleImportSort from "eslint-plugin-simple-import-sort";
|
import simpleImportSort from "eslint-plugin-simple-import-sort"
|
||||||
import globals from "globals";
|
import globals from "globals"
|
||||||
import tseslint from "typescript-eslint";
|
import tseslint from "typescript-eslint"
|
||||||
|
|
||||||
export default [
|
export default [
|
||||||
{ files: ["**/*.{js,mjs,cjs,ts}"] },
|
{ files: ["**/*.{js,mjs,cjs,ts}"] },
|
||||||
@ -19,4 +19,4 @@ export default [
|
|||||||
"simple-import-sort/exports": "error",
|
"simple-import-sort/exports": "error",
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
];
|
]
|
||||||
|
24
index.ts
24
index.ts
@ -1,27 +1,27 @@
|
|||||||
import { manageCIMonitorReq } from "./routes/ci";
|
import { manageCIMonitorReq } from "./routes/ci"
|
||||||
import initSchedule from "./schedule";
|
import initSchedule from "./schedule"
|
||||||
import netTool from "./service/netTool";
|
import netTool from "./service/netTool"
|
||||||
|
|
||||||
// 启动定时任务
|
// 启动定时任务
|
||||||
initSchedule();
|
initSchedule()
|
||||||
|
|
||||||
const server = Bun.serve({
|
const server = Bun.serve({
|
||||||
async fetch(req) {
|
async fetch(req) {
|
||||||
try {
|
try {
|
||||||
const url = new URL(req.url);
|
const url = new URL(req.url)
|
||||||
// 根路由
|
// 根路由
|
||||||
if (url.pathname === "/") return netTool.ok("hello, glade to see you!");
|
if (url.pathname === "/") return netTool.ok("hello, glade to see you!")
|
||||||
// CI 监控
|
// CI 监控
|
||||||
if (url.pathname === "/ci") return manageCIMonitorReq(req);
|
if (url.pathname === "/ci") return manageCIMonitorReq(req)
|
||||||
// 其他
|
// 其他
|
||||||
return netTool.ok("hello, glade to see you!");
|
return netTool.ok("hello, glade to see you!")
|
||||||
} catch (error: any) {
|
} catch (error: any) {
|
||||||
// 错误处理
|
// 错误处理
|
||||||
console.error("🚀 ~ serve ~ error", error);
|
console.error("🚀 ~ serve ~ error", error)
|
||||||
return netTool.serverError(error.message || "server error");
|
return netTool.serverError(error.message || "server error")
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
port: 3000,
|
port: 3000,
|
||||||
});
|
})
|
||||||
|
|
||||||
console.log(`Listening on ${server.hostname}:${server.port}`);
|
console.log(`Listening on ${server.hostname}:${server.port}`)
|
||||||
|
16
package.json
16
package.json
@ -4,9 +4,20 @@
|
|||||||
"type": "module",
|
"type": "module",
|
||||||
"scripts": {
|
"scripts": {
|
||||||
"start": "bun run index.ts",
|
"start": "bun run index.ts",
|
||||||
"lint": "eslint --fix ."
|
"lint": "eslint --fix .",
|
||||||
|
"prepare": "husky",
|
||||||
|
"prettier": "prettier --write ."
|
||||||
|
},
|
||||||
|
"lint-staged": {
|
||||||
|
"*.{js,jsx,ts,tsx}": [
|
||||||
|
"eslint --fix",
|
||||||
|
"prettier --write",
|
||||||
|
"git add"
|
||||||
|
]
|
||||||
},
|
},
|
||||||
"devDependencies": {
|
"devDependencies": {
|
||||||
|
"@commitlint/cli": "^19.3.0",
|
||||||
|
"@commitlint/config-conventional": "^19.2.2",
|
||||||
"@eslint/js": "^9.7.0",
|
"@eslint/js": "^9.7.0",
|
||||||
"@types/lodash": "^4.14.202",
|
"@types/lodash": "^4.14.202",
|
||||||
"@types/node-schedule": "^2.1.6",
|
"@types/node-schedule": "^2.1.6",
|
||||||
@ -14,6 +25,9 @@
|
|||||||
"eslint": "9.x",
|
"eslint": "9.x",
|
||||||
"eslint-plugin-simple-import-sort": "^12.1.1",
|
"eslint-plugin-simple-import-sort": "^12.1.1",
|
||||||
"globals": "^15.8.0",
|
"globals": "^15.8.0",
|
||||||
|
"husky": "^9.1.1",
|
||||||
|
"lint-staged": "^15.2.7",
|
||||||
|
"prettier": "^3.3.3",
|
||||||
"typescript-eslint": "^7.17.0"
|
"typescript-eslint": "^7.17.0"
|
||||||
},
|
},
|
||||||
"peerDependencies": {
|
"peerDependencies": {
|
||||||
|
8
prettier.config.js
Normal file
8
prettier.config.js
Normal file
@ -0,0 +1,8 @@
|
|||||||
|
const config = {
|
||||||
|
trailingComma: "es5",
|
||||||
|
tabWidth: 2,
|
||||||
|
semi: false,
|
||||||
|
singleQuote: false,
|
||||||
|
}
|
||||||
|
|
||||||
|
export default config
|
@ -1,5 +1,5 @@
|
|||||||
import manageRobot from "../../controllers/manageRobot";
|
import manageRobot from "../../controllers/manageRobot"
|
||||||
import netTool from "../../service/netTool";
|
import netTool from "../../service/netTool"
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* 处理管理CI监视器的请求。
|
* 处理管理CI监视器的请求。
|
||||||
@ -7,16 +7,16 @@ import netTool from "../../service/netTool";
|
|||||||
* @returns 响应对象。
|
* @returns 响应对象。
|
||||||
*/
|
*/
|
||||||
export const manageCIMonitorReq = (req: Request) => {
|
export const manageCIMonitorReq = (req: Request) => {
|
||||||
const url = new URL(req.url);
|
const url = new URL(req.url)
|
||||||
if (url.pathname === "/ci") {
|
if (url.pathname === "/ci") {
|
||||||
const chat_id = url.searchParams.get("chat_id");
|
const chat_id = url.searchParams.get("chat_id")
|
||||||
if (!chat_id) return netTool.badRequest("chat_id is required!");
|
if (!chat_id) return netTool.badRequest("chat_id is required!")
|
||||||
manageRobot.sendCIReportByChatId(chat_id);
|
manageRobot.sendCIReportByChatId(chat_id)
|
||||||
return Response.json({
|
return Response.json({
|
||||||
code: 0,
|
code: 0,
|
||||||
msg: "success",
|
msg: "success",
|
||||||
data: "reporting...",
|
data: "reporting...",
|
||||||
});
|
})
|
||||||
}
|
}
|
||||||
return netTool.ok();
|
return netTool.ok()
|
||||||
};
|
}
|
||||||
|
@ -1,15 +1,15 @@
|
|||||||
import { scheduleJob } from "node-schedule";
|
import { scheduleJob } from "node-schedule"
|
||||||
|
|
||||||
import manageRobot from "../controllers/manageRobot";
|
import manageRobot from "../controllers/manageRobot"
|
||||||
import syncPipLine from "./syncPipLine";
|
import syncPipLine from "./syncPipLine"
|
||||||
|
|
||||||
const initSchedule = async () => {
|
const initSchedule = async () => {
|
||||||
// 每周五早上10点发送CI报告
|
// 每周五早上10点发送CI报告
|
||||||
scheduleJob("0 10 * * 5", manageRobot.sendCIReportByCron);
|
scheduleJob("0 10 * * 5", manageRobot.sendCIReportByCron)
|
||||||
// 每15分钟同步一次CI数据
|
// 每15分钟同步一次CI数据
|
||||||
scheduleJob("*/15 * * * *", syncPipLine);
|
scheduleJob("*/15 * * * *", syncPipLine)
|
||||||
// 立即同步一次
|
// 立即同步一次
|
||||||
syncPipLine();
|
syncPipLine()
|
||||||
}
|
}
|
||||||
|
|
||||||
export default initSchedule;
|
export default initSchedule
|
||||||
|
@ -1,19 +1,19 @@
|
|||||||
import managePipeline from "../controllers/managePipeLine";
|
import managePipeline from "../controllers/managePipeLine"
|
||||||
import manageProject from "../controllers/manageProject";
|
import manageProject from "../controllers/manageProject"
|
||||||
import manageUser from "../controllers/manageUser";
|
import manageUser from "../controllers/manageUser"
|
||||||
|
|
||||||
const syncPipLine = async () => {
|
const syncPipLine = async () => {
|
||||||
const fullProjList = await manageProject.getFullProjList();
|
const fullProjList = await manageProject.getFullProjList()
|
||||||
const fullPipelineList = await Promise.all(
|
const fullPipelineList = await Promise.all(
|
||||||
fullProjList.map((v) => managePipeline.getFullPipelineList(v))
|
fullProjList.map((v) => managePipeline.getFullPipelineList(v))
|
||||||
);
|
)
|
||||||
const fullUserMap = await manageUser.getFullUserMap(fullPipelineList);
|
const fullUserMap = await manageUser.getFullUserMap(fullPipelineList)
|
||||||
const fullProjectMap = await manageProject.getFullProjectMap(fullProjList);
|
const fullProjectMap = await manageProject.getFullProjectMap(fullProjList)
|
||||||
await managePipeline.insertFullPipelineList(
|
await managePipeline.insertFullPipelineList(
|
||||||
fullPipelineList,
|
fullPipelineList,
|
||||||
fullUserMap,
|
fullUserMap,
|
||||||
fullProjectMap
|
fullProjectMap
|
||||||
);
|
)
|
||||||
};
|
}
|
||||||
|
|
||||||
export default syncPipLine;
|
export default syncPipLine
|
||||||
|
@ -1,6 +1,6 @@
|
|||||||
import service from "../../service";
|
import service from "../../service"
|
||||||
import { BadgeSetParams } from "../../service/gitlab/badge";
|
import { BadgeSetParams } from "../../service/gitlab/badge"
|
||||||
import { Gitlab } from "../../types/gitlab";
|
import { Gitlab } from "../../types/gitlab"
|
||||||
|
|
||||||
const projectList = [
|
const projectList = [
|
||||||
// "cloud-ml/cloudml-maas",
|
// "cloud-ml/cloudml-maas",
|
||||||
@ -22,39 +22,39 @@ const projectList = [
|
|||||||
// "cloud-ml/knowledge-base",
|
// "cloud-ml/knowledge-base",
|
||||||
// "cloud-ml/ai-proxy",
|
// "cloud-ml/ai-proxy",
|
||||||
"cloud-ml/lark_auth",
|
"cloud-ml/lark_auth",
|
||||||
];
|
]
|
||||||
|
|
||||||
const getNewProjectBadge = async (
|
const getNewProjectBadge = async (
|
||||||
projectDetail: Gitlab.ProjDetail
|
projectDetail: Gitlab.ProjDetail
|
||||||
): Promise<BadgeSetParams[]> => {
|
): Promise<BadgeSetParams[]> => {
|
||||||
// 项目路径 cloud-ml/cloudml-dev
|
// 项目路径 cloud-ml/cloudml-dev
|
||||||
const projectPath = projectDetail.path_with_namespace;
|
const projectPath = projectDetail.path_with_namespace
|
||||||
// 根据项目路径获取sonarqubeId 类似于 cloud-ml/cloudml-dev -> cloud-ml:cloudml-dev
|
// 根据项目路径获取sonarqubeId 类似于 cloud-ml/cloudml-dev -> cloud-ml:cloudml-dev
|
||||||
const sonarqubeId = projectPath.replace("/", ":");
|
const sonarqubeId = projectPath.replace("/", ":")
|
||||||
// 获取项目的badges
|
// 获取项目的badges
|
||||||
const badges: Gitlab.Badge[] = await service.gitlab.badge.get(
|
const badges: Gitlab.Badge[] = await service.gitlab.badge.get(
|
||||||
projectDetail.id
|
projectDetail.id
|
||||||
);
|
)
|
||||||
// 对badges进行补全,可能只有 name: "sonarqube_coverage" 的情况,把剩下的补全
|
// 对badges进行补全,可能只有 name: "sonarqube_coverage" 的情况,把剩下的补全
|
||||||
const badgeNames = badges.map((badge) => badge.name);
|
const badgeNames = badges.map((badge) => badge.name)
|
||||||
const badgeNameSet = new Set(badgeNames);
|
const badgeNameSet = new Set(badgeNames)
|
||||||
const badgeNameList = [
|
const badgeNameList = [
|
||||||
"sonarqube_bugs",
|
"sonarqube_bugs",
|
||||||
"sonarqube_vulnerabilities",
|
"sonarqube_vulnerabilities",
|
||||||
"sonarqube_security_hotspots",
|
"sonarqube_security_hotspots",
|
||||||
"sonarqube_coverage",
|
"sonarqube_coverage",
|
||||||
"sonarqube_quality_gate",
|
"sonarqube_quality_gate",
|
||||||
];
|
]
|
||||||
const diff = [...badgeNameList].filter((x) => !badgeNameSet.has(x));
|
const diff = [...badgeNameList].filter((x) => !badgeNameSet.has(x))
|
||||||
const newBadges: BadgeSetParams[] = diff.map((name) => {
|
const newBadges: BadgeSetParams[] = diff.map((name) => {
|
||||||
const link_url = encodeURI(
|
const link_url = encodeURI(
|
||||||
`https://sonarqube.mioffice.cn/dashboard?id=${sonarqubeId}`
|
`https://sonarqube.mioffice.cn/dashboard?id=${sonarqubeId}`
|
||||||
);
|
)
|
||||||
const metric = name.replace("sonarqube_", "");
|
const metric = name.replace("sonarqube_", "")
|
||||||
const image_url =
|
const image_url =
|
||||||
name !== "sonarqube_quality_gate"
|
name !== "sonarqube_quality_gate"
|
||||||
? `https://sonarqube.mioffice.cn/api/badges/measure?key=${sonarqubeId}&metric=${metric}`
|
? `https://sonarqube.mioffice.cn/api/badges/measure?key=${sonarqubeId}&metric=${metric}`
|
||||||
: `https://sonarqube.mioffice.cn/api/badges/gate?key=${sonarqubeId}`;
|
: `https://sonarqube.mioffice.cn/api/badges/gate?key=${sonarqubeId}`
|
||||||
return {
|
return {
|
||||||
id: projectDetail.id,
|
id: projectDetail.id,
|
||||||
link_url,
|
link_url,
|
||||||
@ -62,39 +62,39 @@ const getNewProjectBadge = async (
|
|||||||
rendered_image_url: image_url,
|
rendered_image_url: image_url,
|
||||||
rendered_link_url: link_url,
|
rendered_link_url: link_url,
|
||||||
name,
|
name,
|
||||||
};
|
}
|
||||||
});
|
})
|
||||||
return newBadges;
|
return newBadges
|
||||||
};
|
}
|
||||||
|
|
||||||
const addNewProjectBadge = async (badgeSetParamsList: BadgeSetParams[]) => {
|
const addNewProjectBadge = async (badgeSetParamsList: BadgeSetParams[]) => {
|
||||||
const chunkSize = 5; // 每次并发请求的数量
|
const chunkSize = 5 // 每次并发请求的数量
|
||||||
for (let i = 0; i < badgeSetParamsList.length; i += chunkSize) {
|
for (let i = 0; i < badgeSetParamsList.length; i += chunkSize) {
|
||||||
const chunk = badgeSetParamsList.slice(i, i + chunkSize);
|
const chunk = badgeSetParamsList.slice(i, i + chunkSize)
|
||||||
const res = await Promise.all(
|
const res = await Promise.all(
|
||||||
chunk.map((badgeSetParams) => service.gitlab.badge.add(badgeSetParams))
|
chunk.map((badgeSetParams) => service.gitlab.badge.add(badgeSetParams))
|
||||||
);
|
)
|
||||||
console.log(res);
|
console.log(res)
|
||||||
}
|
}
|
||||||
};
|
}
|
||||||
|
|
||||||
const main = async () => {
|
const main = async () => {
|
||||||
const errorList: string[] = [];
|
const errorList: string[] = []
|
||||||
const badgeAddParamsList = await Promise.all(
|
const badgeAddParamsList = await Promise.all(
|
||||||
projectList.map(async (projectName) => {
|
projectList.map(async (projectName) => {
|
||||||
const projectDetail = await service.gitlab.project.getDetail(projectName);
|
const projectDetail = await service.gitlab.project.getDetail(projectName)
|
||||||
if (!projectDetail) {
|
if (!projectDetail) {
|
||||||
errorList.push(projectName);
|
errorList.push(projectName)
|
||||||
return [];
|
return []
|
||||||
}
|
}
|
||||||
return await getNewProjectBadge(projectDetail);
|
return await getNewProjectBadge(projectDetail)
|
||||||
})
|
})
|
||||||
);
|
)
|
||||||
|
|
||||||
await addNewProjectBadge(badgeAddParamsList.flat());
|
await addNewProjectBadge(badgeAddParamsList.flat())
|
||||||
};
|
}
|
||||||
|
|
||||||
main();
|
main()
|
||||||
|
|
||||||
// [
|
// [
|
||||||
// {
|
// {
|
||||||
|
@ -1,6 +1,6 @@
|
|||||||
import service from "../../service";
|
import service from "../../service"
|
||||||
import { BadgeSetParams } from "../../service/gitlab/badge";
|
import { BadgeSetParams } from "../../service/gitlab/badge"
|
||||||
import { Gitlab } from "../../types/gitlab";
|
import { Gitlab } from "../../types/gitlab"
|
||||||
|
|
||||||
const projectList = [
|
const projectList = [
|
||||||
"miai-fe/fe/ai-admin-fe",
|
"miai-fe/fe/ai-admin-fe",
|
||||||
@ -53,23 +53,23 @@ const projectList = [
|
|||||||
"miai-fe/fe/ai-schedule-manager-fe",
|
"miai-fe/fe/ai-schedule-manager-fe",
|
||||||
"miai-fe/fe/ai-voiceprint-fe",
|
"miai-fe/fe/ai-voiceprint-fe",
|
||||||
"miai-fe/fe/ai-shortcut-fe",
|
"miai-fe/fe/ai-shortcut-fe",
|
||||||
];
|
]
|
||||||
|
|
||||||
const getProjectId = async (projectName: string) => {
|
const getProjectId = async (projectName: string) => {
|
||||||
const res = await service.gitlab.project.getDetail(projectName);
|
const res = await service.gitlab.project.getDetail(projectName)
|
||||||
return res?.id;
|
return res?.id
|
||||||
};
|
}
|
||||||
|
|
||||||
const getNewProjectBadge = async (
|
const getNewProjectBadge = async (
|
||||||
projectId: number
|
projectId: number
|
||||||
): Promise<BadgeSetParams[]> => {
|
): Promise<BadgeSetParams[]> => {
|
||||||
const badges: Gitlab.Badge[] = await service.gitlab.badge.get(projectId);
|
const badges: Gitlab.Badge[] = await service.gitlab.badge.get(projectId)
|
||||||
|
|
||||||
const replacePath = (value: string) =>
|
const replacePath = (value: string) =>
|
||||||
value.replace(
|
value.replace(
|
||||||
"https://sonarqube.mioffice.cn",
|
"https://sonarqube.mioffice.cn",
|
||||||
"http://scan.sonarqube.xiaomi.srv"
|
"http://scan.sonarqube.xiaomi.srv"
|
||||||
);
|
)
|
||||||
|
|
||||||
return badges.map((badge: Gitlab.Badge) => ({
|
return badges.map((badge: Gitlab.Badge) => ({
|
||||||
id: projectId,
|
id: projectId,
|
||||||
@ -78,36 +78,36 @@ const getNewProjectBadge = async (
|
|||||||
image_url: badge.image_url,
|
image_url: badge.image_url,
|
||||||
rendered_image_url: badge.rendered_image_url,
|
rendered_image_url: badge.rendered_image_url,
|
||||||
rendered_link_url: replacePath(badge.rendered_link_url),
|
rendered_link_url: replacePath(badge.rendered_link_url),
|
||||||
}));
|
}))
|
||||||
};
|
}
|
||||||
|
|
||||||
const setNewProjectBadge = async (badgeSetParamsList: BadgeSetParams[]) => {
|
const setNewProjectBadge = async (badgeSetParamsList: BadgeSetParams[]) => {
|
||||||
const chunkSize = 5; // 每次并发请求的数量
|
const chunkSize = 5 // 每次并发请求的数量
|
||||||
for (let i = 0; i < badgeSetParamsList.length; i += chunkSize) {
|
for (let i = 0; i < badgeSetParamsList.length; i += chunkSize) {
|
||||||
const chunk = badgeSetParamsList.slice(i, i + chunkSize);
|
const chunk = badgeSetParamsList.slice(i, i + chunkSize)
|
||||||
const res = await Promise.all(
|
const res = await Promise.all(
|
||||||
chunk.map((badgeSetParams) => service.gitlab.badge.set(badgeSetParams))
|
chunk.map((badgeSetParams) => service.gitlab.badge.set(badgeSetParams))
|
||||||
);
|
)
|
||||||
console.log(res);
|
console.log(res)
|
||||||
}
|
}
|
||||||
};
|
}
|
||||||
|
|
||||||
const main = async () => {
|
const main = async () => {
|
||||||
const errorList: string[] = [];
|
const errorList: string[] = []
|
||||||
const badgeSetParamsList = await Promise.all(
|
const badgeSetParamsList = await Promise.all(
|
||||||
projectList.map(async (projectName) => {
|
projectList.map(async (projectName) => {
|
||||||
const projectId = await getProjectId(projectName);
|
const projectId = await getProjectId(projectName)
|
||||||
if (!projectId) {
|
if (!projectId) {
|
||||||
errorList.push(projectName);
|
errorList.push(projectName)
|
||||||
return [];
|
return []
|
||||||
}
|
}
|
||||||
return await getNewProjectBadge(projectId);
|
return await getNewProjectBadge(projectId)
|
||||||
})
|
})
|
||||||
);
|
)
|
||||||
|
|
||||||
await setNewProjectBadge(badgeSetParamsList.flat());
|
await setNewProjectBadge(badgeSetParamsList.flat())
|
||||||
|
|
||||||
console.log("errorList", errorList);
|
console.log("errorList", errorList)
|
||||||
};
|
}
|
||||||
|
|
||||||
main();
|
main()
|
||||||
|
@ -9,9 +9,7 @@
|
|||||||
"source": "push",
|
"source": "push",
|
||||||
"status": "pending",
|
"status": "pending",
|
||||||
"detailed_status": "pending",
|
"detailed_status": "pending",
|
||||||
"stages": [
|
"stages": ["print"],
|
||||||
"print"
|
|
||||||
],
|
|
||||||
"created_at": "2024-07-23 09:28:44 +0800",
|
"created_at": "2024-07-23 09:28:44 +0800",
|
||||||
"finished_at": null,
|
"finished_at": null,
|
||||||
"duration": null,
|
"duration": null,
|
||||||
@ -80,4 +78,4 @@
|
|||||||
"environment": null
|
"environment": null
|
||||||
}
|
}
|
||||||
]
|
]
|
||||||
}
|
}
|
||||||
|
@ -9,9 +9,7 @@
|
|||||||
"source": "push",
|
"source": "push",
|
||||||
"status": "running",
|
"status": "running",
|
||||||
"detailed_status": "running",
|
"detailed_status": "running",
|
||||||
"stages": [
|
"stages": ["print"],
|
||||||
"print"
|
|
||||||
],
|
|
||||||
"created_at": "2024-07-23 08:57:14 +0800",
|
"created_at": "2024-07-23 08:57:14 +0800",
|
||||||
"finished_at": null,
|
"finished_at": null,
|
||||||
"duration": null,
|
"duration": null,
|
||||||
@ -78,9 +76,7 @@
|
|||||||
"runner_type": "group_type",
|
"runner_type": "group_type",
|
||||||
"active": true,
|
"active": true,
|
||||||
"is_shared": false,
|
"is_shared": false,
|
||||||
"tags": [
|
"tags": ["cloudml-fe-bj"]
|
||||||
"cloudml-fe-bj"
|
|
||||||
]
|
|
||||||
},
|
},
|
||||||
"artifacts_file": {
|
"artifacts_file": {
|
||||||
"filename": null,
|
"filename": null,
|
||||||
@ -89,4 +85,4 @@
|
|||||||
"environment": null
|
"environment": null
|
||||||
}
|
}
|
||||||
]
|
]
|
||||||
}
|
}
|
||||||
|
@ -9,9 +9,7 @@
|
|||||||
"source": "push",
|
"source": "push",
|
||||||
"status": "success",
|
"status": "success",
|
||||||
"detailed_status": "passed",
|
"detailed_status": "passed",
|
||||||
"stages": [
|
"stages": ["print"],
|
||||||
"print"
|
|
||||||
],
|
|
||||||
"created_at": "2024-07-23 08:57:14 +0800",
|
"created_at": "2024-07-23 08:57:14 +0800",
|
||||||
"finished_at": "2024-07-23 08:57:24 +0800",
|
"finished_at": "2024-07-23 08:57:24 +0800",
|
||||||
"duration": 5,
|
"duration": 5,
|
||||||
@ -78,9 +76,7 @@
|
|||||||
"runner_type": "group_type",
|
"runner_type": "group_type",
|
||||||
"active": true,
|
"active": true,
|
||||||
"is_shared": false,
|
"is_shared": false,
|
||||||
"tags": [
|
"tags": ["cloudml-fe-bj"]
|
||||||
"cloudml-fe-bj"
|
|
||||||
]
|
|
||||||
},
|
},
|
||||||
"artifacts_file": {
|
"artifacts_file": {
|
||||||
"filename": null,
|
"filename": null,
|
||||||
@ -89,4 +85,4 @@
|
|||||||
"environment": null
|
"environment": null
|
||||||
}
|
}
|
||||||
]
|
]
|
||||||
}
|
}
|
||||||
|
@ -1,13 +1,13 @@
|
|||||||
import { Gitlab } from "../../types/gitlab";
|
import { Gitlab } from "../../types/gitlab"
|
||||||
import netTool from "../netTool";
|
import netTool from "../netTool"
|
||||||
import { GITLAB_AUTH_HEADER, GITLAB_BASE_URL, gitlabReqWarp } from "./tools";
|
import { GITLAB_AUTH_HEADER, GITLAB_BASE_URL, gitlabReqWarp } from "./tools"
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* 代表设置 GitLab 徽章的参数。
|
* 代表设置 GitLab 徽章的参数。
|
||||||
*/
|
*/
|
||||||
export type BadgeSetParams = Omit<Gitlab.Badge, "kind" | "name"> & {
|
export type BadgeSetParams = Omit<Gitlab.Badge, "kind" | "name"> & {
|
||||||
badge_id?: number;
|
badge_id?: number
|
||||||
};
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* 为特定项目检索 GitLab 徽章。
|
* 为特定项目检索 GitLab 徽章。
|
||||||
@ -15,12 +15,12 @@ export type BadgeSetParams = Omit<Gitlab.Badge, "kind" | "name"> & {
|
|||||||
* @returns 一个承诺,解析为 GitLab 徽章的数组。
|
* @returns 一个承诺,解析为 GitLab 徽章的数组。
|
||||||
*/
|
*/
|
||||||
const get = async (project_id: number) => {
|
const get = async (project_id: number) => {
|
||||||
const URL = `${GITLAB_BASE_URL}/projects/${project_id}/badges`;
|
const URL = `${GITLAB_BASE_URL}/projects/${project_id}/badges`
|
||||||
return gitlabReqWarp<Gitlab.Badge[]>(
|
return gitlabReqWarp<Gitlab.Badge[]>(
|
||||||
() => netTool.get(URL, {}, GITLAB_AUTH_HEADER),
|
() => netTool.get(URL, {}, GITLAB_AUTH_HEADER),
|
||||||
[]
|
[]
|
||||||
);
|
)
|
||||||
};
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* 设置 GitLab 徽章。
|
* 设置 GitLab 徽章。
|
||||||
@ -28,12 +28,12 @@ const get = async (project_id: number) => {
|
|||||||
* @returns 一个承诺,解析为更新后的徽章。
|
* @returns 一个承诺,解析为更新后的徽章。
|
||||||
*/
|
*/
|
||||||
const set = async (badge: BadgeSetParams) => {
|
const set = async (badge: BadgeSetParams) => {
|
||||||
const URL = `${GITLAB_BASE_URL}/projects/${badge.id}/badges/${badge.badge_id}`;
|
const URL = `${GITLAB_BASE_URL}/projects/${badge.id}/badges/${badge.badge_id}`
|
||||||
return gitlabReqWarp<Gitlab.Badge>(
|
return gitlabReqWarp<Gitlab.Badge>(
|
||||||
() => netTool.put(URL, badge, {}, GITLAB_AUTH_HEADER),
|
() => netTool.put(URL, badge, {}, GITLAB_AUTH_HEADER),
|
||||||
null
|
null
|
||||||
);
|
)
|
||||||
};
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* 添加 GitLab 徽章。
|
* 添加 GitLab 徽章。
|
||||||
@ -41,12 +41,12 @@ const set = async (badge: BadgeSetParams) => {
|
|||||||
* @returns 一个承诺,解析为添加的徽章。
|
* @returns 一个承诺,解析为添加的徽章。
|
||||||
*/
|
*/
|
||||||
const add = async (badge: BadgeSetParams) => {
|
const add = async (badge: BadgeSetParams) => {
|
||||||
const URL = `${GITLAB_BASE_URL}/projects/${badge.id}/badges`;
|
const URL = `${GITLAB_BASE_URL}/projects/${badge.id}/badges`
|
||||||
return gitlabReqWarp<Gitlab.Badge>(
|
return gitlabReqWarp<Gitlab.Badge>(
|
||||||
() => netTool.post(URL, badge, {}, GITLAB_AUTH_HEADER),
|
() => netTool.post(URL, badge, {}, GITLAB_AUTH_HEADER),
|
||||||
null
|
null
|
||||||
);
|
)
|
||||||
};
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* 代表一系列与徽章相关的函数。
|
* 代表一系列与徽章相关的函数。
|
||||||
@ -55,6 +55,6 @@ const badge = {
|
|||||||
get,
|
get,
|
||||||
set,
|
set,
|
||||||
add,
|
add,
|
||||||
};
|
}
|
||||||
|
|
||||||
export default badge;
|
export default badge
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
import netTool from "../netTool";
|
import netTool from "../netTool"
|
||||||
import { GITLAB_AUTH_HEADER, GITLAB_BASE_URL, gitlabReqWarp } from "./tools";
|
import { GITLAB_AUTH_HEADER, GITLAB_BASE_URL, gitlabReqWarp } from "./tools"
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* 检索与特定提交关联的合并请求。
|
* 检索与特定提交关联的合并请求。
|
||||||
@ -8,15 +8,15 @@ import { GITLAB_AUTH_HEADER, GITLAB_BASE_URL, gitlabReqWarp } from "./tools";
|
|||||||
* @returns 一个解析为合并请求数组的promise。
|
* @returns 一个解析为合并请求数组的promise。
|
||||||
*/
|
*/
|
||||||
const getMr = async (project_id: number, sha: string) => {
|
const getMr = async (project_id: number, sha: string) => {
|
||||||
const URL = `${GITLAB_BASE_URL}/projects/${project_id}/repository/commits/${sha}/merge_requests`;
|
const URL = `${GITLAB_BASE_URL}/projects/${project_id}/repository/commits/${sha}/merge_requests`
|
||||||
return gitlabReqWarp<any[]>(
|
return gitlabReqWarp<any[]>(
|
||||||
() => netTool.get(URL, {}, GITLAB_AUTH_HEADER),
|
() => netTool.get(URL, {}, GITLAB_AUTH_HEADER),
|
||||||
[]
|
[]
|
||||||
);
|
)
|
||||||
};
|
}
|
||||||
|
|
||||||
const commit = {
|
const commit = {
|
||||||
getMr,
|
getMr,
|
||||||
};
|
}
|
||||||
|
|
||||||
export default commit;
|
export default commit
|
||||||
|
@ -1,14 +1,13 @@
|
|||||||
|
import badge from "./badge"
|
||||||
import badge from "./badge";
|
import commit from "./commit"
|
||||||
import commit from "./commit";
|
import pipeline from "./pipeline"
|
||||||
import pipeline from "./pipeline";
|
import project from "./project"
|
||||||
import project from "./project";
|
|
||||||
|
|
||||||
const gitlab = {
|
const gitlab = {
|
||||||
project,
|
project,
|
||||||
badge,
|
badge,
|
||||||
commit,
|
commit,
|
||||||
pipeline,
|
pipeline,
|
||||||
};
|
}
|
||||||
|
|
||||||
export default gitlab;
|
export default gitlab
|
||||||
|
@ -1,6 +1,6 @@
|
|||||||
import { Gitlab } from "../../types/gitlab";
|
import { Gitlab } from "../../types/gitlab"
|
||||||
import netTool from "../netTool";
|
import netTool from "../netTool"
|
||||||
import { GITLAB_AUTH_HEADER, GITLAB_BASE_URL, gitlabReqWarp } from "./tools";
|
import { GITLAB_AUTH_HEADER, GITLAB_BASE_URL, gitlabReqWarp } from "./tools"
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* 获取特定GitLab流水线的详细信息。
|
* 获取特定GitLab流水线的详细信息。
|
||||||
@ -14,14 +14,14 @@ const getDetail = async (
|
|||||||
pipeline_id: number,
|
pipeline_id: number,
|
||||||
created_at: string
|
created_at: string
|
||||||
) => {
|
) => {
|
||||||
const URL = `${GITLAB_BASE_URL}/projects/${project_id}/pipelines/${pipeline_id}`;
|
const URL = `${GITLAB_BASE_URL}/projects/${project_id}/pipelines/${pipeline_id}`
|
||||||
const res = await gitlabReqWarp<Gitlab.PipelineDetail>(
|
const res = await gitlabReqWarp<Gitlab.PipelineDetail>(
|
||||||
() => netTool.get(URL, {}, GITLAB_AUTH_HEADER),
|
() => netTool.get(URL, {}, GITLAB_AUTH_HEADER),
|
||||||
null
|
null
|
||||||
);
|
)
|
||||||
if (res === null) return null;
|
if (res === null) return null
|
||||||
return { ...res, created_at };
|
return { ...res, created_at }
|
||||||
};
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* 获取特定项目的GitLab流水线列表。
|
* 获取特定项目的GitLab流水线列表。
|
||||||
@ -30,17 +30,17 @@ const getDetail = async (
|
|||||||
* @returns 一个解析为流水线数组的promise。
|
* @returns 一个解析为流水线数组的promise。
|
||||||
*/
|
*/
|
||||||
const getList = async (project_id: number, page = 1) => {
|
const getList = async (project_id: number, page = 1) => {
|
||||||
const URL = `${GITLAB_BASE_URL}/projects/${project_id}/pipelines`;
|
const URL = `${GITLAB_BASE_URL}/projects/${project_id}/pipelines`
|
||||||
const params = { scope: "finished", per_page: 100, page };
|
const params = { scope: "finished", per_page: 100, page }
|
||||||
return gitlabReqWarp<Gitlab.Pipeline[]>(
|
return gitlabReqWarp<Gitlab.Pipeline[]>(
|
||||||
() => netTool.get(URL, params, GITLAB_AUTH_HEADER),
|
() => netTool.get(URL, params, GITLAB_AUTH_HEADER),
|
||||||
[]
|
[]
|
||||||
);
|
)
|
||||||
};
|
}
|
||||||
|
|
||||||
const pipeline = {
|
const pipeline = {
|
||||||
getDetail,
|
getDetail,
|
||||||
getList,
|
getList,
|
||||||
};
|
}
|
||||||
|
|
||||||
export default pipeline;
|
export default pipeline
|
||||||
|
@ -1,6 +1,6 @@
|
|||||||
import { Gitlab } from "../../types/gitlab";
|
import { Gitlab } from "../../types/gitlab"
|
||||||
import netTool from "../netTool";
|
import netTool from "../netTool"
|
||||||
import { GITLAB_AUTH_HEADER, GITLAB_BASE_URL, gitlabReqWarp } from "./tools";
|
import { GITLAB_AUTH_HEADER, GITLAB_BASE_URL, gitlabReqWarp } from "./tools"
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* 获取 GitLab 项目的详细信息。
|
* 获取 GitLab 项目的详细信息。
|
||||||
@ -9,16 +9,16 @@ import { GITLAB_AUTH_HEADER, GITLAB_BASE_URL, gitlabReqWarp } from "./tools";
|
|||||||
*/
|
*/
|
||||||
const getDetail = async (project_id: number | string) => {
|
const getDetail = async (project_id: number | string) => {
|
||||||
if (typeof project_id === "string")
|
if (typeof project_id === "string")
|
||||||
project_id = encodeURIComponent(project_id);
|
project_id = encodeURIComponent(project_id)
|
||||||
const URL = `${GITLAB_BASE_URL}/projects/${project_id}`;
|
const URL = `${GITLAB_BASE_URL}/projects/${project_id}`
|
||||||
return gitlabReqWarp<Gitlab.ProjDetail>(
|
return gitlabReqWarp<Gitlab.ProjDetail>(
|
||||||
() => netTool.get(URL, {}, GITLAB_AUTH_HEADER),
|
() => netTool.get(URL, {}, GITLAB_AUTH_HEADER),
|
||||||
null
|
null
|
||||||
);
|
)
|
||||||
};
|
}
|
||||||
|
|
||||||
const project = {
|
const project = {
|
||||||
getDetail,
|
getDetail,
|
||||||
};
|
}
|
||||||
|
|
||||||
export default project;
|
export default project
|
||||||
|
@ -1,19 +1,19 @@
|
|||||||
import { Gitlab } from "../../types/gitlab";
|
import { Gitlab } from "../../types/gitlab"
|
||||||
|
|
||||||
export const gitlabReqWarp = async <T = any>(
|
export const gitlabReqWarp = async <T = any>(
|
||||||
func: () => Promise<T>,
|
func: () => Promise<T>,
|
||||||
default_value: any
|
default_value: any
|
||||||
): Promise<T> => {
|
): Promise<T> => {
|
||||||
try {
|
try {
|
||||||
let response = {} as T & Gitlab.Error;
|
let response = {} as T & Gitlab.Error
|
||||||
response = (await func()) as T & Gitlab.Error;
|
response = (await func()) as T & Gitlab.Error
|
||||||
if (response.message === "404 Project Not Found") return default_value;
|
if (response.message === "404 Project Not Found") return default_value
|
||||||
return response;
|
return response
|
||||||
} catch {
|
} catch {
|
||||||
return default_value;
|
return default_value
|
||||||
}
|
}
|
||||||
};
|
}
|
||||||
|
|
||||||
export const GITLAB_BASE_URL = "https://git.n.xiaomi.com/api/v4";
|
export const GITLAB_BASE_URL = "https://git.n.xiaomi.com/api/v4"
|
||||||
|
|
||||||
export const GITLAB_AUTH_HEADER = { "PRIVATE-TOKEN": "Zd1UASPcMwVox5tNS6ep" };
|
export const GITLAB_AUTH_HEADER = { "PRIVATE-TOKEN": "Zd1UASPcMwVox5tNS6ep" }
|
||||||
|
@ -1,9 +1,9 @@
|
|||||||
import gitlab from "./gitlab";
|
import gitlab from "./gitlab"
|
||||||
import message from "./message";
|
import message from "./message"
|
||||||
|
|
||||||
const service = {
|
const service = {
|
||||||
gitlab,
|
gitlab,
|
||||||
message,
|
message,
|
||||||
};
|
}
|
||||||
|
|
||||||
export default service;
|
export default service
|
||||||
|
@ -1,27 +1,27 @@
|
|||||||
import netTool from "../netTool";
|
import netTool from "../netTool"
|
||||||
|
|
||||||
const API_KEY = "1dfz4wlpbbgiky0";
|
const API_KEY = "1dfz4wlpbbgiky0"
|
||||||
const URL = "https://egg.imoaix.cn/message";
|
const URL = "https://egg.imoaix.cn/message"
|
||||||
|
|
||||||
const message = async (body: any) => {
|
const message = async (body: any) => {
|
||||||
try {
|
try {
|
||||||
const res = await netTool.post(URL, {
|
const res = await netTool.post(URL, {
|
||||||
api_key: API_KEY,
|
api_key: API_KEY,
|
||||||
...body,
|
...body,
|
||||||
});
|
})
|
||||||
return res;
|
return res
|
||||||
} catch {
|
} catch {
|
||||||
return null;
|
return null
|
||||||
}
|
}
|
||||||
};
|
}
|
||||||
|
|
||||||
message.byGroupId = async (group_id: string, content: string) => {
|
message.byGroupId = async (group_id: string, content: string) => {
|
||||||
return message({
|
return message({
|
||||||
group_id,
|
group_id,
|
||||||
msg_type: "interactive",
|
msg_type: "interactive",
|
||||||
content,
|
content,
|
||||||
});
|
})
|
||||||
};
|
}
|
||||||
|
|
||||||
message.byChatId = async (chat_id: string, content: string) => {
|
message.byChatId = async (chat_id: string, content: string) => {
|
||||||
return message({
|
return message({
|
||||||
@ -29,8 +29,8 @@ message.byChatId = async (chat_id: string, content: string) => {
|
|||||||
receive_id_type: "chat_id",
|
receive_id_type: "chat_id",
|
||||||
msg_type: "interactive",
|
msg_type: "interactive",
|
||||||
content,
|
content,
|
||||||
});
|
})
|
||||||
};
|
}
|
||||||
|
|
||||||
message.byUserId = async (user_id: string, content: string) => {
|
message.byUserId = async (user_id: string, content: string) => {
|
||||||
return message({
|
return message({
|
||||||
@ -38,7 +38,7 @@ message.byUserId = async (user_id: string, content: string) => {
|
|||||||
receive_id_type: "user_id",
|
receive_id_type: "user_id",
|
||||||
msg_type: "interactive",
|
msg_type: "interactive",
|
||||||
content,
|
content,
|
||||||
});
|
})
|
||||||
};
|
}
|
||||||
|
|
||||||
export default message;
|
export default message
|
||||||
|
@ -1,9 +1,9 @@
|
|||||||
interface NetRequestParams {
|
interface NetRequestParams {
|
||||||
url: string;
|
url: string
|
||||||
method: string;
|
method: string
|
||||||
queryParams?: any;
|
queryParams?: any
|
||||||
payload?: any;
|
payload?: any
|
||||||
additionalHeaders?: any;
|
additionalHeaders?: any
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -32,10 +32,10 @@ const logResponse = (
|
|||||||
responseHeaders: response.headers,
|
responseHeaders: response.headers,
|
||||||
requestBody,
|
requestBody,
|
||||||
responseBody,
|
responseBody,
|
||||||
};
|
}
|
||||||
console.log("🚀 ~ responseLog:", JSON.stringify(responseLog, null, 2));
|
console.log("🚀 ~ responseLog:", JSON.stringify(responseLog, null, 2))
|
||||||
return responseLog;
|
return responseLog
|
||||||
};
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* 发送网络请求并返回一个解析为响应数据的Promise。
|
* 发送网络请求并返回一个解析为响应数据的Promise。
|
||||||
@ -55,13 +55,13 @@ const netTool = async <T = any>({
|
|||||||
additionalHeaders,
|
additionalHeaders,
|
||||||
}: NetRequestParams): Promise<T> => {
|
}: NetRequestParams): Promise<T> => {
|
||||||
// 拼接完整的URL
|
// 拼接完整的URL
|
||||||
let fullUrl = url;
|
let fullUrl = url
|
||||||
if (queryParams) {
|
if (queryParams) {
|
||||||
if (typeof queryParams === "string") {
|
if (typeof queryParams === "string") {
|
||||||
fullUrl = `${url}?${queryParams}`;
|
fullUrl = `${url}?${queryParams}`
|
||||||
} else {
|
} else {
|
||||||
const queryString = new URLSearchParams(queryParams).toString();
|
const queryString = new URLSearchParams(queryParams).toString()
|
||||||
if (queryString) fullUrl = `${url}?${queryString}`;
|
if (queryString) fullUrl = `${url}?${queryString}`
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -69,42 +69,42 @@ const netTool = async <T = any>({
|
|||||||
const headers = {
|
const headers = {
|
||||||
"Content-Type": "application/json",
|
"Content-Type": "application/json",
|
||||||
...additionalHeaders,
|
...additionalHeaders,
|
||||||
};
|
}
|
||||||
|
|
||||||
// 发送请求
|
// 发送请求
|
||||||
const res = await fetch(fullUrl, {
|
const res = await fetch(fullUrl, {
|
||||||
method,
|
method,
|
||||||
body: JSON.stringify(payload),
|
body: JSON.stringify(payload),
|
||||||
headers,
|
headers,
|
||||||
});
|
})
|
||||||
// 获取响应数据
|
// 获取响应数据
|
||||||
let resData: any = null;
|
let resData: any = null
|
||||||
let resText: string = "";
|
let resText: string = ""
|
||||||
|
|
||||||
try {
|
try {
|
||||||
resText = await res.text();
|
resText = await res.text()
|
||||||
resData = JSON.parse(resText);
|
resData = JSON.parse(resText)
|
||||||
} catch {
|
} catch {
|
||||||
/* empty */
|
/* empty */
|
||||||
}
|
}
|
||||||
|
|
||||||
// 记录响应
|
// 记录响应
|
||||||
logResponse(res, method, headers, payload, resData || resText);
|
logResponse(res, method, headers, payload, resData || resText)
|
||||||
if (!res.ok) {
|
if (!res.ok) {
|
||||||
if (resData?.msg) {
|
if (resData?.msg) {
|
||||||
throw new Error(resData.msg);
|
throw new Error(resData.msg)
|
||||||
}
|
}
|
||||||
if (resText) {
|
if (resText) {
|
||||||
throw new Error(resText);
|
throw new Error(resText)
|
||||||
}
|
}
|
||||||
throw new Error("网络响应异常");
|
throw new Error("网络响应异常")
|
||||||
}
|
}
|
||||||
// http 错误码正常,但解析异常
|
// http 错误码正常,但解析异常
|
||||||
if (!resData) {
|
if (!resData) {
|
||||||
throw new Error("解析响应数据异常");
|
throw new Error("解析响应数据异常")
|
||||||
}
|
}
|
||||||
return resData as T;
|
return resData as T
|
||||||
};
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* 发送GET请求并返回一个解析为响应数据的Promise。
|
* 发送GET请求并返回一个解析为响应数据的Promise。
|
||||||
@ -118,8 +118,7 @@ netTool.get = <T = any>(
|
|||||||
url: string,
|
url: string,
|
||||||
queryParams?: any,
|
queryParams?: any,
|
||||||
additionalHeaders?: any
|
additionalHeaders?: any
|
||||||
): Promise<T> =>
|
): Promise<T> => netTool({ url, method: "get", queryParams, additionalHeaders })
|
||||||
netTool({ url, method: "get", queryParams, additionalHeaders });
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* 发送POST请求并返回一个解析为响应数据的Promise。
|
* 发送POST请求并返回一个解析为响应数据的Promise。
|
||||||
@ -136,7 +135,7 @@ netTool.post = <T = any>(
|
|||||||
queryParams?: any,
|
queryParams?: any,
|
||||||
additionalHeaders?: any
|
additionalHeaders?: any
|
||||||
): Promise<T> =>
|
): Promise<T> =>
|
||||||
netTool({ url, method: "post", payload, queryParams, additionalHeaders });
|
netTool({ url, method: "post", payload, queryParams, additionalHeaders })
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* 发送PUT请求并返回一个解析为响应数据的Promise。
|
* 发送PUT请求并返回一个解析为响应数据的Promise。
|
||||||
@ -153,7 +152,7 @@ netTool.put = <T = any>(
|
|||||||
queryParams?: any,
|
queryParams?: any,
|
||||||
additionalHeaders?: any
|
additionalHeaders?: any
|
||||||
): Promise<T> =>
|
): Promise<T> =>
|
||||||
netTool({ url, method: "put", payload, queryParams, additionalHeaders });
|
netTool({ url, method: "put", payload, queryParams, additionalHeaders })
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* 发送DELETE请求并返回一个解析为响应数据的Promise。
|
* 发送DELETE请求并返回一个解析为响应数据的Promise。
|
||||||
@ -170,7 +169,7 @@ netTool.del = <T = any>(
|
|||||||
queryParams?: any,
|
queryParams?: any,
|
||||||
additionalHeaders?: any
|
additionalHeaders?: any
|
||||||
): Promise<T> =>
|
): Promise<T> =>
|
||||||
netTool({ url, method: "delete", payload, queryParams, additionalHeaders });
|
netTool({ url, method: "delete", payload, queryParams, additionalHeaders })
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* 发送PATCH请求并返回一个解析为响应数据的Promise。
|
* 发送PATCH请求并返回一个解析为响应数据的Promise。
|
||||||
@ -187,7 +186,7 @@ netTool.patch = <T = any>(
|
|||||||
queryParams?: any,
|
queryParams?: any,
|
||||||
additionalHeaders?: any
|
additionalHeaders?: any
|
||||||
): Promise<T> =>
|
): Promise<T> =>
|
||||||
netTool({ url, method: "patch", payload, queryParams, additionalHeaders });
|
netTool({ url, method: "patch", payload, queryParams, additionalHeaders })
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* 创建一个表示400 Bad Request的响应对象。
|
* 创建一个表示400 Bad Request的响应对象。
|
||||||
@ -197,7 +196,7 @@ netTool.patch = <T = any>(
|
|||||||
* @returns 一个表示400 Bad Request的响应对象。
|
* @returns 一个表示400 Bad Request的响应对象。
|
||||||
*/
|
*/
|
||||||
netTool.badRequest = (msg: string, requestId?: string) =>
|
netTool.badRequest = (msg: string, requestId?: string) =>
|
||||||
Response.json({ code: 400, msg, requestId }, { status: 400 });
|
Response.json({ code: 400, msg, requestId }, { status: 400 })
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* 创建一个表示404 Not Found的响应对象。
|
* 创建一个表示404 Not Found的响应对象。
|
||||||
@ -207,7 +206,7 @@ netTool.badRequest = (msg: string, requestId?: string) =>
|
|||||||
* @returns 一个表示404 Not Found的响应对象。
|
* @returns 一个表示404 Not Found的响应对象。
|
||||||
*/
|
*/
|
||||||
netTool.notFound = (msg: string, requestId?: string) =>
|
netTool.notFound = (msg: string, requestId?: string) =>
|
||||||
Response.json({ code: 404, msg, requestId }, { status: 404 });
|
Response.json({ code: 404, msg, requestId }, { status: 404 })
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* 创建一个表示500 Internal Server Error的响应对象。
|
* 创建一个表示500 Internal Server Error的响应对象。
|
||||||
@ -218,7 +217,7 @@ netTool.notFound = (msg: string, requestId?: string) =>
|
|||||||
* @returns 一个表示500 Internal Server Error的响应对象。
|
* @returns 一个表示500 Internal Server Error的响应对象。
|
||||||
*/
|
*/
|
||||||
netTool.serverError = (msg: string, data?: any, requestId?: string) =>
|
netTool.serverError = (msg: string, data?: any, requestId?: string) =>
|
||||||
Response.json({ code: 500, msg, data, requestId }, { status: 500 });
|
Response.json({ code: 500, msg, data, requestId }, { status: 500 })
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* 创建一个表示200 OK的响应对象。
|
* 创建一个表示200 OK的响应对象。
|
||||||
@ -228,6 +227,6 @@ netTool.serverError = (msg: string, data?: any, requestId?: string) =>
|
|||||||
* @returns 一个表示200 OK的响应对象。
|
* @returns 一个表示200 OK的响应对象。
|
||||||
*/
|
*/
|
||||||
netTool.ok = (data?: any, requestId?: string) =>
|
netTool.ok = (data?: any, requestId?: string) =>
|
||||||
Response.json({ code: 0, msg: "success", data, requestId });
|
Response.json({ code: 0, msg: "success", data, requestId })
|
||||||
|
|
||||||
export default netTool;
|
export default netTool
|
||||||
|
68
types/db.ts
68
types/db.ts
@ -1,52 +1,52 @@
|
|||||||
import { RecordModel } from "pocketbase";
|
import { RecordModel } from "pocketbase"
|
||||||
|
|
||||||
export namespace DB {
|
export namespace DB {
|
||||||
export interface Pipeline extends RecordModel {
|
export interface Pipeline extends RecordModel {
|
||||||
project_id: string;
|
project_id: string
|
||||||
user_id: string;
|
user_id: string
|
||||||
pipeline_id: number;
|
pipeline_id: number
|
||||||
ref: string;
|
ref: string
|
||||||
status: string;
|
status: string
|
||||||
web_url: string;
|
web_url: string
|
||||||
// 2024-03-06 02:53:59.509Z
|
// 2024-03-06 02:53:59.509Z
|
||||||
created_at: string;
|
created_at: string
|
||||||
started_at: string;
|
started_at: string
|
||||||
finished_at: string;
|
finished_at: string
|
||||||
duration: number;
|
duration: number
|
||||||
queued_duration: number;
|
queued_duration: number
|
||||||
}
|
}
|
||||||
|
|
||||||
export interface Project extends RecordModel {
|
export interface Project extends RecordModel {
|
||||||
project_id: number;
|
project_id: number
|
||||||
description: string;
|
description: string
|
||||||
name: string;
|
name: string
|
||||||
path_with_namespace: string;
|
path_with_namespace: string
|
||||||
web_url: string;
|
web_url: string
|
||||||
avatar_url: string;
|
avatar_url: string
|
||||||
has_new_cicd: boolean;
|
has_new_cicd: boolean
|
||||||
}
|
}
|
||||||
|
|
||||||
export interface User extends RecordModel {
|
export interface User extends RecordModel {
|
||||||
user_id: number;
|
user_id: number
|
||||||
username: string;
|
username: string
|
||||||
name: string;
|
name: string
|
||||||
avatar_url: string;
|
avatar_url: string
|
||||||
web_url: string;
|
web_url: string
|
||||||
}
|
}
|
||||||
|
|
||||||
export interface StatisticsPerWeek extends RecordModel {
|
export interface StatisticsPerWeek extends RecordModel {
|
||||||
week: string;
|
week: string
|
||||||
total_count: number;
|
total_count: number
|
||||||
failed_count: number;
|
failed_count: number
|
||||||
success_count: number;
|
success_count: number
|
||||||
success_rate: number;
|
success_rate: number
|
||||||
duration: number;
|
duration: number
|
||||||
}
|
}
|
||||||
|
|
||||||
export interface StatisticsPerProj extends RecordModel {
|
export interface StatisticsPerProj extends RecordModel {
|
||||||
week: string;
|
week: string
|
||||||
name: string;
|
name: string
|
||||||
duration: number;
|
duration: number
|
||||||
ref: string;
|
ref: string
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -1,59 +1,59 @@
|
|||||||
export namespace Gitlab {
|
export namespace Gitlab {
|
||||||
export interface Error {
|
export interface Error {
|
||||||
message: string;
|
message: string
|
||||||
}
|
}
|
||||||
export interface User {
|
export interface User {
|
||||||
id: number;
|
id: number
|
||||||
username: string;
|
username: string
|
||||||
name: string;
|
name: string
|
||||||
state: string;
|
state: string
|
||||||
avatar_url: string;
|
avatar_url: string
|
||||||
web_url: string;
|
web_url: string
|
||||||
}
|
}
|
||||||
|
|
||||||
export interface ProjDetail {
|
export interface ProjDetail {
|
||||||
id: number;
|
id: number
|
||||||
description: string;
|
description: string
|
||||||
name: string;
|
name: string
|
||||||
path_with_namespace: string;
|
path_with_namespace: string
|
||||||
web_url: string;
|
web_url: string
|
||||||
avatar_url?: any;
|
avatar_url?: any
|
||||||
message?: string;
|
message?: string
|
||||||
}
|
}
|
||||||
|
|
||||||
export interface PipelineDetail {
|
export interface PipelineDetail {
|
||||||
id: number;
|
id: number
|
||||||
project_id: number;
|
project_id: number
|
||||||
ref: string;
|
ref: string
|
||||||
status: string;
|
status: string
|
||||||
web_url: "https://git.n.xiaomi.com/miai-fe/fe/ai-scene-review-fe/-/pipelines/7646046";
|
web_url: "https://git.n.xiaomi.com/miai-fe/fe/ai-scene-review-fe/-/pipelines/7646046"
|
||||||
user: User;
|
user: User
|
||||||
started_at: string;
|
started_at: string
|
||||||
finished_at: string;
|
finished_at: string
|
||||||
duration: number;
|
duration: number
|
||||||
queued_duration: number;
|
queued_duration: number
|
||||||
message?: string;
|
message?: string
|
||||||
}
|
}
|
||||||
|
|
||||||
export interface Pipeline {
|
export interface Pipeline {
|
||||||
id: number;
|
id: number
|
||||||
project_id: number;
|
project_id: number
|
||||||
sha: string;
|
sha: string
|
||||||
ref: string;
|
ref: string
|
||||||
status: string;
|
status: string
|
||||||
source: string;
|
source: string
|
||||||
created_at: string;
|
created_at: string
|
||||||
updated_at: string;
|
updated_at: string
|
||||||
web_url: string;
|
web_url: string
|
||||||
}
|
}
|
||||||
|
|
||||||
export interface Badge {
|
export interface Badge {
|
||||||
id: number;
|
id: number
|
||||||
name: string;
|
name: string
|
||||||
link_url: string;
|
link_url: string
|
||||||
image_url: string;
|
image_url: string
|
||||||
rendered_link_url: string;
|
rendered_link_url: string
|
||||||
rendered_image_url: string;
|
rendered_image_url: string
|
||||||
kind: "project" | "group";
|
kind: "project" | "group"
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -2,10 +2,10 @@ export const managePb404 = async <T>(
|
|||||||
dbFunc: () => Promise<T>
|
dbFunc: () => Promise<T>
|
||||||
): Promise<T | null> => {
|
): Promise<T | null> => {
|
||||||
try {
|
try {
|
||||||
return await dbFunc();
|
return await dbFunc()
|
||||||
} catch (err: any) {
|
} catch (err: any) {
|
||||||
if (err?.message === "The requested resource wasn't found.") {
|
if (err?.message === "The requested resource wasn't found.") {
|
||||||
return null;
|
return null
|
||||||
} else throw err;
|
} else throw err
|
||||||
}
|
}
|
||||||
};
|
}
|
||||||
|
@ -5,22 +5,22 @@
|
|||||||
*/
|
*/
|
||||||
export const calculatePercentageChange = (cur: number, prev: number) => {
|
export const calculatePercentageChange = (cur: number, prev: number) => {
|
||||||
// 计算差值
|
// 计算差值
|
||||||
const diff = cur - prev;
|
const diff = cur - prev
|
||||||
|
|
||||||
if (diff === 0)
|
if (diff === 0)
|
||||||
return {
|
return {
|
||||||
diff,
|
diff,
|
||||||
percentage: "0",
|
percentage: "0",
|
||||||
};
|
}
|
||||||
|
|
||||||
// 计算百分比
|
// 计算百分比
|
||||||
const percentage = Math.abs((diff / prev) * 100).toFixed(1);
|
const percentage = Math.abs((diff / prev) * 100).toFixed(1)
|
||||||
|
|
||||||
return {
|
return {
|
||||||
diff,
|
diff,
|
||||||
percentage,
|
percentage,
|
||||||
};
|
}
|
||||||
};
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* 计算周同比
|
* 计算周同比
|
||||||
@ -36,7 +36,7 @@ export const calculateWeeklyRate = (
|
|||||||
const { diff, percentage } = calculatePercentageChange(
|
const { diff, percentage } = calculatePercentageChange(
|
||||||
Number(cur),
|
Number(cur),
|
||||||
Number(prev)
|
Number(prev)
|
||||||
);
|
)
|
||||||
if (diff > 0)
|
if (diff > 0)
|
||||||
return {
|
return {
|
||||||
text: `<font color='red'>${
|
text: `<font color='red'>${
|
||||||
@ -44,7 +44,7 @@ export const calculateWeeklyRate = (
|
|||||||
}↑${percentage}%</font>`,
|
}↑${percentage}%</font>`,
|
||||||
diff,
|
diff,
|
||||||
percentage,
|
percentage,
|
||||||
};
|
}
|
||||||
if (diff < 0)
|
if (diff < 0)
|
||||||
return {
|
return {
|
||||||
text: `<font color='green'>${
|
text: `<font color='green'>${
|
||||||
@ -52,10 +52,10 @@ export const calculateWeeklyRate = (
|
|||||||
}↓${percentage}%</font>`,
|
}↓${percentage}%</font>`,
|
||||||
diff,
|
diff,
|
||||||
percentage,
|
percentage,
|
||||||
};
|
}
|
||||||
return {
|
return {
|
||||||
text: `<font color='gray'>${needCN ? "较上周 " : ""}0%</font>`,
|
text: `<font color='gray'>${needCN ? "较上周 " : ""}0%</font>`,
|
||||||
diff,
|
diff,
|
||||||
percentage,
|
percentage,
|
||||||
};
|
}
|
||||||
};
|
}
|
||||||
|
@ -1,22 +1,22 @@
|
|||||||
import moment from "moment";
|
import moment from "moment"
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* 获取今天是今年的第几周,like 2024-05
|
* 获取今天是今年的第几周,like 2024-05
|
||||||
*/
|
*/
|
||||||
export const getWeekTimeWithYear = () => {
|
export const getWeekTimeWithYear = () => {
|
||||||
return moment().format("YYYY-WW");
|
return moment().format("YYYY-WW")
|
||||||
};
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* 获取上周是今年的第几周,like 2024-04
|
* 获取上周是今年的第几周,like 2024-04
|
||||||
*/
|
*/
|
||||||
export const getPrevWeekWithYear = () => {
|
export const getPrevWeekWithYear = () => {
|
||||||
return moment().subtract(1, "weeks").format("YYYY-WW");
|
return moment().subtract(1, "weeks").format("YYYY-WW")
|
||||||
};
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* 秒转分钟,保留一位小数
|
* 秒转分钟,保留一位小数
|
||||||
*/
|
*/
|
||||||
export const sec2min = (sec: number) => {
|
export const sec2min = (sec: number) => {
|
||||||
return (sec / 60).toFixed(1);
|
return (sec / 60).toFixed(1)
|
||||||
};
|
}
|
||||||
|
Loading…
x
Reference in New Issue
Block a user