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

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 (441) hide show
  1. package/dist/cli.js +2 -0
  2. package/dist/commands/add-api.js +9 -3
  3. package/dist/commands/add-app.js +0 -2
  4. package/dist/commands/env.d.ts +8 -0
  5. package/dist/commands/env.js +55 -0
  6. package/dist/commands/init.js +7 -9
  7. package/dist/core/project-config.d.ts +1 -0
  8. package/dist/core/prompts.d.ts +2 -0
  9. package/dist/core/prompts.js +33 -32
  10. package/dist/core/workspace.js +30 -9
  11. package/dist/scaffold.js +1 -1
  12. package/dist/usecases/add-api.usecase.d.ts +5 -2
  13. package/dist/usecases/add-api.usecase.js +83 -21
  14. package/dist/usecases/add-app.usecase.d.ts +0 -1
  15. package/dist/usecases/add-app.usecase.js +20 -10
  16. package/dist/usecases/add-feature.usecase.js +26 -23
  17. package/dist/usecases/env.usecase.d.ts +23 -0
  18. package/dist/usecases/env.usecase.js +284 -0
  19. package/dist/usecases/init-scaffold.usecase.d.ts +3 -4
  20. package/dist/usecases/init-scaffold.usecase.js +36 -49
  21. package/package.json +8 -3
  22. package/templates/api-clean/.env.example +3 -0
  23. package/templates/api-clean/app.config.ts +19 -0
  24. package/templates/{scaffold-default/packages/api → api-clean}/examples/security/README.md +2 -2
  25. package/templates/{scaffold-default/packages/api → api-clean}/examples/security/complete/README.md +1 -1
  26. package/templates/{scaffold-default/packages/api → api-clean}/package.json +4 -16
  27. package/templates/api-clean/src/scripts/init-db.ts +50 -0
  28. package/templates/{scaffold-default/packages/api → api-clean}/src/server.ts +15 -11
  29. package/templates/api-clean/src/types/aiko-boot-starter-web.d.ts +8 -0
  30. package/templates/api-system/.env.example +11 -0
  31. package/templates/api-system/.eslintrc.json +6 -0
  32. package/templates/api-system/.swcrc +17 -0
  33. package/templates/api-system/app.config.ts +87 -0
  34. package/templates/api-system/docs/api-document.md +497 -0
  35. package/templates/api-system/docs/final-completion-report.md +565 -0
  36. package/templates/api-system/docs/permission-integration-summary.md +432 -0
  37. package/templates/api-system/examples/security/README.md +664 -0
  38. package/templates/api-system/examples/security/complete/.env.example +26 -0
  39. package/templates/api-system/examples/security/complete/PROJECT_STRUCTURE.md +220 -0
  40. package/templates/api-system/examples/security/complete/README.md +847 -0
  41. package/templates/api-system/examples/security/complete/app.config.ts +69 -0
  42. package/templates/api-system/examples/security/complete/app.ts +63 -0
  43. package/templates/api-system/examples/security/complete/controller/auth.controller.ts +131 -0
  44. package/templates/api-system/examples/security/complete/controller/index.ts +4 -0
  45. package/templates/api-system/examples/security/complete/controller/permission.controller.ts +41 -0
  46. package/templates/api-system/examples/security/complete/controller/role.controller.ts +53 -0
  47. package/templates/api-system/examples/security/complete/controller/user.controller.ts +53 -0
  48. package/templates/api-system/examples/security/complete/dto/change-password.dto.ts +10 -0
  49. package/templates/api-system/examples/security/complete/dto/create-permission.dto.ts +14 -0
  50. package/templates/api-system/examples/security/complete/dto/create-role.dto.ts +11 -0
  51. package/templates/api-system/examples/security/complete/dto/create-user.dto.ts +15 -0
  52. package/templates/api-system/examples/security/complete/dto/index.ts +7 -0
  53. package/templates/api-system/examples/security/complete/dto/login.dto.ts +10 -0
  54. package/templates/api-system/examples/security/complete/dto/oauth-profile.dto.ts +7 -0
  55. package/templates/api-system/examples/security/complete/dto/register.dto.ts +17 -0
  56. package/templates/api-system/examples/security/complete/entity/index.ts +6 -0
  57. package/templates/api-system/examples/security/complete/entity/oauth-account.entity.ts +39 -0
  58. package/templates/api-system/examples/security/complete/entity/permission.entity.ts +31 -0
  59. package/templates/api-system/examples/security/complete/entity/role-permission.entity.ts +19 -0
  60. package/templates/api-system/examples/security/complete/entity/role.entity.ts +25 -0
  61. package/templates/api-system/examples/security/complete/entity/user-role.entity.ts +19 -0
  62. package/templates/api-system/examples/security/complete/entity/user.entity.ts +46 -0
  63. package/templates/api-system/examples/security/complete/init.sql +81 -0
  64. package/templates/api-system/examples/security/complete/middleware/auth.interceptor.ts +39 -0
  65. package/templates/api-system/examples/security/complete/middleware/index.ts +2 -0
  66. package/templates/api-system/examples/security/complete/middleware/permission.interceptor.ts +61 -0
  67. package/templates/api-system/examples/security/complete/package.json +54 -0
  68. package/templates/api-system/examples/security/complete/seed.sql +42 -0
  69. package/templates/api-system/examples/security/complete/service/auth.service.ts +41 -0
  70. package/templates/api-system/examples/security/complete/service/index.ts +5 -0
  71. package/templates/api-system/examples/security/complete/service/oauth.service.ts +82 -0
  72. package/templates/api-system/examples/security/complete/service/permission.service.ts +113 -0
  73. package/templates/api-system/examples/security/complete/service/role.service.ts +85 -0
  74. package/templates/api-system/examples/security/complete/service/user.service.ts +132 -0
  75. package/templates/api-system/examples/security/complete/tests/TEST_REPORT.md +318 -0
  76. package/templates/api-system/examples/security/complete/tests/generate-report.js +335 -0
  77. package/templates/api-system/examples/security/complete/tests/helpers/api-helpers.ts +116 -0
  78. package/templates/api-system/examples/security/complete/tests/helpers/index.ts +2 -0
  79. package/templates/api-system/examples/security/complete/tests/helpers/test-helpers.ts +129 -0
  80. package/templates/api-system/examples/security/complete/tests/integration/auth.api.test.ts +429 -0
  81. package/templates/api-system/examples/security/complete/tests/integration/role.api.test.ts +400 -0
  82. package/templates/api-system/examples/security/complete/tests/integration/user.api.test.ts +459 -0
  83. package/templates/api-system/examples/security/complete/tests/jest.config.js +40 -0
  84. package/templates/api-system/examples/security/complete/tests/run-all-tests.js +135 -0
  85. package/templates/api-system/examples/security/complete/tests/run-tests.js +109 -0
  86. package/templates/api-system/examples/security/complete/tests/setup.ts +19 -0
  87. package/templates/api-system/examples/security/complete/tests/unit/auth.service.test.ts +199 -0
  88. package/templates/api-system/examples/security/complete/tests/unit/permission.service.test.ts +377 -0
  89. package/templates/api-system/examples/security/complete/tests/unit/user.service.test.ts +288 -0
  90. package/templates/api-system/examples/security/complete/tsconfig.json +35 -0
  91. package/templates/api-system/examples/security/jwt/README.md +424 -0
  92. package/templates/api-system/examples/security/local/README.md +499 -0
  93. package/templates/api-system/examples/security/oauth2/README.md +637 -0
  94. package/templates/api-system/examples/security/permission/README.md +943 -0
  95. package/templates/api-system/examples/security/session/README.md +753 -0
  96. package/templates/{api-base → api-system}/package.json +6 -5
  97. package/templates/{api-base → api-system}/scripts/codegen.cjs +0 -1
  98. package/templates/{api-base → api-system}/scripts/codegen.ts +0 -1
  99. package/templates/{scaffold-default/packages/api → api-system}/src/config/security-auto-configuration.ts +5 -2
  100. package/templates/api-system/src/scripts/simple-reset-admin.mjs +39 -0
  101. package/templates/api-system/src/server.ts +42 -0
  102. package/templates/api-system/src/types/aiko-boot-starter-web.d.ts +8 -0
  103. package/templates/api-system/src/types/sqljs.d.ts +18 -0
  104. package/templates/{api-base → api-system}/tsconfig.json +0 -1
  105. package/templates/api-system/tsup.config.ts +14 -0
  106. package/templates/api-system/uploads/.gitkeep +0 -0
  107. package/templates/app-admin/.env.example +18 -0
  108. package/templates/{scaffold-default/packages/admin → app-admin}/package.json +9 -3
  109. package/templates/{scaffold-default/packages/admin → app-admin}/tsconfig.json +1 -1
  110. package/templates/app-admin/vite.config.ts +31 -0
  111. package/templates/app-mobile/.env.example +5 -0
  112. package/templates/{scaffold-default/packages/mobile → app-mobile}/package.json +8 -2
  113. package/templates/{scaffold-default/packages/mobile → app-mobile}/src/main.tsx +2 -1
  114. package/templates/api-base/app.config.ts +0 -155
  115. package/templates/api-base/src/controller/auth.controller.ts +0 -160
  116. package/templates/api-base/src/controller/menu.controller.ts +0 -56
  117. package/templates/api-base/src/controller/role.controller.ts +0 -62
  118. package/templates/api-base/src/controller/user.controller.ts +0 -75
  119. package/templates/api-base/src/dto/auth.dto.ts +0 -31
  120. package/templates/api-base/src/dto/menu.dto.ts +0 -38
  121. package/templates/api-base/src/dto/role.dto.ts +0 -18
  122. package/templates/api-base/src/dto/user.dto.ts +0 -36
  123. package/templates/api-base/src/entity/menu.entity.ts +0 -35
  124. package/templates/api-base/src/entity/role-menu.entity.ts +0 -14
  125. package/templates/api-base/src/entity/role.entity.ts +0 -23
  126. package/templates/api-base/src/entity/user-role.entity.ts +0 -14
  127. package/templates/api-base/src/entity/user.entity.ts +0 -32
  128. package/templates/api-base/src/mapper/menu.mapper.ts +0 -7
  129. package/templates/api-base/src/mapper/role-menu.mapper.ts +0 -7
  130. package/templates/api-base/src/mapper/role.mapper.ts +0 -7
  131. package/templates/api-base/src/mapper/user-role.mapper.ts +0 -7
  132. package/templates/api-base/src/mapper/user.mapper.ts +0 -12
  133. package/templates/api-base/src/scripts/init-db.ts +0 -204
  134. package/templates/api-base/src/server.ts +0 -69
  135. package/templates/api-base/src/service/auth.service.ts +0 -144
  136. package/templates/api-base/src/service/log.request.service.ts +0 -72
  137. package/templates/api-base/src/service/menu.service.ts +0 -94
  138. package/templates/api-base/src/service/role.service.ts +0 -88
  139. package/templates/api-base/src/service/user.service.ts +0 -175
  140. package/templates/api-base/src/utils/jwt.util.ts +0 -39
  141. package/templates/scaffold-default/README.md +0 -114
  142. package/templates/scaffold-default/package.json +0 -26
  143. package/templates/scaffold-default/packages/admin/.env.dev +0 -1
  144. package/templates/scaffold-default/packages/admin/.env.prod +0 -4
  145. package/templates/scaffold-default/packages/admin/.env.stage +0 -4
  146. package/templates/scaffold-default/packages/admin/vite.config.ts +0 -26
  147. package/templates/scaffold-default/packages/api/app.config.ts +0 -163
  148. package/templates/scaffold-default/packages/api/src/controller/cache.controller.ts +0 -106
  149. package/templates/scaffold-default/packages/api/src/controller/mq.controller.ts +0 -35
  150. package/templates/scaffold-default/packages/api/src/controller/upload.controller.ts +0 -83
  151. package/templates/scaffold-default/packages/api/src/dto/cache.dto.ts +0 -24
  152. package/templates/scaffold-default/packages/api/src/dto/mq.dto.ts +0 -16
  153. package/templates/scaffold-default/packages/api/src/service/cache.service.ts +0 -80
  154. package/templates/scaffold-default/packages/api/src/service/log.request.service.ts +0 -158
  155. package/templates/scaffold-default/packages/api/src/service/log.service.ts +0 -123
  156. package/templates/scaffold-default/packages/api/src/service/mq.consumer.service.ts +0 -26
  157. package/templates/scaffold-default/pnpm-workspace.yaml +0 -2
  158. package/templates/scaffold-default/scripts/postinstall.cjs +0 -42
  159. package/templates/scaffold-default/scripts/rebuild-sqlite.cjs +0 -23
  160. /package/templates/{scaffold-default/packages/api → api-clean}/.eslintrc.json +0 -0
  161. /package/templates/{scaffold-default/packages/api → api-clean}/.swcrc +0 -0
  162. /package/templates/{scaffold-default/packages/api → api-clean}/docs/api-document.md +0 -0
  163. /package/templates/{scaffold-default/packages/api → api-clean}/docs/final-completion-report.md +0 -0
  164. /package/templates/{scaffold-default/packages/api → api-clean}/docs/permission-integration-summary.md +0 -0
  165. /package/templates/{scaffold-default/packages/api → api-clean}/examples/security/complete/.env.example +0 -0
  166. /package/templates/{scaffold-default/packages/api → api-clean}/examples/security/complete/PROJECT_STRUCTURE.md +0 -0
  167. /package/templates/{scaffold-default/packages/api → api-clean}/examples/security/complete/app.config.ts +0 -0
  168. /package/templates/{scaffold-default/packages/api → api-clean}/examples/security/complete/app.ts +0 -0
  169. /package/templates/{scaffold-default/packages/api → api-clean}/examples/security/complete/controller/auth.controller.ts +0 -0
  170. /package/templates/{scaffold-default/packages/api → api-clean}/examples/security/complete/controller/index.ts +0 -0
  171. /package/templates/{scaffold-default/packages/api → api-clean}/examples/security/complete/controller/permission.controller.ts +0 -0
  172. /package/templates/{scaffold-default/packages/api → api-clean}/examples/security/complete/controller/role.controller.ts +0 -0
  173. /package/templates/{scaffold-default/packages/api → api-clean}/examples/security/complete/controller/user.controller.ts +0 -0
  174. /package/templates/{scaffold-default/packages/api → api-clean}/examples/security/complete/dto/change-password.dto.ts +0 -0
  175. /package/templates/{scaffold-default/packages/api → api-clean}/examples/security/complete/dto/create-permission.dto.ts +0 -0
  176. /package/templates/{scaffold-default/packages/api → api-clean}/examples/security/complete/dto/create-role.dto.ts +0 -0
  177. /package/templates/{scaffold-default/packages/api → api-clean}/examples/security/complete/dto/create-user.dto.ts +0 -0
  178. /package/templates/{scaffold-default/packages/api → api-clean}/examples/security/complete/dto/index.ts +0 -0
  179. /package/templates/{scaffold-default/packages/api → api-clean}/examples/security/complete/dto/login.dto.ts +0 -0
  180. /package/templates/{scaffold-default/packages/api → api-clean}/examples/security/complete/dto/oauth-profile.dto.ts +0 -0
  181. /package/templates/{scaffold-default/packages/api → api-clean}/examples/security/complete/dto/register.dto.ts +0 -0
  182. /package/templates/{scaffold-default/packages/api → api-clean}/examples/security/complete/entity/index.ts +0 -0
  183. /package/templates/{scaffold-default/packages/api → api-clean}/examples/security/complete/entity/oauth-account.entity.ts +0 -0
  184. /package/templates/{scaffold-default/packages/api → api-clean}/examples/security/complete/entity/permission.entity.ts +0 -0
  185. /package/templates/{scaffold-default/packages/api → api-clean}/examples/security/complete/entity/role-permission.entity.ts +0 -0
  186. /package/templates/{scaffold-default/packages/api → api-clean}/examples/security/complete/entity/role.entity.ts +0 -0
  187. /package/templates/{scaffold-default/packages/api → api-clean}/examples/security/complete/entity/user-role.entity.ts +0 -0
  188. /package/templates/{scaffold-default/packages/api → api-clean}/examples/security/complete/entity/user.entity.ts +0 -0
  189. /package/templates/{scaffold-default/packages/api → api-clean}/examples/security/complete/init.sql +0 -0
  190. /package/templates/{scaffold-default/packages/api → api-clean}/examples/security/complete/middleware/auth.interceptor.ts +0 -0
  191. /package/templates/{scaffold-default/packages/api → api-clean}/examples/security/complete/middleware/index.ts +0 -0
  192. /package/templates/{scaffold-default/packages/api → api-clean}/examples/security/complete/middleware/permission.interceptor.ts +0 -0
  193. /package/templates/{scaffold-default/packages/api → api-clean}/examples/security/complete/package.json +0 -0
  194. /package/templates/{scaffold-default/packages/api → api-clean}/examples/security/complete/seed.sql +0 -0
  195. /package/templates/{scaffold-default/packages/api → api-clean}/examples/security/complete/service/auth.service.ts +0 -0
  196. /package/templates/{scaffold-default/packages/api → api-clean}/examples/security/complete/service/index.ts +0 -0
  197. /package/templates/{scaffold-default/packages/api → api-clean}/examples/security/complete/service/oauth.service.ts +0 -0
  198. /package/templates/{scaffold-default/packages/api → api-clean}/examples/security/complete/service/permission.service.ts +0 -0
  199. /package/templates/{scaffold-default/packages/api → api-clean}/examples/security/complete/service/role.service.ts +0 -0
  200. /package/templates/{scaffold-default/packages/api → api-clean}/examples/security/complete/service/user.service.ts +0 -0
  201. /package/templates/{scaffold-default/packages/api → api-clean}/examples/security/complete/tests/TEST_REPORT.md +0 -0
  202. /package/templates/{scaffold-default/packages/api → api-clean}/examples/security/complete/tests/generate-report.js +0 -0
  203. /package/templates/{scaffold-default/packages/api → api-clean}/examples/security/complete/tests/helpers/api-helpers.ts +0 -0
  204. /package/templates/{scaffold-default/packages/api → api-clean}/examples/security/complete/tests/helpers/index.ts +0 -0
  205. /package/templates/{scaffold-default/packages/api → api-clean}/examples/security/complete/tests/helpers/test-helpers.ts +0 -0
  206. /package/templates/{scaffold-default/packages/api → api-clean}/examples/security/complete/tests/integration/auth.api.test.ts +0 -0
  207. /package/templates/{scaffold-default/packages/api → api-clean}/examples/security/complete/tests/integration/role.api.test.ts +0 -0
  208. /package/templates/{scaffold-default/packages/api → api-clean}/examples/security/complete/tests/integration/user.api.test.ts +0 -0
  209. /package/templates/{scaffold-default/packages/api → api-clean}/examples/security/complete/tests/jest.config.js +0 -0
  210. /package/templates/{scaffold-default/packages/api → api-clean}/examples/security/complete/tests/run-all-tests.js +0 -0
  211. /package/templates/{scaffold-default/packages/api → api-clean}/examples/security/complete/tests/run-tests.js +0 -0
  212. /package/templates/{scaffold-default/packages/api → api-clean}/examples/security/complete/tests/setup.ts +0 -0
  213. /package/templates/{scaffold-default/packages/api → api-clean}/examples/security/complete/tests/unit/auth.service.test.ts +0 -0
  214. /package/templates/{scaffold-default/packages/api → api-clean}/examples/security/complete/tests/unit/permission.service.test.ts +0 -0
  215. /package/templates/{scaffold-default/packages/api → api-clean}/examples/security/complete/tests/unit/user.service.test.ts +0 -0
  216. /package/templates/{scaffold-default/packages/api → api-clean}/examples/security/complete/tsconfig.json +0 -0
  217. /package/templates/{scaffold-default/packages/api → api-clean}/examples/security/jwt/README.md +0 -0
  218. /package/templates/{scaffold-default/packages/api → api-clean}/examples/security/local/README.md +0 -0
  219. /package/templates/{scaffold-default/packages/api → api-clean}/examples/security/oauth2/README.md +0 -0
  220. /package/templates/{scaffold-default/packages/api → api-clean}/examples/security/permission/README.md +0 -0
  221. /package/templates/{scaffold-default/packages/api → api-clean}/examples/security/session/README.md +0 -0
  222. /package/templates/{scaffold-default/packages/api → api-clean}/scripts/codegen.cjs +0 -0
  223. /package/templates/{scaffold-default/packages/api → api-clean}/scripts/codegen.ts +0 -0
  224. /package/templates/{scaffold-default/packages/api → api-clean}/src/scripts/simple-reset-admin.mjs +0 -0
  225. /package/templates/{scaffold-default/packages/api → api-clean}/src/types/sqljs.d.ts +0 -0
  226. /package/templates/{scaffold-default/packages/api → api-clean}/tsconfig.json +0 -0
  227. /package/templates/{scaffold-default/packages/api → api-clean}/tsup.config.ts +0 -0
  228. /package/templates/{scaffold-default/packages/api → api-clean}/uploads/.gitkeep +0 -0
  229. /package/templates/{scaffold-default/packages/api → api-system}/src/controller/auth.controller.ts +0 -0
  230. /package/templates/{scaffold-default/packages/api → api-system}/src/controller/menu.controller.ts +0 -0
  231. /package/templates/{scaffold-default/packages/api → api-system}/src/controller/role.controller.ts +0 -0
  232. /package/templates/{scaffold-default/packages/api → api-system}/src/controller/user.controller.ts +0 -0
  233. /package/templates/{scaffold-default/packages/api → api-system}/src/dto/auth.dto.ts +0 -0
  234. /package/templates/{scaffold-default/packages/api → api-system}/src/dto/menu.dto.ts +0 -0
  235. /package/templates/{scaffold-default/packages/api → api-system}/src/dto/role.dto.ts +0 -0
  236. /package/templates/{scaffold-default/packages/api → api-system}/src/dto/user.dto.ts +0 -0
  237. /package/templates/{scaffold-default/packages/api → api-system}/src/entity/menu.entity.ts +0 -0
  238. /package/templates/{scaffold-default/packages/api → api-system}/src/entity/role-menu.entity.ts +0 -0
  239. /package/templates/{scaffold-default/packages/api → api-system}/src/entity/role.entity.ts +0 -0
  240. /package/templates/{scaffold-default/packages/api → api-system}/src/entity/user-role.entity.ts +0 -0
  241. /package/templates/{scaffold-default/packages/api → api-system}/src/entity/user.entity.ts +0 -0
  242. /package/templates/{scaffold-default/packages/api → api-system}/src/mapper/menu.mapper.ts +0 -0
  243. /package/templates/{scaffold-default/packages/api → api-system}/src/mapper/role-menu.mapper.ts +0 -0
  244. /package/templates/{scaffold-default/packages/api → api-system}/src/mapper/role.mapper.ts +0 -0
  245. /package/templates/{scaffold-default/packages/api → api-system}/src/mapper/user-role.mapper.ts +0 -0
  246. /package/templates/{scaffold-default/packages/api → api-system}/src/mapper/user.mapper.ts +0 -0
  247. /package/templates/{scaffold-default/packages/api → api-system}/src/scripts/assign-role.ts +0 -0
  248. /package/templates/{scaffold-default/packages/api → api-system}/src/scripts/check-user-role.ts +0 -0
  249. /package/templates/{scaffold-default/packages/api → api-system}/src/scripts/check-users.mjs +0 -0
  250. /package/templates/{scaffold-default/packages/api → api-system}/src/scripts/demo-permissions.mjs +0 -0
  251. /package/templates/{scaffold-default/packages/api → api-system}/src/scripts/init-db.ts +0 -0
  252. /package/templates/{scaffold-default/packages/api → api-system}/src/scripts/reset-admin-password.mjs +0 -0
  253. /package/templates/{scaffold-default/packages/api → api-system}/src/scripts/reset-password.ts +0 -0
  254. /package/templates/{scaffold-default/packages/api → api-system}/src/scripts/test-db.ts +0 -0
  255. /package/templates/{scaffold-default/packages/api → api-system}/src/scripts/test-di.ts +0 -0
  256. /package/templates/{scaffold-default/packages/api → api-system}/src/scripts/test-permissions.mjs +0 -0
  257. /package/templates/{scaffold-default/packages/api → api-system}/src/scripts/update-admin-password.mjs +0 -0
  258. /package/templates/{scaffold-default/packages/api → api-system}/src/scripts/user-roles.ts +0 -0
  259. /package/templates/{scaffold-default/packages/api → api-system}/src/service/auth.service.ts +0 -0
  260. /package/templates/{scaffold-default/packages/api → api-system}/src/service/menu.service.ts +0 -0
  261. /package/templates/{scaffold-default/packages/api → api-system}/src/service/role.service.ts +0 -0
  262. /package/templates/{scaffold-default/packages/api → api-system}/src/service/user.service.ts +0 -0
  263. /package/templates/{scaffold-default/packages/api → api-system}/src/utils/auth.utils.js +0 -0
  264. /package/templates/{scaffold-default/packages/api → api-system}/src/utils/jwt.util.ts +0 -0
  265. /package/templates/{scaffold-default/packages/admin → app-admin}/Dockerfile +0 -0
  266. /package/templates/{scaffold-default/packages/admin → app-admin}/README.MD +0 -0
  267. /package/templates/{scaffold-default/packages/admin → app-admin}/components.json +0 -0
  268. /package/templates/{scaffold-default/packages/admin → app-admin}/eslint.config.js +0 -0
  269. /package/templates/{scaffold-default/packages/admin → app-admin}/index.html +0 -0
  270. /package/templates/{scaffold-default/packages/admin → app-admin}/public/vite.svg +0 -0
  271. /package/templates/{scaffold-default/packages/admin → app-admin}/src/App.css +0 -0
  272. /package/templates/{scaffold-default/packages/admin → app-admin}/src/App.tsx +0 -0
  273. /package/templates/{scaffold-default/packages/admin → app-admin}/src/app.config.ts +0 -0
  274. /package/templates/{scaffold-default/packages/admin → app-admin}/src/components/admin-ui/data-table/data-table-pagination.tsx +0 -0
  275. /package/templates/{scaffold-default/packages/admin → app-admin}/src/components/admin-ui/data-table/data-table-sorter.tsx +0 -0
  276. /package/templates/{scaffold-default/packages/admin → app-admin}/src/components/admin-ui/data-table/index.tsx +0 -0
  277. /package/templates/{scaffold-default/packages/admin → app-admin}/src/components/admin-ui/editable-table.tsx +0 -0
  278. /package/templates/{scaffold-default/packages/admin → app-admin}/src/components/admin-ui/form/input-password.tsx +0 -0
  279. /package/templates/{scaffold-default/packages/admin → app-admin}/src/components/admin-ui/form/sign-in-form.tsx +0 -0
  280. /package/templates/{scaffold-default/packages/admin → app-admin}/src/components/admin-ui/layout/error-component.tsx +0 -0
  281. /package/templates/{scaffold-default/packages/admin → app-admin}/src/components/admin-ui/layout/header.tsx +0 -0
  282. /package/templates/{scaffold-default/packages/admin → app-admin}/src/components/admin-ui/layout/language-switcher.tsx +0 -0
  283. /package/templates/{scaffold-default/packages/admin → app-admin}/src/components/admin-ui/layout/layout.tsx +0 -0
  284. /package/templates/{scaffold-default/packages/admin → app-admin}/src/components/admin-ui/layout/loading-overlay.tsx +0 -0
  285. /package/templates/{scaffold-default/packages/admin → app-admin}/src/components/admin-ui/layout/shell-bar.tsx +0 -0
  286. /package/templates/{scaffold-default/packages/admin → app-admin}/src/components/admin-ui/layout/sidebar.tsx +0 -0
  287. /package/templates/{scaffold-default/packages/admin → app-admin}/src/components/admin-ui/layout/user-avatar.tsx +0 -0
  288. /package/templates/{scaffold-default/packages/admin → app-admin}/src/components/admin-ui/list-report.tsx +0 -0
  289. /package/templates/{scaffold-default/packages/admin → app-admin}/src/components/admin-ui/master-detail.tsx +0 -0
  290. /package/templates/{scaffold-default/packages/admin → app-admin}/src/components/admin-ui/notification/toaster.tsx +0 -0
  291. /package/templates/{scaffold-default/packages/admin → app-admin}/src/components/admin-ui/notification/undoable-notification.tsx +0 -0
  292. /package/templates/{scaffold-default/packages/admin → app-admin}/src/components/admin-ui/object-page.tsx +0 -0
  293. /package/templates/{scaffold-default/packages/admin → app-admin}/src/components/admin-ui/theme/theme-provider.tsx +0 -0
  294. /package/templates/{scaffold-default/packages/admin → app-admin}/src/components/admin-ui/theme/theme-select.tsx +0 -0
  295. /package/templates/{scaffold-default/packages/admin → app-admin}/src/components/admin-ui/theme/theme-toggle.tsx +0 -0
  296. /package/templates/{scaffold-default/packages/admin → app-admin}/src/components/ui/accordion.tsx +0 -0
  297. /package/templates/{scaffold-default/packages/admin → app-admin}/src/components/ui/alert-dialog.tsx +0 -0
  298. /package/templates/{scaffold-default/packages/admin → app-admin}/src/components/ui/alert.tsx +0 -0
  299. /package/templates/{scaffold-default/packages/admin → app-admin}/src/components/ui/aspect-ratio.tsx +0 -0
  300. /package/templates/{scaffold-default/packages/admin → app-admin}/src/components/ui/avatar.tsx +0 -0
  301. /package/templates/{scaffold-default/packages/admin → app-admin}/src/components/ui/badge.tsx +0 -0
  302. /package/templates/{scaffold-default/packages/admin → app-admin}/src/components/ui/breadcrumb.tsx +0 -0
  303. /package/templates/{scaffold-default/packages/admin → app-admin}/src/components/ui/button.tsx +0 -0
  304. /package/templates/{scaffold-default/packages/admin → app-admin}/src/components/ui/calendar.tsx +0 -0
  305. /package/templates/{scaffold-default/packages/admin → app-admin}/src/components/ui/card.tsx +0 -0
  306. /package/templates/{scaffold-default/packages/admin → app-admin}/src/components/ui/carousel.tsx +0 -0
  307. /package/templates/{scaffold-default/packages/admin → app-admin}/src/components/ui/chart.tsx +0 -0
  308. /package/templates/{scaffold-default/packages/admin → app-admin}/src/components/ui/checkbox.tsx +0 -0
  309. /package/templates/{scaffold-default/packages/admin → app-admin}/src/components/ui/collapsible.tsx +0 -0
  310. /package/templates/{scaffold-default/packages/admin → app-admin}/src/components/ui/command.tsx +0 -0
  311. /package/templates/{scaffold-default/packages/admin → app-admin}/src/components/ui/context-menu.tsx +0 -0
  312. /package/templates/{scaffold-default/packages/admin → app-admin}/src/components/ui/dialog.tsx +0 -0
  313. /package/templates/{scaffold-default/packages/admin → app-admin}/src/components/ui/drawer.tsx +0 -0
  314. /package/templates/{scaffold-default/packages/admin → app-admin}/src/components/ui/dropdown-menu.tsx +0 -0
  315. /package/templates/{scaffold-default/packages/admin → app-admin}/src/components/ui/form.tsx +0 -0
  316. /package/templates/{scaffold-default/packages/admin → app-admin}/src/components/ui/hover-card.tsx +0 -0
  317. /package/templates/{scaffold-default/packages/admin → app-admin}/src/components/ui/input-otp.tsx +0 -0
  318. /package/templates/{scaffold-default/packages/admin → app-admin}/src/components/ui/input.tsx +0 -0
  319. /package/templates/{scaffold-default/packages/admin → app-admin}/src/components/ui/label.tsx +0 -0
  320. /package/templates/{scaffold-default/packages/admin → app-admin}/src/components/ui/menubar.tsx +0 -0
  321. /package/templates/{scaffold-default/packages/admin → app-admin}/src/components/ui/navigation-menu.tsx +0 -0
  322. /package/templates/{scaffold-default/packages/admin → app-admin}/src/components/ui/pagination.tsx +0 -0
  323. /package/templates/{scaffold-default/packages/admin → app-admin}/src/components/ui/popover.tsx +0 -0
  324. /package/templates/{scaffold-default/packages/admin → app-admin}/src/components/ui/progress.tsx +0 -0
  325. /package/templates/{scaffold-default/packages/admin → app-admin}/src/components/ui/radio-group.tsx +0 -0
  326. /package/templates/{scaffold-default/packages/admin → app-admin}/src/components/ui/resizable.tsx +0 -0
  327. /package/templates/{scaffold-default/packages/admin → app-admin}/src/components/ui/scroll-area.tsx +0 -0
  328. /package/templates/{scaffold-default/packages/admin → app-admin}/src/components/ui/select.tsx +0 -0
  329. /package/templates/{scaffold-default/packages/admin → app-admin}/src/components/ui/separator.tsx +0 -0
  330. /package/templates/{scaffold-default/packages/admin → app-admin}/src/components/ui/sheet.tsx +0 -0
  331. /package/templates/{scaffold-default/packages/admin → app-admin}/src/components/ui/sidebar.tsx +0 -0
  332. /package/templates/{scaffold-default/packages/admin → app-admin}/src/components/ui/skeleton.tsx +0 -0
  333. /package/templates/{scaffold-default/packages/admin → app-admin}/src/components/ui/slider.tsx +0 -0
  334. /package/templates/{scaffold-default/packages/admin → app-admin}/src/components/ui/sonner.tsx +0 -0
  335. /package/templates/{scaffold-default/packages/admin → app-admin}/src/components/ui/switch.tsx +0 -0
  336. /package/templates/{scaffold-default/packages/admin → app-admin}/src/components/ui/table.tsx +0 -0
  337. /package/templates/{scaffold-default/packages/admin → app-admin}/src/components/ui/tabs.tsx +0 -0
  338. /package/templates/{scaffold-default/packages/admin → app-admin}/src/components/ui/textarea.tsx +0 -0
  339. /package/templates/{scaffold-default/packages/admin → app-admin}/src/components/ui/toggle-group.tsx +0 -0
  340. /package/templates/{scaffold-default/packages/admin → app-admin}/src/components/ui/toggle.tsx +0 -0
  341. /package/templates/{scaffold-default/packages/admin → app-admin}/src/components/ui/tooltip.tsx +0 -0
  342. /package/templates/{scaffold-default/packages/admin → app-admin}/src/hooks/use-mobile.ts +0 -0
  343. /package/templates/{scaffold-default/packages/admin → app-admin}/src/i18n.ts +0 -0
  344. /package/templates/{scaffold-default/packages/admin → app-admin}/src/index.tsx +0 -0
  345. /package/templates/{scaffold-default/packages/admin → app-admin}/src/layouts/menu-layout.tsx +0 -0
  346. /package/templates/{scaffold-default/packages/admin → app-admin}/src/layouts/tile-layout.tsx +0 -0
  347. /package/templates/{scaffold-default/packages/admin → app-admin}/src/lib/utils.ts +0 -0
  348. /package/templates/{scaffold-default/packages/admin → app-admin}/src/locales/en.json +0 -0
  349. /package/templates/{scaffold-default/packages/admin → app-admin}/src/locales/zh.json +0 -0
  350. /package/templates/{scaffold-default/packages/admin → app-admin}/src/pages/dashboard.tsx +0 -0
  351. /package/templates/{scaffold-default/packages/admin → app-admin}/src/pages/goods-receipt/CreatePage.tsx +0 -0
  352. /package/templates/{scaffold-default/packages/admin → app-admin}/src/pages/goods-receipt/EditPage.tsx +0 -0
  353. /package/templates/{scaffold-default/packages/admin → app-admin}/src/pages/goods-receipt/ListPage.tsx +0 -0
  354. /package/templates/{scaffold-default/packages/admin → app-admin}/src/pages/goods-receipt/ViewPage.tsx +0 -0
  355. /package/templates/{scaffold-default/packages/admin → app-admin}/src/pages/goods-receipt/index.ts +0 -0
  356. /package/templates/{scaffold-default/packages/admin → app-admin}/src/pages/home-page.tsx +0 -0
  357. /package/templates/{scaffold-default/packages/admin → app-admin}/src/pages/master-data/cost-centers/index.tsx +0 -0
  358. /package/templates/{scaffold-default/packages/admin → app-admin}/src/pages/master-data/currencies/index.tsx +0 -0
  359. /package/templates/{scaffold-default/packages/admin → app-admin}/src/pages/master-data/materials/ListPage.tsx +0 -0
  360. /package/templates/{scaffold-default/packages/admin → app-admin}/src/pages/master-data/materials/ViewPage.tsx +0 -0
  361. /package/templates/{scaffold-default/packages/admin → app-admin}/src/pages/master-data/materials/index.ts +0 -0
  362. /package/templates/{scaffold-default/packages/admin → app-admin}/src/pages/master-data/plants/ListPage.tsx +0 -0
  363. /package/templates/{scaffold-default/packages/admin → app-admin}/src/pages/master-data/plants/ViewPage.tsx +0 -0
  364. /package/templates/{scaffold-default/packages/admin → app-admin}/src/pages/master-data/plants/index.ts +0 -0
  365. /package/templates/{scaffold-default/packages/admin → app-admin}/src/pages/master-data/purchase-organizations/index.tsx +0 -0
  366. /package/templates/{scaffold-default/packages/admin → app-admin}/src/pages/master-data/units-of-measure/index.tsx +0 -0
  367. /package/templates/{scaffold-default/packages/admin → app-admin}/src/pages/master-data/vendors/ListPage.tsx +0 -0
  368. /package/templates/{scaffold-default/packages/admin → app-admin}/src/pages/master-data/vendors/ViewPage.tsx +0 -0
  369. /package/templates/{scaffold-default/packages/admin → app-admin}/src/pages/master-data/vendors/index.ts +0 -0
  370. /package/templates/{scaffold-default/packages/admin → app-admin}/src/pages/placeholder-page.tsx +0 -0
  371. /package/templates/{scaffold-default/packages/admin → app-admin}/src/pages/purchase-orders/ListPage.tsx +0 -0
  372. /package/templates/{scaffold-default/packages/admin → app-admin}/src/pages/purchase-orders/ViewPage.tsx +0 -0
  373. /package/templates/{scaffold-default/packages/admin → app-admin}/src/pages/purchase-orders/index.ts +0 -0
  374. /package/templates/{scaffold-default/packages/admin → app-admin}/src/pages/purchase-requisitions/CreatePage.tsx +0 -0
  375. /package/templates/{scaffold-default/packages/admin → app-admin}/src/pages/purchase-requisitions/EditPage.tsx +0 -0
  376. /package/templates/{scaffold-default/packages/admin → app-admin}/src/pages/purchase-requisitions/ListPage.tsx +0 -0
  377. /package/templates/{scaffold-default/packages/admin → app-admin}/src/pages/purchase-requisitions/ViewPage.tsx +0 -0
  378. /package/templates/{scaffold-default/packages/admin → app-admin}/src/pages/purchase-requisitions/constants.ts +0 -0
  379. /package/templates/{scaffold-default/packages/admin → app-admin}/src/pages/purchase-requisitions/index.ts +0 -0
  380. /package/templates/{scaffold-default/packages/admin → app-admin}/src/pages/reports/PurchaseOrderReport.tsx +0 -0
  381. /package/templates/{scaffold-default/packages/admin → app-admin}/src/pages/reports/PurchaseRequisitionReport.tsx +0 -0
  382. /package/templates/{scaffold-default/packages/admin → app-admin}/src/pages/reports/index.ts +0 -0
  383. /package/templates/{scaffold-default/packages/admin → app-admin}/src/pages/settings-page.tsx +0 -0
  384. /package/templates/{scaffold-default/packages/admin → app-admin}/src/providers/app-config.tsx +0 -0
  385. /package/templates/{scaffold-default/packages/admin → app-admin}/src/providers/auth-provider.ts +0 -0
  386. /package/templates/{scaffold-default/packages/admin → app-admin}/src/routes/auth.ts +0 -0
  387. /package/templates/{scaffold-default/packages/admin → app-admin}/src/routes/index.ts +0 -0
  388. /package/templates/{scaffold-default/packages/admin → app-admin}/src/routes/menu.ts +0 -0
  389. /package/templates/{scaffold-default/packages/admin → app-admin}/src/routes/modules/goods-receipt.ts +0 -0
  390. /package/templates/{scaffold-default/packages/admin → app-admin}/src/routes/modules/master-data.ts +0 -0
  391. /package/templates/{scaffold-default/packages/admin → app-admin}/src/routes/modules/purchase-orders.ts +0 -0
  392. /package/templates/{scaffold-default/packages/admin → app-admin}/src/routes/modules/purchase-requisitions.ts +0 -0
  393. /package/templates/{scaffold-default/packages/admin → app-admin}/src/routes/modules/reports.ts +0 -0
  394. /package/templates/{scaffold-default/packages/admin → app-admin}/src/routes/modules/settings.ts +0 -0
  395. /package/templates/{scaffold-default/packages/admin → app-admin}/src/routes/withSuspense.tsx +0 -0
  396. /package/templates/{scaffold-default/packages/admin → app-admin}/src/theme/amber.css +0 -0
  397. /package/templates/{scaffold-default/packages/admin → app-admin}/src/theme/blue.css +0 -0
  398. /package/templates/{scaffold-default/packages/admin → app-admin}/src/theme/default.css +0 -0
  399. /package/templates/{scaffold-default/packages/admin → app-admin}/src/theme/fiori.css +0 -0
  400. /package/templates/{scaffold-default/packages/admin → app-admin}/src/theme/green.css +0 -0
  401. /package/templates/{scaffold-default/packages/admin → app-admin}/src/theme/index.css +0 -0
  402. /package/templates/{scaffold-default/packages/admin → app-admin}/src/theme/rose.css +0 -0
  403. /package/templates/{scaffold-default/packages/admin → app-admin}/src/theme/violet.css +0 -0
  404. /package/templates/{scaffold-default/packages/admin → app-admin}/src/vite-env.d.ts +0 -0
  405. /package/templates/{scaffold-default/packages/admin → app-admin}/tsconfig.node.json +0 -0
  406. /package/templates/{scaffold-default/packages/core → app-core}/package.json +0 -0
  407. /package/templates/{scaffold-default/packages/core → app-core}/src/auth/auth-client-middleware.ts +0 -0
  408. /package/templates/{scaffold-default/packages/core → app-core}/src/auth/auth-constants.ts +0 -0
  409. /package/templates/{scaffold-default/packages/core → app-core}/src/auth/auth-service.ts +0 -0
  410. /package/templates/{scaffold-default/packages/core → app-core}/src/auth/backend-auth-provider.ts +0 -0
  411. /package/templates/{scaffold-default/packages/core → app-core}/src/auth/default-auth-provider.ts +0 -0
  412. /package/templates/{scaffold-default/packages/core → app-core}/src/auth/index.ts +0 -0
  413. /package/templates/{scaffold-default/packages/core → app-core}/src/auth/types.ts +0 -0
  414. /package/templates/{scaffold-default/packages/core → app-core}/src/authorization/authorization-client-middleware.ts +0 -0
  415. /package/templates/{scaffold-default/packages/core → app-core}/src/authorization/authorization-config.ts +0 -0
  416. /package/templates/{scaffold-default/packages/core → app-core}/src/authorization/authorization-provider.tsx +0 -0
  417. /package/templates/{scaffold-default/packages/core → app-core}/src/authorization/default-authorization-provider.ts +0 -0
  418. /package/templates/{scaffold-default/packages/core → app-core}/src/authorization/index.ts +0 -0
  419. /package/templates/{scaffold-default/packages/core → app-core}/src/authorization/types.ts +0 -0
  420. /package/templates/{scaffold-default/packages/core → app-core}/src/index.ts +0 -0
  421. /package/templates/{scaffold-default/packages/core → app-core}/src/utils/promise-result-cache.ts +0 -0
  422. /package/templates/{scaffold-default/packages/core → app-core}/tsconfig.json +0 -0
  423. /package/templates/{scaffold-default/packages/mobile → app-mobile}/README.md +0 -0
  424. /package/templates/{scaffold-default/packages/mobile → app-mobile}/index.html +0 -0
  425. /package/templates/{scaffold-default/packages/mobile → app-mobile}/postcss.config.mjs +0 -0
  426. /package/templates/{scaffold-default/packages/mobile → app-mobile}/src/App.tsx +0 -0
  427. /package/templates/{scaffold-default/packages/mobile → app-mobile}/src/app/globals.css +0 -0
  428. /package/templates/{scaffold-default/packages/mobile → app-mobile}/src/components/LoginForm.tsx +0 -0
  429. /package/templates/{scaffold-default/packages/mobile → app-mobile}/src/hooks/index.ts +0 -0
  430. /package/templates/{scaffold-default/packages/mobile → app-mobile}/src/lib/utils.ts +0 -0
  431. /package/templates/{scaffold-default/packages/mobile → app-mobile}/src/pages/HomePage.tsx +0 -0
  432. /package/templates/{scaffold-default/packages/mobile → app-mobile}/src/pages/LoginPage.tsx +0 -0
  433. /package/templates/{scaffold-default/packages/mobile → app-mobile}/src/pages/index.ts +0 -0
  434. /package/templates/{scaffold-default/packages/mobile → app-mobile}/src/routes/ProtectedRoute.tsx +0 -0
  435. /package/templates/{scaffold-default/packages/mobile → app-mobile}/src/routes/index.tsx +0 -0
  436. /package/templates/{scaffold-default/packages/mobile → app-mobile}/src/routes/routes.ts +0 -0
  437. /package/templates/{scaffold-default/packages/mobile → app-mobile}/src/types/index.ts +0 -0
  438. /package/templates/{scaffold-default/packages/mobile → app-mobile}/src/vite-env.d.ts +0 -0
  439. /package/templates/{scaffold-default/packages/mobile → app-mobile}/tsconfig.json +0 -0
  440. /package/templates/{scaffold-default/packages/mobile → app-mobile}/tsconfig.node.json +0 -0
  441. /package/templates/{scaffold-default/packages/mobile → app-mobile}/vite.config.ts +0 -0
@@ -0,0 +1,943 @@
1
+ # 权限控制示例
2
+
3
+ 本示例展示如何使用 Aiko Boot Security 的权限控制功能实现细粒度的访问控制。
4
+
5
+ ## 概述
6
+
7
+ 权限控制是保护应用程序资源的重要机制,Aiko Boot Security 提供了基于角色和权限的访问控制(RBAC)。
8
+
9
+ ## 核心概念
10
+
11
+ ### 角色(Role)
12
+
13
+ 角色是用户的身份标识,如 ADMIN、USER、MODERATOR 等。
14
+
15
+ ### 权限(Permission)
16
+
17
+ 权限是具体的操作许可,如 `user:read`、`user:write`、`post:delete` 等。
18
+
19
+ ### 权限表达式
20
+
21
+ 权限表达式用于定义复杂的访问规则,如 `hasRole('ADMIN')`、`hasPermission('user:read')` 等。
22
+
23
+ ## 配置
24
+
25
+ ```typescript
26
+ import type { AppConfig } from '@ai-partner-x/aiko-boot';
27
+
28
+ export default {
29
+ server: {
30
+ port: Number(process.env.PORT) || 3001,
31
+ servlet: {
32
+ contextPath: '/api',
33
+ },
34
+ },
35
+ security: {
36
+ enabled: true,
37
+ jwt: {
38
+ secret: process.env.JWT_SECRET || 'your-secret-key',
39
+ expiresIn: '24h',
40
+ },
41
+ publicPaths: ['/api/auth/login', '/api/auth/register'],
42
+ },
43
+ database: {
44
+ type: 'sqlite',
45
+ filename: './data/app.db',
46
+ },
47
+ } satisfies AppConfig;
48
+ ```
49
+
50
+ ## 实体定义
51
+
52
+ ### User 实体
53
+
54
+ ```typescript
55
+ import { Entity, TableId, TableField, Column } from '@ai-partner-x/aiko-boot-starter-orm';
56
+
57
+ @Entity({ tableName: 'users' })
58
+ export class User {
59
+ @TableId()
60
+ id!: number;
61
+
62
+ @TableField()
63
+ @Column()
64
+ username!: string;
65
+
66
+ @TableField()
67
+ @Column()
68
+ email!: string;
69
+
70
+ @TableField()
71
+ @Column()
72
+ password!: string;
73
+
74
+ @TableField()
75
+ @Column()
76
+ enabled!: boolean;
77
+
78
+ @TableField()
79
+ @Column()
80
+ createdAt!: Date;
81
+
82
+ @TableField()
83
+ @Column()
84
+ updatedAt!: Date;
85
+
86
+ roles?: Role[];
87
+ permissions?: Permission[];
88
+ }
89
+ ```
90
+
91
+ ### Role 实体
92
+
93
+ ```typescript
94
+ import { Entity, TableId, TableField, Column } from '@ai-partner-x/aiko-boot-starter-orm';
95
+
96
+ @Entity({ tableName: 'roles' })
97
+ export class Role {
98
+ @TableId()
99
+ id!: number;
100
+
101
+ @TableField()
102
+ @Column()
103
+ name!: string;
104
+
105
+ @TableField()
106
+ @Column()
107
+ description!: string;
108
+
109
+ @TableField()
110
+ @Column()
111
+ createdAt!: Date;
112
+
113
+ @TableField()
114
+ @Column()
115
+ updatedAt!: Date;
116
+
117
+ permissions?: Permission[];
118
+ }
119
+ ```
120
+
121
+ ### Permission 实体
122
+
123
+ ```typescript
124
+ import { Entity, TableId, TableField, Column } from '@ai-partner-x/aiko-boot-starter-orm';
125
+
126
+ @Entity({ tableName: 'permissions' })
127
+ export class Permission {
128
+ @TableId()
129
+ id!: number;
130
+
131
+ @TableField()
132
+ @Column()
133
+ name!: string;
134
+
135
+ @TableField()
136
+ @Column()
137
+ description!: string;
138
+
139
+ @TableField()
140
+ @Column()
141
+ resource!: string;
142
+
143
+ @TableField()
144
+ @Column()
145
+ action!: string;
146
+
147
+ @TableField()
148
+ @Column()
149
+ createdAt!: Date;
150
+
151
+ @TableField()
152
+ @Column()
153
+ updatedAt!: Date;
154
+ }
155
+ ```
156
+
157
+ ### UserRole 实体(关联表)
158
+
159
+ ```typescript
160
+ import { Entity, TableId, TableField, Column } from '@ai-partner-x/aiko-boot-starter-orm';
161
+
162
+ @Entity({ tableName: 'user_roles' })
163
+ export class UserRole {
164
+ @TableId()
165
+ id!: number;
166
+
167
+ @TableField()
168
+ @Column()
169
+ userId!: number;
170
+
171
+ @TableField()
172
+ @Column()
173
+ roleId!: number;
174
+
175
+ @TableField()
176
+ @Column()
177
+ createdAt!: Date;
178
+ }
179
+ ```
180
+
181
+ ### RolePermission 实体(关联表)
182
+
183
+ ```typescript
184
+ import { Entity, TableId, TableField, Column } from '@ai-partner-x/aiko-boot-starter-orm';
185
+
186
+ @Entity({ tableName: 'role_permissions' })
187
+ export class RolePermission {
188
+ @TableId()
189
+ id!: number;
190
+
191
+ @TableField()
192
+ @Column()
193
+ roleId!: number;
194
+
195
+ @TableField()
196
+ @Column()
197
+ permissionId!: number;
198
+
199
+ @TableField()
200
+ @Column()
201
+ createdAt!: Date;
202
+ }
203
+ ```
204
+
205
+ ## DTO 定义
206
+
207
+ ### CreateUserDto
208
+
209
+ ```typescript
210
+ import { IsEmail, IsNotEmpty, MinLength, IsArray } from '@ai-partner-x/aiko-boot-starter-validation';
211
+
212
+ export class CreateUserDto {
213
+ @IsNotEmpty({ message: '用户名不能为空' })
214
+ username!: string;
215
+
216
+ @IsEmail({}, { message: '邮箱格式不正确' })
217
+ email!: string;
218
+
219
+ @IsNotEmpty({ message: '密码不能为空' })
220
+ @MinLength(8, { message: '密码长度至少 8 位' })
221
+ password!: string;
222
+
223
+ @IsArray({ message: '角色必须是数组' })
224
+ roleIds!: number[];
225
+ }
226
+ ```
227
+
228
+ ### CreateRoleDto
229
+
230
+ ```typescript
231
+ import { IsNotEmpty, IsArray } from '@ai-partner-x/aiko-boot-starter-validation';
232
+
233
+ export class CreateRoleDto {
234
+ @IsNotEmpty({ message: '角色名称不能为空' })
235
+ name!: string;
236
+
237
+ description?: string;
238
+
239
+ @IsArray({ message: '权限必须是数组' })
240
+ permissionIds!: number[];
241
+ }
242
+ ```
243
+
244
+ ### CreatePermissionDto
245
+
246
+ ```typescript
247
+ import { IsNotEmpty } from '@ai-partner-x/aiko-boot-starter-validation';
248
+
249
+ export class CreatePermissionDto {
250
+ @IsNotEmpty({ message: '权限名称不能为空' })
251
+ name!: string;
252
+
253
+ description?: string;
254
+
255
+ @IsNotEmpty({ message: '资源不能为空' })
256
+ resource!: string;
257
+
258
+ @IsNotEmpty({ message: '操作不能为空' })
259
+ action!: string;
260
+ }
261
+ ```
262
+
263
+ ## 服务层
264
+
265
+ ### PermissionService
266
+
267
+ ```typescript
268
+ import { Service, Autowired } from '@ai-partner-x/aiko-boot';
269
+ import { Mapper, BaseMapper } from '@ai-partner-x/aiko-boot-starter-orm';
270
+ import { Permission } from '../entity/permission.entity.js';
271
+ import { Role } from '../entity/role.entity.js';
272
+ import { User } from '../entity/user.entity.js';
273
+
274
+ @Service()
275
+ export class PermissionService {
276
+ @Autowired()
277
+ private permissionMapper!: BaseMapper<Permission>;
278
+
279
+ @Autowired()
280
+ private roleMapper!: BaseMapper<Role>;
281
+
282
+ @Autowired()
283
+ private userMapper!: BaseMapper<User>;
284
+
285
+ async getUserPermissions(userId: number): Promise<Permission[]> {
286
+ const user = await this.userMapper.selectById(userId);
287
+ if (!user || !user.roles) {
288
+ return [];
289
+ }
290
+
291
+ const permissions: Permission[] = [];
292
+ for (const role of user.roles) {
293
+ if (role.permissions) {
294
+ permissions.push(...role.permissions);
295
+ }
296
+ }
297
+
298
+ return permissions;
299
+ }
300
+
301
+ async hasPermission(userId: number, permissionName: string): Promise<boolean> {
302
+ const permissions = await this.getUserPermissions(userId);
303
+ return permissions.some(p => p.name === permissionName);
304
+ }
305
+
306
+ async hasAnyPermission(userId: number, permissionNames: string[]): Promise<boolean> {
307
+ const permissions = await this.getUserPermissions(userId);
308
+ return permissionNames.some(name => permissions.some(p => p.name === name));
309
+ }
310
+
311
+ async hasAllPermissions(userId: number, permissionNames: string[]): Promise<boolean> {
312
+ const permissions = await this.getUserPermissions(userId);
313
+ return permissionNames.every(name => permissions.some(p => p.name === name));
314
+ }
315
+
316
+ async createPermission(dto: CreatePermissionDto): Promise<Permission> {
317
+ const permission = {
318
+ ...dto,
319
+ createdAt: new Date(),
320
+ updatedAt: new Date(),
321
+ };
322
+ const id = await this.permissionMapper.insert(permission);
323
+ return this.permissionMapper.selectById(id) as Promise<Permission>;
324
+ }
325
+
326
+ async assignPermissionToRole(roleId: number, permissionId: number): Promise<void> {
327
+ const rolePermission = {
328
+ roleId,
329
+ permissionId,
330
+ createdAt: new Date(),
331
+ };
332
+ await this.rolePermissionMapper.insert(rolePermission);
333
+ }
334
+ }
335
+ ```
336
+
337
+ ### UserService
338
+
339
+ ```typescript
340
+ import { Service, Autowired } from '@ai-partner-x/aiko-boot';
341
+ import { Mapper, BaseMapper } from '@ai-partner-x/aiko-boot-starter-orm';
342
+ import { User } from '../entity/user.entity.js';
343
+ import { Role } from '../entity/role.entity.js';
344
+ import { PermissionService } from './permission.service.js';
345
+
346
+ @Service()
347
+ export class UserService {
348
+ @Autowired()
349
+ private userMapper!: BaseMapper<User>;
350
+
351
+ @Autowired()
352
+ private roleMapper!: BaseMapper<Role>;
353
+
354
+ @Autowired()
355
+ private permissionService!: PermissionService;
356
+
357
+ async findById(id: number): Promise<User | null> {
358
+ const user = await this.userMapper.selectById(id);
359
+ if (!user) {
360
+ return null;
361
+ }
362
+
363
+ user.roles = await this.getUserRoles(id);
364
+ user.permissions = await this.permissionService.getUserPermissions(id);
365
+
366
+ return user;
367
+ }
368
+
369
+ async getUserRoles(userId: number): Promise<Role[]> {
370
+ const userRoles = await this.userRoleMapper.selectList({
371
+ where: { userId }
372
+ });
373
+
374
+ const roles: Role[] = [];
375
+ for (const userRole of userRoles) {
376
+ const role = await this.roleMapper.selectById(userRole.roleId);
377
+ if (role) {
378
+ role.permissions = await this.getRolePermissions(role.id);
379
+ roles.push(role);
380
+ }
381
+ }
382
+
383
+ return roles;
384
+ }
385
+
386
+ async getRolePermissions(roleId: number): Promise<Permission[]> {
387
+ const rolePermissions = await this.rolePermissionMapper.selectList({
388
+ where: { roleId }
389
+ });
390
+
391
+ const permissions: Permission[] = [];
392
+ for (const rp of rolePermissions) {
393
+ const permission = await this.permissionMapper.selectById(rp.permissionId);
394
+ if (permission) {
395
+ permissions.push(permission);
396
+ }
397
+ }
398
+
399
+ return permissions;
400
+ }
401
+
402
+ async assignRoleToUser(userId: number, roleId: number): Promise<void> {
403
+ const userRole = {
404
+ userId,
405
+ roleId,
406
+ createdAt: new Date(),
407
+ };
408
+ await this.userRoleMapper.insert(userRole);
409
+ }
410
+
411
+ async removeRoleFromUser(userId: number, roleId: number): Promise<void> {
412
+ const userRoles = await this.userRoleMapper.selectList({
413
+ where: { userId, roleId }
414
+ });
415
+
416
+ for (const userRole of userRoles) {
417
+ await this.userRoleMapper.deleteById(userRole.id);
418
+ }
419
+ }
420
+ }
421
+ ```
422
+
423
+ ## 控制器层
424
+
425
+ ### UserController
426
+
427
+ ```typescript
428
+ import { RestController, GetMapping, PostMapping, PutMapping, DeleteMapping, RequestBody, PathVariable } from '@ai-partner-x/aiko-boot-starter-web';
429
+ import { Public, PreAuthorize, RolesAllowed, Secured } from '@ai-partner-x/aiko-boot-starter-security';
430
+ import { UserService } from '../service/user.service.js';
431
+ import { CreateUserDto } from '../dto/create-user.dto.js';
432
+
433
+ @RestController({ path: '/users' })
434
+ export class UserController {
435
+ @Autowired()
436
+ private userService!: UserService;
437
+
438
+ @GetMapping()
439
+ @PreAuthorize("hasRole('ADMIN')")
440
+ async list(): Promise<User[]> {
441
+ return this.userService.findAll();
442
+ }
443
+
444
+ @GetMapping('/{id}')
445
+ @PreAuthorize("hasRole('ADMIN') or #id == authentication.principal.id")
446
+ async getById(@PathVariable('id') id: number): Promise<User> {
447
+ return this.userService.findById(id);
448
+ }
449
+
450
+ @PostMapping()
451
+ @PreAuthorize("hasRole('ADMIN')")
452
+ async create(@RequestBody() dto: CreateUserDto): Promise<User> {
453
+ return this.userService.create(dto);
454
+ }
455
+
456
+ @PutMapping('/{id}')
457
+ @PreAuthorize("hasRole('ADMIN') or #id == authentication.principal.id")
458
+ async update(@PathVariable('id') id: number, @RequestBody() dto: Partial<User>): Promise<User> {
459
+ return this.userService.update(id, dto);
460
+ }
461
+
462
+ @DeleteMapping('/{id}')
463
+ @PreAuthorize("hasRole('ADMIN')")
464
+ async delete(@PathVariable('id') id: number): Promise<boolean> {
465
+ return this.userService.delete(id);
466
+ }
467
+
468
+ @PostMapping('/{id}/roles')
469
+ @PreAuthorize("hasRole('ADMIN')")
470
+ async assignRole(@PathVariable('id') userId: number, @RequestBody() body: { roleId: number }): Promise<void> {
471
+ return this.userService.assignRoleToUser(userId, body.roleId);
472
+ }
473
+
474
+ @DeleteMapping('/{id}/roles/{roleId}')
475
+ @PreAuthorize("hasRole('ADMIN')")
476
+ async removeRole(@PathVariable('id') userId: number, @PathVariable('roleId') roleId: number): Promise<void> {
477
+ return this.userService.removeRoleFromUser(userId, roleId);
478
+ }
479
+ }
480
+ ```
481
+
482
+ ### RoleController
483
+
484
+ ```typescript
485
+ import { RestController, GetMapping, PostMapping, PutMapping, DeleteMapping, RequestBody, PathVariable } from '@ai-partner-x/aiko-boot-starter-web';
486
+ import { PreAuthorize, RolesAllowed } from '@ai-partner-x/aiko-boot-starter-security';
487
+ import { RoleService } from '../service/role.service.js';
488
+ import { CreateRoleDto } from '../dto/create-role.dto.js';
489
+
490
+ @RestController({ path: '/roles' })
491
+ export class RoleController {
492
+ @Autowired()
493
+ private roleService!: RoleService;
494
+
495
+ @GetMapping()
496
+ @PreAuthorize("hasRole('ADMIN')")
497
+ async list(): Promise<Role[]> {
498
+ return this.roleService.findAll();
499
+ }
500
+
501
+ @GetMapping('/{id}')
502
+ @PreAuthorize("hasRole('ADMIN')")
503
+ async getById(@PathVariable('id') id: number): Promise<Role> {
504
+ return this.roleService.findById(id);
505
+ }
506
+
507
+ @PostMapping()
508
+ @PreAuthorize("hasRole('ADMIN')")
509
+ async create(@RequestBody() dto: CreateRoleDto): Promise<Role> {
510
+ return this.roleService.create(dto);
511
+ }
512
+
513
+ @PutMapping('/{id}')
514
+ @PreAuthorize("hasRole('ADMIN')")
515
+ async update(@PathVariable('id') id: number, @RequestBody() dto: Partial<Role>): Promise<Role> {
516
+ return this.roleService.update(id, dto);
517
+ }
518
+
519
+ @DeleteMapping('/{id}')
520
+ @PreAuthorize("hasRole('ADMIN')")
521
+ async delete(@PathVariable('id') id: number): Promise<boolean> {
522
+ return this.roleService.delete(id);
523
+ }
524
+
525
+ @PostMapping('/{id}/permissions')
526
+ @PreAuthorize("hasRole('ADMIN')")
527
+ async assignPermission(@PathVariable('id') roleId: number, @RequestBody() body: { permissionId: number }): Promise<void> {
528
+ return this.roleService.assignPermissionToRole(roleId, body.permissionId);
529
+ }
530
+
531
+ @DeleteMapping('/{id}/permissions/{permissionId}')
532
+ @PreAuthorize("hasRole('ADMIN')")
533
+ async removePermission(@PathVariable('id') roleId: number, @PathVariable('permissionId') permissionId: number): Promise<void> {
534
+ return this.roleService.removePermissionFromRole(roleId, permissionId);
535
+ }
536
+ }
537
+ ```
538
+
539
+ ### PermissionController
540
+
541
+ ```typescript
542
+ import { RestController, GetMapping, PostMapping, PutMapping, DeleteMapping, RequestBody, PathVariable } from '@ai-partner-x/aiko-boot-starter-web';
543
+ import { PreAuthorize, RolesAllowed } from '@ai-partner-x/aiko-boot-starter-security';
544
+ import { PermissionService } from '../service/permission.service.js';
545
+ import { CreatePermissionDto } from '../dto/create-permission.dto.js';
546
+
547
+ @RestController({ path: '/permissions' })
548
+ export class PermissionController {
549
+ @Autowired()
550
+ private permissionService!: PermissionService;
551
+
552
+ @GetMapping()
553
+ @PreAuthorize("hasRole('ADMIN')")
554
+ async list(): Promise<Permission[]> {
555
+ return this.permissionService.findAll();
556
+ }
557
+
558
+ @GetMapping('/{id}')
559
+ @PreAuthorize("hasRole('ADMIN')")
560
+ async getById(@PathVariable('id') id: number): Promise<Permission> {
561
+ return this.permissionService.findById(id);
562
+ }
563
+
564
+ @PostMapping()
565
+ @PreAuthorize("hasRole('ADMIN')")
566
+ async create(@RequestBody() dto: CreatePermissionDto): Promise<Permission> {
567
+ return this.permissionService.create(dto);
568
+ }
569
+
570
+ @PutMapping('/{id}')
571
+ @PreAuthorize("hasRole('ADMIN')")
572
+ async update(@PathVariable('id') id: number, @RequestBody() dto: Partial<Permission>): Promise<Permission> {
573
+ return this.permissionService.update(id, dto);
574
+ }
575
+
576
+ @DeleteMapping('/{id}')
577
+ @PreAuthorize("hasRole('ADMIN')")
578
+ async delete(@PathVariable('id') id: number): Promise<boolean> {
579
+ return this.permissionService.delete(id);
580
+ }
581
+ }
582
+ ```
583
+
584
+ ### PostController(复杂权限示例)
585
+
586
+ ```typescript
587
+ import { RestController, GetMapping, PostMapping, PutMapping, DeleteMapping, RequestBody, PathVariable } from '@ai-partner-x/aiko-boot-starter-web';
588
+ import { Public, PreAuthorize, PostAuthorize, Secured } from '@ai-partner-x/aiko-boot-starter-security';
589
+ import { PostService } from '../service/post.service.js';
590
+
591
+ @RestController({ path: '/posts' })
592
+ export class PostController {
593
+ @Autowired()
594
+ private postService!: PostService;
595
+
596
+ @GetMapping()
597
+ @Public()
598
+ async list(): Promise<Post[]> {
599
+ return this.postService.findAll();
600
+ }
601
+
602
+ @GetMapping('/{id}')
603
+ @Public()
604
+ async getById(@PathVariable('id') id: number): Promise<Post> {
605
+ return this.postService.findById(id);
606
+ }
607
+
608
+ @PostMapping()
609
+ @PreAuthorize("hasRole('ADMIN') or hasPermission('post:create')")
610
+ async create(@RequestBody() dto: CreatePostDto): Promise<Post> {
611
+ const securityContext = SecurityContext.getInstance();
612
+ const currentUser = securityContext.getCurrentUser();
613
+ return this.postService.create(dto, currentUser.id);
614
+ }
615
+
616
+ @PutMapping('/{id}')
617
+ @PreAuthorize("hasRole('ADMIN') or hasPermission('post:update')")
618
+ @PostAuthorize("returnObject.authorId == authentication.principal.id or hasRole('ADMIN')")
619
+ async update(@PathVariable('id') id: number, @RequestBody() dto: Partial<Post>): Promise<Post> {
620
+ return this.postService.update(id, dto);
621
+ }
622
+
623
+ @DeleteMapping('/{id}')
624
+ @PreAuthorize("hasRole('ADMIN') or hasPermission('post:delete')")
625
+ @PostAuthorize("returnObject.authorId == authentication.principal.id or hasRole('ADMIN')")
626
+ async delete(@PathVariable('id') id: number): Promise<boolean> {
627
+ return this.postService.delete(id);
628
+ }
629
+
630
+ @PostMapping('/{id}/publish')
631
+ @Secured('post:publish')
632
+ async publish(@PathVariable('id') id: number): Promise<Post> {
633
+ return this.postService.publish(id);
634
+ }
635
+
636
+ @GetMapping('/drafts')
637
+ @PreAuthorize("hasRole('ADMIN') or hasPermission('post:read')")
638
+ async getDrafts(): Promise<Post[]> {
639
+ const securityContext = SecurityContext.getInstance();
640
+ const currentUser = securityContext.getCurrentUser();
641
+ return this.postService.getDrafts(currentUser.id);
642
+ }
643
+ }
644
+ ```
645
+
646
+ ## 权限装饰器详解
647
+
648
+ ### @Public()
649
+
650
+ 标记端点为公开访问,无需认证。
651
+
652
+ ```typescript
653
+ @GetMapping('/public')
654
+ @Public()
655
+ async publicInfo(): Promise<any> {
656
+ return { message: '公开 API' };
657
+ }
658
+ ```
659
+
660
+ ### @Authenticated()
661
+
662
+ 要求用户已认证。
663
+
664
+ ```typescript
665
+ @GetMapping('/profile')
666
+ @Authenticated()
667
+ async profile(): Promise<any> {
668
+ const securityContext = SecurityContext.getInstance();
669
+ return securityContext.getCurrentUser();
670
+ }
671
+ ```
672
+
673
+ ### @RolesAllowed(...roles)
674
+
675
+ 要求用户具有指定的角色。
676
+
677
+ ```typescript
678
+ @GetMapping('/admin')
679
+ @RolesAllowed('ADMIN')
680
+ async adminOnly(): Promise<any> {
681
+ return { message: '仅管理员可访问' };
682
+ }
683
+
684
+ @GetMapping('/moderator-or-admin')
685
+ @RolesAllowed('MODERATOR', 'ADMIN')
686
+ async moderatorOrAdmin(): Promise<any> {
687
+ return { message: '版主或管理员可访问' };
688
+ }
689
+ ```
690
+
691
+ ### @PreAuthorize(expression)
692
+
693
+ 在方法执行前进行权限检查。
694
+
695
+ ```typescript
696
+ @GetMapping('/users/{id}')
697
+ @PreAuthorize("hasRole('ADMIN') or #id == authentication.principal.id")
698
+ async getUser(@PathVariable('id') id: number): Promise<User> {
699
+ return this.userService.findById(id);
700
+ }
701
+ ```
702
+
703
+ ### @PostAuthorize(expression)
704
+
705
+ 在方法执行后进行权限检查,可以访问返回值。
706
+
707
+ ```typescript
708
+ @PutMapping('/posts/{id}')
709
+ @PreAuthorize("hasRole('ADMIN') or hasPermission('post:update')")
710
+ @PostAuthorize("returnObject.authorId == authentication.principal.id or hasRole('ADMIN')")
711
+ async updatePost(@PathVariable('id') id: number, @RequestBody() dto: Partial<Post>): Promise<Post> {
712
+ return this.postService.update(id, dto);
713
+ }
714
+ ```
715
+
716
+ ### @Secured(...permissions)
717
+
718
+ 要求用户具有指定的权限。
719
+
720
+ ```typescript
721
+ @PostMapping('/posts/{id}/publish')
722
+ @Secured('post:publish')
723
+ async publishPost(@PathVariable('id') id: number): Promise<Post> {
724
+ return this.postService.publish(id);
725
+ }
726
+
727
+ @DeleteMapping('/users/{id}')
728
+ @Secured('user:delete')
729
+ async deleteUser(@PathVariable('id') id: number): Promise<boolean> {
730
+ return this.userService.delete(id);
731
+ }
732
+ ```
733
+
734
+ ## 权限表达式
735
+
736
+ ### 基本表达式
737
+
738
+ ```typescript
739
+ // 检查角色
740
+ hasRole('ADMIN')
741
+ hasRole('USER')
742
+
743
+ // 检查权限
744
+ hasPermission('user:read')
745
+ hasPermission('post:create')
746
+
747
+ // 检查认证状态
748
+ authenticated()
749
+ ```
750
+
751
+ ### 复合表达式
752
+
753
+ ```typescript
754
+ // 或逻辑
755
+ hasRole('ADMIN') or hasRole('MODERATOR')
756
+ hasPermission('user:read') or hasPermission('user:write')
757
+
758
+ // 与逻辑
759
+ hasRole('ADMIN') and hasPermission('user:delete')
760
+
761
+ // 非逻辑
762
+ not hasRole('BANNED')
763
+
764
+ // 混合逻辑
765
+ hasRole('ADMIN') or (hasPermission('post:update') and #id == authentication.principal.id)
766
+ ```
767
+
768
+ ### 参数引用
769
+
770
+ ```typescript
771
+ @PreAuthorize("#id == authentication.principal.id")
772
+ async getUser(@PathVariable('id') id: number): Promise<User> {
773
+ return this.userService.findById(id);
774
+ }
775
+
776
+ @PreAuthorize("#userId == authentication.principal.id")
777
+ async updateUser(@RequestBody() body: { userId: number }): Promise<User> {
778
+ return this.userService.update(body.userId, body);
779
+ }
780
+ ```
781
+
782
+ ### 返回值引用
783
+
784
+ ```typescript
785
+ @PostAuthorize("returnObject.authorId == authentication.principal.id or hasRole('ADMIN')")
786
+ async getPost(@PathVariable('id') id: number): Promise<Post> {
787
+ return this.postService.findById(id);
788
+ }
789
+ ```
790
+
791
+ ## 使用示例
792
+
793
+ ### 1. 创建角色和权限
794
+
795
+ ```bash
796
+ # 创建权限
797
+ curl -X POST http://localhost:3001/api/permissions \
798
+ -H "Content-Type: application/json" \
799
+ -H "Authorization: Bearer YOUR_ADMIN_TOKEN" \
800
+ -d '{
801
+ "name": "user:read",
802
+ "description": "读取用户信息",
803
+ "resource": "user",
804
+ "action": "read"
805
+ }'
806
+
807
+ curl -X POST http://localhost:3001/api/permissions \
808
+ -H "Content-Type: application/json" \
809
+ -H "Authorization: Bearer YOUR_ADMIN_TOKEN" \
810
+ -d '{
811
+ "name": "user:write",
812
+ "description": "写入用户信息",
813
+ "resource": "user",
814
+ "action": "write"
815
+ }'
816
+
817
+ # 创建角色
818
+ curl -X POST http://localhost:3001/api/roles \
819
+ -H "Content-Type: application/json" \
820
+ -H "Authorization: Bearer YOUR_ADMIN_TOKEN" \
821
+ -d '{
822
+ "name": "USER",
823
+ "description": "普通用户",
824
+ "permissionIds": [1, 2]
825
+ }'
826
+ ```
827
+
828
+ ### 2. 分配角色给用户
829
+
830
+ ```bash
831
+ curl -X POST http://localhost:3001/api/users/1/roles \
832
+ -H "Content-Type: application/json" \
833
+ -H "Authorization: Bearer YOUR_ADMIN_TOKEN" \
834
+ -d '{
835
+ "roleId": 1
836
+ }'
837
+ ```
838
+
839
+ ### 3. 测试权限
840
+
841
+ ```bash
842
+ # 普通用户访问自己的信息(成功)
843
+ curl -X GET http://localhost:3001/api/users/1 \
844
+ -H "Authorization: Bearer YOUR_USER_TOKEN"
845
+
846
+ # 普通用户访问其他用户信息(失败)
847
+ curl -X GET http://localhost:3001/api/users/2 \
848
+ -H "Authorization: Bearer YOUR_USER_TOKEN"
849
+
850
+ # 管理员访问所有用户信息(成功)
851
+ curl -X GET http://localhost:3001/api/users \
852
+ -H "Authorization: Bearer YOUR_ADMIN_TOKEN"
853
+ ```
854
+
855
+ ## 最佳实践
856
+
857
+ ### 1. 权限命名规范
858
+
859
+ 使用 `resource:action` 格式:
860
+
861
+ - `user:read` - 读取用户
862
+ - `user:write` - 写入用户
863
+ - `user:delete` - 删除用户
864
+ - `post:create` - 创建文章
865
+ - `post:update` - 更新文章
866
+ - `post:delete` - 删除文章
867
+ - `post:publish` - 发布文章
868
+
869
+ ### 2. 角色设计
870
+
871
+ - **ADMIN** - 管理员,拥有所有权限
872
+ - **MODERATOR** - 版主,拥有内容管理权限
873
+ - **USER** - 普通用户,拥有基本权限
874
+ - **GUEST** - 访客,拥有只读权限
875
+
876
+ ### 3. 权限粒度
877
+
878
+ - 粗粒度:`user:read`、`user:write`
879
+ - 中粒度:`user:read:basic`、`user:read:profile`
880
+ - 细粒度:`user:read:profile:email`、`user:read:profile:phone`
881
+
882
+ ### 4. 权限缓存
883
+
884
+ 使用 Redis 缓存用户权限,减少数据库查询:
885
+
886
+ ```typescript
887
+ async getUserPermissions(userId: number): Promise<Permission[]> {
888
+ const cacheKey = `user:${userId}:permissions`;
889
+ const cached = await redis.get(cacheKey);
890
+
891
+ if (cached) {
892
+ return JSON.parse(cached);
893
+ }
894
+
895
+ const permissions = await this.loadUserPermissions(userId);
896
+ await redis.setex(cacheKey, 3600, JSON.stringify(permissions));
897
+
898
+ return permissions;
899
+ }
900
+ ```
901
+
902
+ ### 5. 权限审计
903
+
904
+ 记录权限检查日志,便于审计和调试:
905
+
906
+ ```typescript
907
+ async checkPermission(userId: number, permissionName: string): Promise<boolean> {
908
+ const hasPermission = await this.hasPermission(userId, permissionName);
909
+
910
+ await this.auditLog.create({
911
+ userId,
912
+ permission: permissionName,
913
+ result: hasPermission,
914
+ timestamp: new Date(),
915
+ });
916
+
917
+ return hasPermission;
918
+ }
919
+ ```
920
+
921
+ ## 常见问题
922
+
923
+ ### Q: @PreAuthorize 和 @Secured 有什么区别?
924
+
925
+ A: @PreAuthorize 支持复杂的权限表达式,@Secured 只能检查权限名称。
926
+
927
+ ### Q: 如何实现动态权限?
928
+
929
+ A: 使用 @PreAuthorize 配合自定义权限服务,在运行时检查权限。
930
+
931
+ ### Q: 如何实现资源级别的权限?
932
+
933
+ A: 使用 @PostAuthorize 检查返回值,或者在方法内部手动检查权限。
934
+
935
+ ### Q: 如何优化权限检查性能?
936
+
937
+ A: 使用缓存、批量查询、减少数据库访问次数。
938
+
939
+ ## 更多信息
940
+
941
+ - [Spring Security 文档](https://docs.spring.io/spring-security/reference/)
942
+ - [RBAC 最佳实践](https://en.wikipedia.org/wiki/Role-based_access_control)
943
+ - [Aiko Boot Security 文档](../../../../../packages/aiko-boot-starter-security/README.md)