agim-cli 1.2.144 → 1.2.148

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 (611) hide show
  1. package/CHANGELOG.md +159 -0
  2. package/dist/cli-ui/setup-llm.d.ts.map +1 -1
  3. package/dist/cli-ui/setup-llm.js +3 -1
  4. package/dist/cli-ui/setup-llm.js.map +1 -1
  5. package/dist/core/circuit-breaker.d.ts +28 -0
  6. package/dist/core/circuit-breaker.d.ts.map +1 -1
  7. package/dist/core/circuit-breaker.js +45 -0
  8. package/dist/core/circuit-breaker.js.map +1 -1
  9. package/dist/core/intent.d.ts.map +1 -1
  10. package/dist/core/intent.js +3 -1
  11. package/dist/core/intent.js.map +1 -1
  12. package/dist/core/llm/agent-loop.d.ts +9 -1
  13. package/dist/core/llm/agent-loop.d.ts.map +1 -1
  14. package/dist/core/llm/agent-loop.js +80 -1
  15. package/dist/core/llm/agent-loop.js.map +1 -1
  16. package/dist/core/llm/anthropic-provider.d.ts.map +1 -1
  17. package/dist/core/llm/anthropic-provider.js +18 -4
  18. package/dist/core/llm/anthropic-provider.js.map +1 -1
  19. package/dist/core/llm/hallucination-detector.d.ts +33 -0
  20. package/dist/core/llm/hallucination-detector.d.ts.map +1 -0
  21. package/dist/core/llm/hallucination-detector.js +103 -0
  22. package/dist/core/llm/hallucination-detector.js.map +1 -0
  23. package/dist/core/llm/imhub-dispatcher.d.ts.map +1 -1
  24. package/dist/core/llm/imhub-dispatcher.js +7 -0
  25. package/dist/core/llm/imhub-dispatcher.js.map +1 -1
  26. package/dist/core/llm/provider-base.d.ts +9 -0
  27. package/dist/core/llm/provider-base.d.ts.map +1 -1
  28. package/dist/core/llm/provider-base.js.map +1 -1
  29. package/dist/core/memory-distill.d.ts.map +1 -1
  30. package/dist/core/memory-distill.js +18 -3
  31. package/dist/core/memory-distill.js.map +1 -1
  32. package/dist/core/memory.d.ts +14 -0
  33. package/dist/core/memory.d.ts.map +1 -1
  34. package/dist/core/memory.js +39 -0
  35. package/dist/core/memory.js.map +1 -1
  36. package/dist/core/message-sink.d.ts +6 -0
  37. package/dist/core/message-sink.d.ts.map +1 -1
  38. package/dist/core/message-sink.js +18 -3
  39. package/dist/core/message-sink.js.map +1 -1
  40. package/dist/core/outbox.d.ts +30 -2
  41. package/dist/core/outbox.d.ts.map +1 -1
  42. package/dist/core/outbox.js +102 -10
  43. package/dist/core/outbox.js.map +1 -1
  44. package/dist/core/reminders.d.ts.map +1 -1
  45. package/dist/core/reminders.js +11 -1
  46. package/dist/core/reminders.js.map +1 -1
  47. package/dist/core/router.d.ts.map +1 -1
  48. package/dist/core/router.js +16 -4
  49. package/dist/core/router.js.map +1 -1
  50. package/dist/core/schedule.d.ts +18 -0
  51. package/dist/core/schedule.d.ts.map +1 -1
  52. package/dist/core/schedule.js +80 -17
  53. package/dist/core/schedule.js.map +1 -1
  54. package/dist/core/sensitive-paths.d.ts.map +1 -1
  55. package/dist/core/sensitive-paths.js +53 -9
  56. package/dist/core/sensitive-paths.js.map +1 -1
  57. package/dist/core/skills/builtin/ECC_LICENSE +21 -0
  58. package/dist/core/skills/builtin/ECC_NOTICE.md +22 -0
  59. package/dist/core/skills/builtin/accessibility/SKILL.md +146 -0
  60. package/dist/core/skills/builtin/agent-eval/SKILL.md +145 -0
  61. package/dist/core/skills/builtin/agent-harness-construction/SKILL.md +73 -0
  62. package/dist/core/skills/builtin/agent-introspection-debugging/SKILL.md +153 -0
  63. package/dist/core/skills/builtin/agentic-engineering/SKILL.md +63 -0
  64. package/dist/core/skills/builtin/ai-first-engineering/SKILL.md +51 -0
  65. package/dist/core/skills/builtin/ai-regression-testing/SKILL.md +385 -0
  66. package/dist/core/skills/builtin/android-clean-architecture/SKILL.md +339 -0
  67. package/dist/core/skills/builtin/angular-developer/SKILL.md +154 -0
  68. package/dist/core/skills/builtin/angular-developer/references/angular-animations.md +160 -0
  69. package/dist/core/skills/builtin/angular-developer/references/angular-aria.md +410 -0
  70. package/dist/core/skills/builtin/angular-developer/references/cli.md +86 -0
  71. package/dist/core/skills/builtin/angular-developer/references/component-harnesses.md +59 -0
  72. package/dist/core/skills/builtin/angular-developer/references/component-styling.md +91 -0
  73. package/dist/core/skills/builtin/angular-developer/references/components.md +117 -0
  74. package/dist/core/skills/builtin/angular-developer/references/creating-services.md +97 -0
  75. package/dist/core/skills/builtin/angular-developer/references/data-resolvers.md +69 -0
  76. package/dist/core/skills/builtin/angular-developer/references/define-routes.md +67 -0
  77. package/dist/core/skills/builtin/angular-developer/references/defining-providers.md +72 -0
  78. package/dist/core/skills/builtin/angular-developer/references/di-fundamentals.md +120 -0
  79. package/dist/core/skills/builtin/angular-developer/references/e2e-testing.md +56 -0
  80. package/dist/core/skills/builtin/angular-developer/references/effects.md +83 -0
  81. package/dist/core/skills/builtin/angular-developer/references/hierarchical-injectors.md +43 -0
  82. package/dist/core/skills/builtin/angular-developer/references/host-elements.md +80 -0
  83. package/dist/core/skills/builtin/angular-developer/references/injection-context.md +63 -0
  84. package/dist/core/skills/builtin/angular-developer/references/inputs.md +101 -0
  85. package/dist/core/skills/builtin/angular-developer/references/linked-signal.md +59 -0
  86. package/dist/core/skills/builtin/angular-developer/references/loading-strategies.md +61 -0
  87. package/dist/core/skills/builtin/angular-developer/references/mcp.md +108 -0
  88. package/dist/core/skills/builtin/angular-developer/references/navigate-to-routes.md +69 -0
  89. package/dist/core/skills/builtin/angular-developer/references/outputs.md +86 -0
  90. package/dist/core/skills/builtin/angular-developer/references/reactive-forms.md +122 -0
  91. package/dist/core/skills/builtin/angular-developer/references/rendering-strategies.md +44 -0
  92. package/dist/core/skills/builtin/angular-developer/references/resource.md +77 -0
  93. package/dist/core/skills/builtin/angular-developer/references/route-animations.md +56 -0
  94. package/dist/core/skills/builtin/angular-developer/references/route-guards.md +52 -0
  95. package/dist/core/skills/builtin/angular-developer/references/router-lifecycle.md +45 -0
  96. package/dist/core/skills/builtin/angular-developer/references/router-testing.md +87 -0
  97. package/dist/core/skills/builtin/angular-developer/references/show-routes-with-outlets.md +68 -0
  98. package/dist/core/skills/builtin/angular-developer/references/signal-forms.md +795 -0
  99. package/dist/core/skills/builtin/angular-developer/references/signals-overview.md +94 -0
  100. package/dist/core/skills/builtin/angular-developer/references/tailwind-css.md +69 -0
  101. package/dist/core/skills/builtin/angular-developer/references/template-driven-forms.md +114 -0
  102. package/dist/core/skills/builtin/angular-developer/references/testing-fundamentals.md +65 -0
  103. package/dist/core/skills/builtin/api-connector-builder/SKILL.md +120 -0
  104. package/dist/core/skills/builtin/api-design/SKILL.md +523 -0
  105. package/dist/core/skills/builtin/architecture-decision-records/SKILL.md +179 -0
  106. package/dist/core/skills/builtin/article-writing/SKILL.md +79 -0
  107. package/dist/core/skills/builtin/automation-audit-ops/SKILL.md +142 -0
  108. package/dist/core/skills/builtin/autonomous-agent-harness/SKILL.md +273 -0
  109. package/dist/core/skills/builtin/autonomous-loops/SKILL.md +610 -0
  110. package/dist/core/skills/builtin/backend-patterns/SKILL.md +561 -0
  111. package/dist/core/skills/builtin/benchmark/SKILL.md +93 -0
  112. package/dist/core/skills/builtin/benchmark-optimization-loop/SKILL.md +69 -0
  113. package/dist/core/skills/builtin/blueprint/SKILL.md +105 -0
  114. package/dist/core/skills/builtin/browser-qa/SKILL.md +87 -0
  115. package/dist/core/skills/builtin/bun-runtime/SKILL.md +84 -0
  116. package/dist/core/skills/builtin/cisco-ios-patterns/SKILL.md +163 -0
  117. package/dist/core/skills/builtin/claude-devfleet/SKILL.md +111 -0
  118. package/dist/core/skills/builtin/click-path-audit/SKILL.md +244 -0
  119. package/dist/core/skills/builtin/clickhouse-io/SKILL.md +439 -0
  120. package/dist/core/skills/builtin/code-tour/SKILL.md +236 -0
  121. package/dist/core/skills/builtin/codebase-onboarding/SKILL.md +233 -0
  122. package/dist/core/skills/builtin/codehealth-mcp/SKILL.md +166 -0
  123. package/dist/core/skills/builtin/coding-standards/SKILL.md +550 -0
  124. package/dist/core/skills/builtin/compose-multiplatform-patterns/SKILL.md +299 -0
  125. package/dist/core/skills/builtin/config-gc/SKILL.md +119 -0
  126. package/dist/core/skills/builtin/content-hash-cache-pattern/SKILL.md +161 -0
  127. package/dist/core/skills/builtin/context-budget/SKILL.md +135 -0
  128. package/dist/core/skills/builtin/continuous-agent-loop/SKILL.md +45 -0
  129. package/dist/core/skills/builtin/continuous-learning/SKILL.md +131 -0
  130. package/dist/core/skills/builtin/continuous-learning/config.json +18 -0
  131. package/dist/core/skills/builtin/continuous-learning/evaluate-session.sh +69 -0
  132. package/dist/core/skills/builtin/continuous-learning-v2/SKILL.md +360 -0
  133. package/dist/core/skills/builtin/continuous-learning-v2/agents/observer-loop.sh +335 -0
  134. package/dist/core/skills/builtin/continuous-learning-v2/agents/observer.md +198 -0
  135. package/dist/core/skills/builtin/continuous-learning-v2/agents/session-guardian.sh +150 -0
  136. package/dist/core/skills/builtin/continuous-learning-v2/agents/start-observer.sh +248 -0
  137. package/dist/core/skills/builtin/continuous-learning-v2/config.json +8 -0
  138. package/dist/core/skills/builtin/continuous-learning-v2/hooks/observe.sh +498 -0
  139. package/dist/core/skills/builtin/continuous-learning-v2/scripts/detect-project.sh +322 -0
  140. package/dist/core/skills/builtin/continuous-learning-v2/scripts/instinct-cli.py +1914 -0
  141. package/dist/core/skills/builtin/continuous-learning-v2/scripts/lib/homunculus-dir.sh +31 -0
  142. package/dist/core/skills/builtin/continuous-learning-v2/scripts/migrate-homunculus.sh +62 -0
  143. package/dist/core/skills/builtin/continuous-learning-v2/scripts/test_parse_instinct.py +1045 -0
  144. package/dist/core/skills/builtin/cost-aware-llm-pipeline/SKILL.md +183 -0
  145. package/dist/core/skills/builtin/cost-tracking/SKILL.md +147 -0
  146. package/dist/core/skills/builtin/council/SKILL.md +203 -0
  147. package/dist/core/skills/builtin/cpp-coding-standards/SKILL.md +723 -0
  148. package/dist/core/skills/builtin/cpp-testing/SKILL.md +324 -0
  149. package/dist/core/skills/builtin/crosspost/SKILL.md +111 -0
  150. package/dist/core/skills/builtin/csharp-testing/SKILL.md +321 -0
  151. package/dist/core/skills/builtin/customs-trade-compliance/SKILL.md +263 -0
  152. package/dist/core/skills/builtin/dart-flutter-patterns/SKILL.md +563 -0
  153. package/dist/core/skills/builtin/dashboard-builder/SKILL.md +108 -0
  154. package/dist/core/skills/builtin/data-scraper-agent/SKILL.md +764 -0
  155. package/dist/core/skills/builtin/data-throughput-accelerator/SKILL.md +72 -0
  156. package/dist/core/skills/builtin/database-migrations/SKILL.md +429 -0
  157. package/dist/core/skills/builtin/deep-research/SKILL.md +159 -0
  158. package/dist/core/skills/builtin/defi-amm-security/SKILL.md +166 -0
  159. package/dist/core/skills/builtin/deployment-patterns/SKILL.md +427 -0
  160. package/dist/core/skills/builtin/design-system/SKILL.md +82 -0
  161. package/dist/core/skills/builtin/django-celery/SKILL.md +457 -0
  162. package/dist/core/skills/builtin/django-patterns/SKILL.md +734 -0
  163. package/dist/core/skills/builtin/django-security/SKILL.md +593 -0
  164. package/dist/core/skills/builtin/django-tdd/SKILL.md +729 -0
  165. package/dist/core/skills/builtin/django-verification/SKILL.md +469 -0
  166. package/dist/core/skills/builtin/dmux-workflows/SKILL.md +191 -0
  167. package/dist/core/skills/builtin/docker-patterns/SKILL.md +364 -0
  168. package/dist/core/skills/builtin/documentation-lookup/SKILL.md +90 -0
  169. package/dist/core/skills/builtin/dotnet-patterns/SKILL.md +321 -0
  170. package/dist/core/skills/builtin/dynamic-workflow-mode/SKILL.md +123 -0
  171. package/dist/core/skills/builtin/e2e-testing/SKILL.md +326 -0
  172. package/dist/core/skills/builtin/email-ops/SKILL.md +121 -0
  173. package/dist/core/skills/builtin/energy-procurement/SKILL.md +228 -0
  174. package/dist/core/skills/builtin/enterprise-agent-ops/SKILL.md +50 -0
  175. package/dist/core/skills/builtin/error-handling/SKILL.md +376 -0
  176. package/dist/core/skills/builtin/eval-harness/SKILL.md +270 -0
  177. package/dist/core/skills/builtin/evm-token-decimals/SKILL.md +130 -0
  178. package/dist/core/skills/builtin/exa-search/SKILL.md +107 -0
  179. package/dist/core/skills/builtin/fal-ai-media/SKILL.md +288 -0
  180. package/dist/core/skills/builtin/fastapi-patterns/SKILL.md +513 -0
  181. package/dist/core/skills/builtin/finance-billing-ops/SKILL.md +127 -0
  182. package/dist/core/skills/builtin/flox-environments/SKILL.md +496 -0
  183. package/dist/core/skills/builtin/flutter-dart-code-review/SKILL.md +435 -0
  184. package/dist/core/skills/builtin/foundation-models-on-device/SKILL.md +243 -0
  185. package/dist/core/skills/builtin/frontend-a11y/SKILL.md +445 -0
  186. package/dist/core/skills/builtin/frontend-design-direction/SKILL.md +92 -0
  187. package/dist/core/skills/builtin/frontend-patterns/SKILL.md +656 -0
  188. package/dist/core/skills/builtin/frontend-slides/SKILL.md +184 -0
  189. package/dist/core/skills/builtin/frontend-slides/STYLE_PRESETS.md +330 -0
  190. package/dist/core/skills/builtin/frontend-slides/animation-patterns.md +122 -0
  191. package/dist/core/skills/builtin/frontend-slides/html-template.md +419 -0
  192. package/dist/core/skills/builtin/frontend-slides/scripts/export-pdf.sh +418 -0
  193. package/dist/core/skills/builtin/frontend-slides/scripts/extract-pptx.py +96 -0
  194. package/dist/core/skills/builtin/frontend-slides/viewport-base.css +153 -0
  195. package/dist/core/skills/builtin/fsharp-testing/SKILL.md +280 -0
  196. package/dist/core/skills/builtin/gan-style-harness/SKILL.md +278 -0
  197. package/dist/core/skills/builtin/gateguard/SKILL.md +132 -0
  198. package/dist/core/skills/builtin/git-workflow/SKILL.md +715 -0
  199. package/dist/core/skills/builtin/github-ops/SKILL.md +144 -0
  200. package/dist/core/skills/builtin/golang-patterns/SKILL.md +674 -0
  201. package/dist/core/skills/builtin/golang-testing/SKILL.md +720 -0
  202. package/dist/core/skills/builtin/healthcare-cdss-patterns/SKILL.md +245 -0
  203. package/dist/core/skills/builtin/healthcare-emr-patterns/SKILL.md +159 -0
  204. package/dist/core/skills/builtin/healthcare-eval-harness/SKILL.md +207 -0
  205. package/dist/core/skills/builtin/healthcare-phi-compliance/SKILL.md +145 -0
  206. package/dist/core/skills/builtin/hermes-imports/SKILL.md +88 -0
  207. package/dist/core/skills/builtin/hexagonal-architecture/SKILL.md +276 -0
  208. package/dist/core/skills/builtin/hipaa-compliance/SKILL.md +78 -0
  209. package/dist/core/skills/builtin/hookify-rules/SKILL.md +128 -0
  210. package/dist/core/skills/builtin/inherit-legacy-style/SKILL.md +156 -0
  211. package/dist/core/skills/builtin/intent-driven-development/SKILL.md +360 -0
  212. package/dist/core/skills/builtin/inventory-demand-planning/SKILL.md +247 -0
  213. package/dist/core/skills/builtin/ios-icon-gen/SKILL.md +157 -0
  214. package/dist/core/skills/builtin/ios-icon-gen/scripts/generate_icons.swift +258 -0
  215. package/dist/core/skills/builtin/ios-icon-gen/scripts/iconify_gen.sh +235 -0
  216. package/dist/core/skills/builtin/iterative-retrieval/SKILL.md +211 -0
  217. package/dist/core/skills/builtin/java-coding-standards/SKILL.md +383 -0
  218. package/dist/core/skills/builtin/jira-integration/SKILL.md +302 -0
  219. package/dist/core/skills/builtin/jpa-patterns/SKILL.md +151 -0
  220. package/dist/core/skills/builtin/knowledge-ops/SKILL.md +154 -0
  221. package/dist/core/skills/builtin/kotlin-coroutines-flows/SKILL.md +284 -0
  222. package/dist/core/skills/builtin/kotlin-exposed-patterns/SKILL.md +719 -0
  223. package/dist/core/skills/builtin/kotlin-ktor-patterns/SKILL.md +689 -0
  224. package/dist/core/skills/builtin/kotlin-patterns/SKILL.md +711 -0
  225. package/dist/core/skills/builtin/kotlin-testing/SKILL.md +824 -0
  226. package/dist/core/skills/builtin/kubernetes-patterns/SKILL.md +755 -0
  227. package/dist/core/skills/builtin/laravel-patterns/SKILL.md +415 -0
  228. package/dist/core/skills/builtin/laravel-plugin-discovery/SKILL.md +229 -0
  229. package/dist/core/skills/builtin/laravel-security/SKILL.md +947 -0
  230. package/dist/core/skills/builtin/laravel-tdd/SKILL.md +674 -0
  231. package/dist/core/skills/builtin/laravel-verification/SKILL.md +179 -0
  232. package/dist/core/skills/builtin/latency-critical-systems/SKILL.md +73 -0
  233. package/dist/core/skills/builtin/lead-intelligence/SKILL.md +321 -0
  234. package/dist/core/skills/builtin/lead-intelligence/agents/enrichment-agent.md +85 -0
  235. package/dist/core/skills/builtin/lead-intelligence/agents/mutual-mapper.md +75 -0
  236. package/dist/core/skills/builtin/lead-intelligence/agents/outreach-drafter.md +98 -0
  237. package/dist/core/skills/builtin/lead-intelligence/agents/signal-scorer.md +60 -0
  238. package/dist/core/skills/builtin/liquid-glass-design/SKILL.md +279 -0
  239. package/dist/core/skills/builtin/llm-trading-agent-security/SKILL.md +146 -0
  240. package/dist/core/skills/builtin/logistics-exception-management/SKILL.md +222 -0
  241. package/dist/core/skills/builtin/make-interfaces-feel-better/SKILL.md +151 -0
  242. package/dist/core/skills/builtin/market-research/SKILL.md +75 -0
  243. package/dist/core/skills/builtin/marketing-campaign/SKILL.md +113 -0
  244. package/dist/core/skills/builtin/mcp-server-patterns/SKILL.md +69 -0
  245. package/dist/core/skills/builtin/messages-ops/SKILL.md +104 -0
  246. package/dist/core/skills/builtin/mle-workflow/SKILL.md +346 -0
  247. package/dist/core/skills/builtin/motion-advanced/SKILL.md +596 -0
  248. package/dist/core/skills/builtin/motion-foundations/SKILL.md +299 -0
  249. package/dist/core/skills/builtin/motion-patterns/SKILL.md +434 -0
  250. package/dist/core/skills/builtin/motion-ui/SKILL.md +575 -0
  251. package/dist/core/skills/builtin/mysql-patterns/SKILL.md +412 -0
  252. package/dist/core/skills/builtin/nanoclaw-repl/SKILL.md +33 -0
  253. package/dist/core/skills/builtin/nestjs-patterns/SKILL.md +230 -0
  254. package/dist/core/skills/builtin/netmiko-ssh-automation/SKILL.md +173 -0
  255. package/dist/core/skills/builtin/network-bgp-diagnostics/SKILL.md +167 -0
  256. package/dist/core/skills/builtin/network-config-validation/SKILL.md +210 -0
  257. package/dist/core/skills/builtin/network-interface-health/SKILL.md +152 -0
  258. package/dist/core/skills/builtin/nextjs-turbopack/SKILL.md +57 -0
  259. package/dist/core/skills/builtin/nodejs-keccak256/SKILL.md +102 -0
  260. package/dist/core/skills/builtin/nutrient-document-processing/SKILL.md +167 -0
  261. package/dist/core/skills/builtin/nuxt4-patterns/SKILL.md +100 -0
  262. package/dist/core/skills/builtin/openclaw-persona-forge/SKILL.md +288 -0
  263. package/dist/core/skills/builtin/openclaw-persona-forge/gacha.py +224 -0
  264. package/dist/core/skills/builtin/openclaw-persona-forge/gacha.sh +5 -0
  265. package/dist/core/skills/builtin/openclaw-persona-forge/references/avatar-style.md +124 -0
  266. package/dist/core/skills/builtin/openclaw-persona-forge/references/boundary-rules.md +53 -0
  267. package/dist/core/skills/builtin/openclaw-persona-forge/references/error-handling.md +53 -0
  268. package/dist/core/skills/builtin/openclaw-persona-forge/references/identity-tension.md +48 -0
  269. package/dist/core/skills/builtin/openclaw-persona-forge/references/naming-system.md +39 -0
  270. package/dist/core/skills/builtin/openclaw-persona-forge/references/output-template.md +166 -0
  271. package/dist/core/skills/builtin/opensource-pipeline/SKILL.md +255 -0
  272. package/dist/core/skills/builtin/orch-add-feature/SKILL.md +44 -0
  273. package/dist/core/skills/builtin/orch-build-mvp/SKILL.md +48 -0
  274. package/dist/core/skills/builtin/orch-change-feature/SKILL.md +42 -0
  275. package/dist/core/skills/builtin/orch-fix-defect/SKILL.md +42 -0
  276. package/dist/core/skills/builtin/orch-pipeline/SKILL.md +120 -0
  277. package/dist/core/skills/builtin/orch-refine-code/SKILL.md +43 -0
  278. package/dist/core/skills/builtin/parallel-execution-optimizer/SKILL.md +72 -0
  279. package/dist/core/skills/builtin/perl-patterns/SKILL.md +504 -0
  280. package/dist/core/skills/builtin/perl-security/SKILL.md +503 -0
  281. package/dist/core/skills/builtin/perl-testing/SKILL.md +475 -0
  282. package/dist/core/skills/builtin/plan-orchestrate/SKILL.md +262 -0
  283. package/dist/core/skills/builtin/plankton-code-quality/SKILL.md +236 -0
  284. package/dist/core/skills/builtin/postgres-patterns/SKILL.md +147 -0
  285. package/dist/core/skills/builtin/prediction-market-oracle-research/SKILL.md +63 -0
  286. package/dist/core/skills/builtin/prediction-market-risk-review/SKILL.md +60 -0
  287. package/dist/core/skills/builtin/prisma-patterns/SKILL.md +371 -0
  288. package/dist/core/skills/builtin/product-capability/SKILL.md +141 -0
  289. package/dist/core/skills/builtin/product-lens/SKILL.md +92 -0
  290. package/dist/core/skills/builtin/production-audit/SKILL.md +206 -0
  291. package/dist/core/skills/builtin/production-scheduling/SKILL.md +238 -0
  292. package/dist/core/skills/builtin/prompt-optimizer/SKILL.md +398 -0
  293. package/dist/core/skills/builtin/python-patterns/SKILL.md +750 -0
  294. package/dist/core/skills/builtin/python-testing/SKILL.md +816 -0
  295. package/dist/core/skills/builtin/pytorch-patterns/SKILL.md +396 -0
  296. package/dist/core/skills/builtin/quality-nonconformance/SKILL.md +260 -0
  297. package/dist/core/skills/builtin/quarkus-patterns/SKILL.md +722 -0
  298. package/dist/core/skills/builtin/quarkus-security/SKILL.md +467 -0
  299. package/dist/core/skills/builtin/quarkus-tdd/SKILL.md +811 -0
  300. package/dist/core/skills/builtin/quarkus-verification/SKILL.md +479 -0
  301. package/dist/core/skills/builtin/ralphinho-rfc-pipeline/SKILL.md +67 -0
  302. package/dist/core/skills/builtin/react-patterns/SKILL.md +341 -0
  303. package/dist/core/skills/builtin/react-performance/SKILL.md +574 -0
  304. package/dist/core/skills/builtin/react-testing/SKILL.md +423 -0
  305. package/dist/core/skills/builtin/recsys-pipeline-architect/SKILL.md +114 -0
  306. package/dist/core/skills/builtin/recursive-decision-ledger/SKILL.md +79 -0
  307. package/dist/core/skills/builtin/redis-patterns/SKILL.md +403 -0
  308. package/dist/core/skills/builtin/regex-vs-llm-structured-text/SKILL.md +220 -0
  309. package/dist/core/skills/builtin/repo-scan/SKILL.md +78 -0
  310. package/dist/core/skills/builtin/research-ops/SKILL.md +112 -0
  311. package/dist/core/skills/builtin/returns-reverse-logistics/SKILL.md +240 -0
  312. package/dist/core/skills/builtin/rules-distill/SKILL.md +264 -0
  313. package/dist/core/skills/builtin/rules-distill/scripts/scan-rules.sh +58 -0
  314. package/dist/core/skills/builtin/rules-distill/scripts/scan-skills.sh +129 -0
  315. package/dist/core/skills/builtin/rust-patterns/SKILL.md +499 -0
  316. package/dist/core/skills/builtin/rust-testing/SKILL.md +500 -0
  317. package/dist/core/skills/builtin/safety-guard/SKILL.md +75 -0
  318. package/dist/core/skills/builtin/santa-method/SKILL.md +306 -0
  319. package/dist/core/skills/builtin/scientific-db-pubmed-database/SKILL.md +175 -0
  320. package/dist/core/skills/builtin/scientific-db-uspto-database/SKILL.md +177 -0
  321. package/dist/core/skills/builtin/scientific-pkg-gget/SKILL.md +166 -0
  322. package/dist/core/skills/builtin/scientific-thinking-literature-review/SKILL.md +192 -0
  323. package/dist/core/skills/builtin/scientific-thinking-scholar-evaluation/SKILL.md +160 -0
  324. package/dist/core/skills/builtin/search-first/SKILL.md +182 -0
  325. package/dist/core/skills/builtin/security-bounty-hunter/SKILL.md +99 -0
  326. package/dist/core/skills/builtin/security-review/SKILL.md +503 -0
  327. package/dist/core/skills/builtin/security-review/cloud-infrastructure-security.md +361 -0
  328. package/dist/core/skills/builtin/security-scan/SKILL.md +165 -0
  329. package/dist/core/skills/builtin/seo/SKILL.md +154 -0
  330. package/dist/core/skills/builtin/skill-comply/SKILL.md +58 -0
  331. package/dist/core/skills/builtin/skill-comply/fixtures/compliant_trace.jsonl +5 -0
  332. package/dist/core/skills/builtin/skill-comply/fixtures/noncompliant_trace.jsonl +3 -0
  333. package/dist/core/skills/builtin/skill-comply/fixtures/tdd_spec.yaml +44 -0
  334. package/dist/core/skills/builtin/skill-comply/prompts/classifier.md +24 -0
  335. package/dist/core/skills/builtin/skill-comply/prompts/scenario_generator.md +62 -0
  336. package/dist/core/skills/builtin/skill-comply/prompts/spec_generator.md +42 -0
  337. package/dist/core/skills/builtin/skill-comply/pyproject.toml +15 -0
  338. package/dist/core/skills/builtin/skill-comply/scripts/__init__.py +0 -0
  339. package/dist/core/skills/builtin/skill-comply/scripts/classifier.py +85 -0
  340. package/dist/core/skills/builtin/skill-comply/scripts/grader.py +124 -0
  341. package/dist/core/skills/builtin/skill-comply/scripts/parser.py +107 -0
  342. package/dist/core/skills/builtin/skill-comply/scripts/report.py +170 -0
  343. package/dist/core/skills/builtin/skill-comply/scripts/run.py +127 -0
  344. package/dist/core/skills/builtin/skill-comply/scripts/runner.py +186 -0
  345. package/dist/core/skills/builtin/skill-comply/scripts/scenario_generator.py +70 -0
  346. package/dist/core/skills/builtin/skill-comply/scripts/spec_generator.py +72 -0
  347. package/dist/core/skills/builtin/skill-comply/scripts/utils.py +13 -0
  348. package/dist/core/skills/builtin/skill-comply/tests/test_grader.py +197 -0
  349. package/dist/core/skills/builtin/skill-comply/tests/test_parser.py +90 -0
  350. package/dist/core/skills/builtin/skill-comply/tests/test_runner.py +172 -0
  351. package/dist/core/skills/builtin/skill-scout/SKILL.md +140 -0
  352. package/dist/core/skills/builtin/skill-stocktake/SKILL.md +194 -0
  353. package/dist/core/skills/builtin/skill-stocktake/scripts/quick-diff.sh +87 -0
  354. package/dist/core/skills/builtin/skill-stocktake/scripts/save-results.sh +56 -0
  355. package/dist/core/skills/builtin/skill-stocktake/scripts/scan.sh +170 -0
  356. package/dist/core/skills/builtin/springboot-patterns/SKILL.md +314 -0
  357. package/dist/core/skills/builtin/springboot-security/SKILL.md +272 -0
  358. package/dist/core/skills/builtin/springboot-tdd/SKILL.md +158 -0
  359. package/dist/core/skills/builtin/springboot-verification/SKILL.md +231 -0
  360. package/dist/core/skills/builtin/strategic-compact/SKILL.md +135 -0
  361. package/dist/core/skills/builtin/swift-actor-persistence/SKILL.md +143 -0
  362. package/dist/core/skills/builtin/swift-concurrency-6-2/SKILL.md +216 -0
  363. package/dist/core/skills/builtin/swift-protocol-di-testing/SKILL.md +190 -0
  364. package/dist/core/skills/builtin/swiftui-patterns/SKILL.md +259 -0
  365. package/dist/core/skills/builtin/tdd-workflow/SKILL.md +463 -0
  366. package/dist/core/skills/builtin/team-agent-orchestration/SKILL.md +110 -0
  367. package/dist/core/skills/builtin/team-builder/SKILL.md +168 -0
  368. package/dist/core/skills/builtin/terminal-ops/SKILL.md +109 -0
  369. package/dist/core/skills/builtin/tinystruct-patterns/SKILL.md +203 -0
  370. package/dist/core/skills/builtin/tinystruct-patterns/references/architecture.md +90 -0
  371. package/dist/core/skills/builtin/tinystruct-patterns/references/data-handling.md +60 -0
  372. package/dist/core/skills/builtin/tinystruct-patterns/references/database.md +99 -0
  373. package/dist/core/skills/builtin/tinystruct-patterns/references/routing.md +64 -0
  374. package/dist/core/skills/builtin/tinystruct-patterns/references/system-usage.md +97 -0
  375. package/dist/core/skills/builtin/tinystruct-patterns/references/testing.md +72 -0
  376. package/dist/core/skills/builtin/token-budget-advisor/SKILL.md +133 -0
  377. package/dist/core/skills/builtin/ui-demo/SKILL.md +465 -0
  378. package/dist/core/skills/builtin/ui-to-vue/SKILL.md +134 -0
  379. package/dist/core/skills/builtin/uncloud/SKILL.md +343 -0
  380. package/dist/core/skills/builtin/unified-notifications-ops/SKILL.md +187 -0
  381. package/dist/core/skills/builtin/verification-loop/SKILL.md +126 -0
  382. package/dist/core/skills/builtin/video-editing/SKILL.md +310 -0
  383. package/dist/core/skills/builtin/videodb/SKILL.md +374 -0
  384. package/dist/core/skills/builtin/videodb/reference/api-reference.md +550 -0
  385. package/dist/core/skills/builtin/videodb/reference/capture-reference.md +407 -0
  386. package/dist/core/skills/builtin/videodb/reference/capture.md +101 -0
  387. package/dist/core/skills/builtin/videodb/reference/editor.md +443 -0
  388. package/dist/core/skills/builtin/videodb/reference/generative.md +331 -0
  389. package/dist/core/skills/builtin/videodb/reference/rtstream-reference.md +564 -0
  390. package/dist/core/skills/builtin/videodb/reference/rtstream.md +65 -0
  391. package/dist/core/skills/builtin/videodb/reference/search.md +230 -0
  392. package/dist/core/skills/builtin/videodb/reference/streaming.md +406 -0
  393. package/dist/core/skills/builtin/videodb/reference/use-cases.md +118 -0
  394. package/dist/core/skills/builtin/videodb/scripts/ws_listener.py +282 -0
  395. package/dist/core/skills/builtin/visa-doc-translate/README.md +86 -0
  396. package/dist/core/skills/builtin/visa-doc-translate/SKILL.md +117 -0
  397. package/dist/core/skills/builtin/vite-patterns/SKILL.md +449 -0
  398. package/dist/core/skills/builtin/windows-desktop-e2e/SKILL.md +887 -0
  399. package/dist/core/skills/builtin/x-api/SKILL.md +234 -0
  400. package/dist/core/skills/loader.js +11 -0
  401. package/dist/core/skills/loader.js.map +1 -1
  402. package/dist/core/types.d.ts +6 -0
  403. package/dist/core/types.d.ts.map +1 -1
  404. package/dist/plugins/agents/native/index.d.ts +47 -8
  405. package/dist/plugins/agents/native/index.d.ts.map +1 -1
  406. package/dist/plugins/agents/native/index.js +253 -102
  407. package/dist/plugins/agents/native/index.js.map +1 -1
  408. package/dist/plugins/agents/native/tool-registry.d.ts +33 -0
  409. package/dist/plugins/agents/native/tool-registry.d.ts.map +1 -0
  410. package/dist/plugins/agents/native/tool-registry.js +82 -0
  411. package/dist/plugins/agents/native/tool-registry.js.map +1 -0
  412. package/dist/plugins/messengers/dingtalk/dingtalk-client.d.ts.map +1 -1
  413. package/dist/plugins/messengers/dingtalk/dingtalk-client.js +11 -11
  414. package/dist/plugins/messengers/dingtalk/dingtalk-client.js.map +1 -1
  415. package/dist/plugins/messengers/feishu/feishu-adapter.d.ts.map +1 -1
  416. package/dist/plugins/messengers/feishu/feishu-adapter.js +9 -5
  417. package/dist/plugins/messengers/feishu/feishu-adapter.js.map +1 -1
  418. package/dist/plugins/messengers/wechat/ilink-adapter.d.ts.map +1 -1
  419. package/dist/plugins/messengers/wechat/ilink-adapter.js +11 -1
  420. package/dist/plugins/messengers/wechat/ilink-adapter.js.map +1 -1
  421. package/dist/web/public/assets/{a2a-DczMMkbl.js → a2a-Cll3P4QN.js} +2 -2
  422. package/dist/web/public/assets/{a2a-DczMMkbl.js.map → a2a-Cll3P4QN.js.map} +1 -1
  423. package/dist/web/public/assets/{activity-cbLHkzca.js → activity-B7T7YFlD.js} +2 -2
  424. package/dist/web/public/assets/{activity-cbLHkzca.js.map → activity-B7T7YFlD.js.map} +1 -1
  425. package/dist/web/public/assets/{admins-C-YsGMj7.js → admins-CN7P018S.js} +2 -2
  426. package/dist/web/public/assets/{admins-C-YsGMj7.js.map → admins-CN7P018S.js.map} +1 -1
  427. package/dist/web/public/assets/{agents-BWfov_1-.js → agents-Bqgq7GBF.js} +2 -2
  428. package/dist/web/public/assets/{agents-BWfov_1-.js.map → agents-Bqgq7GBF.js.map} +1 -1
  429. package/dist/web/public/assets/{approvals-HSssmXKS.js → approvals-C8IUJQ_A.js} +2 -2
  430. package/dist/web/public/assets/{approvals-HSssmXKS.js.map → approvals-C8IUJQ_A.js.map} +1 -1
  431. package/dist/web/public/assets/{arrow-down-BXvC8Al2.js → arrow-down-SLWKqtDc.js} +2 -2
  432. package/dist/web/public/assets/{arrow-down-BXvC8Al2.js.map → arrow-down-SLWKqtDc.js.map} +1 -1
  433. package/dist/web/public/assets/{arrow-up-63xELY5Q.js → arrow-up-BOADc9ce.js} +2 -2
  434. package/dist/web/public/assets/{arrow-up-63xELY5Q.js.map → arrow-up-BOADc9ce.js.map} +1 -1
  435. package/dist/web/public/assets/{asks-COLEFOvK.js → asks-C-j-DypC.js} +2 -2
  436. package/dist/web/public/assets/{asks-COLEFOvK.js.map → asks-C-j-DypC.js.map} +1 -1
  437. package/dist/web/public/assets/{audit-D4ZEiZub.js → audit-DQb-RuXh.js} +2 -2
  438. package/dist/web/public/assets/{audit-D4ZEiZub.js.map → audit-DQb-RuXh.js.map} +1 -1
  439. package/dist/web/public/assets/{bell-Cg2Bvv06.js → bell-CV88-ul6.js} +2 -2
  440. package/dist/web/public/assets/{bell-Cg2Bvv06.js.map → bell-CV88-ul6.js.map} +1 -1
  441. package/dist/web/public/assets/{bgjobs-CEjCzwtd.js → bgjobs-CDrK0d-W.js} +2 -2
  442. package/dist/web/public/assets/{bgjobs-CEjCzwtd.js.map → bgjobs-CDrK0d-W.js.map} +1 -1
  443. package/dist/web/public/assets/{brain-euvl6F6C.js → brain-B7HtSOQU.js} +2 -2
  444. package/dist/web/public/assets/{brain-euvl6F6C.js.map → brain-B7HtSOQU.js.map} +1 -1
  445. package/dist/web/public/assets/{briefcase-DPWLbCnA.js → briefcase-mdzuIa__.js} +2 -2
  446. package/dist/web/public/assets/{briefcase-DPWLbCnA.js.map → briefcase-mdzuIa__.js.map} +1 -1
  447. package/dist/web/public/assets/{browser-ponyfill-BUutOaRz.js → browser-ponyfill-DBWdeCTC.js} +2 -2
  448. package/dist/web/public/assets/{browser-ponyfill-BUutOaRz.js.map → browser-ponyfill-DBWdeCTC.js.map} +1 -1
  449. package/dist/web/public/assets/{chat-Dz9kfaxH.js → chat-CSjtY2rN.js} +3 -3
  450. package/dist/web/public/assets/{chat-Dz9kfaxH.js.map → chat-CSjtY2rN.js.map} +1 -1
  451. package/dist/web/public/assets/{chevron-left-BeIh5thq.js → chevron-left-uSfPn636.js} +2 -2
  452. package/dist/web/public/assets/{chevron-left-BeIh5thq.js.map → chevron-left-uSfPn636.js.map} +1 -1
  453. package/dist/web/public/assets/{chevron-right-uP_l9MMb.js → chevron-right-CtelqacW.js} +2 -2
  454. package/dist/web/public/assets/{chevron-right-uP_l9MMb.js.map → chevron-right-CtelqacW.js.map} +1 -1
  455. package/dist/web/public/assets/{circle-check-CewnjFgv.js → circle-check-8dbL-u7O.js} +2 -2
  456. package/dist/web/public/assets/{circle-check-CewnjFgv.js.map → circle-check-8dbL-u7O.js.map} +1 -1
  457. package/dist/web/public/assets/{circle-check-big-C2RTc48c.js → circle-check-big-D8-svk9a.js} +2 -2
  458. package/dist/web/public/assets/{circle-check-big-C2RTc48c.js.map → circle-check-big-D8-svk9a.js.map} +1 -1
  459. package/dist/web/public/assets/{circle-x-Ccg1HyV-.js → circle-x-rUxzIz5P.js} +2 -2
  460. package/dist/web/public/assets/{circle-x-Ccg1HyV-.js.map → circle-x-rUxzIz5P.js.map} +1 -1
  461. package/dist/web/public/assets/{clock-qxbYSynv.js → clock-CG5dlBGB.js} +2 -2
  462. package/dist/web/public/assets/{clock-qxbYSynv.js.map → clock-CG5dlBGB.js.map} +1 -1
  463. package/dist/web/public/assets/{confirm-dialog-DmJq4Td9.js → confirm-dialog-DlUsSur3.js} +2 -2
  464. package/dist/web/public/assets/{confirm-dialog-DmJq4Td9.js.map → confirm-dialog-DlUsSur3.js.map} +1 -1
  465. package/dist/web/public/assets/{copy-DxSHRdbc.js → copy-DnC76wFT.js} +2 -2
  466. package/dist/web/public/assets/{copy-DxSHRdbc.js.map → copy-DnC76wFT.js.map} +1 -1
  467. package/dist/web/public/assets/{data-table-S7rIjwdO.js → data-table-DswkWUfG.js} +2 -2
  468. package/dist/web/public/assets/{data-table-S7rIjwdO.js.map → data-table-DswkWUfG.js.map} +1 -1
  469. package/dist/web/public/assets/dialog-Ceo4YuXy.js +6 -0
  470. package/dist/web/public/assets/dialog-Ceo4YuXy.js.map +1 -0
  471. package/dist/web/public/assets/{download-OhsGtnO-.js → download-DF-46tS4.js} +2 -2
  472. package/dist/web/public/assets/{download-OhsGtnO-.js.map → download-DF-46tS4.js.map} +1 -1
  473. package/dist/web/public/assets/{email-C1-HxWLF.js → email-CZee26-_.js} +3 -3
  474. package/dist/web/public/assets/{email-C1-HxWLF.js.map → email-CZee26-_.js.map} +1 -1
  475. package/dist/web/public/assets/{empty-state-C-qjOHyu.js → empty-state-D9Hi0Atm.js} +2 -2
  476. package/dist/web/public/assets/{empty-state-C-qjOHyu.js.map → empty-state-D9Hi0Atm.js.map} +1 -1
  477. package/dist/web/public/assets/{external-link-DRVp9-lb.js → external-link-D64iZa9P.js} +2 -2
  478. package/dist/web/public/assets/{external-link-DRVp9-lb.js.map → external-link-D64iZa9P.js.map} +1 -1
  479. package/dist/web/public/assets/{eye-CFhg5BTa.js → eye-sY6WZb7D.js} +2 -2
  480. package/dist/web/public/assets/{eye-CFhg5BTa.js.map → eye-sY6WZb7D.js.map} +1 -1
  481. package/dist/web/public/assets/{facts-CGaLWhzi.js → facts-B7bGGwvi.js} +2 -2
  482. package/dist/web/public/assets/{facts-CGaLWhzi.js.map → facts-B7bGGwvi.js.map} +1 -1
  483. package/dist/web/public/assets/{goals-C-dJANmn.js → goals-BfQbsvZv.js} +2 -2
  484. package/dist/web/public/assets/{goals-C-dJANmn.js.map → goals-BfQbsvZv.js.map} +1 -1
  485. package/dist/web/public/assets/{health-CWcti5h3.js → health-Ba_mY0Ts.js} +2 -2
  486. package/dist/web/public/assets/{health-CWcti5h3.js.map → health-Ba_mY0Ts.js.map} +1 -1
  487. package/dist/web/public/assets/{heart-pulse-DmGhKR2W.js → heart-pulse-BjikOVwU.js} +2 -2
  488. package/dist/web/public/assets/{heart-pulse-DmGhKR2W.js.map → heart-pulse-BjikOVwU.js.map} +1 -1
  489. package/dist/web/public/assets/{heartbeat-kLoGBNCo.js → heartbeat-BM8LlPes.js} +2 -2
  490. package/dist/web/public/assets/{heartbeat-kLoGBNCo.js.map → heartbeat-BM8LlPes.js.map} +1 -1
  491. package/dist/web/public/assets/{hot-BITDoax1.js → hot-BtuLL6n8.js} +2 -2
  492. package/dist/web/public/assets/{hot-BITDoax1.js.map → hot-BtuLL6n8.js.map} +1 -1
  493. package/dist/web/public/assets/index-DEWFfW_Z.js +199 -0
  494. package/dist/web/public/assets/index-DEWFfW_Z.js.map +1 -0
  495. package/dist/web/public/assets/{installed-Co9WrtQ7.js → installed-Xr8p31ij.js} +2 -2
  496. package/dist/web/public/assets/{installed-Co9WrtQ7.js.map → installed-Xr8p31ij.js.map} +1 -1
  497. package/dist/web/public/assets/{jobs-hdHhBEvi.js → jobs-Ddy81Udm.js} +2 -2
  498. package/dist/web/public/assets/{jobs-hdHhBEvi.js.map → jobs-Ddy81Udm.js.map} +1 -1
  499. package/dist/web/public/assets/{layout-CQtbOBag.js → layout-BL74fT-L.js} +2 -2
  500. package/dist/web/public/assets/{layout-CQtbOBag.js.map → layout-BL74fT-L.js.map} +1 -1
  501. package/dist/web/public/assets/{layout-bDMXIKIR.js → layout-Bn2qUxcK.js} +2 -2
  502. package/dist/web/public/assets/{layout-bDMXIKIR.js.map → layout-Bn2qUxcK.js.map} +1 -1
  503. package/dist/web/public/assets/{layout-BMXC1Uh1.js → layout-Bp4SAA8_.js} +2 -2
  504. package/dist/web/public/assets/{layout-BMXC1Uh1.js.map → layout-Bp4SAA8_.js.map} +1 -1
  505. package/dist/web/public/assets/{layout-CysVsySh.js → layout-CZ9pGnW8.js} +2 -2
  506. package/dist/web/public/assets/{layout-CysVsySh.js.map → layout-CZ9pGnW8.js.map} +1 -1
  507. package/dist/web/public/assets/{layout-CyBGneZ9.js → layout-pasFRkKV.js} +2 -2
  508. package/dist/web/public/assets/{layout-CyBGneZ9.js.map → layout-pasFRkKV.js.map} +1 -1
  509. package/dist/web/public/assets/llm-yp7b5xxL.js +7 -0
  510. package/dist/web/public/assets/llm-yp7b5xxL.js.map +1 -0
  511. package/dist/web/public/assets/{loader-circle-9VUMGitw.js → loader-circle-Bbw4pEyE.js} +2 -2
  512. package/dist/web/public/assets/{loader-circle-9VUMGitw.js.map → loader-circle-Bbw4pEyE.js.map} +1 -1
  513. package/dist/web/public/assets/{map-pin-BXYvvHry.js → map-pin-DIXHUQgM.js} +2 -2
  514. package/dist/web/public/assets/{map-pin-BXYvvHry.js.map → map-pin-DIXHUQgM.js.map} +1 -1
  515. package/dist/web/public/assets/{mcp-BgLdlwSn.js → mcp-DyaljIM_.js} +2 -2
  516. package/dist/web/public/assets/{mcp-BgLdlwSn.js.map → mcp-DyaljIM_.js.map} +1 -1
  517. package/dist/web/public/assets/memos-Dkoc157i.js +12 -0
  518. package/dist/web/public/assets/memos-Dkoc157i.js.map +1 -0
  519. package/dist/web/public/assets/{messengers-7Phqea62.js → messengers-CcyGDeUI.js} +2 -2
  520. package/dist/web/public/assets/{messengers-7Phqea62.js.map → messengers-CcyGDeUI.js.map} +1 -1
  521. package/dist/web/public/assets/{mobile-CV5b6D2W.js → mobile-DqzIv4Xb.js} +2 -2
  522. package/dist/web/public/assets/{mobile-CV5b6D2W.js.map → mobile-DqzIv4Xb.js.map} +1 -1
  523. package/dist/web/public/assets/{native-agent-QvIa6LjE.js → native-agent-BQ7WaRGK.js} +2 -2
  524. package/dist/web/public/assets/{native-agent-QvIa6LjE.js.map → native-agent-BQ7WaRGK.js.map} +1 -1
  525. package/dist/web/public/assets/{network-BXhEjGhE.js → network-B_yUFAqC.js} +2 -2
  526. package/dist/web/public/assets/{network-BXhEjGhE.js.map → network-B_yUFAqC.js.map} +1 -1
  527. package/dist/web/public/assets/{outbox-DHQL7TQb.js → outbox-l8aVOZqO.js} +2 -2
  528. package/dist/web/public/assets/{outbox-DHQL7TQb.js.map → outbox-l8aVOZqO.js.map} +1 -1
  529. package/dist/web/public/assets/{pagination-VKuPb1Ot.js → pagination-BAKRGKa9.js} +2 -2
  530. package/dist/web/public/assets/{pagination-VKuPb1Ot.js.map → pagination-BAKRGKa9.js.map} +1 -1
  531. package/dist/web/public/assets/{persona-CWug2GLR.js → persona-D3VL9Rg1.js} +2 -2
  532. package/dist/web/public/assets/{persona-CWug2GLR.js.map → persona-D3VL9Rg1.js.map} +1 -1
  533. package/dist/web/public/assets/{plans-CZoEs5SY.js → plans-BBB5e9my.js} +2 -2
  534. package/dist/web/public/assets/{plans-CZoEs5SY.js.map → plans-BBB5e9my.js.map} +1 -1
  535. package/dist/web/public/assets/{play-CfSn5Vdl.js → play-7-Wd369f.js} +2 -2
  536. package/dist/web/public/assets/{play-CfSn5Vdl.js.map → play-7-Wd369f.js.map} +1 -1
  537. package/dist/web/public/assets/{plus-Z8l4CiqJ.js → plus-B0sfZy-j.js} +2 -2
  538. package/dist/web/public/assets/{plus-Z8l4CiqJ.js.map → plus-B0sfZy-j.js.map} +1 -1
  539. package/dist/web/public/assets/{policy-CutDSEPW.js → policy-BM1WRXH0.js} +2 -2
  540. package/dist/web/public/assets/{policy-CutDSEPW.js.map → policy-BM1WRXH0.js.map} +1 -1
  541. package/dist/web/public/assets/{qr-code-DgU5aiM6.js → qr-code-DcKs5fi3.js} +2 -2
  542. package/dist/web/public/assets/{qr-code-DgU5aiM6.js.map → qr-code-DcKs5fi3.js.map} +1 -1
  543. package/dist/web/public/assets/{react-Cb2sDjhD.js → react-DlP5eolq.js} +2 -2
  544. package/dist/web/public/assets/{react-Cb2sDjhD.js.map → react-DlP5eolq.js.map} +1 -1
  545. package/dist/web/public/assets/{refresh-ccw-D2CWiyU_.js → refresh-ccw-uNKeBeRl.js} +2 -2
  546. package/dist/web/public/assets/{refresh-ccw-D2CWiyU_.js.map → refresh-ccw-uNKeBeRl.js.map} +1 -1
  547. package/dist/web/public/assets/{reminders-Cb6Izedg.js → reminders-DHM8K0_O.js} +2 -2
  548. package/dist/web/public/assets/{reminders-Cb6Izedg.js.map → reminders-DHM8K0_O.js.map} +1 -1
  549. package/dist/web/public/assets/{save-DB0BDYTs.js → save-qwJa5_SA.js} +2 -2
  550. package/dist/web/public/assets/{save-DB0BDYTs.js.map → save-qwJa5_SA.js.map} +1 -1
  551. package/dist/web/public/assets/{schedules-8mSjE14D.js → schedules-Bcd0wbT4.js} +2 -2
  552. package/dist/web/public/assets/{schedules-8mSjE14D.js.map → schedules-Bcd0wbT4.js.map} +1 -1
  553. package/dist/web/public/assets/{search-Con69NhG.js → search-BUlzNWrj.js} +2 -2
  554. package/dist/web/public/assets/{search-Con69NhG.js.map → search-BUlzNWrj.js.map} +1 -1
  555. package/dist/web/public/assets/{search-B4fHilZ0.js → search-i1tP2maJ.js} +2 -2
  556. package/dist/web/public/assets/{search-B4fHilZ0.js.map → search-i1tP2maJ.js.map} +1 -1
  557. package/dist/web/public/assets/{security-BTe3zUg8.js → security-DgJyTT4g.js} +2 -2
  558. package/dist/web/public/assets/{security-BTe3zUg8.js.map → security-DgJyTT4g.js.map} +1 -1
  559. package/dist/web/public/assets/{service-C7SqcwfL.js → service-A0Hzear0.js} +2 -2
  560. package/dist/web/public/assets/{service-C7SqcwfL.js.map → service-A0Hzear0.js.map} +1 -1
  561. package/dist/web/public/assets/{shield-alert-CKFVsGgI.js → shield-alert-DrnN6fz_.js} +2 -2
  562. package/dist/web/public/assets/{shield-alert-CKFVsGgI.js.map → shield-alert-DrnN6fz_.js.map} +1 -1
  563. package/dist/web/public/assets/{status-badge-BSkpyN4D.js → status-badge-Ryzf96Pl.js} +2 -2
  564. package/dist/web/public/assets/{status-badge-BSkpyN4D.js.map → status-badge-Ryzf96Pl.js.map} +1 -1
  565. package/dist/web/public/assets/{subtasks-Bel-I1Sk.js → subtasks-Bzh3o3EF.js} +2 -2
  566. package/dist/web/public/assets/{subtasks-Bel-I1Sk.js.map → subtasks-Bzh3o3EF.js.map} +1 -1
  567. package/dist/web/public/assets/{table-CPn1MRcy.js → table-BbAOSyc8.js} +2 -2
  568. package/dist/web/public/assets/{table-CPn1MRcy.js.map → table-BbAOSyc8.js.map} +1 -1
  569. package/dist/web/public/assets/{topn-Ba3RjcK1.js → topn-DkhYw-Gp.js} +2 -2
  570. package/dist/web/public/assets/{topn-Ba3RjcK1.js.map → topn-DkhYw-Gp.js.map} +1 -1
  571. package/dist/web/public/assets/{trash-2-Dfov8aHD.js → trash-2-CA0cLpnU.js} +2 -2
  572. package/dist/web/public/assets/{trash-2-Dfov8aHD.js.map → trash-2-CA0cLpnU.js.map} +1 -1
  573. package/dist/web/public/assets/{use-background-tasks-BQrEeUwY.js → use-background-tasks-B64YjlA8.js} +2 -2
  574. package/dist/web/public/assets/{use-background-tasks-BQrEeUwY.js.map → use-background-tasks-B64YjlA8.js.map} +1 -1
  575. package/dist/web/public/assets/{use-event-stream-DgGpGKop.js → use-event-stream-I1lMFEfh.js} +2 -2
  576. package/dist/web/public/assets/{use-event-stream-DgGpGKop.js.map → use-event-stream-I1lMFEfh.js.map} +1 -1
  577. package/dist/web/public/assets/{use-llm-admin-DYekqogG.js → use-llm-admin-DY2axI4D.js} +2 -2
  578. package/dist/web/public/assets/{use-llm-admin-DYekqogG.js.map → use-llm-admin-DY2axI4D.js.map} +1 -1
  579. package/dist/web/public/assets/{use-memory-DbJ4pP2Z.js → use-memory-BYEjVWbU.js} +2 -2
  580. package/dist/web/public/assets/{use-memory-DbJ4pP2Z.js.map → use-memory-BYEjVWbU.js.map} +1 -1
  581. package/dist/web/public/assets/{use-observability-C2M6WZ9W.js → use-observability-Coj02yDo.js} +2 -2
  582. package/dist/web/public/assets/{use-observability-C2M6WZ9W.js.map → use-observability-Coj02yDo.js.map} +1 -1
  583. package/dist/web/public/assets/{use-settings-DMdaoWsB.js → use-settings-i1MhlkyC.js} +2 -2
  584. package/dist/web/public/assets/{use-settings-DMdaoWsB.js.map → use-settings-i1MhlkyC.js.map} +1 -1
  585. package/dist/web/public/assets/{use-workspace-BHG7h3jQ.js → use-workspace-DgEM35PY.js} +2 -2
  586. package/dist/web/public/assets/{use-workspace-BHG7h3jQ.js.map → use-workspace-DgEM35PY.js.map} +1 -1
  587. package/dist/web/public/assets/{useQuery-PdiC7-sY.js → useQuery-CY2iazjN.js} +2 -2
  588. package/dist/web/public/assets/{useQuery-PdiC7-sY.js.map → useQuery-CY2iazjN.js.map} +1 -1
  589. package/dist/web/public/assets/{vector-DnZM3OXU.js → vector-Ic76u2hY.js} +2 -2
  590. package/dist/web/public/assets/{vector-DnZM3OXU.js.map → vector-Ic76u2hY.js.map} +1 -1
  591. package/dist/web/public/assets/{viewer-Dz6k0YKp.js → viewer-BXbUN1Rl.js} +2 -2
  592. package/dist/web/public/assets/{viewer-Dz6k0YKp.js.map → viewer-BXbUN1Rl.js.map} +1 -1
  593. package/dist/web/public/assets/{workspace-BnXrWS3j.js → workspace-CUg0JPn6.js} +3 -3
  594. package/dist/web/public/assets/{workspace-BnXrWS3j.js.map → workspace-CUg0JPn6.js.map} +1 -1
  595. package/dist/web/public/assets/{workspaces-CSS_UBEi.js → workspaces-C-wb5FQj.js} +2 -2
  596. package/dist/web/public/assets/{workspaces-CSS_UBEi.js.map → workspaces-C-wb5FQj.js.map} +1 -1
  597. package/dist/web/public/assets/{x-DG-JKVw_.js → x-D1iSuoqg.js} +2 -2
  598. package/dist/web/public/assets/{x-DG-JKVw_.js.map → x-D1iSuoqg.js.map} +1 -1
  599. package/dist/web/public/index.html +2 -2
  600. package/dist/web/server.d.ts.map +1 -1
  601. package/dist/web/server.js +6 -0
  602. package/dist/web/server.js.map +1 -1
  603. package/package.json +1 -1
  604. package/dist/web/public/assets/dialog-bAIDaO-6.js +0 -6
  605. package/dist/web/public/assets/dialog-bAIDaO-6.js.map +0 -1
  606. package/dist/web/public/assets/index-O0BQoyzo.js +0 -199
  607. package/dist/web/public/assets/index-O0BQoyzo.js.map +0 -1
  608. package/dist/web/public/assets/llm-CPIRNQU2.js +0 -7
  609. package/dist/web/public/assets/llm-CPIRNQU2.js.map +0 -1
  610. package/dist/web/public/assets/memos-CfneX9DH.js +0 -12
  611. package/dist/web/public/assets/memos-CfneX9DH.js.map +0 -1
@@ -0,0 +1,513 @@
1
+ ---
2
+ name: fastapi-patterns
3
+ description: [ECC] FastAPI best practices covering project structure, Pydantic v2 schemas, dependency injection, async handlers, authentication, authorization, transactional service layers, and testing with httpx and pytest.
4
+ origin: ECC
5
+ ---
6
+
7
+ # FastAPI Patterns
8
+
9
+ Modern, production-grade FastAPI development: project layout, Pydantic v2 schemas, dependency injection, async patterns, auth, transactional service methods, and testing.
10
+
11
+ ## Project Structure
12
+
13
+ ```text
14
+ my_app/
15
+ |-- app/
16
+ | |-- main.py # App factory, lifespan, middleware
17
+ | |-- config.py # Settings via pydantic-settings
18
+ | |-- dependencies.py # Shared FastAPI dependencies
19
+ | |-- database.py # SQLAlchemy engine + session
20
+ | |-- routers/
21
+ | | `-- users.py
22
+ | |-- models/ # SQLAlchemy ORM models
23
+ | | `-- user.py
24
+ | |-- schemas/ # Pydantic request/response schemas
25
+ | | `-- user.py
26
+ | `-- services/ # Business logic layer
27
+ | `-- user_service.py
28
+ |-- tests/
29
+ | |-- conftest.py
30
+ | `-- test_users.py
31
+ |-- pyproject.toml
32
+ `-- .env
33
+ ```
34
+
35
+ ---
36
+
37
+ ## App Factory and Lifespan
38
+
39
+ ```python
40
+ # app/main.py
41
+ from contextlib import asynccontextmanager
42
+ from fastapi import FastAPI
43
+ from fastapi.middleware.cors import CORSMiddleware
44
+
45
+ from app.config import settings
46
+ from app.database import engine, Base
47
+ from app.routers import users
48
+
49
+
50
+ @asynccontextmanager
51
+ async def lifespan(app: FastAPI):
52
+ # Automatically create tables on startup for ease of use in dev/demo environments.
53
+ # For strict production applications, manage schemas via Alembic migrations instead.
54
+ async with engine.begin() as conn:
55
+ await conn.run_sync(Base.metadata.create_all)
56
+ yield
57
+ # Shutdown: close pooled resources.
58
+ await engine.dispose()
59
+
60
+
61
+ def create_app() -> FastAPI:
62
+ app = FastAPI(
63
+ title=settings.app_name,
64
+ version=settings.app_version,
65
+ lifespan=lifespan,
66
+ )
67
+
68
+ app.add_middleware(
69
+ CORSMiddleware,
70
+ allow_origins=settings.allowed_origins,
71
+ allow_credentials=settings.allow_credentials,
72
+ allow_methods=settings.allowed_methods,
73
+ allow_headers=settings.allowed_headers,
74
+ )
75
+
76
+ app.include_router(users.router, prefix="/users", tags=["users"])
77
+
78
+ return app
79
+
80
+
81
+ app = create_app()
82
+ ```
83
+
84
+ ---
85
+
86
+ ## Configuration with pydantic-settings
87
+
88
+ ```python
89
+ # app/config.py
90
+ from pydantic_settings import BaseSettings, SettingsConfigDict
91
+
92
+
93
+ class Settings(BaseSettings):
94
+ model_config = SettingsConfigDict(env_file=".env", env_file_encoding="utf-8")
95
+
96
+ app_name: str = "My App"
97
+ app_version: str = "0.1.0"
98
+ debug: bool = False
99
+
100
+ database_url: str
101
+ secret_key: str
102
+ algorithm: str = "HS256"
103
+ access_token_expire_minutes: int = 30
104
+
105
+ # Pydantic-settings v2 safely evaluates mutable list literals directly
106
+ allowed_origins: list[str] = ["http://localhost:3000"]
107
+ allowed_methods: list[str] = ["GET", "POST", "PATCH", "DELETE", "OPTIONS"]
108
+ allowed_headers: list[str] = ["Authorization", "Content-Type"]
109
+ allow_credentials: bool = True
110
+
111
+
112
+ settings = Settings()
113
+ ```
114
+
115
+ ---
116
+
117
+ ## Pydantic Schemas (v2)
118
+
119
+ ```python
120
+ # app/schemas/user.py
121
+ from datetime import datetime
122
+ from pydantic import BaseModel, EmailStr, Field, model_validator
123
+
124
+
125
+ class UserBase(BaseModel):
126
+ email: EmailStr
127
+ username: str = Field(min_length=3, max_length=50)
128
+
129
+
130
+ class UserCreate(UserBase):
131
+ password: str = Field(min_length=8)
132
+ password_confirm: str
133
+
134
+ @model_validator(mode="after")
135
+ def passwords_match(self) -> "UserCreate":
136
+ if self.password != self.password_confirm:
137
+ raise ValueError("Passwords do not match")
138
+ return self
139
+
140
+
141
+ class UserUpdate(BaseModel):
142
+ username: str | None = Field(default=None, min_length=3, max_length=50)
143
+ email: EmailStr | None = None
144
+
145
+
146
+ class UserResponse(UserBase):
147
+ id: int
148
+ is_active: bool
149
+ created_at: datetime
150
+
151
+ model_config = {"from_attributes": True}
152
+
153
+
154
+ class UserListResponse(BaseModel):
155
+ total: int
156
+ items: list[UserResponse]
157
+ ```
158
+
159
+ ---
160
+
161
+ ## Dependency Injection
162
+
163
+ ```python
164
+ # app/dependencies.py
165
+ from typing import Annotated, AsyncGenerator
166
+ from fastapi import Depends, HTTPException, status
167
+ from fastapi.security import OAuth2PasswordBearer
168
+ from jose import JWTError, jwt
169
+ from sqlalchemy.ext.asyncio import AsyncSession
170
+
171
+ from app.config import settings
172
+ from app.database import AsyncSessionLocal
173
+ from app.models.user import User
174
+
175
+ oauth2_scheme = OAuth2PasswordBearer(tokenUrl="/users/token")
176
+
177
+
178
+ async def get_db() -> AsyncGenerator[AsyncSession, None]:
179
+ async with AsyncSessionLocal() as session:
180
+ try:
181
+ yield session
182
+ except Exception:
183
+ await session.rollback()
184
+ raise
185
+
186
+
187
+ async def get_current_user(
188
+ token: Annotated[str, Depends(oauth2_scheme)],
189
+ db: Annotated[AsyncSession, Depends(get_db)],
190
+ ) -> User:
191
+ credentials_exception = HTTPException(
192
+ status_code=status.HTTP_401_UNAUTHORIZED,
193
+ detail="Could not validate credentials",
194
+ headers={"WWW-Authenticate": "Bearer"},
195
+ )
196
+ try:
197
+ payload = jwt.decode(token, settings.secret_key, algorithms=[settings.algorithm])
198
+ subject = payload.get("sub")
199
+ if subject is None:
200
+ raise credentials_exception
201
+ user_id = int(subject)
202
+ except (JWTError, TypeError, ValueError):
203
+ raise credentials_exception
204
+
205
+ user = await db.get(User, user_id)
206
+ if user is None:
207
+ raise credentials_exception
208
+ return user
209
+
210
+
211
+ async def get_current_active_user(
212
+ current_user: Annotated[User, Depends(get_current_user)],
213
+ ) -> User:
214
+ if not current_user.is_active:
215
+ raise HTTPException(status_code=status.HTTP_403_FORBIDDEN, detail="Inactive user")
216
+ return current_user
217
+
218
+
219
+ DbDep = Annotated[AsyncSession, Depends(get_db)]
220
+ CurrentUserDep = Annotated[User, Depends(get_current_user)]
221
+ ActiveUserDep = Annotated[User, Depends(get_current_active_user)]
222
+ ```
223
+
224
+ ---
225
+
226
+ ## Router and Endpoint Design
227
+
228
+ ```python
229
+ # app/routers/users.py
230
+ from typing import Annotated
231
+ from fastapi import APIRouter, HTTPException, Query, status
232
+ from fastapi.security import OAuth2PasswordRequestForm
233
+
234
+ from app.dependencies import ActiveUserDep, DbDep
235
+ from app.schemas.user import UserCreate, UserResponse, UserUpdate, UserListResponse
236
+ from app.services.user_service import DuplicateUserError, UserService
237
+
238
+ router = APIRouter()
239
+
240
+
241
+ @router.post("/", response_model=UserResponse, status_code=status.HTTP_201_CREATED)
242
+ async def create_user(payload: UserCreate, db: DbDep) -> UserResponse:
243
+ service = UserService(db)
244
+ try:
245
+ return await service.create(payload)
246
+ except DuplicateUserError:
247
+ raise HTTPException(status_code=400, detail="Email already registered")
248
+
249
+
250
+ @router.get("/me", response_model=UserResponse)
251
+ async def get_me(current_user: ActiveUserDep) -> UserResponse:
252
+ return current_user
253
+
254
+
255
+ @router.get("/", response_model=UserListResponse)
256
+ async def list_users(
257
+ db: DbDep,
258
+ current_user: ActiveUserDep,
259
+ skip: Annotated[int, Query(ge=0)] = 0,
260
+ limit: Annotated[int, Query(ge=1, le=100)] = 20,
261
+ ) -> UserListResponse:
262
+ service = UserService(db)
263
+ users, total = await service.list(skip=skip, limit=limit)
264
+ return UserListResponse(total=total, items=users)
265
+
266
+
267
+ @router.patch("/{user_id}", response_model=UserResponse)
268
+ async def update_user(
269
+ user_id: int,
270
+ payload: UserUpdate,
271
+ db: DbDep,
272
+ current_user: ActiveUserDep,
273
+ ) -> UserResponse:
274
+ if current_user.id != user_id:
275
+ raise HTTPException(status_code=403, detail="Not authorized")
276
+ service = UserService(db)
277
+ try:
278
+ user = await service.update(user_id, payload)
279
+ except DuplicateUserError:
280
+ raise HTTPException(status_code=400, detail="Email already registered")
281
+ if user is None:
282
+ raise HTTPException(status_code=404, detail="User not found")
283
+ return user
284
+
285
+
286
+ @router.post("/token")
287
+ async def login(
288
+ form_data: Annotated[OAuth2PasswordRequestForm, Depends()],
289
+ db: DbDep,
290
+ ) -> dict[str, str]:
291
+ service = UserService(db)
292
+ token = await service.authenticate(form_data.username, form_data.password)
293
+ if token is None:
294
+ raise HTTPException(
295
+ status_code=status.HTTP_401_UNAUTHORIZED,
296
+ detail="Incorrect username or password",
297
+ headers={"WWW-Authenticate": "Bearer"},
298
+ )
299
+ return {"access_token": token, "token_type": "bearer"}
300
+ ```
301
+
302
+ ---
303
+
304
+ ## Service Layer
305
+
306
+ ```python
307
+ # app/services/user_service.py
308
+ from datetime import datetime, timedelta, timezone
309
+
310
+ from jose import jwt
311
+ from passlib.context import CryptContext
312
+ from sqlalchemy import func, select
313
+ from sqlalchemy.exc import IntegrityError
314
+ from sqlalchemy.ext.asyncio import AsyncSession
315
+
316
+ from app.config import settings
317
+ from app.models.user import User
318
+ from app.schemas.user import UserCreate, UserUpdate
319
+
320
+ pwd_context = CryptContext(schemes=["bcrypt"], deprecated="auto")
321
+
322
+
323
+ class DuplicateUserError(Exception):
324
+ """Raised when a unique user field conflicts with an existing row."""
325
+
326
+
327
+ class UserService:
328
+ def __init__(self, db: AsyncSession) -> None:
329
+ self.db = db
330
+
331
+ async def get_by_email(self, email: str) -> User | None:
332
+ result = await self.db.execute(select(User).where(User.email == email))
333
+ return result.scalar_one_or_none()
334
+
335
+ async def create(self, payload: UserCreate) -> User:
336
+ user = User(
337
+ email=payload.email,
338
+ username=payload.username,
339
+ hashed_password=pwd_context.hash(payload.password),
340
+ )
341
+ self.db.add(user)
342
+ try:
343
+ # Rely on atomic DB constraints rather than race-prone application-level prechecks
344
+ await self.db.commit()
345
+ except IntegrityError as exc:
346
+ await self.db.rollback()
347
+ raise DuplicateUserError from exc
348
+ await self.db.refresh(user)
349
+ return user
350
+
351
+ async def list(self, skip: int = 0, limit: int = 20) -> tuple[list[User], int]:
352
+ total_result = await self.db.execute(select(func.count(User.id)))
353
+ total = total_result.scalar_one()
354
+ # Enforce explicit deterministic ordering to ensure reliable pagination
355
+ result = await self.db.execute(
356
+ select(User).order_by(User.id).offset(skip).limit(limit)
357
+ )
358
+ return list(result.scalars()), total
359
+
360
+ async def update(self, user_id: int, payload: UserUpdate) -> User | None:
361
+ user = await self.db.get(User, user_id)
362
+ if user is None:
363
+ return None
364
+ for field, value in payload.model_dump(exclude_unset=True).items():
365
+ setattr(user, field, value)
366
+ try:
367
+ await self.db.commit()
368
+ except IntegrityError as exc:
369
+ await self.db.rollback()
370
+ raise DuplicateUserError from exc
371
+ await self.db.refresh(user)
372
+ return user
373
+
374
+ async def authenticate(self, email: str, password: str) -> str | None:
375
+ user = await self.get_by_email(email)
376
+ if user is None or not pwd_context.verify(password, user.hashed_password):
377
+ return None
378
+ expire = datetime.now(timezone.utc) + timedelta(
379
+ minutes=settings.access_token_expire_minutes
380
+ )
381
+ return jwt.encode(
382
+ {"sub": str(user.id), "exp": expire},
383
+ settings.secret_key,
384
+ algorithm=settings.algorithm,
385
+ )
386
+ ```
387
+
388
+ > **Note on Database Design:** Application-level unique handling requires an underlying unique database index (e.g., `unique=True` on your SQLAlchemy mapping attributes). Without underlying constraints, application layer error-catching cannot safely prevent concurrent race conditions.
389
+
390
+ ---
391
+
392
+ ## Testing with httpx and pytest
393
+
394
+ ```python
395
+ # tests/conftest.py
396
+ import pytest_asyncio
397
+ from httpx import ASGITransport, AsyncClient
398
+ from sqlalchemy.ext.asyncio import AsyncSession, async_sessionmaker, create_async_engine
399
+
400
+ from app.database import Base
401
+ from app.dependencies import get_db
402
+ from app.main import create_app
403
+
404
+ TEST_DATABASE_URL = "sqlite+aiosqlite:///:memory:"
405
+
406
+ engine = create_async_engine(TEST_DATABASE_URL)
407
+ TestingSessionLocal = async_sessionmaker(engine, expire_on_commit=False)
408
+
409
+
410
+ @pytest_asyncio.fixture(autouse=True)
411
+ async def setup_db():
412
+ async with engine.begin() as conn:
413
+ await conn.run_sync(Base.metadata.create_all)
414
+ yield
415
+ async with engine.begin() as conn:
416
+ await conn.run_sync(Base.metadata.drop_all)
417
+
418
+
419
+ @pytest_asyncio.fixture
420
+ async def db_session():
421
+ async with TestingSessionLocal() as session:
422
+ yield session
423
+ await session.rollback()
424
+
425
+
426
+ @pytest_asyncio.fixture
427
+ async def client(db_session: AsyncSession):
428
+ app = create_app()
429
+
430
+ async def override_get_db():
431
+ yield db_session
432
+
433
+ app.dependency_overrides[get_db] = override_get_db
434
+
435
+ async with AsyncClient(
436
+ transport=ASGITransport(app=app), base_url="http://test"
437
+ ) as ac:
438
+ yield ac
439
+
440
+
441
+ @pytest_asyncio.fixture
442
+ async def registered_user(client: AsyncClient) -> dict:
443
+ resp = await client.post("/users/", json={
444
+ "email": "test@example.com",
445
+ "username": "testuser",
446
+ "password": "securepass1",
447
+ "password_confirm": "securepass1",
448
+ })
449
+ assert resp.status_code == 201
450
+ return resp.json()
451
+
452
+
453
+ @pytest_asyncio.fixture
454
+ async def auth_token(client: AsyncClient, registered_user: dict) -> str:
455
+ resp = await client.post("/users/token", data={
456
+ "username": "test@example.com",
457
+ "password": "securepass1",
458
+ })
459
+ assert resp.status_code == 200
460
+ return resp.json()["access_token"]
461
+
462
+
463
+ @pytest_asyncio.fixture
464
+ async def auth_client(client: AsyncClient, auth_token: str) -> AsyncClient:
465
+ client.headers.update({"Authorization": f"Bearer {auth_token}"})
466
+ return client
467
+ ```
468
+
469
+ ---
470
+
471
+ ## Anti-Patterns
472
+
473
+ ```python
474
+ # Bad: business logic inside route handlers.
475
+ @router.post("/users/")
476
+ async def create_user(payload: UserCreate, db: DbDep):
477
+ hashed = bcrypt.hash(payload.password)
478
+ user = User(email=payload.email, hashed_password=hashed)
479
+ db.add(user)
480
+ await db.commit()
481
+ return user
482
+
483
+ # Good: thin route, transactional service handling.
484
+ @router.post("/users/", response_model=UserResponse, status_code=201)
485
+ async def create_user(payload: UserCreate, db: DbDep):
486
+ try:
487
+ return await UserService(db).create(payload)
488
+ except DuplicateUserError:
489
+ raise HTTPException(status_code=400, detail="Email already registered")
490
+
491
+
492
+ # Bad: sync DB calls in async routes block the event loop.
493
+ @router.get("/items/")
494
+ async def list_items(db: Session = Depends(get_db)):
495
+ return db.query(Item).all()
496
+
497
+ # Good: use async SQLAlchemy executions.
498
+ @router.get("/items/")
499
+ async def list_items(db: AsyncSession = Depends(get_db)):
500
+ result = await db.execute(select(Item))
501
+ return result.scalars().all()
502
+ ```
503
+
504
+ ---
505
+
506
+ ## Best Practices
507
+
508
+ - Always declare a typed `response_model` to prevent accidental PII/data leaks and output clean OpenAPI schemas.
509
+ - Consolidate standard middleware dependency injections via type-aliasing: `DbDep = Annotated[AsyncSession, Depends(get_db)]`.
510
+ - Wrap database mutation boundaries gracefully within transactions inside your service layer, catching structural database errors directly.
511
+ - Parse JWT parameters defensively, expecting potential string/integer cast mismatches from modern payload variations.
512
+ - Enforce deterministic sorting (e.g., `.order_by(Model.id)`) on all offset/limit paginated endpoints to avoid data skips.
513
+ - Isolate authorization checks from core authentication dependencies to provide precise REST status signals (`401` vs `403`).
@@ -0,0 +1,127 @@
1
+ ---
2
+ name: finance-billing-ops
3
+ description: [ECC] Evidence-first revenue, pricing, refunds, team-billing, and billing-model truth workflow for ECC. Use when the user wants a sales snapshot, pricing comparison, duplicate-charge diagnosis, or code-backed billing reality instead of generic payments advice.
4
+ origin: ECC
5
+ ---
6
+
7
+ # Finance Billing Ops
8
+
9
+ Use this when the user wants to understand money, pricing, refunds, team-seat logic, or whether the product actually behaves the way the website and sales copy imply.
10
+
11
+ This is broader than `customer-billing-ops`. That skill is for customer remediation. This skill is for operator truth: revenue state, pricing decisions, team billing, and code-backed billing behavior.
12
+
13
+ ## Skill Stack
14
+
15
+ Pull these ECC-native skills into the workflow when relevant:
16
+
17
+ - `customer-billing-ops` for customer-specific remediation and follow-up
18
+ - `research-ops` when competitor pricing or current market evidence matters
19
+ - `market-research` when the answer should end in a pricing recommendation
20
+ - `github-ops` when the billing truth depends on code, backlog, or release state in sibling repos
21
+ - `verification-loop` when the answer depends on proving checkout, seat handling, or entitlement behavior
22
+
23
+ ## When to Use
24
+
25
+ - user asks for Stripe sales, refunds, MRR, or recent customer activity
26
+ - user asks whether team billing, per-seat billing, or quota stacking is real in code
27
+ - user wants competitor pricing comparisons or pricing-model benchmarks
28
+ - the question mixes revenue facts with product implementation truth
29
+
30
+ ## Guardrails
31
+
32
+ - distinguish live data from saved snapshots
33
+ - separate:
34
+ - revenue fact
35
+ - customer impact
36
+ - code-backed product truth
37
+ - recommendation
38
+ - do not say "per seat" unless the actual entitlement path enforces it
39
+ - do not assume duplicate subscriptions imply duplicate value
40
+
41
+ ## Workflow
42
+
43
+ ### 1. Start from the freshest billing evidence
44
+
45
+ Prefer live billing data. If the data is not live, state the snapshot timestamp explicitly.
46
+
47
+ Normalize the picture:
48
+
49
+ - paid sales
50
+ - active subscriptions
51
+ - failed or incomplete checkouts
52
+ - refunds
53
+ - disputes
54
+ - duplicate subscriptions
55
+
56
+ ### 2. Separate customer incidents from product truth
57
+
58
+ If the question is customer-specific, classify first:
59
+
60
+ - duplicate checkout
61
+ - real team intent
62
+ - broken self-serve controls
63
+ - unmet product value
64
+ - failed payment or incomplete setup
65
+
66
+ Then separate that from the broader product question:
67
+
68
+ - does team billing really exist?
69
+ - are seats actually counted?
70
+ - does checkout quantity change entitlement?
71
+ - does the site overstate current behavior?
72
+
73
+ ### 3. Inspect code-backed billing behavior
74
+
75
+ If the answer depends on implementation truth, inspect the code path:
76
+
77
+ - checkout
78
+ - pricing page
79
+ - entitlement calculation
80
+ - seat or quota handling
81
+ - installation vs user usage logic
82
+ - billing portal or self-serve management support
83
+
84
+ ### 4. End with a decision and product gap
85
+
86
+ Report:
87
+
88
+ - sales snapshot
89
+ - issue diagnosis
90
+ - product truth
91
+ - recommended operator action
92
+ - product or backlog gap
93
+
94
+ ## Output Format
95
+
96
+ ```text
97
+ SNAPSHOT
98
+ - timestamp
99
+ - revenue / subscriptions / anomalies
100
+
101
+ CUSTOMER IMPACT
102
+ - who is affected
103
+ - what happened
104
+
105
+ PRODUCT TRUTH
106
+ - what the code actually does
107
+ - what the website or sales copy claims
108
+
109
+ DECISION
110
+ - refund / preserve / convert / no-op
111
+
112
+ PRODUCT GAP
113
+ - exact follow-up item to build or fix
114
+ ```
115
+
116
+ ## Pitfalls
117
+
118
+ - do not conflate failed attempts with net revenue
119
+ - do not infer team billing from marketing language alone
120
+ - do not compare competitor pricing from memory when current evidence is available
121
+ - do not jump from diagnosis straight to refund without classifying the issue
122
+
123
+ ## Verification
124
+
125
+ - the answer includes a live-data statement or snapshot timestamp
126
+ - product-truth claims are code-backed
127
+ - customer-impact and broader pricing/product conclusions are separated cleanly