@ai-partner-x/aiko-boot-cli 0.1.4

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 (389) hide show
  1. package/README.md +187 -0
  2. package/dist/cli.d.ts +2 -0
  3. package/dist/cli.js +30 -0
  4. package/dist/commands/add-api.d.ts +5 -0
  5. package/dist/commands/add-api.js +34 -0
  6. package/dist/commands/add-app.d.ts +5 -0
  7. package/dist/commands/add-app.js +34 -0
  8. package/dist/commands/add-feature.d.ts +5 -0
  9. package/dist/commands/add-feature.js +32 -0
  10. package/dist/commands/init.d.ts +7 -0
  11. package/dist/commands/init.js +42 -0
  12. package/dist/commands/list.d.ts +6 -0
  13. package/dist/commands/list.js +36 -0
  14. package/dist/core/framework-version.d.ts +8 -0
  15. package/dist/core/framework-version.js +40 -0
  16. package/dist/core/logger.d.ts +6 -0
  17. package/dist/core/logger.js +16 -0
  18. package/dist/core/project-config.d.ts +21 -0
  19. package/dist/core/project-config.js +17 -0
  20. package/dist/core/prompts.d.ts +5 -0
  21. package/dist/core/prompts.js +43 -0
  22. package/dist/core/template-utils.d.ts +1 -0
  23. package/dist/core/template-utils.js +42 -0
  24. package/dist/core/workspace.d.ts +10 -0
  25. package/dist/core/workspace.js +141 -0
  26. package/dist/index.d.ts +2 -0
  27. package/dist/index.js +2 -0
  28. package/dist/scaffold.d.ts +7 -0
  29. package/dist/scaffold.js +272 -0
  30. package/dist/usecases/add-api.usecase.d.ts +16 -0
  31. package/dist/usecases/add-api.usecase.js +87 -0
  32. package/dist/usecases/add-app.usecase.d.ts +16 -0
  33. package/dist/usecases/add-app.usecase.js +101 -0
  34. package/dist/usecases/add-feature.usecase.d.ts +15 -0
  35. package/dist/usecases/add-feature.usecase.js +566 -0
  36. package/dist/usecases/init-scaffold.usecase.d.ts +19 -0
  37. package/dist/usecases/init-scaffold.usecase.js +131 -0
  38. package/package.json +43 -0
  39. package/templates/api-base/app.config.ts +155 -0
  40. package/templates/api-base/package.json +55 -0
  41. package/templates/api-base/scripts/codegen.cjs +30 -0
  42. package/templates/api-base/scripts/codegen.ts +10 -0
  43. package/templates/api-base/src/controller/auth.controller.ts +160 -0
  44. package/templates/api-base/src/controller/menu.controller.ts +56 -0
  45. package/templates/api-base/src/controller/role.controller.ts +62 -0
  46. package/templates/api-base/src/controller/user.controller.ts +75 -0
  47. package/templates/api-base/src/dto/auth.dto.ts +31 -0
  48. package/templates/api-base/src/dto/menu.dto.ts +38 -0
  49. package/templates/api-base/src/dto/role.dto.ts +18 -0
  50. package/templates/api-base/src/dto/user.dto.ts +36 -0
  51. package/templates/api-base/src/entity/menu.entity.ts +35 -0
  52. package/templates/api-base/src/entity/role-menu.entity.ts +14 -0
  53. package/templates/api-base/src/entity/role.entity.ts +23 -0
  54. package/templates/api-base/src/entity/user-role.entity.ts +14 -0
  55. package/templates/api-base/src/entity/user.entity.ts +32 -0
  56. package/templates/api-base/src/mapper/menu.mapper.ts +7 -0
  57. package/templates/api-base/src/mapper/role-menu.mapper.ts +7 -0
  58. package/templates/api-base/src/mapper/role.mapper.ts +7 -0
  59. package/templates/api-base/src/mapper/user-role.mapper.ts +7 -0
  60. package/templates/api-base/src/mapper/user.mapper.ts +12 -0
  61. package/templates/api-base/src/scripts/init-db.ts +204 -0
  62. package/templates/api-base/src/server.ts +69 -0
  63. package/templates/api-base/src/service/auth.service.ts +144 -0
  64. package/templates/api-base/src/service/log.request.service.ts +72 -0
  65. package/templates/api-base/src/service/menu.service.ts +94 -0
  66. package/templates/api-base/src/service/role.service.ts +88 -0
  67. package/templates/api-base/src/service/user.service.ts +175 -0
  68. package/templates/api-base/src/utils/jwt.util.ts +39 -0
  69. package/templates/api-base/tsconfig.json +18 -0
  70. package/templates/feature-file/api/upload.controller.ts +97 -0
  71. package/templates/feature-log/api/docs/log-integration-guide.md +54 -0
  72. package/templates/feature-log/api/src/service/log.request.service.ts +112 -0
  73. package/templates/feature-log/api/src/service/log.service.ts +124 -0
  74. package/templates/feature-mq/api/src/controller/mq.controller.ts +35 -0
  75. package/templates/feature-mq/api/src/dto/mq.dto.ts +16 -0
  76. package/templates/feature-mq/api/src/service/mq.consumer.service.ts +26 -0
  77. package/templates/feature-redis/api/src/controller/cache.controller.ts +107 -0
  78. package/templates/feature-redis/api/src/dto/cache.dto.ts +24 -0
  79. package/templates/feature-redis/api/src/service/cache.service.ts +80 -0
  80. package/templates/scaffold-default/README.md +114 -0
  81. package/templates/scaffold-default/package.json +26 -0
  82. package/templates/scaffold-default/packages/admin/.env.dev +1 -0
  83. package/templates/scaffold-default/packages/admin/.env.prod +4 -0
  84. package/templates/scaffold-default/packages/admin/.env.stage +4 -0
  85. package/templates/scaffold-default/packages/admin/Dockerfile +37 -0
  86. package/templates/scaffold-default/packages/admin/README.MD +27 -0
  87. package/templates/scaffold-default/packages/admin/components.json +21 -0
  88. package/templates/scaffold-default/packages/admin/eslint.config.js +28 -0
  89. package/templates/scaffold-default/packages/admin/index.html +50 -0
  90. package/templates/scaffold-default/packages/admin/package.json +100 -0
  91. package/templates/scaffold-default/packages/admin/public/vite.svg +1 -0
  92. package/templates/scaffold-default/packages/admin/src/App.css +82 -0
  93. package/templates/scaffold-default/packages/admin/src/App.tsx +128 -0
  94. package/templates/scaffold-default/packages/admin/src/app.config.ts +3 -0
  95. package/templates/scaffold-default/packages/admin/src/components/admin-ui/data-table/data-table-pagination.tsx +148 -0
  96. package/templates/scaffold-default/packages/admin/src/components/admin-ui/data-table/data-table-sorter.tsx +50 -0
  97. package/templates/scaffold-default/packages/admin/src/components/admin-ui/data-table/index.tsx +266 -0
  98. package/templates/scaffold-default/packages/admin/src/components/admin-ui/editable-table.tsx +292 -0
  99. package/templates/scaffold-default/packages/admin/src/components/admin-ui/form/input-password.tsx +38 -0
  100. package/templates/scaffold-default/packages/admin/src/components/admin-ui/form/sign-in-form.tsx +118 -0
  101. package/templates/scaffold-default/packages/admin/src/components/admin-ui/layout/error-component.tsx +71 -0
  102. package/templates/scaffold-default/packages/admin/src/components/admin-ui/layout/header.tsx +162 -0
  103. package/templates/scaffold-default/packages/admin/src/components/admin-ui/layout/language-switcher.tsx +47 -0
  104. package/templates/scaffold-default/packages/admin/src/components/admin-ui/layout/layout.tsx +42 -0
  105. package/templates/scaffold-default/packages/admin/src/components/admin-ui/layout/loading-overlay.tsx +36 -0
  106. package/templates/scaffold-default/packages/admin/src/components/admin-ui/layout/shell-bar.tsx +232 -0
  107. package/templates/scaffold-default/packages/admin/src/components/admin-ui/layout/sidebar.tsx +193 -0
  108. package/templates/scaffold-default/packages/admin/src/components/admin-ui/layout/user-avatar.tsx +31 -0
  109. package/templates/scaffold-default/packages/admin/src/components/admin-ui/list-report.tsx +306 -0
  110. package/templates/scaffold-default/packages/admin/src/components/admin-ui/master-detail.tsx +382 -0
  111. package/templates/scaffold-default/packages/admin/src/components/admin-ui/notification/toaster.tsx +23 -0
  112. package/templates/scaffold-default/packages/admin/src/components/admin-ui/notification/undoable-notification.tsx +84 -0
  113. package/templates/scaffold-default/packages/admin/src/components/admin-ui/object-page.tsx +539 -0
  114. package/templates/scaffold-default/packages/admin/src/components/admin-ui/theme/theme-provider.tsx +160 -0
  115. package/templates/scaffold-default/packages/admin/src/components/admin-ui/theme/theme-select.tsx +129 -0
  116. package/templates/scaffold-default/packages/admin/src/components/admin-ui/theme/theme-toggle.tsx +90 -0
  117. package/templates/scaffold-default/packages/admin/src/components/ui/accordion.tsx +64 -0
  118. package/templates/scaffold-default/packages/admin/src/components/ui/alert-dialog.tsx +157 -0
  119. package/templates/scaffold-default/packages/admin/src/components/ui/alert.tsx +66 -0
  120. package/templates/scaffold-default/packages/admin/src/components/ui/aspect-ratio.tsx +9 -0
  121. package/templates/scaffold-default/packages/admin/src/components/ui/avatar.tsx +53 -0
  122. package/templates/scaffold-default/packages/admin/src/components/ui/badge.tsx +47 -0
  123. package/templates/scaffold-default/packages/admin/src/components/ui/breadcrumb.tsx +111 -0
  124. package/templates/scaffold-default/packages/admin/src/components/ui/button.tsx +59 -0
  125. package/templates/scaffold-default/packages/admin/src/components/ui/calendar.tsx +74 -0
  126. package/templates/scaffold-default/packages/admin/src/components/ui/card.tsx +92 -0
  127. package/templates/scaffold-default/packages/admin/src/components/ui/carousel.tsx +237 -0
  128. package/templates/scaffold-default/packages/admin/src/components/ui/chart.tsx +351 -0
  129. package/templates/scaffold-default/packages/admin/src/components/ui/checkbox.tsx +32 -0
  130. package/templates/scaffold-default/packages/admin/src/components/ui/collapsible.tsx +33 -0
  131. package/templates/scaffold-default/packages/admin/src/components/ui/command.tsx +182 -0
  132. package/templates/scaffold-default/packages/admin/src/components/ui/context-menu.tsx +252 -0
  133. package/templates/scaffold-default/packages/admin/src/components/ui/dialog.tsx +141 -0
  134. package/templates/scaffold-default/packages/admin/src/components/ui/drawer.tsx +130 -0
  135. package/templates/scaffold-default/packages/admin/src/components/ui/dropdown-menu.tsx +255 -0
  136. package/templates/scaffold-default/packages/admin/src/components/ui/form.tsx +166 -0
  137. package/templates/scaffold-default/packages/admin/src/components/ui/hover-card.tsx +42 -0
  138. package/templates/scaffold-default/packages/admin/src/components/ui/input-otp.tsx +77 -0
  139. package/templates/scaffold-default/packages/admin/src/components/ui/input.tsx +21 -0
  140. package/templates/scaffold-default/packages/admin/src/components/ui/label.tsx +22 -0
  141. package/templates/scaffold-default/packages/admin/src/components/ui/menubar.tsx +274 -0
  142. package/templates/scaffold-default/packages/admin/src/components/ui/navigation-menu.tsx +168 -0
  143. package/templates/scaffold-default/packages/admin/src/components/ui/pagination.tsx +127 -0
  144. package/templates/scaffold-default/packages/admin/src/components/ui/popover.tsx +48 -0
  145. package/templates/scaffold-default/packages/admin/src/components/ui/progress.tsx +29 -0
  146. package/templates/scaffold-default/packages/admin/src/components/ui/radio-group.tsx +45 -0
  147. package/templates/scaffold-default/packages/admin/src/components/ui/resizable.tsx +54 -0
  148. package/templates/scaffold-default/packages/admin/src/components/ui/scroll-area.tsx +58 -0
  149. package/templates/scaffold-default/packages/admin/src/components/ui/select.tsx +183 -0
  150. package/templates/scaffold-default/packages/admin/src/components/ui/separator.tsx +26 -0
  151. package/templates/scaffold-default/packages/admin/src/components/ui/sheet.tsx +139 -0
  152. package/templates/scaffold-default/packages/admin/src/components/ui/sidebar.tsx +740 -0
  153. package/templates/scaffold-default/packages/admin/src/components/ui/skeleton.tsx +13 -0
  154. package/templates/scaffold-default/packages/admin/src/components/ui/slider.tsx +63 -0
  155. package/templates/scaffold-default/packages/admin/src/components/ui/sonner.tsx +23 -0
  156. package/templates/scaffold-default/packages/admin/src/components/ui/switch.tsx +31 -0
  157. package/templates/scaffold-default/packages/admin/src/components/ui/table.tsx +114 -0
  158. package/templates/scaffold-default/packages/admin/src/components/ui/tabs.tsx +66 -0
  159. package/templates/scaffold-default/packages/admin/src/components/ui/textarea.tsx +18 -0
  160. package/templates/scaffold-default/packages/admin/src/components/ui/toggle-group.tsx +73 -0
  161. package/templates/scaffold-default/packages/admin/src/components/ui/toggle.tsx +45 -0
  162. package/templates/scaffold-default/packages/admin/src/components/ui/tooltip.tsx +59 -0
  163. package/templates/scaffold-default/packages/admin/src/hooks/use-mobile.ts +21 -0
  164. package/templates/scaffold-default/packages/admin/src/i18n.ts +20 -0
  165. package/templates/scaffold-default/packages/admin/src/index.tsx +19 -0
  166. package/templates/scaffold-default/packages/admin/src/layouts/menu-layout.tsx +211 -0
  167. package/templates/scaffold-default/packages/admin/src/layouts/tile-layout.tsx +355 -0
  168. package/templates/scaffold-default/packages/admin/src/lib/utils.ts +6 -0
  169. package/templates/scaffold-default/packages/admin/src/locales/en.json +68 -0
  170. package/templates/scaffold-default/packages/admin/src/locales/zh.json +68 -0
  171. package/templates/scaffold-default/packages/admin/src/pages/dashboard.tsx +12 -0
  172. package/templates/scaffold-default/packages/admin/src/pages/goods-receipt/CreatePage.tsx +302 -0
  173. package/templates/scaffold-default/packages/admin/src/pages/goods-receipt/EditPage.tsx +221 -0
  174. package/templates/scaffold-default/packages/admin/src/pages/goods-receipt/ListPage.tsx +283 -0
  175. package/templates/scaffold-default/packages/admin/src/pages/goods-receipt/ViewPage.tsx +280 -0
  176. package/templates/scaffold-default/packages/admin/src/pages/goods-receipt/index.ts +4 -0
  177. package/templates/scaffold-default/packages/admin/src/pages/home-page.tsx +244 -0
  178. package/templates/scaffold-default/packages/admin/src/pages/master-data/cost-centers/index.tsx +461 -0
  179. package/templates/scaffold-default/packages/admin/src/pages/master-data/currencies/index.tsx +255 -0
  180. package/templates/scaffold-default/packages/admin/src/pages/master-data/materials/ListPage.tsx +271 -0
  181. package/templates/scaffold-default/packages/admin/src/pages/master-data/materials/ViewPage.tsx +240 -0
  182. package/templates/scaffold-default/packages/admin/src/pages/master-data/materials/index.ts +2 -0
  183. package/templates/scaffold-default/packages/admin/src/pages/master-data/plants/ListPage.tsx +279 -0
  184. package/templates/scaffold-default/packages/admin/src/pages/master-data/plants/ViewPage.tsx +380 -0
  185. package/templates/scaffold-default/packages/admin/src/pages/master-data/plants/index.ts +2 -0
  186. package/templates/scaffold-default/packages/admin/src/pages/master-data/purchase-organizations/index.tsx +341 -0
  187. package/templates/scaffold-default/packages/admin/src/pages/master-data/units-of-measure/index.tsx +295 -0
  188. package/templates/scaffold-default/packages/admin/src/pages/master-data/vendors/ListPage.tsx +266 -0
  189. package/templates/scaffold-default/packages/admin/src/pages/master-data/vendors/ViewPage.tsx +274 -0
  190. package/templates/scaffold-default/packages/admin/src/pages/master-data/vendors/index.ts +2 -0
  191. package/templates/scaffold-default/packages/admin/src/pages/placeholder-page.tsx +13 -0
  192. package/templates/scaffold-default/packages/admin/src/pages/purchase-orders/ListPage.tsx +289 -0
  193. package/templates/scaffold-default/packages/admin/src/pages/purchase-orders/ViewPage.tsx +343 -0
  194. package/templates/scaffold-default/packages/admin/src/pages/purchase-orders/index.ts +2 -0
  195. package/templates/scaffold-default/packages/admin/src/pages/purchase-requisitions/CreatePage.tsx +398 -0
  196. package/templates/scaffold-default/packages/admin/src/pages/purchase-requisitions/EditPage.tsx +473 -0
  197. package/templates/scaffold-default/packages/admin/src/pages/purchase-requisitions/ListPage.tsx +307 -0
  198. package/templates/scaffold-default/packages/admin/src/pages/purchase-requisitions/ViewPage.tsx +304 -0
  199. package/templates/scaffold-default/packages/admin/src/pages/purchase-requisitions/constants.ts +51 -0
  200. package/templates/scaffold-default/packages/admin/src/pages/purchase-requisitions/index.ts +4 -0
  201. package/templates/scaffold-default/packages/admin/src/pages/reports/PurchaseOrderReport.tsx +312 -0
  202. package/templates/scaffold-default/packages/admin/src/pages/reports/PurchaseRequisitionReport.tsx +303 -0
  203. package/templates/scaffold-default/packages/admin/src/pages/reports/index.ts +2 -0
  204. package/templates/scaffold-default/packages/admin/src/pages/settings-page.tsx +335 -0
  205. package/templates/scaffold-default/packages/admin/src/providers/app-config.tsx +50 -0
  206. package/templates/scaffold-default/packages/admin/src/providers/auth-provider.ts +2 -0
  207. package/templates/scaffold-default/packages/admin/src/routes/auth.ts +9 -0
  208. package/templates/scaffold-default/packages/admin/src/routes/index.ts +85 -0
  209. package/templates/scaffold-default/packages/admin/src/routes/menu.ts +176 -0
  210. package/templates/scaffold-default/packages/admin/src/routes/modules/goods-receipt.ts +31 -0
  211. package/templates/scaffold-default/packages/admin/src/routes/modules/master-data.ts +41 -0
  212. package/templates/scaffold-default/packages/admin/src/routes/modules/purchase-orders.ts +27 -0
  213. package/templates/scaffold-default/packages/admin/src/routes/modules/purchase-requisitions.ts +39 -0
  214. package/templates/scaffold-default/packages/admin/src/routes/modules/reports.ts +33 -0
  215. package/templates/scaffold-default/packages/admin/src/routes/modules/settings.ts +19 -0
  216. package/templates/scaffold-default/packages/admin/src/routes/withSuspense.tsx +21 -0
  217. package/templates/scaffold-default/packages/admin/src/theme/amber.css +27 -0
  218. package/templates/scaffold-default/packages/admin/src/theme/blue.css +27 -0
  219. package/templates/scaffold-default/packages/admin/src/theme/default.css +75 -0
  220. package/templates/scaffold-default/packages/admin/src/theme/fiori.css +180 -0
  221. package/templates/scaffold-default/packages/admin/src/theme/green.css +27 -0
  222. package/templates/scaffold-default/packages/admin/src/theme/index.css +12 -0
  223. package/templates/scaffold-default/packages/admin/src/theme/rose.css +27 -0
  224. package/templates/scaffold-default/packages/admin/src/theme/violet.css +27 -0
  225. package/templates/scaffold-default/packages/admin/src/vite-env.d.ts +1 -0
  226. package/templates/scaffold-default/packages/admin/tsconfig.json +28 -0
  227. package/templates/scaffold-default/packages/admin/tsconfig.node.json +21 -0
  228. package/templates/scaffold-default/packages/admin/vite.config.ts +26 -0
  229. package/templates/scaffold-default/packages/api/.eslintrc.json +6 -0
  230. package/templates/scaffold-default/packages/api/.swcrc +17 -0
  231. package/templates/scaffold-default/packages/api/app.config.ts +163 -0
  232. package/templates/scaffold-default/packages/api/docs/api-document.md +497 -0
  233. package/templates/scaffold-default/packages/api/docs/final-completion-report.md +565 -0
  234. package/templates/scaffold-default/packages/api/docs/permission-integration-summary.md +432 -0
  235. package/templates/scaffold-default/packages/api/examples/security/README.md +664 -0
  236. package/templates/scaffold-default/packages/api/examples/security/complete/.env.example +26 -0
  237. package/templates/scaffold-default/packages/api/examples/security/complete/PROJECT_STRUCTURE.md +220 -0
  238. package/templates/scaffold-default/packages/api/examples/security/complete/README.md +847 -0
  239. package/templates/scaffold-default/packages/api/examples/security/complete/app.config.ts +69 -0
  240. package/templates/scaffold-default/packages/api/examples/security/complete/app.ts +63 -0
  241. package/templates/scaffold-default/packages/api/examples/security/complete/controller/auth.controller.ts +131 -0
  242. package/templates/scaffold-default/packages/api/examples/security/complete/controller/index.ts +4 -0
  243. package/templates/scaffold-default/packages/api/examples/security/complete/controller/permission.controller.ts +41 -0
  244. package/templates/scaffold-default/packages/api/examples/security/complete/controller/role.controller.ts +53 -0
  245. package/templates/scaffold-default/packages/api/examples/security/complete/controller/user.controller.ts +53 -0
  246. package/templates/scaffold-default/packages/api/examples/security/complete/dto/change-password.dto.ts +10 -0
  247. package/templates/scaffold-default/packages/api/examples/security/complete/dto/create-permission.dto.ts +14 -0
  248. package/templates/scaffold-default/packages/api/examples/security/complete/dto/create-role.dto.ts +11 -0
  249. package/templates/scaffold-default/packages/api/examples/security/complete/dto/create-user.dto.ts +15 -0
  250. package/templates/scaffold-default/packages/api/examples/security/complete/dto/index.ts +7 -0
  251. package/templates/scaffold-default/packages/api/examples/security/complete/dto/login.dto.ts +10 -0
  252. package/templates/scaffold-default/packages/api/examples/security/complete/dto/oauth-profile.dto.ts +7 -0
  253. package/templates/scaffold-default/packages/api/examples/security/complete/dto/register.dto.ts +17 -0
  254. package/templates/scaffold-default/packages/api/examples/security/complete/entity/index.ts +6 -0
  255. package/templates/scaffold-default/packages/api/examples/security/complete/entity/oauth-account.entity.ts +39 -0
  256. package/templates/scaffold-default/packages/api/examples/security/complete/entity/permission.entity.ts +31 -0
  257. package/templates/scaffold-default/packages/api/examples/security/complete/entity/role-permission.entity.ts +19 -0
  258. package/templates/scaffold-default/packages/api/examples/security/complete/entity/role.entity.ts +25 -0
  259. package/templates/scaffold-default/packages/api/examples/security/complete/entity/user-role.entity.ts +19 -0
  260. package/templates/scaffold-default/packages/api/examples/security/complete/entity/user.entity.ts +46 -0
  261. package/templates/scaffold-default/packages/api/examples/security/complete/init.sql +81 -0
  262. package/templates/scaffold-default/packages/api/examples/security/complete/middleware/auth.interceptor.ts +39 -0
  263. package/templates/scaffold-default/packages/api/examples/security/complete/middleware/index.ts +2 -0
  264. package/templates/scaffold-default/packages/api/examples/security/complete/middleware/permission.interceptor.ts +61 -0
  265. package/templates/scaffold-default/packages/api/examples/security/complete/package.json +54 -0
  266. package/templates/scaffold-default/packages/api/examples/security/complete/seed.sql +42 -0
  267. package/templates/scaffold-default/packages/api/examples/security/complete/service/auth.service.ts +41 -0
  268. package/templates/scaffold-default/packages/api/examples/security/complete/service/index.ts +5 -0
  269. package/templates/scaffold-default/packages/api/examples/security/complete/service/oauth.service.ts +82 -0
  270. package/templates/scaffold-default/packages/api/examples/security/complete/service/permission.service.ts +113 -0
  271. package/templates/scaffold-default/packages/api/examples/security/complete/service/role.service.ts +85 -0
  272. package/templates/scaffold-default/packages/api/examples/security/complete/service/user.service.ts +132 -0
  273. package/templates/scaffold-default/packages/api/examples/security/complete/tests/TEST_REPORT.md +318 -0
  274. package/templates/scaffold-default/packages/api/examples/security/complete/tests/generate-report.js +335 -0
  275. package/templates/scaffold-default/packages/api/examples/security/complete/tests/helpers/api-helpers.ts +116 -0
  276. package/templates/scaffold-default/packages/api/examples/security/complete/tests/helpers/index.ts +2 -0
  277. package/templates/scaffold-default/packages/api/examples/security/complete/tests/helpers/test-helpers.ts +129 -0
  278. package/templates/scaffold-default/packages/api/examples/security/complete/tests/integration/auth.api.test.ts +429 -0
  279. package/templates/scaffold-default/packages/api/examples/security/complete/tests/integration/role.api.test.ts +400 -0
  280. package/templates/scaffold-default/packages/api/examples/security/complete/tests/integration/user.api.test.ts +459 -0
  281. package/templates/scaffold-default/packages/api/examples/security/complete/tests/jest.config.js +40 -0
  282. package/templates/scaffold-default/packages/api/examples/security/complete/tests/run-all-tests.js +135 -0
  283. package/templates/scaffold-default/packages/api/examples/security/complete/tests/run-tests.js +109 -0
  284. package/templates/scaffold-default/packages/api/examples/security/complete/tests/setup.ts +19 -0
  285. package/templates/scaffold-default/packages/api/examples/security/complete/tests/unit/auth.service.test.ts +199 -0
  286. package/templates/scaffold-default/packages/api/examples/security/complete/tests/unit/permission.service.test.ts +377 -0
  287. package/templates/scaffold-default/packages/api/examples/security/complete/tests/unit/user.service.test.ts +288 -0
  288. package/templates/scaffold-default/packages/api/examples/security/complete/tsconfig.json +35 -0
  289. package/templates/scaffold-default/packages/api/examples/security/jwt/README.md +424 -0
  290. package/templates/scaffold-default/packages/api/examples/security/local/README.md +499 -0
  291. package/templates/scaffold-default/packages/api/examples/security/oauth2/README.md +637 -0
  292. package/templates/scaffold-default/packages/api/examples/security/permission/README.md +943 -0
  293. package/templates/scaffold-default/packages/api/examples/security/session/README.md +753 -0
  294. package/templates/scaffold-default/packages/api/package.json +63 -0
  295. package/templates/scaffold-default/packages/api/scripts/codegen.cjs +29 -0
  296. package/templates/scaffold-default/packages/api/scripts/codegen.ts +9 -0
  297. package/templates/scaffold-default/packages/api/src/config/security-auto-configuration.ts +75 -0
  298. package/templates/scaffold-default/packages/api/src/controller/auth.controller.ts +109 -0
  299. package/templates/scaffold-default/packages/api/src/controller/cache.controller.ts +106 -0
  300. package/templates/scaffold-default/packages/api/src/controller/menu.controller.ts +86 -0
  301. package/templates/scaffold-default/packages/api/src/controller/mq.controller.ts +35 -0
  302. package/templates/scaffold-default/packages/api/src/controller/role.controller.ts +100 -0
  303. package/templates/scaffold-default/packages/api/src/controller/upload.controller.ts +83 -0
  304. package/templates/scaffold-default/packages/api/src/controller/user.controller.ts +102 -0
  305. package/templates/scaffold-default/packages/api/src/dto/auth.dto.ts +30 -0
  306. package/templates/scaffold-default/packages/api/src/dto/cache.dto.ts +24 -0
  307. package/templates/scaffold-default/packages/api/src/dto/menu.dto.ts +37 -0
  308. package/templates/scaffold-default/packages/api/src/dto/mq.dto.ts +16 -0
  309. package/templates/scaffold-default/packages/api/src/dto/role.dto.ts +16 -0
  310. package/templates/scaffold-default/packages/api/src/dto/user.dto.ts +35 -0
  311. package/templates/scaffold-default/packages/api/src/entity/menu.entity.ts +34 -0
  312. package/templates/scaffold-default/packages/api/src/entity/role-menu.entity.ts +13 -0
  313. package/templates/scaffold-default/packages/api/src/entity/role.entity.ts +22 -0
  314. package/templates/scaffold-default/packages/api/src/entity/user-role.entity.ts +13 -0
  315. package/templates/scaffold-default/packages/api/src/entity/user.entity.ts +31 -0
  316. package/templates/scaffold-default/packages/api/src/mapper/menu.mapper.ts +6 -0
  317. package/templates/scaffold-default/packages/api/src/mapper/role-menu.mapper.ts +6 -0
  318. package/templates/scaffold-default/packages/api/src/mapper/role.mapper.ts +6 -0
  319. package/templates/scaffold-default/packages/api/src/mapper/user-role.mapper.ts +6 -0
  320. package/templates/scaffold-default/packages/api/src/mapper/user.mapper.ts +11 -0
  321. package/templates/scaffold-default/packages/api/src/scripts/assign-role.ts +41 -0
  322. package/templates/scaffold-default/packages/api/src/scripts/check-user-role.ts +28 -0
  323. package/templates/scaffold-default/packages/api/src/scripts/check-users.mjs +44 -0
  324. package/templates/scaffold-default/packages/api/src/scripts/demo-permissions.mjs +418 -0
  325. package/templates/scaffold-default/packages/api/src/scripts/init-db.ts +166 -0
  326. package/templates/scaffold-default/packages/api/src/scripts/reset-admin-password.mjs +70 -0
  327. package/templates/scaffold-default/packages/api/src/scripts/reset-password.ts +31 -0
  328. package/templates/scaffold-default/packages/api/src/scripts/simple-reset-admin.mjs +39 -0
  329. package/templates/scaffold-default/packages/api/src/scripts/test-db.ts +36 -0
  330. package/templates/scaffold-default/packages/api/src/scripts/test-di.ts +45 -0
  331. package/templates/scaffold-default/packages/api/src/scripts/test-permissions.mjs +551 -0
  332. package/templates/scaffold-default/packages/api/src/scripts/update-admin-password.mjs +56 -0
  333. package/templates/scaffold-default/packages/api/src/scripts/user-roles.ts +28 -0
  334. package/templates/scaffold-default/packages/api/src/server.ts +38 -0
  335. package/templates/scaffold-default/packages/api/src/service/auth.service.ts +165 -0
  336. package/templates/scaffold-default/packages/api/src/service/cache.service.ts +80 -0
  337. package/templates/scaffold-default/packages/api/src/service/log.request.service.ts +158 -0
  338. package/templates/scaffold-default/packages/api/src/service/log.service.ts +123 -0
  339. package/templates/scaffold-default/packages/api/src/service/menu.service.ts +94 -0
  340. package/templates/scaffold-default/packages/api/src/service/mq.consumer.service.ts +26 -0
  341. package/templates/scaffold-default/packages/api/src/service/role.service.ts +88 -0
  342. package/templates/scaffold-default/packages/api/src/service/user.service.ts +195 -0
  343. package/templates/scaffold-default/packages/api/src/types/sqljs.d.ts +18 -0
  344. package/templates/scaffold-default/packages/api/src/utils/auth.utils.js +0 -0
  345. package/templates/scaffold-default/packages/api/src/utils/jwt.util.ts +29 -0
  346. package/templates/scaffold-default/packages/api/tsconfig.json +17 -0
  347. package/templates/scaffold-default/packages/api/tsup.config.ts +14 -0
  348. package/templates/scaffold-default/packages/api/uploads/.gitkeep +0 -0
  349. package/templates/scaffold-default/packages/core/package.json +28 -0
  350. package/templates/scaffold-default/packages/core/src/auth/auth-client-middleware.ts +22 -0
  351. package/templates/scaffold-default/packages/core/src/auth/auth-constants.ts +6 -0
  352. package/templates/scaffold-default/packages/core/src/auth/auth-service.ts +65 -0
  353. package/templates/scaffold-default/packages/core/src/auth/backend-auth-provider.ts +149 -0
  354. package/templates/scaffold-default/packages/core/src/auth/default-auth-provider.ts +38 -0
  355. package/templates/scaffold-default/packages/core/src/auth/index.ts +12 -0
  356. package/templates/scaffold-default/packages/core/src/auth/types.ts +38 -0
  357. package/templates/scaffold-default/packages/core/src/authorization/authorization-client-middleware.ts +38 -0
  358. package/templates/scaffold-default/packages/core/src/authorization/authorization-config.ts +13 -0
  359. package/templates/scaffold-default/packages/core/src/authorization/authorization-provider.tsx +116 -0
  360. package/templates/scaffold-default/packages/core/src/authorization/default-authorization-provider.ts +26 -0
  361. package/templates/scaffold-default/packages/core/src/authorization/index.ts +15 -0
  362. package/templates/scaffold-default/packages/core/src/authorization/types.ts +42 -0
  363. package/templates/scaffold-default/packages/core/src/index.ts +3 -0
  364. package/templates/scaffold-default/packages/core/src/utils/promise-result-cache.ts +18 -0
  365. package/templates/scaffold-default/packages/core/tsconfig.json +19 -0
  366. package/templates/scaffold-default/packages/mobile/README.md +56 -0
  367. package/templates/scaffold-default/packages/mobile/index.html +13 -0
  368. package/templates/scaffold-default/packages/mobile/package.json +28 -0
  369. package/templates/scaffold-default/packages/mobile/postcss.config.mjs +7 -0
  370. package/templates/scaffold-default/packages/mobile/src/App.tsx +5 -0
  371. package/templates/scaffold-default/packages/mobile/src/app/globals.css +1 -0
  372. package/templates/scaffold-default/packages/mobile/src/components/LoginForm.tsx +83 -0
  373. package/templates/scaffold-default/packages/mobile/src/hooks/index.ts +5 -0
  374. package/templates/scaffold-default/packages/mobile/src/lib/utils.ts +7 -0
  375. package/templates/scaffold-default/packages/mobile/src/main.tsx +19 -0
  376. package/templates/scaffold-default/packages/mobile/src/pages/HomePage.tsx +41 -0
  377. package/templates/scaffold-default/packages/mobile/src/pages/LoginPage.tsx +24 -0
  378. package/templates/scaffold-default/packages/mobile/src/pages/index.ts +2 -0
  379. package/templates/scaffold-default/packages/mobile/src/routes/ProtectedRoute.tsx +35 -0
  380. package/templates/scaffold-default/packages/mobile/src/routes/index.tsx +24 -0
  381. package/templates/scaffold-default/packages/mobile/src/routes/routes.ts +11 -0
  382. package/templates/scaffold-default/packages/mobile/src/types/index.ts +5 -0
  383. package/templates/scaffold-default/packages/mobile/src/vite-env.d.ts +1 -0
  384. package/templates/scaffold-default/packages/mobile/tsconfig.json +23 -0
  385. package/templates/scaffold-default/packages/mobile/tsconfig.node.json +11 -0
  386. package/templates/scaffold-default/packages/mobile/vite.config.ts +20 -0
  387. package/templates/scaffold-default/pnpm-workspace.yaml +2 -0
  388. package/templates/scaffold-default/scripts/postinstall.cjs +42 -0
  389. package/templates/scaffold-default/scripts/rebuild-sqlite.cjs +23 -0
@@ -0,0 +1,69 @@
1
+ /**
2
+ * API Server - aiko-boot (base-api)
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.js';
8
+ import { fileURLToPath } from 'url';
9
+ import { dirname } from 'path';
10
+ import express from 'express';
11
+
12
+ const __dirname = dirname(fileURLToPath(import.meta.url));
13
+
14
+ // 初始化日志系统(自动从app.config.ts加载配置)
15
+ autoInit();
16
+
17
+ // 获取服务器logger
18
+ const logger = getLogger('server');
19
+
20
+ logger.info('Starting API server...', {
21
+ nodeEnv: process.env.NODE_ENV || 'development',
22
+ timestamp: new Date().toISOString(),
23
+ });
24
+
25
+ const app = await createApp({ srcDir: __dirname });
26
+
27
+ // 配置HTTP请求日志中间件
28
+ const expressApp = getExpressApp();
29
+ if (expressApp) {
30
+ // 添加请求日志中间件
31
+ expressApp.use(RequestLogService.requestLogMiddleware);
32
+
33
+ logger.info('Express middleware configured', {
34
+ hasRequestLogging: true,
35
+ });
36
+ }
37
+
38
+ await app.run();
39
+ const port = app.config.get<number>('server.port', 3001);
40
+
41
+ logger.info('API Server started successfully', {
42
+ port,
43
+ baseUrl: `http://localhost:${port}/api`,
44
+ env: process.env.NODE_ENV || 'development',
45
+ timestamp: new Date().toISOString(),
46
+ });
47
+
48
+ // 保留控制台输出以便兼容
49
+ console.log(`\n📡 API: http://localhost:${port}/api\n`);
50
+
51
+ // 全局错误处理
52
+ process.on('uncaughtException', (error) => {
53
+ logger.error('Uncaught Exception', error, {
54
+ timestamp: new Date().toISOString(),
55
+ pid: process.pid,
56
+ });
57
+ });
58
+
59
+ process.on('unhandledRejection', (reason, promise) => {
60
+ logger.error(
61
+ 'Unhandled Rejection',
62
+ reason instanceof Error ? reason : new Error(String(reason)),
63
+ {
64
+ timestamp: new Date().toISOString(),
65
+ pid: process.pid,
66
+ },
67
+ );
68
+ });
69
+
@@ -0,0 +1,144 @@
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 {
12
+ signAccessToken,
13
+ signRefreshToken,
14
+ verifyRefreshToken,
15
+ } from '../utils/jwt.util.js';
16
+
17
+ @Service()
18
+ export class AuthService {
19
+ @Autowired()
20
+ private userMapper!: UserMapper;
21
+
22
+ @Autowired()
23
+ private userRoleMapper!: UserRoleMapper;
24
+
25
+ @Autowired()
26
+ private roleMapper!: RoleMapper;
27
+
28
+ @Autowired()
29
+ private roleMenuMapper!: RoleMenuMapper;
30
+
31
+ @Autowired()
32
+ private menuMapper!: MenuMapper;
33
+
34
+ async login(dto: LoginDto): Promise<LoginResultDto> {
35
+ const user = await this.userMapper.selectByUsername(dto.username);
36
+ if (!user) {
37
+ throw new Error('用户名或密码错误');
38
+ }
39
+ if (user.status === 0) throw new Error('账户已被禁用');
40
+
41
+ const valid = await bcrypt.compare(dto.password, user.passwordHash);
42
+ if (!valid) throw new Error('用户名或密码错误');
43
+
44
+ const userRolesAndPerms = await this.getUserRolesAndPermissions(user.id);
45
+ const roles = userRolesAndPerms.roles;
46
+ const permissions = userRolesAndPerms.permissions;
47
+
48
+ const payload = {
49
+ userId: user.id,
50
+ username: user.username,
51
+ roles,
52
+ permissions,
53
+ };
54
+ return {
55
+ accessToken: signAccessToken(payload),
56
+ refreshToken: signRefreshToken({ userId: user.id }),
57
+ userInfo: {
58
+ id: user.id,
59
+ username: user.username,
60
+ realName: user.realName,
61
+ email: user.email,
62
+ roles,
63
+ permissions,
64
+ },
65
+ };
66
+ }
67
+
68
+ async refreshToken(
69
+ refreshToken: string,
70
+ ): Promise<{ accessToken: string }> {
71
+ const payload = verifyRefreshToken(refreshToken);
72
+ const user = await this.userMapper.selectById(payload.userId);
73
+ if (!user || user.status === 0) throw new Error('用户不存在或已禁用');
74
+
75
+ const userRolesAndPerms = await this.getUserRolesAndPermissions(user.id);
76
+ const roles = userRolesAndPerms.roles;
77
+ const permissions = userRolesAndPerms.permissions;
78
+ const accessToken = signAccessToken({
79
+ userId: user.id,
80
+ username: user.username,
81
+ roles,
82
+ permissions,
83
+ });
84
+ return { accessToken };
85
+ }
86
+
87
+ async getUserInfo(userId: number): Promise<LoginResultDto['userInfo']> {
88
+ const user = await this.userMapper.selectById(userId);
89
+ if (!user) throw new Error('用户不存在');
90
+ const userRolesAndPerms = await this.getUserRolesAndPermissions(userId);
91
+ const roles = userRolesAndPerms.roles;
92
+ const permissions = userRolesAndPerms.permissions;
93
+ const safeUser: any = {};
94
+ for (const key in user) {
95
+ if (key !== 'password') {
96
+ safeUser[key] = (user as any)[key];
97
+ }
98
+ }
99
+ return { ...safeUser, roles, permissions };
100
+ }
101
+
102
+ async logout(token: string): Promise<{ ok: boolean }> {
103
+ // 当前实现中 JWT 为无状态,不做服务端记录,直接返回成功
104
+ return { ok: true };
105
+ }
106
+
107
+ private async getUserRolesAndPermissions(userId: number) {
108
+ const userRoles = await this.userRoleMapper.selectList({ userId });
109
+ if (!userRoles.length)
110
+ return { roles: [] as string[], permissions: [] as string[] };
111
+
112
+ const roleIds: number[] = [];
113
+ for (let i = 0; i < userRoles.length; i++) {
114
+ roleIds.push(userRoles[i].roleId);
115
+ }
116
+ const roles: string[] = [];
117
+ const permissions: string[] = [];
118
+
119
+ for (let i = 0; i < roleIds.length; i++) {
120
+ const roleId = roleIds[i];
121
+ const role = await this.roleMapper.selectById(roleId);
122
+ if (role && role.status === 1) {
123
+ roles.push(role.roleCode);
124
+ const roleMenus = await this.roleMenuMapper.selectList({ roleId });
125
+ for (let j = 0; j < roleMenus.length; j++) {
126
+ const rm = roleMenus[j];
127
+ const menu = await this.menuMapper.selectById(rm.menuId);
128
+ if (menu && menu.permission) permissions.push(menu.permission);
129
+ }
130
+ }
131
+ }
132
+ return {
133
+ roles: [...new Set(roles)],
134
+ permissions: [...new Set(permissions)],
135
+ };
136
+ }
137
+
138
+ async handleOAuthCallback(profile: any, tokens: any): Promise<any> {
139
+ // 这里可以根据第三方 profile 查找或创建本地用户,并返回登录结果。
140
+ // 为保持模板简洁,这里仅抛出未实现错误,由业务方按需扩展。
141
+ throw new Error('OAuth callback handler is not implemented in base-api.');
142
+ }
143
+ }
144
+
@@ -0,0 +1,72 @@
1
+ import { Request, Response, NextFunction } from 'express';
2
+ import { getLogger } from '@ai-partner-x/aiko-boot-starter-log';
3
+
4
+ const logger = getLogger('http');
5
+
6
+ /**
7
+ * HTTP请求日志服务
8
+ * 提供HTTP请求日志记录功能
9
+ */
10
+ export class RequestLogService {
11
+ /**
12
+ * 完整的HTTP请求日志中间件
13
+ * 记录所有HTTP请求的详细信息
14
+ */
15
+ static requestLogMiddleware(req: Request, res: Response, next: NextFunction) {
16
+ const startTime = Date.now();
17
+
18
+ // 获取请求信息
19
+ const requestInfo = {
20
+ method: req.method,
21
+ url: req.url,
22
+ path: req.path,
23
+ query: req.query,
24
+ ip: req.ip || req.connection.remoteAddress,
25
+ userAgent: req.get('user-agent') || 'unknown',
26
+ contentType: req.get('content-type'),
27
+ contentLength: req.get('content-length'),
28
+ referer: req.get('referer'),
29
+ };
30
+
31
+ // 记录请求开始
32
+ logger.http('Request started', requestInfo);
33
+
34
+ // 监听响应完成事件
35
+ res.on('finish', () => {
36
+ const duration = Date.now() - startTime;
37
+
38
+ const responseInfo = {
39
+ ...requestInfo,
40
+ status: res.statusCode,
41
+ statusMessage: res.statusMessage,
42
+ duration: `${duration}ms`,
43
+ responseTime: duration,
44
+ timestamp: new Date().toISOString(),
45
+ };
46
+
47
+ // 根据状态码选择日志级别
48
+ if (res.statusCode >= 500) {
49
+ logger.error('Request completed with server error', responseInfo);
50
+ } else if (res.statusCode >= 400) {
51
+ logger.warn('Request completed with client error', responseInfo);
52
+ } else {
53
+ logger.http('Request completed successfully', responseInfo);
54
+ }
55
+ });
56
+
57
+ // 监听响应关闭事件(客户端提前断开连接)
58
+ res.on('close', () => {
59
+ if (!res.writableFinished) {
60
+ const duration = Date.now() - startTime;
61
+ logger.warn('Request closed by client before completion', {
62
+ ...requestInfo,
63
+ duration: `${duration}ms`,
64
+ timestamp: new Date().toISOString(),
65
+ });
66
+ }
67
+ });
68
+
69
+ next();
70
+ }
71
+ }
72
+
@@ -0,0 +1,94 @@
1
+ import 'reflect-metadata';
2
+ import { Service, Transactional } from '@ai-partner-x/aiko-boot';
3
+ import { Autowired } from '@ai-partner-x/aiko-boot/di/server';
4
+ import { MenuMapper } from '../mapper/menu.mapper.js';
5
+ import { Menu } from '../entity/menu.entity.js';
6
+ import type { CreateMenuDto, UpdateMenuDto, MenuTreeVo } from '../dto/menu.dto.js';
7
+
8
+ @Service()
9
+ export class MenuService {
10
+ @Autowired()
11
+ private menuMapper!: MenuMapper;
12
+
13
+ async getFullTree(): Promise<MenuTreeVo[]> {
14
+ const all = await this.menuMapper.selectList();
15
+ return this.buildTree(all, 0);
16
+ }
17
+
18
+ async getUserMenuTree(permissions: string[]): Promise<MenuTreeVo[]> {
19
+ const all = await this.menuMapper.selectList({ status: 1 } as any);
20
+ const visible: any[] = [];
21
+ for (let i = 0; i < all.length; i++) {
22
+ const m = all[i];
23
+ if (m.menuType === 3) {
24
+ if (m.permission && permissions.includes(m.permission)) {
25
+ visible.push(m);
26
+ }
27
+ } else {
28
+ visible.push(m);
29
+ }
30
+ }
31
+ return this.buildTree(visible, 0);
32
+ }
33
+
34
+ async getById(id: number) {
35
+ const menu = await this.menuMapper.selectById(id);
36
+ if (!menu) throw new Error('菜单不存在');
37
+ return menu;
38
+ }
39
+
40
+ @Transactional()
41
+ async createMenu(dto: CreateMenuDto) {
42
+ const menu: Menu = {
43
+ id: 0,
44
+ parentId: dto.parentId !== undefined ? dto.parentId : 0,
45
+ menuName: dto.menuName,
46
+ menuType: dto.menuType,
47
+ path: dto.path,
48
+ component: dto.component,
49
+ permission: dto.permission,
50
+ icon: dto.icon,
51
+ sortOrder: dto.sortOrder !== undefined ? dto.sortOrder : 0,
52
+ status: dto.status !== undefined ? dto.status : 1,
53
+ };
54
+ const result: number = await this.menuMapper.insert(menu as any);
55
+ if (result !== 1) {
56
+ throw new Error('创建菜单失败');
57
+ }
58
+ const menus = await this.menuMapper.selectList({ id: (menu as any).id } as any);
59
+ return menus[0] || null;
60
+ }
61
+
62
+ @Transactional()
63
+ async updateMenu(id: number, dto: UpdateMenuDto) {
64
+ const menu = await this.menuMapper.selectById(id);
65
+ if (!menu) throw new Error('菜单不存在');
66
+ Object.assign(menu as any, dto);
67
+ await this.menuMapper.updateById(menu as any);
68
+ return menu;
69
+ }
70
+
71
+ async deleteMenu(id: number): Promise<boolean> {
72
+ const children = await this.menuMapper.selectList({ parentId: id } as any);
73
+ if (children.length) throw new Error('存在子菜单,无法删除');
74
+ return this.menuMapper.deleteById(id);
75
+ }
76
+
77
+ private buildTree(menus: any[], parentId: number): MenuTreeVo[] {
78
+ const result: MenuTreeVo[] = [];
79
+ for (let i = 0; i < menus.length; i++) {
80
+ const m = menus[i];
81
+ if (m.parentId === parentId) {
82
+ const children = this.buildTree(menus, m.id);
83
+ const item: MenuTreeVo = { ...(m as any) };
84
+ if (children.length > 0) {
85
+ (item as any).children = children;
86
+ }
87
+ result.push(item);
88
+ }
89
+ }
90
+ result.sort((a, b) => a.sortOrder - b.sortOrder);
91
+ return result;
92
+ }
93
+ }
94
+
@@ -0,0 +1,88 @@
1
+ import 'reflect-metadata';
2
+ import { Service, Transactional } from '@ai-partner-x/aiko-boot';
3
+ import { Autowired } from '@ai-partner-x/aiko-boot/di/server';
4
+ import { RoleMapper } from '../mapper/role.mapper.js';
5
+ import { RoleMenuMapper } from '../mapper/role-menu.mapper.js';
6
+ import type { CreateRoleDto, UpdateRoleDto } from '../dto/role.dto.js';
7
+
8
+ @Service()
9
+ export class RoleService {
10
+ @Autowired()
11
+ private roleMapper!: RoleMapper;
12
+
13
+ @Autowired()
14
+ private roleMenuMapper!: RoleMenuMapper;
15
+
16
+ async listRoles() {
17
+ return this.roleMapper.selectList();
18
+ }
19
+
20
+ async getById(id: number) {
21
+ const role = await this.roleMapper.selectById(id);
22
+ if (!role) throw new Error('角色不存在');
23
+ const roleMenus = await this.roleMenuMapper.selectList({ roleId: id } as any);
24
+ const menuIds: number[] = [];
25
+ for (let i = 0; i < roleMenus.length; i++) {
26
+ const rm = roleMenus[i];
27
+ menuIds.push(rm.menuId);
28
+ }
29
+ return { ...(role as any), menuIds: menuIds };
30
+ }
31
+
32
+ @Transactional()
33
+ async createRole(dto: CreateRoleDto) {
34
+ const exists = await this.roleMapper.selectList({ roleCode: dto.roleCode } as any);
35
+ if (exists.length) throw new Error('角色编码已存在');
36
+ const status = dto.status !== undefined ? dto.status : 1;
37
+ await this.roleMapper.insert({
38
+ roleCode: dto.roleCode,
39
+ roleName: dto.roleName,
40
+ description: dto.description,
41
+ status: status,
42
+ createdAt: new Date().toISOString(),
43
+ } as any);
44
+ const roles = await this.roleMapper.selectList({ roleCode: dto.roleCode } as any);
45
+ const role = roles[0];
46
+ if (!role) throw new Error('创建角色失败');
47
+ if (dto.menuIds !== undefined && dto.menuIds.length > 0)
48
+ await this.assignMenus(role.id, dto.menuIds);
49
+ return role;
50
+ }
51
+
52
+ @Transactional()
53
+ async updateRole(id: number, dto: UpdateRoleDto) {
54
+ const role = await this.roleMapper.selectById(id);
55
+ if (!role) throw new Error('角色不存在');
56
+ if (dto.roleName !== undefined) (role as any).roleName = dto.roleName;
57
+ if (dto.description !== undefined) (role as any).description = dto.description;
58
+ if (dto.status !== undefined) (role as any).status = dto.status;
59
+ await this.roleMapper.updateById(role as any);
60
+ if (dto.menuIds !== undefined) await this.assignMenus(id, dto.menuIds);
61
+ return role;
62
+ }
63
+
64
+ async deleteRole(id: number): Promise<boolean> {
65
+ const role = await this.roleMapper.selectById(id);
66
+ if (!role) throw new Error('角色不存在');
67
+ await this.roleMenuMapper.delete({ roleId: id } as any);
68
+ return this.roleMapper.deleteById(id);
69
+ }
70
+
71
+ async getRoleMenuIds(roleId: number): Promise<number[]> {
72
+ const roleMenus = await this.roleMenuMapper.selectList({ roleId } as any);
73
+ const menuIds: number[] = [];
74
+ for (let i = 0; i < roleMenus.length; i++) {
75
+ const rm = roleMenus[i];
76
+ menuIds.push(rm.menuId);
77
+ }
78
+ return menuIds;
79
+ }
80
+
81
+ private async assignMenus(roleId: number, menuIds: number[]) {
82
+ await this.roleMenuMapper.delete({ roleId } as any);
83
+ for (const menuId of menuIds) {
84
+ await this.roleMenuMapper.insert({ roleId, menuId } as any);
85
+ }
86
+ }
87
+ }
88
+
@@ -0,0 +1,175 @@
1
+ import 'reflect-metadata';
2
+ import { Injectable, Autowired } from '@ai-partner-x/aiko-boot/di/server';
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 type {
8
+ CreateUserDto,
9
+ UpdateUserDto,
10
+ UserPageDto,
11
+ UserVo,
12
+ } from '../dto/user.dto.js';
13
+
14
+ @Injectable()
15
+ export class UserService {
16
+ @Autowired()
17
+ private userMapper!: UserMapper;
18
+
19
+ @Autowired()
20
+ private userRoleMapper!: UserRoleMapper;
21
+
22
+ @Autowired()
23
+ private roleMapper!: RoleMapper;
24
+
25
+ async pageUsers(params: UserPageDto) {
26
+ const allUsers = await this.userMapper.selectList();
27
+ let filtered = allUsers;
28
+ if (params.username) {
29
+ const username = params.username;
30
+ const temp: typeof allUsers = [];
31
+ for (let i = 0; i < filtered.length; i++) {
32
+ const u = filtered[i];
33
+ if (u.username.includes(username)) {
34
+ temp.push(u);
35
+ }
36
+ }
37
+ filtered = temp;
38
+ }
39
+ if (params.status !== undefined) {
40
+ const status = params.status;
41
+ const temp: typeof allUsers = [];
42
+ for (let i = 0; i < filtered.length; i++) {
43
+ const u = filtered[i];
44
+ if (u.status === status) {
45
+ temp.push(u);
46
+ }
47
+ }
48
+ filtered = temp;
49
+ }
50
+ const pageNo = params.pageNo || 1;
51
+ const pageSize = params.pageSize || 10;
52
+ const total = filtered.length;
53
+ const start = (pageNo - 1) * pageSize;
54
+ const end = start + pageSize;
55
+ const records = filtered.slice(start, end);
56
+ const usersWithRoles: UserVo[] = [];
57
+ for (let i = 0; i < records.length; i++) {
58
+ const u = records[i];
59
+ const vo = await this.toVo(this.parseEntityDates(u));
60
+ usersWithRoles.push(vo);
61
+ }
62
+ return {
63
+ records: usersWithRoles,
64
+ total,
65
+ pageNo,
66
+ pageSize,
67
+ totalPages: Math.ceil(total / pageSize),
68
+ };
69
+ }
70
+
71
+ async getById(id: number): Promise<UserVo> {
72
+ const user = await this.userMapper.selectById(id);
73
+ if (!user) throw new Error('用户不存在');
74
+ return this.toVo(this.parseEntityDates(user));
75
+ }
76
+
77
+ async createUser(dto: CreateUserDto): Promise<UserVo> {
78
+ const exists = await this.userMapper.selectByUsername(dto.username);
79
+ if (exists) throw new Error('用户名已存在');
80
+ const hashed = await bcrypt.hash(dto.password, 10);
81
+ const status = dto.status !== undefined ? dto.status : 1;
82
+ await this.userMapper.insert({
83
+ username: dto.username,
84
+ passwordHash: hashed,
85
+ realName: dto.realName,
86
+ email: dto.email,
87
+ phone: dto.phone,
88
+ status: status,
89
+ createdAt: new Date().toISOString(),
90
+ updatedAt: new Date().toISOString(),
91
+ } as any);
92
+ const user = await this.userMapper.selectByUsername(dto.username);
93
+ if (!user) throw new Error('创建用户失败');
94
+ if (dto.roleIds !== undefined && dto.roleIds.length > 0)
95
+ await this.assignRoles(user.id, dto.roleIds);
96
+ return this.toVo(this.parseEntityDates(user));
97
+ }
98
+
99
+ async updateUser(id: number, dto: UpdateUserDto): Promise<UserVo> {
100
+ let user = await this.userMapper.selectById(id);
101
+ if (!user) throw new Error('用户不存在');
102
+ user = this.parseEntityDates(user);
103
+ if (dto.realName !== undefined) (user as any).realName = dto.realName;
104
+ if (dto.email !== undefined) (user as any).email = dto.email;
105
+ if (dto.phone !== undefined) (user as any).phone = dto.phone;
106
+ if (dto.status !== undefined) (user as any).status = dto.status;
107
+ (user as any).updatedAt = new Date().toISOString();
108
+ await this.userMapper.updateById(this.formatEntityDates(user) as any);
109
+ if (dto.roleIds !== undefined) await this.assignRoles(id, dto.roleIds);
110
+ return this.toVo(this.parseEntityDates(user));
111
+ }
112
+
113
+ async deleteUser(id: number): Promise<boolean> {
114
+ const user = await this.userMapper.selectById(id);
115
+ if (!user) throw new Error('用户不存在');
116
+ await this.userRoleMapper.delete({ userId: id } as any);
117
+ return this.userMapper.deleteById(id);
118
+ }
119
+
120
+ async resetPassword(id: number, newPassword: string): Promise<void> {
121
+ let user = await this.userMapper.selectById(id);
122
+ if (!user) throw new Error('用户不存在');
123
+ user = this.parseEntityDates(user);
124
+ (user as any).passwordHash = await bcrypt.hash(newPassword, 10);
125
+ (user as any).updatedAt = new Date().toISOString();
126
+ await this.userMapper.updateById(this.formatEntityDates(user) as any);
127
+ }
128
+
129
+ private async assignRoles(userId: number, roleIds: number[]) {
130
+ await this.userRoleMapper.delete({ userId } as any);
131
+ for (const roleId of roleIds) {
132
+ await this.userRoleMapper.insert({ userId, roleId } as any);
133
+ }
134
+ }
135
+
136
+ private parseEntityDates(entity: any): any {
137
+ const parsed = { ...entity };
138
+ if (parsed.createdAt && typeof parsed.createdAt === 'string') {
139
+ parsed.createdAt = new Date(parsed.createdAt);
140
+ }
141
+ if (parsed.updatedAt && typeof parsed.updatedAt === 'string') {
142
+ parsed.updatedAt = new Date(parsed.updatedAt);
143
+ }
144
+ return parsed;
145
+ }
146
+
147
+ private formatEntityDates(entity: any): any {
148
+ const formatted = { ...entity };
149
+ if (formatted.createdAt && formatted.createdAt instanceof Date) {
150
+ formatted.createdAt = formatted.createdAt.toISOString();
151
+ }
152
+ if (formatted.updatedAt && formatted.updatedAt instanceof Date) {
153
+ formatted.updatedAt = formatted.updatedAt.toISOString();
154
+ }
155
+ return formatted;
156
+ }
157
+
158
+ private async toVo(user: any): Promise<UserVo> {
159
+ const userRoles = await this.userRoleMapper.selectList({ userId: user.id } as any);
160
+ const roles: string[] = [];
161
+ for (let i = 0; i < userRoles.length; i++) {
162
+ const ur = userRoles[i];
163
+ const role = await this.roleMapper.selectById(ur.roleId);
164
+ if (role) roles.push(role.roleCode);
165
+ }
166
+ const safe: any = {};
167
+ for (const key in user) {
168
+ if (key !== 'password') {
169
+ safe[key] = user[key];
170
+ }
171
+ }
172
+ return { ...(safe as any), roles } as UserVo;
173
+ }
174
+ }
175
+
@@ -0,0 +1,39 @@
1
+ import jwt from 'jsonwebtoken';
2
+
3
+ const JWT_SECRET =
4
+ process.env.JWT_SECRET || 'ai-first-admin-secret-change-in-production';
5
+ const JWT_REFRESH_SECRET =
6
+ process.env.JWT_REFRESH_SECRET ||
7
+ 'ai-first-refresh-secret-change-in-production';
8
+ const ACCESS_TOKEN_EXPIRES = '2h';
9
+ const REFRESH_TOKEN_EXPIRES = '7d';
10
+
11
+ export interface JwtPayload {
12
+ userId: number;
13
+ username: string;
14
+ roles: string[];
15
+ permissions: string[];
16
+ }
17
+
18
+ export function signAccessToken(payload: JwtPayload): string {
19
+ return jwt.sign(payload, JWT_SECRET, { expiresIn: ACCESS_TOKEN_EXPIRES });
20
+ }
21
+
22
+ export function signRefreshToken(
23
+ payload: Pick<JwtPayload, 'userId'>,
24
+ ): string {
25
+ return jwt.sign(payload, JWT_REFRESH_SECRET, {
26
+ expiresIn: REFRESH_TOKEN_EXPIRES,
27
+ });
28
+ }
29
+
30
+ export function verifyAccessToken(token: string): JwtPayload {
31
+ return jwt.verify(token, JWT_SECRET) as JwtPayload;
32
+ }
33
+
34
+ export function verifyRefreshToken(
35
+ token: string,
36
+ ): Pick<JwtPayload, 'userId'> {
37
+ return jwt.verify(token, JWT_REFRESH_SECRET) as Pick<JwtPayload, 'userId'>;
38
+ }
39
+
@@ -0,0 +1,18 @@
1
+ {
2
+ "compilerOptions": {
3
+ "target": "ES2022",
4
+ "module": "ESNext",
5
+ "moduleResolution": "bundler",
6
+ "strict": true,
7
+ "esModuleInterop": true,
8
+ "skipLibCheck": true,
9
+ "experimentalDecorators": true,
10
+ "emitDecoratorMetadata": true,
11
+ "declaration": true,
12
+ "outDir": "./dist",
13
+ "rootDir": "./src"
14
+ },
15
+ "include": ["src/**/*"],
16
+ "exclude": ["node_modules", "dist"]
17
+ }
18
+