javi-forge 1.2.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 (228) hide show
  1. package/ci-local/ci-local.sh +20 -8
  2. package/package.json +1 -1
  3. package/ai-config/.skillignore +0 -15
  4. package/ai-config/AUTO_INVOKE.md +0 -300
  5. package/ai-config/agents/_TEMPLATE.md +0 -93
  6. package/ai-config/agents/business/api-designer.md +0 -1657
  7. package/ai-config/agents/business/business-analyst.md +0 -1331
  8. package/ai-config/agents/business/product-strategist.md +0 -206
  9. package/ai-config/agents/business/project-manager.md +0 -178
  10. package/ai-config/agents/business/requirements-analyst.md +0 -1277
  11. package/ai-config/agents/business/technical-writer.md +0 -1679
  12. package/ai-config/agents/creative/ux-designer.md +0 -205
  13. package/ai-config/agents/data-ai/ai-engineer.md +0 -487
  14. package/ai-config/agents/data-ai/analytics-engineer.md +0 -953
  15. package/ai-config/agents/data-ai/data-engineer.md +0 -173
  16. package/ai-config/agents/data-ai/data-scientist.md +0 -672
  17. package/ai-config/agents/data-ai/mlops-engineer.md +0 -814
  18. package/ai-config/agents/data-ai/prompt-engineer.md +0 -772
  19. package/ai-config/agents/development/angular-expert.md +0 -620
  20. package/ai-config/agents/development/backend-architect.md +0 -795
  21. package/ai-config/agents/development/database-specialist.md +0 -212
  22. package/ai-config/agents/development/frontend-specialist.md +0 -686
  23. package/ai-config/agents/development/fullstack-engineer.md +0 -668
  24. package/ai-config/agents/development/golang-pro.md +0 -338
  25. package/ai-config/agents/development/java-enterprise.md +0 -400
  26. package/ai-config/agents/development/javascript-pro.md +0 -422
  27. package/ai-config/agents/development/nextjs-pro.md +0 -474
  28. package/ai-config/agents/development/python-pro.md +0 -570
  29. package/ai-config/agents/development/react-pro.md +0 -487
  30. package/ai-config/agents/development/rust-pro.md +0 -246
  31. package/ai-config/agents/development/spring-boot-4-expert.md +0 -326
  32. package/ai-config/agents/development/typescript-pro.md +0 -336
  33. package/ai-config/agents/development/vue-specialist.md +0 -605
  34. package/ai-config/agents/infrastructure/cloud-architect.md +0 -472
  35. package/ai-config/agents/infrastructure/deployment-manager.md +0 -358
  36. package/ai-config/agents/infrastructure/devops-engineer.md +0 -455
  37. package/ai-config/agents/infrastructure/incident-responder.md +0 -519
  38. package/ai-config/agents/infrastructure/kubernetes-expert.md +0 -705
  39. package/ai-config/agents/infrastructure/monitoring-specialist.md +0 -674
  40. package/ai-config/agents/infrastructure/performance-engineer.md +0 -658
  41. package/ai-config/agents/orchestrator.md +0 -241
  42. package/ai-config/agents/quality/accessibility-auditor.md +0 -1204
  43. package/ai-config/agents/quality/code-reviewer-compact.md +0 -123
  44. package/ai-config/agents/quality/code-reviewer.md +0 -363
  45. package/ai-config/agents/quality/dependency-manager.md +0 -743
  46. package/ai-config/agents/quality/e2e-test-specialist.md +0 -1005
  47. package/ai-config/agents/quality/performance-tester.md +0 -1086
  48. package/ai-config/agents/quality/security-auditor.md +0 -133
  49. package/ai-config/agents/quality/test-engineer.md +0 -453
  50. package/ai-config/agents/specialists/api-designer.md +0 -87
  51. package/ai-config/agents/specialists/backend-architect.md +0 -73
  52. package/ai-config/agents/specialists/code-reviewer.md +0 -77
  53. package/ai-config/agents/specialists/db-optimizer.md +0 -75
  54. package/ai-config/agents/specialists/devops-engineer.md +0 -83
  55. package/ai-config/agents/specialists/documentation-writer.md +0 -78
  56. package/ai-config/agents/specialists/frontend-developer.md +0 -75
  57. package/ai-config/agents/specialists/performance-analyst.md +0 -82
  58. package/ai-config/agents/specialists/refactor-specialist.md +0 -74
  59. package/ai-config/agents/specialists/security-auditor.md +0 -74
  60. package/ai-config/agents/specialists/test-engineer.md +0 -81
  61. package/ai-config/agents/specialists/ux-consultant.md +0 -76
  62. package/ai-config/agents/specialized/agent-generator.md +0 -1190
  63. package/ai-config/agents/specialized/blockchain-developer.md +0 -149
  64. package/ai-config/agents/specialized/code-migrator.md +0 -892
  65. package/ai-config/agents/specialized/context-manager.md +0 -978
  66. package/ai-config/agents/specialized/documentation-writer.md +0 -1078
  67. package/ai-config/agents/specialized/ecommerce-expert.md +0 -1756
  68. package/ai-config/agents/specialized/embedded-engineer.md +0 -1714
  69. package/ai-config/agents/specialized/error-detective.md +0 -1034
  70. package/ai-config/agents/specialized/fintech-specialist.md +0 -1659
  71. package/ai-config/agents/specialized/freelance-project-planner-v2.md +0 -1988
  72. package/ai-config/agents/specialized/freelance-project-planner-v3.md +0 -2136
  73. package/ai-config/agents/specialized/freelance-project-planner-v4.md +0 -4503
  74. package/ai-config/agents/specialized/freelance-project-planner.md +0 -722
  75. package/ai-config/agents/specialized/game-developer.md +0 -1963
  76. package/ai-config/agents/specialized/healthcare-dev.md +0 -1620
  77. package/ai-config/agents/specialized/mobile-developer.md +0 -188
  78. package/ai-config/agents/specialized/parallel-plan-executor.md +0 -506
  79. package/ai-config/agents/specialized/plan-executor.md +0 -485
  80. package/ai-config/agents/specialized/solo-dev-planner-modular/00-INDEX.md +0 -485
  81. package/ai-config/agents/specialized/solo-dev-planner-modular/01-CORE.md +0 -3493
  82. package/ai-config/agents/specialized/solo-dev-planner-modular/02-SELF-CORRECTION.md +0 -778
  83. package/ai-config/agents/specialized/solo-dev-planner-modular/03-PROGRESSIVE-SETUP.md +0 -918
  84. package/ai-config/agents/specialized/solo-dev-planner-modular/04-DEPLOYMENT.md +0 -1537
  85. package/ai-config/agents/specialized/solo-dev-planner-modular/05-TESTING.md +0 -2633
  86. package/ai-config/agents/specialized/solo-dev-planner-modular/06-OPERATIONS.md +0 -5610
  87. package/ai-config/agents/specialized/solo-dev-planner-modular/INSTALL.md +0 -335
  88. package/ai-config/agents/specialized/solo-dev-planner-modular/QUICK-REFERENCE.txt +0 -215
  89. package/ai-config/agents/specialized/solo-dev-planner-modular/README.md +0 -260
  90. package/ai-config/agents/specialized/solo-dev-planner-modular/START-HERE.md +0 -379
  91. package/ai-config/agents/specialized/solo-dev-planner-modular/WORKFLOW-DIAGRAM.md +0 -355
  92. package/ai-config/agents/specialized/solo-dev-planner-modular/solo-dev-planner.md +0 -279
  93. package/ai-config/agents/specialized/template-writer.md +0 -347
  94. package/ai-config/agents/specialized/test-runner.md +0 -99
  95. package/ai-config/agents/specialized/vibekanban-smart-worker.md +0 -244
  96. package/ai-config/agents/specialized/wave-executor.md +0 -138
  97. package/ai-config/agents/specialized/workflow-optimizer.md +0 -1114
  98. package/ai-config/commands/git/changelog.md +0 -32
  99. package/ai-config/commands/git/ci-local.md +0 -70
  100. package/ai-config/commands/git/commit.md +0 -35
  101. package/ai-config/commands/git/fix-issue.md +0 -23
  102. package/ai-config/commands/git/pr-create.md +0 -42
  103. package/ai-config/commands/git/pr-review.md +0 -50
  104. package/ai-config/commands/git/worktree.md +0 -39
  105. package/ai-config/commands/refactoring/cleanup.md +0 -24
  106. package/ai-config/commands/refactoring/dead-code.md +0 -40
  107. package/ai-config/commands/refactoring/extract.md +0 -31
  108. package/ai-config/commands/testing/e2e.md +0 -30
  109. package/ai-config/commands/testing/tdd.md +0 -36
  110. package/ai-config/commands/testing/test-coverage.md +0 -30
  111. package/ai-config/commands/testing/test-fix.md +0 -24
  112. package/ai-config/commands/workflow/generate-agents-md.md +0 -85
  113. package/ai-config/commands/workflow/planning.md +0 -47
  114. package/ai-config/commands/workflows/compound.md +0 -89
  115. package/ai-config/commands/workflows/diagnose.md +0 -70
  116. package/ai-config/commands/workflows/discover.md +0 -86
  117. package/ai-config/commands/workflows/plan.md +0 -77
  118. package/ai-config/commands/workflows/review.md +0 -78
  119. package/ai-config/commands/workflows/work.md +0 -75
  120. package/ai-config/config.yaml +0 -18
  121. package/ai-config/hooks/_TEMPLATE.md +0 -96
  122. package/ai-config/hooks/block-dangerous-commands.md +0 -75
  123. package/ai-config/hooks/commit-guard.md +0 -90
  124. package/ai-config/hooks/context-loader.md +0 -73
  125. package/ai-config/hooks/improve-prompt.md +0 -91
  126. package/ai-config/hooks/learning-log.md +0 -72
  127. package/ai-config/hooks/model-router.md +0 -86
  128. package/ai-config/hooks/secret-scanner.md +0 -64
  129. package/ai-config/hooks/skill-validator.md +0 -102
  130. package/ai-config/hooks/task-artifact.md +0 -114
  131. package/ai-config/hooks/validate-workflow.md +0 -100
  132. package/ai-config/prompts/base.md +0 -71
  133. package/ai-config/prompts/modes/debug.md +0 -34
  134. package/ai-config/prompts/modes/deploy.md +0 -40
  135. package/ai-config/prompts/modes/research.md +0 -32
  136. package/ai-config/prompts/modes/review.md +0 -33
  137. package/ai-config/prompts/review-policy.md +0 -79
  138. package/ai-config/skills/_TEMPLATE.md +0 -157
  139. package/ai-config/skills/backend/api-gateway/SKILL.md +0 -254
  140. package/ai-config/skills/backend/bff-concepts/SKILL.md +0 -239
  141. package/ai-config/skills/backend/bff-spring/SKILL.md +0 -364
  142. package/ai-config/skills/backend/chi-router/SKILL.md +0 -396
  143. package/ai-config/skills/backend/error-handling/SKILL.md +0 -255
  144. package/ai-config/skills/backend/exceptions-spring/SKILL.md +0 -323
  145. package/ai-config/skills/backend/fastapi/SKILL.md +0 -302
  146. package/ai-config/skills/backend/gateway-spring/SKILL.md +0 -390
  147. package/ai-config/skills/backend/go-backend/SKILL.md +0 -457
  148. package/ai-config/skills/backend/gradle-multimodule/SKILL.md +0 -274
  149. package/ai-config/skills/backend/graphql-concepts/SKILL.md +0 -352
  150. package/ai-config/skills/backend/graphql-spring/SKILL.md +0 -398
  151. package/ai-config/skills/backend/grpc-concepts/SKILL.md +0 -283
  152. package/ai-config/skills/backend/grpc-spring/SKILL.md +0 -445
  153. package/ai-config/skills/backend/jwt-auth/SKILL.md +0 -412
  154. package/ai-config/skills/backend/notifications-concepts/SKILL.md +0 -259
  155. package/ai-config/skills/backend/recommendations-concepts/SKILL.md +0 -261
  156. package/ai-config/skills/backend/search-concepts/SKILL.md +0 -263
  157. package/ai-config/skills/backend/search-spring/SKILL.md +0 -375
  158. package/ai-config/skills/backend/spring-boot-4/SKILL.md +0 -172
  159. package/ai-config/skills/backend/websockets/SKILL.md +0 -532
  160. package/ai-config/skills/data-ai/ai-ml/SKILL.md +0 -423
  161. package/ai-config/skills/data-ai/analytics-concepts/SKILL.md +0 -195
  162. package/ai-config/skills/data-ai/analytics-spring/SKILL.md +0 -340
  163. package/ai-config/skills/data-ai/duckdb-analytics/SKILL.md +0 -440
  164. package/ai-config/skills/data-ai/langchain/SKILL.md +0 -238
  165. package/ai-config/skills/data-ai/mlflow/SKILL.md +0 -302
  166. package/ai-config/skills/data-ai/onnx-inference/SKILL.md +0 -290
  167. package/ai-config/skills/data-ai/powerbi/SKILL.md +0 -352
  168. package/ai-config/skills/data-ai/pytorch/SKILL.md +0 -274
  169. package/ai-config/skills/data-ai/scikit-learn/SKILL.md +0 -321
  170. package/ai-config/skills/data-ai/vector-db/SKILL.md +0 -301
  171. package/ai-config/skills/database/graph-databases/SKILL.md +0 -218
  172. package/ai-config/skills/database/graph-spring/SKILL.md +0 -361
  173. package/ai-config/skills/database/pgx-postgres/SKILL.md +0 -512
  174. package/ai-config/skills/database/redis-cache/SKILL.md +0 -343
  175. package/ai-config/skills/database/sqlite-embedded/SKILL.md +0 -388
  176. package/ai-config/skills/database/timescaledb/SKILL.md +0 -320
  177. package/ai-config/skills/docs/api-documentation/SKILL.md +0 -293
  178. package/ai-config/skills/docs/docs-spring/SKILL.md +0 -377
  179. package/ai-config/skills/docs/mustache-templates/SKILL.md +0 -190
  180. package/ai-config/skills/docs/technical-docs/SKILL.md +0 -447
  181. package/ai-config/skills/frontend/astro-ssr/SKILL.md +0 -441
  182. package/ai-config/skills/frontend/frontend-design/SKILL.md +0 -54
  183. package/ai-config/skills/frontend/frontend-web/SKILL.md +0 -368
  184. package/ai-config/skills/frontend/mantine-ui/SKILL.md +0 -396
  185. package/ai-config/skills/frontend/tanstack-query/SKILL.md +0 -439
  186. package/ai-config/skills/frontend/zod-validation/SKILL.md +0 -417
  187. package/ai-config/skills/frontend/zustand-state/SKILL.md +0 -350
  188. package/ai-config/skills/infrastructure/chaos-engineering/SKILL.md +0 -244
  189. package/ai-config/skills/infrastructure/chaos-spring/SKILL.md +0 -378
  190. package/ai-config/skills/infrastructure/devops-infra/SKILL.md +0 -435
  191. package/ai-config/skills/infrastructure/docker-containers/SKILL.md +0 -420
  192. package/ai-config/skills/infrastructure/kubernetes/SKILL.md +0 -456
  193. package/ai-config/skills/infrastructure/opentelemetry/SKILL.md +0 -546
  194. package/ai-config/skills/infrastructure/traefik-proxy/SKILL.md +0 -474
  195. package/ai-config/skills/infrastructure/woodpecker-ci/SKILL.md +0 -315
  196. package/ai-config/skills/mobile/ionic-capacitor/SKILL.md +0 -504
  197. package/ai-config/skills/mobile/mobile-ionic/SKILL.md +0 -448
  198. package/ai-config/skills/prompt-improver/SKILL.md +0 -125
  199. package/ai-config/skills/quality/ghagga-review/SKILL.md +0 -216
  200. package/ai-config/skills/references/hooks-patterns/SKILL.md +0 -238
  201. package/ai-config/skills/references/mcp-servers/SKILL.md +0 -275
  202. package/ai-config/skills/references/plugins-reference/SKILL.md +0 -110
  203. package/ai-config/skills/references/skills-reference/SKILL.md +0 -420
  204. package/ai-config/skills/references/subagent-templates/SKILL.md +0 -193
  205. package/ai-config/skills/systems-iot/modbus-protocol/SKILL.md +0 -410
  206. package/ai-config/skills/systems-iot/mqtt-rumqttc/SKILL.md +0 -408
  207. package/ai-config/skills/systems-iot/rust-systems/SKILL.md +0 -386
  208. package/ai-config/skills/systems-iot/tokio-async/SKILL.md +0 -324
  209. package/ai-config/skills/testing/playwright-e2e/SKILL.md +0 -289
  210. package/ai-config/skills/testing/testcontainers/SKILL.md +0 -299
  211. package/ai-config/skills/testing/vitest-testing/SKILL.md +0 -381
  212. package/ai-config/skills/workflow/ci-local-guide/SKILL.md +0 -118
  213. package/ai-config/skills/workflow/claude-automation-recommender/SKILL.md +0 -299
  214. package/ai-config/skills/workflow/claude-md-improver/SKILL.md +0 -158
  215. package/ai-config/skills/workflow/finishing-a-development-branch/SKILL.md +0 -117
  216. package/ai-config/skills/workflow/git-github/SKILL.md +0 -334
  217. package/ai-config/skills/workflow/git-github/references/examples.md +0 -160
  218. package/ai-config/skills/workflow/git-workflow/SKILL.md +0 -214
  219. package/ai-config/skills/workflow/ide-plugins/SKILL.md +0 -277
  220. package/ai-config/skills/workflow/ide-plugins-intellij/SKILL.md +0 -401
  221. package/ai-config/skills/workflow/obsidian-brain-workflow/SKILL.md +0 -199
  222. package/ai-config/skills/workflow/using-git-worktrees/SKILL.md +0 -100
  223. package/ai-config/skills/workflow/verification-before-completion/SKILL.md +0 -73
  224. package/ai-config/skills/workflow/wave-workflow/SKILL.md +0 -178
  225. package/schemas/agent.schema.json +0 -34
  226. package/schemas/ai-config.schema.json +0 -28
  227. package/schemas/plugin.schema.json +0 -62
  228. 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