arboris-cli 1.0.0 → 1.1.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 (451) hide show
  1. package/dist/cli.mjs +382 -0
  2. package/manifest.json +323 -0
  3. package/package.json +22 -10
  4. package/prisma/skills/accessibility/SKILL.md +147 -0
  5. package/prisma/skills/agent-architecture-audit/SKILL.md +257 -0
  6. package/prisma/skills/agent-eval/SKILL.md +146 -0
  7. package/prisma/skills/agent-harness-construction/SKILL.md +74 -0
  8. package/prisma/skills/agent-introspection-debugging/SKILL.md +154 -0
  9. package/prisma/skills/agent-payment-x402/SKILL.md +225 -0
  10. package/prisma/skills/agent-self-evaluation/SKILL.md +182 -0
  11. package/prisma/skills/agent-self-evaluation/examples/high-score-example.md +87 -0
  12. package/prisma/skills/agent-self-evaluation/examples/low-score-example.md +86 -0
  13. package/prisma/skills/agent-self-evaluation/references/evaluation-criteria.md +71 -0
  14. package/prisma/skills/agent-self-evaluation/references/hook-integration.md +64 -0
  15. package/prisma/skills/agent-self-evaluation/scripts/evaluate.py +408 -0
  16. package/prisma/skills/agent-self-evaluation/templates/evaluation-report.md +86 -0
  17. package/prisma/skills/agent-sort/SKILL.md +216 -0
  18. package/prisma/skills/agentic-engineering/SKILL.md +64 -0
  19. package/prisma/skills/agentic-os/SKILL.md +388 -0
  20. package/prisma/skills/ai-first-engineering/SKILL.md +52 -0
  21. package/prisma/skills/ai-regression-testing/SKILL.md +386 -0
  22. package/prisma/skills/android-clean-architecture/SKILL.md +340 -0
  23. package/prisma/skills/angular-developer/SKILL.md +155 -0
  24. package/prisma/skills/angular-developer/references/angular-animations.md +160 -0
  25. package/prisma/skills/angular-developer/references/angular-aria.md +410 -0
  26. package/prisma/skills/angular-developer/references/cli.md +86 -0
  27. package/prisma/skills/angular-developer/references/component-harnesses.md +59 -0
  28. package/prisma/skills/angular-developer/references/component-styling.md +91 -0
  29. package/prisma/skills/angular-developer/references/components.md +117 -0
  30. package/prisma/skills/angular-developer/references/creating-services.md +97 -0
  31. package/prisma/skills/angular-developer/references/data-resolvers.md +69 -0
  32. package/prisma/skills/angular-developer/references/define-routes.md +67 -0
  33. package/prisma/skills/angular-developer/references/defining-providers.md +72 -0
  34. package/prisma/skills/angular-developer/references/di-fundamentals.md +120 -0
  35. package/prisma/skills/angular-developer/references/e2e-testing.md +56 -0
  36. package/prisma/skills/angular-developer/references/effects.md +83 -0
  37. package/prisma/skills/angular-developer/references/hierarchical-injectors.md +43 -0
  38. package/prisma/skills/angular-developer/references/host-elements.md +80 -0
  39. package/prisma/skills/angular-developer/references/injection-context.md +63 -0
  40. package/prisma/skills/angular-developer/references/inputs.md +101 -0
  41. package/prisma/skills/angular-developer/references/linked-signal.md +59 -0
  42. package/prisma/skills/angular-developer/references/loading-strategies.md +61 -0
  43. package/prisma/skills/angular-developer/references/mcp.md +108 -0
  44. package/prisma/skills/angular-developer/references/navigate-to-routes.md +69 -0
  45. package/prisma/skills/angular-developer/references/outputs.md +86 -0
  46. package/prisma/skills/angular-developer/references/reactive-forms.md +122 -0
  47. package/prisma/skills/angular-developer/references/rendering-strategies.md +44 -0
  48. package/prisma/skills/angular-developer/references/resource.md +77 -0
  49. package/prisma/skills/angular-developer/references/route-animations.md +56 -0
  50. package/prisma/skills/angular-developer/references/route-guards.md +52 -0
  51. package/prisma/skills/angular-developer/references/router-lifecycle.md +45 -0
  52. package/prisma/skills/angular-developer/references/router-testing.md +87 -0
  53. package/prisma/skills/angular-developer/references/show-routes-with-outlets.md +68 -0
  54. package/prisma/skills/angular-developer/references/signal-forms.md +795 -0
  55. package/prisma/skills/angular-developer/references/signals-overview.md +94 -0
  56. package/prisma/skills/angular-developer/references/tailwind-css.md +69 -0
  57. package/prisma/skills/angular-developer/references/template-driven-forms.md +114 -0
  58. package/prisma/skills/angular-developer/references/testing-fundamentals.md +65 -0
  59. package/prisma/skills/api-connector-builder/SKILL.md +121 -0
  60. package/prisma/skills/api-design/SKILL.md +524 -0
  61. package/prisma/skills/architecture-decision-records/SKILL.md +180 -0
  62. package/prisma/skills/article-writing/SKILL.md +80 -0
  63. package/prisma/skills/automation-audit-ops/SKILL.md +143 -0
  64. package/prisma/skills/autonomous-agent-harness/SKILL.md +274 -0
  65. package/prisma/skills/autonomous-loops/SKILL.md +611 -0
  66. package/prisma/skills/backend-patterns/SKILL.md +562 -0
  67. package/prisma/skills/benchmark/SKILL.md +94 -0
  68. package/prisma/skills/benchmark-methodology/SKILL.md +190 -0
  69. package/prisma/skills/benchmark-optimization-loop/SKILL.md +70 -0
  70. package/prisma/skills/blender-motion-state-inspection/SKILL.md +165 -0
  71. package/prisma/skills/blueprint/SKILL.md +106 -0
  72. package/prisma/skills/brand-discovery/SKILL.md +145 -0
  73. package/prisma/skills/brand-discovery/references/10_purpose-why.md +40 -0
  74. package/prisma/skills/brand-discovery/references/20_positioning.md +44 -0
  75. package/prisma/skills/brand-discovery/references/30_audience-niche.md +52 -0
  76. package/prisma/skills/brand-discovery/references/40_personality-archetype.md +57 -0
  77. package/prisma/skills/brand-discovery/references/50_voice-tone.md +59 -0
  78. package/prisma/skills/brand-discovery/references/60_narrative-story.md +50 -0
  79. package/prisma/skills/brand-discovery/references/70_founder-tension.md +49 -0
  80. package/prisma/skills/brand-discovery/references/90_SYNTHESIS.md +133 -0
  81. package/prisma/skills/brand-voice/SKILL.md +98 -0
  82. package/prisma/skills/brand-voice/references/voice-profile-schema.md +55 -0
  83. package/prisma/skills/browser-qa/SKILL.md +105 -0
  84. package/prisma/skills/bun-runtime/SKILL.md +85 -0
  85. package/prisma/skills/canary-watch/SKILL.md +108 -0
  86. package/prisma/skills/carrier-relationship-management/SKILL.md +212 -0
  87. package/prisma/skills/cisco-ios-patterns/SKILL.md +164 -0
  88. package/prisma/skills/ck/SKILL.md +148 -0
  89. package/prisma/skills/ck/commands/forget.mjs +44 -0
  90. package/prisma/skills/ck/commands/info.mjs +24 -0
  91. package/prisma/skills/ck/commands/init.mjs +143 -0
  92. package/prisma/skills/ck/commands/list.mjs +40 -0
  93. package/prisma/skills/ck/commands/migrate.mjs +202 -0
  94. package/prisma/skills/ck/commands/resume.mjs +36 -0
  95. package/prisma/skills/ck/commands/save.mjs +210 -0
  96. package/prisma/skills/ck/commands/shared.mjs +387 -0
  97. package/prisma/skills/ck/hooks/session-start.mjs +224 -0
  98. package/prisma/skills/claude-devfleet/SKILL.md +112 -0
  99. package/prisma/skills/click-path-audit/SKILL.md +245 -0
  100. package/prisma/skills/clickhouse-io/SKILL.md +440 -0
  101. package/prisma/skills/code-tour/SKILL.md +254 -0
  102. package/prisma/skills/codebase-onboarding/SKILL.md +234 -0
  103. package/prisma/skills/codehealth-mcp/SKILL.md +167 -0
  104. package/prisma/skills/coding-standards/SKILL.md +551 -0
  105. package/prisma/skills/competitive-platform-analysis/SKILL.md +214 -0
  106. package/prisma/skills/competitive-report-structure/SKILL.md +162 -0
  107. package/prisma/skills/compose-multiplatform-patterns/SKILL.md +300 -0
  108. package/prisma/skills/config-gc/SKILL.md +120 -0
  109. package/prisma/skills/configure-ecc/SKILL.md +385 -0
  110. package/prisma/skills/connections-optimizer/SKILL.md +190 -0
  111. package/prisma/skills/content-engine/SKILL.md +132 -0
  112. package/prisma/skills/content-hash-cache-pattern/SKILL.md +162 -0
  113. package/prisma/skills/context-budget/SKILL.md +136 -0
  114. package/prisma/skills/continuous-agent-loop/SKILL.md +46 -0
  115. package/prisma/skills/continuous-learning/SKILL.md +132 -0
  116. package/prisma/skills/continuous-learning/config.json +18 -0
  117. package/prisma/skills/continuous-learning/evaluate-session.sh +69 -0
  118. package/prisma/skills/continuous-learning-v2/SKILL.md +361 -0
  119. package/prisma/skills/continuous-learning-v2/agents/observer-loop.sh +359 -0
  120. package/prisma/skills/continuous-learning-v2/agents/observer.md +189 -0
  121. package/prisma/skills/continuous-learning-v2/agents/session-guardian.sh +150 -0
  122. package/prisma/skills/continuous-learning-v2/agents/start-observer.sh +248 -0
  123. package/prisma/skills/continuous-learning-v2/config.json +8 -0
  124. package/prisma/skills/continuous-learning-v2/hooks/observe.sh +585 -0
  125. package/prisma/skills/continuous-learning-v2/scripts/detect-project.sh +322 -0
  126. package/prisma/skills/continuous-learning-v2/scripts/instinct-cli.py +1956 -0
  127. package/prisma/skills/continuous-learning-v2/scripts/lib/homunculus-dir.sh +31 -0
  128. package/prisma/skills/continuous-learning-v2/scripts/migrate-homunculus.sh +68 -0
  129. package/prisma/skills/continuous-learning-v2/scripts/test_parse_instinct.py +1421 -0
  130. package/prisma/skills/cost-aware-llm-pipeline/SKILL.md +184 -0
  131. package/prisma/skills/cost-tracking/SKILL.md +97 -0
  132. package/prisma/skills/council/SKILL.md +204 -0
  133. package/prisma/skills/cpp-coding-standards/SKILL.md +724 -0
  134. package/prisma/skills/cpp-testing/SKILL.md +325 -0
  135. package/prisma/skills/crosspost/SKILL.md +112 -0
  136. package/prisma/skills/csharp-testing/SKILL.md +322 -0
  137. package/prisma/skills/customer-billing-ops/SKILL.md +141 -0
  138. package/prisma/skills/customs-trade-compliance/SKILL.md +263 -0
  139. package/prisma/skills/dart-flutter-patterns/SKILL.md +564 -0
  140. package/prisma/skills/dashboard-builder/SKILL.md +109 -0
  141. package/prisma/skills/data-scraper-agent/SKILL.md +765 -0
  142. package/prisma/skills/data-throughput-accelerator/SKILL.md +73 -0
  143. package/prisma/skills/database-migrations/SKILL.md +430 -0
  144. package/prisma/skills/deep-research/SKILL.md +160 -0
  145. package/prisma/skills/defi-amm-security/SKILL.md +167 -0
  146. package/prisma/skills/delivery-gate/SKILL.md +126 -0
  147. package/prisma/skills/delivery-gate/hooks/quality-gate.py +220 -0
  148. package/prisma/skills/deployment-patterns/SKILL.md +428 -0
  149. package/prisma/skills/design-system/SKILL.md +83 -0
  150. package/prisma/skills/django-celery/SKILL.md +458 -0
  151. package/prisma/skills/django-patterns/SKILL.md +735 -0
  152. package/prisma/skills/django-security/SKILL.md +644 -0
  153. package/prisma/skills/django-tdd/SKILL.md +730 -0
  154. package/prisma/skills/django-verification/SKILL.md +470 -0
  155. package/prisma/skills/dmux-workflows/SKILL.md +192 -0
  156. package/prisma/skills/docker-patterns/SKILL.md +365 -0
  157. package/prisma/skills/documentation-lookup/SKILL.md +91 -0
  158. package/prisma/skills/dotnet-patterns/SKILL.md +322 -0
  159. package/prisma/skills/dynamic-workflow-mode/SKILL.md +124 -0
  160. package/prisma/skills/e2e-testing/SKILL.md +327 -0
  161. package/prisma/skills/ecc-guide/SKILL.md +190 -0
  162. package/prisma/skills/ecc-recipes/SKILL.md +149 -0
  163. package/prisma/skills/ecc-tools-cost-audit/SKILL.md +161 -0
  164. package/prisma/skills/email-ops/SKILL.md +122 -0
  165. package/prisma/skills/energy-procurement/SKILL.md +228 -0
  166. package/prisma/skills/enterprise-agent-ops/SKILL.md +51 -0
  167. package/prisma/skills/error-handling/SKILL.md +377 -0
  168. package/prisma/skills/eval-harness/SKILL.md +271 -0
  169. package/prisma/skills/evm-token-decimals/SKILL.md +131 -0
  170. package/prisma/skills/exa-search/SKILL.md +108 -0
  171. package/prisma/skills/fal-ai-media/SKILL.md +289 -0
  172. package/prisma/skills/fastapi-patterns/SKILL.md +514 -0
  173. package/prisma/skills/finance-billing-ops/SKILL.md +128 -0
  174. package/prisma/skills/flox-environments/SKILL.md +497 -0
  175. package/prisma/skills/flutter-dart-code-review/SKILL.md +436 -0
  176. package/prisma/skills/foundation-models-on-device/SKILL.md +243 -0
  177. package/prisma/skills/frontend-a11y/SKILL.md +446 -0
  178. package/prisma/skills/frontend-design-direction/SKILL.md +93 -0
  179. package/prisma/skills/frontend-patterns/SKILL.md +657 -0
  180. package/prisma/skills/frontend-slides/SKILL.md +185 -0
  181. package/prisma/skills/frontend-slides/STYLE_PRESETS.md +330 -0
  182. package/prisma/skills/frontend-slides/animation-patterns.md +122 -0
  183. package/prisma/skills/frontend-slides/html-template.md +419 -0
  184. package/prisma/skills/frontend-slides/scripts/export-pdf.sh +418 -0
  185. package/prisma/skills/frontend-slides/scripts/extract-pptx.py +96 -0
  186. package/prisma/skills/frontend-slides/viewport-base.css +153 -0
  187. package/prisma/skills/fsharp-testing/SKILL.md +281 -0
  188. package/prisma/skills/gan-style-harness/SKILL.md +279 -0
  189. package/prisma/skills/gateguard/SKILL.md +133 -0
  190. package/prisma/skills/generating-python-installer/SKILL.md +820 -0
  191. package/prisma/skills/git-workflow/SKILL.md +716 -0
  192. package/prisma/skills/github-ops/SKILL.md +145 -0
  193. package/prisma/skills/golang-patterns/SKILL.md +675 -0
  194. package/prisma/skills/golang-testing/SKILL.md +721 -0
  195. package/prisma/skills/google-workspace-ops/SKILL.md +96 -0
  196. package/prisma/skills/growth-log/SKILL.md +128 -0
  197. package/prisma/skills/healthcare-cdss-patterns/SKILL.md +246 -0
  198. package/prisma/skills/healthcare-emr-patterns/SKILL.md +160 -0
  199. package/prisma/skills/healthcare-eval-harness/SKILL.md +208 -0
  200. package/prisma/skills/healthcare-phi-compliance/SKILL.md +146 -0
  201. package/prisma/skills/hermes-imports/SKILL.md +89 -0
  202. package/prisma/skills/hexagonal-architecture/SKILL.md +277 -0
  203. package/prisma/skills/hipaa-compliance/SKILL.md +79 -0
  204. package/prisma/skills/homelab-network-readiness/SKILL.md +170 -0
  205. package/prisma/skills/homelab-network-setup/SKILL.md +130 -0
  206. package/prisma/skills/homelab-pihole-dns/SKILL.md +275 -0
  207. package/prisma/skills/homelab-vlan-segmentation/SKILL.md +312 -0
  208. package/prisma/skills/homelab-wireguard-vpn/SKILL.md +306 -0
  209. package/prisma/skills/hookify-rules/SKILL.md +128 -0
  210. package/prisma/skills/inherit-legacy-style/SKILL.md +157 -0
  211. package/prisma/skills/intent-driven-development/SKILL.md +360 -0
  212. package/prisma/skills/inventory-demand-planning/SKILL.md +247 -0
  213. package/prisma/skills/investor-materials/SKILL.md +97 -0
  214. package/prisma/skills/investor-outreach/SKILL.md +92 -0
  215. package/prisma/skills/ios-icon-gen/SKILL.md +158 -0
  216. package/prisma/skills/ios-icon-gen/scripts/generate_icons.swift +258 -0
  217. package/prisma/skills/ios-icon-gen/scripts/iconify_gen.sh +235 -0
  218. package/prisma/skills/iterative-retrieval/SKILL.md +212 -0
  219. package/prisma/skills/ito-basket-compare/SKILL.md +64 -0
  220. package/prisma/skills/ito-data-atlas-agent/SKILL.md +64 -0
  221. package/prisma/skills/ito-market-intelligence/SKILL.md +61 -0
  222. package/prisma/skills/ito-trade-planner/SKILL.md +68 -0
  223. package/prisma/skills/java-coding-standards/SKILL.md +384 -0
  224. package/prisma/skills/jira-integration/SKILL.md +303 -0
  225. package/prisma/skills/jpa-patterns/SKILL.md +152 -0
  226. package/prisma/skills/knowledge-ops/SKILL.md +155 -0
  227. package/prisma/skills/kotlin-coroutines-flows/SKILL.md +285 -0
  228. package/prisma/skills/kotlin-exposed-patterns/SKILL.md +720 -0
  229. package/prisma/skills/kotlin-ktor-patterns/SKILL.md +690 -0
  230. package/prisma/skills/kotlin-patterns/SKILL.md +712 -0
  231. package/prisma/skills/kotlin-testing/SKILL.md +825 -0
  232. package/prisma/skills/kubernetes-patterns/SKILL.md +756 -0
  233. package/prisma/skills/laravel-patterns/SKILL.md +416 -0
  234. package/prisma/skills/laravel-plugin-discovery/SKILL.md +230 -0
  235. package/prisma/skills/laravel-security/SKILL.md +948 -0
  236. package/prisma/skills/laravel-tdd/SKILL.md +675 -0
  237. package/prisma/skills/laravel-verification/SKILL.md +180 -0
  238. package/prisma/skills/latency-critical-systems/SKILL.md +74 -0
  239. package/prisma/skills/lead-intelligence/SKILL.md +322 -0
  240. package/prisma/skills/lead-intelligence/agents/enrichment-agent.md +85 -0
  241. package/prisma/skills/lead-intelligence/agents/mutual-mapper.md +75 -0
  242. package/prisma/skills/lead-intelligence/agents/outreach-drafter.md +98 -0
  243. package/prisma/skills/lead-intelligence/agents/signal-scorer.md +60 -0
  244. package/prisma/skills/liquid-glass-design/SKILL.md +279 -0
  245. package/prisma/skills/llm-trading-agent-security/SKILL.md +147 -0
  246. package/prisma/skills/logistics-exception-management/SKILL.md +222 -0
  247. package/prisma/skills/loop-design-check/SKILL.md +143 -0
  248. package/prisma/skills/mailtrap-email-integration/SKILL.md +77 -0
  249. package/prisma/skills/make-interfaces-feel-better/SKILL.md +152 -0
  250. package/prisma/skills/manim-video/SKILL.md +90 -0
  251. package/prisma/skills/manim-video/assets/network_graph_scene.py +52 -0
  252. package/prisma/skills/market-research/SKILL.md +76 -0
  253. package/prisma/skills/marketing-campaign/SKILL.md +114 -0
  254. package/prisma/skills/mcp-server-patterns/SKILL.md +70 -0
  255. package/prisma/skills/messages-ops/SKILL.md +105 -0
  256. package/prisma/skills/ml-adoption-playbook/SKILL.md +57 -0
  257. package/prisma/skills/mle-workflow/SKILL.md +347 -0
  258. package/prisma/skills/motion-advanced/SKILL.md +596 -0
  259. package/prisma/skills/motion-foundations/SKILL.md +299 -0
  260. package/prisma/skills/motion-patterns/SKILL.md +434 -0
  261. package/prisma/skills/motion-ui/SKILL.md +576 -0
  262. package/prisma/skills/mysql-patterns/SKILL.md +413 -0
  263. package/prisma/skills/nanoclaw-repl/SKILL.md +34 -0
  264. package/prisma/skills/nestjs-patterns/SKILL.md +231 -0
  265. package/prisma/skills/netmiko-ssh-automation/SKILL.md +174 -0
  266. package/prisma/skills/network-bgp-diagnostics/SKILL.md +168 -0
  267. package/prisma/skills/network-config-validation/SKILL.md +211 -0
  268. package/prisma/skills/network-interface-health/SKILL.md +153 -0
  269. package/prisma/skills/nextjs-turbopack/SKILL.md +58 -0
  270. package/prisma/skills/nodejs-keccak256/SKILL.md +103 -0
  271. package/prisma/skills/nutrient-document-processing/SKILL.md +168 -0
  272. package/prisma/skills/nuxt4-patterns/SKILL.md +101 -0
  273. package/prisma/skills/openclaw-persona-forge/SKILL.md +289 -0
  274. package/prisma/skills/openclaw-persona-forge/gacha.py +224 -0
  275. package/prisma/skills/openclaw-persona-forge/gacha.sh +5 -0
  276. package/prisma/skills/openclaw-persona-forge/references/avatar-style.md +124 -0
  277. package/prisma/skills/openclaw-persona-forge/references/boundary-rules.md +53 -0
  278. package/prisma/skills/openclaw-persona-forge/references/error-handling.md +53 -0
  279. package/prisma/skills/openclaw-persona-forge/references/identity-tension.md +48 -0
  280. package/prisma/skills/openclaw-persona-forge/references/naming-system.md +39 -0
  281. package/prisma/skills/openclaw-persona-forge/references/output-template.md +166 -0
  282. package/prisma/skills/opensource-pipeline/SKILL.md +256 -0
  283. package/prisma/skills/orch-add-feature/SKILL.md +45 -0
  284. package/prisma/skills/orch-build-mvp/SKILL.md +49 -0
  285. package/prisma/skills/orch-change-feature/SKILL.md +43 -0
  286. package/prisma/skills/orch-fix-defect/SKILL.md +43 -0
  287. package/prisma/skills/orch-pipeline/SKILL.md +121 -0
  288. package/prisma/skills/orch-refine-code/SKILL.md +44 -0
  289. package/prisma/skills/parallel-execution-optimizer/SKILL.md +73 -0
  290. package/prisma/skills/perl-patterns/SKILL.md +505 -0
  291. package/prisma/skills/perl-security/SKILL.md +504 -0
  292. package/prisma/skills/perl-testing/SKILL.md +476 -0
  293. package/prisma/skills/plan-orchestrate/SKILL.md +263 -0
  294. package/prisma/skills/plankton-code-quality/SKILL.md +237 -0
  295. package/prisma/skills/postgres-patterns/SKILL.md +148 -0
  296. package/prisma/skills/prediction-market-oracle-research/SKILL.md +64 -0
  297. package/prisma/skills/prediction-market-risk-review/SKILL.md +61 -0
  298. package/prisma/skills/prisma-patterns/SKILL.md +401 -0
  299. package/prisma/skills/product-capability/SKILL.md +142 -0
  300. package/prisma/skills/product-lens/SKILL.md +93 -0
  301. package/prisma/skills/production-audit/SKILL.md +207 -0
  302. package/prisma/skills/production-scheduling/SKILL.md +238 -0
  303. package/prisma/skills/project-flow-ops/SKILL.md +112 -0
  304. package/prisma/skills/prompt-optimizer/SKILL.md +398 -0
  305. package/prisma/skills/python-patterns/SKILL.md +751 -0
  306. package/prisma/skills/python-testing/SKILL.md +817 -0
  307. package/prisma/skills/pytorch-patterns/SKILL.md +397 -0
  308. package/prisma/skills/quality-nonconformance/SKILL.md +260 -0
  309. package/prisma/skills/quarkus-patterns/SKILL.md +723 -0
  310. package/prisma/skills/quarkus-security/SKILL.md +468 -0
  311. package/prisma/skills/quarkus-tdd/SKILL.md +812 -0
  312. package/prisma/skills/quarkus-verification/SKILL.md +480 -0
  313. package/prisma/skills/ralphinho-rfc-pipeline/SKILL.md +68 -0
  314. package/prisma/skills/react-native-patterns/SKILL.md +326 -0
  315. package/prisma/skills/react-patterns/SKILL.md +342 -0
  316. package/prisma/skills/react-performance/SKILL.md +575 -0
  317. package/prisma/skills/react-testing/SKILL.md +424 -0
  318. package/prisma/skills/recsys-pipeline-architect/SKILL.md +115 -0
  319. package/prisma/skills/recursive-decision-ledger/SKILL.md +80 -0
  320. package/prisma/skills/redis-patterns/SKILL.md +404 -0
  321. package/prisma/skills/regex-vs-llm-structured-text/SKILL.md +221 -0
  322. package/prisma/skills/remotion-video-creation/SKILL.md +43 -0
  323. package/prisma/skills/remotion-video-creation/rules/3d.md +86 -0
  324. package/prisma/skills/remotion-video-creation/rules/animations.md +29 -0
  325. package/prisma/skills/remotion-video-creation/rules/assets/charts-bar-chart.tsx +173 -0
  326. package/prisma/skills/remotion-video-creation/rules/assets/text-animations-typewriter.tsx +100 -0
  327. package/prisma/skills/remotion-video-creation/rules/assets/text-animations-word-highlight.tsx +108 -0
  328. package/prisma/skills/remotion-video-creation/rules/assets.md +78 -0
  329. package/prisma/skills/remotion-video-creation/rules/audio.md +172 -0
  330. package/prisma/skills/remotion-video-creation/rules/calculate-metadata.md +104 -0
  331. package/prisma/skills/remotion-video-creation/rules/can-decode.md +75 -0
  332. package/prisma/skills/remotion-video-creation/rules/charts.md +58 -0
  333. package/prisma/skills/remotion-video-creation/rules/compositions.md +146 -0
  334. package/prisma/skills/remotion-video-creation/rules/display-captions.md +126 -0
  335. package/prisma/skills/remotion-video-creation/rules/extract-frames.md +229 -0
  336. package/prisma/skills/remotion-video-creation/rules/fonts.md +152 -0
  337. package/prisma/skills/remotion-video-creation/rules/get-audio-duration.md +58 -0
  338. package/prisma/skills/remotion-video-creation/rules/get-video-dimensions.md +68 -0
  339. package/prisma/skills/remotion-video-creation/rules/get-video-duration.md +58 -0
  340. package/prisma/skills/remotion-video-creation/rules/gifs.md +138 -0
  341. package/prisma/skills/remotion-video-creation/rules/images.md +130 -0
  342. package/prisma/skills/remotion-video-creation/rules/import-srt-captions.md +67 -0
  343. package/prisma/skills/remotion-video-creation/rules/lottie.md +67 -0
  344. package/prisma/skills/remotion-video-creation/rules/measuring-dom-nodes.md +34 -0
  345. package/prisma/skills/remotion-video-creation/rules/measuring-text.md +143 -0
  346. package/prisma/skills/remotion-video-creation/rules/sequencing.md +106 -0
  347. package/prisma/skills/remotion-video-creation/rules/tailwind.md +11 -0
  348. package/prisma/skills/remotion-video-creation/rules/text-animations.md +20 -0
  349. package/prisma/skills/remotion-video-creation/rules/timing.md +179 -0
  350. package/prisma/skills/remotion-video-creation/rules/transcribe-captions.md +19 -0
  351. package/prisma/skills/remotion-video-creation/rules/transitions.md +122 -0
  352. package/prisma/skills/remotion-video-creation/rules/trimming.md +52 -0
  353. package/prisma/skills/remotion-video-creation/rules/videos.md +171 -0
  354. package/prisma/skills/repo-scan/SKILL.md +79 -0
  355. package/prisma/skills/research-ops/SKILL.md +113 -0
  356. package/prisma/skills/returns-reverse-logistics/SKILL.md +240 -0
  357. package/prisma/skills/rules-distill/SKILL.md +265 -0
  358. package/prisma/skills/rules-distill/scripts/scan-rules.sh +58 -0
  359. package/prisma/skills/rules-distill/scripts/scan-skills.sh +129 -0
  360. package/prisma/skills/rust-patterns/SKILL.md +500 -0
  361. package/prisma/skills/rust-testing/SKILL.md +501 -0
  362. package/prisma/skills/safety-guard/SKILL.md +76 -0
  363. package/prisma/skills/santa-method/SKILL.md +307 -0
  364. package/prisma/skills/scientific-db-pubmed-database/SKILL.md +176 -0
  365. package/prisma/skills/scientific-db-uspto-database/SKILL.md +178 -0
  366. package/prisma/skills/scientific-pkg-gget/SKILL.md +167 -0
  367. package/prisma/skills/scientific-thinking-literature-review/SKILL.md +193 -0
  368. package/prisma/skills/scientific-thinking-scholar-evaluation/SKILL.md +161 -0
  369. package/prisma/skills/search-first/SKILL.md +183 -0
  370. package/prisma/skills/security-bounty-hunter/SKILL.md +100 -0
  371. package/prisma/skills/security-review/SKILL.md +504 -0
  372. package/prisma/skills/security-review/cloud-infrastructure-security.md +361 -0
  373. package/prisma/skills/security-scan/SKILL.md +166 -0
  374. package/prisma/skills/seo/SKILL.md +155 -0
  375. package/prisma/skills/skill-comply/SKILL.md +59 -0
  376. package/prisma/skills/skill-comply/fixtures/compliant_trace.jsonl +5 -0
  377. package/prisma/skills/skill-comply/fixtures/noncompliant_trace.jsonl +3 -0
  378. package/prisma/skills/skill-comply/fixtures/tdd_spec.yaml +44 -0
  379. package/prisma/skills/skill-comply/prompts/classifier.md +24 -0
  380. package/prisma/skills/skill-comply/prompts/scenario_generator.md +62 -0
  381. package/prisma/skills/skill-comply/prompts/spec_generator.md +42 -0
  382. package/prisma/skills/skill-comply/pyproject.toml +15 -0
  383. package/prisma/skills/skill-comply/scripts/__init__.py +0 -0
  384. package/prisma/skills/skill-comply/scripts/classifier.py +85 -0
  385. package/prisma/skills/skill-comply/scripts/grader.py +124 -0
  386. package/prisma/skills/skill-comply/scripts/parser.py +107 -0
  387. package/prisma/skills/skill-comply/scripts/report.py +170 -0
  388. package/prisma/skills/skill-comply/scripts/run.py +127 -0
  389. package/prisma/skills/skill-comply/scripts/runner.py +194 -0
  390. package/prisma/skills/skill-comply/scripts/scenario_generator.py +70 -0
  391. package/prisma/skills/skill-comply/scripts/spec_generator.py +72 -0
  392. package/prisma/skills/skill-comply/scripts/utils.py +13 -0
  393. package/prisma/skills/skill-comply/tests/test_grader.py +197 -0
  394. package/prisma/skills/skill-comply/tests/test_parser.py +90 -0
  395. package/prisma/skills/skill-comply/tests/test_runner.py +172 -0
  396. package/prisma/skills/skill-scout/SKILL.md +141 -0
  397. package/prisma/skills/skill-stocktake/SKILL.md +195 -0
  398. package/prisma/skills/skill-stocktake/scripts/quick-diff.sh +87 -0
  399. package/prisma/skills/skill-stocktake/scripts/save-results.sh +56 -0
  400. package/prisma/skills/skill-stocktake/scripts/scan.sh +170 -0
  401. package/prisma/skills/social-graph-ranker/SKILL.md +155 -0
  402. package/prisma/skills/social-publisher/SKILL.md +130 -0
  403. package/prisma/skills/springboot-patterns/SKILL.md +315 -0
  404. package/prisma/skills/springboot-security/SKILL.md +273 -0
  405. package/prisma/skills/springboot-tdd/SKILL.md +159 -0
  406. package/prisma/skills/springboot-verification/SKILL.md +232 -0
  407. package/prisma/skills/strategic-compact/SKILL.md +136 -0
  408. package/prisma/skills/swift-actor-persistence/SKILL.md +144 -0
  409. package/prisma/skills/swift-concurrency-6-2/SKILL.md +216 -0
  410. package/prisma/skills/swift-protocol-di-testing/SKILL.md +191 -0
  411. package/prisma/skills/swiftui-patterns/SKILL.md +259 -0
  412. package/prisma/skills/taste/SKILL.md +264 -0
  413. package/prisma/skills/taste/references/genre-taxonomy.md +87 -0
  414. package/prisma/skills/tdd-workflow/SKILL.md +583 -0
  415. package/prisma/skills/team-agent-orchestration/SKILL.md +111 -0
  416. package/prisma/skills/team-builder/SKILL.md +169 -0
  417. package/prisma/skills/terminal-ops/SKILL.md +110 -0
  418. package/prisma/skills/tinystruct-patterns/SKILL.md +279 -0
  419. package/prisma/skills/tinystruct-patterns/references/architecture.md +90 -0
  420. package/prisma/skills/tinystruct-patterns/references/data-handling.md +60 -0
  421. package/prisma/skills/tinystruct-patterns/references/database.md +99 -0
  422. package/prisma/skills/tinystruct-patterns/references/routing.md +64 -0
  423. package/prisma/skills/tinystruct-patterns/references/system-usage.md +97 -0
  424. package/prisma/skills/tinystruct-patterns/references/testing.md +72 -0
  425. package/prisma/skills/token-budget-advisor/SKILL.md +134 -0
  426. package/prisma/skills/ui-demo/SKILL.md +466 -0
  427. package/prisma/skills/ui-to-vue/SKILL.md +135 -0
  428. package/prisma/skills/uncloud/SKILL.md +344 -0
  429. package/prisma/skills/unified-notifications-ops/SKILL.md +188 -0
  430. package/prisma/skills/verification-loop/SKILL.md +127 -0
  431. package/prisma/skills/video-editing/SKILL.md +311 -0
  432. package/prisma/skills/videodb/SKILL.md +375 -0
  433. package/prisma/skills/videodb/reference/api-reference.md +550 -0
  434. package/prisma/skills/videodb/reference/capture-reference.md +407 -0
  435. package/prisma/skills/videodb/reference/capture.md +101 -0
  436. package/prisma/skills/videodb/reference/editor.md +443 -0
  437. package/prisma/skills/videodb/reference/generative.md +331 -0
  438. package/prisma/skills/videodb/reference/rtstream-reference.md +564 -0
  439. package/prisma/skills/videodb/reference/rtstream.md +65 -0
  440. package/prisma/skills/videodb/reference/search.md +230 -0
  441. package/prisma/skills/videodb/reference/streaming.md +406 -0
  442. package/prisma/skills/videodb/reference/use-cases.md +118 -0
  443. package/prisma/skills/videodb/scripts/ws_listener.py +282 -0
  444. package/prisma/skills/visa-doc-translate/README.md +86 -0
  445. package/prisma/skills/visa-doc-translate/SKILL.md +117 -0
  446. package/prisma/skills/vite-patterns/SKILL.md +450 -0
  447. package/prisma/skills/vue-patterns/SKILL.md +471 -0
  448. package/prisma/skills/windows-desktop-e2e/SKILL.md +888 -0
  449. package/prisma/skills/workspace-surface-audit/SKILL.md +126 -0
  450. package/prisma/skills/x-api/SKILL.md +235 -0
  451. package/run.mjs +0 -10
@@ -0,0 +1,514 @@
1
+ ---
2
+ name: fastapi-patterns
3
+ description: 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
+ metadata:
5
+ origin: ECC
6
+ ---
7
+
8
+ # FastAPI Patterns
9
+
10
+ Modern, production-grade FastAPI development: project layout, Pydantic v2 schemas, dependency injection, async patterns, auth, transactional service methods, and testing.
11
+
12
+ ## Project Structure
13
+
14
+ ```text
15
+ my_app/
16
+ |-- app/
17
+ | |-- main.py # App factory, lifespan, middleware
18
+ | |-- config.py # Settings via pydantic-settings
19
+ | |-- dependencies.py # Shared FastAPI dependencies
20
+ | |-- database.py # SQLAlchemy engine + session
21
+ | |-- routers/
22
+ | | `-- users.py
23
+ | |-- models/ # SQLAlchemy ORM models
24
+ | | `-- user.py
25
+ | |-- schemas/ # Pydantic request/response schemas
26
+ | | `-- user.py
27
+ | `-- services/ # Business logic layer
28
+ | `-- user_service.py
29
+ |-- tests/
30
+ | |-- conftest.py
31
+ | `-- test_users.py
32
+ |-- pyproject.toml
33
+ `-- .env
34
+ ```
35
+
36
+ ---
37
+
38
+ ## App Factory and Lifespan
39
+
40
+ ```python
41
+ # app/main.py
42
+ from contextlib import asynccontextmanager
43
+ from fastapi import FastAPI
44
+ from fastapi.middleware.cors import CORSMiddleware
45
+
46
+ from app.config import settings
47
+ from app.database import engine, Base
48
+ from app.routers import users
49
+
50
+
51
+ @asynccontextmanager
52
+ async def lifespan(app: FastAPI):
53
+ # Automatically create tables on startup for ease of use in dev/demo environments.
54
+ # For strict production applications, manage schemas via Alembic migrations instead.
55
+ async with engine.begin() as conn:
56
+ await conn.run_sync(Base.metadata.create_all)
57
+ yield
58
+ # Shutdown: close pooled resources.
59
+ await engine.dispose()
60
+
61
+
62
+ def create_app() -> FastAPI:
63
+ app = FastAPI(
64
+ title=settings.app_name,
65
+ version=settings.app_version,
66
+ lifespan=lifespan,
67
+ )
68
+
69
+ app.add_middleware(
70
+ CORSMiddleware,
71
+ allow_origins=settings.allowed_origins,
72
+ allow_credentials=settings.allow_credentials,
73
+ allow_methods=settings.allowed_methods,
74
+ allow_headers=settings.allowed_headers,
75
+ )
76
+
77
+ app.include_router(users.router, prefix="/users", tags=["users"])
78
+
79
+ return app
80
+
81
+
82
+ app = create_app()
83
+ ```
84
+
85
+ ---
86
+
87
+ ## Configuration with pydantic-settings
88
+
89
+ ```python
90
+ # app/config.py
91
+ from pydantic_settings import BaseSettings, SettingsConfigDict
92
+
93
+
94
+ class Settings(BaseSettings):
95
+ model_config = SettingsConfigDict(env_file=".env", env_file_encoding="utf-8")
96
+
97
+ app_name: str = "My App"
98
+ app_version: str = "0.1.0"
99
+ debug: bool = False
100
+
101
+ database_url: str
102
+ secret_key: str
103
+ algorithm: str = "HS256"
104
+ access_token_expire_minutes: int = 30
105
+
106
+ # Pydantic-settings v2 safely evaluates mutable list literals directly
107
+ allowed_origins: list[str] = ["http://localhost:3000"]
108
+ allowed_methods: list[str] = ["GET", "POST", "PATCH", "DELETE", "OPTIONS"]
109
+ allowed_headers: list[str] = ["Authorization", "Content-Type"]
110
+ allow_credentials: bool = True
111
+
112
+
113
+ settings = Settings()
114
+ ```
115
+
116
+ ---
117
+
118
+ ## Pydantic Schemas (v2)
119
+
120
+ ```python
121
+ # app/schemas/user.py
122
+ from datetime import datetime
123
+ from pydantic import BaseModel, EmailStr, Field, model_validator
124
+
125
+
126
+ class UserBase(BaseModel):
127
+ email: EmailStr
128
+ username: str = Field(min_length=3, max_length=50)
129
+
130
+
131
+ class UserCreate(UserBase):
132
+ password: str = Field(min_length=8)
133
+ password_confirm: str
134
+
135
+ @model_validator(mode="after")
136
+ def passwords_match(self) -> "UserCreate":
137
+ if self.password != self.password_confirm:
138
+ raise ValueError("Passwords do not match")
139
+ return self
140
+
141
+
142
+ class UserUpdate(BaseModel):
143
+ username: str | None = Field(default=None, min_length=3, max_length=50)
144
+ email: EmailStr | None = None
145
+
146
+
147
+ class UserResponse(UserBase):
148
+ id: int
149
+ is_active: bool
150
+ created_at: datetime
151
+
152
+ model_config = {"from_attributes": True}
153
+
154
+
155
+ class UserListResponse(BaseModel):
156
+ total: int
157
+ items: list[UserResponse]
158
+ ```
159
+
160
+ ---
161
+
162
+ ## Dependency Injection
163
+
164
+ ```python
165
+ # app/dependencies.py
166
+ from typing import Annotated, AsyncGenerator
167
+ from fastapi import Depends, HTTPException, status
168
+ from fastapi.security import OAuth2PasswordBearer
169
+ from jose import JWTError, jwt
170
+ from sqlalchemy.ext.asyncio import AsyncSession
171
+
172
+ from app.config import settings
173
+ from app.database import AsyncSessionLocal
174
+ from app.models.user import User
175
+
176
+ oauth2_scheme = OAuth2PasswordBearer(tokenUrl="/users/token")
177
+
178
+
179
+ async def get_db() -> AsyncGenerator[AsyncSession, None]:
180
+ async with AsyncSessionLocal() as session:
181
+ try:
182
+ yield session
183
+ except Exception:
184
+ await session.rollback()
185
+ raise
186
+
187
+
188
+ async def get_current_user(
189
+ token: Annotated[str, Depends(oauth2_scheme)],
190
+ db: Annotated[AsyncSession, Depends(get_db)],
191
+ ) -> User:
192
+ credentials_exception = HTTPException(
193
+ status_code=status.HTTP_401_UNAUTHORIZED,
194
+ detail="Could not validate credentials",
195
+ headers={"WWW-Authenticate": "Bearer"},
196
+ )
197
+ try:
198
+ payload = jwt.decode(token, settings.secret_key, algorithms=[settings.algorithm])
199
+ subject = payload.get("sub")
200
+ if subject is None:
201
+ raise credentials_exception
202
+ user_id = int(subject)
203
+ except (JWTError, TypeError, ValueError):
204
+ raise credentials_exception
205
+
206
+ user = await db.get(User, user_id)
207
+ if user is None:
208
+ raise credentials_exception
209
+ return user
210
+
211
+
212
+ async def get_current_active_user(
213
+ current_user: Annotated[User, Depends(get_current_user)],
214
+ ) -> User:
215
+ if not current_user.is_active:
216
+ raise HTTPException(status_code=status.HTTP_403_FORBIDDEN, detail="Inactive user")
217
+ return current_user
218
+
219
+
220
+ DbDep = Annotated[AsyncSession, Depends(get_db)]
221
+ CurrentUserDep = Annotated[User, Depends(get_current_user)]
222
+ ActiveUserDep = Annotated[User, Depends(get_current_active_user)]
223
+ ```
224
+
225
+ ---
226
+
227
+ ## Router and Endpoint Design
228
+
229
+ ```python
230
+ # app/routers/users.py
231
+ from typing import Annotated
232
+ from fastapi import APIRouter, HTTPException, Query, status
233
+ from fastapi.security import OAuth2PasswordRequestForm
234
+
235
+ from app.dependencies import ActiveUserDep, DbDep
236
+ from app.schemas.user import UserCreate, UserResponse, UserUpdate, UserListResponse
237
+ from app.services.user_service import DuplicateUserError, UserService
238
+
239
+ router = APIRouter()
240
+
241
+
242
+ @router.post("/", response_model=UserResponse, status_code=status.HTTP_201_CREATED)
243
+ async def create_user(payload: UserCreate, db: DbDep) -> UserResponse:
244
+ service = UserService(db)
245
+ try:
246
+ return await service.create(payload)
247
+ except DuplicateUserError:
248
+ raise HTTPException(status_code=400, detail="Email already registered")
249
+
250
+
251
+ @router.get("/me", response_model=UserResponse)
252
+ async def get_me(current_user: ActiveUserDep) -> UserResponse:
253
+ return current_user
254
+
255
+
256
+ @router.get("/", response_model=UserListResponse)
257
+ async def list_users(
258
+ db: DbDep,
259
+ current_user: ActiveUserDep,
260
+ skip: Annotated[int, Query(ge=0)] = 0,
261
+ limit: Annotated[int, Query(ge=1, le=100)] = 20,
262
+ ) -> UserListResponse:
263
+ service = UserService(db)
264
+ users, total = await service.list(skip=skip, limit=limit)
265
+ return UserListResponse(total=total, items=users)
266
+
267
+
268
+ @router.patch("/{user_id}", response_model=UserResponse)
269
+ async def update_user(
270
+ user_id: int,
271
+ payload: UserUpdate,
272
+ db: DbDep,
273
+ current_user: ActiveUserDep,
274
+ ) -> UserResponse:
275
+ if current_user.id != user_id:
276
+ raise HTTPException(status_code=403, detail="Not authorized")
277
+ service = UserService(db)
278
+ try:
279
+ user = await service.update(user_id, payload)
280
+ except DuplicateUserError:
281
+ raise HTTPException(status_code=400, detail="Email already registered")
282
+ if user is None:
283
+ raise HTTPException(status_code=404, detail="User not found")
284
+ return user
285
+
286
+
287
+ @router.post("/token")
288
+ async def login(
289
+ form_data: Annotated[OAuth2PasswordRequestForm, Depends()],
290
+ db: DbDep,
291
+ ) -> dict[str, str]:
292
+ service = UserService(db)
293
+ token = await service.authenticate(form_data.username, form_data.password)
294
+ if token is None:
295
+ raise HTTPException(
296
+ status_code=status.HTTP_401_UNAUTHORIZED,
297
+ detail="Incorrect username or password",
298
+ headers={"WWW-Authenticate": "Bearer"},
299
+ )
300
+ return {"access_token": token, "token_type": "bearer"}
301
+ ```
302
+
303
+ ---
304
+
305
+ ## Service Layer
306
+
307
+ ```python
308
+ # app/services/user_service.py
309
+ from datetime import datetime, timedelta, timezone
310
+
311
+ from jose import jwt
312
+ from passlib.context import CryptContext
313
+ from sqlalchemy import func, select
314
+ from sqlalchemy.exc import IntegrityError
315
+ from sqlalchemy.ext.asyncio import AsyncSession
316
+
317
+ from app.config import settings
318
+ from app.models.user import User
319
+ from app.schemas.user import UserCreate, UserUpdate
320
+
321
+ pwd_context = CryptContext(schemes=["bcrypt"], deprecated="auto")
322
+
323
+
324
+ class DuplicateUserError(Exception):
325
+ """Raised when a unique user field conflicts with an existing row."""
326
+
327
+
328
+ class UserService:
329
+ def __init__(self, db: AsyncSession) -> None:
330
+ self.db = db
331
+
332
+ async def get_by_email(self, email: str) -> User | None:
333
+ result = await self.db.execute(select(User).where(User.email == email))
334
+ return result.scalar_one_or_none()
335
+
336
+ async def create(self, payload: UserCreate) -> User:
337
+ user = User(
338
+ email=payload.email,
339
+ username=payload.username,
340
+ hashed_password=pwd_context.hash(payload.password),
341
+ )
342
+ self.db.add(user)
343
+ try:
344
+ # Rely on atomic DB constraints rather than race-prone application-level prechecks
345
+ await self.db.commit()
346
+ except IntegrityError as exc:
347
+ await self.db.rollback()
348
+ raise DuplicateUserError from exc
349
+ await self.db.refresh(user)
350
+ return user
351
+
352
+ async def list(self, skip: int = 0, limit: int = 20) -> tuple[list[User], int]:
353
+ total_result = await self.db.execute(select(func.count(User.id)))
354
+ total = total_result.scalar_one()
355
+ # Enforce explicit deterministic ordering to ensure reliable pagination
356
+ result = await self.db.execute(
357
+ select(User).order_by(User.id).offset(skip).limit(limit)
358
+ )
359
+ return list(result.scalars()), total
360
+
361
+ async def update(self, user_id: int, payload: UserUpdate) -> User | None:
362
+ user = await self.db.get(User, user_id)
363
+ if user is None:
364
+ return None
365
+ for field, value in payload.model_dump(exclude_unset=True).items():
366
+ setattr(user, field, value)
367
+ try:
368
+ await self.db.commit()
369
+ except IntegrityError as exc:
370
+ await self.db.rollback()
371
+ raise DuplicateUserError from exc
372
+ await self.db.refresh(user)
373
+ return user
374
+
375
+ async def authenticate(self, email: str, password: str) -> str | None:
376
+ user = await self.get_by_email(email)
377
+ if user is None or not pwd_context.verify(password, user.hashed_password):
378
+ return None
379
+ expire = datetime.now(timezone.utc) + timedelta(
380
+ minutes=settings.access_token_expire_minutes
381
+ )
382
+ return jwt.encode(
383
+ {"sub": str(user.id), "exp": expire},
384
+ settings.secret_key,
385
+ algorithm=settings.algorithm,
386
+ )
387
+ ```
388
+
389
+ > **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.
390
+
391
+ ---
392
+
393
+ ## Testing with httpx and pytest
394
+
395
+ ```python
396
+ # tests/conftest.py
397
+ import pytest_asyncio
398
+ from httpx import ASGITransport, AsyncClient
399
+ from sqlalchemy.ext.asyncio import AsyncSession, async_sessionmaker, create_async_engine
400
+
401
+ from app.database import Base
402
+ from app.dependencies import get_db
403
+ from app.main import create_app
404
+
405
+ TEST_DATABASE_URL = "sqlite+aiosqlite:///:memory:"
406
+
407
+ engine = create_async_engine(TEST_DATABASE_URL)
408
+ TestingSessionLocal = async_sessionmaker(engine, expire_on_commit=False)
409
+
410
+
411
+ @pytest_asyncio.fixture(autouse=True)
412
+ async def setup_db():
413
+ async with engine.begin() as conn:
414
+ await conn.run_sync(Base.metadata.create_all)
415
+ yield
416
+ async with engine.begin() as conn:
417
+ await conn.run_sync(Base.metadata.drop_all)
418
+
419
+
420
+ @pytest_asyncio.fixture
421
+ async def db_session():
422
+ async with TestingSessionLocal() as session:
423
+ yield session
424
+ await session.rollback()
425
+
426
+
427
+ @pytest_asyncio.fixture
428
+ async def client(db_session: AsyncSession):
429
+ app = create_app()
430
+
431
+ async def override_get_db():
432
+ yield db_session
433
+
434
+ app.dependency_overrides[get_db] = override_get_db
435
+
436
+ async with AsyncClient(
437
+ transport=ASGITransport(app=app), base_url="http://test"
438
+ ) as ac:
439
+ yield ac
440
+
441
+
442
+ @pytest_asyncio.fixture
443
+ async def registered_user(client: AsyncClient) -> dict:
444
+ resp = await client.post("/users/", json={
445
+ "email": "test@example.com",
446
+ "username": "testuser",
447
+ "password": "securepass1",
448
+ "password_confirm": "securepass1",
449
+ })
450
+ assert resp.status_code == 201
451
+ return resp.json()
452
+
453
+
454
+ @pytest_asyncio.fixture
455
+ async def auth_token(client: AsyncClient, registered_user: dict) -> str:
456
+ resp = await client.post("/users/token", data={
457
+ "username": "test@example.com",
458
+ "password": "securepass1",
459
+ })
460
+ assert resp.status_code == 200
461
+ return resp.json()["access_token"]
462
+
463
+
464
+ @pytest_asyncio.fixture
465
+ async def auth_client(client: AsyncClient, auth_token: str) -> AsyncClient:
466
+ client.headers.update({"Authorization": f"Bearer {auth_token}"})
467
+ return client
468
+ ```
469
+
470
+ ---
471
+
472
+ ## Anti-Patterns
473
+
474
+ ```python
475
+ # Bad: business logic inside route handlers.
476
+ @router.post("/users/")
477
+ async def create_user(payload: UserCreate, db: DbDep):
478
+ hashed = bcrypt.hash(payload.password)
479
+ user = User(email=payload.email, hashed_password=hashed)
480
+ db.add(user)
481
+ await db.commit()
482
+ return user
483
+
484
+ # Good: thin route, transactional service handling.
485
+ @router.post("/users/", response_model=UserResponse, status_code=201)
486
+ async def create_user(payload: UserCreate, db: DbDep):
487
+ try:
488
+ return await UserService(db).create(payload)
489
+ except DuplicateUserError:
490
+ raise HTTPException(status_code=400, detail="Email already registered")
491
+
492
+
493
+ # Bad: sync DB calls in async routes block the event loop.
494
+ @router.get("/items/")
495
+ async def list_items(db: Session = Depends(get_db)):
496
+ return db.query(Item).all()
497
+
498
+ # Good: use async SQLAlchemy executions.
499
+ @router.get("/items/")
500
+ async def list_items(db: AsyncSession = Depends(get_db)):
501
+ result = await db.execute(select(Item))
502
+ return result.scalars().all()
503
+ ```
504
+
505
+ ---
506
+
507
+ ## Best Practices
508
+
509
+ - Always declare a typed `response_model` to prevent accidental PII/data leaks and output clean OpenAPI schemas.
510
+ - Consolidate standard middleware dependency injections via type-aliasing: `DbDep = Annotated[AsyncSession, Depends(get_db)]`.
511
+ - Wrap database mutation boundaries gracefully within transactions inside your service layer, catching structural database errors directly.
512
+ - Parse JWT parameters defensively, expecting potential string/integer cast mismatches from modern payload variations.
513
+ - Enforce deterministic sorting (e.g., `.order_by(Model.id)`) on all offset/limit paginated endpoints to avoid data skips.
514
+ - Isolate authorization checks from core authentication dependencies to provide precise REST status signals (`401` vs `403`).
@@ -0,0 +1,128 @@
1
+ ---
2
+ name: finance-billing-ops
3
+ description: 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
+ metadata:
5
+ origin: ECC
6
+ ---
7
+
8
+ # Finance Billing Ops
9
+
10
+ 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.
11
+
12
+ 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.
13
+
14
+ ## Skill Stack
15
+
16
+ Pull these ECC-native skills into the workflow when relevant:
17
+
18
+ - `customer-billing-ops` for customer-specific remediation and follow-up
19
+ - `research-ops` when competitor pricing or current market evidence matters
20
+ - `market-research` when the answer should end in a pricing recommendation
21
+ - `github-ops` when the billing truth depends on code, backlog, or release state in sibling repos
22
+ - `verification-loop` when the answer depends on proving checkout, seat handling, or entitlement behavior
23
+
24
+ ## When to Use
25
+
26
+ - user asks for Stripe sales, refunds, MRR, or recent customer activity
27
+ - user asks whether team billing, per-seat billing, or quota stacking is real in code
28
+ - user wants competitor pricing comparisons or pricing-model benchmarks
29
+ - the question mixes revenue facts with product implementation truth
30
+
31
+ ## Guardrails
32
+
33
+ - distinguish live data from saved snapshots
34
+ - separate:
35
+ - revenue fact
36
+ - customer impact
37
+ - code-backed product truth
38
+ - recommendation
39
+ - do not say "per seat" unless the actual entitlement path enforces it
40
+ - do not assume duplicate subscriptions imply duplicate value
41
+
42
+ ## Workflow
43
+
44
+ ### 1. Start from the freshest billing evidence
45
+
46
+ Prefer live billing data. If the data is not live, state the snapshot timestamp explicitly.
47
+
48
+ Normalize the picture:
49
+
50
+ - paid sales
51
+ - active subscriptions
52
+ - failed or incomplete checkouts
53
+ - refunds
54
+ - disputes
55
+ - duplicate subscriptions
56
+
57
+ ### 2. Separate customer incidents from product truth
58
+
59
+ If the question is customer-specific, classify first:
60
+
61
+ - duplicate checkout
62
+ - real team intent
63
+ - broken self-serve controls
64
+ - unmet product value
65
+ - failed payment or incomplete setup
66
+
67
+ Then separate that from the broader product question:
68
+
69
+ - does team billing really exist?
70
+ - are seats actually counted?
71
+ - does checkout quantity change entitlement?
72
+ - does the site overstate current behavior?
73
+
74
+ ### 3. Inspect code-backed billing behavior
75
+
76
+ If the answer depends on implementation truth, inspect the code path:
77
+
78
+ - checkout
79
+ - pricing page
80
+ - entitlement calculation
81
+ - seat or quota handling
82
+ - installation vs user usage logic
83
+ - billing portal or self-serve management support
84
+
85
+ ### 4. End with a decision and product gap
86
+
87
+ Report:
88
+
89
+ - sales snapshot
90
+ - issue diagnosis
91
+ - product truth
92
+ - recommended operator action
93
+ - product or backlog gap
94
+
95
+ ## Output Format
96
+
97
+ ```text
98
+ SNAPSHOT
99
+ - timestamp
100
+ - revenue / subscriptions / anomalies
101
+
102
+ CUSTOMER IMPACT
103
+ - who is affected
104
+ - what happened
105
+
106
+ PRODUCT TRUTH
107
+ - what the code actually does
108
+ - what the website or sales copy claims
109
+
110
+ DECISION
111
+ - refund / preserve / convert / no-op
112
+
113
+ PRODUCT GAP
114
+ - exact follow-up item to build or fix
115
+ ```
116
+
117
+ ## Pitfalls
118
+
119
+ - do not conflate failed attempts with net revenue
120
+ - do not infer team billing from marketing language alone
121
+ - do not compare competitor pricing from memory when current evidence is available
122
+ - do not jump from diagnosis straight to refund without classifying the issue
123
+
124
+ ## Verification
125
+
126
+ - the answer includes a live-data statement or snapshot timestamp
127
+ - product-truth claims are code-backed
128
+ - customer-impact and broader pricing/product conclusions are separated cleanly