javi-forge 1.1.0 → 1.3.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (238) hide show
  1. package/ci-local/ci-local.sh +38 -10
  2. package/ci-local/hooks/pre-commit +10 -155
  3. package/ci-local/hooks/pre-push +12 -29
  4. package/dist/commands/ci.d.ts +33 -0
  5. package/dist/commands/ci.js +341 -0
  6. package/dist/commands/init.js +5 -0
  7. package/dist/index.js +39 -5
  8. package/dist/lib/docker.d.ts +43 -0
  9. package/dist/lib/docker.js +223 -0
  10. package/dist/ui/CI.d.ts +9 -0
  11. package/dist/ui/CI.js +91 -0
  12. package/package.json +9 -1
  13. package/ai-config/.skillignore +0 -15
  14. package/ai-config/AUTO_INVOKE.md +0 -300
  15. package/ai-config/agents/_TEMPLATE.md +0 -93
  16. package/ai-config/agents/business/api-designer.md +0 -1657
  17. package/ai-config/agents/business/business-analyst.md +0 -1331
  18. package/ai-config/agents/business/product-strategist.md +0 -206
  19. package/ai-config/agents/business/project-manager.md +0 -178
  20. package/ai-config/agents/business/requirements-analyst.md +0 -1277
  21. package/ai-config/agents/business/technical-writer.md +0 -1679
  22. package/ai-config/agents/creative/ux-designer.md +0 -205
  23. package/ai-config/agents/data-ai/ai-engineer.md +0 -487
  24. package/ai-config/agents/data-ai/analytics-engineer.md +0 -953
  25. package/ai-config/agents/data-ai/data-engineer.md +0 -173
  26. package/ai-config/agents/data-ai/data-scientist.md +0 -672
  27. package/ai-config/agents/data-ai/mlops-engineer.md +0 -814
  28. package/ai-config/agents/data-ai/prompt-engineer.md +0 -772
  29. package/ai-config/agents/development/angular-expert.md +0 -620
  30. package/ai-config/agents/development/backend-architect.md +0 -795
  31. package/ai-config/agents/development/database-specialist.md +0 -212
  32. package/ai-config/agents/development/frontend-specialist.md +0 -686
  33. package/ai-config/agents/development/fullstack-engineer.md +0 -668
  34. package/ai-config/agents/development/golang-pro.md +0 -338
  35. package/ai-config/agents/development/java-enterprise.md +0 -400
  36. package/ai-config/agents/development/javascript-pro.md +0 -422
  37. package/ai-config/agents/development/nextjs-pro.md +0 -474
  38. package/ai-config/agents/development/python-pro.md +0 -570
  39. package/ai-config/agents/development/react-pro.md +0 -487
  40. package/ai-config/agents/development/rust-pro.md +0 -246
  41. package/ai-config/agents/development/spring-boot-4-expert.md +0 -326
  42. package/ai-config/agents/development/typescript-pro.md +0 -336
  43. package/ai-config/agents/development/vue-specialist.md +0 -605
  44. package/ai-config/agents/infrastructure/cloud-architect.md +0 -472
  45. package/ai-config/agents/infrastructure/deployment-manager.md +0 -358
  46. package/ai-config/agents/infrastructure/devops-engineer.md +0 -455
  47. package/ai-config/agents/infrastructure/incident-responder.md +0 -519
  48. package/ai-config/agents/infrastructure/kubernetes-expert.md +0 -705
  49. package/ai-config/agents/infrastructure/monitoring-specialist.md +0 -674
  50. package/ai-config/agents/infrastructure/performance-engineer.md +0 -658
  51. package/ai-config/agents/orchestrator.md +0 -241
  52. package/ai-config/agents/quality/accessibility-auditor.md +0 -1204
  53. package/ai-config/agents/quality/code-reviewer-compact.md +0 -123
  54. package/ai-config/agents/quality/code-reviewer.md +0 -363
  55. package/ai-config/agents/quality/dependency-manager.md +0 -743
  56. package/ai-config/agents/quality/e2e-test-specialist.md +0 -1005
  57. package/ai-config/agents/quality/performance-tester.md +0 -1086
  58. package/ai-config/agents/quality/security-auditor.md +0 -133
  59. package/ai-config/agents/quality/test-engineer.md +0 -453
  60. package/ai-config/agents/specialists/api-designer.md +0 -87
  61. package/ai-config/agents/specialists/backend-architect.md +0 -73
  62. package/ai-config/agents/specialists/code-reviewer.md +0 -77
  63. package/ai-config/agents/specialists/db-optimizer.md +0 -75
  64. package/ai-config/agents/specialists/devops-engineer.md +0 -83
  65. package/ai-config/agents/specialists/documentation-writer.md +0 -78
  66. package/ai-config/agents/specialists/frontend-developer.md +0 -75
  67. package/ai-config/agents/specialists/performance-analyst.md +0 -82
  68. package/ai-config/agents/specialists/refactor-specialist.md +0 -74
  69. package/ai-config/agents/specialists/security-auditor.md +0 -74
  70. package/ai-config/agents/specialists/test-engineer.md +0 -81
  71. package/ai-config/agents/specialists/ux-consultant.md +0 -76
  72. package/ai-config/agents/specialized/agent-generator.md +0 -1190
  73. package/ai-config/agents/specialized/blockchain-developer.md +0 -149
  74. package/ai-config/agents/specialized/code-migrator.md +0 -892
  75. package/ai-config/agents/specialized/context-manager.md +0 -978
  76. package/ai-config/agents/specialized/documentation-writer.md +0 -1078
  77. package/ai-config/agents/specialized/ecommerce-expert.md +0 -1756
  78. package/ai-config/agents/specialized/embedded-engineer.md +0 -1714
  79. package/ai-config/agents/specialized/error-detective.md +0 -1034
  80. package/ai-config/agents/specialized/fintech-specialist.md +0 -1659
  81. package/ai-config/agents/specialized/freelance-project-planner-v2.md +0 -1988
  82. package/ai-config/agents/specialized/freelance-project-planner-v3.md +0 -2136
  83. package/ai-config/agents/specialized/freelance-project-planner-v4.md +0 -4503
  84. package/ai-config/agents/specialized/freelance-project-planner.md +0 -722
  85. package/ai-config/agents/specialized/game-developer.md +0 -1963
  86. package/ai-config/agents/specialized/healthcare-dev.md +0 -1620
  87. package/ai-config/agents/specialized/mobile-developer.md +0 -188
  88. package/ai-config/agents/specialized/parallel-plan-executor.md +0 -506
  89. package/ai-config/agents/specialized/plan-executor.md +0 -485
  90. package/ai-config/agents/specialized/solo-dev-planner-modular/00-INDEX.md +0 -485
  91. package/ai-config/agents/specialized/solo-dev-planner-modular/01-CORE.md +0 -3493
  92. package/ai-config/agents/specialized/solo-dev-planner-modular/02-SELF-CORRECTION.md +0 -778
  93. package/ai-config/agents/specialized/solo-dev-planner-modular/03-PROGRESSIVE-SETUP.md +0 -918
  94. package/ai-config/agents/specialized/solo-dev-planner-modular/04-DEPLOYMENT.md +0 -1537
  95. package/ai-config/agents/specialized/solo-dev-planner-modular/05-TESTING.md +0 -2633
  96. package/ai-config/agents/specialized/solo-dev-planner-modular/06-OPERATIONS.md +0 -5610
  97. package/ai-config/agents/specialized/solo-dev-planner-modular/INSTALL.md +0 -335
  98. package/ai-config/agents/specialized/solo-dev-planner-modular/QUICK-REFERENCE.txt +0 -215
  99. package/ai-config/agents/specialized/solo-dev-planner-modular/README.md +0 -260
  100. package/ai-config/agents/specialized/solo-dev-planner-modular/START-HERE.md +0 -379
  101. package/ai-config/agents/specialized/solo-dev-planner-modular/WORKFLOW-DIAGRAM.md +0 -355
  102. package/ai-config/agents/specialized/solo-dev-planner-modular/solo-dev-planner.md +0 -279
  103. package/ai-config/agents/specialized/template-writer.md +0 -347
  104. package/ai-config/agents/specialized/test-runner.md +0 -99
  105. package/ai-config/agents/specialized/vibekanban-smart-worker.md +0 -244
  106. package/ai-config/agents/specialized/wave-executor.md +0 -138
  107. package/ai-config/agents/specialized/workflow-optimizer.md +0 -1114
  108. package/ai-config/commands/git/changelog.md +0 -32
  109. package/ai-config/commands/git/ci-local.md +0 -70
  110. package/ai-config/commands/git/commit.md +0 -35
  111. package/ai-config/commands/git/fix-issue.md +0 -23
  112. package/ai-config/commands/git/pr-create.md +0 -42
  113. package/ai-config/commands/git/pr-review.md +0 -50
  114. package/ai-config/commands/git/worktree.md +0 -39
  115. package/ai-config/commands/refactoring/cleanup.md +0 -24
  116. package/ai-config/commands/refactoring/dead-code.md +0 -40
  117. package/ai-config/commands/refactoring/extract.md +0 -31
  118. package/ai-config/commands/testing/e2e.md +0 -30
  119. package/ai-config/commands/testing/tdd.md +0 -36
  120. package/ai-config/commands/testing/test-coverage.md +0 -30
  121. package/ai-config/commands/testing/test-fix.md +0 -24
  122. package/ai-config/commands/workflow/generate-agents-md.md +0 -85
  123. package/ai-config/commands/workflow/planning.md +0 -47
  124. package/ai-config/commands/workflows/compound.md +0 -89
  125. package/ai-config/commands/workflows/diagnose.md +0 -70
  126. package/ai-config/commands/workflows/discover.md +0 -86
  127. package/ai-config/commands/workflows/plan.md +0 -77
  128. package/ai-config/commands/workflows/review.md +0 -78
  129. package/ai-config/commands/workflows/work.md +0 -75
  130. package/ai-config/config.yaml +0 -18
  131. package/ai-config/hooks/_TEMPLATE.md +0 -96
  132. package/ai-config/hooks/block-dangerous-commands.md +0 -75
  133. package/ai-config/hooks/commit-guard.md +0 -90
  134. package/ai-config/hooks/context-loader.md +0 -73
  135. package/ai-config/hooks/improve-prompt.md +0 -91
  136. package/ai-config/hooks/learning-log.md +0 -72
  137. package/ai-config/hooks/model-router.md +0 -86
  138. package/ai-config/hooks/secret-scanner.md +0 -64
  139. package/ai-config/hooks/skill-validator.md +0 -102
  140. package/ai-config/hooks/task-artifact.md +0 -114
  141. package/ai-config/hooks/validate-workflow.md +0 -100
  142. package/ai-config/prompts/base.md +0 -71
  143. package/ai-config/prompts/modes/debug.md +0 -34
  144. package/ai-config/prompts/modes/deploy.md +0 -40
  145. package/ai-config/prompts/modes/research.md +0 -32
  146. package/ai-config/prompts/modes/review.md +0 -33
  147. package/ai-config/prompts/review-policy.md +0 -79
  148. package/ai-config/skills/_TEMPLATE.md +0 -157
  149. package/ai-config/skills/backend/api-gateway/SKILL.md +0 -254
  150. package/ai-config/skills/backend/bff-concepts/SKILL.md +0 -239
  151. package/ai-config/skills/backend/bff-spring/SKILL.md +0 -364
  152. package/ai-config/skills/backend/chi-router/SKILL.md +0 -396
  153. package/ai-config/skills/backend/error-handling/SKILL.md +0 -255
  154. package/ai-config/skills/backend/exceptions-spring/SKILL.md +0 -323
  155. package/ai-config/skills/backend/fastapi/SKILL.md +0 -302
  156. package/ai-config/skills/backend/gateway-spring/SKILL.md +0 -390
  157. package/ai-config/skills/backend/go-backend/SKILL.md +0 -457
  158. package/ai-config/skills/backend/gradle-multimodule/SKILL.md +0 -274
  159. package/ai-config/skills/backend/graphql-concepts/SKILL.md +0 -352
  160. package/ai-config/skills/backend/graphql-spring/SKILL.md +0 -398
  161. package/ai-config/skills/backend/grpc-concepts/SKILL.md +0 -283
  162. package/ai-config/skills/backend/grpc-spring/SKILL.md +0 -445
  163. package/ai-config/skills/backend/jwt-auth/SKILL.md +0 -412
  164. package/ai-config/skills/backend/notifications-concepts/SKILL.md +0 -259
  165. package/ai-config/skills/backend/recommendations-concepts/SKILL.md +0 -261
  166. package/ai-config/skills/backend/search-concepts/SKILL.md +0 -263
  167. package/ai-config/skills/backend/search-spring/SKILL.md +0 -375
  168. package/ai-config/skills/backend/spring-boot-4/SKILL.md +0 -172
  169. package/ai-config/skills/backend/websockets/SKILL.md +0 -532
  170. package/ai-config/skills/data-ai/ai-ml/SKILL.md +0 -423
  171. package/ai-config/skills/data-ai/analytics-concepts/SKILL.md +0 -195
  172. package/ai-config/skills/data-ai/analytics-spring/SKILL.md +0 -340
  173. package/ai-config/skills/data-ai/duckdb-analytics/SKILL.md +0 -440
  174. package/ai-config/skills/data-ai/langchain/SKILL.md +0 -238
  175. package/ai-config/skills/data-ai/mlflow/SKILL.md +0 -302
  176. package/ai-config/skills/data-ai/onnx-inference/SKILL.md +0 -290
  177. package/ai-config/skills/data-ai/powerbi/SKILL.md +0 -352
  178. package/ai-config/skills/data-ai/pytorch/SKILL.md +0 -274
  179. package/ai-config/skills/data-ai/scikit-learn/SKILL.md +0 -321
  180. package/ai-config/skills/data-ai/vector-db/SKILL.md +0 -301
  181. package/ai-config/skills/database/graph-databases/SKILL.md +0 -218
  182. package/ai-config/skills/database/graph-spring/SKILL.md +0 -361
  183. package/ai-config/skills/database/pgx-postgres/SKILL.md +0 -512
  184. package/ai-config/skills/database/redis-cache/SKILL.md +0 -343
  185. package/ai-config/skills/database/sqlite-embedded/SKILL.md +0 -388
  186. package/ai-config/skills/database/timescaledb/SKILL.md +0 -320
  187. package/ai-config/skills/docs/api-documentation/SKILL.md +0 -293
  188. package/ai-config/skills/docs/docs-spring/SKILL.md +0 -377
  189. package/ai-config/skills/docs/mustache-templates/SKILL.md +0 -190
  190. package/ai-config/skills/docs/technical-docs/SKILL.md +0 -447
  191. package/ai-config/skills/frontend/astro-ssr/SKILL.md +0 -441
  192. package/ai-config/skills/frontend/frontend-design/SKILL.md +0 -54
  193. package/ai-config/skills/frontend/frontend-web/SKILL.md +0 -368
  194. package/ai-config/skills/frontend/mantine-ui/SKILL.md +0 -396
  195. package/ai-config/skills/frontend/tanstack-query/SKILL.md +0 -439
  196. package/ai-config/skills/frontend/zod-validation/SKILL.md +0 -417
  197. package/ai-config/skills/frontend/zustand-state/SKILL.md +0 -350
  198. package/ai-config/skills/infrastructure/chaos-engineering/SKILL.md +0 -244
  199. package/ai-config/skills/infrastructure/chaos-spring/SKILL.md +0 -378
  200. package/ai-config/skills/infrastructure/devops-infra/SKILL.md +0 -435
  201. package/ai-config/skills/infrastructure/docker-containers/SKILL.md +0 -420
  202. package/ai-config/skills/infrastructure/kubernetes/SKILL.md +0 -456
  203. package/ai-config/skills/infrastructure/opentelemetry/SKILL.md +0 -546
  204. package/ai-config/skills/infrastructure/traefik-proxy/SKILL.md +0 -474
  205. package/ai-config/skills/infrastructure/woodpecker-ci/SKILL.md +0 -315
  206. package/ai-config/skills/mobile/ionic-capacitor/SKILL.md +0 -504
  207. package/ai-config/skills/mobile/mobile-ionic/SKILL.md +0 -448
  208. package/ai-config/skills/prompt-improver/SKILL.md +0 -125
  209. package/ai-config/skills/quality/ghagga-review/SKILL.md +0 -216
  210. package/ai-config/skills/references/hooks-patterns/SKILL.md +0 -238
  211. package/ai-config/skills/references/mcp-servers/SKILL.md +0 -275
  212. package/ai-config/skills/references/plugins-reference/SKILL.md +0 -110
  213. package/ai-config/skills/references/skills-reference/SKILL.md +0 -420
  214. package/ai-config/skills/references/subagent-templates/SKILL.md +0 -193
  215. package/ai-config/skills/systems-iot/modbus-protocol/SKILL.md +0 -410
  216. package/ai-config/skills/systems-iot/mqtt-rumqttc/SKILL.md +0 -408
  217. package/ai-config/skills/systems-iot/rust-systems/SKILL.md +0 -386
  218. package/ai-config/skills/systems-iot/tokio-async/SKILL.md +0 -324
  219. package/ai-config/skills/testing/playwright-e2e/SKILL.md +0 -289
  220. package/ai-config/skills/testing/testcontainers/SKILL.md +0 -299
  221. package/ai-config/skills/testing/vitest-testing/SKILL.md +0 -381
  222. package/ai-config/skills/workflow/ci-local-guide/SKILL.md +0 -118
  223. package/ai-config/skills/workflow/claude-automation-recommender/SKILL.md +0 -299
  224. package/ai-config/skills/workflow/claude-md-improver/SKILL.md +0 -158
  225. package/ai-config/skills/workflow/finishing-a-development-branch/SKILL.md +0 -117
  226. package/ai-config/skills/workflow/git-github/SKILL.md +0 -334
  227. package/ai-config/skills/workflow/git-github/references/examples.md +0 -160
  228. package/ai-config/skills/workflow/git-workflow/SKILL.md +0 -214
  229. package/ai-config/skills/workflow/ide-plugins/SKILL.md +0 -277
  230. package/ai-config/skills/workflow/ide-plugins-intellij/SKILL.md +0 -401
  231. package/ai-config/skills/workflow/obsidian-brain-workflow/SKILL.md +0 -199
  232. package/ai-config/skills/workflow/using-git-worktrees/SKILL.md +0 -100
  233. package/ai-config/skills/workflow/verification-before-completion/SKILL.md +0 -73
  234. package/ai-config/skills/workflow/wave-workflow/SKILL.md +0 -178
  235. package/schemas/agent.schema.json +0 -34
  236. package/schemas/ai-config.schema.json +0 -28
  237. package/schemas/plugin.schema.json +0 -62
  238. package/schemas/skill.schema.json +0 -44
@@ -1,323 +0,0 @@
1
- ---
2
- name: exceptions-spring
3
- description: >
4
- Spring Boot exception handling. @ControllerAdvice, ProblemDetail, GlobalExceptionHandler.
5
- Trigger: apigen-exceptions, GlobalExceptionHandler, @ControllerAdvice, ProblemDetail
6
- tools:
7
- - Read
8
- - Write
9
- - Edit
10
- - Bash
11
- - Grep
12
- metadata:
13
- author: apigen-team
14
- version: "1.0"
15
- tags: [exceptions, spring-boot, error-handling, java]
16
- scope: ["apigen-exceptions/**", "apigen-core/**/exception/**"]
17
- ---
18
-
19
- # Exceptions Spring Boot (apigen-exceptions)
20
-
21
- ## Configuration
22
-
23
- ```yaml
24
- apigen:
25
- exceptions:
26
- include-stack-trace: ${DEBUG:false}
27
- include-message: always
28
- log-full-details: true
29
- problem-base-uri: https://api.example.com/errors
30
- ```
31
-
32
- ## Base Exception Hierarchy
33
-
34
- ```java
35
- public abstract class ApiException extends RuntimeException {
36
-
37
- private final String errorCode;
38
- private final HttpStatus status;
39
- private final Map<String, Object> context;
40
-
41
- protected ApiException(String message, String errorCode,
42
- HttpStatus status, Map<String, Object> context) {
43
- super(message);
44
- this.errorCode = errorCode;
45
- this.status = status;
46
- this.context = context != null ? context : Map.of();
47
- }
48
-
49
- public ProblemDetail toProblemDetail(String baseUri) {
50
- ProblemDetail problem = ProblemDetail.forStatus(status);
51
- problem.setType(URI.create(baseUri + "/" + errorCode));
52
- problem.setTitle(getTitle());
53
- problem.setDetail(getMessage());
54
- problem.setProperty("code", errorCode);
55
- context.forEach(problem::setProperty);
56
- return problem;
57
- }
58
-
59
- protected abstract String getTitle();
60
- }
61
- ```
62
-
63
- ## Specific Exception Types
64
-
65
- ```java
66
- // Validation exceptions
67
- public class ValidationException extends ApiException {
68
- private final List<FieldError> fieldErrors;
69
-
70
- public ValidationException(List<FieldError> errors) {
71
- super("Validation failed", "VAL-001", HttpStatus.BAD_REQUEST,
72
- Map.of("errors", errors));
73
- this.fieldErrors = errors;
74
- }
75
-
76
- @Override
77
- protected String getTitle() {
78
- return "Validation Error";
79
- }
80
- }
81
-
82
- // Resource exceptions
83
- public class ResourceNotFoundException extends ApiException {
84
- public ResourceNotFoundException(String resourceType, Object id) {
85
- super(String.format("%s with id '%s' not found", resourceType, id),
86
- "RES-404", HttpStatus.NOT_FOUND,
87
- Map.of("resourceType", resourceType, "resourceId", id));
88
- }
89
-
90
- @Override
91
- protected String getTitle() {
92
- return "Resource Not Found";
93
- }
94
- }
95
-
96
- // Business rule exceptions
97
- public class BusinessRuleException extends ApiException {
98
- public BusinessRuleException(String message, String code,
99
- Map<String, Object> context) {
100
- super(message, code, HttpStatus.UNPROCESSABLE_ENTITY, context);
101
- }
102
-
103
- @Override
104
- protected String getTitle() {
105
- return "Business Rule Violation";
106
- }
107
- }
108
-
109
- // Conflict exceptions
110
- public class ConflictException extends ApiException {
111
- public ConflictException(String message, String field, Object value) {
112
- super(message, "CONFLICT-409", HttpStatus.CONFLICT,
113
- Map.of("field", field, "value", value));
114
- }
115
-
116
- @Override
117
- protected String getTitle() {
118
- return "Resource Conflict";
119
- }
120
- }
121
- ```
122
-
123
- ## Global Exception Handler
124
-
125
- ```java
126
- @RestControllerAdvice
127
- @Order(Ordered.HIGHEST_PRECEDENCE)
128
- public class GlobalExceptionHandler {
129
-
130
- private final ExceptionProperties properties;
131
- private final MessageSource messageSource;
132
-
133
- @ExceptionHandler(ApiException.class)
134
- public ProblemDetail handleApiException(ApiException ex,
135
- HttpServletRequest request) {
136
- log.warn("API Exception: {} - {}", ex.getErrorCode(), ex.getMessage());
137
-
138
- ProblemDetail problem = ex.toProblemDetail(properties.getProblemBaseUri());
139
- problem.setInstance(URI.create(request.getRequestURI()));
140
- return problem;
141
- }
142
-
143
- @ExceptionHandler(MethodArgumentNotValidException.class)
144
- public ProblemDetail handleValidation(MethodArgumentNotValidException ex,
145
- HttpServletRequest request) {
146
- List<FieldError> errors = ex.getBindingResult().getFieldErrors()
147
- .stream()
148
- .map(fe -> new FieldError(fe.getField(),
149
- fe.getDefaultMessage(),
150
- fe.getRejectedValue()))
151
- .toList();
152
-
153
- ProblemDetail problem = ProblemDetail.forStatus(HttpStatus.BAD_REQUEST);
154
- problem.setType(URI.create(properties.getProblemBaseUri() + "/validation"));
155
- problem.setTitle("Validation Error");
156
- problem.setDetail(errors.size() + " validation errors");
157
- problem.setProperty("errors", errors);
158
- problem.setInstance(URI.create(request.getRequestURI()));
159
- return problem;
160
- }
161
-
162
- @ExceptionHandler(ConstraintViolationException.class)
163
- public ProblemDetail handleConstraintViolation(
164
- ConstraintViolationException ex,
165
- HttpServletRequest request) {
166
-
167
- List<FieldError> errors = ex.getConstraintViolations().stream()
168
- .map(cv -> new FieldError(
169
- getPropertyPath(cv),
170
- cv.getMessage(),
171
- cv.getInvalidValue()))
172
- .toList();
173
-
174
- ProblemDetail problem = ProblemDetail.forStatus(HttpStatus.BAD_REQUEST);
175
- problem.setType(URI.create(properties.getProblemBaseUri() + "/validation"));
176
- problem.setTitle("Constraint Violation");
177
- problem.setProperty("errors", errors);
178
- return problem;
179
- }
180
-
181
- @ExceptionHandler(DataIntegrityViolationException.class)
182
- public ProblemDetail handleDataIntegrity(DataIntegrityViolationException ex) {
183
- log.error("Data integrity violation", ex);
184
-
185
- ProblemDetail problem = ProblemDetail.forStatus(HttpStatus.CONFLICT);
186
- problem.setTitle("Data Conflict");
187
- problem.setDetail("A data integrity constraint was violated");
188
- // Don't expose DB details
189
- return problem;
190
- }
191
-
192
- @ExceptionHandler(OptimisticLockingFailureException.class)
193
- public ProblemDetail handleOptimisticLock(
194
- OptimisticLockingFailureException ex) {
195
-
196
- ProblemDetail problem = ProblemDetail.forStatus(HttpStatus.CONFLICT);
197
- problem.setTitle("Concurrent Modification");
198
- problem.setDetail("Resource was modified by another request");
199
- problem.setProperty("code", "OPT-LOCK-001");
200
- return problem;
201
- }
202
-
203
- @ExceptionHandler(Exception.class)
204
- public ProblemDetail handleGeneric(Exception ex, HttpServletRequest request) {
205
- String requestId = UUID.randomUUID().toString();
206
- log.error("Unhandled exception [requestId={}]", requestId, ex);
207
-
208
- ProblemDetail problem = ProblemDetail.forStatus(
209
- HttpStatus.INTERNAL_SERVER_ERROR);
210
- problem.setTitle("Internal Server Error");
211
- problem.setDetail("An unexpected error occurred");
212
- problem.setProperty("requestId", requestId);
213
-
214
- if (properties.isIncludeStackTrace()) {
215
- problem.setProperty("stackTrace", getStackTrace(ex));
216
- }
217
-
218
- return problem;
219
- }
220
- }
221
- ```
222
-
223
- ## Field Error Record
224
-
225
- ```java
226
- public record FieldError(
227
- String field,
228
- String message,
229
- @JsonInclude(JsonInclude.Include.NON_NULL)
230
- Object rejectedValue
231
- ) {}
232
- ```
233
-
234
- ## Exception Auto Configuration
235
-
236
- ```java
237
- @AutoConfiguration
238
- @ConditionalOnWebApplication(type = ConditionalOnWebApplication.Type.SERVLET)
239
- @EnableConfigurationProperties(ExceptionProperties.class)
240
- public class ExceptionAutoConfiguration {
241
-
242
- @Bean
243
- @ConditionalOnMissingBean
244
- public GlobalExceptionHandler globalExceptionHandler(
245
- ExceptionProperties properties,
246
- @Autowired(required = false) MessageSource messageSource) {
247
- return new GlobalExceptionHandler(properties, messageSource);
248
- }
249
-
250
- @Bean
251
- public ProblemDetailExceptionHandler problemDetailHandler() {
252
- return new ProblemDetailExceptionHandler();
253
- }
254
- }
255
- ```
256
-
257
- ## Custom Problem Detail Extensions
258
-
259
- ```java
260
- public class ExtendedProblemDetail extends ProblemDetail {
261
-
262
- private String code;
263
- private Instant timestamp;
264
- private String traceId;
265
-
266
- public static ExtendedProblemDetail forStatusAndCode(
267
- HttpStatus status, String code) {
268
-
269
- ExtendedProblemDetail problem = new ExtendedProblemDetail();
270
- problem.setStatus(status.value());
271
- problem.setCode(code);
272
- problem.setTimestamp(Instant.now());
273
- problem.setTraceId(MDC.get("traceId"));
274
- return problem;
275
- }
276
- }
277
- ```
278
-
279
- ## Testing Exceptions
280
-
281
- ```java
282
- @WebMvcTest(UserController.class)
283
- class UserControllerExceptionTest {
284
-
285
- @Autowired
286
- private MockMvc mockMvc;
287
-
288
- @MockBean
289
- private UserService userService;
290
-
291
- @Test
292
- void shouldReturnProblemDetailForNotFound() throws Exception {
293
- when(userService.findById(any()))
294
- .thenThrow(new ResourceNotFoundException("User", "123"));
295
-
296
- mockMvc.perform(get("/api/users/123"))
297
- .andExpect(status().isNotFound())
298
- .andExpect(content().contentType(MediaType.APPLICATION_PROBLEM_JSON))
299
- .andExpect(jsonPath("$.type").value(containsString("RES-404")))
300
- .andExpect(jsonPath("$.title").value("Resource Not Found"))
301
- .andExpect(jsonPath("$.resourceType").value("User"));
302
- }
303
-
304
- @Test
305
- void shouldReturnValidationErrors() throws Exception {
306
- String invalidJson = """
307
- {"email": "not-an-email", "name": ""}
308
- """;
309
-
310
- mockMvc.perform(post("/api/users")
311
- .contentType(MediaType.APPLICATION_JSON)
312
- .content(invalidJson))
313
- .andExpect(status().isBadRequest())
314
- .andExpect(jsonPath("$.errors").isArray())
315
- .andExpect(jsonPath("$.errors.length()").value(2));
316
- }
317
- }
318
- ```
319
-
320
- ## Related Skills
321
-
322
- - `error-handling`: Error handling concepts
323
- - `spring-boot-4`: Spring Boot 4.0 patterns
@@ -1,302 +0,0 @@
1
- ---
2
- name: fastapi
3
- description: >
4
- FastAPI development patterns with Pydantic v2, async services, and dependency injection.
5
- Trigger: fastapi, python api, async python, pydantic, uvicorn
6
- tools:
7
- - Read
8
- - Write
9
- - Bash
10
- - Grep
11
- metadata:
12
- author: plataforma-industrial
13
- version: "2.0"
14
- tags: [python, fastapi, async, api]
15
- updated: "2026-02"
16
- ---
17
-
18
- # FastAPI Development Skill
19
-
20
- ## Stack
21
-
22
- ```txt
23
- fastapi==0.110.0
24
- uvicorn[standard]==0.28.0
25
- pydantic==2.6.3
26
- pydantic-settings==2.2.1
27
- python-jose[cryptography]==3.3.0
28
- httpx==0.27.0
29
- redis==5.0.2
30
- sqlalchemy==2.0.28
31
- asyncpg==0.29.0
32
- ```
33
-
34
- ## Project Structure
35
-
36
- ```
37
- src/
38
- └── app_name/
39
- ├── __init__.py
40
- ├── main.py
41
- ├── config.py
42
- ├── api/
43
- │ ├── deps.py
44
- │ └── routes/
45
- ├── core/
46
- │ ├── security.py
47
- │ └── exceptions.py
48
- ├── models/
49
- ├── services/
50
- └── agents/
51
- ```
52
-
53
- ## Main Application Pattern
54
-
55
- ```python
56
- # main.py
57
- from contextlib import asynccontextmanager
58
- from fastapi import FastAPI
59
- from fastapi.middleware.cors import CORSMiddleware
60
-
61
- @asynccontextmanager
62
- async def lifespan(app: FastAPI):
63
- # Startup
64
- app.state.cache = CacheService()
65
- await app.state.cache.connect()
66
- yield
67
- # Shutdown
68
- await app.state.cache.disconnect()
69
-
70
- app = FastAPI(
71
- title="Service Name",
72
- version="1.0.0",
73
- lifespan=lifespan,
74
- )
75
-
76
- app.add_middleware(
77
- CORSMiddleware,
78
- allow_origins=settings.cors_origins,
79
- allow_credentials=True,
80
- allow_methods=["*"],
81
- allow_headers=["*"],
82
- )
83
-
84
- app.include_router(health.router, tags=["health"])
85
- app.include_router(api.router, prefix="/api/v1", tags=["api"])
86
- ```
87
-
88
- ## Configuration with Pydantic Settings
89
-
90
- ```python
91
- # config.py
92
- from pydantic_settings import BaseSettings
93
- from functools import lru_cache
94
-
95
- class Settings(BaseSettings):
96
- debug: bool = False
97
- cors_origins: list[str] = ["*"]
98
- database_url: str = "postgresql+asyncpg://user:pass@localhost/db"
99
- redis_url: str = "redis://localhost:6379"
100
- jwt_secret: str
101
- jwt_algorithm: str = "HS256"
102
-
103
- class Config:
104
- env_file = ".env"
105
- case_sensitive = False
106
-
107
- @lru_cache
108
- def get_settings() -> Settings:
109
- return Settings()
110
-
111
- settings = get_settings()
112
- ```
113
-
114
- ## Pydantic Models (v2)
115
-
116
- ```python
117
- from pydantic import BaseModel, Field
118
- from datetime import datetime
119
- from typing import Literal
120
-
121
- class MessageRequest(BaseModel):
122
- content: str = Field(..., min_length=1, max_length=4000)
123
- context: dict | None = None
124
-
125
- class MessageResponse(BaseModel):
126
- message: str
127
- timestamp: datetime = Field(default_factory=datetime.utcnow)
128
-
129
- class StreamChunk(BaseModel):
130
- content: str
131
- done: bool = False
132
- ```
133
-
134
- ## Dependency Injection
135
-
136
- ```python
137
- # api/deps.py
138
- from typing import Annotated
139
- from fastapi import Depends, HTTPException, Header, status
140
- from jose import jwt, JWTError
141
-
142
- async def get_cache(request: Request) -> CacheService:
143
- return request.app.state.cache
144
-
145
- async def verify_token(authorization: Annotated[str, Header()]) -> dict:
146
- if not authorization.startswith("Bearer "):
147
- raise HTTPException(status_code=status.HTTP_401_UNAUTHORIZED)
148
-
149
- token = authorization[7:]
150
- try:
151
- return jwt.decode(token, settings.jwt_secret, algorithms=[settings.jwt_algorithm])
152
- except JWTError:
153
- raise HTTPException(status_code=status.HTTP_401_UNAUTHORIZED)
154
-
155
- async def get_tenant_id(token: Annotated[dict, Depends(verify_token)]) -> str:
156
- tenant_id = token.get("tenant_id")
157
- if not tenant_id:
158
- raise HTTPException(status_code=status.HTTP_403_FORBIDDEN)
159
- return tenant_id
160
-
161
- # Type aliases
162
- TokenDep = Annotated[dict, Depends(verify_token)]
163
- TenantDep = Annotated[str, Depends(get_tenant_id)]
164
- CacheDep = Annotated[CacheService, Depends(get_cache)]
165
- ```
166
-
167
- ## Routes
168
-
169
- ```python
170
- # api/routes/items.py
171
- from fastapi import APIRouter
172
- from fastapi.responses import StreamingResponse
173
-
174
- router = APIRouter()
175
-
176
- @router.post("", response_model=ItemResponse)
177
- async def create_item(
178
- request: ItemRequest,
179
- tenant_id: TenantDep,
180
- service: ServiceDep,
181
- ):
182
- return await service.create(request, tenant_id)
183
-
184
- @router.post("/stream")
185
- async def stream_response(request: Request, service: ServiceDep):
186
- async def generate():
187
- async for chunk in service.stream():
188
- yield f"data: {chunk.model_dump_json()}\n\n"
189
- yield "data: [DONE]\n\n"
190
-
191
- return StreamingResponse(generate(), media_type="text/event-stream")
192
- ```
193
-
194
- ## Async Service Pattern
195
-
196
- ```python
197
- class ItemService:
198
- def __init__(self, cache: CacheService):
199
- self.cache = cache
200
-
201
- async def create(self, request: ItemRequest, tenant_id: str) -> ItemResponse:
202
- # Business logic here
203
- result = await self._process(request)
204
- await self.cache.set(f"item:{result.id}", result.model_dump())
205
- return result
206
-
207
- async def stream(self) -> AsyncIterator[StreamChunk]:
208
- async for chunk in self._generate():
209
- yield StreamChunk(content=chunk)
210
- yield StreamChunk(content="", done=True)
211
- ```
212
-
213
- ## Redis Cache Service
214
-
215
- ```python
216
- import redis.asyncio as redis
217
- import json
218
-
219
- class CacheService:
220
- def __init__(self):
221
- self.redis: redis.Redis | None = None
222
-
223
- async def connect(self):
224
- self.redis = await redis.from_url(settings.redis_url)
225
-
226
- async def disconnect(self):
227
- if self.redis:
228
- await self.redis.close()
229
-
230
- async def get(self, key: str) -> Any | None:
231
- data = await self.redis.get(key)
232
- return json.loads(data) if data else None
233
-
234
- async def set(self, key: str, value: Any, ttl: int = 3600):
235
- await self.redis.setex(key, ttl, json.dumps(value))
236
- ```
237
-
238
- ## Custom Exceptions
239
-
240
- ```python
241
- from fastapi import HTTPException, status
242
-
243
- class ServiceException(HTTPException):
244
- def __init__(self, detail: str):
245
- super().__init__(status_code=status.HTTP_500_INTERNAL_SERVER_ERROR, detail=detail)
246
-
247
- class RateLimitException(HTTPException):
248
- def __init__(self):
249
- super().__init__(status_code=status.HTTP_429_TOO_MANY_REQUESTS, detail="Rate limit exceeded")
250
- ```
251
-
252
- ## Testing
253
-
254
- ```python
255
- import pytest
256
- from httpx import AsyncClient
257
- from app.main import app
258
-
259
- @pytest.fixture
260
- async def client():
261
- async with AsyncClient(app=app, base_url="http://test") as ac:
262
- yield ac
263
-
264
- @pytest.mark.asyncio
265
- async def test_create_item(client: AsyncClient, auth_token: str):
266
- response = await client.post(
267
- "/api/v1/items",
268
- json={"content": "test"},
269
- headers={"Authorization": f"Bearer {auth_token}"},
270
- )
271
- assert response.status_code == 200
272
- assert "message" in response.json()
273
- ```
274
-
275
- ## Dockerfile
276
-
277
- ```dockerfile
278
- FROM python:3.12-slim
279
- WORKDIR /app
280
- COPY requirements.txt .
281
- RUN pip install --no-cache-dir -r requirements.txt
282
- COPY src/ ./src/
283
- ENV PYTHONPATH=/app/src
284
- EXPOSE 8000
285
- CMD ["uvicorn", "app.main:app", "--host", "0.0.0.0", "--port", "8000"]
286
- ```
287
-
288
- ## Best Practices
289
-
290
- 1. **Use Pydantic v2**: `BaseModel`, `Field`, `model_dump()`, `model_validate_json()`
291
- 2. **Async everywhere**: Never block the event loop with sync operations
292
- 3. **Dependency injection**: Use `Annotated[Type, Depends(fn)]` pattern
293
- 4. **Type hints**: Full typing for all functions and returns
294
- 5. **Lifespan context**: Manage startup/shutdown with `asynccontextmanager`
295
-
296
- ## Related Skills
297
-
298
- - `jwt-auth`: Authentication patterns
299
- - `redis-cache`: Response caching
300
- - `langchain`: LLM integration
301
- - `opentelemetry`: Observability
302
- - `docker-containers`: Deployment