init
This commit is contained in:
commit
260aa8c350
32
.devcontainer/devcontainer.json
Normal file
32
.devcontainer/devcontainer.json
Normal 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
1
.gitattributes
vendored
Normal file
@ -0,0 +1 @@
|
||||
*.lockb binary diff=lockb
|
57
.gitea/workflows/cicd.yaml.template
Normal file
57
.gitea/workflows/cicd.yaml.template
Normal 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
58
.gitignore
vendored
Normal 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
13
Dockerfile
Normal 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"]
|
7
controllers/template/index.ts
Normal file
7
controllers/template/index.ts
Normal file
@ -0,0 +1,7 @@
|
||||
const templateFunc = () => {}
|
||||
|
||||
const template = {
|
||||
templateFunc,
|
||||
};
|
||||
|
||||
export default template;
|
7
db/index.ts
Normal file
7
db/index.ts
Normal file
@ -0,0 +1,7 @@
|
||||
import user from "./user";
|
||||
|
||||
const db = {
|
||||
user,
|
||||
};
|
||||
|
||||
export default db;
|
7
db/pbClient.ts
Normal file
7
db/pbClient.ts
Normal 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
46
db/user/index.ts
Normal 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
9
docker-compose.yml
Normal 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
26
eslint.config.js
Normal 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
14
index.ts
Normal 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
25
package.json
Normal 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
182
readme.md
Normal 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,如何创建视图
|
||||
|
||||
# 机器人
|
||||
|
||||
卡片 ID:ctp_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
10
schedule/index.ts
Normal 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
28
service/index.ts
Normal 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
22
tsconfig.json
Normal 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
0
types/index.ts
Normal file
10
utils/pbTools.ts
Normal file
10
utils/pbTools.ts
Normal 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;
|
||||
}
|
||||
};
|
Loading…
x
Reference in New Issue
Block a user