agim-cli 1.2.147 → 1.2.149
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.
- package/CHANGELOG.md +158 -0
- package/dist/core/skills/builtin/ECC_LICENSE +21 -0
- package/dist/core/skills/builtin/ECC_NOTICE.md +28 -0
- package/dist/core/skills/builtin/accessibility/SKILL.md +146 -0
- package/dist/core/skills/builtin/agent-eval/SKILL.md +145 -0
- package/dist/core/skills/builtin/agent-harness-construction/SKILL.md +73 -0
- package/dist/core/skills/builtin/agent-introspection-debugging/SKILL.md +153 -0
- package/dist/core/skills/builtin/agentic-engineering/SKILL.md +63 -0
- package/dist/core/skills/builtin/ai-first-engineering/SKILL.md +51 -0
- package/dist/core/skills/builtin/ai-regression-testing/SKILL.md +385 -0
- package/dist/core/skills/builtin/android-clean-architecture/SKILL.md +339 -0
- package/dist/core/skills/builtin/angular-developer/SKILL.md +154 -0
- package/dist/core/skills/builtin/angular-developer/references/angular-animations.md +160 -0
- package/dist/core/skills/builtin/angular-developer/references/angular-aria.md +410 -0
- package/dist/core/skills/builtin/angular-developer/references/cli.md +86 -0
- package/dist/core/skills/builtin/angular-developer/references/component-harnesses.md +59 -0
- package/dist/core/skills/builtin/angular-developer/references/component-styling.md +91 -0
- package/dist/core/skills/builtin/angular-developer/references/components.md +117 -0
- package/dist/core/skills/builtin/angular-developer/references/creating-services.md +97 -0
- package/dist/core/skills/builtin/angular-developer/references/data-resolvers.md +69 -0
- package/dist/core/skills/builtin/angular-developer/references/define-routes.md +67 -0
- package/dist/core/skills/builtin/angular-developer/references/defining-providers.md +72 -0
- package/dist/core/skills/builtin/angular-developer/references/di-fundamentals.md +120 -0
- package/dist/core/skills/builtin/angular-developer/references/e2e-testing.md +56 -0
- package/dist/core/skills/builtin/angular-developer/references/effects.md +83 -0
- package/dist/core/skills/builtin/angular-developer/references/hierarchical-injectors.md +43 -0
- package/dist/core/skills/builtin/angular-developer/references/host-elements.md +80 -0
- package/dist/core/skills/builtin/angular-developer/references/injection-context.md +63 -0
- package/dist/core/skills/builtin/angular-developer/references/inputs.md +101 -0
- package/dist/core/skills/builtin/angular-developer/references/linked-signal.md +59 -0
- package/dist/core/skills/builtin/angular-developer/references/loading-strategies.md +61 -0
- package/dist/core/skills/builtin/angular-developer/references/mcp.md +108 -0
- package/dist/core/skills/builtin/angular-developer/references/navigate-to-routes.md +69 -0
- package/dist/core/skills/builtin/angular-developer/references/outputs.md +86 -0
- package/dist/core/skills/builtin/angular-developer/references/reactive-forms.md +122 -0
- package/dist/core/skills/builtin/angular-developer/references/rendering-strategies.md +44 -0
- package/dist/core/skills/builtin/angular-developer/references/resource.md +77 -0
- package/dist/core/skills/builtin/angular-developer/references/route-animations.md +56 -0
- package/dist/core/skills/builtin/angular-developer/references/route-guards.md +52 -0
- package/dist/core/skills/builtin/angular-developer/references/router-lifecycle.md +45 -0
- package/dist/core/skills/builtin/angular-developer/references/router-testing.md +87 -0
- package/dist/core/skills/builtin/angular-developer/references/show-routes-with-outlets.md +68 -0
- package/dist/core/skills/builtin/angular-developer/references/signal-forms.md +795 -0
- package/dist/core/skills/builtin/angular-developer/references/signals-overview.md +94 -0
- package/dist/core/skills/builtin/angular-developer/references/tailwind-css.md +69 -0
- package/dist/core/skills/builtin/angular-developer/references/template-driven-forms.md +114 -0
- package/dist/core/skills/builtin/angular-developer/references/testing-fundamentals.md +65 -0
- package/dist/core/skills/builtin/api-connector-builder/SKILL.md +120 -0
- package/dist/core/skills/builtin/api-design/SKILL.md +523 -0
- package/dist/core/skills/builtin/architecture-decision-records/SKILL.md +179 -0
- package/dist/core/skills/builtin/article-writing/SKILL.md +79 -0
- package/dist/core/skills/builtin/automation-audit-ops/SKILL.md +142 -0
- package/dist/core/skills/builtin/autonomous-agent-harness/SKILL.md +273 -0
- package/dist/core/skills/builtin/autonomous-loops/SKILL.md +610 -0
- package/dist/core/skills/builtin/backend-patterns/SKILL.md +561 -0
- package/dist/core/skills/builtin/benchmark/SKILL.md +93 -0
- package/dist/core/skills/builtin/benchmark-optimization-loop/SKILL.md +69 -0
- package/dist/core/skills/builtin/blueprint/SKILL.md +105 -0
- package/dist/core/skills/builtin/browser-qa/SKILL.md +87 -0
- package/dist/core/skills/builtin/bun-runtime/SKILL.md +84 -0
- package/dist/core/skills/builtin/cisco-ios-patterns/SKILL.md +163 -0
- package/dist/core/skills/builtin/claude-devfleet/SKILL.md +111 -0
- package/dist/core/skills/builtin/click-path-audit/SKILL.md +244 -0
- package/dist/core/skills/builtin/clickhouse-io/SKILL.md +439 -0
- package/dist/core/skills/builtin/code-tour/SKILL.md +236 -0
- package/dist/core/skills/builtin/codebase-onboarding/SKILL.md +233 -0
- package/dist/core/skills/builtin/codehealth-mcp/SKILL.md +166 -0
- package/dist/core/skills/builtin/coding-standards/SKILL.md +550 -0
- package/dist/core/skills/builtin/compose-multiplatform-patterns/SKILL.md +299 -0
- package/dist/core/skills/builtin/config-gc/SKILL.md +119 -0
- package/dist/core/skills/builtin/content-engine/SKILL.md +131 -0
- package/dist/core/skills/builtin/content-hash-cache-pattern/SKILL.md +161 -0
- package/dist/core/skills/builtin/context-budget/SKILL.md +135 -0
- package/dist/core/skills/builtin/continuous-agent-loop/SKILL.md +45 -0
- package/dist/core/skills/builtin/continuous-learning/SKILL.md +131 -0
- package/dist/core/skills/builtin/continuous-learning/config.json +18 -0
- package/dist/core/skills/builtin/continuous-learning/evaluate-session.sh +69 -0
- package/dist/core/skills/builtin/continuous-learning-v2/SKILL.md +360 -0
- package/dist/core/skills/builtin/continuous-learning-v2/agents/observer-loop.sh +335 -0
- package/dist/core/skills/builtin/continuous-learning-v2/agents/observer.md +198 -0
- package/dist/core/skills/builtin/continuous-learning-v2/agents/session-guardian.sh +150 -0
- package/dist/core/skills/builtin/continuous-learning-v2/agents/start-observer.sh +248 -0
- package/dist/core/skills/builtin/continuous-learning-v2/config.json +8 -0
- package/dist/core/skills/builtin/continuous-learning-v2/hooks/observe.sh +498 -0
- package/dist/core/skills/builtin/continuous-learning-v2/scripts/detect-project.sh +322 -0
- package/dist/core/skills/builtin/continuous-learning-v2/scripts/instinct-cli.py +1914 -0
- package/dist/core/skills/builtin/continuous-learning-v2/scripts/lib/homunculus-dir.sh +31 -0
- package/dist/core/skills/builtin/continuous-learning-v2/scripts/migrate-homunculus.sh +62 -0
- package/dist/core/skills/builtin/continuous-learning-v2/scripts/test_parse_instinct.py +1045 -0
- package/dist/core/skills/builtin/cost-aware-llm-pipeline/SKILL.md +183 -0
- package/dist/core/skills/builtin/cost-tracking/SKILL.md +147 -0
- package/dist/core/skills/builtin/council/SKILL.md +203 -0
- package/dist/core/skills/builtin/cpp-coding-standards/SKILL.md +723 -0
- package/dist/core/skills/builtin/cpp-testing/SKILL.md +324 -0
- package/dist/core/skills/builtin/crosspost/SKILL.md +111 -0
- package/dist/core/skills/builtin/csharp-testing/SKILL.md +321 -0
- package/dist/core/skills/builtin/customs-trade-compliance/SKILL.md +263 -0
- package/dist/core/skills/builtin/dart-flutter-patterns/SKILL.md +563 -0
- package/dist/core/skills/builtin/dashboard-builder/SKILL.md +108 -0
- package/dist/core/skills/builtin/data-scraper-agent/SKILL.md +764 -0
- package/dist/core/skills/builtin/data-throughput-accelerator/SKILL.md +72 -0
- package/dist/core/skills/builtin/database-migrations/SKILL.md +429 -0
- package/dist/core/skills/builtin/deep-research/SKILL.md +159 -0
- package/dist/core/skills/builtin/defi-amm-security/SKILL.md +166 -0
- package/dist/core/skills/builtin/deployment-patterns/SKILL.md +427 -0
- package/dist/core/skills/builtin/design-system/SKILL.md +82 -0
- package/dist/core/skills/builtin/django-celery/SKILL.md +457 -0
- package/dist/core/skills/builtin/django-patterns/SKILL.md +734 -0
- package/dist/core/skills/builtin/django-security/SKILL.md +593 -0
- package/dist/core/skills/builtin/django-tdd/SKILL.md +729 -0
- package/dist/core/skills/builtin/django-verification/SKILL.md +469 -0
- package/dist/core/skills/builtin/dmux-workflows/SKILL.md +191 -0
- package/dist/core/skills/builtin/docker-patterns/SKILL.md +364 -0
- package/dist/core/skills/builtin/documentation-lookup/SKILL.md +90 -0
- package/dist/core/skills/builtin/dotnet-patterns/SKILL.md +321 -0
- package/dist/core/skills/builtin/dynamic-workflow-mode/SKILL.md +123 -0
- package/dist/core/skills/builtin/e2e-testing/SKILL.md +326 -0
- package/dist/core/skills/builtin/email-ops/SKILL.md +121 -0
- package/dist/core/skills/builtin/energy-procurement/SKILL.md +228 -0
- package/dist/core/skills/builtin/enterprise-agent-ops/SKILL.md +50 -0
- package/dist/core/skills/builtin/error-handling/SKILL.md +376 -0
- package/dist/core/skills/builtin/eval-harness/SKILL.md +270 -0
- package/dist/core/skills/builtin/evm-token-decimals/SKILL.md +130 -0
- package/dist/core/skills/builtin/exa-search/SKILL.md +107 -0
- package/dist/core/skills/builtin/fal-ai-media/SKILL.md +288 -0
- package/dist/core/skills/builtin/fastapi-patterns/SKILL.md +513 -0
- package/dist/core/skills/builtin/finance-billing-ops/SKILL.md +127 -0
- package/dist/core/skills/builtin/flox-environments/SKILL.md +496 -0
- package/dist/core/skills/builtin/flutter-dart-code-review/SKILL.md +435 -0
- package/dist/core/skills/builtin/foundation-models-on-device/SKILL.md +243 -0
- package/dist/core/skills/builtin/frontend-a11y/SKILL.md +445 -0
- package/dist/core/skills/builtin/frontend-design-direction/SKILL.md +92 -0
- package/dist/core/skills/builtin/frontend-patterns/SKILL.md +656 -0
- package/dist/core/skills/builtin/frontend-slides/SKILL.md +184 -0
- package/dist/core/skills/builtin/frontend-slides/STYLE_PRESETS.md +330 -0
- package/dist/core/skills/builtin/frontend-slides/animation-patterns.md +122 -0
- package/dist/core/skills/builtin/frontend-slides/html-template.md +419 -0
- package/dist/core/skills/builtin/frontend-slides/scripts/export-pdf.sh +418 -0
- package/dist/core/skills/builtin/frontend-slides/scripts/extract-pptx.py +96 -0
- package/dist/core/skills/builtin/frontend-slides/viewport-base.css +153 -0
- package/dist/core/skills/builtin/fsharp-testing/SKILL.md +280 -0
- package/dist/core/skills/builtin/gan-style-harness/SKILL.md +278 -0
- package/dist/core/skills/builtin/gateguard/SKILL.md +132 -0
- package/dist/core/skills/builtin/git-workflow/SKILL.md +715 -0
- package/dist/core/skills/builtin/github-ops/SKILL.md +144 -0
- package/dist/core/skills/builtin/golang-patterns/SKILL.md +674 -0
- package/dist/core/skills/builtin/golang-testing/SKILL.md +720 -0
- package/dist/core/skills/builtin/healthcare-cdss-patterns/SKILL.md +245 -0
- package/dist/core/skills/builtin/healthcare-emr-patterns/SKILL.md +159 -0
- package/dist/core/skills/builtin/healthcare-eval-harness/SKILL.md +207 -0
- package/dist/core/skills/builtin/healthcare-phi-compliance/SKILL.md +145 -0
- package/dist/core/skills/builtin/hermes-imports/SKILL.md +88 -0
- package/dist/core/skills/builtin/hexagonal-architecture/SKILL.md +276 -0
- package/dist/core/skills/builtin/hipaa-compliance/SKILL.md +78 -0
- package/dist/core/skills/builtin/hookify-rules/SKILL.md +128 -0
- package/dist/core/skills/builtin/inherit-legacy-style/SKILL.md +156 -0
- package/dist/core/skills/builtin/intent-driven-development/SKILL.md +360 -0
- package/dist/core/skills/builtin/inventory-demand-planning/SKILL.md +247 -0
- package/dist/core/skills/builtin/ios-icon-gen/SKILL.md +157 -0
- package/dist/core/skills/builtin/ios-icon-gen/scripts/generate_icons.swift +258 -0
- package/dist/core/skills/builtin/ios-icon-gen/scripts/iconify_gen.sh +235 -0
- package/dist/core/skills/builtin/iterative-retrieval/SKILL.md +211 -0
- package/dist/core/skills/builtin/java-coding-standards/SKILL.md +383 -0
- package/dist/core/skills/builtin/jira-integration/SKILL.md +302 -0
- package/dist/core/skills/builtin/jpa-patterns/SKILL.md +151 -0
- package/dist/core/skills/builtin/knowledge-ops/SKILL.md +154 -0
- package/dist/core/skills/builtin/kotlin-coroutines-flows/SKILL.md +284 -0
- package/dist/core/skills/builtin/kotlin-exposed-patterns/SKILL.md +719 -0
- package/dist/core/skills/builtin/kotlin-ktor-patterns/SKILL.md +689 -0
- package/dist/core/skills/builtin/kotlin-patterns/SKILL.md +711 -0
- package/dist/core/skills/builtin/kotlin-testing/SKILL.md +824 -0
- package/dist/core/skills/builtin/kubernetes-patterns/SKILL.md +755 -0
- package/dist/core/skills/builtin/laravel-patterns/SKILL.md +415 -0
- package/dist/core/skills/builtin/laravel-plugin-discovery/SKILL.md +229 -0
- package/dist/core/skills/builtin/laravel-security/SKILL.md +947 -0
- package/dist/core/skills/builtin/laravel-tdd/SKILL.md +674 -0
- package/dist/core/skills/builtin/laravel-verification/SKILL.md +179 -0
- package/dist/core/skills/builtin/latency-critical-systems/SKILL.md +73 -0
- package/dist/core/skills/builtin/lead-intelligence/SKILL.md +321 -0
- package/dist/core/skills/builtin/lead-intelligence/agents/enrichment-agent.md +85 -0
- package/dist/core/skills/builtin/lead-intelligence/agents/mutual-mapper.md +75 -0
- package/dist/core/skills/builtin/lead-intelligence/agents/outreach-drafter.md +98 -0
- package/dist/core/skills/builtin/lead-intelligence/agents/signal-scorer.md +60 -0
- package/dist/core/skills/builtin/liquid-glass-design/SKILL.md +279 -0
- package/dist/core/skills/builtin/llm-trading-agent-security/SKILL.md +146 -0
- package/dist/core/skills/builtin/logistics-exception-management/SKILL.md +222 -0
- package/dist/core/skills/builtin/make-interfaces-feel-better/SKILL.md +151 -0
- package/dist/core/skills/builtin/market-research/SKILL.md +75 -0
- package/dist/core/skills/builtin/marketing-campaign/SKILL.md +113 -0
- package/dist/core/skills/builtin/mcp-server-patterns/SKILL.md +69 -0
- package/dist/core/skills/builtin/messages-ops/SKILL.md +104 -0
- package/dist/core/skills/builtin/mle-workflow/SKILL.md +346 -0
- package/dist/core/skills/builtin/motion-advanced/SKILL.md +596 -0
- package/dist/core/skills/builtin/motion-foundations/SKILL.md +299 -0
- package/dist/core/skills/builtin/motion-patterns/SKILL.md +434 -0
- package/dist/core/skills/builtin/motion-ui/SKILL.md +575 -0
- package/dist/core/skills/builtin/mysql-patterns/SKILL.md +412 -0
- package/dist/core/skills/builtin/nanoclaw-repl/SKILL.md +33 -0
- package/dist/core/skills/builtin/nestjs-patterns/SKILL.md +230 -0
- package/dist/core/skills/builtin/netmiko-ssh-automation/SKILL.md +173 -0
- package/dist/core/skills/builtin/network-bgp-diagnostics/SKILL.md +167 -0
- package/dist/core/skills/builtin/network-config-validation/SKILL.md +210 -0
- package/dist/core/skills/builtin/network-interface-health/SKILL.md +152 -0
- package/dist/core/skills/builtin/nextjs-turbopack/SKILL.md +57 -0
- package/dist/core/skills/builtin/nodejs-keccak256/SKILL.md +102 -0
- package/dist/core/skills/builtin/nutrient-document-processing/SKILL.md +167 -0
- package/dist/core/skills/builtin/nuxt4-patterns/SKILL.md +100 -0
- package/dist/core/skills/builtin/openclaw-persona-forge/SKILL.md +288 -0
- package/dist/core/skills/builtin/openclaw-persona-forge/gacha.py +224 -0
- package/dist/core/skills/builtin/openclaw-persona-forge/gacha.sh +5 -0
- package/dist/core/skills/builtin/openclaw-persona-forge/references/avatar-style.md +124 -0
- package/dist/core/skills/builtin/openclaw-persona-forge/references/boundary-rules.md +53 -0
- package/dist/core/skills/builtin/openclaw-persona-forge/references/error-handling.md +53 -0
- package/dist/core/skills/builtin/openclaw-persona-forge/references/identity-tension.md +48 -0
- package/dist/core/skills/builtin/openclaw-persona-forge/references/naming-system.md +39 -0
- package/dist/core/skills/builtin/openclaw-persona-forge/references/output-template.md +166 -0
- package/dist/core/skills/builtin/opensource-pipeline/SKILL.md +255 -0
- package/dist/core/skills/builtin/orch-add-feature/SKILL.md +44 -0
- package/dist/core/skills/builtin/orch-build-mvp/SKILL.md +48 -0
- package/dist/core/skills/builtin/orch-change-feature/SKILL.md +42 -0
- package/dist/core/skills/builtin/orch-fix-defect/SKILL.md +42 -0
- package/dist/core/skills/builtin/orch-pipeline/SKILL.md +120 -0
- package/dist/core/skills/builtin/orch-refine-code/SKILL.md +43 -0
- package/dist/core/skills/builtin/parallel-execution-optimizer/SKILL.md +72 -0
- package/dist/core/skills/builtin/perl-patterns/SKILL.md +504 -0
- package/dist/core/skills/builtin/perl-security/SKILL.md +503 -0
- package/dist/core/skills/builtin/perl-testing/SKILL.md +475 -0
- package/dist/core/skills/builtin/plan-orchestrate/SKILL.md +262 -0
- package/dist/core/skills/builtin/plankton-code-quality/SKILL.md +236 -0
- package/dist/core/skills/builtin/postgres-patterns/SKILL.md +147 -0
- package/dist/core/skills/builtin/prediction-market-oracle-research/SKILL.md +63 -0
- package/dist/core/skills/builtin/prediction-market-risk-review/SKILL.md +60 -0
- package/dist/core/skills/builtin/prisma-patterns/SKILL.md +371 -0
- package/dist/core/skills/builtin/product-capability/SKILL.md +141 -0
- package/dist/core/skills/builtin/product-lens/SKILL.md +92 -0
- package/dist/core/skills/builtin/production-audit/SKILL.md +206 -0
- package/dist/core/skills/builtin/production-scheduling/SKILL.md +238 -0
- package/dist/core/skills/builtin/prompt-optimizer/SKILL.md +398 -0
- package/dist/core/skills/builtin/python-patterns/SKILL.md +750 -0
- package/dist/core/skills/builtin/python-testing/SKILL.md +816 -0
- package/dist/core/skills/builtin/pytorch-patterns/SKILL.md +396 -0
- package/dist/core/skills/builtin/quality-nonconformance/SKILL.md +260 -0
- package/dist/core/skills/builtin/quarkus-patterns/SKILL.md +722 -0
- package/dist/core/skills/builtin/quarkus-security/SKILL.md +467 -0
- package/dist/core/skills/builtin/quarkus-tdd/SKILL.md +811 -0
- package/dist/core/skills/builtin/quarkus-verification/SKILL.md +479 -0
- package/dist/core/skills/builtin/ralphinho-rfc-pipeline/SKILL.md +67 -0
- package/dist/core/skills/builtin/react-patterns/SKILL.md +341 -0
- package/dist/core/skills/builtin/react-performance/SKILL.md +574 -0
- package/dist/core/skills/builtin/react-testing/SKILL.md +423 -0
- package/dist/core/skills/builtin/recsys-pipeline-architect/SKILL.md +114 -0
- package/dist/core/skills/builtin/recursive-decision-ledger/SKILL.md +79 -0
- package/dist/core/skills/builtin/redis-patterns/SKILL.md +403 -0
- package/dist/core/skills/builtin/regex-vs-llm-structured-text/SKILL.md +220 -0
- package/dist/core/skills/builtin/repo-scan/SKILL.md +78 -0
- package/dist/core/skills/builtin/research-ops/SKILL.md +112 -0
- package/dist/core/skills/builtin/returns-reverse-logistics/SKILL.md +240 -0
- package/dist/core/skills/builtin/rules-distill/SKILL.md +264 -0
- package/dist/core/skills/builtin/rules-distill/scripts/scan-rules.sh +58 -0
- package/dist/core/skills/builtin/rules-distill/scripts/scan-skills.sh +129 -0
- package/dist/core/skills/builtin/rust-patterns/SKILL.md +499 -0
- package/dist/core/skills/builtin/rust-testing/SKILL.md +500 -0
- package/dist/core/skills/builtin/safety-guard/SKILL.md +75 -0
- package/dist/core/skills/builtin/santa-method/SKILL.md +306 -0
- package/dist/core/skills/builtin/scientific-db-pubmed-database/SKILL.md +175 -0
- package/dist/core/skills/builtin/scientific-db-uspto-database/SKILL.md +177 -0
- package/dist/core/skills/builtin/scientific-pkg-gget/SKILL.md +166 -0
- package/dist/core/skills/builtin/scientific-thinking-literature-review/SKILL.md +192 -0
- package/dist/core/skills/builtin/scientific-thinking-scholar-evaluation/SKILL.md +160 -0
- package/dist/core/skills/builtin/search-first/SKILL.md +182 -0
- package/dist/core/skills/builtin/security-bounty-hunter/SKILL.md +99 -0
- package/dist/core/skills/builtin/security-review/SKILL.md +503 -0
- package/dist/core/skills/builtin/security-review/cloud-infrastructure-security.md +361 -0
- package/dist/core/skills/builtin/security-scan/SKILL.md +165 -0
- package/dist/core/skills/builtin/seo/SKILL.md +154 -0
- package/dist/core/skills/builtin/skill-comply/SKILL.md +58 -0
- package/dist/core/skills/builtin/skill-comply/fixtures/compliant_trace.jsonl +5 -0
- package/dist/core/skills/builtin/skill-comply/fixtures/noncompliant_trace.jsonl +3 -0
- package/dist/core/skills/builtin/skill-comply/fixtures/tdd_spec.yaml +44 -0
- package/dist/core/skills/builtin/skill-comply/prompts/classifier.md +24 -0
- package/dist/core/skills/builtin/skill-comply/prompts/scenario_generator.md +62 -0
- package/dist/core/skills/builtin/skill-comply/prompts/spec_generator.md +42 -0
- package/dist/core/skills/builtin/skill-comply/pyproject.toml +15 -0
- package/dist/core/skills/builtin/skill-comply/scripts/__init__.py +0 -0
- package/dist/core/skills/builtin/skill-comply/scripts/classifier.py +85 -0
- package/dist/core/skills/builtin/skill-comply/scripts/grader.py +124 -0
- package/dist/core/skills/builtin/skill-comply/scripts/parser.py +107 -0
- package/dist/core/skills/builtin/skill-comply/scripts/report.py +170 -0
- package/dist/core/skills/builtin/skill-comply/scripts/run.py +127 -0
- package/dist/core/skills/builtin/skill-comply/scripts/runner.py +186 -0
- package/dist/core/skills/builtin/skill-comply/scripts/scenario_generator.py +70 -0
- package/dist/core/skills/builtin/skill-comply/scripts/spec_generator.py +72 -0
- package/dist/core/skills/builtin/skill-comply/scripts/utils.py +13 -0
- package/dist/core/skills/builtin/skill-comply/tests/test_grader.py +197 -0
- package/dist/core/skills/builtin/skill-comply/tests/test_parser.py +90 -0
- package/dist/core/skills/builtin/skill-comply/tests/test_runner.py +172 -0
- package/dist/core/skills/builtin/skill-scout/SKILL.md +140 -0
- package/dist/core/skills/builtin/skill-stocktake/SKILL.md +194 -0
- package/dist/core/skills/builtin/skill-stocktake/scripts/quick-diff.sh +87 -0
- package/dist/core/skills/builtin/skill-stocktake/scripts/save-results.sh +56 -0
- package/dist/core/skills/builtin/skill-stocktake/scripts/scan.sh +170 -0
- package/dist/core/skills/builtin/springboot-patterns/SKILL.md +314 -0
- package/dist/core/skills/builtin/springboot-security/SKILL.md +272 -0
- package/dist/core/skills/builtin/springboot-tdd/SKILL.md +158 -0
- package/dist/core/skills/builtin/springboot-verification/SKILL.md +231 -0
- package/dist/core/skills/builtin/strategic-compact/SKILL.md +135 -0
- package/dist/core/skills/builtin/swift-actor-persistence/SKILL.md +143 -0
- package/dist/core/skills/builtin/swift-concurrency-6-2/SKILL.md +216 -0
- package/dist/core/skills/builtin/swift-protocol-di-testing/SKILL.md +190 -0
- package/dist/core/skills/builtin/swiftui-patterns/SKILL.md +259 -0
- package/dist/core/skills/builtin/tdd-workflow/SKILL.md +463 -0
- package/dist/core/skills/builtin/team-agent-orchestration/SKILL.md +110 -0
- package/dist/core/skills/builtin/team-builder/SKILL.md +168 -0
- package/dist/core/skills/builtin/terminal-ops/SKILL.md +109 -0
- package/dist/core/skills/builtin/tinystruct-patterns/SKILL.md +203 -0
- package/dist/core/skills/builtin/tinystruct-patterns/references/architecture.md +90 -0
- package/dist/core/skills/builtin/tinystruct-patterns/references/data-handling.md +60 -0
- package/dist/core/skills/builtin/tinystruct-patterns/references/database.md +99 -0
- package/dist/core/skills/builtin/tinystruct-patterns/references/routing.md +64 -0
- package/dist/core/skills/builtin/tinystruct-patterns/references/system-usage.md +97 -0
- package/dist/core/skills/builtin/tinystruct-patterns/references/testing.md +72 -0
- package/dist/core/skills/builtin/token-budget-advisor/SKILL.md +133 -0
- package/dist/core/skills/builtin/ui-demo/SKILL.md +465 -0
- package/dist/core/skills/builtin/ui-to-vue/SKILL.md +134 -0
- package/dist/core/skills/builtin/uncloud/SKILL.md +343 -0
- package/dist/core/skills/builtin/unified-notifications-ops/SKILL.md +187 -0
- package/dist/core/skills/builtin/verification-loop/SKILL.md +126 -0
- package/dist/core/skills/builtin/video-editing/SKILL.md +310 -0
- package/dist/core/skills/builtin/videodb/SKILL.md +374 -0
- package/dist/core/skills/builtin/videodb/reference/api-reference.md +550 -0
- package/dist/core/skills/builtin/videodb/reference/capture-reference.md +407 -0
- package/dist/core/skills/builtin/videodb/reference/capture.md +101 -0
- package/dist/core/skills/builtin/videodb/reference/editor.md +443 -0
- package/dist/core/skills/builtin/videodb/reference/generative.md +331 -0
- package/dist/core/skills/builtin/videodb/reference/rtstream-reference.md +564 -0
- package/dist/core/skills/builtin/videodb/reference/rtstream.md +65 -0
- package/dist/core/skills/builtin/videodb/reference/search.md +230 -0
- package/dist/core/skills/builtin/videodb/reference/streaming.md +406 -0
- package/dist/core/skills/builtin/videodb/reference/use-cases.md +118 -0
- package/dist/core/skills/builtin/videodb/scripts/ws_listener.py +282 -0
- package/dist/core/skills/builtin/visa-doc-translate/README.md +86 -0
- package/dist/core/skills/builtin/visa-doc-translate/SKILL.md +117 -0
- package/dist/core/skills/builtin/vite-patterns/SKILL.md +449 -0
- package/dist/core/skills/builtin/windows-desktop-e2e/SKILL.md +887 -0
- package/dist/core/skills/builtin/x-api/SKILL.md +234 -0
- package/dist/core/skills/loader.d.ts +23 -12
- package/dist/core/skills/loader.d.ts.map +1 -1
- package/dist/core/skills/loader.js +105 -2
- package/dist/core/skills/loader.js.map +1 -1
- package/package.json +1 -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
|