create-aomex 0.0.38 → 0.0.40

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
package/dist/index.js CHANGED
@@ -183,7 +183,7 @@ var { error } = await terminal3.runTasks([
183
183
  {
184
184
  label: "dependencies",
185
185
  pkgs: [
186
- "@aomex/core",
186
+ "@aomex/common",
187
187
  "@aomex/web",
188
188
  "@aomex/cors",
189
189
  "@aomex/etag",
package/dist/index.js.map CHANGED
@@ -1 +1 @@
1
- {"version":3,"sources":["../src/index.ts","../src/input-project.ts","../src/select-node-version.ts"],"sourcesContent":["#!/usr/bin/env node\n\nimport path from 'node:path/posix';\nimport yargsParser from 'yargs-parser';\nimport {\n cp,\n mkdir,\n readdir,\n readFile,\n rename,\n stat,\n writeFile,\n} from 'node:fs/promises';\nimport { terminal } from '@aomex/console';\nimport { execSync, spawn } from 'node:child_process';\nimport { setTimeout } from 'node:timers/promises';\nimport { inputProject } from './input-project';\nimport { selectNodeVersion } from './select-node-version';\n\nconst runShell = async (command: string) => {\n await new Promise((resolve, reject) => {\n const stream = spawn(command, {\n cwd: process.cwd(),\n env: process.env,\n shell: true,\n });\n stream.on('close', resolve);\n stream.on('error', reject);\n });\n};\n\nconst argv = yargsParser(process.argv.slice(2));\nconst templateDir = path.join(import.meta.dirname, '..', 'templates');\nconst projectName = await inputProject(argv);\nconst pnpmVersion =\n execSync(`npm view pnpm version`, {\n encoding: 'utf8',\n }).replaceAll('\\n', '') || '0.0.0';\n\nconst nodeVersion = await selectNodeVersion(argv);\nconst targetDir = path.resolve(projectName);\n\nconst { error } = await terminal.runTasks([\n {\n title: '创建目录',\n task: async () => {\n await mkdir(targetDir, { recursive: true });\n process.chdir(targetDir);\n await setTimeout(500);\n },\n },\n {\n title: '复制模板文件',\n task: async () => {\n const variables = {\n projectName,\n nodeVersion,\n pnpmVersion,\n };\n await cp(templateDir, targetDir, { recursive: true });\n const files = await readdir(targetDir, { recursive: true });\n for (const file of files) {\n const isFile = (await stat(path.join(targetDir, file))).isFile();\n if (!isFile) continue;\n const fileAbsolutePath = path.resolve(file);\n let fileContent = await readFile(fileAbsolutePath, 'utf8');\n Object.entries(variables).forEach(([key, value]) => {\n fileContent = fileContent.replaceAll(`{{${key}}}`, value);\n });\n await writeFile(fileAbsolutePath, fileContent);\n }\n // .gitignore这个文件在安装的时候总是被忽略\n await rename(\n path.join(targetDir, 'gitignore'),\n path.join(targetDir, '.gitignore'),\n );\n await setTimeout(500);\n },\n },\n {\n title: 'git初始化',\n skip: async () => {\n return !/\\d\\.\\d/.test(execSync('git -v', { encoding: 'utf8' }));\n },\n task: async () => {\n await runShell('git init');\n await setTimeout(500);\n },\n },\n {\n title: '增加volta配置',\n skip: async () => {\n return !/\\d\\.\\d/.test(execSync('volta -v', { encoding: 'utf8' }));\n },\n task: async () => {\n await runShell(`volta pin node@${nodeVersion}`);\n await runShell(`volta pin pnpm@${pnpmVersion}`);\n },\n },\n {\n title: '安装插件',\n task: async (_, task) => {\n const packages: { label: string; pkgs: string[]; dev?: boolean }[] = [\n {\n label: 'dependencies',\n pkgs: [\n '@aomex/core',\n '@aomex/web',\n '@aomex/cors',\n '@aomex/etag',\n '@aomex/compress',\n '@aomex/http-logger',\n '@aomex/response-time',\n '@aomex/console',\n '@aomex/cron',\n '@aomex/helmet',\n '@aomex/openapi',\n '@aomex/swagger-ui',\n '@aomex/async-trace',\n '@prisma/client',\n '@aomex/cache',\n '@aomex/cache-redis-adapter',\n '@aomex/auth',\n 'ioredis',\n ],\n },\n {\n label: 'dev dependencies',\n pkgs: [\n 'prisma',\n 'typescript',\n 'tsx',\n 'tsc-alias',\n '@types/node',\n 'husky',\n 'prettier',\n '@commitlint/cli',\n '@commitlint/config-conventional',\n 'eslint',\n '@typescript-eslint/eslint-plugin',\n '@typescript-eslint/parser',\n 'eslint-plugin-check-file',\n ],\n dev: true,\n },\n ];\n\n for (let i = 0; i < packages.length; ++i) {\n const { pkgs, dev, label } = packages[i]!;\n task.suffix = terminal.style('gray', label);\n await runShell(`pnpm add ${pkgs.join(' ')} ${dev ? '-D' : ''}`);\n }\n task.suffix = '';\n },\n },\n {\n title: '生成prisma客户端',\n task: async () => {\n await runShell('npx prisma generate');\n },\n },\n]);\n\nif (error) process.exit(1);\n\nconsole.log(\n '\\n项目创建成功:' +\n terminal.style(['blue', 'underline'], process.cwd()) +\n '\\n' +\n '启动项目可执行如下指令:' +\n '\\n\\n' +\n terminal.style('green', `cd ${projectName} && pnpm start`) +\n '\\n',\n);\n","import { terminal } from '@aomex/console';\nimport { input } from '@inquirer/prompts';\nimport { existsSync, readdirSync } from 'node:fs';\nimport sanitize from 'sanitize-filename';\nimport path from 'path';\n\nconst kebabCase = (value: string) => {\n value = sanitize(value);\n if (!value.length) return '';\n value = value[0]!.toLowerCase() + value.slice(1);\n return value\n .replace(/[A-Z]/g, (i) => '-' + i.toLowerCase())\n .replace(/_/g, '-')\n .replace(/-+/g, '-');\n};\n\nexport const inputProject = async (argv: Record<string, any>) => {\n let projectName: string = argv['project'];\n if (projectName) {\n projectName = kebabCase(projectName);\n const available = isProjectDirAvailable(projectName);\n if (typeof available === 'string') {\n terminal.printError(available);\n process.exit(1);\n }\n } else {\n projectName = await input({\n message: '请输入项目',\n validate(value) {\n return isProjectDirAvailable(value);\n },\n transformer(value) {\n return kebabCase(value);\n },\n });\n projectName = kebabCase(projectName);\n }\n\n return projectName;\n};\n\nconst isProjectDirAvailable = (projectName: string): string | true => {\n projectName = projectName.trim();\n if (!projectName.trim()) return '项目名不能为空';\n const targetDir = path.resolve(projectName);\n if (!existsSync(targetDir)) return true;\n if (!readdirSync(targetDir).length) return true;\n return `目录 \"${targetDir}\" 包含其它文件,请手动删除或选择新的项目名`;\n};\n","import { terminal } from '@aomex/console';\nimport { select } from '@inquirer/prompts';\n\nexport const selectNodeVersion = async (argv: Record<string, any>) => {\n let nodeVersion: string = argv['node'];\n if (!nodeVersion) {\n const nodes = await promise;\n nodeVersion = await select({\n message: '请选择node.js稳定版本',\n choices: nodes.map((item) => {\n return {\n name: `${item.version} ${terminal.style(['gray', 'dim'], item.date)}`,\n value: item.version,\n short: item.version,\n };\n }),\n loop: false,\n });\n }\n\n return nodeVersion.replace(/^v/, '');\n};\n\nconst fetchNodes = async () => {\n const response = await fetch(\n 'https://nodejs.org/download/release/index.json',\n );\n const nodes = (await response.json()) as {\n version: string;\n lts: boolean;\n date: string;\n }[];\n\n return nodes\n .map((item) => {\n return {\n ...item,\n versions: item.version.slice(1).split('.').map(Number) as [\n number,\n number,\n number,\n ],\n };\n })\n .filter((item) => {\n if (!item.lts) return false;\n if (item.versions[0] === 20 && item.versions[1] >= 13) return true;\n if (item.versions[0] >= 22) return true;\n return false;\n });\n};\n\nconst promise = fetchNodes();\n"],"mappings":"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","names":["path","terminal","argv","projectName","targetDir","terminal","argv","nodeVersion","path","terminal"]}
1
+ {"version":3,"sources":["../src/index.ts","../src/input-project.ts","../src/select-node-version.ts"],"sourcesContent":["#!/usr/bin/env node\n\nimport path from 'node:path/posix';\nimport yargsParser from 'yargs-parser';\nimport {\n cp,\n mkdir,\n readdir,\n readFile,\n rename,\n stat,\n writeFile,\n} from 'node:fs/promises';\nimport { terminal } from '@aomex/console';\nimport { execSync, spawn } from 'node:child_process';\nimport { setTimeout } from 'node:timers/promises';\nimport { inputProject } from './input-project';\nimport { selectNodeVersion } from './select-node-version';\n\nconst runShell = async (command: string) => {\n await new Promise((resolve, reject) => {\n const stream = spawn(command, {\n cwd: process.cwd(),\n env: process.env,\n shell: true,\n });\n stream.on('close', resolve);\n stream.on('error', reject);\n });\n};\n\nconst argv = yargsParser(process.argv.slice(2));\nconst templateDir = path.join(import.meta.dirname, '..', 'templates');\nconst projectName = await inputProject(argv);\nconst pnpmVersion =\n execSync(`npm view pnpm version`, {\n encoding: 'utf8',\n }).replaceAll('\\n', '') || '0.0.0';\n\nconst nodeVersion = await selectNodeVersion(argv);\nconst targetDir = path.resolve(projectName);\n\nconst { error } = await terminal.runTasks([\n {\n title: '创建目录',\n task: async () => {\n await mkdir(targetDir, { recursive: true });\n process.chdir(targetDir);\n await setTimeout(500);\n },\n },\n {\n title: '复制模板文件',\n task: async () => {\n const variables = {\n projectName,\n nodeVersion,\n pnpmVersion,\n };\n await cp(templateDir, targetDir, { recursive: true });\n const files = await readdir(targetDir, { recursive: true });\n for (const file of files) {\n const isFile = (await stat(path.join(targetDir, file))).isFile();\n if (!isFile) continue;\n const fileAbsolutePath = path.resolve(file);\n let fileContent = await readFile(fileAbsolutePath, 'utf8');\n Object.entries(variables).forEach(([key, value]) => {\n fileContent = fileContent.replaceAll(`{{${key}}}`, value);\n });\n await writeFile(fileAbsolutePath, fileContent);\n }\n // .gitignore这个文件在安装的时候总是被忽略\n await rename(\n path.join(targetDir, 'gitignore'),\n path.join(targetDir, '.gitignore'),\n );\n await setTimeout(500);\n },\n },\n {\n title: 'git初始化',\n skip: async () => {\n return !/\\d\\.\\d/.test(execSync('git -v', { encoding: 'utf8' }));\n },\n task: async () => {\n await runShell('git init');\n await setTimeout(500);\n },\n },\n {\n title: '增加volta配置',\n skip: async () => {\n return !/\\d\\.\\d/.test(execSync('volta -v', { encoding: 'utf8' }));\n },\n task: async () => {\n await runShell(`volta pin node@${nodeVersion}`);\n await runShell(`volta pin pnpm@${pnpmVersion}`);\n },\n },\n {\n title: '安装插件',\n task: async (_, task) => {\n const packages: { label: string; pkgs: string[]; dev?: boolean }[] = [\n {\n label: 'dependencies',\n pkgs: [\n '@aomex/common',\n '@aomex/web',\n '@aomex/cors',\n '@aomex/etag',\n '@aomex/compress',\n '@aomex/http-logger',\n '@aomex/response-time',\n '@aomex/console',\n '@aomex/cron',\n '@aomex/helmet',\n '@aomex/openapi',\n '@aomex/swagger-ui',\n '@aomex/async-trace',\n '@prisma/client',\n '@aomex/cache',\n '@aomex/cache-redis-adapter',\n '@aomex/auth',\n 'ioredis',\n ],\n },\n {\n label: 'dev dependencies',\n pkgs: [\n 'prisma',\n 'typescript',\n 'tsx',\n 'tsc-alias',\n '@types/node',\n 'husky',\n 'prettier',\n '@commitlint/cli',\n '@commitlint/config-conventional',\n 'eslint',\n '@typescript-eslint/eslint-plugin',\n '@typescript-eslint/parser',\n 'eslint-plugin-check-file',\n ],\n dev: true,\n },\n ];\n\n for (let i = 0; i < packages.length; ++i) {\n const { pkgs, dev, label } = packages[i]!;\n task.suffix = terminal.style('gray', label);\n await runShell(`pnpm add ${pkgs.join(' ')} ${dev ? '-D' : ''}`);\n }\n task.suffix = '';\n },\n },\n {\n title: '生成prisma客户端',\n task: async () => {\n await runShell('npx prisma generate');\n },\n },\n]);\n\nif (error) process.exit(1);\n\nconsole.log(\n '\\n项目创建成功:' +\n terminal.style(['blue', 'underline'], process.cwd()) +\n '\\n' +\n '启动项目可执行如下指令:' +\n '\\n\\n' +\n terminal.style('green', `cd ${projectName} && pnpm start`) +\n '\\n',\n);\n","import { terminal } from '@aomex/console';\nimport { input } from '@inquirer/prompts';\nimport { existsSync, readdirSync } from 'node:fs';\nimport sanitize from 'sanitize-filename';\nimport path from 'path';\n\nconst kebabCase = (value: string) => {\n value = sanitize(value);\n if (!value.length) return '';\n value = value[0]!.toLowerCase() + value.slice(1);\n return value\n .replace(/[A-Z]/g, (i) => '-' + i.toLowerCase())\n .replace(/_/g, '-')\n .replace(/-+/g, '-');\n};\n\nexport const inputProject = async (argv: Record<string, any>) => {\n let projectName: string = argv['project'];\n if (projectName) {\n projectName = kebabCase(projectName);\n const available = isProjectDirAvailable(projectName);\n if (typeof available === 'string') {\n terminal.printError(available);\n process.exit(1);\n }\n } else {\n projectName = await input({\n message: '请输入项目',\n validate(value) {\n return isProjectDirAvailable(value);\n },\n transformer(value) {\n return kebabCase(value);\n },\n });\n projectName = kebabCase(projectName);\n }\n\n return projectName;\n};\n\nconst isProjectDirAvailable = (projectName: string): string | true => {\n projectName = projectName.trim();\n if (!projectName.trim()) return '项目名不能为空';\n const targetDir = path.resolve(projectName);\n if (!existsSync(targetDir)) return true;\n if (!readdirSync(targetDir).length) return true;\n return `目录 \"${targetDir}\" 包含其它文件,请手动删除或选择新的项目名`;\n};\n","import { terminal } from '@aomex/console';\nimport { select } from '@inquirer/prompts';\n\nexport const selectNodeVersion = async (argv: Record<string, any>) => {\n let nodeVersion: string = argv['node'];\n if (!nodeVersion) {\n const nodes = await promise;\n nodeVersion = await select({\n message: '请选择node.js稳定版本',\n choices: nodes.map((item) => {\n return {\n name: `${item.version} ${terminal.style(['gray', 'dim'], item.date)}`,\n value: item.version,\n short: item.version,\n };\n }),\n loop: false,\n });\n }\n\n return nodeVersion.replace(/^v/, '');\n};\n\nconst fetchNodes = async () => {\n const response = await fetch(\n 'https://nodejs.org/download/release/index.json',\n );\n const nodes = (await response.json()) as {\n version: string;\n lts: boolean;\n date: string;\n }[];\n\n return nodes\n .map((item) => {\n return {\n ...item,\n versions: item.version.slice(1).split('.').map(Number) as [\n number,\n number,\n number,\n ],\n };\n })\n .filter((item) => {\n if (!item.lts) return false;\n if (item.versions[0] === 20 && item.versions[1] >= 13) return true;\n if (item.versions[0] >= 22) return true;\n return false;\n });\n};\n\nconst promise = fetchNodes();\n"],"mappings":"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","names":["path","terminal","argv","projectName","targetDir","terminal","argv","nodeVersion","path","terminal"]}
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "create-aomex",
3
- "version": "0.0.38",
3
+ "version": "0.0.40",
4
4
  "repository": "git@github.com:aomex/create-aomex.git",
5
5
  "license": "MIT",
6
6
  "type": "module",
@@ -30,27 +30,27 @@
30
30
  "bin": "dist/index.js",
31
31
  "packageManager": "pnpm@9.4.0",
32
32
  "dependencies": {
33
- "@aomex/console": "^3.3.0",
34
- "@aomex/core": "^3.3.0",
33
+ "@aomex/common": "^3.9.0",
34
+ "@aomex/console": "^3.9.0",
35
35
  "@inquirer/prompts": "^6.0.1",
36
36
  "sanitize-filename": "^1.6.3",
37
37
  "yargs-parser": "^21.1.1"
38
38
  },
39
39
  "devDependencies": {
40
- "@aomex/async-trace": "^3.3.0",
40
+ "@aomex/async-trace": "^3.9.0",
41
41
  "@aomex/auth": "^3.4.0",
42
- "@aomex/cache": "^3.3.0",
43
- "@aomex/cache-redis-adapter": "^3.3.0",
44
- "@aomex/compress": "^3.3.0",
45
- "@aomex/cors": "^3.3.0",
46
- "@aomex/cron": "^3.3.0",
47
- "@aomex/etag": "^3.3.0",
48
- "@aomex/helmet": "^3.3.0",
49
- "@aomex/http-logger": "^3.3.0",
50
- "@aomex/openapi": "^3.3.0",
51
- "@aomex/response-time": "^3.3.0",
52
- "@aomex/swagger-ui": "^3.3.0",
53
- "@aomex/web": "^3.3.0",
42
+ "@aomex/cache": "^3.9.0",
43
+ "@aomex/cache-redis-adapter": "^3.9.0",
44
+ "@aomex/compress": "^3.9.0",
45
+ "@aomex/cors": "^3.9.0",
46
+ "@aomex/cron": "^3.9.0",
47
+ "@aomex/etag": "^3.9.0",
48
+ "@aomex/helmet": "^3.9.0",
49
+ "@aomex/http-logger": "^3.9.0",
50
+ "@aomex/openapi": "^3.9.0",
51
+ "@aomex/response-time": "^3.9.0",
52
+ "@aomex/swagger-ui": "^3.9.0",
53
+ "@aomex/web": "^3.9.0",
54
54
  "@prisma/client": "^5.17.0",
55
55
  "@types/lodash.kebabcase": "^4.1.9",
56
56
  "@types/node": "^20.14.12",
@@ -4,9 +4,10 @@ services:
4
4
  command: npx prisma migrate deploy
5
5
  api:
6
6
  image: {{projectName}}:integration
7
- restart: always
8
7
  ports:
9
8
  - 3000:3000
10
9
  cron:
11
10
  image: {{projectName}}:integration
12
11
  command: npx aomex cron:start
12
+ environment:
13
+ - NODE_COMPILE_CACHE=/tmp
@@ -4,9 +4,10 @@ services:
4
4
  command: npx prisma migrate deploy
5
5
  api:
6
6
  image: {{projectName}}:production
7
- restart: always
8
7
  ports:
9
8
  - 3000:3000
10
9
  cron:
11
10
  image: {{projectName}}:production
12
11
  command: npx aomex cron:start
12
+ environment:
13
+ - NODE_COMPILE_CACHE=/tmp
@@ -6,4 +6,5 @@ build/
6
6
  coverage/
7
7
  .DS_Store
8
8
  *.log
9
- volumes/
9
+ volumes/
10
+ logs/
@@ -8,11 +8,11 @@ generator client {
8
8
  provider = "prisma-client-js"
9
9
  }
10
10
 
11
- model user {
12
- id Int @id @default(autoincrement())
13
- name String @unique
14
- age Int @default(17)
15
- address String?
16
- created_at DateTime @default(now()) @db.Timestamp(0)
17
- updated_at DateTime @default(now()) @updatedAt() @db.Timestamp(0)
18
- }
11
+ // model user {
12
+ // id Int @id @default(autoincrement())
13
+ // name String @unique
14
+ // age Int @default(17)
15
+ // address String?
16
+ // created_at DateTime @default(now()) @db.Timestamp(0)
17
+ // updated_at DateTime @default(now()) @updatedAt() @db.Timestamp(0)
18
+ // }
@@ -1,26 +1,23 @@
1
1
  import { commanders, ConsoleApp } from '@aomex/console';
2
2
  import { crons } from '@aomex/cron';
3
- import { traceMiddleware } from '@aomex/async-trace';
4
- import { services } from '@services';
3
+ import { redisCache } from '@services/cache.service';
4
+ import { openapi } from '@aomex/openapi';
5
+ import { logger } from '@services/logger';
5
6
 
6
7
  const app = new ConsoleApp({
7
8
  language: 'zh_CN',
8
9
  mount: [
9
10
  crons({
10
11
  commanders: './src/commanders',
11
- cache: services.cache.instance,
12
- }),
13
- traceMiddleware('生命周期', async (_record) => {
14
- // 根据 record.delta 上报慢日志
15
- // console.log(record);
12
+ cache: redisCache,
16
13
  }),
14
+ openapi({ routers: './src/routers' }),
17
15
  commanders('./src/commanders'),
18
16
  ],
19
17
  });
20
18
 
21
19
  app.on('error', (err) => {
22
- // 上报错误日志
23
- app.log(err);
20
+ logger.error(err.stack || '');
24
21
  });
25
22
 
26
23
  const code = await app.run();
@@ -1,5 +1,5 @@
1
1
  import { Commander, options } from '@aomex/console';
2
- import { rule } from '@aomex/core';
2
+ import { rule } from '@aomex/common';
3
3
  import { cron } from '@aomex/cron';
4
4
  import timers from 'node:timers/promises';
5
5
 
@@ -1,4 +1,4 @@
1
- import { I18n } from '@aomex/core';
1
+ import { I18n } from '@aomex/common';
2
2
  import { resources } from './locales';
3
3
 
4
4
  export const i18n = new I18n({
@@ -1,6 +1,15 @@
1
- import { I18n } from '@aomex/core';
1
+ import { I18n } from '@aomex/common';
2
2
  import { zh } from './zh-cn';
3
3
 
4
4
  export const en = I18n.satisfies(zh).define({
5
- hello: 'Hello World',
5
+ hello: I18n.message('Hello World. {{count}}', {
6
+ count: {
7
+ type: 'plural',
8
+ plural: {
9
+ '1': 'You are first here',
10
+ '2-4': 'Good job',
11
+ '5-n': 'Research other features plz',
12
+ },
13
+ },
14
+ }),
6
15
  });
@@ -1,4 +1,4 @@
1
- import { I18n } from '@aomex/core';
1
+ import { I18n } from '@aomex/common';
2
2
 
3
3
  export const zh = I18n.define({
4
4
  hello: I18n.message('你好,世界。{{count}}', {
@@ -1,4 +1,4 @@
1
- import { middleware } from '@aomex/core';
1
+ import { middleware } from '@aomex/common';
2
2
 
3
3
  let _counter = 0;
4
4
 
@@ -0,0 +1,9 @@
1
+ import { Logger } from '@aomex/common';
2
+ import { httpLogger as aomexHttpLogger } from '@aomex/http-logger';
3
+
4
+ export const httpLogger = aomexHttpLogger({
5
+ transports: [
6
+ new Logger.transport.Console(),
7
+ new Logger.transport.File({ file: 'logs/http/%year%-%month%-%day%/%hour%.log' }),
8
+ ],
9
+ });
@@ -1,4 +1,4 @@
1
- import { middleware, I18n } from '@aomex/core';
1
+ import { middleware, I18n } from '@aomex/common';
2
2
 
3
3
  export const i18nProvider = middleware.web((ctx, next) => {
4
4
  // 动态选择i18n语言包
@@ -1,6 +1,7 @@
1
1
  import { traceMiddleware } from '@aomex/async-trace';
2
+ import type { WebContext } from '@aomex/web';
2
3
 
3
- export const trace = traceMiddleware('生命周期', async (_record) => {
4
+ export const trace = traceMiddleware('生命周期', async (_record, _ctx: WebContext) => {
4
5
  // 根据 record.delta 上报慢日志
5
6
  // console.log(record);
6
7
  });
@@ -1,4 +1,4 @@
1
- import { rule } from '@aomex/core';
1
+ import { rule } from '@aomex/common';
2
2
  import { response, Router } from '@aomex/web';
3
3
  import { counter } from '@middleware/counter.md';
4
4
  import { i18n } from '../i18n';
@@ -1,25 +1,8 @@
1
- import { Service } from '@aomex/core';
1
+ import { Service } from '@aomex/common';
2
2
  import { Caching } from '@aomex/cache';
3
3
  import { redisAdapter } from '@aomex/cache-redis-adapter';
4
4
  import { configs } from '@configs';
5
5
 
6
- const cache = new Caching(redisAdapter(configs.redis));
6
+ export const redisCache = new Caching(redisAdapter(configs.redis));
7
7
 
8
- export class CacheService extends Service {
9
- public readonly instance = cache;
10
-
11
- protected readonly demoKey = 'demo';
12
-
13
- getHotRankings() {
14
- return cache.get<DemoItem[]>(this.demoKey, []);
15
- }
16
-
17
- setHotRankings(value: DemoItem[]) {
18
- return cache.set(this.demoKey, value, 30_000);
19
- }
20
- }
21
-
22
- interface DemoItem {
23
- id: number;
24
- name: string;
25
- }
8
+ export class CacheService extends Service {}
@@ -1,13 +1,11 @@
1
- import { combineServices } from '@aomex/core';
2
- import { UserService } from './user.service';
1
+ import { combineServices } from '@aomex/common';
3
2
  import { CacheService } from './cache.service';
4
3
 
5
4
  export const services = await combineServices({
6
- user: UserService,
7
5
  cache: CacheService,
8
6
  });
9
7
 
10
- declare module '@aomex/core' {
8
+ declare module '@aomex/common' {
11
9
  type T = typeof services;
12
10
  export interface CombinedServices extends T {}
13
11
  }
@@ -0,0 +1,17 @@
1
+ import { Logger } from '@aomex/common';
2
+
3
+ export const logger = Logger.create({
4
+ levels: ['debug', 'info', 'warn', 'error'],
5
+ transports: [
6
+ {
7
+ transport: new Logger.transport.Console(),
8
+ level: { from: 'debug' },
9
+ },
10
+ {
11
+ transport: new Logger.transport.File({
12
+ file: `logs/error-%year%-%month%-%day%.log`,
13
+ }),
14
+ level: ['error'],
15
+ },
16
+ ],
17
+ });
@@ -2,7 +2,7 @@ import { PrismaClient } from '@prisma/client';
2
2
  import * as runtime from '@prisma/client/runtime/library';
3
3
 
4
4
  export const prisma = new PrismaClient({
5
- log: ['error'],
5
+ log: ['warn'],
6
6
  });
7
7
 
8
8
  export type PrismaTX = Omit<PrismaClient, runtime.ITXClientDenyList>;
@@ -1,33 +1,33 @@
1
1
  import { routers, WebApp } from '@aomex/web';
2
2
  import { cors } from '@aomex/cors';
3
3
  import { compress } from '@aomex/compress';
4
- import { httpLogger } from '@aomex/http-logger';
5
4
  import { etag } from '@aomex/etag';
6
5
  import { helmet } from '@aomex/helmet';
7
6
  import { responseTime } from '@aomex/response-time';
8
7
  import { swagger } from '@middleware/swagger.md';
9
8
  import { i18nProvider } from '@middleware/i18n-provider.md';
10
9
  import { trace } from '@middleware/trace.md';
10
+ import { httpLogger } from '@middleware/http-logger.md';
11
+ import { logger } from '@services/logger';
11
12
 
12
13
  export const app = new WebApp({
13
14
  language: 'zh_CN',
14
15
  mount: [
15
16
  i18nProvider,
16
- httpLogger(),
17
+ httpLogger,
17
18
  responseTime,
18
- trace,
19
19
  cors(),
20
20
  compress(),
21
21
  etag(),
22
22
  swagger,
23
23
  helmet(),
24
+ trace,
24
25
  routers('./src/routers'),
25
26
  ],
26
27
  });
27
28
 
28
29
  app.on('error', (err, ctx) => {
29
- // 上报错误日志
30
- app.log(err, ctx);
30
+ logger.error(err.stack || '');
31
31
  ctx.response.body = {
32
32
  status: ctx.response.statusCode,
33
33
  message: ctx.response.body,
@@ -1,71 +0,0 @@
1
- import { services } from '@services';
2
- import { body, params, response, Router } from '@aomex/web';
3
- import { rule } from '@aomex/core';
4
-
5
- export const router = new Router({
6
- prefix: '/users',
7
- });
8
-
9
- router.get('/', {
10
- mount: [
11
- response({
12
- statusCode: 200,
13
- content: rule.array({
14
- id: rule.int(),
15
- name: rule.string(),
16
- age: rule.int(),
17
- address: rule.string().nullable(),
18
- }),
19
- }),
20
- ],
21
- action: async (ctx) => {
22
- const users = await services.user.findAll();
23
- ctx.send(200, users);
24
- },
25
- });
26
-
27
- router.get('/:id', {
28
- mount: [
29
- params({
30
- id: rule.int().min(1),
31
- }),
32
- response({
33
- statusCode: 200,
34
- content: {
35
- id: rule.int(),
36
- name: rule.string(),
37
- age: rule.int(),
38
- address: rule.string().nullable(),
39
- },
40
- }),
41
- response({
42
- statusCode: 404,
43
- description: '用户不存在',
44
- }),
45
- ],
46
- action: async (ctx) => {
47
- const { id } = ctx.params;
48
- const user = await services.user.findById(id);
49
- if (!user) {
50
- return ctx.throw(404, '用户不存在');
51
- }
52
- ctx.send(user);
53
- },
54
- });
55
-
56
- router.post('/', {
57
- mount: [
58
- body({
59
- name: rule.string(),
60
- age: rule.number(),
61
- address: rule.string().optional(),
62
- }),
63
- response({
64
- statusCode: 201,
65
- }),
66
- ],
67
- action: async (ctx) => {
68
- await services.user.createUser(ctx.body);
69
- ctx.send(201, null);
70
- },
71
- });
@@ -1,28 +0,0 @@
1
- import { Service } from '@aomex/core';
2
- import { prisma } from './prisma';
3
- import { Prisma } from '@prisma/client';
4
- import { traceMethod } from '@aomex/async-trace';
5
-
6
- export class UserService extends Service {
7
- @traceMethod()
8
- async findAll() {
9
- return prisma.user.findMany({
10
- orderBy: [
11
- {
12
- id: Prisma.SortOrder.desc,
13
- },
14
- ],
15
- });
16
- }
17
-
18
- @traceMethod((userId) => `User.find(${userId})`)
19
- async findById(userId: number) {
20
- return prisma.user.findUnique({ where: { id: userId } });
21
- }
22
-
23
- async createUser(data: Prisma.userCreateArgs['data']) {
24
- await prisma.user.create({
25
- data: data,
26
- });
27
- }
28
- }