feat(base): 优化项目基础开发环境
This commit is contained in:
parent
004044a085
commit
68248c1d24
@ -1,8 +1,12 @@
|
||||
{
|
||||
"name": "replace_me",
|
||||
"image": "micr.cloud.mioffice.cn/zhaoyingbo/dev:bun",
|
||||
"remoteUser": "bun",
|
||||
"containerUser": "bun",
|
||||
"image": "mcr.microsoft.com/devcontainers/typescript-node:20",
|
||||
"features": {
|
||||
"ghcr.io/devcontainers/features/docker-in-docker:2": {
|
||||
"version": "latest",
|
||||
"moby": true
|
||||
}
|
||||
},
|
||||
"customizations": {
|
||||
"vscode": {
|
||||
"settings": {
|
||||
@ -10,20 +14,25 @@
|
||||
"editor.guides.bracketPairs": true,
|
||||
"editor.defaultFormatter": "esbenp.prettier-vscode",
|
||||
"editor.formatOnSave": true,
|
||||
"github.copilot.chat.localeOverride": "zh-CN"
|
||||
"github.copilot.chat.localeOverride": "zh-CN",
|
||||
"editor.codeActionsOnSave": {
|
||||
"source.fixAll.eslint": "always"
|
||||
}
|
||||
},
|
||||
"extensions": [
|
||||
"dbaeumer.vscode-eslint",
|
||||
"esbenp.prettier-vscode",
|
||||
"jock.svg",
|
||||
"GitHub.copilot",
|
||||
"eamodio.gitlens",
|
||||
"unifiedjs.vscode-mdx",
|
||||
"oderwat.indent-rainbow",
|
||||
"jock.svg",
|
||||
"ChakrounAnas.turbo-console-log",
|
||||
"dbaeumer.vscode-eslint",
|
||||
"esbenp.prettier-vscode",
|
||||
"Gruntfuggly.todo-tree",
|
||||
"ChakrounAnas.turbo-console-log",
|
||||
"streetsidesoftware.code-spell-checker",
|
||||
"MS-CEINTL.vscode-language-pack-zh-hans",
|
||||
"GitHub.copilot"
|
||||
"Prisma.prisma"
|
||||
]
|
||||
}
|
||||
}
|
||||
},
|
||||
"onCreateCommand": "curl -fsSL https://bun.sh/install | bash"
|
||||
}
|
||||
|
7
.env
Normal file
7
.env
Normal file
@ -0,0 +1,7 @@
|
||||
# Environment variables declared in this file are automatically made available to Prisma.
|
||||
# See the documentation for more detail: https://pris.ly/d/prisma-schema#accessing-environment-variables-from-the-schema
|
||||
|
||||
# Prisma supports the native connection string format for PostgreSQL, MySQL, SQLite, SQL Server, MongoDB and CockroachDB.
|
||||
# See the documentation for all the connection string options: https://pris.ly/d/connection-strings
|
||||
|
||||
DATABASE_URL="mysql://root:rootpassword@localhost:3306/testdb"
|
4
.gitignore
vendored
4
.gitignore
vendored
@ -49,8 +49,8 @@ profile-*
|
||||
.idea
|
||||
|
||||
# vscode
|
||||
.vscode
|
||||
*code-workspace
|
||||
# .vscode
|
||||
# *code-workspace
|
||||
|
||||
# clinic
|
||||
profile*
|
||||
|
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
|
3
.vscode/settings.json
vendored
Normal file
3
.vscode/settings.json
vendored
Normal file
@ -0,0 +1,3 @@
|
||||
{
|
||||
"cSpell.words": ["bunx", "oxlint"]
|
||||
}
|
2
bunfig.toml
Normal file
2
bunfig.toml
Normal file
@ -0,0 +1,2 @@
|
||||
[install]
|
||||
registry = "https://registry.npmjs.org"
|
1
commitlint.config.js
Normal file
1
commitlint.config.js
Normal file
@ -0,0 +1 @@
|
||||
export default { extends: ["@commitlint/config-conventional"] }
|
@ -2,6 +2,6 @@ const templateFunc = () => {}
|
||||
|
||||
const template = {
|
||||
templateFunc,
|
||||
};
|
||||
}
|
||||
|
||||
export default template;
|
||||
export default template
|
||||
|
@ -1,7 +1,7 @@
|
||||
import user from "./user";
|
||||
import user from "./user"
|
||||
|
||||
const db = {
|
||||
user,
|
||||
};
|
||||
}
|
||||
|
||||
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
|
||||
|
19
db/prisma/user/index.ts
Normal file
19
db/prisma/user/index.ts
Normal file
@ -0,0 +1,19 @@
|
||||
import { PrismaClient } from "@prisma/client"
|
||||
|
||||
const prisma = new PrismaClient()
|
||||
|
||||
const main = async () => {
|
||||
const user = await prisma.user.create({
|
||||
data: {
|
||||
name: "Alice",
|
||||
email: "alice@example.com",
|
||||
},
|
||||
})
|
||||
console.log(user)
|
||||
const users = await prisma.user.findMany()
|
||||
console.log(users)
|
||||
}
|
||||
|
||||
main().then(() => {
|
||||
prisma.$disconnect()
|
||||
})
|
@ -1,19 +1,19 @@
|
||||
import { RecordModel } from "pocketbase";
|
||||
import { managePb404 } from "../../utils/pbTools";
|
||||
import pbClient from "../pbClient";
|
||||
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;
|
||||
user_id: number
|
||||
username: string
|
||||
name: string
|
||||
avatar_url: string
|
||||
web_url: string
|
||||
}
|
||||
|
||||
const getOne = (id: string) =>
|
||||
managePb404<UserRecordModel>(
|
||||
async () => await pbClient.collection("user").getOne(id)
|
||||
);
|
||||
)
|
||||
|
||||
const getOneByUserId = (user_id: number) => {
|
||||
return managePb404<UserRecordModel>(
|
||||
@ -23,24 +23,24 @@ const getOneByUserId = (user_id: number) => {
|
||||
.getFirstListItem(`user_id="${user_id}"`, {
|
||||
sort: "-created",
|
||||
})
|
||||
);
|
||||
};
|
||||
)
|
||||
}
|
||||
|
||||
const create = async (data: Partial<UserRecordModel>) =>
|
||||
await pbClient.collection("user").create<UserRecordModel>(data);
|
||||
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);
|
||||
};
|
||||
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;
|
||||
export default user
|
||||
|
16
docker/mysql/dev.docker-compose.yml
Normal file
16
docker/mysql/dev.docker-compose.yml
Normal file
@ -0,0 +1,16 @@
|
||||
version: "3.8"
|
||||
|
||||
services:
|
||||
mysql:
|
||||
image: micr.cloud.mioffice.cn/zhaoyingbo/mysql:8.0
|
||||
container_name: mysql
|
||||
environment:
|
||||
MYSQL_ROOT_PASSWORD: rootpassword
|
||||
MYSQL_DATABASE: testdb
|
||||
ports:
|
||||
- "3306:3306"
|
||||
volumes:
|
||||
- mysql-data:/var/lib/mysql
|
||||
|
||||
volumes:
|
||||
mysql-data:
|
1
docker/mysql/run.sh
Normal file
1
docker/mysql/run.sh
Normal file
@ -0,0 +1 @@
|
||||
docker compose -f dev.docker-compose.yml up -d
|
1
docker/mysql/stop.sh
Normal file
1
docker/mysql/stop.sh
Normal file
@ -0,0 +1 @@
|
||||
docker compose -f dev.docker-compose.yml down
|
@ -1,26 +1,26 @@
|
||||
export default {
|
||||
languageOptions: {
|
||||
ecmaVersion: 2021,
|
||||
sourceType: 'module',
|
||||
sourceType: "module",
|
||||
},
|
||||
ignores: ['*.js', '*.cjs', '*.mjs', '/src/Backup/*'],
|
||||
plugins: ['@typescript-eslint', 'simple-import-sort'],
|
||||
ignores: ["*.js", "*.cjs", "*.mjs", "/src/Backup/*"],
|
||||
plugins: ["@typescript-eslint", "simple-import-sort"],
|
||||
extends: [
|
||||
'eslint:recommended',
|
||||
'plugin:@typescript-eslint/recommended',
|
||||
'prettier',
|
||||
"eslint:recommended",
|
||||
"plugin:@typescript-eslint/recommended",
|
||||
"prettier",
|
||||
],
|
||||
rules: {
|
||||
'@typescript-eslint/no-explicit-any': 'off',
|
||||
'simple-import-sort/imports': 'error',
|
||||
'simple-import-sort/exports': 'error',
|
||||
"@typescript-eslint/no-explicit-any": "off",
|
||||
"simple-import-sort/imports": "error",
|
||||
"simple-import-sort/exports": "error",
|
||||
},
|
||||
overrides: [
|
||||
{
|
||||
files: ['**/*.ts', '**/*.tsx'],
|
||||
files: ["**/*.ts", "**/*.tsx"],
|
||||
languageOptions: {
|
||||
parser: '@typescript-eslint/parser',
|
||||
parser: "@typescript-eslint/parser",
|
||||
},
|
||||
},
|
||||
],
|
||||
};
|
||||
}
|
||||
|
12
index.ts
12
index.ts
@ -1,14 +1,14 @@
|
||||
import { initSchedule } from "./schedule";
|
||||
import { initSchedule } from "./schedule"
|
||||
|
||||
initSchedule()
|
||||
|
||||
Bun.serve({
|
||||
async fetch(req) {
|
||||
const url = new URL(req.url);
|
||||
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");
|
||||
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,
|
||||
});
|
||||
})
|
||||
|
29
package.json
29
package.json
@ -3,21 +3,40 @@
|
||||
"module": "index.ts",
|
||||
"type": "module",
|
||||
"scripts": {
|
||||
"start": "bun run index.ts"
|
||||
"start": "NODE_ENV=production bun run index.ts",
|
||||
"dev": "NODE_ENV=dev bun run index.ts --watch",
|
||||
"lint": "oxlint --fix .",
|
||||
"prepare": "husky",
|
||||
"prettier": "prettier --write .",
|
||||
"init:prisma": "bunx prisma migrate dev --name init"
|
||||
},
|
||||
"lint-staged": {
|
||||
"*.{js,jsx,ts,tsx}": [
|
||||
"oxlint --fix",
|
||||
"eslint --fix",
|
||||
"prettier --write"
|
||||
]
|
||||
},
|
||||
"devDependencies": {
|
||||
"eslint": "^9.2.0",
|
||||
"bun-types": "latest",
|
||||
"@commitlint/cli": "^19.5.0",
|
||||
"@commitlint/config-conventional": "^19.5.0",
|
||||
"@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"
|
||||
"@typescript-eslint/parser": "^7.8.0",
|
||||
"bun-types": "latest",
|
||||
"eslint": "^9.2.0",
|
||||
"eslint-plugin-simple-import-sort": "^12.1.0",
|
||||
"husky": "^9.1.6",
|
||||
"lint-staged": "^15.2.10",
|
||||
"oxlint": "^0.11.1",
|
||||
"prettier": "^3.3.3"
|
||||
},
|
||||
"peerDependencies": {
|
||||
"typescript": "^5.0.0"
|
||||
},
|
||||
"dependencies": {
|
||||
"@prisma/client": "^5.22.0",
|
||||
"lodash": "^4.17.21",
|
||||
"node-schedule": "^2.1.1",
|
||||
"pocketbase": "^0.21.1"
|
||||
|
6
prettier.config.js
Normal file
6
prettier.config.js
Normal file
@ -0,0 +1,6 @@
|
||||
export default {
|
||||
trailingComma: "es5",
|
||||
tabWidth: 2,
|
||||
semi: false,
|
||||
singleQuote: false,
|
||||
}
|
9
prisma/migrations/20241116101639_init/migration.sql
Normal file
9
prisma/migrations/20241116101639_init/migration.sql
Normal file
@ -0,0 +1,9 @@
|
||||
-- CreateTable
|
||||
CREATE TABLE `User` (
|
||||
`id` INTEGER NOT NULL AUTO_INCREMENT,
|
||||
`name` VARCHAR(191) NOT NULL,
|
||||
`email` VARCHAR(191) NOT NULL,
|
||||
|
||||
UNIQUE INDEX `User_email_key`(`email`),
|
||||
PRIMARY KEY (`id`)
|
||||
) DEFAULT CHARACTER SET utf8mb4 COLLATE utf8mb4_unicode_ci;
|
3
prisma/migrations/migration_lock.toml
Normal file
3
prisma/migrations/migration_lock.toml
Normal file
@ -0,0 +1,3 @@
|
||||
# Please do not edit this file manually
|
||||
# It should be added in your version-control system (i.e. Git)
|
||||
provider = "mysql"
|
20
prisma/schema.prisma
Normal file
20
prisma/schema.prisma
Normal file
@ -0,0 +1,20 @@
|
||||
// This is your Prisma schema file,
|
||||
// learn more about it in the docs: https://pris.ly/d/prisma-schema
|
||||
|
||||
// Looking for ways to speed up your queries, or scale easily with your serverless or edge functions?
|
||||
// Try Prisma Accelerate: https://pris.ly/cli/accelerate-init
|
||||
|
||||
generator client {
|
||||
provider = "prisma-client-js"
|
||||
}
|
||||
|
||||
datasource db {
|
||||
provider = "mysql"
|
||||
url = env("DATABASE_URL")
|
||||
}
|
||||
|
||||
model User {
|
||||
id Int @id @default(autoincrement())
|
||||
name String
|
||||
email String @unique
|
||||
}
|
@ -1,10 +1,10 @@
|
||||
import schedule from 'node-schedule'
|
||||
import schedule from "node-schedule"
|
||||
|
||||
const func = () => {}
|
||||
|
||||
export const initSchedule = async () => {
|
||||
// 定时任务,每15分钟刷新一次token
|
||||
schedule.scheduleJob('*/15 * * * *', func);
|
||||
schedule.scheduleJob("*/15 * * * *", func)
|
||||
// 立即执行一次
|
||||
func()
|
||||
}
|
||||
|
@ -1,7 +1,7 @@
|
||||
const fetchGetParams = {
|
||||
method: "GET",
|
||||
headers: { "PRIVATE-TOKEN": "Zd1UASPcMwVox5tNS6ep" },
|
||||
};
|
||||
}
|
||||
|
||||
/**
|
||||
* 获取项目详情
|
||||
@ -12,17 +12,17 @@ const fetchTemplate = async (id: number) => {
|
||||
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;
|
||||
)
|
||||
const body = (await response.json()) as any
|
||||
if (body.message === "404 Project Not Found") return null
|
||||
return body
|
||||
} catch {
|
||||
return null;
|
||||
return null
|
||||
}
|
||||
};
|
||||
}
|
||||
|
||||
const service = {
|
||||
fetchTemplate,
|
||||
};
|
||||
}
|
||||
|
||||
export default service;
|
||||
export default service
|
||||
|
@ -0,0 +1,4 @@
|
||||
export interface FakeType {
|
||||
name: string
|
||||
age: number
|
||||
}
|
@ -1,9 +1,9 @@
|
||||
export const managePb404 = async <T>(dbFunc: Function): Promise<T | null> => {
|
||||
try {
|
||||
return await dbFunc();
|
||||
return await dbFunc()
|
||||
} catch (err: any) {
|
||||
if (err?.message === "The requested resource wasn't found.") {
|
||||
return null;
|
||||
} else throw err;
|
||||
return null
|
||||
} else throw err
|
||||
}
|
||||
};
|
||||
}
|
||||
|
Loading…
x
Reference in New Issue
Block a user