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