create-aiko 0.1.0

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.
Files changed (317) hide show
  1. package/README.md +46 -0
  2. package/dist/cli.d.ts +2 -0
  3. package/dist/cli.js +7 -0
  4. package/dist/create.d.ts +1 -0
  5. package/dist/create.js +174 -0
  6. package/dist/index.d.ts +2 -0
  7. package/dist/index.js +2 -0
  8. package/dist/scaffold.d.ts +7 -0
  9. package/dist/scaffold.js +310 -0
  10. package/package.json +37 -0
  11. package/template/README.md +63 -0
  12. package/template/docs/better-sqlite3-design.md +101 -0
  13. package/template/docs/shared-vs-shared-auth.md +54 -0
  14. package/template/package.json +23 -0
  15. package/template/packages/admin/.env.dev +1 -0
  16. package/template/packages/admin/.env.prod +4 -0
  17. package/template/packages/admin/.env.stage +4 -0
  18. package/template/packages/admin/Dockerfile +37 -0
  19. package/template/packages/admin/README.MD +27 -0
  20. package/template/packages/admin/components.json +21 -0
  21. package/template/packages/admin/eslint.config.js +28 -0
  22. package/template/packages/admin/index.html +50 -0
  23. package/template/packages/admin/package.json +100 -0
  24. package/template/packages/admin/public/vite.svg +1 -0
  25. package/template/packages/admin/src/App.css +82 -0
  26. package/template/packages/admin/src/App.tsx +128 -0
  27. package/template/packages/admin/src/app.config.ts +3 -0
  28. package/template/packages/admin/src/components/admin-ui/approval-flow.tsx +205 -0
  29. package/template/packages/admin/src/components/admin-ui/data-table/data-table-pagination.tsx +148 -0
  30. package/template/packages/admin/src/components/admin-ui/data-table/data-table-sorter.tsx +50 -0
  31. package/template/packages/admin/src/components/admin-ui/data-table/index.tsx +296 -0
  32. package/template/packages/admin/src/components/admin-ui/editable-table.tsx +292 -0
  33. package/template/packages/admin/src/components/admin-ui/form/input-password.tsx +38 -0
  34. package/template/packages/admin/src/components/admin-ui/form/sign-in-form.tsx +104 -0
  35. package/template/packages/admin/src/components/admin-ui/layout/error-component.tsx +71 -0
  36. package/template/packages/admin/src/components/admin-ui/layout/header.tsx +148 -0
  37. package/template/packages/admin/src/components/admin-ui/layout/language-switcher.tsx +47 -0
  38. package/template/packages/admin/src/components/admin-ui/layout/layout.tsx +42 -0
  39. package/template/packages/admin/src/components/admin-ui/layout/loading-overlay.tsx +36 -0
  40. package/template/packages/admin/src/components/admin-ui/layout/shell-bar.tsx +232 -0
  41. package/template/packages/admin/src/components/admin-ui/layout/sidebar.tsx +193 -0
  42. package/template/packages/admin/src/components/admin-ui/layout/user-avatar.tsx +31 -0
  43. package/template/packages/admin/src/components/admin-ui/list-report.tsx +313 -0
  44. package/template/packages/admin/src/components/admin-ui/master-detail.tsx +382 -0
  45. package/template/packages/admin/src/components/admin-ui/notification/toaster.tsx +23 -0
  46. package/template/packages/admin/src/components/admin-ui/notification/undoable-notification.tsx +84 -0
  47. package/template/packages/admin/src/components/admin-ui/object-page.tsx +539 -0
  48. package/template/packages/admin/src/components/admin-ui/process-stepper.tsx +204 -0
  49. package/template/packages/admin/src/components/admin-ui/theme/theme-provider.tsx +160 -0
  50. package/template/packages/admin/src/components/admin-ui/theme/theme-select.tsx +129 -0
  51. package/template/packages/admin/src/components/admin-ui/theme/theme-toggle.tsx +90 -0
  52. package/template/packages/admin/src/components/admin-ui/timeline.tsx +137 -0
  53. package/template/packages/admin/src/components/admin-ui/tree-navigator.tsx +243 -0
  54. package/template/packages/admin/src/components/ui/accordion.tsx +64 -0
  55. package/template/packages/admin/src/components/ui/alert-dialog.tsx +157 -0
  56. package/template/packages/admin/src/components/ui/alert.tsx +66 -0
  57. package/template/packages/admin/src/components/ui/aspect-ratio.tsx +9 -0
  58. package/template/packages/admin/src/components/ui/avatar.tsx +53 -0
  59. package/template/packages/admin/src/components/ui/badge.tsx +47 -0
  60. package/template/packages/admin/src/components/ui/breadcrumb.tsx +111 -0
  61. package/template/packages/admin/src/components/ui/button.tsx +59 -0
  62. package/template/packages/admin/src/components/ui/calendar.tsx +74 -0
  63. package/template/packages/admin/src/components/ui/card.tsx +92 -0
  64. package/template/packages/admin/src/components/ui/carousel.tsx +237 -0
  65. package/template/packages/admin/src/components/ui/chart.tsx +351 -0
  66. package/template/packages/admin/src/components/ui/checkbox.tsx +32 -0
  67. package/template/packages/admin/src/components/ui/collapsible.tsx +33 -0
  68. package/template/packages/admin/src/components/ui/command.tsx +182 -0
  69. package/template/packages/admin/src/components/ui/context-menu.tsx +252 -0
  70. package/template/packages/admin/src/components/ui/dialog.tsx +141 -0
  71. package/template/packages/admin/src/components/ui/drawer.tsx +130 -0
  72. package/template/packages/admin/src/components/ui/dropdown-menu.tsx +255 -0
  73. package/template/packages/admin/src/components/ui/form.tsx +166 -0
  74. package/template/packages/admin/src/components/ui/hover-card.tsx +42 -0
  75. package/template/packages/admin/src/components/ui/input-otp.tsx +77 -0
  76. package/template/packages/admin/src/components/ui/input.tsx +21 -0
  77. package/template/packages/admin/src/components/ui/label.tsx +22 -0
  78. package/template/packages/admin/src/components/ui/menubar.tsx +274 -0
  79. package/template/packages/admin/src/components/ui/navigation-menu.tsx +168 -0
  80. package/template/packages/admin/src/components/ui/pagination.tsx +127 -0
  81. package/template/packages/admin/src/components/ui/popover.tsx +48 -0
  82. package/template/packages/admin/src/components/ui/progress.tsx +29 -0
  83. package/template/packages/admin/src/components/ui/radio-group.tsx +45 -0
  84. package/template/packages/admin/src/components/ui/resizable.tsx +54 -0
  85. package/template/packages/admin/src/components/ui/scroll-area.tsx +58 -0
  86. package/template/packages/admin/src/components/ui/select.tsx +183 -0
  87. package/template/packages/admin/src/components/ui/separator.tsx +26 -0
  88. package/template/packages/admin/src/components/ui/sheet.tsx +139 -0
  89. package/template/packages/admin/src/components/ui/sidebar.tsx +740 -0
  90. package/template/packages/admin/src/components/ui/skeleton.tsx +13 -0
  91. package/template/packages/admin/src/components/ui/slider.tsx +63 -0
  92. package/template/packages/admin/src/components/ui/sonner.tsx +23 -0
  93. package/template/packages/admin/src/components/ui/switch.tsx +31 -0
  94. package/template/packages/admin/src/components/ui/table.tsx +114 -0
  95. package/template/packages/admin/src/components/ui/tabs.tsx +66 -0
  96. package/template/packages/admin/src/components/ui/textarea.tsx +18 -0
  97. package/template/packages/admin/src/components/ui/toggle-group.tsx +73 -0
  98. package/template/packages/admin/src/components/ui/toggle.tsx +45 -0
  99. package/template/packages/admin/src/components/ui/tooltip.tsx +59 -0
  100. package/template/packages/admin/src/hooks/use-mobile.ts +21 -0
  101. package/template/packages/admin/src/i18n.ts +20 -0
  102. package/template/packages/admin/src/index.tsx +18 -0
  103. package/template/packages/admin/src/layouts/menu-layout.tsx +211 -0
  104. package/template/packages/admin/src/layouts/tile-layout.tsx +355 -0
  105. package/template/packages/admin/src/lib/utils.ts +6 -0
  106. package/template/packages/admin/src/locales/en.json +68 -0
  107. package/template/packages/admin/src/locales/zh.json +68 -0
  108. package/template/packages/admin/src/pages/dashboard.tsx +12 -0
  109. package/template/packages/admin/src/pages/goods-receipt/CreatePage.tsx +302 -0
  110. package/template/packages/admin/src/pages/goods-receipt/EditPage.tsx +221 -0
  111. package/template/packages/admin/src/pages/goods-receipt/ListPage.tsx +283 -0
  112. package/template/packages/admin/src/pages/goods-receipt/ViewPage.tsx +280 -0
  113. package/template/packages/admin/src/pages/goods-receipt/index.ts +4 -0
  114. package/template/packages/admin/src/pages/home-page.tsx +244 -0
  115. package/template/packages/admin/src/pages/login-page.tsx +91 -0
  116. package/template/packages/admin/src/pages/master-data/cost-centers/index.tsx +461 -0
  117. package/template/packages/admin/src/pages/master-data/currencies/index.tsx +255 -0
  118. package/template/packages/admin/src/pages/master-data/materials/ListPage.tsx +271 -0
  119. package/template/packages/admin/src/pages/master-data/materials/ViewPage.tsx +240 -0
  120. package/template/packages/admin/src/pages/master-data/materials/index.ts +2 -0
  121. package/template/packages/admin/src/pages/master-data/plants/ListPage.tsx +279 -0
  122. package/template/packages/admin/src/pages/master-data/plants/ViewPage.tsx +380 -0
  123. package/template/packages/admin/src/pages/master-data/plants/index.ts +2 -0
  124. package/template/packages/admin/src/pages/master-data/purchase-organizations/index.tsx +341 -0
  125. package/template/packages/admin/src/pages/master-data/units-of-measure/index.tsx +295 -0
  126. package/template/packages/admin/src/pages/master-data/vendors/ListPage.tsx +266 -0
  127. package/template/packages/admin/src/pages/master-data/vendors/ViewPage.tsx +274 -0
  128. package/template/packages/admin/src/pages/master-data/vendors/index.ts +2 -0
  129. package/template/packages/admin/src/pages/placeholder-page.tsx +13 -0
  130. package/template/packages/admin/src/pages/purchase-orders/ListPage.tsx +289 -0
  131. package/template/packages/admin/src/pages/purchase-orders/ViewPage.tsx +343 -0
  132. package/template/packages/admin/src/pages/purchase-orders/index.ts +2 -0
  133. package/template/packages/admin/src/pages/purchase-requisitions/CreatePage.tsx +398 -0
  134. package/template/packages/admin/src/pages/purchase-requisitions/EditPage.tsx +473 -0
  135. package/template/packages/admin/src/pages/purchase-requisitions/ListPage.tsx +307 -0
  136. package/template/packages/admin/src/pages/purchase-requisitions/ViewPage.tsx +304 -0
  137. package/template/packages/admin/src/pages/purchase-requisitions/constants.ts +51 -0
  138. package/template/packages/admin/src/pages/purchase-requisitions/index.ts +4 -0
  139. package/template/packages/admin/src/pages/reports/PurchaseOrderReport.tsx +312 -0
  140. package/template/packages/admin/src/pages/reports/PurchaseRequisitionReport.tsx +303 -0
  141. package/template/packages/admin/src/pages/reports/index.ts +2 -0
  142. package/template/packages/admin/src/pages/settings-page.tsx +335 -0
  143. package/template/packages/admin/src/providers/app-config.tsx +50 -0
  144. package/template/packages/admin/src/routes/auth.ts +6 -0
  145. package/template/packages/admin/src/routes/index.ts +85 -0
  146. package/template/packages/admin/src/routes/menu.ts +176 -0
  147. package/template/packages/admin/src/routes/modules/goods-receipt.ts +31 -0
  148. package/template/packages/admin/src/routes/modules/master-data.ts +41 -0
  149. package/template/packages/admin/src/routes/modules/purchase-orders.ts +27 -0
  150. package/template/packages/admin/src/routes/modules/purchase-requisitions.ts +39 -0
  151. package/template/packages/admin/src/routes/modules/reports.ts +33 -0
  152. package/template/packages/admin/src/routes/modules/settings.ts +19 -0
  153. package/template/packages/admin/src/routes/withSuspense.tsx +21 -0
  154. package/template/packages/admin/src/theme/amber.css +27 -0
  155. package/template/packages/admin/src/theme/blue.css +27 -0
  156. package/template/packages/admin/src/theme/default.css +75 -0
  157. package/template/packages/admin/src/theme/fiori.css +180 -0
  158. package/template/packages/admin/src/theme/green.css +27 -0
  159. package/template/packages/admin/src/theme/index.css +12 -0
  160. package/template/packages/admin/src/theme/rose.css +27 -0
  161. package/template/packages/admin/src/theme/violet.css +27 -0
  162. package/template/packages/admin/src/vite-env.d.ts +1 -0
  163. package/template/packages/admin/tsconfig.json +28 -0
  164. package/template/packages/admin/tsconfig.node.json +21 -0
  165. package/template/packages/admin/vite.config.ts +26 -0
  166. package/template/packages/api/.eslintrc.json +6 -0
  167. package/template/packages/api/.swcrc +17 -0
  168. package/template/packages/api/app.config.ts +160 -0
  169. package/template/packages/api/docs/api-document.md +497 -0
  170. package/template/packages/api/examples/security/README.md +664 -0
  171. package/template/packages/api/examples/security/complete/.env.example +26 -0
  172. package/template/packages/api/examples/security/complete/PROJECT_STRUCTURE.md +220 -0
  173. package/template/packages/api/examples/security/complete/README.md +847 -0
  174. package/template/packages/api/examples/security/complete/app.config.ts +69 -0
  175. package/template/packages/api/examples/security/complete/app.ts +63 -0
  176. package/template/packages/api/examples/security/complete/controller/auth.controller.ts +131 -0
  177. package/template/packages/api/examples/security/complete/controller/index.ts +4 -0
  178. package/template/packages/api/examples/security/complete/controller/permission.controller.ts +41 -0
  179. package/template/packages/api/examples/security/complete/controller/role.controller.ts +53 -0
  180. package/template/packages/api/examples/security/complete/controller/user.controller.ts +53 -0
  181. package/template/packages/api/examples/security/complete/dto/change-password.dto.ts +10 -0
  182. package/template/packages/api/examples/security/complete/dto/create-permission.dto.ts +14 -0
  183. package/template/packages/api/examples/security/complete/dto/create-role.dto.ts +11 -0
  184. package/template/packages/api/examples/security/complete/dto/create-user.dto.ts +15 -0
  185. package/template/packages/api/examples/security/complete/dto/index.ts +7 -0
  186. package/template/packages/api/examples/security/complete/dto/login.dto.ts +10 -0
  187. package/template/packages/api/examples/security/complete/dto/oauth-profile.dto.ts +7 -0
  188. package/template/packages/api/examples/security/complete/dto/register.dto.ts +17 -0
  189. package/template/packages/api/examples/security/complete/entity/index.ts +6 -0
  190. package/template/packages/api/examples/security/complete/entity/oauth-account.entity.ts +39 -0
  191. package/template/packages/api/examples/security/complete/entity/permission.entity.ts +31 -0
  192. package/template/packages/api/examples/security/complete/entity/role-permission.entity.ts +19 -0
  193. package/template/packages/api/examples/security/complete/entity/role.entity.ts +25 -0
  194. package/template/packages/api/examples/security/complete/entity/user-role.entity.ts +19 -0
  195. package/template/packages/api/examples/security/complete/entity/user.entity.ts +46 -0
  196. package/template/packages/api/examples/security/complete/init.sql +81 -0
  197. package/template/packages/api/examples/security/complete/middleware/auth.interceptor.ts +39 -0
  198. package/template/packages/api/examples/security/complete/middleware/index.ts +2 -0
  199. package/template/packages/api/examples/security/complete/middleware/permission.interceptor.ts +61 -0
  200. package/template/packages/api/examples/security/complete/package.json +54 -0
  201. package/template/packages/api/examples/security/complete/seed.sql +42 -0
  202. package/template/packages/api/examples/security/complete/service/auth.service.ts +41 -0
  203. package/template/packages/api/examples/security/complete/service/index.ts +5 -0
  204. package/template/packages/api/examples/security/complete/service/oauth.service.ts +82 -0
  205. package/template/packages/api/examples/security/complete/service/permission.service.ts +113 -0
  206. package/template/packages/api/examples/security/complete/service/role.service.ts +85 -0
  207. package/template/packages/api/examples/security/complete/service/user.service.ts +132 -0
  208. package/template/packages/api/examples/security/complete/tests/TEST_REPORT.md +318 -0
  209. package/template/packages/api/examples/security/complete/tests/generate-report.js +335 -0
  210. package/template/packages/api/examples/security/complete/tests/helpers/api-helpers.ts +116 -0
  211. package/template/packages/api/examples/security/complete/tests/helpers/index.ts +2 -0
  212. package/template/packages/api/examples/security/complete/tests/helpers/test-helpers.ts +129 -0
  213. package/template/packages/api/examples/security/complete/tests/integration/auth.api.test.ts +429 -0
  214. package/template/packages/api/examples/security/complete/tests/integration/role.api.test.ts +400 -0
  215. package/template/packages/api/examples/security/complete/tests/integration/user.api.test.ts +459 -0
  216. package/template/packages/api/examples/security/complete/tests/jest.config.js +40 -0
  217. package/template/packages/api/examples/security/complete/tests/run-all-tests.js +135 -0
  218. package/template/packages/api/examples/security/complete/tests/run-tests.js +109 -0
  219. package/template/packages/api/examples/security/complete/tests/setup.ts +19 -0
  220. package/template/packages/api/examples/security/complete/tests/unit/auth.service.test.ts +199 -0
  221. package/template/packages/api/examples/security/complete/tests/unit/permission.service.test.ts +377 -0
  222. package/template/packages/api/examples/security/complete/tests/unit/user.service.test.ts +288 -0
  223. package/template/packages/api/examples/security/complete/tsconfig.json +35 -0
  224. package/template/packages/api/examples/security/jwt/README.md +424 -0
  225. package/template/packages/api/examples/security/local/README.md +499 -0
  226. package/template/packages/api/examples/security/oauth2/README.md +637 -0
  227. package/template/packages/api/examples/security/permission/README.md +943 -0
  228. package/template/packages/api/examples/security/session/README.md +753 -0
  229. package/template/packages/api/package.json +56 -0
  230. package/template/packages/api/src/controller/auth.controller.ts +127 -0
  231. package/template/packages/api/src/controller/cache.controller.ts +106 -0
  232. package/template/packages/api/src/controller/menu.controller.ts +46 -0
  233. package/template/packages/api/src/controller/mq.controller.ts +35 -0
  234. package/template/packages/api/src/controller/role.controller.ts +51 -0
  235. package/template/packages/api/src/controller/upload.controller.ts +85 -0
  236. package/template/packages/api/src/controller/user.controller.ts +57 -0
  237. package/template/packages/api/src/dto/auth.dto.ts +30 -0
  238. package/template/packages/api/src/dto/cache.dto.ts +24 -0
  239. package/template/packages/api/src/dto/menu.dto.ts +37 -0
  240. package/template/packages/api/src/dto/mq.dto.ts +16 -0
  241. package/template/packages/api/src/dto/role.dto.ts +16 -0
  242. package/template/packages/api/src/dto/user.dto.ts +35 -0
  243. package/template/packages/api/src/entity/menu.entity.ts +34 -0
  244. package/template/packages/api/src/entity/role-menu.entity.ts +13 -0
  245. package/template/packages/api/src/entity/role.entity.ts +22 -0
  246. package/template/packages/api/src/entity/user-role.entity.ts +13 -0
  247. package/template/packages/api/src/entity/user.entity.ts +31 -0
  248. package/template/packages/api/src/mapper/menu.mapper.ts +6 -0
  249. package/template/packages/api/src/mapper/role-menu.mapper.ts +6 -0
  250. package/template/packages/api/src/mapper/role.mapper.ts +6 -0
  251. package/template/packages/api/src/mapper/user-role.mapper.ts +6 -0
  252. package/template/packages/api/src/mapper/user.mapper.ts +11 -0
  253. package/template/packages/api/src/scripts/init-db.ts +185 -0
  254. package/template/packages/api/src/server.ts +76 -0
  255. package/template/packages/api/src/service/auth.service.ts +106 -0
  256. package/template/packages/api/src/service/cache.service.ts +80 -0
  257. package/template/packages/api/src/service/log.request.service.ts +158 -0
  258. package/template/packages/api/src/service/log.service.ts +123 -0
  259. package/template/packages/api/src/service/menu.service.ts +94 -0
  260. package/template/packages/api/src/service/mq.consumer.service.ts +26 -0
  261. package/template/packages/api/src/service/role.service.ts +88 -0
  262. package/template/packages/api/src/service/user.service.ts +170 -0
  263. package/template/packages/api/src/types/sqljs.d.ts +18 -0
  264. package/template/packages/api/src/utils/auth.utils.js +0 -0
  265. package/template/packages/api/src/utils/jwt.util.ts +29 -0
  266. package/template/packages/api/tsconfig.json +17 -0
  267. package/template/packages/api/tsup.config.ts +9 -0
  268. package/template/packages/api/uploads/.gitkeep +0 -0
  269. package/template/packages/core/package.json +31 -0
  270. package/template/packages/core/src/auth/auth-client-middleware.ts +22 -0
  271. package/template/packages/core/src/auth/auth-service.ts +65 -0
  272. package/template/packages/core/src/auth/default-auth-provider.ts +37 -0
  273. package/template/packages/core/src/auth/index.ts +10 -0
  274. package/template/packages/core/src/auth/types.ts +38 -0
  275. package/template/packages/core/src/authorization/authorization-client-middleware.ts +38 -0
  276. package/template/packages/core/src/authorization/authorization-config.ts +13 -0
  277. package/template/packages/core/src/authorization/authorization-provider.tsx +116 -0
  278. package/template/packages/core/src/authorization/default-authorization-provider.ts +26 -0
  279. package/template/packages/core/src/authorization/index.ts +15 -0
  280. package/template/packages/core/src/authorization/types.ts +42 -0
  281. package/template/packages/core/src/index.ts +3 -0
  282. package/template/packages/core/src/utils/promise-result-cache.ts +18 -0
  283. package/template/packages/core/tsconfig.json +19 -0
  284. package/template/packages/mobile/README.md +55 -0
  285. package/template/packages/mobile/index.html +13 -0
  286. package/template/packages/mobile/package.json +30 -0
  287. package/template/packages/mobile/postcss.config.mjs +7 -0
  288. package/template/packages/mobile/src/App.tsx +5 -0
  289. package/template/packages/mobile/src/app/globals.css +1 -0
  290. package/template/packages/mobile/src/components/LoginForm.tsx +76 -0
  291. package/template/packages/mobile/src/hooks/index.ts +5 -0
  292. package/template/packages/mobile/src/lib/utils.ts +7 -0
  293. package/template/packages/mobile/src/main.tsx +13 -0
  294. package/template/packages/mobile/src/pages/HomePage.tsx +35 -0
  295. package/template/packages/mobile/src/pages/LoginPage.tsx +35 -0
  296. package/template/packages/mobile/src/pages/index.ts +2 -0
  297. package/template/packages/mobile/src/routes/ProtectedRoute.tsx +29 -0
  298. package/template/packages/mobile/src/routes/index.tsx +24 -0
  299. package/template/packages/mobile/src/routes/routes.ts +11 -0
  300. package/template/packages/mobile/src/types/index.ts +5 -0
  301. package/template/packages/mobile/src/vite-env.d.ts +1 -0
  302. package/template/packages/mobile/tsconfig.json +23 -0
  303. package/template/packages/mobile/tsconfig.node.json +11 -0
  304. package/template/packages/mobile/vite.config.ts +19 -0
  305. package/template/packages/shared/package.json +20 -0
  306. package/template/packages/shared/src/constants.ts +8 -0
  307. package/template/packages/shared/src/index.ts +5 -0
  308. package/template/packages/shared/tsconfig.json +13 -0
  309. package/template/packages/shared-auth/package.json +29 -0
  310. package/template/packages/shared-auth/src/AuthContext.tsx +177 -0
  311. package/template/packages/shared-auth/src/AuthProviderWrapper.tsx +29 -0
  312. package/template/packages/shared-auth/src/index.ts +10 -0
  313. package/template/packages/shared-auth/tsconfig.json +14 -0
  314. package/template/pnpm-lock.yaml +16349 -0
  315. package/template/pnpm-workspace.yaml +3 -0
  316. package/template/scripts/postinstall.cjs +42 -0
  317. package/template/scripts/rebuild-sqlite.cjs +23 -0
@@ -0,0 +1,16 @@
1
+ import { IsNotEmpty, IsEmail, IsOptional, Length, Min, Max, IsInt } from '@ai-partner-x/aiko-boot-starter-validation';
2
+ export class CreateRoleDto {
3
+ @IsNotEmpty({ message: '用户名不能为空' })
4
+ roleCode!: string;
5
+ roleName!: string;
6
+ description?: string;
7
+ status: number = 1;
8
+ menuIds?: number[];
9
+ }
10
+
11
+ export class UpdateRoleDto {
12
+ roleName?: string;
13
+ description?: string;
14
+ status?: number;
15
+ menuIds?: number[];
16
+ }
@@ -0,0 +1,35 @@
1
+ export class CreateUserDto {
2
+ username!: string;
3
+ password!: string;
4
+ realName?: string;
5
+ email?: string;
6
+ phone?: string;
7
+ status: number = 1;
8
+ roleIds?: number[];
9
+ }
10
+
11
+ export class UpdateUserDto {
12
+ realName?: string;
13
+ email?: string;
14
+ phone?: string;
15
+ status?: number;
16
+ roleIds?: number[];
17
+ }
18
+
19
+ export class UserPageDto {
20
+ pageNo: number = 1;
21
+ pageSize: number = 10;
22
+ username?: string;
23
+ status?: number;
24
+ }
25
+
26
+ export interface UserVo {
27
+ id: number;
28
+ username: string;
29
+ realName?: string;
30
+ email?: string;
31
+ phone?: string;
32
+ status: number;
33
+ roles: string[];
34
+ createdAt?: Date;
35
+ }
@@ -0,0 +1,34 @@
1
+ import { Entity, TableId, TableField } from '@ai-partner-x/aiko-boot-starter-orm';
2
+
3
+ @Entity({ tableName: 'sys_menu' })
4
+ export class Menu {
5
+ @TableId({ type: 'AUTO' })
6
+ id!: number;
7
+
8
+ @TableField({ column: 'parent_id' })
9
+ parentId!: number; // 0 为顶级
10
+
11
+ @TableField({ column: 'menu_name' })
12
+ menuName!: string;
13
+
14
+ @TableField({ column: 'menu_type' })
15
+ menuType!: number; // 1目录 2菜单 3按钮
16
+
17
+ @TableField({ column: 'path' })
18
+ path?: string;
19
+
20
+ @TableField({ column: 'component' })
21
+ component?: string;
22
+
23
+ @TableField({ column: 'permission' })
24
+ permission?: string; // 权限标识,如 sys:user:list
25
+
26
+ @TableField({ column: 'icon' })
27
+ icon?: string;
28
+
29
+ @TableField({ column: 'sort_order' })
30
+ sortOrder!: number;
31
+
32
+ @TableField({ column: 'status' })
33
+ status!: number;
34
+ }
@@ -0,0 +1,13 @@
1
+ import { Entity, TableId, TableField } from '@ai-partner-x/aiko-boot-starter-orm';
2
+
3
+ @Entity({ tableName: 'sys_role_menu' })
4
+ export class RoleMenu {
5
+ @TableId({ type: 'AUTO' })
6
+ id!: number;
7
+
8
+ @TableField({ column: 'role_id' })
9
+ roleId!: number;
10
+
11
+ @TableField({ column: 'menu_id' })
12
+ menuId!: number;
13
+ }
@@ -0,0 +1,22 @@
1
+ import { Entity, TableId, TableField } from '@ai-partner-x/aiko-boot-starter-orm';
2
+
3
+ @Entity({ tableName: 'sys_role' })
4
+ export class Role {
5
+ @TableId({ type: 'AUTO' })
6
+ id!: number;
7
+
8
+ @TableField({ column: 'role_code' })
9
+ roleCode!: string;
10
+
11
+ @TableField({ column: 'role_name' })
12
+ roleName!: string;
13
+
14
+ @TableField({ column: 'description' })
15
+ description?: string;
16
+
17
+ @TableField({ column: 'status' })
18
+ status!: number;
19
+
20
+ @TableField({ column: 'created_at' })
21
+ createdAt?: Date;
22
+ }
@@ -0,0 +1,13 @@
1
+ import { Entity, TableId, TableField } from '@ai-partner-x/aiko-boot-starter-orm';
2
+
3
+ @Entity({ tableName: 'sys_user_role' })
4
+ export class UserRole {
5
+ @TableId({ type: 'AUTO' })
6
+ id!: number;
7
+
8
+ @TableField({ column: 'user_id' })
9
+ userId!: number;
10
+
11
+ @TableField({ column: 'role_id' })
12
+ roleId!: number;
13
+ }
@@ -0,0 +1,31 @@
1
+ import { Entity, TableId, TableField } from '@ai-partner-x/aiko-boot-starter-orm';
2
+
3
+ @Entity({ tableName: 'sys_user' })
4
+ export class User {
5
+ @TableId({ type: 'AUTO' })
6
+ id!: number;
7
+
8
+ @TableField({ column: 'user_name' })
9
+ username!: string;
10
+
11
+ @TableField({ column: 'password_hash' })
12
+ passwordHash!: string;
13
+
14
+ @TableField({ column: 'real_name' })
15
+ realName?: string;
16
+
17
+ @TableField({ column: 'email' })
18
+ email!: string;
19
+
20
+ @TableField({ column: 'phone' })
21
+ phone?: string;
22
+
23
+ @TableField({ column: 'status' })
24
+ status!: number; // 1启用 0禁用
25
+
26
+ @TableField({ column: 'created_at' })
27
+ createdAt?: Date;
28
+
29
+ @TableField({ column: 'updated_at' })
30
+ updatedAt?: Date;
31
+ }
@@ -0,0 +1,6 @@
1
+ import 'reflect-metadata';
2
+ import { Mapper, BaseMapper } from '@ai-partner-x/aiko-boot-starter-orm';
3
+ import { Menu } from '../entity/menu.entity.js';
4
+
5
+ @Mapper(Menu)
6
+ export class MenuMapper extends BaseMapper<Menu> {}
@@ -0,0 +1,6 @@
1
+ import 'reflect-metadata';
2
+ import { Mapper, BaseMapper } from '@ai-partner-x/aiko-boot-starter-orm';
3
+ import { RoleMenu } from '../entity/role-menu.entity.js';
4
+
5
+ @Mapper(RoleMenu)
6
+ export class RoleMenuMapper extends BaseMapper<RoleMenu> {}
@@ -0,0 +1,6 @@
1
+ import 'reflect-metadata';
2
+ import { Mapper, BaseMapper } from '@ai-partner-x/aiko-boot-starter-orm';
3
+ import { Role } from '../entity/role.entity.js';
4
+
5
+ @Mapper(Role)
6
+ export class RoleMapper extends BaseMapper<Role> {}
@@ -0,0 +1,6 @@
1
+ import 'reflect-metadata';
2
+ import { Mapper, BaseMapper } from '@ai-partner-x/aiko-boot-starter-orm';
3
+ import { UserRole } from '../entity/user-role.entity.js';
4
+
5
+ @Mapper(UserRole)
6
+ export class UserRoleMapper extends BaseMapper<UserRole> {}
@@ -0,0 +1,11 @@
1
+ import 'reflect-metadata';
2
+ import { Mapper, BaseMapper } from '@ai-partner-x/aiko-boot-starter-orm';
3
+ import { User } from '../entity/user.entity.js';
4
+
5
+ @Mapper(User)
6
+ export class UserMapper extends BaseMapper<User> {
7
+ async selectByUsername(username: string): Promise<User | null> {
8
+ const user = await this.selectOne({ username });
9
+ return user;
10
+ }
11
+ }
@@ -0,0 +1,185 @@
1
+ /**
2
+ * 初始化数据库表结构和默认数据
3
+ * 运行: pnpm init-db
4
+ */
5
+ import 'reflect-metadata';
6
+ import { fileURLToPath } from 'url';
7
+ import { pathToFileURL } from 'url';
8
+ import { dirname, join } from 'path';
9
+ import { mkdirSync, existsSync, readdirSync, statSync } from 'fs';
10
+ import { createKyselyDatabase, getKyselyDatabase } from '@ai-partner-x/aiko-boot-starter-orm';
11
+ import bcrypt from 'bcryptjs';
12
+
13
+ const __dirname = dirname(fileURLToPath(import.meta.url));
14
+ const dbPath = join(__dirname, '../../data/app.db');
15
+ const dir = dirname(dbPath);
16
+ if (!existsSync(dir)) {
17
+ mkdirSync(dir, { recursive: true });
18
+ }
19
+
20
+ await createKyselyDatabase({
21
+ type: 'sqlite',
22
+ filename: join(__dirname, '../../data/app.db'),
23
+ });
24
+
25
+ const db = getKyselyDatabase();
26
+
27
+ // 建表
28
+ await db.schema.createTable('sys_user').ifNotExists()
29
+ .addColumn('id', 'integer', (col: any) => col.primaryKey().autoIncrement())
30
+ .addColumn('user_name', 'varchar(50)', (col: any) => col.notNull().unique())
31
+ .addColumn('password_hash', 'varchar(255)', (col: any) => col.notNull())
32
+ .addColumn('real_name', 'varchar(50)')
33
+ .addColumn('email', 'varchar(100)')
34
+ .addColumn('phone', 'varchar(20)')
35
+ .addColumn('status', 'integer', (col: any) => col.notNull().defaultTo(1))
36
+ .addColumn('created_at', 'datetime')
37
+ .addColumn('updated_at', 'datetime')
38
+ .execute();
39
+
40
+ await db.schema.createTable('sys_role').ifNotExists()
41
+ .addColumn('id', 'integer', (col: any) => col.primaryKey().autoIncrement())
42
+ .addColumn('role_code', 'varchar(50)', (col: any) => col.notNull().unique())
43
+ .addColumn('role_name', 'varchar(50)', (col: any) => col.notNull())
44
+ .addColumn('description', 'varchar(255)')
45
+ .addColumn('status', 'integer', (col: any) => col.notNull().defaultTo(1))
46
+ .addColumn('created_at', 'datetime')
47
+ .execute();
48
+
49
+ await db.schema.createTable('sys_menu').ifNotExists()
50
+ .addColumn('id', 'integer', (col: any) => col.primaryKey().autoIncrement())
51
+ .addColumn('parent_id', 'integer', (col: any) => col.notNull().defaultTo(0))
52
+ .addColumn('menu_name', 'varchar(50)', (col: any) => col.notNull())
53
+ .addColumn('menu_type', 'integer', (col: any) => col.notNull())
54
+ .addColumn('path', 'varchar(255)')
55
+ .addColumn('component', 'varchar(255)')
56
+ .addColumn('permission', 'varchar(100)')
57
+ .addColumn('icon', 'varchar(100)')
58
+ .addColumn('sort_order', 'integer', (col: any) => col.notNull().defaultTo(0))
59
+ .addColumn('status', 'integer', (col: any) => col.notNull().defaultTo(1))
60
+ .execute();
61
+
62
+ await db.schema.createTable('sys_user_role').ifNotExists()
63
+ .addColumn('id', 'integer', (col: any) => col.primaryKey().autoIncrement())
64
+ .addColumn('user_id', 'integer', (col: any) => col.notNull())
65
+ .addColumn('role_id', 'integer', (col: any) => col.notNull())
66
+ .execute();
67
+
68
+ await db.schema.createTable('sys_role_menu').ifNotExists()
69
+ .addColumn('id', 'integer', (col: any) => col.primaryKey().autoIncrement())
70
+ .addColumn('role_id', 'integer', (col: any) => col.notNull())
71
+ .addColumn('menu_id', 'integer', (col: any) => col.notNull())
72
+ .execute();
73
+
74
+ console.log('✅ 表结构创建完成');
75
+
76
+ // 初始化超级管理员角色
77
+ const roles = await db.selectFrom('sys_role').where('role_code', '=', 'SUPER_ADMIN').selectAll().execute();
78
+ let adminRoleId: number;
79
+ if (!roles.length) {
80
+ const result = await db.insertInto('sys_role')
81
+ .values({ role_code: 'SUPER_ADMIN', role_name: '超级管理员', description: '拥有全部权限', status: 1, created_at: new Date().toISOString() })
82
+ .executeTakeFirst();
83
+ adminRoleId = Number(result.insertId);
84
+ console.log('✅ 超级管理员角色创建完成');
85
+ } else {
86
+ adminRoleId = roles[0].id as number;
87
+ }
88
+
89
+ // 初始化默认菜单
90
+ const menuCount = await db.selectFrom('sys_menu').select(db.fn.count('id').as('cnt')).executeTakeFirst();
91
+ const cntValue = menuCount && menuCount.cnt;
92
+ if (Number(cntValue) === 0) {
93
+ // 系统管理目录
94
+ const sysDir = await db.insertInto('sys_menu')
95
+ .values({ parent_id: 0, menu_name: '系统管理', menu_type: 1, icon: 'Settings', sort_order: 100, status: 1 })
96
+ .executeTakeFirst();
97
+ const sysDirId = Number(sysDir.insertId);
98
+
99
+ const menus = [
100
+ { parent_id: sysDirId, menu_name: '用户管理', menu_type: 2, path: '/sys/user', permission: 'sys:user:list', sort_order: 1, status: 1 },
101
+ { parent_id: sysDirId, menu_name: '角色管理', menu_type: 2, path: '/sys/role', permission: 'sys:role:list', sort_order: 2, status: 1 },
102
+ { parent_id: sysDirId, menu_name: '菜单管理', menu_type: 2, path: '/sys/menu', permission: 'sys:menu:list', sort_order: 3, status: 1 },
103
+ ];
104
+ const insertedMenus = await db.insertInto('sys_menu').values(menus).executeTakeFirst();
105
+ console.log('✅ 默认菜单创建完成');
106
+
107
+ // 给超级管理员角色分配所有菜单
108
+ const allMenus = await db.selectFrom('sys_menu').select('id').execute();
109
+ for (const m of allMenus) {
110
+ await db.insertInto('sys_role_menu').values({ role_id: adminRoleId, menu_id: m.id as number }).execute();
111
+ }
112
+ }
113
+
114
+ // 初始化 admin 账号
115
+ const users = await db.selectFrom('sys_user').where('user_name', '=', 'admin').selectAll().execute();
116
+ if (!users.length) {
117
+ const hashed = await bcrypt.hash('admin123', 10);
118
+ const result = await db.insertInto('sys_user')
119
+ .values({ user_name: 'admin', password_hash: hashed, email: 'admin@example.com', real_name: '超级管理员', status: 1, created_at: new Date().toISOString(), updated_at: new Date().toISOString() })
120
+ .executeTakeFirst();
121
+ const adminUserId = Number(result.insertId);
122
+ await db.insertInto('sys_user_role').values({ user_id: adminUserId, role_id: adminRoleId }).execute();
123
+ console.log('✅ admin 账号创建完成 (密码: Admin@123)');
124
+ } else {
125
+ console.log('ℹ️ admin 账号已存在,跳过');
126
+ }
127
+
128
+ await runGeneratedModuleInitializers(db);
129
+
130
+ console.log('\n🎉 数据库初始化完成!');
131
+ process.exit(0);
132
+
133
+ async function runGeneratedModuleInitializers(dbInstance: any) {
134
+ const generatedDir = join(__dirname, 'generated');
135
+ if (!existsSync(generatedDir)) {
136
+ return;
137
+ }
138
+
139
+ const files = collectModuleFiles(generatedDir);
140
+ let loaded = 0;
141
+
142
+ for (const filePath of files) {
143
+ const fileUrl = pathToFileURL(filePath).href;
144
+ const mod = await import(fileUrl);
145
+ if (typeof mod.registerModuleTables === 'function') {
146
+ await mod.registerModuleTables(dbInstance);
147
+ loaded += 1;
148
+ console.log(`✅ 已加载业务模块初始化脚本: ${filePath}`);
149
+ }
150
+ }
151
+
152
+ if (loaded === 0) {
153
+ console.log('ℹ️ 未发现业务模块初始化脚本,跳过');
154
+ }
155
+ }
156
+
157
+ function collectModuleFiles(baseDir: string): string[] {
158
+ const results: string[] = [];
159
+ const files = readdirSync(baseDir);
160
+
161
+ for (const file of files) {
162
+ const filePath = join(baseDir, file);
163
+ const stat = statSync(filePath);
164
+ if (stat.isDirectory()) {
165
+ results.push(...collectModuleFiles(filePath));
166
+ continue;
167
+ }
168
+
169
+ if (!isModuleFile(file)) {
170
+ continue;
171
+ }
172
+
173
+ results.push(filePath);
174
+ }
175
+
176
+ return results;
177
+ }
178
+
179
+ function isModuleFile(filename: string): boolean {
180
+ if (filename.endsWith('.d.ts') || filename.endsWith('.spec.ts') || filename.endsWith('.test.ts')) {
181
+ return false;
182
+ }
183
+
184
+ return filename.endsWith('.ts') || filename.endsWith('.js') || filename.endsWith('.mjs');
185
+ }
@@ -0,0 +1,76 @@
1
+ /**
2
+ * API Server - aiko-boot
3
+ */
4
+ import { createApp } from '@ai-partner-x/aiko-boot';
5
+ import { getExpressApp } from '@ai-partner-x/aiko-boot-starter-web';
6
+ import { autoInit, getLogger } from '@ai-partner-x/aiko-boot-starter-log';
7
+ import { RequestLogService } from './service/log.request.service';
8
+ import { fileURLToPath } from 'url';
9
+ import { dirname, join } from 'path';
10
+ import multer from 'multer';
11
+ import express from 'express';
12
+
13
+ const __dirname = dirname(fileURLToPath(import.meta.url));
14
+
15
+ // 初始化日志系统(自动从app.config.ts加载配置)
16
+ autoInit();
17
+
18
+ // 获取服务器logger
19
+ const logger = getLogger('server');
20
+
21
+ logger.info('Starting API server...', {
22
+ nodeEnv: process.env.NODE_ENV || 'development',
23
+ timestamp: new Date().toISOString(),
24
+ });
25
+
26
+ const app = await createApp({ srcDir: __dirname });
27
+
28
+ // 配置HTTP请求日志中间件
29
+ const expressApp = getExpressApp();
30
+ if (expressApp) {
31
+ // 添加请求日志中间件
32
+ expressApp.use(RequestLogService.requestLogMiddleware);
33
+
34
+ // 配置 multer 文件上传中间件
35
+ const upload = multer({ storage: multer.memoryStorage() });
36
+ expressApp.post('/api/upload', upload.single('file'));
37
+ expressApp.post('/api/upload/multiple', upload.array('files', 10));
38
+
39
+ // 配置静态文件服务(本地存储时访问上传的文件)
40
+ const uploadsDir = join(__dirname, '..', 'uploads');
41
+ expressApp.use('/api/uploads', express.static(uploadsDir));
42
+
43
+ logger.info('Express middleware configured', {
44
+ hasRequestLogging: true,
45
+ hasFileUpload: true,
46
+ hasStaticFiles: true,
47
+ });
48
+ }
49
+
50
+ await app.run();
51
+ const port = app.config.get<number>('server.port', 3001);
52
+
53
+ logger.info('API Server started successfully', {
54
+ port,
55
+ baseUrl: `http://localhost:${port}/api`,
56
+ env: process.env.NODE_ENV || 'development',
57
+ timestamp: new Date().toISOString(),
58
+ });
59
+
60
+ // 保留控制台输出以便兼容
61
+ console.log(`\n📡 API: http://localhost:${port}/api\n`);
62
+
63
+ // 全局错误处理
64
+ process.on('uncaughtException', (error) => {
65
+ logger.error('Uncaught Exception', error, {
66
+ timestamp: new Date().toISOString(),
67
+ pid: process.pid,
68
+ });
69
+ });
70
+
71
+ process.on('unhandledRejection', (reason, promise) => {
72
+ logger.error('Unhandled Rejection', reason instanceof Error ? reason : new Error(String(reason)), {
73
+ timestamp: new Date().toISOString(),
74
+ pid: process.pid,
75
+ });
76
+ });
@@ -0,0 +1,106 @@
1
+ import 'reflect-metadata';
2
+ import { Service, Autowired } from '@ai-partner-x/aiko-boot';
3
+ import bcrypt from 'bcryptjs';
4
+ import { UserMapper } from '../mapper/user.mapper.js';
5
+ import { UserRoleMapper } from '../mapper/user-role.mapper.js';
6
+ import { RoleMapper } from '../mapper/role.mapper.js';
7
+ import { RoleMenuMapper } from '../mapper/role-menu.mapper.js';
8
+ import { MenuMapper } from '../mapper/menu.mapper.js';
9
+
10
+ import type { LoginDto, LoginResultDto } from '../dto/auth.dto.js';
11
+ import { signAccessToken, signRefreshToken, verifyRefreshToken } from '../utils/jwt.util.js';
12
+
13
+ @Service()
14
+ export class AuthService {
15
+ @Autowired()
16
+ private userMapper!: UserMapper;
17
+
18
+ @Autowired()
19
+ private userRoleMapper!: UserRoleMapper;
20
+
21
+ @Autowired()
22
+ private roleMapper!: RoleMapper;
23
+
24
+ @Autowired()
25
+ private roleMenuMapper!: RoleMenuMapper;
26
+
27
+ @Autowired()
28
+ private menuMapper!: MenuMapper;
29
+
30
+ async login(dto: LoginDto): Promise<LoginResultDto> {
31
+ const user = await this.userMapper.selectByUsername(dto.username);
32
+ if (!user) {
33
+ throw new Error('用户名或密码错误');
34
+ }
35
+ if (user.status === 0) throw new Error('账户已被禁用');
36
+
37
+ const valid = await bcrypt.compare(dto.password, user.passwordHash);
38
+ if (!valid) throw new Error('用户名或密码错误');
39
+
40
+ const userRolesAndPerms = await this.getUserRolesAndPermissions(user.id);
41
+ const roles = userRolesAndPerms.roles;
42
+ const permissions = userRolesAndPerms.permissions;
43
+
44
+ const payload = { userId: user.id, username: user.username, roles, permissions };
45
+ return {
46
+ accessToken: signAccessToken(payload),
47
+ refreshToken: signRefreshToken({ userId: user.id }),
48
+ userInfo: { id: user.id, username: user.username, realName: user.realName, email: user.email, roles, permissions },
49
+ };
50
+ }
51
+
52
+ async refreshToken(refreshToken: string): Promise<{ accessToken: string }> {
53
+ const payload = verifyRefreshToken(refreshToken);
54
+ const user = await this.userMapper.selectById(payload.userId);
55
+ if (!user || user.status === 0) throw new Error('用户不存在或已禁用');
56
+
57
+ const userRolesAndPerms = await this.getUserRolesAndPermissions(user.id);
58
+ const roles = userRolesAndPerms.roles;
59
+ const permissions = userRolesAndPerms.permissions;
60
+ const accessToken = signAccessToken({ userId: user.id, username: user.username, roles, permissions });
61
+ return { accessToken };
62
+ }
63
+
64
+ async getUserInfo(userId: number): Promise<LoginResultDto['userInfo']> {
65
+ const user = await this.userMapper.selectById(userId);
66
+ if (!user) throw new Error('用户不存在');
67
+ const userRolesAndPerms = await this.getUserRolesAndPermissions(userId);
68
+ const roles = userRolesAndPerms.roles;
69
+ const permissions = userRolesAndPerms.permissions;
70
+ const safeUser: any = {};
71
+ for (const key in user) {
72
+ if (key !== 'password') {
73
+ safeUser[key] = user[key];
74
+ }
75
+ }
76
+ return { ...safeUser, roles, permissions };
77
+ }
78
+
79
+ private async getUserRolesAndPermissions(userId: number) {
80
+ const userRoles = await this.userRoleMapper.selectList({ userId });
81
+ if (!userRoles.length) return { roles: [] as string[], permissions: [] as string[] };
82
+
83
+ const roleIds: number[] = [];
84
+ for (let i = 0; i < userRoles.length; i++) {
85
+ roleIds.push(userRoles[i].roleId);
86
+ }
87
+ const roles: string[] = [];
88
+ const permissions: string[] = [];
89
+
90
+ for (let i = 0; i < roleIds.length; i++) {
91
+ const roleId = roleIds[i];
92
+ const role = await this.roleMapper.selectById(roleId);
93
+ if (role && role.status === 1) {
94
+ roles.push(role.roleCode);
95
+ const roleMenus = await this.roleMenuMapper.selectList({ roleId });
96
+ for (let j = 0; j < roleMenus.length; j++) {
97
+ const rm = roleMenus[j];
98
+ const menu = await this.menuMapper.selectById(rm.menuId);
99
+ if (menu && menu.permission) permissions.push(menu.permission);
100
+ }
101
+ }
102
+ }
103
+ return { roles: [...new Set(roles)], permissions: [...new Set(permissions)] };
104
+ }
105
+ }
106
+
@@ -0,0 +1,80 @@
1
+ import 'reflect-metadata';
2
+ import { Service } from '@ai-partner-x/aiko-boot';
3
+ import type { CacheManager } from '@ai-partner-x/aiko-boot-starter-cache';
4
+ import { getCacheManager, isCacheManagerInitialized } from '@ai-partner-x/aiko-boot-starter-cache';
5
+ import type { CacheGetDto, CachePutDto, CacheEvictDto, CacheClearDto } from '../dto/cache.dto.js';
6
+
7
+ @Service()
8
+ export class CacheService {
9
+ private getCacheManagerInstance(): CacheManager {
10
+ if (!isCacheManagerInitialized()) {
11
+ throw new Error('Cache is not initialized. Enable cache in app.config.ts and ensure Redis is running.');
12
+ }
13
+ return getCacheManager();
14
+ }
15
+
16
+ /**
17
+ * 获取缓存值
18
+ * @param name 缓存命名空间
19
+ * @param key 缓存键
20
+ * @returns 缓存值(JSON 字符串),未命中返回 null
21
+ */
22
+ async get(dto: CacheGetDto): Promise<string | null> {
23
+ const cache = this.getCacheManagerInstance().getCache(dto.name);
24
+ const value = await cache.get(dto.key);
25
+ return value;
26
+ }
27
+
28
+ /**
29
+ * 设置缓存值
30
+ * @param name 缓存命名空间
31
+ * @param key 缓存键
32
+ * @param value 缓存值(任意类型,会自动序列化为 JSON 字符串存储)
33
+ * @param ttlSeconds 可选过期时间(秒)
34
+ */
35
+ async put(dto: CachePutDto): Promise<void> {
36
+ if (dto.value === undefined) {
37
+ throw new Error('Cache value is required and must be JSON-serializable.');
38
+ }
39
+
40
+ let value: string;
41
+ try {
42
+ value = JSON.stringify(dto.value);
43
+ } catch (err) {
44
+ throw new Error('Cache value is not JSON-serializable.');
45
+ }
46
+ const cache = this.getCacheManagerInstance().getCache(dto.name);
47
+ await cache.put(dto.key, value, dto.ttlSeconds);
48
+ }
49
+
50
+ /**
51
+ * 删除缓存条目
52
+ * @param name 缓存命名空间
53
+ * @param key 缓存键
54
+ * @param allEntries 是否清除所有条目
55
+ */
56
+ async evict(dto: CacheEvictDto): Promise<void> {
57
+ const cache = this.getCacheManagerInstance().getCache(dto.name);
58
+ if (dto.allEntries) {
59
+ await cache.clear();
60
+ } else {
61
+ await cache.evict(dto.key);
62
+ }
63
+ }
64
+
65
+ /**
66
+ * 清空缓存命名空间
67
+ * @param name 缓存命名空间
68
+ */
69
+ async clear(dto: CacheClearDto): Promise<void> {
70
+ const cache = this.getCacheManagerInstance().getCache(dto.name);
71
+ await cache.clear();
72
+ }
73
+
74
+ /**
75
+ * 检查缓存是否已初始化
76
+ */
77
+ isInitialized(): boolean {
78
+ return isCacheManagerInitialized();
79
+ }
80
+ }