arboris-cli 1.0.0 → 1.1.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 (451) hide show
  1. package/dist/cli.mjs +420 -0
  2. package/manifest.json +602 -0
  3. package/package.json +22 -10
  4. package/prisma/skills/accessibility/SKILL.md +147 -0
  5. package/prisma/skills/agent-architecture-audit/SKILL.md +257 -0
  6. package/prisma/skills/agent-eval/SKILL.md +146 -0
  7. package/prisma/skills/agent-harness-construction/SKILL.md +74 -0
  8. package/prisma/skills/agent-introspection-debugging/SKILL.md +154 -0
  9. package/prisma/skills/agent-payment-x402/SKILL.md +225 -0
  10. package/prisma/skills/agent-self-evaluation/SKILL.md +182 -0
  11. package/prisma/skills/agent-self-evaluation/examples/high-score-example.md +87 -0
  12. package/prisma/skills/agent-self-evaluation/examples/low-score-example.md +86 -0
  13. package/prisma/skills/agent-self-evaluation/references/evaluation-criteria.md +71 -0
  14. package/prisma/skills/agent-self-evaluation/references/hook-integration.md +64 -0
  15. package/prisma/skills/agent-self-evaluation/scripts/evaluate.py +408 -0
  16. package/prisma/skills/agent-self-evaluation/templates/evaluation-report.md +86 -0
  17. package/prisma/skills/agent-sort/SKILL.md +216 -0
  18. package/prisma/skills/agentic-engineering/SKILL.md +64 -0
  19. package/prisma/skills/agentic-os/SKILL.md +388 -0
  20. package/prisma/skills/ai-first-engineering/SKILL.md +52 -0
  21. package/prisma/skills/ai-regression-testing/SKILL.md +386 -0
  22. package/prisma/skills/android-clean-architecture/SKILL.md +340 -0
  23. package/prisma/skills/angular-developer/SKILL.md +155 -0
  24. package/prisma/skills/angular-developer/references/angular-animations.md +160 -0
  25. package/prisma/skills/angular-developer/references/angular-aria.md +410 -0
  26. package/prisma/skills/angular-developer/references/cli.md +86 -0
  27. package/prisma/skills/angular-developer/references/component-harnesses.md +59 -0
  28. package/prisma/skills/angular-developer/references/component-styling.md +91 -0
  29. package/prisma/skills/angular-developer/references/components.md +117 -0
  30. package/prisma/skills/angular-developer/references/creating-services.md +97 -0
  31. package/prisma/skills/angular-developer/references/data-resolvers.md +69 -0
  32. package/prisma/skills/angular-developer/references/define-routes.md +67 -0
  33. package/prisma/skills/angular-developer/references/defining-providers.md +72 -0
  34. package/prisma/skills/angular-developer/references/di-fundamentals.md +120 -0
  35. package/prisma/skills/angular-developer/references/e2e-testing.md +56 -0
  36. package/prisma/skills/angular-developer/references/effects.md +83 -0
  37. package/prisma/skills/angular-developer/references/hierarchical-injectors.md +43 -0
  38. package/prisma/skills/angular-developer/references/host-elements.md +80 -0
  39. package/prisma/skills/angular-developer/references/injection-context.md +63 -0
  40. package/prisma/skills/angular-developer/references/inputs.md +101 -0
  41. package/prisma/skills/angular-developer/references/linked-signal.md +59 -0
  42. package/prisma/skills/angular-developer/references/loading-strategies.md +61 -0
  43. package/prisma/skills/angular-developer/references/mcp.md +108 -0
  44. package/prisma/skills/angular-developer/references/navigate-to-routes.md +69 -0
  45. package/prisma/skills/angular-developer/references/outputs.md +86 -0
  46. package/prisma/skills/angular-developer/references/reactive-forms.md +122 -0
  47. package/prisma/skills/angular-developer/references/rendering-strategies.md +44 -0
  48. package/prisma/skills/angular-developer/references/resource.md +77 -0
  49. package/prisma/skills/angular-developer/references/route-animations.md +56 -0
  50. package/prisma/skills/angular-developer/references/route-guards.md +52 -0
  51. package/prisma/skills/angular-developer/references/router-lifecycle.md +45 -0
  52. package/prisma/skills/angular-developer/references/router-testing.md +87 -0
  53. package/prisma/skills/angular-developer/references/show-routes-with-outlets.md +68 -0
  54. package/prisma/skills/angular-developer/references/signal-forms.md +795 -0
  55. package/prisma/skills/angular-developer/references/signals-overview.md +94 -0
  56. package/prisma/skills/angular-developer/references/tailwind-css.md +69 -0
  57. package/prisma/skills/angular-developer/references/template-driven-forms.md +114 -0
  58. package/prisma/skills/angular-developer/references/testing-fundamentals.md +65 -0
  59. package/prisma/skills/api-connector-builder/SKILL.md +121 -0
  60. package/prisma/skills/api-design/SKILL.md +524 -0
  61. package/prisma/skills/architecture-decision-records/SKILL.md +180 -0
  62. package/prisma/skills/article-writing/SKILL.md +80 -0
  63. package/prisma/skills/automation-audit-ops/SKILL.md +143 -0
  64. package/prisma/skills/autonomous-agent-harness/SKILL.md +274 -0
  65. package/prisma/skills/autonomous-loops/SKILL.md +611 -0
  66. package/prisma/skills/backend-patterns/SKILL.md +562 -0
  67. package/prisma/skills/benchmark/SKILL.md +94 -0
  68. package/prisma/skills/benchmark-methodology/SKILL.md +190 -0
  69. package/prisma/skills/benchmark-optimization-loop/SKILL.md +70 -0
  70. package/prisma/skills/blender-motion-state-inspection/SKILL.md +165 -0
  71. package/prisma/skills/blueprint/SKILL.md +106 -0
  72. package/prisma/skills/brand-discovery/SKILL.md +145 -0
  73. package/prisma/skills/brand-discovery/references/10_purpose-why.md +40 -0
  74. package/prisma/skills/brand-discovery/references/20_positioning.md +44 -0
  75. package/prisma/skills/brand-discovery/references/30_audience-niche.md +52 -0
  76. package/prisma/skills/brand-discovery/references/40_personality-archetype.md +57 -0
  77. package/prisma/skills/brand-discovery/references/50_voice-tone.md +59 -0
  78. package/prisma/skills/brand-discovery/references/60_narrative-story.md +50 -0
  79. package/prisma/skills/brand-discovery/references/70_founder-tension.md +49 -0
  80. package/prisma/skills/brand-discovery/references/90_SYNTHESIS.md +133 -0
  81. package/prisma/skills/brand-voice/SKILL.md +98 -0
  82. package/prisma/skills/brand-voice/references/voice-profile-schema.md +55 -0
  83. package/prisma/skills/browser-qa/SKILL.md +105 -0
  84. package/prisma/skills/bun-runtime/SKILL.md +85 -0
  85. package/prisma/skills/canary-watch/SKILL.md +108 -0
  86. package/prisma/skills/carrier-relationship-management/SKILL.md +212 -0
  87. package/prisma/skills/cisco-ios-patterns/SKILL.md +164 -0
  88. package/prisma/skills/ck/SKILL.md +148 -0
  89. package/prisma/skills/ck/commands/forget.mjs +44 -0
  90. package/prisma/skills/ck/commands/info.mjs +24 -0
  91. package/prisma/skills/ck/commands/init.mjs +143 -0
  92. package/prisma/skills/ck/commands/list.mjs +40 -0
  93. package/prisma/skills/ck/commands/migrate.mjs +202 -0
  94. package/prisma/skills/ck/commands/resume.mjs +36 -0
  95. package/prisma/skills/ck/commands/save.mjs +210 -0
  96. package/prisma/skills/ck/commands/shared.mjs +387 -0
  97. package/prisma/skills/ck/hooks/session-start.mjs +224 -0
  98. package/prisma/skills/claude-devfleet/SKILL.md +112 -0
  99. package/prisma/skills/click-path-audit/SKILL.md +245 -0
  100. package/prisma/skills/clickhouse-io/SKILL.md +440 -0
  101. package/prisma/skills/code-tour/SKILL.md +254 -0
  102. package/prisma/skills/codebase-onboarding/SKILL.md +234 -0
  103. package/prisma/skills/codehealth-mcp/SKILL.md +167 -0
  104. package/prisma/skills/coding-standards/SKILL.md +551 -0
  105. package/prisma/skills/competitive-platform-analysis/SKILL.md +214 -0
  106. package/prisma/skills/competitive-report-structure/SKILL.md +162 -0
  107. package/prisma/skills/compose-multiplatform-patterns/SKILL.md +300 -0
  108. package/prisma/skills/config-gc/SKILL.md +120 -0
  109. package/prisma/skills/configure-ecc/SKILL.md +385 -0
  110. package/prisma/skills/connections-optimizer/SKILL.md +190 -0
  111. package/prisma/skills/content-engine/SKILL.md +132 -0
  112. package/prisma/skills/content-hash-cache-pattern/SKILL.md +162 -0
  113. package/prisma/skills/context-budget/SKILL.md +136 -0
  114. package/prisma/skills/continuous-agent-loop/SKILL.md +46 -0
  115. package/prisma/skills/continuous-learning/SKILL.md +132 -0
  116. package/prisma/skills/continuous-learning/config.json +18 -0
  117. package/prisma/skills/continuous-learning/evaluate-session.sh +69 -0
  118. package/prisma/skills/continuous-learning-v2/SKILL.md +361 -0
  119. package/prisma/skills/continuous-learning-v2/agents/observer-loop.sh +359 -0
  120. package/prisma/skills/continuous-learning-v2/agents/observer.md +189 -0
  121. package/prisma/skills/continuous-learning-v2/agents/session-guardian.sh +150 -0
  122. package/prisma/skills/continuous-learning-v2/agents/start-observer.sh +248 -0
  123. package/prisma/skills/continuous-learning-v2/config.json +8 -0
  124. package/prisma/skills/continuous-learning-v2/hooks/observe.sh +585 -0
  125. package/prisma/skills/continuous-learning-v2/scripts/detect-project.sh +322 -0
  126. package/prisma/skills/continuous-learning-v2/scripts/instinct-cli.py +1956 -0
  127. package/prisma/skills/continuous-learning-v2/scripts/lib/homunculus-dir.sh +31 -0
  128. package/prisma/skills/continuous-learning-v2/scripts/migrate-homunculus.sh +68 -0
  129. package/prisma/skills/continuous-learning-v2/scripts/test_parse_instinct.py +1421 -0
  130. package/prisma/skills/cost-aware-llm-pipeline/SKILL.md +184 -0
  131. package/prisma/skills/cost-tracking/SKILL.md +97 -0
  132. package/prisma/skills/council/SKILL.md +204 -0
  133. package/prisma/skills/cpp-coding-standards/SKILL.md +724 -0
  134. package/prisma/skills/cpp-testing/SKILL.md +325 -0
  135. package/prisma/skills/crosspost/SKILL.md +112 -0
  136. package/prisma/skills/csharp-testing/SKILL.md +322 -0
  137. package/prisma/skills/customer-billing-ops/SKILL.md +141 -0
  138. package/prisma/skills/customs-trade-compliance/SKILL.md +263 -0
  139. package/prisma/skills/dart-flutter-patterns/SKILL.md +564 -0
  140. package/prisma/skills/dashboard-builder/SKILL.md +109 -0
  141. package/prisma/skills/data-scraper-agent/SKILL.md +765 -0
  142. package/prisma/skills/data-throughput-accelerator/SKILL.md +73 -0
  143. package/prisma/skills/database-migrations/SKILL.md +430 -0
  144. package/prisma/skills/deep-research/SKILL.md +160 -0
  145. package/prisma/skills/defi-amm-security/SKILL.md +167 -0
  146. package/prisma/skills/delivery-gate/SKILL.md +126 -0
  147. package/prisma/skills/delivery-gate/hooks/quality-gate.py +220 -0
  148. package/prisma/skills/deployment-patterns/SKILL.md +428 -0
  149. package/prisma/skills/design-system/SKILL.md +83 -0
  150. package/prisma/skills/django-celery/SKILL.md +458 -0
  151. package/prisma/skills/django-patterns/SKILL.md +735 -0
  152. package/prisma/skills/django-security/SKILL.md +644 -0
  153. package/prisma/skills/django-tdd/SKILL.md +730 -0
  154. package/prisma/skills/django-verification/SKILL.md +470 -0
  155. package/prisma/skills/dmux-workflows/SKILL.md +192 -0
  156. package/prisma/skills/docker-patterns/SKILL.md +365 -0
  157. package/prisma/skills/documentation-lookup/SKILL.md +91 -0
  158. package/prisma/skills/dotnet-patterns/SKILL.md +322 -0
  159. package/prisma/skills/dynamic-workflow-mode/SKILL.md +124 -0
  160. package/prisma/skills/e2e-testing/SKILL.md +327 -0
  161. package/prisma/skills/ecc-guide/SKILL.md +190 -0
  162. package/prisma/skills/ecc-recipes/SKILL.md +149 -0
  163. package/prisma/skills/ecc-tools-cost-audit/SKILL.md +161 -0
  164. package/prisma/skills/email-ops/SKILL.md +122 -0
  165. package/prisma/skills/energy-procurement/SKILL.md +228 -0
  166. package/prisma/skills/enterprise-agent-ops/SKILL.md +51 -0
  167. package/prisma/skills/error-handling/SKILL.md +377 -0
  168. package/prisma/skills/eval-harness/SKILL.md +271 -0
  169. package/prisma/skills/evm-token-decimals/SKILL.md +131 -0
  170. package/prisma/skills/exa-search/SKILL.md +108 -0
  171. package/prisma/skills/fal-ai-media/SKILL.md +289 -0
  172. package/prisma/skills/fastapi-patterns/SKILL.md +514 -0
  173. package/prisma/skills/finance-billing-ops/SKILL.md +128 -0
  174. package/prisma/skills/flox-environments/SKILL.md +497 -0
  175. package/prisma/skills/flutter-dart-code-review/SKILL.md +436 -0
  176. package/prisma/skills/foundation-models-on-device/SKILL.md +243 -0
  177. package/prisma/skills/frontend-a11y/SKILL.md +446 -0
  178. package/prisma/skills/frontend-design-direction/SKILL.md +93 -0
  179. package/prisma/skills/frontend-patterns/SKILL.md +657 -0
  180. package/prisma/skills/frontend-slides/SKILL.md +185 -0
  181. package/prisma/skills/frontend-slides/STYLE_PRESETS.md +330 -0
  182. package/prisma/skills/frontend-slides/animation-patterns.md +122 -0
  183. package/prisma/skills/frontend-slides/html-template.md +419 -0
  184. package/prisma/skills/frontend-slides/scripts/export-pdf.sh +418 -0
  185. package/prisma/skills/frontend-slides/scripts/extract-pptx.py +96 -0
  186. package/prisma/skills/frontend-slides/viewport-base.css +153 -0
  187. package/prisma/skills/fsharp-testing/SKILL.md +281 -0
  188. package/prisma/skills/gan-style-harness/SKILL.md +279 -0
  189. package/prisma/skills/gateguard/SKILL.md +133 -0
  190. package/prisma/skills/generating-python-installer/SKILL.md +820 -0
  191. package/prisma/skills/git-workflow/SKILL.md +716 -0
  192. package/prisma/skills/github-ops/SKILL.md +145 -0
  193. package/prisma/skills/golang-patterns/SKILL.md +675 -0
  194. package/prisma/skills/golang-testing/SKILL.md +721 -0
  195. package/prisma/skills/google-workspace-ops/SKILL.md +96 -0
  196. package/prisma/skills/growth-log/SKILL.md +128 -0
  197. package/prisma/skills/healthcare-cdss-patterns/SKILL.md +246 -0
  198. package/prisma/skills/healthcare-emr-patterns/SKILL.md +160 -0
  199. package/prisma/skills/healthcare-eval-harness/SKILL.md +208 -0
  200. package/prisma/skills/healthcare-phi-compliance/SKILL.md +146 -0
  201. package/prisma/skills/hermes-imports/SKILL.md +89 -0
  202. package/prisma/skills/hexagonal-architecture/SKILL.md +277 -0
  203. package/prisma/skills/hipaa-compliance/SKILL.md +79 -0
  204. package/prisma/skills/homelab-network-readiness/SKILL.md +170 -0
  205. package/prisma/skills/homelab-network-setup/SKILL.md +130 -0
  206. package/prisma/skills/homelab-pihole-dns/SKILL.md +275 -0
  207. package/prisma/skills/homelab-vlan-segmentation/SKILL.md +312 -0
  208. package/prisma/skills/homelab-wireguard-vpn/SKILL.md +306 -0
  209. package/prisma/skills/hookify-rules/SKILL.md +128 -0
  210. package/prisma/skills/inherit-legacy-style/SKILL.md +157 -0
  211. package/prisma/skills/intent-driven-development/SKILL.md +360 -0
  212. package/prisma/skills/inventory-demand-planning/SKILL.md +247 -0
  213. package/prisma/skills/investor-materials/SKILL.md +97 -0
  214. package/prisma/skills/investor-outreach/SKILL.md +92 -0
  215. package/prisma/skills/ios-icon-gen/SKILL.md +158 -0
  216. package/prisma/skills/ios-icon-gen/scripts/generate_icons.swift +258 -0
  217. package/prisma/skills/ios-icon-gen/scripts/iconify_gen.sh +235 -0
  218. package/prisma/skills/iterative-retrieval/SKILL.md +212 -0
  219. package/prisma/skills/ito-basket-compare/SKILL.md +64 -0
  220. package/prisma/skills/ito-data-atlas-agent/SKILL.md +64 -0
  221. package/prisma/skills/ito-market-intelligence/SKILL.md +61 -0
  222. package/prisma/skills/ito-trade-planner/SKILL.md +68 -0
  223. package/prisma/skills/java-coding-standards/SKILL.md +384 -0
  224. package/prisma/skills/jira-integration/SKILL.md +303 -0
  225. package/prisma/skills/jpa-patterns/SKILL.md +152 -0
  226. package/prisma/skills/knowledge-ops/SKILL.md +155 -0
  227. package/prisma/skills/kotlin-coroutines-flows/SKILL.md +285 -0
  228. package/prisma/skills/kotlin-exposed-patterns/SKILL.md +720 -0
  229. package/prisma/skills/kotlin-ktor-patterns/SKILL.md +690 -0
  230. package/prisma/skills/kotlin-patterns/SKILL.md +712 -0
  231. package/prisma/skills/kotlin-testing/SKILL.md +825 -0
  232. package/prisma/skills/kubernetes-patterns/SKILL.md +756 -0
  233. package/prisma/skills/laravel-patterns/SKILL.md +416 -0
  234. package/prisma/skills/laravel-plugin-discovery/SKILL.md +230 -0
  235. package/prisma/skills/laravel-security/SKILL.md +948 -0
  236. package/prisma/skills/laravel-tdd/SKILL.md +675 -0
  237. package/prisma/skills/laravel-verification/SKILL.md +180 -0
  238. package/prisma/skills/latency-critical-systems/SKILL.md +74 -0
  239. package/prisma/skills/lead-intelligence/SKILL.md +322 -0
  240. package/prisma/skills/lead-intelligence/agents/enrichment-agent.md +85 -0
  241. package/prisma/skills/lead-intelligence/agents/mutual-mapper.md +75 -0
  242. package/prisma/skills/lead-intelligence/agents/outreach-drafter.md +98 -0
  243. package/prisma/skills/lead-intelligence/agents/signal-scorer.md +60 -0
  244. package/prisma/skills/liquid-glass-design/SKILL.md +279 -0
  245. package/prisma/skills/llm-trading-agent-security/SKILL.md +147 -0
  246. package/prisma/skills/logistics-exception-management/SKILL.md +222 -0
  247. package/prisma/skills/loop-design-check/SKILL.md +143 -0
  248. package/prisma/skills/mailtrap-email-integration/SKILL.md +77 -0
  249. package/prisma/skills/make-interfaces-feel-better/SKILL.md +152 -0
  250. package/prisma/skills/manim-video/SKILL.md +90 -0
  251. package/prisma/skills/manim-video/assets/network_graph_scene.py +52 -0
  252. package/prisma/skills/market-research/SKILL.md +76 -0
  253. package/prisma/skills/marketing-campaign/SKILL.md +114 -0
  254. package/prisma/skills/mcp-server-patterns/SKILL.md +70 -0
  255. package/prisma/skills/messages-ops/SKILL.md +105 -0
  256. package/prisma/skills/ml-adoption-playbook/SKILL.md +57 -0
  257. package/prisma/skills/mle-workflow/SKILL.md +347 -0
  258. package/prisma/skills/motion-advanced/SKILL.md +596 -0
  259. package/prisma/skills/motion-foundations/SKILL.md +299 -0
  260. package/prisma/skills/motion-patterns/SKILL.md +434 -0
  261. package/prisma/skills/motion-ui/SKILL.md +576 -0
  262. package/prisma/skills/mysql-patterns/SKILL.md +413 -0
  263. package/prisma/skills/nanoclaw-repl/SKILL.md +34 -0
  264. package/prisma/skills/nestjs-patterns/SKILL.md +231 -0
  265. package/prisma/skills/netmiko-ssh-automation/SKILL.md +174 -0
  266. package/prisma/skills/network-bgp-diagnostics/SKILL.md +168 -0
  267. package/prisma/skills/network-config-validation/SKILL.md +211 -0
  268. package/prisma/skills/network-interface-health/SKILL.md +153 -0
  269. package/prisma/skills/nextjs-turbopack/SKILL.md +58 -0
  270. package/prisma/skills/nodejs-keccak256/SKILL.md +103 -0
  271. package/prisma/skills/nutrient-document-processing/SKILL.md +168 -0
  272. package/prisma/skills/nuxt4-patterns/SKILL.md +101 -0
  273. package/prisma/skills/openclaw-persona-forge/SKILL.md +289 -0
  274. package/prisma/skills/openclaw-persona-forge/gacha.py +224 -0
  275. package/prisma/skills/openclaw-persona-forge/gacha.sh +5 -0
  276. package/prisma/skills/openclaw-persona-forge/references/avatar-style.md +124 -0
  277. package/prisma/skills/openclaw-persona-forge/references/boundary-rules.md +53 -0
  278. package/prisma/skills/openclaw-persona-forge/references/error-handling.md +53 -0
  279. package/prisma/skills/openclaw-persona-forge/references/identity-tension.md +48 -0
  280. package/prisma/skills/openclaw-persona-forge/references/naming-system.md +39 -0
  281. package/prisma/skills/openclaw-persona-forge/references/output-template.md +166 -0
  282. package/prisma/skills/opensource-pipeline/SKILL.md +256 -0
  283. package/prisma/skills/orch-add-feature/SKILL.md +45 -0
  284. package/prisma/skills/orch-build-mvp/SKILL.md +49 -0
  285. package/prisma/skills/orch-change-feature/SKILL.md +43 -0
  286. package/prisma/skills/orch-fix-defect/SKILL.md +43 -0
  287. package/prisma/skills/orch-pipeline/SKILL.md +121 -0
  288. package/prisma/skills/orch-refine-code/SKILL.md +44 -0
  289. package/prisma/skills/parallel-execution-optimizer/SKILL.md +73 -0
  290. package/prisma/skills/perl-patterns/SKILL.md +505 -0
  291. package/prisma/skills/perl-security/SKILL.md +504 -0
  292. package/prisma/skills/perl-testing/SKILL.md +476 -0
  293. package/prisma/skills/plan-orchestrate/SKILL.md +263 -0
  294. package/prisma/skills/plankton-code-quality/SKILL.md +237 -0
  295. package/prisma/skills/postgres-patterns/SKILL.md +148 -0
  296. package/prisma/skills/prediction-market-oracle-research/SKILL.md +64 -0
  297. package/prisma/skills/prediction-market-risk-review/SKILL.md +61 -0
  298. package/prisma/skills/prisma-patterns/SKILL.md +401 -0
  299. package/prisma/skills/product-capability/SKILL.md +142 -0
  300. package/prisma/skills/product-lens/SKILL.md +93 -0
  301. package/prisma/skills/production-audit/SKILL.md +207 -0
  302. package/prisma/skills/production-scheduling/SKILL.md +238 -0
  303. package/prisma/skills/project-flow-ops/SKILL.md +112 -0
  304. package/prisma/skills/prompt-optimizer/SKILL.md +398 -0
  305. package/prisma/skills/python-patterns/SKILL.md +751 -0
  306. package/prisma/skills/python-testing/SKILL.md +817 -0
  307. package/prisma/skills/pytorch-patterns/SKILL.md +397 -0
  308. package/prisma/skills/quality-nonconformance/SKILL.md +260 -0
  309. package/prisma/skills/quarkus-patterns/SKILL.md +723 -0
  310. package/prisma/skills/quarkus-security/SKILL.md +468 -0
  311. package/prisma/skills/quarkus-tdd/SKILL.md +812 -0
  312. package/prisma/skills/quarkus-verification/SKILL.md +480 -0
  313. package/prisma/skills/ralphinho-rfc-pipeline/SKILL.md +68 -0
  314. package/prisma/skills/react-native-patterns/SKILL.md +326 -0
  315. package/prisma/skills/react-patterns/SKILL.md +342 -0
  316. package/prisma/skills/react-performance/SKILL.md +575 -0
  317. package/prisma/skills/react-testing/SKILL.md +424 -0
  318. package/prisma/skills/recsys-pipeline-architect/SKILL.md +115 -0
  319. package/prisma/skills/recursive-decision-ledger/SKILL.md +80 -0
  320. package/prisma/skills/redis-patterns/SKILL.md +404 -0
  321. package/prisma/skills/regex-vs-llm-structured-text/SKILL.md +221 -0
  322. package/prisma/skills/remotion-video-creation/SKILL.md +43 -0
  323. package/prisma/skills/remotion-video-creation/rules/3d.md +86 -0
  324. package/prisma/skills/remotion-video-creation/rules/animations.md +29 -0
  325. package/prisma/skills/remotion-video-creation/rules/assets/charts-bar-chart.tsx +173 -0
  326. package/prisma/skills/remotion-video-creation/rules/assets/text-animations-typewriter.tsx +100 -0
  327. package/prisma/skills/remotion-video-creation/rules/assets/text-animations-word-highlight.tsx +108 -0
  328. package/prisma/skills/remotion-video-creation/rules/assets.md +78 -0
  329. package/prisma/skills/remotion-video-creation/rules/audio.md +172 -0
  330. package/prisma/skills/remotion-video-creation/rules/calculate-metadata.md +104 -0
  331. package/prisma/skills/remotion-video-creation/rules/can-decode.md +75 -0
  332. package/prisma/skills/remotion-video-creation/rules/charts.md +58 -0
  333. package/prisma/skills/remotion-video-creation/rules/compositions.md +146 -0
  334. package/prisma/skills/remotion-video-creation/rules/display-captions.md +126 -0
  335. package/prisma/skills/remotion-video-creation/rules/extract-frames.md +229 -0
  336. package/prisma/skills/remotion-video-creation/rules/fonts.md +152 -0
  337. package/prisma/skills/remotion-video-creation/rules/get-audio-duration.md +58 -0
  338. package/prisma/skills/remotion-video-creation/rules/get-video-dimensions.md +68 -0
  339. package/prisma/skills/remotion-video-creation/rules/get-video-duration.md +58 -0
  340. package/prisma/skills/remotion-video-creation/rules/gifs.md +138 -0
  341. package/prisma/skills/remotion-video-creation/rules/images.md +130 -0
  342. package/prisma/skills/remotion-video-creation/rules/import-srt-captions.md +67 -0
  343. package/prisma/skills/remotion-video-creation/rules/lottie.md +67 -0
  344. package/prisma/skills/remotion-video-creation/rules/measuring-dom-nodes.md +34 -0
  345. package/prisma/skills/remotion-video-creation/rules/measuring-text.md +143 -0
  346. package/prisma/skills/remotion-video-creation/rules/sequencing.md +106 -0
  347. package/prisma/skills/remotion-video-creation/rules/tailwind.md +11 -0
  348. package/prisma/skills/remotion-video-creation/rules/text-animations.md +20 -0
  349. package/prisma/skills/remotion-video-creation/rules/timing.md +179 -0
  350. package/prisma/skills/remotion-video-creation/rules/transcribe-captions.md +19 -0
  351. package/prisma/skills/remotion-video-creation/rules/transitions.md +122 -0
  352. package/prisma/skills/remotion-video-creation/rules/trimming.md +52 -0
  353. package/prisma/skills/remotion-video-creation/rules/videos.md +171 -0
  354. package/prisma/skills/repo-scan/SKILL.md +79 -0
  355. package/prisma/skills/research-ops/SKILL.md +113 -0
  356. package/prisma/skills/returns-reverse-logistics/SKILL.md +240 -0
  357. package/prisma/skills/rules-distill/SKILL.md +265 -0
  358. package/prisma/skills/rules-distill/scripts/scan-rules.sh +58 -0
  359. package/prisma/skills/rules-distill/scripts/scan-skills.sh +129 -0
  360. package/prisma/skills/rust-patterns/SKILL.md +500 -0
  361. package/prisma/skills/rust-testing/SKILL.md +501 -0
  362. package/prisma/skills/safety-guard/SKILL.md +76 -0
  363. package/prisma/skills/santa-method/SKILL.md +307 -0
  364. package/prisma/skills/scientific-db-pubmed-database/SKILL.md +176 -0
  365. package/prisma/skills/scientific-db-uspto-database/SKILL.md +178 -0
  366. package/prisma/skills/scientific-pkg-gget/SKILL.md +167 -0
  367. package/prisma/skills/scientific-thinking-literature-review/SKILL.md +193 -0
  368. package/prisma/skills/scientific-thinking-scholar-evaluation/SKILL.md +161 -0
  369. package/prisma/skills/search-first/SKILL.md +183 -0
  370. package/prisma/skills/security-bounty-hunter/SKILL.md +100 -0
  371. package/prisma/skills/security-review/SKILL.md +504 -0
  372. package/prisma/skills/security-review/cloud-infrastructure-security.md +361 -0
  373. package/prisma/skills/security-scan/SKILL.md +166 -0
  374. package/prisma/skills/seo/SKILL.md +155 -0
  375. package/prisma/skills/skill-comply/SKILL.md +59 -0
  376. package/prisma/skills/skill-comply/fixtures/compliant_trace.jsonl +5 -0
  377. package/prisma/skills/skill-comply/fixtures/noncompliant_trace.jsonl +3 -0
  378. package/prisma/skills/skill-comply/fixtures/tdd_spec.yaml +44 -0
  379. package/prisma/skills/skill-comply/prompts/classifier.md +24 -0
  380. package/prisma/skills/skill-comply/prompts/scenario_generator.md +62 -0
  381. package/prisma/skills/skill-comply/prompts/spec_generator.md +42 -0
  382. package/prisma/skills/skill-comply/pyproject.toml +15 -0
  383. package/prisma/skills/skill-comply/scripts/__init__.py +0 -0
  384. package/prisma/skills/skill-comply/scripts/classifier.py +85 -0
  385. package/prisma/skills/skill-comply/scripts/grader.py +124 -0
  386. package/prisma/skills/skill-comply/scripts/parser.py +107 -0
  387. package/prisma/skills/skill-comply/scripts/report.py +170 -0
  388. package/prisma/skills/skill-comply/scripts/run.py +127 -0
  389. package/prisma/skills/skill-comply/scripts/runner.py +194 -0
  390. package/prisma/skills/skill-comply/scripts/scenario_generator.py +70 -0
  391. package/prisma/skills/skill-comply/scripts/spec_generator.py +72 -0
  392. package/prisma/skills/skill-comply/scripts/utils.py +13 -0
  393. package/prisma/skills/skill-comply/tests/test_grader.py +197 -0
  394. package/prisma/skills/skill-comply/tests/test_parser.py +90 -0
  395. package/prisma/skills/skill-comply/tests/test_runner.py +172 -0
  396. package/prisma/skills/skill-scout/SKILL.md +141 -0
  397. package/prisma/skills/skill-stocktake/SKILL.md +195 -0
  398. package/prisma/skills/skill-stocktake/scripts/quick-diff.sh +87 -0
  399. package/prisma/skills/skill-stocktake/scripts/save-results.sh +56 -0
  400. package/prisma/skills/skill-stocktake/scripts/scan.sh +170 -0
  401. package/prisma/skills/social-graph-ranker/SKILL.md +155 -0
  402. package/prisma/skills/social-publisher/SKILL.md +130 -0
  403. package/prisma/skills/springboot-patterns/SKILL.md +315 -0
  404. package/prisma/skills/springboot-security/SKILL.md +273 -0
  405. package/prisma/skills/springboot-tdd/SKILL.md +159 -0
  406. package/prisma/skills/springboot-verification/SKILL.md +232 -0
  407. package/prisma/skills/strategic-compact/SKILL.md +136 -0
  408. package/prisma/skills/swift-actor-persistence/SKILL.md +144 -0
  409. package/prisma/skills/swift-concurrency-6-2/SKILL.md +216 -0
  410. package/prisma/skills/swift-protocol-di-testing/SKILL.md +191 -0
  411. package/prisma/skills/swiftui-patterns/SKILL.md +259 -0
  412. package/prisma/skills/taste/SKILL.md +264 -0
  413. package/prisma/skills/taste/references/genre-taxonomy.md +87 -0
  414. package/prisma/skills/tdd-workflow/SKILL.md +583 -0
  415. package/prisma/skills/team-agent-orchestration/SKILL.md +111 -0
  416. package/prisma/skills/team-builder/SKILL.md +169 -0
  417. package/prisma/skills/terminal-ops/SKILL.md +110 -0
  418. package/prisma/skills/tinystruct-patterns/SKILL.md +279 -0
  419. package/prisma/skills/tinystruct-patterns/references/architecture.md +90 -0
  420. package/prisma/skills/tinystruct-patterns/references/data-handling.md +60 -0
  421. package/prisma/skills/tinystruct-patterns/references/database.md +99 -0
  422. package/prisma/skills/tinystruct-patterns/references/routing.md +64 -0
  423. package/prisma/skills/tinystruct-patterns/references/system-usage.md +97 -0
  424. package/prisma/skills/tinystruct-patterns/references/testing.md +72 -0
  425. package/prisma/skills/token-budget-advisor/SKILL.md +134 -0
  426. package/prisma/skills/ui-demo/SKILL.md +466 -0
  427. package/prisma/skills/ui-to-vue/SKILL.md +135 -0
  428. package/prisma/skills/uncloud/SKILL.md +344 -0
  429. package/prisma/skills/unified-notifications-ops/SKILL.md +188 -0
  430. package/prisma/skills/verification-loop/SKILL.md +127 -0
  431. package/prisma/skills/video-editing/SKILL.md +311 -0
  432. package/prisma/skills/videodb/SKILL.md +375 -0
  433. package/prisma/skills/videodb/reference/api-reference.md +550 -0
  434. package/prisma/skills/videodb/reference/capture-reference.md +407 -0
  435. package/prisma/skills/videodb/reference/capture.md +101 -0
  436. package/prisma/skills/videodb/reference/editor.md +443 -0
  437. package/prisma/skills/videodb/reference/generative.md +331 -0
  438. package/prisma/skills/videodb/reference/rtstream-reference.md +564 -0
  439. package/prisma/skills/videodb/reference/rtstream.md +65 -0
  440. package/prisma/skills/videodb/reference/search.md +230 -0
  441. package/prisma/skills/videodb/reference/streaming.md +406 -0
  442. package/prisma/skills/videodb/reference/use-cases.md +118 -0
  443. package/prisma/skills/videodb/scripts/ws_listener.py +282 -0
  444. package/prisma/skills/visa-doc-translate/README.md +86 -0
  445. package/prisma/skills/visa-doc-translate/SKILL.md +117 -0
  446. package/prisma/skills/vite-patterns/SKILL.md +450 -0
  447. package/prisma/skills/vue-patterns/SKILL.md +471 -0
  448. package/prisma/skills/windows-desktop-e2e/SKILL.md +888 -0
  449. package/prisma/skills/workspace-surface-audit/SKILL.md +126 -0
  450. package/prisma/skills/x-api/SKILL.md +235 -0
  451. package/run.mjs +0 -10
@@ -0,0 +1,387 @@
1
+ /**
2
+ * ck — Context Keeper v2
3
+ * shared.mjs — common utilities for all command scripts
4
+ *
5
+ * No external dependencies. Node.js stdlib only.
6
+ */
7
+
8
+ import { readFileSync, writeFileSync, existsSync, mkdirSync } from 'fs';
9
+ import { resolve } from 'path';
10
+ import { homedir } from 'os';
11
+ import { spawnSync } from 'child_process';
12
+ import { randomBytes } from 'crypto';
13
+
14
+ // ─── Paths ────────────────────────────────────────────────────────────────────
15
+
16
+ export const CK_HOME = resolve(homedir(), '.claude', 'ck');
17
+ export const CONTEXTS_DIR = resolve(CK_HOME, 'contexts');
18
+ export const PROJECTS_FILE = resolve(CK_HOME, 'projects.json');
19
+ export const CURRENT_SESSION = resolve(CK_HOME, 'current-session.json');
20
+ export const SKILL_FILE = resolve(homedir(), '.claude', 'skills', 'ck', 'SKILL.md');
21
+
22
+ // ─── JSON I/O ─────────────────────────────────────────────────────────────────
23
+
24
+ export function readJson(filePath) {
25
+ try {
26
+ if (!existsSync(filePath)) return null;
27
+ return JSON.parse(readFileSync(filePath, 'utf8'));
28
+ } catch {
29
+ return null;
30
+ }
31
+ }
32
+
33
+ export function writeJson(filePath, data) {
34
+ const dir = resolve(filePath, '..');
35
+ mkdirSync(dir, { recursive: true });
36
+ writeFileSync(filePath, JSON.stringify(data, null, 2) + '\n', 'utf8');
37
+ }
38
+
39
+ export function readProjects() {
40
+ return readJson(PROJECTS_FILE) || {};
41
+ }
42
+
43
+ export function writeProjects(projects) {
44
+ writeJson(PROJECTS_FILE, projects);
45
+ }
46
+
47
+ // ─── Context I/O ──────────────────────────────────────────────────────────────
48
+
49
+ export function contextPath(contextDir) {
50
+ return resolve(CONTEXTS_DIR, contextDir, 'context.json');
51
+ }
52
+
53
+ export function contextMdPath(contextDir) {
54
+ return resolve(CONTEXTS_DIR, contextDir, 'CONTEXT.md');
55
+ }
56
+
57
+ export function loadContext(contextDir) {
58
+ return readJson(contextPath(contextDir));
59
+ }
60
+
61
+ export function saveContext(contextDir, data) {
62
+ const dir = resolve(CONTEXTS_DIR, contextDir);
63
+ mkdirSync(dir, { recursive: true });
64
+ writeJson(contextPath(contextDir), data);
65
+ writeFileSync(contextMdPath(contextDir), renderContextMd(data), 'utf8');
66
+ }
67
+
68
+ /**
69
+ * Resolve which project to operate on.
70
+ * @param {string|undefined} arg — undefined = cwd match, number string = alphabetical index, else name search
71
+ * @param {string} cwd
72
+ * @returns {{ name, contextDir, projectPath, context } | null}
73
+ */
74
+ export function resolveContext(arg, cwd) {
75
+ const projects = readProjects();
76
+ const entries = Object.entries(projects); // [path, {name, contextDir, lastUpdated}]
77
+
78
+ if (!arg) {
79
+ // Match by cwd
80
+ const entry = projects[cwd];
81
+ if (!entry) return null;
82
+ const context = loadContext(entry.contextDir);
83
+ if (!context) return null;
84
+ return { name: entry.name, contextDir: entry.contextDir, projectPath: cwd, context };
85
+ }
86
+
87
+ // Collect all contexts sorted alphabetically by contextDir
88
+ const sorted = entries
89
+ .map(([path, info]) => ({ path, ...info }))
90
+ .sort((a, b) => a.contextDir.localeCompare(b.contextDir));
91
+
92
+ const asNumber = parseInt(arg, 10);
93
+ if (!isNaN(asNumber) && String(asNumber) === arg) {
94
+ // Number-based lookup (1-indexed)
95
+ const item = sorted[asNumber - 1];
96
+ if (!item) return null;
97
+ const context = loadContext(item.contextDir);
98
+ if (!context) return null;
99
+ return { name: item.name, contextDir: item.contextDir, projectPath: item.path, context };
100
+ }
101
+
102
+ // Name-based lookup: exact > prefix > substring (case-insensitive)
103
+ const lower = arg.toLowerCase();
104
+ let match =
105
+ sorted.find(e => e.name.toLowerCase() === lower) ||
106
+ sorted.find(e => e.name.toLowerCase().startsWith(lower)) ||
107
+ sorted.find(e => e.name.toLowerCase().includes(lower));
108
+
109
+ if (!match) return null;
110
+ const context = loadContext(match.contextDir);
111
+ if (!context) return null;
112
+ return { name: match.name, contextDir: match.contextDir, projectPath: match.path, context };
113
+ }
114
+
115
+ // ─── Date helpers ─────────────────────────────────────────────────────────────
116
+
117
+ export function today() {
118
+ return new Date().toISOString().slice(0, 10);
119
+ }
120
+
121
+ export function daysAgoLabel(dateStr) {
122
+ if (!dateStr) return 'unknown';
123
+ const diff = Math.floor((Date.now() - new Date(dateStr)) / 86_400_000);
124
+ if (diff === 0) return 'Today';
125
+ if (diff === 1) return '1 day ago';
126
+ return `${diff} days ago`;
127
+ }
128
+
129
+ export function stalenessIcon(dateStr) {
130
+ if (!dateStr) return '○';
131
+ const diff = Math.floor((Date.now() - new Date(dateStr)) / 86_400_000);
132
+ if (diff < 1) return '●';
133
+ if (diff <= 5) return '◐';
134
+ return '○';
135
+ }
136
+
137
+ // ─── ID generation ────────────────────────────────────────────────────────────
138
+
139
+ export function shortId() {
140
+ return randomBytes(4).toString('hex');
141
+ }
142
+
143
+ // ─── Git helpers ──────────────────────────────────────────────────────────────
144
+
145
+ function runGit(args, cwd) {
146
+ try {
147
+ const result = spawnSync('git', ['-C', cwd, ...args], {
148
+ timeout: 3000,
149
+ stdio: 'pipe',
150
+ encoding: 'utf8',
151
+ });
152
+ if (result.status !== 0) return null;
153
+ return result.stdout.trim();
154
+ } catch {
155
+ return null;
156
+ }
157
+ }
158
+
159
+ export function gitLogSince(projectPath, sinceDate) {
160
+ if (!sinceDate) return null;
161
+ return runGit(['log', '--oneline', `--since=${sinceDate}`], projectPath);
162
+ }
163
+
164
+ export function gitSummary(projectPath, sinceDate) {
165
+ const log = gitLogSince(projectPath, sinceDate);
166
+ if (!log) return null;
167
+ const commits = log.split('\n').filter(Boolean).length;
168
+ if (commits === 0) return null;
169
+
170
+ // Count unique files changed: use a separate runGit call to avoid nested shell substitution
171
+ const countStr = runGit(['rev-list', '--count', 'HEAD', `--since=${sinceDate}`], projectPath);
172
+ const revCount = countStr ? parseInt(countStr, 10) : commits;
173
+ const diff = runGit(['diff', '--shortstat', `HEAD~${Math.min(revCount, 50)}..HEAD`], projectPath);
174
+
175
+ if (diff) {
176
+ const filesMatch = diff.match(/(\d+) file/);
177
+ const files = filesMatch ? parseInt(filesMatch[1]) : '?';
178
+ return `${commits} commit${commits !== 1 ? 's' : ''}, ${files} file${files !== 1 ? 's' : ''} changed`;
179
+ }
180
+ return `${commits} commit${commits !== 1 ? 's' : ''}`;
181
+ }
182
+
183
+ // ─── Native memory path encoding ──────────────────────────────────────────────
184
+
185
+ export function encodeProjectPath(absolutePath) {
186
+ // "/Users/sree/dev/app" -> "-Users-sree-dev-app"
187
+ return absolutePath.replace(/\//g, '-');
188
+ }
189
+
190
+ export function nativeMemoryDir(absolutePath) {
191
+ const encoded = encodeProjectPath(absolutePath);
192
+ return resolve(homedir(), '.claude', 'projects', encoded, 'memory');
193
+ }
194
+
195
+ // ─── Rendering ────────────────────────────────────────────────────────────────
196
+
197
+ /** Render the human-readable CONTEXT.md from context.json */
198
+ export function renderContextMd(ctx) {
199
+ const latest = ctx.sessions?.[ctx.sessions.length - 1] || null;
200
+ const lines = [
201
+ `<!-- Generated by ck v2 — edit context.json instead -->`,
202
+ `# Project: ${ctx.displayName ?? ctx.name}`,
203
+ `> Path: ${ctx.path}`,
204
+ ];
205
+ if (ctx.repo) lines.push(`> Repo: ${ctx.repo}`);
206
+ const sessionCount = ctx.sessions?.length || 0;
207
+ lines.push(`> Last Session: ${ctx.sessions?.[sessionCount - 1]?.date || 'never'} | Sessions: ${sessionCount}`);
208
+ lines.push(``);
209
+ lines.push(`## What This Is`);
210
+ lines.push(ctx.description || '_Not set._');
211
+ lines.push(``);
212
+ lines.push(`## Tech Stack`);
213
+ lines.push(Array.isArray(ctx.stack) ? ctx.stack.join(', ') : (ctx.stack || '_Not set._'));
214
+ lines.push(``);
215
+ lines.push(`## Current Goal`);
216
+ lines.push(ctx.goal || '_Not set._');
217
+ lines.push(``);
218
+ lines.push(`## Where I Left Off`);
219
+ lines.push(latest?.leftOff || '_Not yet recorded. Run /ck:save after your first session._');
220
+ lines.push(``);
221
+ lines.push(`## Next Steps`);
222
+ if (latest?.nextSteps?.length) {
223
+ latest.nextSteps.forEach((s, i) => lines.push(`${i + 1}. ${s}`));
224
+ } else {
225
+ lines.push(`_Not yet recorded._`);
226
+ }
227
+ lines.push(``);
228
+ lines.push(`## Blockers`);
229
+ if (latest?.blockers?.length) {
230
+ latest.blockers.forEach(b => lines.push(`- ${b}`));
231
+ } else {
232
+ lines.push(`- None`);
233
+ }
234
+ lines.push(``);
235
+ lines.push(`## Do Not Do`);
236
+ if (ctx.constraints?.length) {
237
+ ctx.constraints.forEach(c => lines.push(`- ${c}`));
238
+ } else {
239
+ lines.push(`- None specified`);
240
+ }
241
+ lines.push(``);
242
+
243
+ // All decisions across sessions
244
+ const allDecisions = (ctx.sessions || []).flatMap(s =>
245
+ (s.decisions || []).map(d => ({ ...d, date: s.date }))
246
+ );
247
+ lines.push(`## Decisions Made`);
248
+ lines.push(`| Decision | Why | Date |`);
249
+ lines.push(`|----------|-----|------|`);
250
+ if (allDecisions.length) {
251
+ allDecisions.forEach(d => lines.push(`| ${d.what} | ${d.why || ''} | ${d.date || ''} |`));
252
+ } else {
253
+ lines.push(`| _(none yet)_ | | |`);
254
+ }
255
+ lines.push(``);
256
+
257
+ // Session history (most recent first)
258
+ if (ctx.sessions?.length > 1) {
259
+ lines.push(`## Session History`);
260
+ const reversed = [...ctx.sessions].reverse();
261
+ reversed.forEach(s => {
262
+ lines.push(`### ${s.date} — ${s.summary || 'Session'}`);
263
+ if (s.gitActivity) lines.push(`_${s.gitActivity}_`);
264
+ if (s.leftOff) lines.push(`**Left off:** ${s.leftOff}`);
265
+ });
266
+ lines.push(``);
267
+ }
268
+
269
+ return lines.join('\n');
270
+ }
271
+
272
+ /** Render the bordered briefing box used by /ck:resume */
273
+ export function renderBriefingBox(ctx, _meta = {}) {
274
+ const latest = ctx.sessions?.[ctx.sessions.length - 1] || {};
275
+ const W = 57;
276
+ const pad = (str, w) => {
277
+ const s = String(str || '');
278
+ return s.length > w ? s.slice(0, w - 1) + '…' : s.padEnd(w);
279
+ };
280
+ const row = (label, value) => `│ ${label} → ${pad(value, W - label.length - 7)}│`;
281
+
282
+ const when = daysAgoLabel(ctx.sessions?.[ctx.sessions.length - 1]?.date);
283
+ const sessions = ctx.sessions?.length || 0;
284
+ const shortSessId = latest.id?.slice(0, 8) || null;
285
+
286
+ const lines = [
287
+ `┌${'─'.repeat(W)}┐`,
288
+ `│ RESUMING: ${pad(ctx.displayName ?? ctx.name, W - 12)}│`,
289
+ `│ Last session: ${pad(`${when} | Sessions: ${sessions}`, W - 16)}│`,
290
+ ];
291
+ if (shortSessId) lines.push(`│ Session ID: ${pad(shortSessId, W - 14)}│`);
292
+ lines.push(`├${'─'.repeat(W)}┤`);
293
+ lines.push(row('WHAT IT IS', ctx.description || '—'));
294
+ lines.push(row('STACK ', Array.isArray(ctx.stack) ? ctx.stack.join(', ') : (ctx.stack || '—')));
295
+ lines.push(row('PATH ', ctx.path));
296
+ if (ctx.repo) lines.push(row('REPO ', ctx.repo));
297
+ lines.push(row('GOAL ', ctx.goal || '—'));
298
+ lines.push(`├${'─'.repeat(W)}┤`);
299
+ lines.push(`│ WHERE I LEFT OFF${' '.repeat(W - 18)}│`);
300
+ const leftOffLines = (latest.leftOff || '—').split('\n').filter(Boolean);
301
+ leftOffLines.forEach(l => lines.push(`│ • ${pad(l, W - 7)}│`));
302
+ lines.push(`├${'─'.repeat(W)}┤`);
303
+ lines.push(`│ NEXT STEPS${' '.repeat(W - 12)}│`);
304
+ const steps = latest.nextSteps || [];
305
+ if (steps.length) {
306
+ steps.forEach((s, i) => lines.push(`│ ${i + 1}. ${pad(s, W - 8)}│`));
307
+ } else {
308
+ lines.push(`│ —${' '.repeat(W - 5)}│`);
309
+ }
310
+ const blockers = latest.blockers?.length ? latest.blockers.join(', ') : 'None';
311
+ lines.push(`│ BLOCKERS → ${pad(blockers, W - 13)}│`);
312
+ if (latest.gitActivity) {
313
+ lines.push(`│ GIT → ${pad(latest.gitActivity, W - 13)}│`);
314
+ }
315
+ lines.push(`└${'─'.repeat(W)}┘`);
316
+ return lines.join('\n');
317
+ }
318
+
319
+ /** Render compact info block used by /ck:info */
320
+ export function renderInfoBlock(ctx) {
321
+ const latest = ctx.sessions?.[ctx.sessions.length - 1] || {};
322
+ const sep = '─'.repeat(44);
323
+ const lines = [
324
+ `ck: ${ctx.displayName ?? ctx.name}`,
325
+ sep,
326
+ ];
327
+ lines.push(`PATH ${ctx.path}`);
328
+ if (ctx.repo) lines.push(`REPO ${ctx.repo}`);
329
+ if (latest.id) lines.push(`SESSION ${latest.id.slice(0, 8)}`);
330
+ lines.push(`GOAL ${ctx.goal || '—'}`);
331
+ lines.push(sep);
332
+ lines.push(`WHERE I LEFT OFF`);
333
+ (latest.leftOff || '—').split('\n').filter(Boolean).forEach(l => lines.push(` • ${l}`));
334
+ lines.push(`NEXT STEPS`);
335
+ (latest.nextSteps || []).forEach((s, i) => lines.push(` ${i + 1}. ${s}`));
336
+ if (!latest.nextSteps?.length) lines.push(` —`);
337
+ lines.push(`BLOCKERS`);
338
+ if (latest.blockers?.length) {
339
+ latest.blockers.forEach(b => lines.push(` • ${b}`));
340
+ } else {
341
+ lines.push(` • None`);
342
+ }
343
+ return lines.join('\n');
344
+ }
345
+
346
+ /** Render ASCII list table used by /ck:list */
347
+ export function renderListTable(entries, cwd, _todayStr) {
348
+ // entries: [{name, contextDir, path, context, lastUpdated}]
349
+ // Sorted alphabetically by contextDir before calling
350
+ const rows = entries.map((e, i) => {
351
+ const isHere = e.path === cwd;
352
+ const latest = e.context?.sessions?.[e.context.sessions.length - 1] || {};
353
+ const when = daysAgoLabel(latest.date);
354
+ const icon = stalenessIcon(latest.date);
355
+ const statusLabel = icon === '●' ? '● Active' : icon === '◐' ? '◐ Warm' : '○ Stale';
356
+ const sessId = latest.id ? latest.id.slice(0, 8) : '—';
357
+ const summary = (latest.summary || '—').slice(0, 34);
358
+ const displayName = ((e.context?.displayName ?? e.name) + (isHere ? ' <-' : '')).slice(0, 18);
359
+ return {
360
+ num: String(i + 1),
361
+ name: displayName,
362
+ status: statusLabel,
363
+ when: when.slice(0, 10),
364
+ sessId,
365
+ summary,
366
+ };
367
+ });
368
+
369
+ const cols = {
370
+ num: Math.max(1, ...rows.map(r => r.num.length)),
371
+ name: Math.max(7, ...rows.map(r => r.name.length)),
372
+ status: Math.max(6, ...rows.map(r => r.status.length)),
373
+ when: Math.max(9, ...rows.map(r => r.when.length)),
374
+ sessId: Math.max(7, ...rows.map(r => r.sessId.length)),
375
+ summary: Math.max(12, ...rows.map(r => r.summary.length)),
376
+ };
377
+
378
+ const hr = `+${'-'.repeat(cols.num + 2)}+${'-'.repeat(cols.name + 2)}+${'-'.repeat(cols.status + 2)}+${'-'.repeat(cols.when + 2)}+${'-'.repeat(cols.sessId + 2)}+${'-'.repeat(cols.summary + 2)}+`;
379
+ const cell = (val, width) => ` ${val.padEnd(width)} `;
380
+ const headerRow = `|${cell('#', cols.num)}|${cell('Project', cols.name)}|${cell('Status', cols.status)}|${cell('Last Seen', cols.when)}|${cell('Session', cols.sessId)}|${cell('Last Summary', cols.summary)}|`;
381
+
382
+ const dataRows = rows.map(r =>
383
+ `|${cell(r.num, cols.num)}|${cell(r.name, cols.name)}|${cell(r.status, cols.status)}|${cell(r.when, cols.when)}|${cell(r.sessId, cols.sessId)}|${cell(r.summary, cols.summary)}|`
384
+ );
385
+
386
+ return [hr, headerRow, hr, ...dataRows, hr].join('\n');
387
+ }
@@ -0,0 +1,224 @@
1
+ #!/usr/bin/env node
2
+ /**
3
+ * ck — Context Keeper v2
4
+ * session-start.mjs — inject compact project context on session start.
5
+ *
6
+ * Injects ~100 tokens (not ~2,500 like v1).
7
+ * SKILL.md is injected separately (still small at ~50 lines).
8
+ *
9
+ * Features:
10
+ * - Compact 5-line summary for registered projects
11
+ * - Unsaved session detection → "Last session wasn't saved. Run /ck:save."
12
+ * - Git activity since last session
13
+ * - Goal mismatch detection vs CLAUDE.md
14
+ * - Mini portfolio for unregistered directories
15
+ */
16
+
17
+ import { readFileSync, writeFileSync, existsSync } from 'fs';
18
+ import { resolve } from 'path';
19
+ import { homedir } from 'os';
20
+ import { spawnSync } from 'child_process';
21
+
22
+ const CK_HOME = resolve(homedir(), '.claude', 'ck');
23
+ const PROJECTS_FILE = resolve(CK_HOME, 'projects.json');
24
+ const CURRENT_SESSION = resolve(CK_HOME, 'current-session.json');
25
+ const SKILL_FILE = resolve(homedir(), '.claude', 'skills', 'ck', 'SKILL.md');
26
+
27
+ // ─── Helpers ──────────────────────────────────────────────────────────────────
28
+
29
+ function readJson(p) {
30
+ try { return JSON.parse(readFileSync(p, 'utf8')); } catch { return null; }
31
+ }
32
+
33
+ function daysAgo(dateStr) {
34
+ if (!dateStr) return 'unknown';
35
+ const diff = Math.floor((Date.now() - new Date(dateStr)) / 86_400_000);
36
+ if (diff === 0) return 'today';
37
+ if (diff === 1) return '1 day ago';
38
+ return `${diff} days ago`;
39
+ }
40
+
41
+ function stalenessIcon(dateStr) {
42
+ if (!dateStr) return '○';
43
+ const diff = Math.floor((Date.now() - new Date(dateStr)) / 86_400_000);
44
+ return diff < 1 ? '●' : diff <= 5 ? '◐' : '○';
45
+ }
46
+
47
+ function gitLogSince(projectPath, sinceDate) {
48
+ if (!sinceDate || !existsSync(resolve(projectPath, '.git'))) return null;
49
+ try {
50
+ const result = spawnSync(
51
+ 'git',
52
+ ['-C', projectPath, 'log', '--oneline', `--since=${sinceDate}`],
53
+ { timeout: 3000, stdio: 'pipe', encoding: 'utf8' },
54
+ );
55
+ if (result.status !== 0) return null;
56
+ const output = result.stdout.trim();
57
+ const commits = output.split('\n').filter(Boolean).length;
58
+ return commits > 0 ? `${commits} commit${commits !== 1 ? 's' : ''} since last session` : null;
59
+ } catch { return null; }
60
+ }
61
+
62
+ function extractClaudeMdGoal(projectPath) {
63
+ const p = resolve(projectPath, 'CLAUDE.md');
64
+ if (!existsSync(p)) return null;
65
+ try {
66
+ const md = readFileSync(p, 'utf8');
67
+ const m = md.match(/## Current Goal\n([\s\S]*?)(?=\n## |$)/);
68
+ return m ? m[1].trim().split('\n')[0].trim() : null;
69
+ } catch { return null; }
70
+ }
71
+
72
+ // ─── Session ID from stdin ────────────────────────────────────────────────────
73
+
74
+ function readSessionId() {
75
+ try {
76
+ const raw = readFileSync(0, 'utf8');
77
+ return JSON.parse(raw).session_id || null;
78
+ } catch { return null; }
79
+ }
80
+
81
+ // ─── Main ─────────────────────────────────────────────────────────────────────
82
+
83
+ function main() {
84
+ const cwd = process.env.PWD || process.cwd();
85
+ const sessionId = readSessionId();
86
+
87
+ // Load skill (always inject — now only ~50 lines)
88
+ const skill = existsSync(SKILL_FILE) ? readFileSync(SKILL_FILE, 'utf8') : '';
89
+
90
+ const projects = readJson(PROJECTS_FILE) || {};
91
+ const entry = projects[cwd];
92
+
93
+ // Read previous session BEFORE overwriting current-session.json
94
+ const prevSession = readJson(CURRENT_SESSION);
95
+
96
+ // Write current-session.json
97
+ try {
98
+ writeFileSync(CURRENT_SESSION, JSON.stringify({
99
+ sessionId,
100
+ projectPath: cwd,
101
+ projectName: entry?.name || null,
102
+ startedAt: new Date().toISOString(),
103
+ }, null, 2), 'utf8');
104
+ } catch { /* non-fatal */ }
105
+
106
+ const parts = [];
107
+ if (skill) parts.push(skill);
108
+
109
+ // ── REGISTERED PROJECT ────────────────────────────────────────────────────
110
+ if (entry?.contextDir) {
111
+ const contextFile = resolve(CK_HOME, 'contexts', entry.contextDir, 'context.json');
112
+ const context = readJson(contextFile);
113
+
114
+ if (context) {
115
+ const latest = context.sessions?.[context.sessions.length - 1] || {};
116
+ const sessionDate = latest.date || context.createdAt;
117
+ const sessionCount = context.sessions?.length || 0;
118
+ const displayName = context.displayName ?? context.name;
119
+
120
+ // ── Compact summary block (~100 tokens) ──────────────────────────────
121
+ const summaryLines = [
122
+ `ck: ${displayName} | ${daysAgo(sessionDate)} | ${sessionCount} session${sessionCount !== 1 ? 's' : ''}`,
123
+ `Goal: ${context.goal || '—'}`,
124
+ latest.leftOff ? `Left off: ${latest.leftOff.split('\n')[0]}` : null,
125
+ latest.nextSteps?.length ? `Next: ${latest.nextSteps.slice(0, 2).join(' · ')}` : null,
126
+ ].filter(Boolean);
127
+
128
+ // ── Unsaved session detection ─────────────────────────────────────────
129
+ if (prevSession?.sessionId && prevSession.sessionId !== sessionId) {
130
+ // Check if previous session ID exists in sessions array
131
+ const alreadySaved = context.sessions?.some(s => s.id === prevSession.sessionId);
132
+ if (!alreadySaved) {
133
+ summaryLines.push(`WARNING Last session wasn't saved — run /ck:save to capture it`);
134
+ }
135
+ }
136
+
137
+ // ── Git activity ──────────────────────────────────────────────────────
138
+ const gitLine = gitLogSince(cwd, sessionDate);
139
+ if (gitLine) summaryLines.push(`Git: ${gitLine}`);
140
+
141
+ // ── Goal mismatch detection ───────────────────────────────────────────
142
+ const claudeMdGoal = extractClaudeMdGoal(cwd);
143
+ if (claudeMdGoal && context.goal &&
144
+ claudeMdGoal.toLowerCase().trim() !== context.goal.toLowerCase().trim()) {
145
+ summaryLines.push(`WARNING Goal mismatch — ck: "${context.goal.slice(0, 40)}" · CLAUDE.md: "${claudeMdGoal.slice(0, 40)}"`);
146
+ summaryLines.push(` Run /ck:save with updated goal to sync`);
147
+ }
148
+
149
+ parts.push([
150
+ `---`,
151
+ `## ck: ${displayName}`,
152
+ ``,
153
+ summaryLines.join('\n'),
154
+ ].join('\n'));
155
+
156
+ // Instruct Claude to display compact briefing at session start
157
+ parts.push([
158
+ `---`,
159
+ `## ck: SESSION START`,
160
+ ``,
161
+ `IMPORTANT: Display the following as your FIRST message, verbatim:`,
162
+ ``,
163
+ '```',
164
+ summaryLines.join('\n'),
165
+ '```',
166
+ ``,
167
+ `After the block, add one line: "Ready — what are we working on?"`,
168
+ `If you see WARNING lines above, mention them briefly after the block.`,
169
+ ].join('\n'));
170
+
171
+ return parts;
172
+ }
173
+ }
174
+
175
+ // ── NOT IN A REGISTERED PROJECT ────────────────────────────────────────────
176
+ const entries = Object.entries(projects);
177
+ if (entries.length === 0) return parts;
178
+
179
+ // Load and sort by most recent
180
+ const recent = entries
181
+ .map(([path, info]) => {
182
+ const ctx = readJson(resolve(CK_HOME, 'contexts', info.contextDir, 'context.json'));
183
+ const latest = ctx?.sessions?.[ctx.sessions.length - 1] || {};
184
+ return { name: info.name, path, lastDate: latest.date || '', summary: latest.summary || '—', ctx };
185
+ })
186
+ .sort((a, b) => (b.lastDate > a.lastDate ? 1 : -1))
187
+ .slice(0, 3);
188
+
189
+ const miniRows = recent.map(p => {
190
+ const icon = stalenessIcon(p.lastDate);
191
+ const when = daysAgo(p.lastDate);
192
+ const name = p.name.padEnd(16).slice(0, 16);
193
+ const whenStr = when.padEnd(12).slice(0, 12);
194
+ const summary = p.summary.slice(0, 32);
195
+ return ` ${name} ${icon} ${whenStr} ${summary}`;
196
+ });
197
+
198
+ const miniStatus = [
199
+ `ck — recent projects:`,
200
+ ` ${'PROJECT'.padEnd(16)} S ${'LAST SEEN'.padEnd(12)} LAST SESSION`,
201
+ ` ${'─'.repeat(68)}`,
202
+ ...miniRows,
203
+ ``,
204
+ `Run /ck:list · /ck:resume <name> · /ck:init to register this folder`,
205
+ ].join('\n');
206
+
207
+ parts.push([
208
+ `---`,
209
+ `## ck: SESSION START`,
210
+ ``,
211
+ `IMPORTANT: Display the following as your FIRST message, verbatim:`,
212
+ ``,
213
+ '```',
214
+ miniStatus,
215
+ '```',
216
+ ].join('\n'));
217
+
218
+ return parts;
219
+ }
220
+
221
+ const parts = main();
222
+ if (parts.length > 0) {
223
+ console.log(JSON.stringify({ additionalContext: parts.join('\n\n---\n\n') }));
224
+ }