This commit is contained in:
zhaoyingbo 2024-05-06 03:28:58 +00:00
commit 260aa8c350
20 changed files with 554 additions and 0 deletions

View File

@ -0,0 +1,32 @@
{
"name": "replace_me",
"image": "micr.cloud.mioffice.cn/zhaoyingbo/dev:bun",
"remoteUser": "bun",
"containerUser": "bun",
"customizations": {
"vscode": {
"settings": {
"files.autoSave": "afterDelay",
"editor.guides.bracketPairs": true,
"editor.defaultFormatter": "esbenp.prettier-vscode",
"editor.formatOnSave": true,
"editor.codeActionsOnSave": {
"source.fixAll.stylelint": true
}
},
"extensions": [
"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"
]
}
}
}

1
.gitattributes vendored Normal file
View File

@ -0,0 +1 @@
*.lockb binary diff=lockb

View File

@ -0,0 +1,57 @@
name: CI Monitor CI/CD
on: [push]
jobs:
build-image:
runs-on: ubuntu-latest
container: catthehacker/ubuntu:act-latest
steps:
- name: Check out repository code
uses: actions/checkout@v3
- name: Set up QEMU
uses: docker/setup-qemu-action@v2
- name: Set up Docker Buildx
uses: docker/setup-buildx-action@v2
- name: Login to Docker Hub
uses: docker/login-action@v2
with:
registry: git.yingbo.im:333
username: ${{ secrets.DOCKER_USERNAME }}
password: ${{ secrets.DOCKER_PASSWORD }}
- name: Build and push
uses: docker/build-push-action@v4
with:
push: true
tags: git.yingbo.im:333/zhaoyingbo/ci_monitor:${{ github.sha }}
deploy:
needs: build-image
runs-on: ubuntu-latest
container: catthehacker/ubuntu:act-latest
steps:
# 检出代码
- name: Check out repository code
uses: actions/checkout@v3
# 使用scp命令将docker-compose.yml文件上传到服务器
- name: Upload docker-compose.yml to server
uses: appleboy/scp-action@master
with:
host: ${{ secrets.SERVER_HOST }}
username: ${{ secrets.SERVER_USERNAME }}
key: ${{ secrets.SERVER_KEY }}
port: ${{ secrets.SERVER_PORT }}
source: docker-compose.yml
target: /home/${{ secrets.SERVER_USERNAME }}/docker/ci_monitor
# 登录服务器执行docker-compose命令
- name: Login to the server and execute docker-compose command
uses: appleboy/ssh-action@master
with:
host: ${{ secrets.SERVER_HOST }}
username: ${{ secrets.SERVER_USERNAME }}
key: ${{ secrets.SERVER_KEY }}
port: ${{ secrets.SERVER_PORT }}
script: |
docker login -u ${{ secrets.DOCKER_USERNAME }} -p ${{ secrets.DOCKER_PASSWORD }} git.yingbo.im:333
cd /home/${{ secrets.SERVER_USERNAME }}/docker/ci_monitor
sed -i "s/sha/${{ github.sha }}/g" docker-compose.yml
docker compose up -d --force-recreate --no-deps ci_monitor

58
.gitignore vendored Normal file
View File

@ -0,0 +1,58 @@
# Logs
logs
*.log
npm-debug.log*
# Runtime data
pids
*.pid
*.seed
# Directory for instrumented libs generated by jscoverage/JSCover
lib-cov
# Coverage directory used by tools like istanbul
coverage
# nyc test coverage
.nyc_output
# Grunt intermediate storage (http://gruntjs.com/creating-plugins#storing-task-files)
.grunt
# node-waf configuration
.lock-wscript
# Compiled binary addons (http://nodejs.org/api/addons.html)
build/Release
# Dependency directories
node_modules
jspm_packages
# Optional npm cache directory
.npm
# Optional REPL history
.node_repl_history
# 0x
profile-*
# mac files
.DS_Store
# vim swap files
*.swp
# webstorm
.idea
# vscode
.vscode
*code-workspace
# clinic
profile*
*clinic*
*flamegraph*

13
Dockerfile Normal file
View File

@ -0,0 +1,13 @@
FROM micr.cloud.mioffice.cn/zhaoyingbo/bun:alpine-cn
WORKDIR /app
COPY package*.json ./
COPY bun.lockb ./
RUN bun install
COPY . .
CMD ["bun", "start"]

BIN
bun.lockb Executable file

Binary file not shown.

View File

@ -0,0 +1,7 @@
const templateFunc = () => {}
const template = {
templateFunc,
};
export default template;

7
db/index.ts Normal file
View File

@ -0,0 +1,7 @@
import user from "./user";
const db = {
user,
};
export default db;

7
db/pbClient.ts Normal file
View File

@ -0,0 +1,7 @@
import PocketBase from "pocketbase";
const pbClient = new PocketBase("https://ci-pb.xiaomiwh.cn");
pbClient.autoCancellation(false);
export default pbClient;

46
db/user/index.ts Normal file
View File

@ -0,0 +1,46 @@
import { RecordModel } from "pocketbase";
import { managePb404 } from "../../utils/pbTools";
import pbClient from "../pbClient";
export interface UserRecordModel extends RecordModel {
user_id: number;
username: string;
name: string;
avatar_url: string;
web_url: string;
}
const getOne = (id: string) =>
managePb404(
async () => await pbClient.collection("user").getOne(id)
) as Promise<UserRecordModel>;
const getOneByUserId = (user_id: number) => {
return managePb404(
async () =>
await pbClient
.collection("user")
.getFirstListItem(`user_id="${user_id}"`, {
sort: "-created",
})
) as Promise<UserRecordModel>;
};
const create = async (data: Partial<UserRecordModel>) =>
await pbClient.collection("user").create<UserRecordModel>(data);
const upsert = async (data: Partial<UserRecordModel>) => {
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,
getOne,
getOneByUserId,
};
export default user;

9
docker-compose.yml Normal file
View File

@ -0,0 +1,9 @@
version: "3"
services:
replace_me:
image: git.yingbo.im:333/zhaoyingbo/replace_me:sha
container_name: replace_me
restart: always
ports:
- 3001:3000

26
eslint.config.js Normal file
View File

@ -0,0 +1,26 @@
export default {
languageOptions: {
ecmaVersion: 2021,
sourceType: 'module',
},
ignores: ['*.js', '*.cjs', '*.mjs', '/src/Backup/*'],
plugins: ['@typescript-eslint', 'simple-import-sort'],
extends: [
'eslint:recommended',
'plugin:@typescript-eslint/recommended',
'prettier',
],
rules: {
'@typescript-eslint/no-explicit-any': 'off',
'simple-import-sort/imports': 'error',
'simple-import-sort/exports': 'error',
},
overrides: [
{
files: ['**/*.ts', '**/*.tsx'],
languageOptions: {
parser: '@typescript-eslint/parser',
},
},
],
};

14
index.ts Normal file
View File

@ -0,0 +1,14 @@
import { initSchedule } from "./schedule";
initSchedule()
Bun.serve({
async fetch(req) {
const url = new URL(req.url);
// 根路由
if (url.pathname === "/") return new Response("hello, glade to see you!");
if (url.pathname === '/ci') return new Response("OK")
return new Response("OK");
},
port: 3000,
});

25
package.json Normal file
View File

@ -0,0 +1,25 @@
{
"name": "replace_me",
"module": "index.ts",
"type": "module",
"scripts": {
"start": "bun run index.ts"
},
"devDependencies": {
"eslint": "^9.2.0",
"bun-types": "latest",
"@types/lodash": "^4.14.202",
"@types/node-schedule": "^2.1.6",
"eslint-plugin-simple-import-sort": "^12.1.0",
"@typescript-eslint/eslint-plugin": "^7.8.0",
"@typescript-eslint/parser": "^7.8.0"
},
"peerDependencies": {
"typescript": "^5.0.0"
},
"dependencies": {
"lodash": "^4.17.21",
"node-schedule": "^2.1.1",
"pocketbase": "^0.21.1"
}
}

182
readme.md Normal file
View File

@ -0,0 +1,182 @@
# CI 监控
监听新 projId自动补全内容获取从 20240101 到当前的所有流水线信息
监听功能未知原因不好用,先不做了,改手动遍历了
拿到 project_id 后,获取数据表中最新的 pipeline 的 Id然后比对接口中的 ID 进行填充
如果没有 pipeline 的 id直接从接口中获取 20240101 到当前的流水线信息
先从数据中获取用户信息填充,随后在根据填充完的用户信息获取全部的 userid 的列表,再写 pipeline 表
# 图表库(不用了)
https://g2plot.antv.antgroup.com/examples
# 数据信息
project 信息
```js
{
id: 'aaa',
project_id: 131366,
description: "场景复现平台-展示设备(移动、音箱、小爱建议、车载、手表等设备)上小爱执行结果及相关处理流程",
name: "ai-scene-review-fe",
path_with_namespace: "miai-fe/fe/ai-scene-review-fe",
web_url: "https://git.n.xiaomi.com/miai-fe/fe/ai-scene-review-fe",
avatar_url: null,
has_new_cicd: false,
}
```
pipeline 信息
```js
{
id: 'bbb',
project_id: 'aaa',
user_id: 'ccc',
pipeline_id: 7646046,
ref: "preview",
status: "success",
web_url: "https://git.n.xiaomi.com/miai-fe/fe/ai-scene-review-fe/-/pipelines/7646046",
started_at: "2024-03-01T16:47:40.192+08:00",
finished_at: "2024-03-01T16:49:30.624+08:00",
duration: 100,
queued_duration: 6,
}
```
user 信息
```js
{
id: 'ccc',
user_id: 10011,
username: "zhaoyingbo",
name: "赵英博",
avatar_url: "https://git.n.xiaomi.com/uploads/-/system/user/avatar/10011/avatar.png",
web_url: "https://git.n.xiaomi.com/zhaoyingbo"
}
```
每周每个项目按分支的运行时长统计 SQL `statisticsPerProj`
```SQL
SELECT
(ROW_NUMBER() OVER()) as id,
strftime('%Y-%W', datetime(pip.started_at)) AS week,
p.name AS name,
ROUND(AVG(pip.duration/60.0), 1) AS duration,
pip.ref
FROM project p
JOIN pipeline pip ON p.id = pip.project_id
GROUP BY name, week, pip.ref;
```
每周流水线运行统计 SQL `statisticsPerWeek`
```SQL
SELECT
(ROW_NUMBER() OVER()) as id,
strftime('%Y-%W', datetime(started_at)) AS week,
COUNT(*) AS total_count,
SUM(CASE WHEN status = 'success' THEN 0 ELSE 1 END) AS failed_count,
SUM(CASE WHEN status = 'success' THEN 1 ELSE 0 END) AS success_count,
ROUND(SUM(CASE WHEN status = 'success' THEN 1 ELSE 0 END) * 100.0 / COUNT(*), 1) AS success_rate,
ROUND(AVG(duration/60.0), 1) AS duration
FROM pipeline
GROUP BY week;
```
# GPT
我有一个 sqlite 数据库,表如下
project 表
```js
{
id: 'aaa',
project_id: 131366,
description: "场景复现平台-展示设备(移动、音箱、小爱建议、车载、手表等设备)上小爱执行结果及相关处理流程",
name: "ai-scene-review-fe",
path_with_namespace: "miai-fe/fe/ai-scene-review-fe",
web_url: "https://git.n.xiaomi.com/miai-fe/fe/ai-scene-review-fe",
avatar_url: null,
has_new_cicd: false,
}
```
pipeline 表
```js
{
id: 'bbb',
project_id: 'aaa',
user_id: 'ccc',
pipeline_id: 7646046,
ref: "preview",
status: "success",
web_url: "https://git.n.xiaomi.com/miai-fe/fe/ai-scene-review-fe/-/pipelines/7646046",
started_at: "2024-03-01T16:47:40.192+08:00",
finished_at: "2024-03-01T16:49:30.624+08:00",
duration: 100,
queued_duration: 6,
}
```
user 表
```js
{
id: 'ccc',
user_id: 10011,
username: "zhaoyingbo",
name: "赵英博",
avatar_url: "https://git.n.xiaomi.com/uploads/-/system/user/avatar/10011/avatar.png",
web_url: "https://git.n.xiaomi.com/zhaoyingbo"
}
```
我想按天展示每个项目的 pipline 按 ref 区分的平均 duration如何创建视图
# 机器人
卡片 IDctp_AAyVLS6Q37cL
JSON 示例
```json
{
"total_count": "29", // OK
"group_table": [
{
"project_name": "ai-ak-fe",
"project_ref": "master",
"project_duration": "1.4",
"project_duration_rate": "<font color='green'>↓12%</font>"
},
{
"project_name": "ai-class-schedule-fe",
"project_ref": "preview",
"project_duration": "3.2",
"project_duration_rate": "<font color='red'>↑5%</font>"
},
{
"project_name": "ai-scene-review-fe",
"project_ref": "staging",
"project_duration": "5.4",
"project_duration_rate": "<font color='green'>↓6%</font>"
}
],
"weekly_count_rate": "<font color='red'>较上周 ↑5%</font>", // OK
"weekly_duration_rate": "<font color='green'>较上周 ↓12%</font>", // OK
"weekly_success_rate": "<font color='red'>较上周 ↑12%</font>", // OK
"duration": "0.9", // OK
"success_rate": "100", // OK
"has_new_cicd_count": "15", // OK
"without_new_cicd_count": "20" // OK
}
```

10
schedule/index.ts Normal file
View File

@ -0,0 +1,10 @@
import schedule from 'node-schedule'
const func = () => {}
export const initSchedule = async () => {
// 定时任务每15分钟刷新一次token
schedule.scheduleJob('*/15 * * * *', func);
// 立即执行一次
func()
}

28
service/index.ts Normal file
View File

@ -0,0 +1,28 @@
const fetchGetParams = {
method: "GET",
headers: { "PRIVATE-TOKEN": "Zd1UASPcMwVox5tNS6ep" },
};
/**
*
* @param id id
*/
const fetchTemplate = async (id: number) => {
try {
const response = await fetch(
`https://git.n.xiaomi.com/api/v4/projects/${id}`,
fetchGetParams
);
const body = (await response.json()) as any;
if (body.message === "404 Project Not Found") return null;
return body;
} catch {
return null;
}
};
const service = {
fetchTemplate,
};
export default service;

22
tsconfig.json Normal file
View File

@ -0,0 +1,22 @@
{
"compilerOptions": {
"lib": ["ESNext"],
"module": "esnext",
"target": "esnext",
"moduleResolution": "bundler",
"moduleDetection": "force",
"allowImportingTsExtensions": true,
"noEmit": true,
"composite": true,
"strict": true,
"downlevelIteration": true,
"skipLibCheck": true,
"jsx": "react-jsx",
"allowSyntheticDefaultImports": true,
"forceConsistentCasingInFileNames": true,
"allowJs": true,
"types": [
"bun-types" // add Bun global
]
}
}

0
types/index.ts Normal file
View File

10
utils/pbTools.ts Normal file
View File

@ -0,0 +1,10 @@
export const managePb404 = async (dbFunc: Function) => {
try {
return await dbFunc();
} catch (err: any) {
// 没有这个提醒就返回空
if (err?.message === "The requested resource wasn't found.") {
return null;
} else throw err;
}
};