@heyai-rules/pilo-masterkit 2.1.0 → 3.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.
- package/.agent/agents/PILO_MASTER.md +77 -77
- package/.agent/agents/architect.md +211 -211
- package/.agent/agents/backend-specialist.md +263 -263
- package/.agent/agents/build-error-resolver.md +114 -114
- package/.agent/agents/chief-of-staff.md +151 -151
- package/.agent/agents/code-archaeologist.md +106 -106
- package/.agent/agents/code-reviewer.md +237 -237
- package/.agent/agents/cpp-build-resolver.md +90 -90
- package/.agent/agents/cpp-reviewer.md +72 -72
- package/.agent/agents/csharp-reviewer.md +101 -101
- package/.agent/agents/dart-build-resolver.md +201 -201
- package/.agent/agents/database-architect.md +226 -226
- package/.agent/agents/database-reviewer.md +91 -91
- package/.agent/agents/debugger.md +225 -225
- package/.agent/agents/devops-engineer.md +242 -242
- package/.agent/agents/doc-updater.md +107 -107
- package/.agent/agents/docs-lookup.md +68 -68
- package/.agent/agents/documentation-writer.md +104 -104
- package/.agent/agents/e2e-runner.md +107 -107
- package/.agent/agents/explorer-agent.md +73 -73
- package/.agent/agents/flutter-reviewer.md +243 -243
- package/.agent/agents/frontend-specialist.md +593 -593
- package/.agent/agents/game-developer.md +162 -162
- package/.agent/agents/gan-evaluator.md +209 -209
- package/.agent/agents/gan-generator.md +131 -131
- package/.agent/agents/gan-planner.md +99 -99
- package/.agent/agents/go-build-resolver.md +94 -94
- package/.agent/agents/go-reviewer.md +76 -76
- package/.agent/agents/harness-optimizer.md +35 -35
- package/.agent/agents/healthcare-reviewer.md +83 -83
- package/.agent/agents/java-build-resolver.md +153 -153
- package/.agent/agents/java-reviewer.md +92 -92
- package/.agent/agents/kotlin-build-resolver.md +118 -118
- package/.agent/agents/kotlin-reviewer.md +159 -159
- package/.agent/agents/loop-operator.md +36 -36
- package/.agent/agents/mobile-developer.md +377 -377
- package/.agent/agents/opensource-forker.md +198 -198
- package/.agent/agents/opensource-packager.md +249 -249
- package/.agent/agents/opensource-sanitizer.md +188 -188
- package/.agent/agents/orchestrator.md +416 -416
- package/.agent/agents/penetration-tester.md +188 -188
- package/.agent/agents/performance-optimizer.md +446 -446
- package/.agent/agents/personas/athena-agent/agent.json +10 -10
- package/.agent/agents/personas/athena-agent/athena-backend-logic-architecture-profile.md +3 -3
- package/.agent/agents/personas/athena-agent/context-files/agents.md +1 -1
- package/.agent/agents/personas/athena-agent/context-files/identity.md +1 -1
- package/.agent/agents/personas/athena-agent/context-files/soul.md +1 -1
- package/.agent/agents/personas/athena-agent/context-files/user-predefined.md +1 -1
- package/.agent/agents/personas/athena-agent/user-context-files/system/bootstrap.md +1 -1
- package/.agent/agents/personas/athena-agent/user-context-files/system/user.md +1 -1
- package/.agent/agents/personas/da-vinci-agent/agent.json +10 -10
- package/.agent/agents/personas/da-vinci-agent/context-files/agents.md +1 -1
- package/.agent/agents/personas/da-vinci-agent/context-files/identity.md +1 -1
- package/.agent/agents/personas/da-vinci-agent/context-files/soul.md +1 -1
- package/.agent/agents/personas/da-vinci-agent/context-files/user-predefined.md +1 -1
- package/.agent/agents/personas/da-vinci-agent/da-vinci-frontend-ui-ux-design-profile.md +3 -3
- package/.agent/agents/personas/da-vinci-agent/user-context-files/system/bootstrap.md +1 -1
- package/.agent/agents/personas/da-vinci-agent/user-context-files/system/user.md +1 -1
- package/.agent/agents/personas/duong-tang-agent/agent.json +10 -10
- package/.agent/agents/personas/duong-tang-agent/context-files/agents.md +1 -1
- package/.agent/agents/personas/duong-tang-agent/context-files/identity.md +1 -1
- package/.agent/agents/personas/duong-tang-agent/context-files/soul.md +1 -1
- package/.agent/agents/personas/duong-tang-agent/context-files/user-predefined.md +1 -1
- package/.agent/agents/personas/duong-tang-agent/tang-monk-quality-testing-documentation-profile.md +3 -3
- package/.agent/agents/personas/duong-tang-agent/user-context-files/system/bootstrap.md +1 -1
- package/.agent/agents/personas/duong-tang-agent/user-context-files/system/user.md +1 -1
- package/.agent/agents/personas/gia-cat-luong-agent/agent.json +10 -10
- package/.agent/agents/personas/gia-cat-luong-agent/context-files/agents.md +1 -1
- package/.agent/agents/personas/gia-cat-luong-agent/context-files/identity.md +1 -1
- package/.agent/agents/personas/gia-cat-luong-agent/context-files/soul.md +1 -1
- package/.agent/agents/personas/gia-cat-luong-agent/context-files/user-predefined.md +1 -1
- package/.agent/agents/personas/gia-cat-luong-agent/kongming-research-strategy-analysis-profile.md +3 -3
- package/.agent/agents/personas/gia-cat-luong-agent/user-context-files/system/bootstrap.md +1 -1
- package/.agent/agents/personas/gia-cat-luong-agent/user-context-files/system/user.md +1 -1
- package/.agent/agents/personas/mihata-agent/agent.json +10 -10
- package/.agent/agents/personas/mihata-agent/context-files/agents.md +1 -1
- package/.agent/agents/personas/mihata-agent/context-files/identity.md +1 -1
- package/.agent/agents/personas/mihata-agent/context-files/soul.md +1 -1
- package/.agent/agents/personas/mihata-agent/context-files/user-predefined.md +1 -1
- package/.agent/agents/personas/mihata-agent/mihata-multi-agent-orchestration-profile.md +3 -3
- package/.agent/agents/personas/mihata-agent/user-context-files/system/bootstrap.md +1 -1
- package/.agent/agents/personas/mihata-agent/user-context-files/system/user.md +1 -1
- package/.agent/agents/personas/tesla-agent/agent.json +10 -10
- package/.agent/agents/personas/tesla-agent/context-files/agents.md +1 -1
- package/.agent/agents/personas/tesla-agent/context-files/identity.md +1 -1
- package/.agent/agents/personas/tesla-agent/context-files/soul.md +1 -1
- package/.agent/agents/personas/tesla-agent/context-files/user-predefined.md +1 -1
- package/.agent/agents/personas/tesla-agent/tesla-fullstack-system-optimization-profile.md +3 -3
- package/.agent/agents/personas/tesla-agent/user-context-files/system/bootstrap.md +1 -1
- package/.agent/agents/personas/tesla-agent/user-context-files/system/user.md +1 -1
- package/.agent/agents/personas/tu-ma-y-agent/agent.json +10 -10
- package/.agent/agents/personas/tu-ma-y-agent/context-files/agents.md +1 -1
- package/.agent/agents/personas/tu-ma-y-agent/context-files/identity.md +1 -1
- package/.agent/agents/personas/tu-ma-y-agent/context-files/soul.md +1 -1
- package/.agent/agents/personas/tu-ma-y-agent/context-files/user-predefined.md +1 -1
- package/.agent/agents/personas/tu-ma-y-agent/simayi-feasibility-risk-control-profile.md +3 -3
- package/.agent/agents/personas/tu-ma-y-agent/user-context-files/system/bootstrap.md +1 -1
- package/.agent/agents/personas/tu-ma-y-agent/user-context-files/system/user.md +1 -1
- package/.agent/agents/personas/venti-agent/agent.json +10 -10
- package/.agent/agents/personas/venti-agent/context-files/agents.md +1 -1
- package/.agent/agents/personas/venti-agent/context-files/identity.md +1 -1
- package/.agent/agents/personas/venti-agent/context-files/soul.md +1 -1
- package/.agent/agents/personas/venti-agent/context-files/user-predefined.md +1 -1
- package/.agent/agents/personas/venti-agent/user-context-files/system/bootstrap.md +1 -1
- package/.agent/agents/personas/venti-agent/user-context-files/system/user.md +1 -1
- package/.agent/agents/personas/venti-agent/venti-learning-communication-mentoring-profile.md +3 -3
- package/.agent/agents/planner.md +212 -212
- package/.agent/agents/product-manager.md +112 -112
- package/.agent/agents/product-owner.md +95 -95
- package/.agent/agents/project-planner.md +406 -406
- package/.agent/agents/python-reviewer.md +98 -98
- package/.agent/agents/pytorch-build-resolver.md +120 -120
- package/.agent/agents/qa-automation-engineer.md +103 -103
- package/.agent/agents/refactor-cleaner.md +85 -85
- package/.agent/agents/rust-build-resolver.md +148 -148
- package/.agent/agents/rust-reviewer.md +94 -94
- package/.agent/agents/security-auditor.md +170 -170
- package/.agent/agents/security-reviewer.md +108 -108
- package/.agent/agents/seo-specialist.md +111 -111
- package/.agent/agents/tdd-guide.md +91 -91
- package/.agent/agents/test-engineer.md +158 -158
- package/.agent/agents/typescript-reviewer.md +112 -112
- package/.agent/contexts/dev.md +20 -20
- package/.agent/contexts/research.md +26 -26
- package/.agent/contexts/review.md +22 -22
- package/.agent/hooks/hooks.json +395 -395
- package/.agent/hooks/readme.md +222 -222
- package/.agent/mcp-configs/mcp-servers.json +181 -181
- package/.agent/rules/ARCHITECTURAL_BLUEPRINTS.md +62 -62
- package/.agent/rules/CODE_CRAFTSMANSHIP.md +69 -69
- package/.agent/rules/CORE_RULES.md +72 -72
- package/.agent/rules/PROJECT_MAP.md +58 -58
- package/.agent/rules/QUALITY_ASSURANCE.md +54 -54
- package/.agent/rules/SECURITY_ARMOR.md +44 -44
- package/.agent/rules/VERSION_ORCHESTRATION.md +64 -64
- package/.agent/rules/WORKFLOW_ORCHESTRATION.md +55 -55
- package/.agent/rules/common/agents.md +50 -50
- package/.agent/rules/common/code-review.md +124 -124
- package/.agent/rules/common/coding-style.md +48 -48
- package/.agent/rules/common/development-workflow.md +44 -44
- package/.agent/rules/common/git-workflow.md +24 -24
- package/.agent/rules/common/hooks.md +30 -30
- package/.agent/rules/common/patterns.md +31 -31
- package/.agent/rules/common/performance.md +55 -55
- package/.agent/rules/common/security.md +29 -29
- package/.agent/rules/common/testing.md +29 -29
- package/.agent/rules/cpp/coding-style.md +44 -44
- package/.agent/rules/cpp/hooks.md +39 -39
- package/.agent/rules/cpp/patterns.md +51 -51
- package/.agent/rules/cpp/security.md +51 -51
- package/.agent/rules/cpp/testing.md +44 -44
- package/.agent/rules/csharp/coding-style.md +72 -72
- package/.agent/rules/csharp/hooks.md +25 -25
- package/.agent/rules/csharp/patterns.md +50 -50
- package/.agent/rules/csharp/security.md +58 -58
- package/.agent/rules/csharp/testing.md +46 -46
- package/.agent/rules/dart/coding-style.md +159 -159
- package/.agent/rules/dart/hooks.md +66 -66
- package/.agent/rules/dart/patterns.md +261 -261
- package/.agent/rules/dart/security.md +135 -135
- package/.agent/rules/dart/testing.md +215 -215
- package/.agent/rules/golang/coding-style.md +32 -32
- package/.agent/rules/golang/hooks.md +17 -17
- package/.agent/rules/golang/patterns.md +45 -45
- package/.agent/rules/golang/security.md +34 -34
- package/.agent/rules/golang/testing.md +31 -31
- package/.agent/rules/java/coding-style.md +114 -114
- package/.agent/rules/java/hooks.md +18 -18
- package/.agent/rules/java/patterns.md +146 -146
- package/.agent/rules/java/security.md +100 -100
- package/.agent/rules/java/testing.md +131 -131
- package/.agent/rules/kotlin/coding-style.md +86 -86
- package/.agent/rules/kotlin/hooks.md +17 -17
- package/.agent/rules/kotlin/patterns.md +146 -146
- package/.agent/rules/kotlin/security.md +82 -82
- package/.agent/rules/kotlin/testing.md +128 -128
- package/.agent/rules/perl/coding-style.md +46 -46
- package/.agent/rules/perl/hooks.md +22 -22
- package/.agent/rules/perl/patterns.md +76 -76
- package/.agent/rules/perl/security.md +69 -69
- package/.agent/rules/perl/testing.md +54 -54
- package/.agent/rules/php/coding-style.md +40 -40
- package/.agent/rules/php/hooks.md +24 -24
- package/.agent/rules/php/patterns.md +33 -33
- package/.agent/rules/php/security.md +37 -37
- package/.agent/rules/php/testing.md +39 -39
- package/.agent/rules/python/coding-style.md +42 -42
- package/.agent/rules/python/hooks.md +19 -19
- package/.agent/rules/python/patterns.md +39 -39
- package/.agent/rules/python/security.md +30 -30
- package/.agent/rules/python/testing.md +38 -38
- package/.agent/rules/readme.md +111 -111
- package/.agent/rules/rust/coding-style.md +151 -151
- package/.agent/rules/rust/hooks.md +16 -16
- package/.agent/rules/rust/patterns.md +168 -168
- package/.agent/rules/rust/security.md +141 -141
- package/.agent/rules/rust/testing.md +154 -154
- package/.agent/rules/swift/coding-style.md +47 -47
- package/.agent/rules/swift/hooks.md +20 -20
- package/.agent/rules/swift/patterns.md +66 -66
- package/.agent/rules/swift/security.md +33 -33
- package/.agent/rules/swift/testing.md +45 -45
- package/.agent/rules/typescript/coding-style.md +199 -199
- package/.agent/rules/typescript/hooks.md +22 -22
- package/.agent/rules/typescript/patterns.md +52 -52
- package/.agent/rules/typescript/security.md +28 -28
- package/.agent/rules/typescript/testing.md +18 -18
- package/.agent/rules/web/coding-style.md +96 -96
- package/.agent/rules/web/design-quality.md +63 -63
- package/.agent/rules/web/hooks.md +120 -120
- package/.agent/rules/web/patterns.md +79 -79
- package/.agent/rules/web/performance.md +64 -64
- package/.agent/rules/web/security.md +57 -57
- package/.agent/rules/web/testing.md +55 -55
- package/.agent/rules/zh/agents.md +50 -50
- package/.agent/rules/zh/code-review.md +124 -124
- package/.agent/rules/zh/coding-style.md +48 -48
- package/.agent/rules/zh/development-workflow.md +44 -44
- package/.agent/rules/zh/git-workflow.md +24 -24
- package/.agent/rules/zh/hooks.md +30 -30
- package/.agent/rules/zh/patterns.md +31 -31
- package/.agent/rules/zh/performance.md +55 -55
- package/.agent/rules/zh/readme.md +108 -108
- package/.agent/rules/zh/security.md +29 -29
- package/.agent/rules/zh/testing.md +29 -29
- package/.agent/scripts/auto_preview.py +148 -148
- package/.agent/scripts/checklist.py +217 -217
- package/.agent/scripts/session_manager.py +120 -120
- package/.agent/scripts/verify_all.py +327 -327
- package/.agent/skills/agent-eval/SKILL.md +145 -145
- package/.agent/skills/agent-harness-construction/SKILL.md +73 -73
- package/.agent/skills/agent-payment-x402/SKILL.md +178 -178
- package/.agent/skills/agentic-engineering/SKILL.md +63 -63
- package/.agent/skills/ai-first-engineering/SKILL.md +51 -51
- package/.agent/skills/ai-regression-testing/SKILL.md +385 -385
- package/.agent/skills/android-clean-architecture/SKILL.md +339 -339
- package/.agent/skills/api-design/SKILL.md +523 -523
- package/.agent/skills/api-patterns/SKILL.md +81 -81
- package/.agent/skills/api-patterns/api-style.md +42 -42
- package/.agent/skills/api-patterns/auth.md +24 -24
- package/.agent/skills/api-patterns/documentation.md +26 -26
- package/.agent/skills/api-patterns/graphql.md +41 -41
- package/.agent/skills/api-patterns/rate-limiting.md +31 -31
- package/.agent/skills/api-patterns/response.md +37 -37
- package/.agent/skills/api-patterns/rest.md +40 -40
- package/.agent/skills/api-patterns/scripts/api_validator.py +211 -211
- package/.agent/skills/api-patterns/security-testing.md +122 -122
- package/.agent/skills/api-patterns/trpc.md +41 -41
- package/.agent/skills/api-patterns/versioning.md +22 -22
- package/.agent/skills/app-builder/SKILL.md +75 -75
- package/.agent/skills/app-builder/agent-coordination.md +71 -71
- package/.agent/skills/app-builder/feature-building.md +53 -53
- package/.agent/skills/app-builder/project-detection.md +34 -34
- package/.agent/skills/app-builder/scaffolding.md +118 -118
- package/.agent/skills/app-builder/tech-stack.md +41 -41
- package/.agent/skills/app-builder/templates/SKILL.md +39 -39
- package/.agent/skills/app-builder/templates/astro-static/TEMPLATE.md +76 -76
- package/.agent/skills/app-builder/templates/chrome-extension/TEMPLATE.md +92 -92
- package/.agent/skills/app-builder/templates/cli-tool/TEMPLATE.md +88 -88
- package/.agent/skills/app-builder/templates/electron-desktop/TEMPLATE.md +88 -88
- package/.agent/skills/app-builder/templates/express-api/TEMPLATE.md +83 -83
- package/.agent/skills/app-builder/templates/flutter-app/TEMPLATE.md +90 -90
- package/.agent/skills/app-builder/templates/monorepo-turborepo/TEMPLATE.md +90 -90
- package/.agent/skills/app-builder/templates/nextjs-fullstack/TEMPLATE.md +122 -122
- package/.agent/skills/app-builder/templates/nextjs-saas/TEMPLATE.md +122 -122
- package/.agent/skills/app-builder/templates/nextjs-static/TEMPLATE.md +169 -169
- package/.agent/skills/app-builder/templates/nuxt-app/TEMPLATE.md +134 -134
- package/.agent/skills/app-builder/templates/python-fastapi/TEMPLATE.md +83 -83
- package/.agent/skills/app-builder/templates/react-native-app/TEMPLATE.md +119 -119
- package/.agent/skills/architecture/SKILL.md +55 -55
- package/.agent/skills/architecture/context-discovery.md +43 -43
- package/.agent/skills/architecture/examples.md +94 -94
- package/.agent/skills/architecture/pattern-selection.md +68 -68
- package/.agent/skills/architecture/patterns-reference.md +50 -50
- package/.agent/skills/architecture/trade-off-analysis.md +77 -77
- package/.agent/skills/architecture-decision-records/SKILL.md +179 -179
- package/.agent/skills/article-writing/SKILL.md +79 -79
- package/.agent/skills/autonomous-agent-harness/SKILL.md +267 -267
- package/.agent/skills/autonomous-loops/SKILL.md +610 -610
- package/.agent/skills/backend-patterns/SKILL.md +598 -598
- package/.agent/skills/bash-linux/SKILL.md +199 -199
- package/.agent/skills/behavioral-modes/SKILL.md +242 -242
- package/.agent/skills/benchmark/SKILL.md +93 -93
- package/.agent/skills/blueprint/SKILL.md +105 -105
- package/.agent/skills/brainstorming/SKILL.md +163 -163
- package/.agent/skills/brainstorming/dynamic-questioning.md +350 -350
- package/.agent/skills/brand-voice/SKILL.md +97 -97
- package/.agent/skills/brand-voice/references/voice-profile-schema.md +55 -55
- package/.agent/skills/browser-qa/SKILL.md +87 -87
- package/.agent/skills/bun-runtime/SKILL.md +84 -84
- package/.agent/skills/canary-watch/SKILL.md +99 -99
- package/.agent/skills/carrier-relationship-management/SKILL.md +212 -212
- package/.agent/skills/ck/SKILL.md +147 -147
- package/.agent/skills/ck/commands/forget.mjs +44 -44
- package/.agent/skills/ck/commands/info.mjs +24 -24
- package/.agent/skills/ck/commands/init.mjs +143 -143
- package/.agent/skills/ck/commands/list.mjs +40 -40
- package/.agent/skills/ck/commands/migrate.mjs +202 -202
- package/.agent/skills/ck/commands/resume.mjs +36 -36
- package/.agent/skills/ck/commands/save.mjs +210 -210
- package/.agent/skills/ck/commands/shared.mjs +387 -387
- package/.agent/skills/ck/hooks/session-start.mjs +224 -224
- package/.agent/skills/claude-api/SKILL.md +337 -337
- package/.agent/skills/claude-devfleet/SKILL.md +103 -103
- package/.agent/skills/clean-code/SKILL.md +201 -201
- package/.agent/skills/click-path-audit/SKILL.md +244 -244
- package/.agent/skills/clickhouse-io/SKILL.md +439 -439
- package/.agent/skills/code-review-checklist/SKILL.md +109 -109
- package/.agent/skills/codebase-onboarding/SKILL.md +233 -233
- package/.agent/skills/coding-standards/SKILL.md +530 -530
- package/.agent/skills/compose-multiplatform-patterns/SKILL.md +299 -299
- package/.agent/skills/configure-ecc/SKILL.md +367 -367
- package/.agent/skills/connections-optimizer/SKILL.md +189 -189
- package/.agent/skills/content-engine/SKILL.md +131 -131
- package/.agent/skills/content-hash-cache-pattern/SKILL.md +161 -161
- package/.agent/skills/context-budget/SKILL.md +135 -135
- package/.agent/skills/continuous-agent-loop/SKILL.md +45 -45
- package/.agent/skills/continuous-learning/SKILL.md +119 -119
- package/.agent/skills/continuous-learning/config.json +18 -18
- package/.agent/skills/continuous-learning/evaluate-session.sh +69 -69
- package/.agent/skills/continuous-learning-v2/SKILL.md +365 -365
- package/.agent/skills/continuous-learning-v2/agents/observer-loop.sh +271 -271
- package/.agent/skills/continuous-learning-v2/agents/observer.md +198 -198
- package/.agent/skills/continuous-learning-v2/agents/session-guardian.sh +150 -150
- package/.agent/skills/continuous-learning-v2/agents/start-observer.sh +244 -244
- package/.agent/skills/continuous-learning-v2/config.json +8 -8
- package/.agent/skills/continuous-learning-v2/hooks/observe.sh +428 -428
- package/.agent/skills/continuous-learning-v2/scripts/detect-project.sh +228 -228
- package/.agent/skills/continuous-learning-v2/scripts/instinct-cli.py +1426 -1426
- package/.agent/skills/continuous-learning-v2/scripts/test-parse-instinct.py +984 -984
- package/.agent/skills/cost-aware-llm-pipeline/SKILL.md +183 -183
- package/.agent/skills/cpp-coding-standards/SKILL.md +723 -723
- package/.agent/skills/cpp-testing/SKILL.md +324 -324
- package/.agent/skills/crosspost/SKILL.md +111 -111
- package/.agent/skills/csharp-testing/SKILL.md +321 -321
- package/.agent/skills/customer-billing-ops/SKILL.md +140 -140
- package/.agent/skills/customs-trade-compliance/SKILL.md +263 -263
- package/.agent/skills/dart-flutter-patterns/SKILL.md +563 -563
- package/.agent/skills/data-scraper-agent/SKILL.md +764 -764
- package/.agent/skills/database-design/SKILL.md +52 -52
- package/.agent/skills/database-design/database-selection.md +43 -43
- package/.agent/skills/database-design/indexing.md +39 -39
- package/.agent/skills/database-design/migrations.md +48 -48
- package/.agent/skills/database-design/optimization.md +36 -36
- package/.agent/skills/database-design/orm-selection.md +30 -30
- package/.agent/skills/database-design/schema-design.md +56 -56
- package/.agent/skills/database-design/scripts/schema_validator.py +172 -172
- package/.agent/skills/database-migrations/SKILL.md +429 -429
- package/.agent/skills/deep-research/SKILL.md +155 -155
- package/.agent/skills/deployment-patterns/SKILL.md +427 -427
- package/.agent/skills/deployment-procedures/SKILL.md +241 -241
- package/.agent/skills/design-system/SKILL.md +82 -82
- package/.agent/skills/django-patterns/SKILL.md +734 -734
- package/.agent/skills/django-security/SKILL.md +593 -593
- package/.agent/skills/django-tdd/SKILL.md +729 -729
- package/.agent/skills/django-verification/SKILL.md +469 -469
- package/.agent/skills/dmux-workflows/SKILL.md +191 -191
- package/.agent/skills/doc.md +177 -177
- package/.agent/skills/docker-patterns/SKILL.md +364 -364
- package/.agent/skills/documentation-lookup/SKILL.md +90 -90
- package/.agent/skills/documentation-templates/SKILL.md +194 -194
- package/.agent/skills/dotnet-patterns/SKILL.md +321 -321
- package/.agent/skills/e2e-testing/SKILL.md +326 -326
- package/.agent/skills/energy-procurement/SKILL.md +228 -228
- package/.agent/skills/enterprise-agent-ops/SKILL.md +50 -50
- package/.agent/skills/eval-harness/SKILL.md +270 -270
- package/.agent/skills/exa-search/SKILL.md +103 -103
- package/.agent/skills/fal-ai-media/SKILL.md +284 -284
- package/.agent/skills/flutter-dart-code-review/SKILL.md +435 -435
- package/.agent/skills/foundation-models-on-device/SKILL.md +243 -243
- package/.agent/skills/frontend-design/SKILL.md +452 -452
- package/.agent/skills/frontend-design/animation-guide.md +331 -331
- package/.agent/skills/frontend-design/color-system.md +311 -311
- package/.agent/skills/frontend-design/decision-trees.md +418 -418
- package/.agent/skills/frontend-design/motion-graphics.md +306 -306
- package/.agent/skills/frontend-design/scripts/accessibility_checker.py +183 -183
- package/.agent/skills/frontend-design/scripts/ux_audit.py +722 -722
- package/.agent/skills/frontend-design/typography-system.md +345 -345
- package/.agent/skills/frontend-design/ux-psychology.md +1116 -1116
- package/.agent/skills/frontend-design/visual-effects.md +383 -383
- package/.agent/skills/frontend-patterns/SKILL.md +642 -642
- package/.agent/skills/frontend-slides/SKILL.md +184 -184
- package/.agent/skills/frontend-slides/style-presets.md +330 -330
- package/.agent/skills/game-development/2d-games/SKILL.md +119 -119
- package/.agent/skills/game-development/3d-games/SKILL.md +135 -135
- package/.agent/skills/game-development/SKILL.md +167 -167
- package/.agent/skills/game-development/game-art/SKILL.md +185 -185
- package/.agent/skills/game-development/game-audio/SKILL.md +190 -190
- package/.agent/skills/game-development/game-design/SKILL.md +129 -129
- package/.agent/skills/game-development/mobile-games/SKILL.md +108 -108
- package/.agent/skills/game-development/multiplayer/SKILL.md +132 -132
- package/.agent/skills/game-development/pc-games/SKILL.md +144 -144
- package/.agent/skills/game-development/vr-ar/SKILL.md +123 -123
- package/.agent/skills/game-development/web-games/SKILL.md +150 -150
- package/.agent/skills/gan-style-harness/SKILL.md +278 -278
- package/.agent/skills/geo-fundamentals/SKILL.md +156 -156
- package/.agent/skills/geo-fundamentals/scripts/geo_checker.py +289 -289
- package/.agent/skills/git-workflow/SKILL.md +715 -715
- package/.agent/skills/golang-patterns/SKILL.md +674 -674
- package/.agent/skills/golang-testing/SKILL.md +720 -720
- package/.agent/skills/google-workspace-ops/SKILL.md +95 -95
- package/.agent/skills/healthcare-cdss-patterns/SKILL.md +245 -245
- package/.agent/skills/healthcare-emr-patterns/SKILL.md +159 -159
- package/.agent/skills/healthcare-eval-harness/SKILL.md +207 -207
- package/.agent/skills/healthcare-phi-compliance/SKILL.md +145 -145
- package/.agent/skills/hexagonal-architecture/SKILL.md +276 -276
- package/.agent/skills/i18n-localization/SKILL.md +154 -154
- package/.agent/skills/i18n-localization/scripts/i18n_checker.py +241 -241
- package/.agent/skills/intelligent-routing/SKILL.md +335 -335
- package/.agent/skills/inventory-demand-planning/SKILL.md +247 -247
- package/.agent/skills/investor-materials/SKILL.md +96 -96
- package/.agent/skills/investor-outreach/SKILL.md +91 -91
- package/.agent/skills/iterative-retrieval/SKILL.md +211 -211
- package/.agent/skills/java-coding-standards/SKILL.md +147 -147
- package/.agent/skills/jira-integration/SKILL.md +293 -293
- package/.agent/skills/jpa-patterns/SKILL.md +151 -151
- package/.agent/skills/kotlin-coroutines-flows/SKILL.md +284 -284
- package/.agent/skills/kotlin-exposed-patterns/SKILL.md +719 -719
- package/.agent/skills/kotlin-ktor-patterns/SKILL.md +689 -689
- package/.agent/skills/kotlin-patterns/SKILL.md +711 -711
- package/.agent/skills/kotlin-testing/SKILL.md +824 -824
- package/.agent/skills/laravel-patterns/SKILL.md +415 -415
- package/.agent/skills/laravel-plugin-discovery/SKILL.md +229 -229
- package/.agent/skills/laravel-security/SKILL.md +285 -285
- package/.agent/skills/laravel-tdd/SKILL.md +283 -283
- package/.agent/skills/laravel-verification/SKILL.md +179 -179
- package/.agent/skills/lead-intelligence/SKILL.md +321 -321
- package/.agent/skills/lead-intelligence/agents/enrichment-agent.md +85 -85
- package/.agent/skills/lead-intelligence/agents/mutual-mapper.md +75 -75
- package/.agent/skills/lead-intelligence/agents/outreach-drafter.md +98 -98
- package/.agent/skills/lead-intelligence/agents/signal-scorer.md +60 -60
- package/.agent/skills/lint-and-validate/SKILL.md +45 -45
- package/.agent/skills/lint-and-validate/scripts/lint_runner.py +184 -184
- package/.agent/skills/lint-and-validate/scripts/type_coverage.py +173 -173
- package/.agent/skills/liquid-glass-design/SKILL.md +279 -279
- package/.agent/skills/logistics-exception-management/SKILL.md +222 -222
- package/.agent/skills/manim-video/SKILL.md +89 -89
- package/.agent/skills/manim-video/assets/network-graph-scene.py +52 -52
- package/.agent/skills/market-research/SKILL.md +75 -75
- package/.agent/skills/mcp-server-patterns/SKILL.md +67 -67
- package/.agent/skills/mobile-design/SKILL.md +394 -394
- package/.agent/skills/mobile-design/decision-trees.md +516 -516
- package/.agent/skills/mobile-design/mobile-backend.md +491 -491
- package/.agent/skills/mobile-design/mobile-color-system.md +420 -420
- package/.agent/skills/mobile-design/mobile-debugging.md +122 -122
- package/.agent/skills/mobile-design/mobile-design-thinking.md +357 -357
- package/.agent/skills/mobile-design/mobile-navigation.md +458 -458
- package/.agent/skills/mobile-design/mobile-performance.md +767 -767
- package/.agent/skills/mobile-design/mobile-testing.md +356 -356
- package/.agent/skills/mobile-design/mobile-typography.md +433 -433
- package/.agent/skills/mobile-design/platform-android.md +666 -666
- package/.agent/skills/mobile-design/platform-ios.md +561 -561
- package/.agent/skills/mobile-design/scripts/mobile_audit.py +670 -670
- package/.agent/skills/mobile-design/touch-psychology.md +537 -537
- package/.agent/skills/nanoclaw-repl/SKILL.md +33 -33
- package/.agent/skills/nestjs-patterns/SKILL.md +230 -230
- package/.agent/skills/nextjs-react-expert/1-async-eliminating-waterfalls.md +351 -351
- package/.agent/skills/nextjs-react-expert/2-bundle-bundle-size-optimization.md +240 -240
- package/.agent/skills/nextjs-react-expert/3-server-server-side-performance.md +490 -490
- package/.agent/skills/nextjs-react-expert/4-client-client-side-data-fetching.md +264 -264
- package/.agent/skills/nextjs-react-expert/5-rerender-re-render-optimization.md +581 -581
- package/.agent/skills/nextjs-react-expert/6-rendering-rendering-performance.md +432 -432
- package/.agent/skills/nextjs-react-expert/7-js-javascript-performance.md +684 -684
- package/.agent/skills/nextjs-react-expert/8-advanced-advanced-patterns.md +150 -150
- package/.agent/skills/nextjs-react-expert/9-cache-components.md +103 -103
- package/.agent/skills/nextjs-react-expert/SKILL.md +293 -293
- package/.agent/skills/nextjs-react-expert/scripts/convert_rules.py +222 -222
- package/.agent/skills/nextjs-react-expert/scripts/react_performance_checker.py +252 -252
- package/.agent/skills/nextjs-turbopack/SKILL.md +44 -44
- package/.agent/skills/nodejs-best-practices/SKILL.md +333 -333
- package/.agent/skills/nutrient-document-processing/SKILL.md +167 -167
- package/.agent/skills/nuxt4-patterns/SKILL.md +100 -100
- package/.agent/skills/openclaw-persona-forge/SKILL.md +296 -296
- package/.agent/skills/openclaw-persona-forge/gacha.py +224 -224
- package/.agent/skills/openclaw-persona-forge/gacha.sh +5 -5
- package/.agent/skills/openclaw-persona-forge/references/avatar-style.md +124 -124
- package/.agent/skills/openclaw-persona-forge/references/boundary-rules.md +53 -53
- package/.agent/skills/openclaw-persona-forge/references/error-handling.md +53 -53
- package/.agent/skills/openclaw-persona-forge/references/identity-tension.md +48 -48
- package/.agent/skills/openclaw-persona-forge/references/naming-system.md +39 -39
- package/.agent/skills/openclaw-persona-forge/references/output-template.md +166 -166
- package/.agent/skills/opensource-pipeline/SKILL.md +255 -255
- package/.agent/skills/parallel-agents/SKILL.md +175 -175
- package/.agent/skills/performance-profiling/SKILL.md +143 -143
- package/.agent/skills/performance-profiling/scripts/lighthouse_audit.py +76 -76
- package/.agent/skills/perl-patterns/SKILL.md +504 -504
- package/.agent/skills/perl-security/SKILL.md +503 -503
- package/.agent/skills/perl-testing/SKILL.md +475 -475
- package/.agent/skills/plan-writing/SKILL.md +152 -152
- package/.agent/skills/plankton-code-quality/SKILL.md +236 -236
- package/.agent/skills/postgres-patterns/SKILL.md +147 -147
- package/.agent/skills/powershell-windows/SKILL.md +167 -167
- package/.agent/skills/product-lens/SKILL.md +85 -85
- package/.agent/skills/production-scheduling/SKILL.md +238 -238
- package/.agent/skills/project-flow-ops/SKILL.md +111 -111
- package/.agent/skills/project-guidelines-example/SKILL.md +349 -349
- package/.agent/skills/prompt-optimizer/SKILL.md +397 -397
- package/.agent/skills/python-patterns/SKILL.md +750 -750
- package/.agent/skills/python-testing/SKILL.md +816 -816
- package/.agent/skills/pytorch-patterns/SKILL.md +396 -396
- package/.agent/skills/quality-nonconformance/SKILL.md +260 -260
- package/.agent/skills/ralphinho-rfc-pipeline/SKILL.md +67 -67
- package/.agent/skills/red-team-tactics/SKILL.md +199 -199
- package/.agent/skills/regex-vs-llm-structured-text/SKILL.md +220 -220
- package/.agent/skills/remotion-video-creation/SKILL.md +43 -43
- package/.agent/skills/remotion-video-creation/rules/3d.md +86 -86
- package/.agent/skills/remotion-video-creation/rules/animations.md +29 -29
- package/.agent/skills/remotion-video-creation/rules/assets/charts-bar-chart.tsx +173 -173
- package/.agent/skills/remotion-video-creation/rules/assets/text-animations-typewriter.tsx +100 -100
- package/.agent/skills/remotion-video-creation/rules/assets/text-animations-word-highlight.tsx +108 -108
- package/.agent/skills/remotion-video-creation/rules/assets.md +78 -78
- package/.agent/skills/remotion-video-creation/rules/audio.md +172 -172
- package/.agent/skills/remotion-video-creation/rules/calculate-metadata.md +104 -104
- package/.agent/skills/remotion-video-creation/rules/can-decode.md +75 -75
- package/.agent/skills/remotion-video-creation/rules/charts.md +58 -58
- package/.agent/skills/remotion-video-creation/rules/compositions.md +146 -146
- package/.agent/skills/remotion-video-creation/rules/display-captions.md +126 -126
- package/.agent/skills/remotion-video-creation/rules/extract-frames.md +229 -229
- package/.agent/skills/remotion-video-creation/rules/fonts.md +152 -152
- package/.agent/skills/remotion-video-creation/rules/get-audio-duration.md +58 -58
- package/.agent/skills/remotion-video-creation/rules/get-video-dimensions.md +68 -68
- package/.agent/skills/remotion-video-creation/rules/get-video-duration.md +58 -58
- package/.agent/skills/remotion-video-creation/rules/gifs.md +138 -138
- package/.agent/skills/remotion-video-creation/rules/images.md +130 -130
- package/.agent/skills/remotion-video-creation/rules/import-srt-captions.md +67 -67
- package/.agent/skills/remotion-video-creation/rules/lottie.md +67 -67
- package/.agent/skills/remotion-video-creation/rules/measuring-dom-nodes.md +34 -34
- package/.agent/skills/remotion-video-creation/rules/measuring-text.md +143 -143
- package/.agent/skills/remotion-video-creation/rules/sequencing.md +106 -106
- package/.agent/skills/remotion-video-creation/rules/tailwind.md +11 -11
- package/.agent/skills/remotion-video-creation/rules/text-animations.md +20 -20
- package/.agent/skills/remotion-video-creation/rules/timing.md +179 -179
- package/.agent/skills/remotion-video-creation/rules/transcribe-captions.md +19 -19
- package/.agent/skills/remotion-video-creation/rules/transitions.md +122 -122
- package/.agent/skills/remotion-video-creation/rules/trimming.md +52 -52
- package/.agent/skills/remotion-video-creation/rules/videos.md +171 -171
- package/.agent/skills/repo-scan/SKILL.md +63 -63
- package/.agent/skills/returns-reverse-logistics/SKILL.md +240 -240
- package/.agent/skills/rules-distill/SKILL.md +264 -264
- package/.agent/skills/rules-distill/scripts/scan-rules.sh +58 -58
- package/.agent/skills/rules-distill/scripts/scan-skills.sh +129 -129
- package/.agent/skills/rust-patterns/SKILL.md +499 -499
- package/.agent/skills/rust-pro/SKILL.md +175 -175
- package/.agent/skills/rust-testing/SKILL.md +500 -500
- package/.agent/skills/safety-guard/SKILL.md +75 -75
- package/.agent/skills/santa-method/SKILL.md +306 -306
- package/.agent/skills/search-first/SKILL.md +161 -161
- package/.agent/skills/security-review/SKILL.md +495 -495
- package/.agent/skills/security-review/cloud-infrastructure-security.md +361 -361
- package/.agent/skills/security-scan/SKILL.md +165 -165
- package/.agent/skills/seo-fundamentals/SKILL.md +129 -129
- package/.agent/skills/seo-fundamentals/scripts/seo_checker.py +219 -219
- package/.agent/skills/server-management/SKILL.md +161 -161
- package/.agent/skills/skill-comply/SKILL.md +58 -58
- package/.agent/skills/skill-comply/fixtures/compliant-trace.jsonl +5 -5
- package/.agent/skills/skill-comply/fixtures/noncompliant-trace.jsonl +3 -3
- package/.agent/skills/skill-comply/fixtures/tdd-spec.yaml +44 -44
- package/.agent/skills/skill-comply/prompts/classifier.md +24 -24
- package/.agent/skills/skill-comply/prompts/scenario-generator.md +62 -62
- package/.agent/skills/skill-comply/prompts/spec-generator.md +42 -42
- package/.agent/skills/skill-comply/pyproject.toml +15 -15
- package/.agent/skills/skill-comply/scripts/classifier.py +85 -85
- package/.agent/skills/skill-comply/scripts/grader.py +122 -122
- package/.agent/skills/skill-comply/scripts/parser.py +107 -107
- package/.agent/skills/skill-comply/scripts/report.py +170 -170
- package/.agent/skills/skill-comply/scripts/run.py +127 -127
- package/.agent/skills/skill-comply/scripts/runner.py +161 -161
- package/.agent/skills/skill-comply/scripts/scenario-generator.py +70 -70
- package/.agent/skills/skill-comply/scripts/spec-generator.py +72 -72
- package/.agent/skills/skill-comply/scripts/utils.py +13 -13
- package/.agent/skills/skill-comply/tests/test-grader.py +137 -137
- package/.agent/skills/skill-comply/tests/test-parser.py +90 -90
- package/.agent/skills/skill-stocktake/SKILL.md +193 -193
- package/.agent/skills/skill-stocktake/scripts/quick-diff.sh +87 -87
- package/.agent/skills/skill-stocktake/scripts/save-results.sh +56 -56
- package/.agent/skills/skill-stocktake/scripts/scan.sh +170 -170
- package/.agent/skills/social-graph-ranker/SKILL.md +154 -154
- package/.agent/skills/springboot-patterns/SKILL.md +314 -314
- package/.agent/skills/springboot-security/SKILL.md +272 -272
- package/.agent/skills/springboot-tdd/SKILL.md +158 -158
- package/.agent/skills/springboot-verification/SKILL.md +231 -231
- package/.agent/skills/strategic-compact/SKILL.md +131 -131
- package/.agent/skills/strategic-compact/suggest-compact.sh +54 -54
- package/.agent/skills/swift-actor-persistence/SKILL.md +143 -143
- package/.agent/skills/swift-concurrency-6-2/SKILL.md +216 -216
- package/.agent/skills/swift-protocol-di-testing/SKILL.md +190 -190
- package/.agent/skills/swiftui-patterns/SKILL.md +259 -259
- package/.agent/skills/systematic-debugging/SKILL.md +109 -109
- package/.agent/skills/tailwind-patterns/SKILL.md +269 -269
- package/.agent/skills/tdd-workflow/SKILL.md +463 -463
- package/.agent/skills/team-builder/SKILL.md +168 -168
- package/.agent/skills/testing-patterns/SKILL.md +178 -178
- package/.agent/skills/testing-patterns/scripts/test_runner.py +219 -219
- package/.agent/skills/token-budget-advisor/SKILL.md +133 -133
- package/.agent/skills/ui-demo/SKILL.md +465 -465
- package/.agent/skills/ui-ux-pro-max/SKILL.md +292 -292
- package/.agent/skills/ui-ux-pro-max/data/charts.csv +26 -26
- package/.agent/skills/ui-ux-pro-max/data/colors.csv +97 -97
- package/.agent/skills/ui-ux-pro-max/data/icons.csv +101 -101
- package/.agent/skills/ui-ux-pro-max/data/landing.csv +31 -31
- package/.agent/skills/ui-ux-pro-max/data/products.csv +96 -96
- package/.agent/skills/ui-ux-pro-max/data/react-performance.csv +45 -45
- package/.agent/skills/ui-ux-pro-max/data/stacks/astro.csv +54 -54
- package/.agent/skills/ui-ux-pro-max/data/stacks/flutter.csv +53 -53
- package/.agent/skills/ui-ux-pro-max/data/stacks/html-tailwind.csv +56 -56
- package/.agent/skills/ui-ux-pro-max/data/stacks/jetpack-compose.csv +53 -53
- package/.agent/skills/ui-ux-pro-max/data/stacks/nextjs.csv +53 -53
- package/.agent/skills/ui-ux-pro-max/data/stacks/nuxt-ui.csv +51 -51
- package/.agent/skills/ui-ux-pro-max/data/stacks/nuxtjs.csv +59 -59
- package/.agent/skills/ui-ux-pro-max/data/stacks/react-native.csv +52 -52
- package/.agent/skills/ui-ux-pro-max/data/stacks/react.csv +54 -54
- package/.agent/skills/ui-ux-pro-max/data/stacks/shadcn.csv +61 -61
- package/.agent/skills/ui-ux-pro-max/data/stacks/svelte.csv +54 -54
- package/.agent/skills/ui-ux-pro-max/data/stacks/swiftui.csv +51 -51
- package/.agent/skills/ui-ux-pro-max/data/stacks/vue.csv +50 -50
- package/.agent/skills/ui-ux-pro-max/data/styles.csv +68 -68
- package/.agent/skills/ui-ux-pro-max/data/typography.csv +57 -57
- package/.agent/skills/ui-ux-pro-max/data/ui-reasoning.csv +101 -101
- package/.agent/skills/ui-ux-pro-max/data/ux-guidelines.csv +99 -99
- package/.agent/skills/ui-ux-pro-max/data/web-interface.csv +31 -31
- package/.agent/skills/ui-ux-pro-max/scripts/core.py +253 -253
- package/.agent/skills/ui-ux-pro-max/scripts/design_system.py +1067 -1067
- package/.agent/skills/ui-ux-pro-max/scripts/search.py +114 -114
- package/.agent/skills/verification-loop/SKILL.md +126 -126
- package/.agent/skills/video-editing/SKILL.md +310 -310
- package/.agent/skills/videodb/SKILL.md +374 -374
- package/.agent/skills/videodb/reference/api-reference.md +550 -550
- package/.agent/skills/videodb/reference/capture-reference.md +407 -407
- package/.agent/skills/videodb/reference/capture.md +101 -101
- package/.agent/skills/videodb/reference/editor.md +443 -443
- package/.agent/skills/videodb/reference/generative.md +331 -331
- package/.agent/skills/videodb/reference/rtstream-reference.md +564 -564
- package/.agent/skills/videodb/reference/rtstream.md +65 -65
- package/.agent/skills/videodb/reference/search.md +230 -230
- package/.agent/skills/videodb/reference/streaming.md +406 -406
- package/.agent/skills/videodb/reference/use-cases.md +118 -118
- package/.agent/skills/videodb/scripts/ws-listener.py +282 -282
- package/.agent/skills/visa-doc-translate/SKILL.md +117 -117
- package/.agent/skills/visa-doc-translate/readme.md +86 -86
- package/.agent/skills/vulnerability-scanner/SKILL.md +276 -276
- package/.agent/skills/vulnerability-scanner/checklists.md +121 -121
- package/.agent/skills/vulnerability-scanner/scripts/security_scan.py +458 -458
- package/.agent/skills/web-design-guidelines/SKILL.md +57 -57
- package/.agent/skills/webapp-testing/SKILL.md +187 -187
- package/.agent/skills/webapp-testing/scripts/playwright_runner.py +173 -173
- package/.agent/skills/workspace-surface-audit/SKILL.md +125 -125
- package/.agent/skills/x-api/SKILL.md +230 -230
- package/.agent/tasks/lessons.md +40 -40
- package/.agent/tasks/todo.md +33 -33
- package/.agent/tasks/two-track-merge-contract.md +1 -1
- package/.agent/workflows/aside.md +164 -164
- package/.agent/workflows/brainstorm.md +113 -113
- package/.agent/workflows/build-fix.md +62 -62
- package/.agent/workflows/checkpoint.md +74 -74
- package/.agent/workflows/claw.md +23 -23
- package/.agent/workflows/clean-memory.md +34 -34
- package/.agent/workflows/code-review.md +289 -289
- package/.agent/workflows/context-budget.md +23 -23
- package/.agent/workflows/cpp-build.md +173 -173
- package/.agent/workflows/cpp-review.md +132 -132
- package/.agent/workflows/cpp-test.md +251 -251
- package/.agent/workflows/create.md +59 -59
- package/.agent/workflows/debug.md +103 -103
- package/.agent/workflows/deploy.md +176 -176
- package/.agent/workflows/devfleet.md +23 -23
- package/.agent/workflows/docs.md +23 -23
- package/.agent/workflows/e2e.md +268 -268
- package/.agent/workflows/enhance.md +63 -63
- package/.agent/workflows/eval.md +23 -23
- package/.agent/workflows/evolve.md +178 -178
- package/.agent/workflows/flutter-build.md +164 -164
- package/.agent/workflows/flutter-review.md +116 -116
- package/.agent/workflows/flutter-test.md +144 -144
- package/.agent/workflows/gan-build.md +99 -99
- package/.agent/workflows/gan-design.md +35 -35
- package/.agent/workflows/go-build.md +183 -183
- package/.agent/workflows/go-review.md +148 -148
- package/.agent/workflows/go-test.md +268 -268
- package/.agent/workflows/gradle-build.md +70 -70
- package/.agent/workflows/harness-audit.md +73 -73
- package/.agent/workflows/init-docs.md +46 -46
- package/.agent/workflows/instinct-export.md +66 -66
- package/.agent/workflows/instinct-import.md +114 -114
- package/.agent/workflows/instinct-status.md +59 -59
- package/.agent/workflows/jira.md +106 -106
- package/.agent/workflows/kotlin-build.md +174 -174
- package/.agent/workflows/kotlin-review.md +140 -140
- package/.agent/workflows/kotlin-test.md +312 -312
- package/.agent/workflows/learn-eval.md +116 -116
- package/.agent/workflows/learn.md +70 -70
- package/.agent/workflows/loop-start.md +32 -32
- package/.agent/workflows/loop-status.md +24 -24
- package/.agent/workflows/model-route.md +26 -26
- package/.agent/workflows/multi-backend.md +158 -158
- package/.agent/workflows/multi-execute.md +315 -315
- package/.agent/workflows/multi-frontend.md +158 -158
- package/.agent/workflows/multi-plan.md +268 -268
- package/.agent/workflows/multi-workflow.md +191 -191
- package/.agent/workflows/orchestrate.md +135 -135
- package/.agent/workflows/plan.md +117 -117
- package/.agent/workflows/pm2.md +272 -272
- package/.agent/workflows/preview.md +81 -81
- package/.agent/workflows/projects.md +39 -39
- package/.agent/workflows/promote.md +41 -41
- package/.agent/workflows/prompt-optimize.md +23 -23
- package/.agent/workflows/prp-commit.md +112 -112
- package/.agent/workflows/prp-implement.md +385 -385
- package/.agent/workflows/prp-plan.md +502 -502
- package/.agent/workflows/prp-pr.md +184 -184
- package/.agent/workflows/prp-prd.md +447 -447
- package/.agent/workflows/prune.md +31 -31
- package/.agent/workflows/python-review.md +297 -297
- package/.agent/workflows/quality-gate.md +29 -29
- package/.agent/workflows/refactor-clean.md +80 -80
- package/.agent/workflows/resume-session.md +156 -156
- package/.agent/workflows/rules-distill.md +20 -20
- package/.agent/workflows/rust-build.md +187 -187
- package/.agent/workflows/rust-review.md +142 -142
- package/.agent/workflows/rust-test.md +308 -308
- package/.agent/workflows/santa-loop.md +175 -175
- package/.agent/workflows/save-session.md +275 -275
- package/.agent/workflows/sessions.md +333 -333
- package/.agent/workflows/setup-pm.md +80 -80
- package/.agent/workflows/skill-create.md +174 -174
- package/.agent/workflows/skill-health.md +54 -54
- package/.agent/workflows/status.md +86 -86
- package/.agent/workflows/tdd.md +231 -231
- package/.agent/workflows/test-coverage.md +69 -69
- package/.agent/workflows/test.md +144 -144
- package/.agent/workflows/ui-ux-pro-max.md +295 -295
- package/.agent/workflows/update-codemaps.md +72 -72
- package/.agent/workflows/update-docs.md +84 -84
- package/.agent/workflows/verify.md +23 -23
- package/LICENSE +176 -176
- package/README.md +144 -144
- package/package.json +1 -1
- package/scripts/release-check.js +55 -55
- package/src/bin/cli.js +424 -354
- package/src/lib/installer.js +223 -11
|
@@ -1,711 +1,711 @@
|
|
|
1
|
-
---
|
|
2
|
-
name: kotlin-patterns
|
|
3
|
-
description: Idiomatic Kotlin patterns, best practices, and conventions for building robust, efficient, and maintainable Kotlin applications with coroutines, null safety, and DSL builders.
|
|
4
|
-
origin: ECC
|
|
5
|
-
---
|
|
6
|
-
|
|
7
|
-
# Kotlin Development Patterns
|
|
8
|
-
|
|
9
|
-
Idiomatic Kotlin patterns and best practices for building robust, efficient, and maintainable applications.
|
|
10
|
-
|
|
11
|
-
## When to Use
|
|
12
|
-
|
|
13
|
-
- Writing new Kotlin code
|
|
14
|
-
- Reviewing Kotlin code
|
|
15
|
-
- Refactoring existing Kotlin code
|
|
16
|
-
- Designing Kotlin modules or libraries
|
|
17
|
-
- Configuring Gradle Kotlin DSL builds
|
|
18
|
-
|
|
19
|
-
## How It Works
|
|
20
|
-
|
|
21
|
-
This skill enforces idiomatic Kotlin conventions across seven key areas: null safety using the type system and safe-call operators, immutability via `val` and `copy()` on data classes, sealed classes and interfaces for exhaustive type hierarchies, structured concurrency with coroutines and `Flow`, extension functions for adding behaviour without inheritance, type-safe DSL builders using `@DslMarker` and lambda receivers, and Gradle Kotlin DSL for build configuration.
|
|
22
|
-
|
|
23
|
-
## Examples
|
|
24
|
-
|
|
25
|
-
**Null safety with Elvis operator:**
|
|
26
|
-
```kotlin
|
|
27
|
-
fun getUserEmail(userId: String): String {
|
|
28
|
-
val user = userRepository.findById(userId)
|
|
29
|
-
return user?.email ?: "unknown@example.com"
|
|
30
|
-
}
|
|
31
|
-
```
|
|
32
|
-
|
|
33
|
-
**Sealed class for exhaustive results:**
|
|
34
|
-
```kotlin
|
|
35
|
-
sealed class Result<out T> {
|
|
36
|
-
data class Success<T>(val data: T) : Result<T>()
|
|
37
|
-
data class Failure(val error: AppError) : Result<Nothing>()
|
|
38
|
-
data object Loading : Result<Nothing>()
|
|
39
|
-
}
|
|
40
|
-
```
|
|
41
|
-
|
|
42
|
-
**Structured concurrency with async/await:**
|
|
43
|
-
```kotlin
|
|
44
|
-
suspend fun fetchUserWithPosts(userId: String): UserProfile =
|
|
45
|
-
coroutineScope {
|
|
46
|
-
val user = async { userService.getUser(userId) }
|
|
47
|
-
val posts = async { postService.getUserPosts(userId) }
|
|
48
|
-
UserProfile(user = user.await(), posts = posts.await())
|
|
49
|
-
}
|
|
50
|
-
```
|
|
51
|
-
|
|
52
|
-
## Core Principles
|
|
53
|
-
|
|
54
|
-
### 1. Null Safety
|
|
55
|
-
|
|
56
|
-
Kotlin's type system distinguishes nullable and non-nullable types. Leverage it fully.
|
|
57
|
-
|
|
58
|
-
```kotlin
|
|
59
|
-
// Good: Use non-nullable types by default
|
|
60
|
-
fun getUser(id: String): User {
|
|
61
|
-
return userRepository.findById(id)
|
|
62
|
-
?: throw UserNotFoundException("User $id not found")
|
|
63
|
-
}
|
|
64
|
-
|
|
65
|
-
// Good: Safe calls and Elvis operator
|
|
66
|
-
fun getUserEmail(userId: String): String {
|
|
67
|
-
val user = userRepository.findById(userId)
|
|
68
|
-
return user?.email ?: "unknown@example.com"
|
|
69
|
-
}
|
|
70
|
-
|
|
71
|
-
// Bad: Force-unwrapping nullable types
|
|
72
|
-
fun getUserEmail(userId: String): String {
|
|
73
|
-
val user = userRepository.findById(userId)
|
|
74
|
-
return user!!.email // Throws NPE if null
|
|
75
|
-
}
|
|
76
|
-
```
|
|
77
|
-
|
|
78
|
-
### 2. Immutability by Default
|
|
79
|
-
|
|
80
|
-
Prefer `val` over `var`, immutable collections over mutable ones.
|
|
81
|
-
|
|
82
|
-
```kotlin
|
|
83
|
-
// Good: Immutable data
|
|
84
|
-
data class User(
|
|
85
|
-
val id: String,
|
|
86
|
-
val name: String,
|
|
87
|
-
val email: String,
|
|
88
|
-
)
|
|
89
|
-
|
|
90
|
-
// Good: Transform with copy()
|
|
91
|
-
fun updateEmail(user: User, newEmail: String): User =
|
|
92
|
-
user.copy(email = newEmail)
|
|
93
|
-
|
|
94
|
-
// Good: Immutable collections
|
|
95
|
-
val users: List<User> = listOf(user1, user2)
|
|
96
|
-
val filtered = users.filter { it.email.isNotBlank() }
|
|
97
|
-
|
|
98
|
-
// Bad: Mutable state
|
|
99
|
-
var currentUser: User? = null // Avoid mutable global state
|
|
100
|
-
val mutableUsers = mutableListOf<User>() // Avoid unless truly needed
|
|
101
|
-
```
|
|
102
|
-
|
|
103
|
-
### 3. Expression Bodies and Single-Expression Functions
|
|
104
|
-
|
|
105
|
-
Use expression bodies for concise, readable functions.
|
|
106
|
-
|
|
107
|
-
```kotlin
|
|
108
|
-
// Good: Expression body
|
|
109
|
-
fun isAdult(age: Int): Boolean = age >= 18
|
|
110
|
-
|
|
111
|
-
fun formatFullName(first: String, last: String): String =
|
|
112
|
-
"$first $last".trim()
|
|
113
|
-
|
|
114
|
-
fun User.displayName(): String =
|
|
115
|
-
name.ifBlank { email.substringBefore('@') }
|
|
116
|
-
|
|
117
|
-
// Good: When as expression
|
|
118
|
-
fun statusMessage(code: Int): String = when (code) {
|
|
119
|
-
200 -> "OK"
|
|
120
|
-
404 -> "Not Found"
|
|
121
|
-
500 -> "Internal Server Error"
|
|
122
|
-
else -> "Unknown status: $code"
|
|
123
|
-
}
|
|
124
|
-
|
|
125
|
-
// Bad: Unnecessary block body
|
|
126
|
-
fun isAdult(age: Int): Boolean {
|
|
127
|
-
return age >= 18
|
|
128
|
-
}
|
|
129
|
-
```
|
|
130
|
-
|
|
131
|
-
### 4. Data Classes for Value Objects
|
|
132
|
-
|
|
133
|
-
Use data classes for types that primarily hold data.
|
|
134
|
-
|
|
135
|
-
```kotlin
|
|
136
|
-
// Good: Data class with copy, equals, hashCode, toString
|
|
137
|
-
data class CreateUserRequest(
|
|
138
|
-
val name: String,
|
|
139
|
-
val email: String,
|
|
140
|
-
val role: Role = Role.USER,
|
|
141
|
-
)
|
|
142
|
-
|
|
143
|
-
// Good: Value class for type safety (zero overhead at runtime)
|
|
144
|
-
@JvmInline
|
|
145
|
-
value class UserId(val value: String) {
|
|
146
|
-
init {
|
|
147
|
-
require(value.isNotBlank()) { "UserId cannot be blank" }
|
|
148
|
-
}
|
|
149
|
-
}
|
|
150
|
-
|
|
151
|
-
@JvmInline
|
|
152
|
-
value class Email(val value: String) {
|
|
153
|
-
init {
|
|
154
|
-
require('@' in value) { "Invalid email: $value" }
|
|
155
|
-
}
|
|
156
|
-
}
|
|
157
|
-
|
|
158
|
-
fun getUser(id: UserId): User = userRepository.findById(id)
|
|
159
|
-
```
|
|
160
|
-
|
|
161
|
-
## Sealed Classes and Interfaces
|
|
162
|
-
|
|
163
|
-
### Modeling Restricted Hierarchies
|
|
164
|
-
|
|
165
|
-
```kotlin
|
|
166
|
-
// Good: Sealed class for exhaustive when
|
|
167
|
-
sealed class Result<out T> {
|
|
168
|
-
data class Success<T>(val data: T) : Result<T>()
|
|
169
|
-
data class Failure(val error: AppError) : Result<Nothing>()
|
|
170
|
-
data object Loading : Result<Nothing>()
|
|
171
|
-
}
|
|
172
|
-
|
|
173
|
-
fun <T> Result<T>.getOrNull(): T? = when (this) {
|
|
174
|
-
is Result.Success -> data
|
|
175
|
-
is Result.Failure -> null
|
|
176
|
-
is Result.Loading -> null
|
|
177
|
-
}
|
|
178
|
-
|
|
179
|
-
fun <T> Result<T>.getOrThrow(): T = when (this) {
|
|
180
|
-
is Result.Success -> data
|
|
181
|
-
is Result.Failure -> throw error.toException()
|
|
182
|
-
is Result.Loading -> throw IllegalStateException("Still loading")
|
|
183
|
-
}
|
|
184
|
-
```
|
|
185
|
-
|
|
186
|
-
### Sealed Interfaces for API Responses
|
|
187
|
-
|
|
188
|
-
```kotlin
|
|
189
|
-
sealed interface ApiError {
|
|
190
|
-
val message: String
|
|
191
|
-
|
|
192
|
-
data class NotFound(override val message: String) : ApiError
|
|
193
|
-
data class Unauthorized(override val message: String) : ApiError
|
|
194
|
-
data class Validation(
|
|
195
|
-
override val message: String,
|
|
196
|
-
val field: String,
|
|
197
|
-
) : ApiError
|
|
198
|
-
data class Internal(
|
|
199
|
-
override val message: String,
|
|
200
|
-
val cause: Throwable? = null,
|
|
201
|
-
) : ApiError
|
|
202
|
-
}
|
|
203
|
-
|
|
204
|
-
fun ApiError.toStatusCode(): Int = when (this) {
|
|
205
|
-
is ApiError.NotFound -> 404
|
|
206
|
-
is ApiError.Unauthorized -> 401
|
|
207
|
-
is ApiError.Validation -> 422
|
|
208
|
-
is ApiError.Internal -> 500
|
|
209
|
-
}
|
|
210
|
-
```
|
|
211
|
-
|
|
212
|
-
## Scope Functions
|
|
213
|
-
|
|
214
|
-
### When to Use Each
|
|
215
|
-
|
|
216
|
-
```kotlin
|
|
217
|
-
// let: Transform nullable or scoped result
|
|
218
|
-
val length: Int? = name?.let { it.trim().length }
|
|
219
|
-
|
|
220
|
-
// apply: Configure an object (returns the object)
|
|
221
|
-
val user = User().apply {
|
|
222
|
-
name = "Alice"
|
|
223
|
-
email = "alice@example.com"
|
|
224
|
-
}
|
|
225
|
-
|
|
226
|
-
// also: Side effects (returns the object)
|
|
227
|
-
val user = createUser(request).also { logger.info("Created user: ${it.id}") }
|
|
228
|
-
|
|
229
|
-
// run: Execute a block with receiver (returns result)
|
|
230
|
-
val result = connection.run {
|
|
231
|
-
prepareStatement(sql)
|
|
232
|
-
executeQuery()
|
|
233
|
-
}
|
|
234
|
-
|
|
235
|
-
// with: Non-extension form of run
|
|
236
|
-
val csv = with(StringBuilder()) {
|
|
237
|
-
appendLine("name,email")
|
|
238
|
-
users.forEach { appendLine("${it.name},${it.email}") }
|
|
239
|
-
toString()
|
|
240
|
-
}
|
|
241
|
-
```
|
|
242
|
-
|
|
243
|
-
### Anti-Patterns
|
|
244
|
-
|
|
245
|
-
```kotlin
|
|
246
|
-
// Bad: Nesting scope functions
|
|
247
|
-
user?.let { u ->
|
|
248
|
-
u.address?.let { addr ->
|
|
249
|
-
addr.city?.let { city ->
|
|
250
|
-
println(city) // Hard to read
|
|
251
|
-
}
|
|
252
|
-
}
|
|
253
|
-
}
|
|
254
|
-
|
|
255
|
-
// Good: Chain safe calls instead
|
|
256
|
-
val city = user?.address?.city
|
|
257
|
-
city?.let { println(it) }
|
|
258
|
-
```
|
|
259
|
-
|
|
260
|
-
## Extension Functions
|
|
261
|
-
|
|
262
|
-
### Adding Functionality Without Inheritance
|
|
263
|
-
|
|
264
|
-
```kotlin
|
|
265
|
-
// Good: Domain-specific extensions
|
|
266
|
-
fun String.toSlug(): String =
|
|
267
|
-
lowercase()
|
|
268
|
-
.replace(Regex("[^a-z0-9\\s-]"), "")
|
|
269
|
-
.replace(Regex("\\s+"), "-")
|
|
270
|
-
.trim('-')
|
|
271
|
-
|
|
272
|
-
fun Instant.toLocalDate(zone: ZoneId = ZoneId.systemDefault()): LocalDate =
|
|
273
|
-
atZone(zone).toLocalDate()
|
|
274
|
-
|
|
275
|
-
// Good: Collection extensions
|
|
276
|
-
fun <T> List<T>.second(): T = this[1]
|
|
277
|
-
|
|
278
|
-
fun <T> List<T>.secondOrNull(): T? = getOrNull(1)
|
|
279
|
-
|
|
280
|
-
// Good: Scoped extensions (not polluting global namespace)
|
|
281
|
-
class UserService {
|
|
282
|
-
private fun User.isActive(): Boolean =
|
|
283
|
-
status == Status.ACTIVE && lastLogin.isAfter(Instant.now().minus(30, ChronoUnit.DAYS))
|
|
284
|
-
|
|
285
|
-
fun getActiveUsers(): List<User> = userRepository.findAll().filter { it.isActive() }
|
|
286
|
-
}
|
|
287
|
-
```
|
|
288
|
-
|
|
289
|
-
## Coroutines
|
|
290
|
-
|
|
291
|
-
### Structured Concurrency
|
|
292
|
-
|
|
293
|
-
```kotlin
|
|
294
|
-
// Good: Structured concurrency with coroutineScope
|
|
295
|
-
suspend fun fetchUserWithPosts(userId: String): UserProfile =
|
|
296
|
-
coroutineScope {
|
|
297
|
-
val userDeferred = async { userService.getUser(userId) }
|
|
298
|
-
val postsDeferred = async { postService.getUserPosts(userId) }
|
|
299
|
-
|
|
300
|
-
UserProfile(
|
|
301
|
-
user = userDeferred.await(),
|
|
302
|
-
posts = postsDeferred.await(),
|
|
303
|
-
)
|
|
304
|
-
}
|
|
305
|
-
|
|
306
|
-
// Good: supervisorScope when children can fail independently
|
|
307
|
-
suspend fun fetchDashboard(userId: String): Dashboard =
|
|
308
|
-
supervisorScope {
|
|
309
|
-
val user = async { userService.getUser(userId) }
|
|
310
|
-
val notifications = async { notificationService.getRecent(userId) }
|
|
311
|
-
val recommendations = async { recommendationService.getFor(userId) }
|
|
312
|
-
|
|
313
|
-
Dashboard(
|
|
314
|
-
user = user.await(),
|
|
315
|
-
notifications = try {
|
|
316
|
-
notifications.await()
|
|
317
|
-
} catch (e: CancellationException) {
|
|
318
|
-
throw e
|
|
319
|
-
} catch (e: Exception) {
|
|
320
|
-
emptyList()
|
|
321
|
-
},
|
|
322
|
-
recommendations = try {
|
|
323
|
-
recommendations.await()
|
|
324
|
-
} catch (e: CancellationException) {
|
|
325
|
-
throw e
|
|
326
|
-
} catch (e: Exception) {
|
|
327
|
-
emptyList()
|
|
328
|
-
},
|
|
329
|
-
)
|
|
330
|
-
}
|
|
331
|
-
```
|
|
332
|
-
|
|
333
|
-
### Flow for Reactive Streams
|
|
334
|
-
|
|
335
|
-
```kotlin
|
|
336
|
-
// Good: Cold flow with proper error handling
|
|
337
|
-
fun observeUsers(): Flow<List<User>> = flow {
|
|
338
|
-
while (currentCoroutineContext().isActive) {
|
|
339
|
-
val users = userRepository.findAll()
|
|
340
|
-
emit(users)
|
|
341
|
-
delay(5.seconds)
|
|
342
|
-
}
|
|
343
|
-
}.catch { e ->
|
|
344
|
-
logger.error("Error observing users", e)
|
|
345
|
-
emit(emptyList())
|
|
346
|
-
}
|
|
347
|
-
|
|
348
|
-
// Good: Flow operators
|
|
349
|
-
fun searchUsers(query: Flow<String>): Flow<List<User>> =
|
|
350
|
-
query
|
|
351
|
-
.debounce(300.milliseconds)
|
|
352
|
-
.distinctUntilChanged()
|
|
353
|
-
.filter { it.length >= 2 }
|
|
354
|
-
.mapLatest { q -> userRepository.search(q) }
|
|
355
|
-
.catch { emit(emptyList()) }
|
|
356
|
-
```
|
|
357
|
-
|
|
358
|
-
### Cancellation and Cleanup
|
|
359
|
-
|
|
360
|
-
```kotlin
|
|
361
|
-
// Good: Respect cancellation
|
|
362
|
-
suspend fun processItems(items: List<Item>) {
|
|
363
|
-
items.forEach { item ->
|
|
364
|
-
ensureActive() // Check cancellation before expensive work
|
|
365
|
-
processItem(item)
|
|
366
|
-
}
|
|
367
|
-
}
|
|
368
|
-
|
|
369
|
-
// Good: Cleanup with try/finally
|
|
370
|
-
suspend fun acquireAndProcess() {
|
|
371
|
-
val resource = acquireResource()
|
|
372
|
-
try {
|
|
373
|
-
resource.process()
|
|
374
|
-
} finally {
|
|
375
|
-
withContext(NonCancellable) {
|
|
376
|
-
resource.release() // Always release, even on cancellation
|
|
377
|
-
}
|
|
378
|
-
}
|
|
379
|
-
}
|
|
380
|
-
```
|
|
381
|
-
|
|
382
|
-
## Delegation
|
|
383
|
-
|
|
384
|
-
### Property Delegation
|
|
385
|
-
|
|
386
|
-
```kotlin
|
|
387
|
-
// Lazy initialization
|
|
388
|
-
val expensiveData: List<User> by lazy {
|
|
389
|
-
userRepository.findAll()
|
|
390
|
-
}
|
|
391
|
-
|
|
392
|
-
// Observable property
|
|
393
|
-
var name: String by Delegates.observable("initial") { _, old, new ->
|
|
394
|
-
logger.info("Name changed from '$old' to '$new'")
|
|
395
|
-
}
|
|
396
|
-
|
|
397
|
-
// Map-backed properties
|
|
398
|
-
class Config(private val map: Map<String, Any?>) {
|
|
399
|
-
val host: String by map
|
|
400
|
-
val port: Int by map
|
|
401
|
-
val debug: Boolean by map
|
|
402
|
-
}
|
|
403
|
-
|
|
404
|
-
val config = Config(mapOf("host" to "localhost", "port" to 8080, "debug" to true))
|
|
405
|
-
```
|
|
406
|
-
|
|
407
|
-
### Interface Delegation
|
|
408
|
-
|
|
409
|
-
```kotlin
|
|
410
|
-
// Good: Delegate interface implementation
|
|
411
|
-
class LoggingUserRepository(
|
|
412
|
-
private val delegate: UserRepository,
|
|
413
|
-
private val logger: Logger,
|
|
414
|
-
) : UserRepository by delegate {
|
|
415
|
-
// Only override what you need to add logging to
|
|
416
|
-
override suspend fun findById(id: String): User? {
|
|
417
|
-
logger.info("Finding user by id: $id")
|
|
418
|
-
return delegate.findById(id).also {
|
|
419
|
-
logger.info("Found user: ${it?.name ?: "null"}")
|
|
420
|
-
}
|
|
421
|
-
}
|
|
422
|
-
}
|
|
423
|
-
```
|
|
424
|
-
|
|
425
|
-
## DSL Builders
|
|
426
|
-
|
|
427
|
-
### Type-Safe Builders
|
|
428
|
-
|
|
429
|
-
```kotlin
|
|
430
|
-
// Good: DSL with @DslMarker
|
|
431
|
-
@DslMarker
|
|
432
|
-
annotation class HtmlDsl
|
|
433
|
-
|
|
434
|
-
@HtmlDsl
|
|
435
|
-
class HTML {
|
|
436
|
-
private val children = mutableListOf<Element>()
|
|
437
|
-
|
|
438
|
-
fun head(init: Head.() -> Unit) {
|
|
439
|
-
children += Head().apply(init)
|
|
440
|
-
}
|
|
441
|
-
|
|
442
|
-
fun body(init: Body.() -> Unit) {
|
|
443
|
-
children += Body().apply(init)
|
|
444
|
-
}
|
|
445
|
-
|
|
446
|
-
override fun toString(): String = children.joinToString("\n")
|
|
447
|
-
}
|
|
448
|
-
|
|
449
|
-
fun html(init: HTML.() -> Unit): HTML = HTML().apply(init)
|
|
450
|
-
|
|
451
|
-
// Usage
|
|
452
|
-
val page = html {
|
|
453
|
-
head { title("My Page") }
|
|
454
|
-
body {
|
|
455
|
-
h1("Welcome")
|
|
456
|
-
p("Hello, World!")
|
|
457
|
-
}
|
|
458
|
-
}
|
|
459
|
-
```
|
|
460
|
-
|
|
461
|
-
### Configuration DSL
|
|
462
|
-
|
|
463
|
-
```kotlin
|
|
464
|
-
data class ServerConfig(
|
|
465
|
-
val host: String = "0.0.0.0",
|
|
466
|
-
val port: Int = 8080,
|
|
467
|
-
val ssl: SslConfig? = null,
|
|
468
|
-
val database: DatabaseConfig? = null,
|
|
469
|
-
)
|
|
470
|
-
|
|
471
|
-
data class SslConfig(val certPath: String, val keyPath: String)
|
|
472
|
-
data class DatabaseConfig(val url: String, val maxPoolSize: Int = 10)
|
|
473
|
-
|
|
474
|
-
class ServerConfigBuilder {
|
|
475
|
-
var host: String = "0.0.0.0"
|
|
476
|
-
var port: Int = 8080
|
|
477
|
-
private var ssl: SslConfig? = null
|
|
478
|
-
private var database: DatabaseConfig? = null
|
|
479
|
-
|
|
480
|
-
fun ssl(certPath: String, keyPath: String) {
|
|
481
|
-
ssl = SslConfig(certPath, keyPath)
|
|
482
|
-
}
|
|
483
|
-
|
|
484
|
-
fun database(url: String, maxPoolSize: Int = 10) {
|
|
485
|
-
database = DatabaseConfig(url, maxPoolSize)
|
|
486
|
-
}
|
|
487
|
-
|
|
488
|
-
fun build(): ServerConfig = ServerConfig(host, port, ssl, database)
|
|
489
|
-
}
|
|
490
|
-
|
|
491
|
-
fun serverConfig(init: ServerConfigBuilder.() -> Unit): ServerConfig =
|
|
492
|
-
ServerConfigBuilder().apply(init).build()
|
|
493
|
-
|
|
494
|
-
// Usage
|
|
495
|
-
val config = serverConfig {
|
|
496
|
-
host = "0.0.0.0"
|
|
497
|
-
port = 443
|
|
498
|
-
ssl("/certs/cert.pem", "/certs/key.pem")
|
|
499
|
-
database("jdbc:postgresql://localhost:5432/mydb", maxPoolSize = 20)
|
|
500
|
-
}
|
|
501
|
-
```
|
|
502
|
-
|
|
503
|
-
## Sequences for Lazy Evaluation
|
|
504
|
-
|
|
505
|
-
```kotlin
|
|
506
|
-
// Good: Use sequences for large collections with multiple operations
|
|
507
|
-
val result = users.asSequence()
|
|
508
|
-
.filter { it.isActive }
|
|
509
|
-
.map { it.email }
|
|
510
|
-
.filter { it.endsWith("@company.com") }
|
|
511
|
-
.take(10)
|
|
512
|
-
.toList()
|
|
513
|
-
|
|
514
|
-
// Good: Generate infinite sequences
|
|
515
|
-
val fibonacci: Sequence<Long> = sequence {
|
|
516
|
-
var a = 0L
|
|
517
|
-
var b = 1L
|
|
518
|
-
while (true) {
|
|
519
|
-
yield(a)
|
|
520
|
-
val next = a + b
|
|
521
|
-
a = b
|
|
522
|
-
b = next
|
|
523
|
-
}
|
|
524
|
-
}
|
|
525
|
-
|
|
526
|
-
val first20 = fibonacci.take(20).toList()
|
|
527
|
-
```
|
|
528
|
-
|
|
529
|
-
## Gradle Kotlin DSL
|
|
530
|
-
|
|
531
|
-
### build.gradle.kts Configuration
|
|
532
|
-
|
|
533
|
-
```kotlin
|
|
534
|
-
// Check for latest versions: https://kotlinlang.org/docs/releases.html
|
|
535
|
-
plugins {
|
|
536
|
-
kotlin("jvm") version "2.3.10"
|
|
537
|
-
kotlin("plugin.serialization") version "2.3.10"
|
|
538
|
-
id("io.ktor.plugin") version "3.4.0"
|
|
539
|
-
id("org.jetbrains.kotlinx.kover") version "0.9.7"
|
|
540
|
-
id("io.gitlab.arturbosch.detekt") version "1.23.8"
|
|
541
|
-
}
|
|
542
|
-
|
|
543
|
-
group = "com.example"
|
|
544
|
-
version = "1.0.0"
|
|
545
|
-
|
|
546
|
-
kotlin {
|
|
547
|
-
jvmToolchain(21)
|
|
548
|
-
}
|
|
549
|
-
|
|
550
|
-
dependencies {
|
|
551
|
-
// Ktor
|
|
552
|
-
implementation("io.ktor:ktor-server-core:3.4.0")
|
|
553
|
-
implementation("io.ktor:ktor-server-netty:3.4.0")
|
|
554
|
-
implementation("io.ktor:ktor-server-content-negotiation:3.4.0")
|
|
555
|
-
implementation("io.ktor:ktor-serialization-kotlinx-json:3.4.0")
|
|
556
|
-
|
|
557
|
-
// Exposed
|
|
558
|
-
implementation("org.jetbrains.exposed:exposed-core:1.0.0")
|
|
559
|
-
implementation("org.jetbrains.exposed:exposed-dao:1.0.0")
|
|
560
|
-
implementation("org.jetbrains.exposed:exposed-jdbc:1.0.0")
|
|
561
|
-
implementation("org.jetbrains.exposed:exposed-kotlin-datetime:1.0.0")
|
|
562
|
-
|
|
563
|
-
// Koin
|
|
564
|
-
implementation("io.insert-koin:koin-ktor:4.2.0")
|
|
565
|
-
|
|
566
|
-
// Coroutines
|
|
567
|
-
implementation("org.jetbrains.kotlinx:kotlinx-coroutines-core:1.10.2")
|
|
568
|
-
|
|
569
|
-
// Testing
|
|
570
|
-
testImplementation("io.kotest:kotest-runner-junit5:6.1.4")
|
|
571
|
-
testImplementation("io.kotest:kotest-assertions-core:6.1.4")
|
|
572
|
-
testImplementation("io.kotest:kotest-property:6.1.4")
|
|
573
|
-
testImplementation("io.mockk:mockk:1.14.9")
|
|
574
|
-
testImplementation("io.ktor:ktor-server-test-host:3.4.0")
|
|
575
|
-
testImplementation("org.jetbrains.kotlinx:kotlinx-coroutines-test:1.10.2")
|
|
576
|
-
}
|
|
577
|
-
|
|
578
|
-
tasks.withType<Test> {
|
|
579
|
-
useJUnitPlatform()
|
|
580
|
-
}
|
|
581
|
-
|
|
582
|
-
detekt {
|
|
583
|
-
config.setFrom(files("config/detekt/detekt.yml"))
|
|
584
|
-
buildUponDefaultConfig = true
|
|
585
|
-
}
|
|
586
|
-
```
|
|
587
|
-
|
|
588
|
-
## Error Handling Patterns
|
|
589
|
-
|
|
590
|
-
### Result Type for Domain Operations
|
|
591
|
-
|
|
592
|
-
```kotlin
|
|
593
|
-
// Good: Use Kotlin's Result or a custom sealed class
|
|
594
|
-
suspend fun createUser(request: CreateUserRequest): Result<User> = runCatching {
|
|
595
|
-
require(request.name.isNotBlank()) { "Name cannot be blank" }
|
|
596
|
-
require('@' in request.email) { "Invalid email format" }
|
|
597
|
-
|
|
598
|
-
val user = User(
|
|
599
|
-
id = UserId(UUID.randomUUID().toString()),
|
|
600
|
-
name = request.name,
|
|
601
|
-
email = Email(request.email),
|
|
602
|
-
)
|
|
603
|
-
userRepository.save(user)
|
|
604
|
-
user
|
|
605
|
-
}
|
|
606
|
-
|
|
607
|
-
// Good: Chain results
|
|
608
|
-
val displayName = createUser(request)
|
|
609
|
-
.map { it.name }
|
|
610
|
-
.getOrElse { "Unknown" }
|
|
611
|
-
```
|
|
612
|
-
|
|
613
|
-
### require, check, error
|
|
614
|
-
|
|
615
|
-
```kotlin
|
|
616
|
-
// Good: Preconditions with clear messages
|
|
617
|
-
fun withdraw(account: Account, amount: Money): Account {
|
|
618
|
-
require(amount.value > 0) { "Amount must be positive: $amount" }
|
|
619
|
-
check(account.balance >= amount) { "Insufficient balance: ${account.balance} < $amount" }
|
|
620
|
-
|
|
621
|
-
return account.copy(balance = account.balance - amount)
|
|
622
|
-
}
|
|
623
|
-
```
|
|
624
|
-
|
|
625
|
-
## Collection Operations
|
|
626
|
-
|
|
627
|
-
### Idiomatic Collection Processing
|
|
628
|
-
|
|
629
|
-
```kotlin
|
|
630
|
-
// Good: Chained operations
|
|
631
|
-
val activeAdminEmails: List<String> = users
|
|
632
|
-
.filter { it.role == Role.ADMIN && it.isActive }
|
|
633
|
-
.sortedBy { it.name }
|
|
634
|
-
.map { it.email }
|
|
635
|
-
|
|
636
|
-
// Good: Grouping and aggregation
|
|
637
|
-
val usersByRole: Map<Role, List<User>> = users.groupBy { it.role }
|
|
638
|
-
|
|
639
|
-
val oldestByRole: Map<Role, User?> = users.groupBy { it.role }
|
|
640
|
-
.mapValues { (_, users) -> users.minByOrNull { it.createdAt } }
|
|
641
|
-
|
|
642
|
-
// Good: Associate for map creation
|
|
643
|
-
val usersById: Map<UserId, User> = users.associateBy { it.id }
|
|
644
|
-
|
|
645
|
-
// Good: Partition for splitting
|
|
646
|
-
val (active, inactive) = users.partition { it.isActive }
|
|
647
|
-
```
|
|
648
|
-
|
|
649
|
-
## Quick Reference: Kotlin Idioms
|
|
650
|
-
|
|
651
|
-
| Idiom | Description |
|
|
652
|
-
|-------|-------------|
|
|
653
|
-
| `val` over `var` | Prefer immutable variables |
|
|
654
|
-
| `data class` | For value objects with equals/hashCode/copy |
|
|
655
|
-
| `sealed class/interface` | For restricted type hierarchies |
|
|
656
|
-
| `value class` | For type-safe wrappers with zero overhead |
|
|
657
|
-
| Expression `when` | Exhaustive pattern matching |
|
|
658
|
-
| Safe call `?.` | Null-safe member access |
|
|
659
|
-
| Elvis `?:` | Default value for nullables |
|
|
660
|
-
| `let`/`apply`/`also`/`run`/`with` | Scope functions for clean code |
|
|
661
|
-
| Extension functions | Add behavior without inheritance |
|
|
662
|
-
| `copy()` | Immutable updates on data classes |
|
|
663
|
-
| `require`/`check` | Precondition assertions |
|
|
664
|
-
| Coroutine `async`/`await` | Structured concurrent execution |
|
|
665
|
-
| `Flow` | Cold reactive streams |
|
|
666
|
-
| `sequence` | Lazy evaluation |
|
|
667
|
-
| Delegation `by` | Reuse implementation without inheritance |
|
|
668
|
-
|
|
669
|
-
## Anti-Patterns to Avoid
|
|
670
|
-
|
|
671
|
-
```kotlin
|
|
672
|
-
// Bad: Force-unwrapping nullable types
|
|
673
|
-
val name = user!!.name
|
|
674
|
-
|
|
675
|
-
// Bad: Platform type leakage from Java
|
|
676
|
-
fun getLength(s: String) = s.length // Safe
|
|
677
|
-
fun getLength(s: String?) = s?.length ?: 0 // Handle nulls from Java
|
|
678
|
-
|
|
679
|
-
// Bad: Mutable data classes
|
|
680
|
-
data class MutableUser(var name: String, var email: String)
|
|
681
|
-
|
|
682
|
-
// Bad: Using exceptions for control flow
|
|
683
|
-
try {
|
|
684
|
-
val user = findUser(id)
|
|
685
|
-
} catch (e: NotFoundException) {
|
|
686
|
-
// Don't use exceptions for expected cases
|
|
687
|
-
}
|
|
688
|
-
|
|
689
|
-
// Good: Use nullable return or Result
|
|
690
|
-
val user: User? = findUserOrNull(id)
|
|
691
|
-
|
|
692
|
-
// Bad: Ignoring coroutine scope
|
|
693
|
-
GlobalScope.launch { /* Avoid GlobalScope */ }
|
|
694
|
-
|
|
695
|
-
// Good: Use structured concurrency
|
|
696
|
-
coroutineScope {
|
|
697
|
-
launch { /* Properly scoped */ }
|
|
698
|
-
}
|
|
699
|
-
|
|
700
|
-
// Bad: Deeply nested scope functions
|
|
701
|
-
user?.let { u ->
|
|
702
|
-
u.address?.let { a ->
|
|
703
|
-
a.city?.let { c -> process(c) }
|
|
704
|
-
}
|
|
705
|
-
}
|
|
706
|
-
|
|
707
|
-
// Good: Direct null-safe chain
|
|
708
|
-
user?.address?.city?.let { process(it) }
|
|
709
|
-
```
|
|
710
|
-
|
|
711
|
-
**Remember**: Kotlin code should be concise but readable. Leverage the type system for safety, prefer immutability, and use coroutines for concurrency. When in doubt, let the compiler help you.
|
|
1
|
+
---
|
|
2
|
+
name: kotlin-patterns
|
|
3
|
+
description: Idiomatic Kotlin patterns, best practices, and conventions for building robust, efficient, and maintainable Kotlin applications with coroutines, null safety, and DSL builders.
|
|
4
|
+
origin: ECC
|
|
5
|
+
---
|
|
6
|
+
|
|
7
|
+
# Kotlin Development Patterns
|
|
8
|
+
|
|
9
|
+
Idiomatic Kotlin patterns and best practices for building robust, efficient, and maintainable applications.
|
|
10
|
+
|
|
11
|
+
## When to Use
|
|
12
|
+
|
|
13
|
+
- Writing new Kotlin code
|
|
14
|
+
- Reviewing Kotlin code
|
|
15
|
+
- Refactoring existing Kotlin code
|
|
16
|
+
- Designing Kotlin modules or libraries
|
|
17
|
+
- Configuring Gradle Kotlin DSL builds
|
|
18
|
+
|
|
19
|
+
## How It Works
|
|
20
|
+
|
|
21
|
+
This skill enforces idiomatic Kotlin conventions across seven key areas: null safety using the type system and safe-call operators, immutability via `val` and `copy()` on data classes, sealed classes and interfaces for exhaustive type hierarchies, structured concurrency with coroutines and `Flow`, extension functions for adding behaviour without inheritance, type-safe DSL builders using `@DslMarker` and lambda receivers, and Gradle Kotlin DSL for build configuration.
|
|
22
|
+
|
|
23
|
+
## Examples
|
|
24
|
+
|
|
25
|
+
**Null safety with Elvis operator:**
|
|
26
|
+
```kotlin
|
|
27
|
+
fun getUserEmail(userId: String): String {
|
|
28
|
+
val user = userRepository.findById(userId)
|
|
29
|
+
return user?.email ?: "unknown@example.com"
|
|
30
|
+
}
|
|
31
|
+
```
|
|
32
|
+
|
|
33
|
+
**Sealed class for exhaustive results:**
|
|
34
|
+
```kotlin
|
|
35
|
+
sealed class Result<out T> {
|
|
36
|
+
data class Success<T>(val data: T) : Result<T>()
|
|
37
|
+
data class Failure(val error: AppError) : Result<Nothing>()
|
|
38
|
+
data object Loading : Result<Nothing>()
|
|
39
|
+
}
|
|
40
|
+
```
|
|
41
|
+
|
|
42
|
+
**Structured concurrency with async/await:**
|
|
43
|
+
```kotlin
|
|
44
|
+
suspend fun fetchUserWithPosts(userId: String): UserProfile =
|
|
45
|
+
coroutineScope {
|
|
46
|
+
val user = async { userService.getUser(userId) }
|
|
47
|
+
val posts = async { postService.getUserPosts(userId) }
|
|
48
|
+
UserProfile(user = user.await(), posts = posts.await())
|
|
49
|
+
}
|
|
50
|
+
```
|
|
51
|
+
|
|
52
|
+
## Core Principles
|
|
53
|
+
|
|
54
|
+
### 1. Null Safety
|
|
55
|
+
|
|
56
|
+
Kotlin's type system distinguishes nullable and non-nullable types. Leverage it fully.
|
|
57
|
+
|
|
58
|
+
```kotlin
|
|
59
|
+
// Good: Use non-nullable types by default
|
|
60
|
+
fun getUser(id: String): User {
|
|
61
|
+
return userRepository.findById(id)
|
|
62
|
+
?: throw UserNotFoundException("User $id not found")
|
|
63
|
+
}
|
|
64
|
+
|
|
65
|
+
// Good: Safe calls and Elvis operator
|
|
66
|
+
fun getUserEmail(userId: String): String {
|
|
67
|
+
val user = userRepository.findById(userId)
|
|
68
|
+
return user?.email ?: "unknown@example.com"
|
|
69
|
+
}
|
|
70
|
+
|
|
71
|
+
// Bad: Force-unwrapping nullable types
|
|
72
|
+
fun getUserEmail(userId: String): String {
|
|
73
|
+
val user = userRepository.findById(userId)
|
|
74
|
+
return user!!.email // Throws NPE if null
|
|
75
|
+
}
|
|
76
|
+
```
|
|
77
|
+
|
|
78
|
+
### 2. Immutability by Default
|
|
79
|
+
|
|
80
|
+
Prefer `val` over `var`, immutable collections over mutable ones.
|
|
81
|
+
|
|
82
|
+
```kotlin
|
|
83
|
+
// Good: Immutable data
|
|
84
|
+
data class User(
|
|
85
|
+
val id: String,
|
|
86
|
+
val name: String,
|
|
87
|
+
val email: String,
|
|
88
|
+
)
|
|
89
|
+
|
|
90
|
+
// Good: Transform with copy()
|
|
91
|
+
fun updateEmail(user: User, newEmail: String): User =
|
|
92
|
+
user.copy(email = newEmail)
|
|
93
|
+
|
|
94
|
+
// Good: Immutable collections
|
|
95
|
+
val users: List<User> = listOf(user1, user2)
|
|
96
|
+
val filtered = users.filter { it.email.isNotBlank() }
|
|
97
|
+
|
|
98
|
+
// Bad: Mutable state
|
|
99
|
+
var currentUser: User? = null // Avoid mutable global state
|
|
100
|
+
val mutableUsers = mutableListOf<User>() // Avoid unless truly needed
|
|
101
|
+
```
|
|
102
|
+
|
|
103
|
+
### 3. Expression Bodies and Single-Expression Functions
|
|
104
|
+
|
|
105
|
+
Use expression bodies for concise, readable functions.
|
|
106
|
+
|
|
107
|
+
```kotlin
|
|
108
|
+
// Good: Expression body
|
|
109
|
+
fun isAdult(age: Int): Boolean = age >= 18
|
|
110
|
+
|
|
111
|
+
fun formatFullName(first: String, last: String): String =
|
|
112
|
+
"$first $last".trim()
|
|
113
|
+
|
|
114
|
+
fun User.displayName(): String =
|
|
115
|
+
name.ifBlank { email.substringBefore('@') }
|
|
116
|
+
|
|
117
|
+
// Good: When as expression
|
|
118
|
+
fun statusMessage(code: Int): String = when (code) {
|
|
119
|
+
200 -> "OK"
|
|
120
|
+
404 -> "Not Found"
|
|
121
|
+
500 -> "Internal Server Error"
|
|
122
|
+
else -> "Unknown status: $code"
|
|
123
|
+
}
|
|
124
|
+
|
|
125
|
+
// Bad: Unnecessary block body
|
|
126
|
+
fun isAdult(age: Int): Boolean {
|
|
127
|
+
return age >= 18
|
|
128
|
+
}
|
|
129
|
+
```
|
|
130
|
+
|
|
131
|
+
### 4. Data Classes for Value Objects
|
|
132
|
+
|
|
133
|
+
Use data classes for types that primarily hold data.
|
|
134
|
+
|
|
135
|
+
```kotlin
|
|
136
|
+
// Good: Data class with copy, equals, hashCode, toString
|
|
137
|
+
data class CreateUserRequest(
|
|
138
|
+
val name: String,
|
|
139
|
+
val email: String,
|
|
140
|
+
val role: Role = Role.USER,
|
|
141
|
+
)
|
|
142
|
+
|
|
143
|
+
// Good: Value class for type safety (zero overhead at runtime)
|
|
144
|
+
@JvmInline
|
|
145
|
+
value class UserId(val value: String) {
|
|
146
|
+
init {
|
|
147
|
+
require(value.isNotBlank()) { "UserId cannot be blank" }
|
|
148
|
+
}
|
|
149
|
+
}
|
|
150
|
+
|
|
151
|
+
@JvmInline
|
|
152
|
+
value class Email(val value: String) {
|
|
153
|
+
init {
|
|
154
|
+
require('@' in value) { "Invalid email: $value" }
|
|
155
|
+
}
|
|
156
|
+
}
|
|
157
|
+
|
|
158
|
+
fun getUser(id: UserId): User = userRepository.findById(id)
|
|
159
|
+
```
|
|
160
|
+
|
|
161
|
+
## Sealed Classes and Interfaces
|
|
162
|
+
|
|
163
|
+
### Modeling Restricted Hierarchies
|
|
164
|
+
|
|
165
|
+
```kotlin
|
|
166
|
+
// Good: Sealed class for exhaustive when
|
|
167
|
+
sealed class Result<out T> {
|
|
168
|
+
data class Success<T>(val data: T) : Result<T>()
|
|
169
|
+
data class Failure(val error: AppError) : Result<Nothing>()
|
|
170
|
+
data object Loading : Result<Nothing>()
|
|
171
|
+
}
|
|
172
|
+
|
|
173
|
+
fun <T> Result<T>.getOrNull(): T? = when (this) {
|
|
174
|
+
is Result.Success -> data
|
|
175
|
+
is Result.Failure -> null
|
|
176
|
+
is Result.Loading -> null
|
|
177
|
+
}
|
|
178
|
+
|
|
179
|
+
fun <T> Result<T>.getOrThrow(): T = when (this) {
|
|
180
|
+
is Result.Success -> data
|
|
181
|
+
is Result.Failure -> throw error.toException()
|
|
182
|
+
is Result.Loading -> throw IllegalStateException("Still loading")
|
|
183
|
+
}
|
|
184
|
+
```
|
|
185
|
+
|
|
186
|
+
### Sealed Interfaces for API Responses
|
|
187
|
+
|
|
188
|
+
```kotlin
|
|
189
|
+
sealed interface ApiError {
|
|
190
|
+
val message: String
|
|
191
|
+
|
|
192
|
+
data class NotFound(override val message: String) : ApiError
|
|
193
|
+
data class Unauthorized(override val message: String) : ApiError
|
|
194
|
+
data class Validation(
|
|
195
|
+
override val message: String,
|
|
196
|
+
val field: String,
|
|
197
|
+
) : ApiError
|
|
198
|
+
data class Internal(
|
|
199
|
+
override val message: String,
|
|
200
|
+
val cause: Throwable? = null,
|
|
201
|
+
) : ApiError
|
|
202
|
+
}
|
|
203
|
+
|
|
204
|
+
fun ApiError.toStatusCode(): Int = when (this) {
|
|
205
|
+
is ApiError.NotFound -> 404
|
|
206
|
+
is ApiError.Unauthorized -> 401
|
|
207
|
+
is ApiError.Validation -> 422
|
|
208
|
+
is ApiError.Internal -> 500
|
|
209
|
+
}
|
|
210
|
+
```
|
|
211
|
+
|
|
212
|
+
## Scope Functions
|
|
213
|
+
|
|
214
|
+
### When to Use Each
|
|
215
|
+
|
|
216
|
+
```kotlin
|
|
217
|
+
// let: Transform nullable or scoped result
|
|
218
|
+
val length: Int? = name?.let { it.trim().length }
|
|
219
|
+
|
|
220
|
+
// apply: Configure an object (returns the object)
|
|
221
|
+
val user = User().apply {
|
|
222
|
+
name = "Alice"
|
|
223
|
+
email = "alice@example.com"
|
|
224
|
+
}
|
|
225
|
+
|
|
226
|
+
// also: Side effects (returns the object)
|
|
227
|
+
val user = createUser(request).also { logger.info("Created user: ${it.id}") }
|
|
228
|
+
|
|
229
|
+
// run: Execute a block with receiver (returns result)
|
|
230
|
+
val result = connection.run {
|
|
231
|
+
prepareStatement(sql)
|
|
232
|
+
executeQuery()
|
|
233
|
+
}
|
|
234
|
+
|
|
235
|
+
// with: Non-extension form of run
|
|
236
|
+
val csv = with(StringBuilder()) {
|
|
237
|
+
appendLine("name,email")
|
|
238
|
+
users.forEach { appendLine("${it.name},${it.email}") }
|
|
239
|
+
toString()
|
|
240
|
+
}
|
|
241
|
+
```
|
|
242
|
+
|
|
243
|
+
### Anti-Patterns
|
|
244
|
+
|
|
245
|
+
```kotlin
|
|
246
|
+
// Bad: Nesting scope functions
|
|
247
|
+
user?.let { u ->
|
|
248
|
+
u.address?.let { addr ->
|
|
249
|
+
addr.city?.let { city ->
|
|
250
|
+
println(city) // Hard to read
|
|
251
|
+
}
|
|
252
|
+
}
|
|
253
|
+
}
|
|
254
|
+
|
|
255
|
+
// Good: Chain safe calls instead
|
|
256
|
+
val city = user?.address?.city
|
|
257
|
+
city?.let { println(it) }
|
|
258
|
+
```
|
|
259
|
+
|
|
260
|
+
## Extension Functions
|
|
261
|
+
|
|
262
|
+
### Adding Functionality Without Inheritance
|
|
263
|
+
|
|
264
|
+
```kotlin
|
|
265
|
+
// Good: Domain-specific extensions
|
|
266
|
+
fun String.toSlug(): String =
|
|
267
|
+
lowercase()
|
|
268
|
+
.replace(Regex("[^a-z0-9\\s-]"), "")
|
|
269
|
+
.replace(Regex("\\s+"), "-")
|
|
270
|
+
.trim('-')
|
|
271
|
+
|
|
272
|
+
fun Instant.toLocalDate(zone: ZoneId = ZoneId.systemDefault()): LocalDate =
|
|
273
|
+
atZone(zone).toLocalDate()
|
|
274
|
+
|
|
275
|
+
// Good: Collection extensions
|
|
276
|
+
fun <T> List<T>.second(): T = this[1]
|
|
277
|
+
|
|
278
|
+
fun <T> List<T>.secondOrNull(): T? = getOrNull(1)
|
|
279
|
+
|
|
280
|
+
// Good: Scoped extensions (not polluting global namespace)
|
|
281
|
+
class UserService {
|
|
282
|
+
private fun User.isActive(): Boolean =
|
|
283
|
+
status == Status.ACTIVE && lastLogin.isAfter(Instant.now().minus(30, ChronoUnit.DAYS))
|
|
284
|
+
|
|
285
|
+
fun getActiveUsers(): List<User> = userRepository.findAll().filter { it.isActive() }
|
|
286
|
+
}
|
|
287
|
+
```
|
|
288
|
+
|
|
289
|
+
## Coroutines
|
|
290
|
+
|
|
291
|
+
### Structured Concurrency
|
|
292
|
+
|
|
293
|
+
```kotlin
|
|
294
|
+
// Good: Structured concurrency with coroutineScope
|
|
295
|
+
suspend fun fetchUserWithPosts(userId: String): UserProfile =
|
|
296
|
+
coroutineScope {
|
|
297
|
+
val userDeferred = async { userService.getUser(userId) }
|
|
298
|
+
val postsDeferred = async { postService.getUserPosts(userId) }
|
|
299
|
+
|
|
300
|
+
UserProfile(
|
|
301
|
+
user = userDeferred.await(),
|
|
302
|
+
posts = postsDeferred.await(),
|
|
303
|
+
)
|
|
304
|
+
}
|
|
305
|
+
|
|
306
|
+
// Good: supervisorScope when children can fail independently
|
|
307
|
+
suspend fun fetchDashboard(userId: String): Dashboard =
|
|
308
|
+
supervisorScope {
|
|
309
|
+
val user = async { userService.getUser(userId) }
|
|
310
|
+
val notifications = async { notificationService.getRecent(userId) }
|
|
311
|
+
val recommendations = async { recommendationService.getFor(userId) }
|
|
312
|
+
|
|
313
|
+
Dashboard(
|
|
314
|
+
user = user.await(),
|
|
315
|
+
notifications = try {
|
|
316
|
+
notifications.await()
|
|
317
|
+
} catch (e: CancellationException) {
|
|
318
|
+
throw e
|
|
319
|
+
} catch (e: Exception) {
|
|
320
|
+
emptyList()
|
|
321
|
+
},
|
|
322
|
+
recommendations = try {
|
|
323
|
+
recommendations.await()
|
|
324
|
+
} catch (e: CancellationException) {
|
|
325
|
+
throw e
|
|
326
|
+
} catch (e: Exception) {
|
|
327
|
+
emptyList()
|
|
328
|
+
},
|
|
329
|
+
)
|
|
330
|
+
}
|
|
331
|
+
```
|
|
332
|
+
|
|
333
|
+
### Flow for Reactive Streams
|
|
334
|
+
|
|
335
|
+
```kotlin
|
|
336
|
+
// Good: Cold flow with proper error handling
|
|
337
|
+
fun observeUsers(): Flow<List<User>> = flow {
|
|
338
|
+
while (currentCoroutineContext().isActive) {
|
|
339
|
+
val users = userRepository.findAll()
|
|
340
|
+
emit(users)
|
|
341
|
+
delay(5.seconds)
|
|
342
|
+
}
|
|
343
|
+
}.catch { e ->
|
|
344
|
+
logger.error("Error observing users", e)
|
|
345
|
+
emit(emptyList())
|
|
346
|
+
}
|
|
347
|
+
|
|
348
|
+
// Good: Flow operators
|
|
349
|
+
fun searchUsers(query: Flow<String>): Flow<List<User>> =
|
|
350
|
+
query
|
|
351
|
+
.debounce(300.milliseconds)
|
|
352
|
+
.distinctUntilChanged()
|
|
353
|
+
.filter { it.length >= 2 }
|
|
354
|
+
.mapLatest { q -> userRepository.search(q) }
|
|
355
|
+
.catch { emit(emptyList()) }
|
|
356
|
+
```
|
|
357
|
+
|
|
358
|
+
### Cancellation and Cleanup
|
|
359
|
+
|
|
360
|
+
```kotlin
|
|
361
|
+
// Good: Respect cancellation
|
|
362
|
+
suspend fun processItems(items: List<Item>) {
|
|
363
|
+
items.forEach { item ->
|
|
364
|
+
ensureActive() // Check cancellation before expensive work
|
|
365
|
+
processItem(item)
|
|
366
|
+
}
|
|
367
|
+
}
|
|
368
|
+
|
|
369
|
+
// Good: Cleanup with try/finally
|
|
370
|
+
suspend fun acquireAndProcess() {
|
|
371
|
+
val resource = acquireResource()
|
|
372
|
+
try {
|
|
373
|
+
resource.process()
|
|
374
|
+
} finally {
|
|
375
|
+
withContext(NonCancellable) {
|
|
376
|
+
resource.release() // Always release, even on cancellation
|
|
377
|
+
}
|
|
378
|
+
}
|
|
379
|
+
}
|
|
380
|
+
```
|
|
381
|
+
|
|
382
|
+
## Delegation
|
|
383
|
+
|
|
384
|
+
### Property Delegation
|
|
385
|
+
|
|
386
|
+
```kotlin
|
|
387
|
+
// Lazy initialization
|
|
388
|
+
val expensiveData: List<User> by lazy {
|
|
389
|
+
userRepository.findAll()
|
|
390
|
+
}
|
|
391
|
+
|
|
392
|
+
// Observable property
|
|
393
|
+
var name: String by Delegates.observable("initial") { _, old, new ->
|
|
394
|
+
logger.info("Name changed from '$old' to '$new'")
|
|
395
|
+
}
|
|
396
|
+
|
|
397
|
+
// Map-backed properties
|
|
398
|
+
class Config(private val map: Map<String, Any?>) {
|
|
399
|
+
val host: String by map
|
|
400
|
+
val port: Int by map
|
|
401
|
+
val debug: Boolean by map
|
|
402
|
+
}
|
|
403
|
+
|
|
404
|
+
val config = Config(mapOf("host" to "localhost", "port" to 8080, "debug" to true))
|
|
405
|
+
```
|
|
406
|
+
|
|
407
|
+
### Interface Delegation
|
|
408
|
+
|
|
409
|
+
```kotlin
|
|
410
|
+
// Good: Delegate interface implementation
|
|
411
|
+
class LoggingUserRepository(
|
|
412
|
+
private val delegate: UserRepository,
|
|
413
|
+
private val logger: Logger,
|
|
414
|
+
) : UserRepository by delegate {
|
|
415
|
+
// Only override what you need to add logging to
|
|
416
|
+
override suspend fun findById(id: String): User? {
|
|
417
|
+
logger.info("Finding user by id: $id")
|
|
418
|
+
return delegate.findById(id).also {
|
|
419
|
+
logger.info("Found user: ${it?.name ?: "null"}")
|
|
420
|
+
}
|
|
421
|
+
}
|
|
422
|
+
}
|
|
423
|
+
```
|
|
424
|
+
|
|
425
|
+
## DSL Builders
|
|
426
|
+
|
|
427
|
+
### Type-Safe Builders
|
|
428
|
+
|
|
429
|
+
```kotlin
|
|
430
|
+
// Good: DSL with @DslMarker
|
|
431
|
+
@DslMarker
|
|
432
|
+
annotation class HtmlDsl
|
|
433
|
+
|
|
434
|
+
@HtmlDsl
|
|
435
|
+
class HTML {
|
|
436
|
+
private val children = mutableListOf<Element>()
|
|
437
|
+
|
|
438
|
+
fun head(init: Head.() -> Unit) {
|
|
439
|
+
children += Head().apply(init)
|
|
440
|
+
}
|
|
441
|
+
|
|
442
|
+
fun body(init: Body.() -> Unit) {
|
|
443
|
+
children += Body().apply(init)
|
|
444
|
+
}
|
|
445
|
+
|
|
446
|
+
override fun toString(): String = children.joinToString("\n")
|
|
447
|
+
}
|
|
448
|
+
|
|
449
|
+
fun html(init: HTML.() -> Unit): HTML = HTML().apply(init)
|
|
450
|
+
|
|
451
|
+
// Usage
|
|
452
|
+
val page = html {
|
|
453
|
+
head { title("My Page") }
|
|
454
|
+
body {
|
|
455
|
+
h1("Welcome")
|
|
456
|
+
p("Hello, World!")
|
|
457
|
+
}
|
|
458
|
+
}
|
|
459
|
+
```
|
|
460
|
+
|
|
461
|
+
### Configuration DSL
|
|
462
|
+
|
|
463
|
+
```kotlin
|
|
464
|
+
data class ServerConfig(
|
|
465
|
+
val host: String = "0.0.0.0",
|
|
466
|
+
val port: Int = 8080,
|
|
467
|
+
val ssl: SslConfig? = null,
|
|
468
|
+
val database: DatabaseConfig? = null,
|
|
469
|
+
)
|
|
470
|
+
|
|
471
|
+
data class SslConfig(val certPath: String, val keyPath: String)
|
|
472
|
+
data class DatabaseConfig(val url: String, val maxPoolSize: Int = 10)
|
|
473
|
+
|
|
474
|
+
class ServerConfigBuilder {
|
|
475
|
+
var host: String = "0.0.0.0"
|
|
476
|
+
var port: Int = 8080
|
|
477
|
+
private var ssl: SslConfig? = null
|
|
478
|
+
private var database: DatabaseConfig? = null
|
|
479
|
+
|
|
480
|
+
fun ssl(certPath: String, keyPath: String) {
|
|
481
|
+
ssl = SslConfig(certPath, keyPath)
|
|
482
|
+
}
|
|
483
|
+
|
|
484
|
+
fun database(url: String, maxPoolSize: Int = 10) {
|
|
485
|
+
database = DatabaseConfig(url, maxPoolSize)
|
|
486
|
+
}
|
|
487
|
+
|
|
488
|
+
fun build(): ServerConfig = ServerConfig(host, port, ssl, database)
|
|
489
|
+
}
|
|
490
|
+
|
|
491
|
+
fun serverConfig(init: ServerConfigBuilder.() -> Unit): ServerConfig =
|
|
492
|
+
ServerConfigBuilder().apply(init).build()
|
|
493
|
+
|
|
494
|
+
// Usage
|
|
495
|
+
val config = serverConfig {
|
|
496
|
+
host = "0.0.0.0"
|
|
497
|
+
port = 443
|
|
498
|
+
ssl("/certs/cert.pem", "/certs/key.pem")
|
|
499
|
+
database("jdbc:postgresql://localhost:5432/mydb", maxPoolSize = 20)
|
|
500
|
+
}
|
|
501
|
+
```
|
|
502
|
+
|
|
503
|
+
## Sequences for Lazy Evaluation
|
|
504
|
+
|
|
505
|
+
```kotlin
|
|
506
|
+
// Good: Use sequences for large collections with multiple operations
|
|
507
|
+
val result = users.asSequence()
|
|
508
|
+
.filter { it.isActive }
|
|
509
|
+
.map { it.email }
|
|
510
|
+
.filter { it.endsWith("@company.com") }
|
|
511
|
+
.take(10)
|
|
512
|
+
.toList()
|
|
513
|
+
|
|
514
|
+
// Good: Generate infinite sequences
|
|
515
|
+
val fibonacci: Sequence<Long> = sequence {
|
|
516
|
+
var a = 0L
|
|
517
|
+
var b = 1L
|
|
518
|
+
while (true) {
|
|
519
|
+
yield(a)
|
|
520
|
+
val next = a + b
|
|
521
|
+
a = b
|
|
522
|
+
b = next
|
|
523
|
+
}
|
|
524
|
+
}
|
|
525
|
+
|
|
526
|
+
val first20 = fibonacci.take(20).toList()
|
|
527
|
+
```
|
|
528
|
+
|
|
529
|
+
## Gradle Kotlin DSL
|
|
530
|
+
|
|
531
|
+
### build.gradle.kts Configuration
|
|
532
|
+
|
|
533
|
+
```kotlin
|
|
534
|
+
// Check for latest versions: https://kotlinlang.org/docs/releases.html
|
|
535
|
+
plugins {
|
|
536
|
+
kotlin("jvm") version "2.3.10"
|
|
537
|
+
kotlin("plugin.serialization") version "2.3.10"
|
|
538
|
+
id("io.ktor.plugin") version "3.4.0"
|
|
539
|
+
id("org.jetbrains.kotlinx.kover") version "0.9.7"
|
|
540
|
+
id("io.gitlab.arturbosch.detekt") version "1.23.8"
|
|
541
|
+
}
|
|
542
|
+
|
|
543
|
+
group = "com.example"
|
|
544
|
+
version = "1.0.0"
|
|
545
|
+
|
|
546
|
+
kotlin {
|
|
547
|
+
jvmToolchain(21)
|
|
548
|
+
}
|
|
549
|
+
|
|
550
|
+
dependencies {
|
|
551
|
+
// Ktor
|
|
552
|
+
implementation("io.ktor:ktor-server-core:3.4.0")
|
|
553
|
+
implementation("io.ktor:ktor-server-netty:3.4.0")
|
|
554
|
+
implementation("io.ktor:ktor-server-content-negotiation:3.4.0")
|
|
555
|
+
implementation("io.ktor:ktor-serialization-kotlinx-json:3.4.0")
|
|
556
|
+
|
|
557
|
+
// Exposed
|
|
558
|
+
implementation("org.jetbrains.exposed:exposed-core:1.0.0")
|
|
559
|
+
implementation("org.jetbrains.exposed:exposed-dao:1.0.0")
|
|
560
|
+
implementation("org.jetbrains.exposed:exposed-jdbc:1.0.0")
|
|
561
|
+
implementation("org.jetbrains.exposed:exposed-kotlin-datetime:1.0.0")
|
|
562
|
+
|
|
563
|
+
// Koin
|
|
564
|
+
implementation("io.insert-koin:koin-ktor:4.2.0")
|
|
565
|
+
|
|
566
|
+
// Coroutines
|
|
567
|
+
implementation("org.jetbrains.kotlinx:kotlinx-coroutines-core:1.10.2")
|
|
568
|
+
|
|
569
|
+
// Testing
|
|
570
|
+
testImplementation("io.kotest:kotest-runner-junit5:6.1.4")
|
|
571
|
+
testImplementation("io.kotest:kotest-assertions-core:6.1.4")
|
|
572
|
+
testImplementation("io.kotest:kotest-property:6.1.4")
|
|
573
|
+
testImplementation("io.mockk:mockk:1.14.9")
|
|
574
|
+
testImplementation("io.ktor:ktor-server-test-host:3.4.0")
|
|
575
|
+
testImplementation("org.jetbrains.kotlinx:kotlinx-coroutines-test:1.10.2")
|
|
576
|
+
}
|
|
577
|
+
|
|
578
|
+
tasks.withType<Test> {
|
|
579
|
+
useJUnitPlatform()
|
|
580
|
+
}
|
|
581
|
+
|
|
582
|
+
detekt {
|
|
583
|
+
config.setFrom(files("config/detekt/detekt.yml"))
|
|
584
|
+
buildUponDefaultConfig = true
|
|
585
|
+
}
|
|
586
|
+
```
|
|
587
|
+
|
|
588
|
+
## Error Handling Patterns
|
|
589
|
+
|
|
590
|
+
### Result Type for Domain Operations
|
|
591
|
+
|
|
592
|
+
```kotlin
|
|
593
|
+
// Good: Use Kotlin's Result or a custom sealed class
|
|
594
|
+
suspend fun createUser(request: CreateUserRequest): Result<User> = runCatching {
|
|
595
|
+
require(request.name.isNotBlank()) { "Name cannot be blank" }
|
|
596
|
+
require('@' in request.email) { "Invalid email format" }
|
|
597
|
+
|
|
598
|
+
val user = User(
|
|
599
|
+
id = UserId(UUID.randomUUID().toString()),
|
|
600
|
+
name = request.name,
|
|
601
|
+
email = Email(request.email),
|
|
602
|
+
)
|
|
603
|
+
userRepository.save(user)
|
|
604
|
+
user
|
|
605
|
+
}
|
|
606
|
+
|
|
607
|
+
// Good: Chain results
|
|
608
|
+
val displayName = createUser(request)
|
|
609
|
+
.map { it.name }
|
|
610
|
+
.getOrElse { "Unknown" }
|
|
611
|
+
```
|
|
612
|
+
|
|
613
|
+
### require, check, error
|
|
614
|
+
|
|
615
|
+
```kotlin
|
|
616
|
+
// Good: Preconditions with clear messages
|
|
617
|
+
fun withdraw(account: Account, amount: Money): Account {
|
|
618
|
+
require(amount.value > 0) { "Amount must be positive: $amount" }
|
|
619
|
+
check(account.balance >= amount) { "Insufficient balance: ${account.balance} < $amount" }
|
|
620
|
+
|
|
621
|
+
return account.copy(balance = account.balance - amount)
|
|
622
|
+
}
|
|
623
|
+
```
|
|
624
|
+
|
|
625
|
+
## Collection Operations
|
|
626
|
+
|
|
627
|
+
### Idiomatic Collection Processing
|
|
628
|
+
|
|
629
|
+
```kotlin
|
|
630
|
+
// Good: Chained operations
|
|
631
|
+
val activeAdminEmails: List<String> = users
|
|
632
|
+
.filter { it.role == Role.ADMIN && it.isActive }
|
|
633
|
+
.sortedBy { it.name }
|
|
634
|
+
.map { it.email }
|
|
635
|
+
|
|
636
|
+
// Good: Grouping and aggregation
|
|
637
|
+
val usersByRole: Map<Role, List<User>> = users.groupBy { it.role }
|
|
638
|
+
|
|
639
|
+
val oldestByRole: Map<Role, User?> = users.groupBy { it.role }
|
|
640
|
+
.mapValues { (_, users) -> users.minByOrNull { it.createdAt } }
|
|
641
|
+
|
|
642
|
+
// Good: Associate for map creation
|
|
643
|
+
val usersById: Map<UserId, User> = users.associateBy { it.id }
|
|
644
|
+
|
|
645
|
+
// Good: Partition for splitting
|
|
646
|
+
val (active, inactive) = users.partition { it.isActive }
|
|
647
|
+
```
|
|
648
|
+
|
|
649
|
+
## Quick Reference: Kotlin Idioms
|
|
650
|
+
|
|
651
|
+
| Idiom | Description |
|
|
652
|
+
|-------|-------------|
|
|
653
|
+
| `val` over `var` | Prefer immutable variables |
|
|
654
|
+
| `data class` | For value objects with equals/hashCode/copy |
|
|
655
|
+
| `sealed class/interface` | For restricted type hierarchies |
|
|
656
|
+
| `value class` | For type-safe wrappers with zero overhead |
|
|
657
|
+
| Expression `when` | Exhaustive pattern matching |
|
|
658
|
+
| Safe call `?.` | Null-safe member access |
|
|
659
|
+
| Elvis `?:` | Default value for nullables |
|
|
660
|
+
| `let`/`apply`/`also`/`run`/`with` | Scope functions for clean code |
|
|
661
|
+
| Extension functions | Add behavior without inheritance |
|
|
662
|
+
| `copy()` | Immutable updates on data classes |
|
|
663
|
+
| `require`/`check` | Precondition assertions |
|
|
664
|
+
| Coroutine `async`/`await` | Structured concurrent execution |
|
|
665
|
+
| `Flow` | Cold reactive streams |
|
|
666
|
+
| `sequence` | Lazy evaluation |
|
|
667
|
+
| Delegation `by` | Reuse implementation without inheritance |
|
|
668
|
+
|
|
669
|
+
## Anti-Patterns to Avoid
|
|
670
|
+
|
|
671
|
+
```kotlin
|
|
672
|
+
// Bad: Force-unwrapping nullable types
|
|
673
|
+
val name = user!!.name
|
|
674
|
+
|
|
675
|
+
// Bad: Platform type leakage from Java
|
|
676
|
+
fun getLength(s: String) = s.length // Safe
|
|
677
|
+
fun getLength(s: String?) = s?.length ?: 0 // Handle nulls from Java
|
|
678
|
+
|
|
679
|
+
// Bad: Mutable data classes
|
|
680
|
+
data class MutableUser(var name: String, var email: String)
|
|
681
|
+
|
|
682
|
+
// Bad: Using exceptions for control flow
|
|
683
|
+
try {
|
|
684
|
+
val user = findUser(id)
|
|
685
|
+
} catch (e: NotFoundException) {
|
|
686
|
+
// Don't use exceptions for expected cases
|
|
687
|
+
}
|
|
688
|
+
|
|
689
|
+
// Good: Use nullable return or Result
|
|
690
|
+
val user: User? = findUserOrNull(id)
|
|
691
|
+
|
|
692
|
+
// Bad: Ignoring coroutine scope
|
|
693
|
+
GlobalScope.launch { /* Avoid GlobalScope */ }
|
|
694
|
+
|
|
695
|
+
// Good: Use structured concurrency
|
|
696
|
+
coroutineScope {
|
|
697
|
+
launch { /* Properly scoped */ }
|
|
698
|
+
}
|
|
699
|
+
|
|
700
|
+
// Bad: Deeply nested scope functions
|
|
701
|
+
user?.let { u ->
|
|
702
|
+
u.address?.let { a ->
|
|
703
|
+
a.city?.let { c -> process(c) }
|
|
704
|
+
}
|
|
705
|
+
}
|
|
706
|
+
|
|
707
|
+
// Good: Direct null-safe chain
|
|
708
|
+
user?.address?.city?.let { process(it) }
|
|
709
|
+
```
|
|
710
|
+
|
|
711
|
+
**Remember**: Kotlin code should be concise but readable. Leverage the type system for safety, prefer immutability, and use coroutines for concurrency. When in doubt, let the compiler help you.
|