@heytherevibin/skillforge 0.2.1

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (402) hide show
  1. package/CHANGELOG.md +16 -0
  2. package/CODE_OF_CONDUCT.md +34 -0
  3. package/CONTRIBUTING.md +38 -0
  4. package/LICENSE +21 -0
  5. package/README.md +337 -0
  6. package/RELEASING.md +93 -0
  7. package/SECURITY.md +31 -0
  8. package/STRATEGY.md +26 -0
  9. package/bin/cli.js +547 -0
  10. package/lib/packs.js +184 -0
  11. package/package.json +38 -0
  12. package/python/app/__init__.py +0 -0
  13. package/python/app/__pycache__/__init__.cpython-312.pyc +0 -0
  14. package/python/app/__pycache__/auth.cpython-312.pyc +0 -0
  15. package/python/app/__pycache__/main.cpython-312.pyc +0 -0
  16. package/python/app/auth.py +63 -0
  17. package/python/app/cli.py +78 -0
  18. package/python/app/db_paths.py +26 -0
  19. package/python/app/events_cli.py +175 -0
  20. package/python/app/main.py +647 -0
  21. package/python/app/materialize.py +138 -0
  22. package/python/app/mcp_server.py +610 -0
  23. package/python/app/route_cli.py +117 -0
  24. package/python/requirements-dev.txt +1 -0
  25. package/python/requirements.txt +7 -0
  26. package/python/tests/test_db_paths.py +41 -0
  27. package/skills/accessibility/SKILL.md +145 -0
  28. package/skills/agent-architecture-audit/SKILL.md +256 -0
  29. package/skills/agent-eval/SKILL.md +144 -0
  30. package/skills/agent-harness-construction/SKILL.md +72 -0
  31. package/skills/agent-introspection-debugging/SKILL.md +152 -0
  32. package/skills/agent-payment-x402/SKILL.md +224 -0
  33. package/skills/agent-sort/SKILL.md +214 -0
  34. package/skills/agentic-engineering/SKILL.md +62 -0
  35. package/skills/agentic-os/SKILL.md +386 -0
  36. package/skills/ai-first-engineering/SKILL.md +50 -0
  37. package/skills/ai-regression-testing/SKILL.md +384 -0
  38. package/skills/android-clean-architecture/SKILL.md +338 -0
  39. package/skills/angular-developer/SKILL.md +153 -0
  40. package/skills/angular-developer/references/angular-animations.md +160 -0
  41. package/skills/angular-developer/references/angular-aria.md +410 -0
  42. package/skills/angular-developer/references/cli.md +86 -0
  43. package/skills/angular-developer/references/component-harnesses.md +59 -0
  44. package/skills/angular-developer/references/component-styling.md +91 -0
  45. package/skills/angular-developer/references/components.md +117 -0
  46. package/skills/angular-developer/references/creating-services.md +97 -0
  47. package/skills/angular-developer/references/data-resolvers.md +69 -0
  48. package/skills/angular-developer/references/define-routes.md +67 -0
  49. package/skills/angular-developer/references/defining-providers.md +72 -0
  50. package/skills/angular-developer/references/di-fundamentals.md +120 -0
  51. package/skills/angular-developer/references/e2e-testing.md +56 -0
  52. package/skills/angular-developer/references/effects.md +83 -0
  53. package/skills/angular-developer/references/hierarchical-injectors.md +43 -0
  54. package/skills/angular-developer/references/host-elements.md +80 -0
  55. package/skills/angular-developer/references/injection-context.md +63 -0
  56. package/skills/angular-developer/references/inputs.md +101 -0
  57. package/skills/angular-developer/references/linked-signal.md +59 -0
  58. package/skills/angular-developer/references/loading-strategies.md +61 -0
  59. package/skills/angular-developer/references/mcp.md +108 -0
  60. package/skills/angular-developer/references/navigate-to-routes.md +69 -0
  61. package/skills/angular-developer/references/outputs.md +86 -0
  62. package/skills/angular-developer/references/reactive-forms.md +122 -0
  63. package/skills/angular-developer/references/rendering-strategies.md +44 -0
  64. package/skills/angular-developer/references/resource.md +77 -0
  65. package/skills/angular-developer/references/route-animations.md +56 -0
  66. package/skills/angular-developer/references/route-guards.md +52 -0
  67. package/skills/angular-developer/references/router-lifecycle.md +45 -0
  68. package/skills/angular-developer/references/router-testing.md +87 -0
  69. package/skills/angular-developer/references/show-routes-with-outlets.md +68 -0
  70. package/skills/angular-developer/references/signal-forms.md +795 -0
  71. package/skills/angular-developer/references/signals-overview.md +94 -0
  72. package/skills/angular-developer/references/tailwind-css.md +69 -0
  73. package/skills/angular-developer/references/template-driven-forms.md +114 -0
  74. package/skills/angular-developer/references/testing-fundamentals.md +65 -0
  75. package/skills/api-connector-builder/SKILL.md +120 -0
  76. package/skills/api-design/SKILL.md +522 -0
  77. package/skills/architecture-decision-records/SKILL.md +178 -0
  78. package/skills/article-writing/SKILL.md +78 -0
  79. package/skills/automation-audit-ops/SKILL.md +141 -0
  80. package/skills/autonomous-agent-harness/SKILL.md +272 -0
  81. package/skills/autonomous-loops/SKILL.md +609 -0
  82. package/skills/backend-patterns/SKILL.md +560 -0
  83. package/skills/benchmark/SKILL.md +92 -0
  84. package/skills/blueprint/SKILL.md +104 -0
  85. package/skills/browser-qa/SKILL.md +86 -0
  86. package/skills/bun-runtime/SKILL.md +83 -0
  87. package/skills/canary-watch/SKILL.md +98 -0
  88. package/skills/carrier-relationship-management/SKILL.md +211 -0
  89. package/skills/cisco-ios-patterns/SKILL.md +163 -0
  90. package/skills/ck/SKILL.md +147 -0
  91. package/skills/ck/commands/forget.mjs +44 -0
  92. package/skills/ck/commands/info.mjs +24 -0
  93. package/skills/ck/commands/init.mjs +143 -0
  94. package/skills/ck/commands/list.mjs +40 -0
  95. package/skills/ck/commands/migrate.mjs +202 -0
  96. package/skills/ck/commands/resume.mjs +36 -0
  97. package/skills/ck/commands/save.mjs +210 -0
  98. package/skills/ck/commands/shared.mjs +387 -0
  99. package/skills/ck/hooks/session-start.mjs +224 -0
  100. package/skills/claude-devfleet/SKILL.md +103 -0
  101. package/skills/click-path-audit/SKILL.md +244 -0
  102. package/skills/clickhouse-io/SKILL.md +438 -0
  103. package/skills/code-tour/SKILL.md +235 -0
  104. package/skills/codebase-onboarding/SKILL.md +232 -0
  105. package/skills/coding-standards/SKILL.md +548 -0
  106. package/skills/compose-multiplatform-patterns/SKILL.md +298 -0
  107. package/skills/connections-optimizer/SKILL.md +188 -0
  108. package/skills/content-engine/SKILL.md +126 -0
  109. package/skills/content-hash-cache-pattern/SKILL.md +160 -0
  110. package/skills/context-budget/SKILL.md +134 -0
  111. package/skills/continuous-agent-loop/SKILL.md +44 -0
  112. package/skills/continuous-learning/SKILL.md +129 -0
  113. package/skills/continuous-learning/config.json +18 -0
  114. package/skills/continuous-learning/evaluate-session.sh +69 -0
  115. package/skills/continuous-learning-v2/SKILL.md +358 -0
  116. package/skills/continuous-learning-v2/agents/observer-loop.sh +322 -0
  117. package/skills/continuous-learning-v2/agents/observer.md +198 -0
  118. package/skills/continuous-learning-v2/agents/session-guardian.sh +150 -0
  119. package/skills/continuous-learning-v2/agents/start-observer.sh +248 -0
  120. package/skills/continuous-learning-v2/config.json +8 -0
  121. package/skills/continuous-learning-v2/hooks/observe.sh +476 -0
  122. package/skills/continuous-learning-v2/scripts/detect-project.sh +288 -0
  123. package/skills/continuous-learning-v2/scripts/instinct-cli.py +1519 -0
  124. package/skills/continuous-learning-v2/scripts/lib/homunculus-dir.sh +31 -0
  125. package/skills/continuous-learning-v2/scripts/migrate-homunculus.sh +62 -0
  126. package/skills/continuous-learning-v2/scripts/test_parse_instinct.py +1018 -0
  127. package/skills/cost-aware-llm-pipeline/SKILL.md +182 -0
  128. package/skills/cost-tracking/SKILL.md +147 -0
  129. package/skills/council/SKILL.md +202 -0
  130. package/skills/cpp-coding-standards/SKILL.md +722 -0
  131. package/skills/cpp-testing/SKILL.md +323 -0
  132. package/skills/crosspost/SKILL.md +110 -0
  133. package/skills/csharp-testing/SKILL.md +320 -0
  134. package/skills/customer-billing-ops/SKILL.md +139 -0
  135. package/skills/customs-trade-compliance/SKILL.md +262 -0
  136. package/skills/dart-flutter-patterns/SKILL.md +562 -0
  137. package/skills/dashboard-builder/SKILL.md +108 -0
  138. package/skills/data-scraper-agent/SKILL.md +764 -0
  139. package/skills/database-migrations/SKILL.md +428 -0
  140. package/skills/deep-research/SKILL.md +158 -0
  141. package/skills/defi-amm-security/SKILL.md +166 -0
  142. package/skills/deployment-patterns/SKILL.md +426 -0
  143. package/skills/design-system/SKILL.md +81 -0
  144. package/skills/django-celery/SKILL.md +456 -0
  145. package/skills/django-patterns/SKILL.md +733 -0
  146. package/skills/django-security/SKILL.md +592 -0
  147. package/skills/django-tdd/SKILL.md +728 -0
  148. package/skills/django-verification/SKILL.md +468 -0
  149. package/skills/dmux-workflows/SKILL.md +190 -0
  150. package/skills/docker-patterns/SKILL.md +363 -0
  151. package/skills/documentation-lookup/SKILL.md +89 -0
  152. package/skills/dotnet-patterns/SKILL.md +320 -0
  153. package/skills/e2e-testing/SKILL.md +325 -0
  154. package/skills/email-ops/SKILL.md +120 -0
  155. package/skills/energy-procurement/SKILL.md +227 -0
  156. package/skills/enterprise-agent-ops/SKILL.md +49 -0
  157. package/skills/error-handling/SKILL.md +375 -0
  158. package/skills/eval-harness/SKILL.md +269 -0
  159. package/skills/evm-token-decimals/SKILL.md +130 -0
  160. package/skills/exa-search/SKILL.md +106 -0
  161. package/skills/fal-ai-media/SKILL.md +287 -0
  162. package/skills/fastapi-patterns/SKILL.md +327 -0
  163. package/skills/finance-billing-ops/SKILL.md +126 -0
  164. package/skills/flox-environments/SKILL.md +496 -0
  165. package/skills/flutter-dart-code-review/SKILL.md +434 -0
  166. package/skills/foundation-models-on-device/SKILL.md +243 -0
  167. package/skills/frontend-design-direction/SKILL.md +92 -0
  168. package/skills/frontend-patterns/SKILL.md +641 -0
  169. package/skills/frontend-slides/SKILL.md +183 -0
  170. package/skills/frontend-slides/STYLE_PRESETS.md +330 -0
  171. package/skills/frontend-slides/animation-patterns.md +122 -0
  172. package/skills/frontend-slides/html-template.md +419 -0
  173. package/skills/frontend-slides/scripts/export-pdf.sh +418 -0
  174. package/skills/frontend-slides/scripts/extract-pptx.py +96 -0
  175. package/skills/frontend-slides/viewport-base.css +153 -0
  176. package/skills/fsharp-testing/SKILL.md +279 -0
  177. package/skills/gan-style-harness/SKILL.md +278 -0
  178. package/skills/gateguard/SKILL.md +125 -0
  179. package/skills/git-workflow/SKILL.md +714 -0
  180. package/skills/github-ops/SKILL.md +143 -0
  181. package/skills/golang-patterns/SKILL.md +673 -0
  182. package/skills/golang-testing/SKILL.md +719 -0
  183. package/skills/google-workspace-ops/SKILL.md +94 -0
  184. package/skills/healthcare-cdss-patterns/SKILL.md +245 -0
  185. package/skills/healthcare-emr-patterns/SKILL.md +159 -0
  186. package/skills/healthcare-eval-harness/SKILL.md +207 -0
  187. package/skills/healthcare-phi-compliance/SKILL.md +145 -0
  188. package/skills/hermes-imports/SKILL.md +87 -0
  189. package/skills/hexagonal-architecture/SKILL.md +275 -0
  190. package/skills/hipaa-compliance/SKILL.md +78 -0
  191. package/skills/homelab-network-readiness/SKILL.md +169 -0
  192. package/skills/homelab-network-setup/SKILL.md +129 -0
  193. package/skills/homelab-pihole-dns/SKILL.md +274 -0
  194. package/skills/homelab-vlan-segmentation/SKILL.md +311 -0
  195. package/skills/homelab-wireguard-vpn/SKILL.md +305 -0
  196. package/skills/hookify-rules/SKILL.md +128 -0
  197. package/skills/inventory-demand-planning/SKILL.md +246 -0
  198. package/skills/investor-materials/SKILL.md +95 -0
  199. package/skills/investor-outreach/SKILL.md +90 -0
  200. package/skills/ios-icon-gen/SKILL.md +157 -0
  201. package/skills/ios-icon-gen/scripts/generate_icons.swift +258 -0
  202. package/skills/ios-icon-gen/scripts/iconify_gen.sh +235 -0
  203. package/skills/iterative-retrieval/SKILL.md +209 -0
  204. package/skills/java-coding-standards/SKILL.md +382 -0
  205. package/skills/jira-integration/SKILL.md +292 -0
  206. package/skills/jpa-patterns/SKILL.md +150 -0
  207. package/skills/knowledge-ops/SKILL.md +153 -0
  208. package/skills/kotlin-coroutines-flows/SKILL.md +283 -0
  209. package/skills/kotlin-exposed-patterns/SKILL.md +718 -0
  210. package/skills/kotlin-ktor-patterns/SKILL.md +688 -0
  211. package/skills/kotlin-patterns/SKILL.md +710 -0
  212. package/skills/kotlin-testing/SKILL.md +823 -0
  213. package/skills/laravel-patterns/SKILL.md +414 -0
  214. package/skills/laravel-plugin-discovery/SKILL.md +228 -0
  215. package/skills/laravel-security/SKILL.md +284 -0
  216. package/skills/laravel-tdd/SKILL.md +282 -0
  217. package/skills/laravel-verification/SKILL.md +178 -0
  218. package/skills/lead-intelligence/SKILL.md +320 -0
  219. package/skills/lead-intelligence/agents/enrichment-agent.md +85 -0
  220. package/skills/lead-intelligence/agents/mutual-mapper.md +75 -0
  221. package/skills/lead-intelligence/agents/outreach-drafter.md +98 -0
  222. package/skills/lead-intelligence/agents/signal-scorer.md +60 -0
  223. package/skills/liquid-glass-design/SKILL.md +279 -0
  224. package/skills/llm-trading-agent-security/SKILL.md +146 -0
  225. package/skills/logistics-exception-management/SKILL.md +221 -0
  226. package/skills/make-interfaces-feel-better/SKILL.md +151 -0
  227. package/skills/manim-video/SKILL.md +88 -0
  228. package/skills/manim-video/assets/network_graph_scene.py +52 -0
  229. package/skills/market-research/SKILL.md +74 -0
  230. package/skills/mcp-server-patterns/SKILL.md +68 -0
  231. package/skills/messages-ops/SKILL.md +103 -0
  232. package/skills/mle-workflow/SKILL.md +345 -0
  233. package/skills/motion-advanced/SKILL.md +596 -0
  234. package/skills/motion-foundations/SKILL.md +299 -0
  235. package/skills/motion-patterns/SKILL.md +435 -0
  236. package/skills/motion-ui/SKILL.md +574 -0
  237. package/skills/mysql-patterns/SKILL.md +411 -0
  238. package/skills/nanoclaw-repl/SKILL.md +32 -0
  239. package/skills/nestjs-patterns/SKILL.md +229 -0
  240. package/skills/netmiko-ssh-automation/SKILL.md +173 -0
  241. package/skills/network-bgp-diagnostics/SKILL.md +167 -0
  242. package/skills/network-config-validation/SKILL.md +210 -0
  243. package/skills/network-interface-health/SKILL.md +152 -0
  244. package/skills/nextjs-turbopack/SKILL.md +43 -0
  245. package/skills/nodejs-keccak256/SKILL.md +102 -0
  246. package/skills/nutrient-document-processing/SKILL.md +166 -0
  247. package/skills/nuxt4-patterns/SKILL.md +99 -0
  248. package/skills/openclaw-persona-forge/SKILL.md +288 -0
  249. package/skills/openclaw-persona-forge/gacha.py +224 -0
  250. package/skills/openclaw-persona-forge/gacha.sh +5 -0
  251. package/skills/openclaw-persona-forge/references/avatar-style.md +124 -0
  252. package/skills/openclaw-persona-forge/references/boundary-rules.md +53 -0
  253. package/skills/openclaw-persona-forge/references/error-handling.md +53 -0
  254. package/skills/openclaw-persona-forge/references/identity-tension.md +48 -0
  255. package/skills/openclaw-persona-forge/references/naming-system.md +39 -0
  256. package/skills/openclaw-persona-forge/references/output-template.md +166 -0
  257. package/skills/opensource-pipeline/SKILL.md +254 -0
  258. package/skills/perl-patterns/SKILL.md +503 -0
  259. package/skills/perl-security/SKILL.md +502 -0
  260. package/skills/perl-testing/SKILL.md +474 -0
  261. package/skills/plan-orchestrate/SKILL.md +253 -0
  262. package/skills/plankton-code-quality/SKILL.md +236 -0
  263. package/skills/postgres-patterns/SKILL.md +146 -0
  264. package/skills/product-capability/SKILL.md +140 -0
  265. package/skills/product-lens/SKILL.md +91 -0
  266. package/skills/production-audit/SKILL.md +206 -0
  267. package/skills/production-scheduling/SKILL.md +237 -0
  268. package/skills/project-flow-ops/SKILL.md +110 -0
  269. package/skills/prompt-optimizer/SKILL.md +398 -0
  270. package/skills/python-patterns/SKILL.md +749 -0
  271. package/skills/python-testing/SKILL.md +815 -0
  272. package/skills/pytorch-patterns/SKILL.md +395 -0
  273. package/skills/quality-nonconformance/SKILL.md +259 -0
  274. package/skills/quarkus-patterns/SKILL.md +721 -0
  275. package/skills/quarkus-security/SKILL.md +466 -0
  276. package/skills/quarkus-tdd/SKILL.md +810 -0
  277. package/skills/quarkus-verification/SKILL.md +478 -0
  278. package/skills/ralphinho-rfc-pipeline/SKILL.md +66 -0
  279. package/skills/redis-patterns/SKILL.md +402 -0
  280. package/skills/regex-vs-llm-structured-text/SKILL.md +219 -0
  281. package/skills/remotion-video-creation/SKILL.md +43 -0
  282. package/skills/remotion-video-creation/rules/3d.md +86 -0
  283. package/skills/remotion-video-creation/rules/animations.md +29 -0
  284. package/skills/remotion-video-creation/rules/assets/charts-bar-chart.tsx +173 -0
  285. package/skills/remotion-video-creation/rules/assets/text-animations-typewriter.tsx +100 -0
  286. package/skills/remotion-video-creation/rules/assets/text-animations-word-highlight.tsx +108 -0
  287. package/skills/remotion-video-creation/rules/assets.md +78 -0
  288. package/skills/remotion-video-creation/rules/audio.md +172 -0
  289. package/skills/remotion-video-creation/rules/calculate-metadata.md +104 -0
  290. package/skills/remotion-video-creation/rules/can-decode.md +75 -0
  291. package/skills/remotion-video-creation/rules/charts.md +58 -0
  292. package/skills/remotion-video-creation/rules/compositions.md +146 -0
  293. package/skills/remotion-video-creation/rules/display-captions.md +126 -0
  294. package/skills/remotion-video-creation/rules/extract-frames.md +229 -0
  295. package/skills/remotion-video-creation/rules/fonts.md +152 -0
  296. package/skills/remotion-video-creation/rules/get-audio-duration.md +58 -0
  297. package/skills/remotion-video-creation/rules/get-video-dimensions.md +68 -0
  298. package/skills/remotion-video-creation/rules/get-video-duration.md +58 -0
  299. package/skills/remotion-video-creation/rules/gifs.md +138 -0
  300. package/skills/remotion-video-creation/rules/images.md +130 -0
  301. package/skills/remotion-video-creation/rules/import-srt-captions.md +67 -0
  302. package/skills/remotion-video-creation/rules/lottie.md +67 -0
  303. package/skills/remotion-video-creation/rules/measuring-dom-nodes.md +34 -0
  304. package/skills/remotion-video-creation/rules/measuring-text.md +143 -0
  305. package/skills/remotion-video-creation/rules/sequencing.md +106 -0
  306. package/skills/remotion-video-creation/rules/tailwind.md +11 -0
  307. package/skills/remotion-video-creation/rules/text-animations.md +20 -0
  308. package/skills/remotion-video-creation/rules/timing.md +179 -0
  309. package/skills/remotion-video-creation/rules/transcribe-captions.md +19 -0
  310. package/skills/remotion-video-creation/rules/transitions.md +122 -0
  311. package/skills/remotion-video-creation/rules/trimming.md +52 -0
  312. package/skills/remotion-video-creation/rules/videos.md +171 -0
  313. package/skills/repo-scan/SKILL.md +78 -0
  314. package/skills/research-ops/SKILL.md +111 -0
  315. package/skills/returns-reverse-logistics/SKILL.md +239 -0
  316. package/skills/rules-distill/SKILL.md +263 -0
  317. package/skills/rules-distill/scripts/scan-rules.sh +58 -0
  318. package/skills/rules-distill/scripts/scan-skills.sh +129 -0
  319. package/skills/rust-patterns/SKILL.md +498 -0
  320. package/skills/rust-testing/SKILL.md +499 -0
  321. package/skills/safety-guard/SKILL.md +74 -0
  322. package/skills/santa-method/SKILL.md +306 -0
  323. package/skills/scientific-db-pubmed-database/SKILL.md +175 -0
  324. package/skills/scientific-db-uspto-database/SKILL.md +177 -0
  325. package/skills/scientific-pkg-gget/SKILL.md +166 -0
  326. package/skills/scientific-thinking-literature-review/SKILL.md +192 -0
  327. package/skills/scientific-thinking-scholar-evaluation/SKILL.md +160 -0
  328. package/skills/search-first/SKILL.md +181 -0
  329. package/skills/security-bounty-hunter/SKILL.md +99 -0
  330. package/skills/security-review/SKILL.md +502 -0
  331. package/skills/security-review/cloud-infrastructure-security.md +361 -0
  332. package/skills/seo/SKILL.md +153 -0
  333. package/skills/skill-comply/SKILL.md +57 -0
  334. package/skills/skill-comply/fixtures/compliant_trace.jsonl +5 -0
  335. package/skills/skill-comply/fixtures/noncompliant_trace.jsonl +3 -0
  336. package/skills/skill-comply/fixtures/tdd_spec.yaml +44 -0
  337. package/skills/skill-comply/prompts/classifier.md +24 -0
  338. package/skills/skill-comply/prompts/scenario_generator.md +62 -0
  339. package/skills/skill-comply/prompts/spec_generator.md +42 -0
  340. package/skills/skill-comply/pyproject.toml +15 -0
  341. package/skills/skill-comply/scripts/__init__.py +0 -0
  342. package/skills/skill-comply/scripts/classifier.py +85 -0
  343. package/skills/skill-comply/scripts/grader.py +124 -0
  344. package/skills/skill-comply/scripts/parser.py +107 -0
  345. package/skills/skill-comply/scripts/report.py +170 -0
  346. package/skills/skill-comply/scripts/run.py +127 -0
  347. package/skills/skill-comply/scripts/runner.py +186 -0
  348. package/skills/skill-comply/scripts/scenario_generator.py +70 -0
  349. package/skills/skill-comply/scripts/spec_generator.py +72 -0
  350. package/skills/skill-comply/scripts/utils.py +13 -0
  351. package/skills/skill-comply/tests/test_grader.py +197 -0
  352. package/skills/skill-comply/tests/test_parser.py +90 -0
  353. package/skills/skill-comply/tests/test_runner.py +172 -0
  354. package/skills/skill-scout/SKILL.md +139 -0
  355. package/skills/skill-stocktake/SKILL.md +193 -0
  356. package/skills/skill-stocktake/scripts/quick-diff.sh +87 -0
  357. package/skills/skill-stocktake/scripts/save-results.sh +56 -0
  358. package/skills/skill-stocktake/scripts/scan.sh +170 -0
  359. package/skills/social-graph-ranker/SKILL.md +153 -0
  360. package/skills/springboot-patterns/SKILL.md +313 -0
  361. package/skills/springboot-security/SKILL.md +271 -0
  362. package/skills/springboot-tdd/SKILL.md +157 -0
  363. package/skills/springboot-verification/SKILL.md +230 -0
  364. package/skills/strategic-compact/SKILL.md +129 -0
  365. package/skills/strategic-compact/suggest-compact.sh +54 -0
  366. package/skills/swift-actor-persistence/SKILL.md +142 -0
  367. package/skills/swift-concurrency-6-2/SKILL.md +216 -0
  368. package/skills/swift-protocol-di-testing/SKILL.md +189 -0
  369. package/skills/swiftui-patterns/SKILL.md +259 -0
  370. package/skills/tdd-workflow/SKILL.md +462 -0
  371. package/skills/team-builder/SKILL.md +166 -0
  372. package/skills/terminal-ops/SKILL.md +108 -0
  373. package/skills/tinystruct-patterns/SKILL.md +130 -0
  374. package/skills/tinystruct-patterns/references/architecture.md +77 -0
  375. package/skills/tinystruct-patterns/references/data-handling.md +35 -0
  376. package/skills/tinystruct-patterns/references/routing.md +57 -0
  377. package/skills/tinystruct-patterns/references/system-usage.md +74 -0
  378. package/skills/tinystruct-patterns/references/testing.md +59 -0
  379. package/skills/token-budget-advisor/SKILL.md +133 -0
  380. package/skills/ui-demo/SKILL.md +464 -0
  381. package/skills/ui-to-vue/SKILL.md +134 -0
  382. package/skills/unified-notifications-ops/SKILL.md +186 -0
  383. package/skills/verification-loop/SKILL.md +125 -0
  384. package/skills/video-editing/SKILL.md +309 -0
  385. package/skills/videodb/SKILL.md +373 -0
  386. package/skills/videodb/reference/api-reference.md +550 -0
  387. package/skills/videodb/reference/capture-reference.md +407 -0
  388. package/skills/videodb/reference/capture.md +101 -0
  389. package/skills/videodb/reference/editor.md +443 -0
  390. package/skills/videodb/reference/generative.md +331 -0
  391. package/skills/videodb/reference/rtstream-reference.md +564 -0
  392. package/skills/videodb/reference/rtstream.md +65 -0
  393. package/skills/videodb/reference/search.md +230 -0
  394. package/skills/videodb/reference/streaming.md +406 -0
  395. package/skills/videodb/reference/use-cases.md +118 -0
  396. package/skills/videodb/scripts/ws_listener.py +282 -0
  397. package/skills/visa-doc-translate/README.md +86 -0
  398. package/skills/visa-doc-translate/SKILL.md +117 -0
  399. package/skills/vite-patterns/SKILL.md +448 -0
  400. package/skills/windows-desktop-e2e/SKILL.md +787 -0
  401. package/skills/workspace-surface-audit/SKILL.md +124 -0
  402. package/skills/x-api/SKILL.md +233 -0
@@ -0,0 +1,710 @@
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
+ ---
5
+
6
+ # Kotlin Development Patterns
7
+
8
+ Idiomatic Kotlin patterns and best practices for building robust, efficient, and maintainable applications.
9
+
10
+ ## When to Use
11
+
12
+ - Writing new Kotlin code
13
+ - Reviewing Kotlin code
14
+ - Refactoring existing Kotlin code
15
+ - Designing Kotlin modules or libraries
16
+ - Configuring Gradle Kotlin DSL builds
17
+
18
+ ## How It Works
19
+
20
+ 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.
21
+
22
+ ## Examples
23
+
24
+ **Null safety with Elvis operator:**
25
+ ```kotlin
26
+ fun getUserEmail(userId: String): String {
27
+ val user = userRepository.findById(userId)
28
+ return user?.email ?: "unknown@example.com"
29
+ }
30
+ ```
31
+
32
+ **Sealed class for exhaustive results:**
33
+ ```kotlin
34
+ sealed class Result<out T> {
35
+ data class Success<T>(val data: T) : Result<T>()
36
+ data class Failure(val error: AppError) : Result<Nothing>()
37
+ data object Loading : Result<Nothing>()
38
+ }
39
+ ```
40
+
41
+ **Structured concurrency with async/await:**
42
+ ```kotlin
43
+ suspend fun fetchUserWithPosts(userId: String): UserProfile =
44
+ coroutineScope {
45
+ val user = async { userService.getUser(userId) }
46
+ val posts = async { postService.getUserPosts(userId) }
47
+ UserProfile(user = user.await(), posts = posts.await())
48
+ }
49
+ ```
50
+
51
+ ## Core Principles
52
+
53
+ ### 1. Null Safety
54
+
55
+ Kotlin's type system distinguishes nullable and non-nullable types. Leverage it fully.
56
+
57
+ ```kotlin
58
+ // Good: Use non-nullable types by default
59
+ fun getUser(id: String): User {
60
+ return userRepository.findById(id)
61
+ ?: throw UserNotFoundException("User $id not found")
62
+ }
63
+
64
+ // Good: Safe calls and Elvis operator
65
+ fun getUserEmail(userId: String): String {
66
+ val user = userRepository.findById(userId)
67
+ return user?.email ?: "unknown@example.com"
68
+ }
69
+
70
+ // Bad: Force-unwrapping nullable types
71
+ fun getUserEmail(userId: String): String {
72
+ val user = userRepository.findById(userId)
73
+ return user!!.email // Throws NPE if null
74
+ }
75
+ ```
76
+
77
+ ### 2. Immutability by Default
78
+
79
+ Prefer `val` over `var`, immutable collections over mutable ones.
80
+
81
+ ```kotlin
82
+ // Good: Immutable data
83
+ data class User(
84
+ val id: String,
85
+ val name: String,
86
+ val email: String,
87
+ )
88
+
89
+ // Good: Transform with copy()
90
+ fun updateEmail(user: User, newEmail: String): User =
91
+ user.copy(email = newEmail)
92
+
93
+ // Good: Immutable collections
94
+ val users: List<User> = listOf(user1, user2)
95
+ val filtered = users.filter { it.email.isNotBlank() }
96
+
97
+ // Bad: Mutable state
98
+ var currentUser: User? = null // Avoid mutable global state
99
+ val mutableUsers = mutableListOf<User>() // Avoid unless truly needed
100
+ ```
101
+
102
+ ### 3. Expression Bodies and Single-Expression Functions
103
+
104
+ Use expression bodies for concise, readable functions.
105
+
106
+ ```kotlin
107
+ // Good: Expression body
108
+ fun isAdult(age: Int): Boolean = age >= 18
109
+
110
+ fun formatFullName(first: String, last: String): String =
111
+ "$first $last".trim()
112
+
113
+ fun User.displayName(): String =
114
+ name.ifBlank { email.substringBefore('@') }
115
+
116
+ // Good: When as expression
117
+ fun statusMessage(code: Int): String = when (code) {
118
+ 200 -> "OK"
119
+ 404 -> "Not Found"
120
+ 500 -> "Internal Server Error"
121
+ else -> "Unknown status: $code"
122
+ }
123
+
124
+ // Bad: Unnecessary block body
125
+ fun isAdult(age: Int): Boolean {
126
+ return age >= 18
127
+ }
128
+ ```
129
+
130
+ ### 4. Data Classes for Value Objects
131
+
132
+ Use data classes for types that primarily hold data.
133
+
134
+ ```kotlin
135
+ // Good: Data class with copy, equals, hashCode, toString
136
+ data class CreateUserRequest(
137
+ val name: String,
138
+ val email: String,
139
+ val role: Role = Role.USER,
140
+ )
141
+
142
+ // Good: Value class for type safety (zero overhead at runtime)
143
+ @JvmInline
144
+ value class UserId(val value: String) {
145
+ init {
146
+ require(value.isNotBlank()) { "UserId cannot be blank" }
147
+ }
148
+ }
149
+
150
+ @JvmInline
151
+ value class Email(val value: String) {
152
+ init {
153
+ require('@' in value) { "Invalid email: $value" }
154
+ }
155
+ }
156
+
157
+ fun getUser(id: UserId): User = userRepository.findById(id)
158
+ ```
159
+
160
+ ## Sealed Classes and Interfaces
161
+
162
+ ### Modeling Restricted Hierarchies
163
+
164
+ ```kotlin
165
+ // Good: Sealed class for exhaustive when
166
+ sealed class Result<out T> {
167
+ data class Success<T>(val data: T) : Result<T>()
168
+ data class Failure(val error: AppError) : Result<Nothing>()
169
+ data object Loading : Result<Nothing>()
170
+ }
171
+
172
+ fun <T> Result<T>.getOrNull(): T? = when (this) {
173
+ is Result.Success -> data
174
+ is Result.Failure -> null
175
+ is Result.Loading -> null
176
+ }
177
+
178
+ fun <T> Result<T>.getOrThrow(): T = when (this) {
179
+ is Result.Success -> data
180
+ is Result.Failure -> throw error.toException()
181
+ is Result.Loading -> throw IllegalStateException("Still loading")
182
+ }
183
+ ```
184
+
185
+ ### Sealed Interfaces for API Responses
186
+
187
+ ```kotlin
188
+ sealed interface ApiError {
189
+ val message: String
190
+
191
+ data class NotFound(override val message: String) : ApiError
192
+ data class Unauthorized(override val message: String) : ApiError
193
+ data class Validation(
194
+ override val message: String,
195
+ val field: String,
196
+ ) : ApiError
197
+ data class Internal(
198
+ override val message: String,
199
+ val cause: Throwable? = null,
200
+ ) : ApiError
201
+ }
202
+
203
+ fun ApiError.toStatusCode(): Int = when (this) {
204
+ is ApiError.NotFound -> 404
205
+ is ApiError.Unauthorized -> 401
206
+ is ApiError.Validation -> 422
207
+ is ApiError.Internal -> 500
208
+ }
209
+ ```
210
+
211
+ ## Scope Functions
212
+
213
+ ### When to Use Each
214
+
215
+ ```kotlin
216
+ // let: Transform nullable or scoped result
217
+ val length: Int? = name?.let { it.trim().length }
218
+
219
+ // apply: Configure an object (returns the object)
220
+ val user = User().apply {
221
+ name = "Alice"
222
+ email = "alice@example.com"
223
+ }
224
+
225
+ // also: Side effects (returns the object)
226
+ val user = createUser(request).also { logger.info("Created user: ${it.id}") }
227
+
228
+ // run: Execute a block with receiver (returns result)
229
+ val result = connection.run {
230
+ prepareStatement(sql)
231
+ executeQuery()
232
+ }
233
+
234
+ // with: Non-extension form of run
235
+ val csv = with(StringBuilder()) {
236
+ appendLine("name,email")
237
+ users.forEach { appendLine("${it.name},${it.email}") }
238
+ toString()
239
+ }
240
+ ```
241
+
242
+ ### Anti-Patterns
243
+
244
+ ```kotlin
245
+ // Bad: Nesting scope functions
246
+ user?.let { u ->
247
+ u.address?.let { addr ->
248
+ addr.city?.let { city ->
249
+ println(city) // Hard to read
250
+ }
251
+ }
252
+ }
253
+
254
+ // Good: Chain safe calls instead
255
+ val city = user?.address?.city
256
+ city?.let { println(it) }
257
+ ```
258
+
259
+ ## Extension Functions
260
+
261
+ ### Adding Functionality Without Inheritance
262
+
263
+ ```kotlin
264
+ // Good: Domain-specific extensions
265
+ fun String.toSlug(): String =
266
+ lowercase()
267
+ .replace(Regex("[^a-z0-9\\s-]"), "")
268
+ .replace(Regex("\\s+"), "-")
269
+ .trim('-')
270
+
271
+ fun Instant.toLocalDate(zone: ZoneId = ZoneId.systemDefault()): LocalDate =
272
+ atZone(zone).toLocalDate()
273
+
274
+ // Good: Collection extensions
275
+ fun <T> List<T>.second(): T = this[1]
276
+
277
+ fun <T> List<T>.secondOrNull(): T? = getOrNull(1)
278
+
279
+ // Good: Scoped extensions (not polluting global namespace)
280
+ class UserService {
281
+ private fun User.isActive(): Boolean =
282
+ status == Status.ACTIVE && lastLogin.isAfter(Instant.now().minus(30, ChronoUnit.DAYS))
283
+
284
+ fun getActiveUsers(): List<User> = userRepository.findAll().filter { it.isActive() }
285
+ }
286
+ ```
287
+
288
+ ## Coroutines
289
+
290
+ ### Structured Concurrency
291
+
292
+ ```kotlin
293
+ // Good: Structured concurrency with coroutineScope
294
+ suspend fun fetchUserWithPosts(userId: String): UserProfile =
295
+ coroutineScope {
296
+ val userDeferred = async { userService.getUser(userId) }
297
+ val postsDeferred = async { postService.getUserPosts(userId) }
298
+
299
+ UserProfile(
300
+ user = userDeferred.await(),
301
+ posts = postsDeferred.await(),
302
+ )
303
+ }
304
+
305
+ // Good: supervisorScope when children can fail independently
306
+ suspend fun fetchDashboard(userId: String): Dashboard =
307
+ supervisorScope {
308
+ val user = async { userService.getUser(userId) }
309
+ val notifications = async { notificationService.getRecent(userId) }
310
+ val recommendations = async { recommendationService.getFor(userId) }
311
+
312
+ Dashboard(
313
+ user = user.await(),
314
+ notifications = try {
315
+ notifications.await()
316
+ } catch (e: CancellationException) {
317
+ throw e
318
+ } catch (e: Exception) {
319
+ emptyList()
320
+ },
321
+ recommendations = try {
322
+ recommendations.await()
323
+ } catch (e: CancellationException) {
324
+ throw e
325
+ } catch (e: Exception) {
326
+ emptyList()
327
+ },
328
+ )
329
+ }
330
+ ```
331
+
332
+ ### Flow for Reactive Streams
333
+
334
+ ```kotlin
335
+ // Good: Cold flow with proper error handling
336
+ fun observeUsers(): Flow<List<User>> = flow {
337
+ while (currentCoroutineContext().isActive) {
338
+ val users = userRepository.findAll()
339
+ emit(users)
340
+ delay(5.seconds)
341
+ }
342
+ }.catch { e ->
343
+ logger.error("Error observing users", e)
344
+ emit(emptyList())
345
+ }
346
+
347
+ // Good: Flow operators
348
+ fun searchUsers(query: Flow<String>): Flow<List<User>> =
349
+ query
350
+ .debounce(300.milliseconds)
351
+ .distinctUntilChanged()
352
+ .filter { it.length >= 2 }
353
+ .mapLatest { q -> userRepository.search(q) }
354
+ .catch { emit(emptyList()) }
355
+ ```
356
+
357
+ ### Cancellation and Cleanup
358
+
359
+ ```kotlin
360
+ // Good: Respect cancellation
361
+ suspend fun processItems(items: List<Item>) {
362
+ items.forEach { item ->
363
+ ensureActive() // Check cancellation before expensive work
364
+ processItem(item)
365
+ }
366
+ }
367
+
368
+ // Good: Cleanup with try/finally
369
+ suspend fun acquireAndProcess() {
370
+ val resource = acquireResource()
371
+ try {
372
+ resource.process()
373
+ } finally {
374
+ withContext(NonCancellable) {
375
+ resource.release() // Always release, even on cancellation
376
+ }
377
+ }
378
+ }
379
+ ```
380
+
381
+ ## Delegation
382
+
383
+ ### Property Delegation
384
+
385
+ ```kotlin
386
+ // Lazy initialization
387
+ val expensiveData: List<User> by lazy {
388
+ userRepository.findAll()
389
+ }
390
+
391
+ // Observable property
392
+ var name: String by Delegates.observable("initial") { _, old, new ->
393
+ logger.info("Name changed from '$old' to '$new'")
394
+ }
395
+
396
+ // Map-backed properties
397
+ class Config(private val map: Map<String, Any?>) {
398
+ val host: String by map
399
+ val port: Int by map
400
+ val debug: Boolean by map
401
+ }
402
+
403
+ val config = Config(mapOf("host" to "localhost", "port" to 8080, "debug" to true))
404
+ ```
405
+
406
+ ### Interface Delegation
407
+
408
+ ```kotlin
409
+ // Good: Delegate interface implementation
410
+ class LoggingUserRepository(
411
+ private val delegate: UserRepository,
412
+ private val logger: Logger,
413
+ ) : UserRepository by delegate {
414
+ // Only override what you need to add logging to
415
+ override suspend fun findById(id: String): User? {
416
+ logger.info("Finding user by id: $id")
417
+ return delegate.findById(id).also {
418
+ logger.info("Found user: ${it?.name ?: "null"}")
419
+ }
420
+ }
421
+ }
422
+ ```
423
+
424
+ ## DSL Builders
425
+
426
+ ### Type-Safe Builders
427
+
428
+ ```kotlin
429
+ // Good: DSL with @DslMarker
430
+ @DslMarker
431
+ annotation class HtmlDsl
432
+
433
+ @HtmlDsl
434
+ class HTML {
435
+ private val children = mutableListOf<Element>()
436
+
437
+ fun head(init: Head.() -> Unit) {
438
+ children += Head().apply(init)
439
+ }
440
+
441
+ fun body(init: Body.() -> Unit) {
442
+ children += Body().apply(init)
443
+ }
444
+
445
+ override fun toString(): String = children.joinToString("\n")
446
+ }
447
+
448
+ fun html(init: HTML.() -> Unit): HTML = HTML().apply(init)
449
+
450
+ // Usage
451
+ val page = html {
452
+ head { title("My Page") }
453
+ body {
454
+ h1("Welcome")
455
+ p("Hello, World!")
456
+ }
457
+ }
458
+ ```
459
+
460
+ ### Configuration DSL
461
+
462
+ ```kotlin
463
+ data class ServerConfig(
464
+ val host: String = "0.0.0.0",
465
+ val port: Int = 8080,
466
+ val ssl: SslConfig? = null,
467
+ val database: DatabaseConfig? = null,
468
+ )
469
+
470
+ data class SslConfig(val certPath: String, val keyPath: String)
471
+ data class DatabaseConfig(val url: String, val maxPoolSize: Int = 10)
472
+
473
+ class ServerConfigBuilder {
474
+ var host: String = "0.0.0.0"
475
+ var port: Int = 8080
476
+ private var ssl: SslConfig? = null
477
+ private var database: DatabaseConfig? = null
478
+
479
+ fun ssl(certPath: String, keyPath: String) {
480
+ ssl = SslConfig(certPath, keyPath)
481
+ }
482
+
483
+ fun database(url: String, maxPoolSize: Int = 10) {
484
+ database = DatabaseConfig(url, maxPoolSize)
485
+ }
486
+
487
+ fun build(): ServerConfig = ServerConfig(host, port, ssl, database)
488
+ }
489
+
490
+ fun serverConfig(init: ServerConfigBuilder.() -> Unit): ServerConfig =
491
+ ServerConfigBuilder().apply(init).build()
492
+
493
+ // Usage
494
+ val config = serverConfig {
495
+ host = "0.0.0.0"
496
+ port = 443
497
+ ssl("/certs/cert.pem", "/certs/key.pem")
498
+ database("jdbc:postgresql://localhost:5432/mydb", maxPoolSize = 20)
499
+ }
500
+ ```
501
+
502
+ ## Sequences for Lazy Evaluation
503
+
504
+ ```kotlin
505
+ // Good: Use sequences for large collections with multiple operations
506
+ val result = users.asSequence()
507
+ .filter { it.isActive }
508
+ .map { it.email }
509
+ .filter { it.endsWith("@company.com") }
510
+ .take(10)
511
+ .toList()
512
+
513
+ // Good: Generate infinite sequences
514
+ val fibonacci: Sequence<Long> = sequence {
515
+ var a = 0L
516
+ var b = 1L
517
+ while (true) {
518
+ yield(a)
519
+ val next = a + b
520
+ a = b
521
+ b = next
522
+ }
523
+ }
524
+
525
+ val first20 = fibonacci.take(20).toList()
526
+ ```
527
+
528
+ ## Gradle Kotlin DSL
529
+
530
+ ### build.gradle.kts Configuration
531
+
532
+ ```kotlin
533
+ // Check for latest versions: https://kotlinlang.org/docs/releases.html
534
+ plugins {
535
+ kotlin("jvm") version "2.3.10"
536
+ kotlin("plugin.serialization") version "2.3.10"
537
+ id("io.ktor.plugin") version "3.4.0"
538
+ id("org.jetbrains.kotlinx.kover") version "0.9.7"
539
+ id("io.gitlab.arturbosch.detekt") version "1.23.8"
540
+ }
541
+
542
+ group = "com.example"
543
+ version = "1.0.0"
544
+
545
+ kotlin {
546
+ jvmToolchain(21)
547
+ }
548
+
549
+ dependencies {
550
+ // Ktor
551
+ implementation("io.ktor:ktor-server-core:3.4.0")
552
+ implementation("io.ktor:ktor-server-netty:3.4.0")
553
+ implementation("io.ktor:ktor-server-content-negotiation:3.4.0")
554
+ implementation("io.ktor:ktor-serialization-kotlinx-json:3.4.0")
555
+
556
+ // Exposed
557
+ implementation("org.jetbrains.exposed:exposed-core:1.0.0")
558
+ implementation("org.jetbrains.exposed:exposed-dao:1.0.0")
559
+ implementation("org.jetbrains.exposed:exposed-jdbc:1.0.0")
560
+ implementation("org.jetbrains.exposed:exposed-kotlin-datetime:1.0.0")
561
+
562
+ // Koin
563
+ implementation("io.insert-koin:koin-ktor:4.2.0")
564
+
565
+ // Coroutines
566
+ implementation("org.jetbrains.kotlinx:kotlinx-coroutines-core:1.10.2")
567
+
568
+ // Testing
569
+ testImplementation("io.kotest:kotest-runner-junit5:6.1.4")
570
+ testImplementation("io.kotest:kotest-assertions-core:6.1.4")
571
+ testImplementation("io.kotest:kotest-property:6.1.4")
572
+ testImplementation("io.mockk:mockk:1.14.9")
573
+ testImplementation("io.ktor:ktor-server-test-host:3.4.0")
574
+ testImplementation("org.jetbrains.kotlinx:kotlinx-coroutines-test:1.10.2")
575
+ }
576
+
577
+ tasks.withType<Test> {
578
+ useJUnitPlatform()
579
+ }
580
+
581
+ detekt {
582
+ config.setFrom(files("config/detekt/detekt.yml"))
583
+ buildUponDefaultConfig = true
584
+ }
585
+ ```
586
+
587
+ ## Error Handling Patterns
588
+
589
+ ### Result Type for Domain Operations
590
+
591
+ ```kotlin
592
+ // Good: Use Kotlin's Result or a custom sealed class
593
+ suspend fun createUser(request: CreateUserRequest): Result<User> = runCatching {
594
+ require(request.name.isNotBlank()) { "Name cannot be blank" }
595
+ require('@' in request.email) { "Invalid email format" }
596
+
597
+ val user = User(
598
+ id = UserId(UUID.randomUUID().toString()),
599
+ name = request.name,
600
+ email = Email(request.email),
601
+ )
602
+ userRepository.save(user)
603
+ user
604
+ }
605
+
606
+ // Good: Chain results
607
+ val displayName = createUser(request)
608
+ .map { it.name }
609
+ .getOrElse { "Unknown" }
610
+ ```
611
+
612
+ ### require, check, error
613
+
614
+ ```kotlin
615
+ // Good: Preconditions with clear messages
616
+ fun withdraw(account: Account, amount: Money): Account {
617
+ require(amount.value > 0) { "Amount must be positive: $amount" }
618
+ check(account.balance >= amount) { "Insufficient balance: ${account.balance} < $amount" }
619
+
620
+ return account.copy(balance = account.balance - amount)
621
+ }
622
+ ```
623
+
624
+ ## Collection Operations
625
+
626
+ ### Idiomatic Collection Processing
627
+
628
+ ```kotlin
629
+ // Good: Chained operations
630
+ val activeAdminEmails: List<String> = users
631
+ .filter { it.role == Role.ADMIN && it.isActive }
632
+ .sortedBy { it.name }
633
+ .map { it.email }
634
+
635
+ // Good: Grouping and aggregation
636
+ val usersByRole: Map<Role, List<User>> = users.groupBy { it.role }
637
+
638
+ val oldestByRole: Map<Role, User?> = users.groupBy { it.role }
639
+ .mapValues { (_, users) -> users.minByOrNull { it.createdAt } }
640
+
641
+ // Good: Associate for map creation
642
+ val usersById: Map<UserId, User> = users.associateBy { it.id }
643
+
644
+ // Good: Partition for splitting
645
+ val (active, inactive) = users.partition { it.isActive }
646
+ ```
647
+
648
+ ## Quick Reference: Kotlin Idioms
649
+
650
+ | Idiom | Description |
651
+ |-------|-------------|
652
+ | `val` over `var` | Prefer immutable variables |
653
+ | `data class` | For value objects with equals/hashCode/copy |
654
+ | `sealed class/interface` | For restricted type hierarchies |
655
+ | `value class` | For type-safe wrappers with zero overhead |
656
+ | Expression `when` | Exhaustive pattern matching |
657
+ | Safe call `?.` | Null-safe member access |
658
+ | Elvis `?:` | Default value for nullables |
659
+ | `let`/`apply`/`also`/`run`/`with` | Scope functions for clean code |
660
+ | Extension functions | Add behavior without inheritance |
661
+ | `copy()` | Immutable updates on data classes |
662
+ | `require`/`check` | Precondition assertions |
663
+ | Coroutine `async`/`await` | Structured concurrent execution |
664
+ | `Flow` | Cold reactive streams |
665
+ | `sequence` | Lazy evaluation |
666
+ | Delegation `by` | Reuse implementation without inheritance |
667
+
668
+ ## Anti-Patterns to Avoid
669
+
670
+ ```kotlin
671
+ // Bad: Force-unwrapping nullable types
672
+ val name = user!!.name
673
+
674
+ // Bad: Platform type leakage from Java
675
+ fun getLength(s: String) = s.length // Safe
676
+ fun getLength(s: String?) = s?.length ?: 0 // Handle nulls from Java
677
+
678
+ // Bad: Mutable data classes
679
+ data class MutableUser(var name: String, var email: String)
680
+
681
+ // Bad: Using exceptions for control flow
682
+ try {
683
+ val user = findUser(id)
684
+ } catch (e: NotFoundException) {
685
+ // Don't use exceptions for expected cases
686
+ }
687
+
688
+ // Good: Use nullable return or Result
689
+ val user: User? = findUserOrNull(id)
690
+
691
+ // Bad: Ignoring coroutine scope
692
+ GlobalScope.launch { /* Avoid GlobalScope */ }
693
+
694
+ // Good: Use structured concurrency
695
+ coroutineScope {
696
+ launch { /* Properly scoped */ }
697
+ }
698
+
699
+ // Bad: Deeply nested scope functions
700
+ user?.let { u ->
701
+ u.address?.let { a ->
702
+ a.city?.let { c -> process(c) }
703
+ }
704
+ }
705
+
706
+ // Good: Direct null-safe chain
707
+ user?.address?.city?.let { process(it) }
708
+ ```
709
+
710
+ **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.