nemonix-cli 1.0.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (281) hide show
  1. package/LICENSE +21 -0
  2. package/README.md +186 -0
  3. package/dist/cli.js +6870 -0
  4. package/package.json +48 -0
  5. package/src/config/claude-code-config/.claude/commands/setup-tmux.md +0 -0
  6. package/src/config/claude-code-config/CHANGELOG.md +261 -0
  7. package/src/config/claude-code-config/README.md +1 -0
  8. package/src/config/claude-code-config/agents/action.md +36 -0
  9. package/src/config/claude-code-config/agents/clean-code-runner.md +61 -0
  10. package/src/config/claude-code-config/agents/code-reviewer.md +172 -0
  11. package/src/config/claude-code-config/agents/code-simplifier.md +49 -0
  12. package/src/config/claude-code-config/agents/explore-codebase.md +65 -0
  13. package/src/config/claude-code-config/agents/explore-docs.md +65 -0
  14. package/src/config/claude-code-config/agents/explore-fast.md +22 -0
  15. package/src/config/claude-code-config/agents/implementer.md +181 -0
  16. package/src/config/claude-code-config/agents/snipper.md +36 -0
  17. package/src/config/claude-code-config/agents/websearch.md +42 -0
  18. package/src/config/claude-code-config/commands/prompts/create-vitejs-app.md +272 -0
  19. package/src/config/claude-code-config/commands/prompts/nextjs-add-prisma-db.md +136 -0
  20. package/src/config/claude-code-config/commands/prompts/nextjs-setup-better-auth.md +173 -0
  21. package/src/config/claude-code-config/commands/prompts/nextjs-setup-project.md +200 -0
  22. package/src/config/claude-code-config/commands/prompts/prompt.md +55 -0
  23. package/src/config/claude-code-config/commands/prompts/saas-challenge-idea.md +135 -0
  24. package/src/config/claude-code-config/commands/prompts/saas-create-architecture.md +242 -0
  25. package/src/config/claude-code-config/commands/prompts/saas-create-headline.md +132 -0
  26. package/src/config/claude-code-config/commands/prompts/saas-create-landing-copywritting.md +267 -0
  27. package/src/config/claude-code-config/commands/prompts/saas-create-legals-docs.md +176 -0
  28. package/src/config/claude-code-config/commands/prompts/saas-create-logos.md +240 -0
  29. package/src/config/claude-code-config/commands/prompts/saas-create-prd.md +195 -0
  30. package/src/config/claude-code-config/commands/prompts/saas-create-tasks.md +240 -0
  31. package/src/config/claude-code-config/commands/prompts/saas-define-pricing.md +293 -0
  32. package/src/config/claude-code-config/commands/prompts/saas-find-domain-name.md +190 -0
  33. package/src/config/claude-code-config/commands/prompts/saas-implement-landing-page.md +257 -0
  34. package/src/config/claude-code-config/commands/prompts/setup-tmux.md +160 -0
  35. package/src/config/claude-code-config/commands/prompts/tools.md +148 -0
  36. package/src/config/claude-code-config/hooks/hook-post-file.ts +162 -0
  37. package/src/config/claude-code-config/scripts/.claude/skills/fix-on-my-computer/SKILL.md +81 -0
  38. package/src/config/claude-code-config/scripts/CLAUDE.md +50 -0
  39. package/src/config/claude-code-config/scripts/auto-rename-session/CLAUDE.md +59 -0
  40. package/src/config/claude-code-config/scripts/auto-rename-session/__tests__/shared.test.ts +185 -0
  41. package/src/config/claude-code-config/scripts/auto-rename-session/__tests__/title-generation.test.ts +44 -0
  42. package/src/config/claude-code-config/scripts/auto-rename-session/fixtures/sample-transcript.jsonl +3 -0
  43. package/src/config/claude-code-config/scripts/auto-rename-session/fixtures/transcript-short-message.jsonl +2 -0
  44. package/src/config/claude-code-config/scripts/auto-rename-session/fixtures/transcript-with-title.jsonl +3 -0
  45. package/src/config/claude-code-config/scripts/auto-rename-session/src/index.ts +113 -0
  46. package/src/config/claude-code-config/scripts/auto-rename-session/src/rename-all.ts +148 -0
  47. package/src/config/claude-code-config/scripts/auto-rename-session/src/shared.ts +157 -0
  48. package/src/config/claude-code-config/scripts/auto-rename-session/src/worker.ts +58 -0
  49. package/src/config/claude-code-config/scripts/biome.json +37 -0
  50. package/src/config/claude-code-config/scripts/claude-code-ai/CLAUDE.md +84 -0
  51. package/src/config/claude-code-config/scripts/claude-code-ai/__tests__/claude.test.ts +19 -0
  52. package/src/config/claude-code-config/scripts/claude-code-ai/__tests__/paths.test.ts +68 -0
  53. package/src/config/claude-code-config/scripts/claude-code-ai/claude.ts +56 -0
  54. package/src/config/claude-code-config/scripts/claude-code-ai/cli.ts +46 -0
  55. package/src/config/claude-code-config/scripts/claude-code-ai/helper/credentials.ts +76 -0
  56. package/src/config/claude-code-config/scripts/claude-code-ai/helper/index.ts +9 -0
  57. package/src/config/claude-code-config/scripts/claude-code-ai/helper/paths.ts +51 -0
  58. package/src/config/claude-code-config/scripts/command-validator/CLAUDE.md +112 -0
  59. package/src/config/claude-code-config/scripts/command-validator/README.md +147 -0
  60. package/src/config/claude-code-config/scripts/command-validator/src/__tests__/validator.test.ts +99 -0
  61. package/src/config/claude-code-config/scripts/command-validator/src/cli.ts +121 -0
  62. package/src/config/claude-code-config/scripts/command-validator/src/lib/security-rules.ts +174 -0
  63. package/src/config/claude-code-config/scripts/command-validator/src/lib/types.ts +34 -0
  64. package/src/config/claude-code-config/scripts/command-validator/src/lib/validator.ts +90 -0
  65. package/src/config/claude-code-config/scripts/package.json +43 -0
  66. package/src/config/claude-code-config/scripts/statusline/CLAUDE.md +205 -0
  67. package/src/config/claude-code-config/scripts/statusline/README.md +193 -0
  68. package/src/config/claude-code-config/scripts/statusline/__tests__/context.test.ts +229 -0
  69. package/src/config/claude-code-config/scripts/statusline/__tests__/formatters.test.ts +108 -0
  70. package/src/config/claude-code-config/scripts/statusline/__tests__/statusline.test.ts +309 -0
  71. package/src/config/claude-code-config/scripts/statusline/data/.gitkeep +0 -0
  72. package/src/config/claude-code-config/scripts/statusline/defaults.json +82 -0
  73. package/src/config/claude-code-config/scripts/statusline/docs/ARCHITECTURE.md +166 -0
  74. package/src/config/claude-code-config/scripts/statusline/fixtures/mock-transcript.jsonl +4 -0
  75. package/src/config/claude-code-config/scripts/statusline/fixtures/test-input.json +35 -0
  76. package/src/config/claude-code-config/scripts/statusline/src/analyze-daily-usage.ts +151 -0
  77. package/src/config/claude-code-config/scripts/statusline/src/commands/interactive-config.ts +515 -0
  78. package/src/config/claude-code-config/scripts/statusline/src/debug-payloads.ts +195 -0
  79. package/src/config/claude-code-config/scripts/statusline/src/index.ts +224 -0
  80. package/src/config/claude-code-config/scripts/statusline/src/lib/config-types.ts +110 -0
  81. package/src/config/claude-code-config/scripts/statusline/src/lib/config.ts +21 -0
  82. package/src/config/claude-code-config/scripts/statusline/src/lib/context.ts +103 -0
  83. package/src/config/claude-code-config/scripts/statusline/src/lib/features/limits/commands/weekly-analysis.ts +108 -0
  84. package/src/config/claude-code-config/scripts/statusline/src/lib/features/limits/index.ts +111 -0
  85. package/src/config/claude-code-config/scripts/statusline/src/lib/features/limits/types.ts +15 -0
  86. package/src/config/claude-code-config/scripts/statusline/src/lib/features/spend/commands/migrate-to-sqlite.ts +136 -0
  87. package/src/config/claude-code-config/scripts/statusline/src/lib/features/spend/commands/spend-day.ts +79 -0
  88. package/src/config/claude-code-config/scripts/statusline/src/lib/features/spend/commands/spend-month.ts +66 -0
  89. package/src/config/claude-code-config/scripts/statusline/src/lib/features/spend/commands/spend-project.ts +85 -0
  90. package/src/config/claude-code-config/scripts/statusline/src/lib/features/spend/database.ts +395 -0
  91. package/src/config/claude-code-config/scripts/statusline/src/lib/features/spend/index.ts +178 -0
  92. package/src/config/claude-code-config/scripts/statusline/src/lib/features/spend/payload-logger.ts +161 -0
  93. package/src/config/claude-code-config/scripts/statusline/src/lib/features/spend/types.ts +37 -0
  94. package/src/config/claude-code-config/scripts/statusline/src/lib/formatters.ts +426 -0
  95. package/src/config/claude-code-config/scripts/statusline/src/lib/git.ts +100 -0
  96. package/src/config/claude-code-config/scripts/statusline/src/lib/menu-factories.ts +224 -0
  97. package/src/config/claude-code-config/scripts/statusline/src/lib/presets.ts +177 -0
  98. package/src/config/claude-code-config/scripts/statusline/src/lib/render-pure.ts +516 -0
  99. package/src/config/claude-code-config/scripts/statusline/src/lib/types.ts +36 -0
  100. package/src/config/claude-code-config/scripts/statusline/src/lib/utils.ts +15 -0
  101. package/src/config/claude-code-config/scripts/statusline/src/stats.ts +119 -0
  102. package/src/config/claude-code-config/scripts/statusline/src/tests/spend-v2.test.ts +377 -0
  103. package/src/config/claude-code-config/scripts/statusline/src/tools/fixed-cost-calc.ts +317 -0
  104. package/src/config/claude-code-config/scripts/statusline/statusline.config.free.json +79 -0
  105. package/src/config/claude-code-config/scripts/statusline/statusline.config.json +100 -0
  106. package/src/config/claude-code-config/scripts/statusline/test-with-fixtures.ts +37 -0
  107. package/src/config/claude-code-config/scripts/statusline/test.ts +20 -0
  108. package/src/config/claude-code-config/scripts/statusline/tsconfig.json +27 -0
  109. package/src/config/claude-code-config/scripts/tsconfig.json +27 -0
  110. package/src/config/claude-code-config/settings.json +92 -0
  111. package/src/config/claude-code-config/skills/aibuilder-create-saas/SKILL.md +176 -0
  112. package/src/config/claude-code-config/skills/aibuilder-create-saas/references/architecture-template.md +436 -0
  113. package/src/config/claude-code-config/skills/aibuilder-create-saas/references/challenge-framework.md +289 -0
  114. package/src/config/claude-code-config/skills/aibuilder-create-saas/references/discovery-framework.md +338 -0
  115. package/src/config/claude-code-config/skills/aibuilder-create-saas/references/prd-template.md +452 -0
  116. package/src/config/claude-code-config/skills/aibuilder-create-saas/references/task-template.md +263 -0
  117. package/src/config/claude-code-config/skills/aibuilder-create-saas/references/tools.md +132 -0
  118. package/src/config/claude-code-config/skills/aibuilder-create-saas/scripts/rename-project.sh +42 -0
  119. package/src/config/claude-code-config/skills/aibuilder-create-saas/scripts/setup.sh +104 -0
  120. package/src/config/claude-code-config/skills/aibuilder-create-saas/steps/step-00-init.md +174 -0
  121. package/src/config/claude-code-config/skills/aibuilder-create-saas/steps/step-01-discovery.md +342 -0
  122. package/src/config/claude-code-config/skills/aibuilder-create-saas/steps/step-02-brainstorm.md +230 -0
  123. package/src/config/claude-code-config/skills/aibuilder-create-saas/steps/step-03-validate.md +372 -0
  124. package/src/config/claude-code-config/skills/aibuilder-create-saas/steps/step-04-prd.md +364 -0
  125. package/src/config/claude-code-config/skills/aibuilder-create-saas/steps/step-05-architecture.md +904 -0
  126. package/src/config/claude-code-config/skills/aibuilder-create-saas/steps/step-06-tasks.md +446 -0
  127. package/src/config/claude-code-config/skills/explore/SKILL.md +119 -0
  128. package/src/config/claude-code-config/skills/frontend-design/LICENSE.txt +177 -0
  129. package/src/config/claude-code-config/skills/frontend-design/SKILL.md +42 -0
  130. package/src/config/claude-code-config/skills/git-commit/SKILL.md +42 -0
  131. package/src/config/claude-code-config/skills/git-create-pr/SKILL.md +53 -0
  132. package/src/config/claude-code-config/skills/git-fix-pr-comments/SKILL.md +51 -0
  133. package/src/config/claude-code-config/skills/git-merge/SKILL.md +68 -0
  134. package/src/config/claude-code-config/skills/marketing-copywriting/SKILL.md +72 -0
  135. package/src/config/claude-code-config/skills/meta-claude-memory/SKILL.md +694 -0
  136. package/src/config/claude-code-config/skills/meta-claude-memory/references/comprehensive-example.md +175 -0
  137. package/src/config/claude-code-config/skills/meta-claude-memory/references/project-patterns.md +334 -0
  138. package/src/config/claude-code-config/skills/meta-claude-memory/references/prompting-techniques.md +411 -0
  139. package/src/config/claude-code-config/skills/meta-claude-memory/references/rules-directory-guide.md +298 -0
  140. package/src/config/claude-code-config/skills/meta-claude-memory/references/section-templates.md +347 -0
  141. package/src/config/claude-code-config/skills/meta-hooks-creator/SKILL.md +357 -0
  142. package/src/config/claude-code-config/skills/meta-hooks-creator/references/command-vs-prompt.md +287 -0
  143. package/src/config/claude-code-config/skills/meta-hooks-creator/references/examples.md +689 -0
  144. package/src/config/claude-code-config/skills/meta-hooks-creator/references/hook-types.md +495 -0
  145. package/src/config/claude-code-config/skills/meta-hooks-creator/references/input-output-schemas.md +503 -0
  146. package/src/config/claude-code-config/skills/meta-hooks-creator/references/matchers.md +517 -0
  147. package/src/config/claude-code-config/skills/meta-hooks-creator/references/troubleshooting.md +653 -0
  148. package/src/config/claude-code-config/skills/meta-prompt-creator/SKILL.md +285 -0
  149. package/src/config/claude-code-config/skills/meta-prompt-creator/references/anthropic-best-practices.md +126 -0
  150. package/src/config/claude-code-config/skills/meta-prompt-creator/references/anti-patterns.md +57 -0
  151. package/src/config/claude-code-config/skills/meta-prompt-creator/references/clarity-principles.md +54 -0
  152. package/src/config/claude-code-config/skills/meta-prompt-creator/references/context-management.md +389 -0
  153. package/src/config/claude-code-config/skills/meta-prompt-creator/references/few-shot-patterns.md +47 -0
  154. package/src/config/claude-code-config/skills/meta-prompt-creator/references/openai-best-practices.md +50 -0
  155. package/src/config/claude-code-config/skills/meta-prompt-creator/references/prompt-templates.md +110 -0
  156. package/src/config/claude-code-config/skills/meta-prompt-creator/references/reasoning-techniques.md +52 -0
  157. package/src/config/claude-code-config/skills/meta-prompt-creator/references/system-prompt-patterns.md +48 -0
  158. package/src/config/claude-code-config/skills/meta-prompt-creator/references/xml-structure.md +36 -0
  159. package/src/config/claude-code-config/skills/meta-skill-creator/LICENSE.txt +202 -0
  160. package/src/config/claude-code-config/skills/meta-skill-creator/SKILL.md +421 -0
  161. package/src/config/claude-code-config/skills/meta-skill-creator/package.json +5 -0
  162. package/src/config/claude-code-config/skills/meta-skill-creator/references/output-patterns.md +82 -0
  163. package/src/config/claude-code-config/skills/meta-skill-creator/references/progressive-disclosure-patterns.md +374 -0
  164. package/src/config/claude-code-config/skills/meta-skill-creator/references/prompting-integration.md +363 -0
  165. package/src/config/claude-code-config/skills/meta-skill-creator/references/real-world-examples.md +513 -0
  166. package/src/config/claude-code-config/skills/meta-skill-creator/references/script-patterns.md +385 -0
  167. package/src/config/claude-code-config/skills/meta-skill-creator/references/workflows.md +28 -0
  168. package/src/config/claude-code-config/skills/meta-skill-creator/references/xml-tag-guide.md +606 -0
  169. package/src/config/claude-code-config/skills/meta-skill-creator/scripts/init-skill.ts +214 -0
  170. package/src/config/claude-code-config/skills/meta-skill-creator/scripts/package-skill.ts +146 -0
  171. package/src/config/claude-code-config/skills/meta-skill-creator/scripts/validate.ts +138 -0
  172. package/src/config/claude-code-config/skills/meta-skill-workflow-creator/SKILL.md +390 -0
  173. package/src/config/claude-code-config/skills/meta-skill-workflow-creator/references/ask-patterns.md +225 -0
  174. package/src/config/claude-code-config/skills/meta-skill-workflow-creator/references/prompt-engineering.md +377 -0
  175. package/src/config/claude-code-config/skills/meta-skill-workflow-creator/references/state-management.md +275 -0
  176. package/src/config/claude-code-config/skills/meta-skill-workflow-creator/references/step-template.md +334 -0
  177. package/src/config/claude-code-config/skills/meta-skill-workflow-creator/references/workflow-patterns.md +300 -0
  178. package/src/config/claude-code-config/skills/meta-subagent-creator/SKILL.md +451 -0
  179. package/src/config/claude-code-config/skills/meta-subagent-creator/references/context-management.md +567 -0
  180. package/src/config/claude-code-config/skills/meta-subagent-creator/references/debugging-agents.md +714 -0
  181. package/src/config/claude-code-config/skills/meta-subagent-creator/references/error-handling-and-recovery.md +502 -0
  182. package/src/config/claude-code-config/skills/meta-subagent-creator/references/evaluation-and-testing.md +374 -0
  183. package/src/config/claude-code-config/skills/meta-subagent-creator/references/orchestration-patterns.md +591 -0
  184. package/src/config/claude-code-config/skills/meta-subagent-creator/references/subagents.md +599 -0
  185. package/src/config/claude-code-config/skills/meta-subagent-creator/references/writing-subagent-prompts.md +513 -0
  186. package/src/config/claude-code-config/skills/ralph-loop/SKILL.md +117 -0
  187. package/src/config/claude-code-config/skills/ralph-loop/scripts/setup.sh +278 -0
  188. package/src/config/claude-code-config/skills/ralph-loop/steps/step-00-init.md +215 -0
  189. package/src/config/claude-code-config/skills/ralph-loop/steps/step-01-interactive-prd.md +366 -0
  190. package/src/config/claude-code-config/skills/ralph-loop/steps/step-02-create-stories.md +273 -0
  191. package/src/config/claude-code-config/skills/ralph-loop/steps/step-03-finish.md +245 -0
  192. package/src/config/claude-code-config/skills/ralph-tasks/SKILL.md +88 -0
  193. package/src/config/claude-code-config/skills/ralph-tasks/scripts/add-task.sh +36 -0
  194. package/src/config/claude-code-config/skills/ralph-tasks/scripts/list-tasks.sh +66 -0
  195. package/src/config/claude-code-config/skills/ralph-tasks/scripts/remove-task.sh +47 -0
  196. package/src/config/claude-code-config/skills/ralph-tasks/scripts/setup.sh +201 -0
  197. package/src/config/claude-code-config/skills/ralph-tasks/steps/action-add-search.md +131 -0
  198. package/src/config/claude-code-config/skills/ralph-tasks/steps/action-add.md +46 -0
  199. package/src/config/claude-code-config/skills/ralph-tasks/steps/action-init.md +123 -0
  200. package/src/config/claude-code-config/skills/ralph-tasks/steps/action-list.md +58 -0
  201. package/src/config/claude-code-config/skills/ralph-tasks/steps/action-remove.md +48 -0
  202. package/src/config/claude-code-config/skills/ralph-tasks/steps/action-setup.md +46 -0
  203. package/src/config/claude-code-config/skills/ralph-tasks/steps/action-wake-up.md +62 -0
  204. package/src/config/claude-code-config/skills/utils-fix-errors/SKILL.md +61 -0
  205. package/src/config/claude-code-config/skills/utils-fix-grammar/SKILL.md +59 -0
  206. package/src/config/claude-code-config/skills/utils-oneshot/SKILL.md +56 -0
  207. package/src/config/claude-code-config/skills/utils-refactor/SKILL.md +89 -0
  208. package/src/config/claude-code-config/skills/utils-save-docs/SKILL.md +74 -0
  209. package/src/config/claude-code-config/skills/utils-ultrathink/SKILL.md +42 -0
  210. package/src/config/claude-code-config/skills/workflow-apex/SKILL.md +116 -0
  211. package/src/config/claude-code-config/skills/workflow-apex/scripts/setup-templates.sh +144 -0
  212. package/src/config/claude-code-config/skills/workflow-apex/scripts/update-progress.sh +80 -0
  213. package/src/config/claude-code-config/skills/workflow-apex/steps/step-00-init.md +273 -0
  214. package/src/config/claude-code-config/skills/workflow-apex/steps/step-00b-branch.md +126 -0
  215. package/src/config/claude-code-config/skills/workflow-apex/steps/step-00b-economy.md +244 -0
  216. package/src/config/claude-code-config/skills/workflow-apex/steps/step-00b-interactive.md +165 -0
  217. package/src/config/claude-code-config/skills/workflow-apex/steps/step-00b-save.md +123 -0
  218. package/src/config/claude-code-config/skills/workflow-apex/steps/step-01-analyze.md +361 -0
  219. package/src/config/claude-code-config/skills/workflow-apex/steps/step-02-plan.md +422 -0
  220. package/src/config/claude-code-config/skills/workflow-apex/steps/step-02b-tasks.md +301 -0
  221. package/src/config/claude-code-config/skills/workflow-apex/steps/step-03-execute-teams.md +297 -0
  222. package/src/config/claude-code-config/skills/workflow-apex/steps/step-03-execute.md +239 -0
  223. package/src/config/claude-code-config/skills/workflow-apex/steps/step-04-validate.md +264 -0
  224. package/src/config/claude-code-config/skills/workflow-apex/steps/step-05-examine.md +294 -0
  225. package/src/config/claude-code-config/skills/workflow-apex/steps/step-06-resolve.md +237 -0
  226. package/src/config/claude-code-config/skills/workflow-apex/steps/step-07-tests.md +250 -0
  227. package/src/config/claude-code-config/skills/workflow-apex/steps/step-08-run-tests.md +308 -0
  228. package/src/config/claude-code-config/skills/workflow-apex/steps/step-09-finish.md +193 -0
  229. package/src/config/claude-code-config/skills/workflow-apex/templates/00-context.md +53 -0
  230. package/src/config/claude-code-config/skills/workflow-apex/templates/01-analyze.md +10 -0
  231. package/src/config/claude-code-config/skills/workflow-apex/templates/02-plan.md +10 -0
  232. package/src/config/claude-code-config/skills/workflow-apex/templates/03-execute.md +10 -0
  233. package/src/config/claude-code-config/skills/workflow-apex/templates/04-validate.md +10 -0
  234. package/src/config/claude-code-config/skills/workflow-apex/templates/05-examine.md +10 -0
  235. package/src/config/claude-code-config/skills/workflow-apex/templates/06-resolve.md +10 -0
  236. package/src/config/claude-code-config/skills/workflow-apex/templates/07-tests.md +10 -0
  237. package/src/config/claude-code-config/skills/workflow-apex/templates/08-run-tests.md +10 -0
  238. package/src/config/claude-code-config/skills/workflow-apex/templates/09-finish.md +10 -0
  239. package/src/config/claude-code-config/skills/workflow-apex/templates/README.md +195 -0
  240. package/src/config/claude-code-config/skills/workflow-apex/templates/step-complete.md +7 -0
  241. package/src/config/claude-code-config/skills/workflow-brainstorm/SKILL.md +127 -0
  242. package/src/config/claude-code-config/skills/workflow-brainstorm/steps/step-01-explore.md +230 -0
  243. package/src/config/claude-code-config/skills/workflow-brainstorm/steps/step-02-challenge.md +238 -0
  244. package/src/config/claude-code-config/skills/workflow-brainstorm/steps/step-03-synthesize.md +337 -0
  245. package/src/config/claude-code-config/skills/workflow-brainstorm/steps/step-04-action.md +285 -0
  246. package/src/config/claude-code-config/skills/workflow-ci-fixer/SKILL.md +150 -0
  247. package/src/config/claude-code-config/skills/workflow-ci-fixer/references/cli-commands.md +122 -0
  248. package/src/config/claude-code-config/skills/workflow-ci-fixer/references/github-cli.md +243 -0
  249. package/src/config/claude-code-config/skills/workflow-ci-fixer/references/troubleshooting.md +362 -0
  250. package/src/config/claude-code-config/skills/workflow-ci-fixer/references/vercel-cli.md +192 -0
  251. package/src/config/claude-code-config/skills/workflow-ci-fixer/steps/step-00-init.md +157 -0
  252. package/src/config/claude-code-config/skills/workflow-ci-fixer/steps/step-01-watch-ci.md +192 -0
  253. package/src/config/claude-code-config/skills/workflow-ci-fixer/steps/step-02-analyze-errors.md +263 -0
  254. package/src/config/claude-code-config/skills/workflow-ci-fixer/steps/step-03-fix-locally.md +312 -0
  255. package/src/config/claude-code-config/skills/workflow-ci-fixer/steps/step-04-commit-push.md +206 -0
  256. package/src/config/claude-code-config/skills/workflow-ci-fixer/steps/step-05-cleanup.md +196 -0
  257. package/src/config/claude-code-config/skills/workflow-clean-code/SKILL.md +129 -0
  258. package/src/config/claude-code-config/skills/workflow-clean-code/references/general-clean-code.md +347 -0
  259. package/src/config/claude-code-config/skills/workflow-clean-code/references/nextjs-clean-code.md +458 -0
  260. package/src/config/claude-code-config/skills/workflow-clean-code/references/react-clean-code.md +406 -0
  261. package/src/config/claude-code-config/skills/workflow-clean-code/references/tanstack-query-best-practices.md +516 -0
  262. package/src/config/claude-code-config/skills/workflow-clean-code/references/zustand-best-practices.md +491 -0
  263. package/src/config/claude-code-config/skills/workflow-clean-code/steps/step-01-scan.md +137 -0
  264. package/src/config/claude-code-config/skills/workflow-clean-code/steps/step-02-apply.md +152 -0
  265. package/src/config/claude-code-config/skills/workflow-clean-code/steps/step-03-verify.md +143 -0
  266. package/src/config/claude-code-config/skills/workflow-debug/SKILL.md +126 -0
  267. package/src/config/claude-code-config/skills/workflow-debug/references/log-technique.md +285 -0
  268. package/src/config/claude-code-config/skills/workflow-debug/steps/step-00-init.md +94 -0
  269. package/src/config/claude-code-config/skills/workflow-debug/steps/step-01-analyze.md +180 -0
  270. package/src/config/claude-code-config/skills/workflow-debug/steps/step-01b-log-instrumentation.md +271 -0
  271. package/src/config/claude-code-config/skills/workflow-debug/steps/step-02-find-solutions.md +141 -0
  272. package/src/config/claude-code-config/skills/workflow-debug/steps/step-03-propose.md +142 -0
  273. package/src/config/claude-code-config/skills/workflow-debug/steps/step-04-fix.md +176 -0
  274. package/src/config/claude-code-config/skills/workflow-debug/steps/step-05-verify.md +279 -0
  275. package/src/config/claude-code-config/skills/workflow-review-code/SKILL.md +219 -0
  276. package/src/config/claude-code-config/skills/workflow-review-code/references/clean-code-principles.md +140 -0
  277. package/src/config/claude-code-config/skills/workflow-review-code/references/code-quality-metrics.md +174 -0
  278. package/src/config/claude-code-config/skills/workflow-review-code/references/feedback-patterns.md +149 -0
  279. package/src/config/claude-code-config/skills/workflow-review-code/references/security-checklist.md +127 -0
  280. package/src/config/claude-code-config/song/finish.mp3 +0 -0
  281. package/src/config/claude-code-config/song/need-human.mp3 +0 -0
@@ -0,0 +1,108 @@
1
+ #!/usr/bin/env bun
2
+
3
+ import { colors, formatProgressBar } from "../../../formatters";
4
+ import { getUsageLimits } from "../index";
5
+
6
+ const WEEKLY_HOURS = 168; // 7 days * 24 hours
7
+
8
+ function calculateTimeProgress(resetsAt: string | null): {
9
+ hoursRemaining: number;
10
+ hoursElapsed: number;
11
+ timeElapsedPercent: number;
12
+ } {
13
+ if (!resetsAt) {
14
+ return {
15
+ hoursRemaining: 0,
16
+ hoursElapsed: WEEKLY_HOURS,
17
+ timeElapsedPercent: 100,
18
+ };
19
+ }
20
+
21
+ const resetDate = new Date(resetsAt);
22
+ const now = new Date();
23
+ const diffMs = resetDate.getTime() - now.getTime();
24
+ const hoursRemaining = Math.max(0, diffMs / 3600000);
25
+ const hoursElapsed = WEEKLY_HOURS - hoursRemaining;
26
+ const timeElapsedPercent = (hoursElapsed / WEEKLY_HOURS) * 100;
27
+
28
+ return { hoursRemaining, hoursElapsed, timeElapsedPercent };
29
+ }
30
+
31
+ function formatDelta(delta: number): string {
32
+ const sign = delta >= 0 ? "+" : "";
33
+ const value = `${sign}${delta.toFixed(1)}%`;
34
+
35
+ if (delta > 5) return colors.green(value);
36
+ if (delta > 0) return colors.lightGray(value);
37
+ if (delta > -5) return colors.yellow(value);
38
+ if (delta > -15) return colors.orange(value);
39
+ return colors.red(value);
40
+ }
41
+
42
+ async function main() {
43
+ const limits = await getUsageLimits();
44
+
45
+ console.log("");
46
+ console.log(colors.bold("šŸ“Š Weekly Usage Analysis"));
47
+ console.log(colors.gray("─".repeat(40)));
48
+
49
+ if (!limits.seven_day) {
50
+ console.log(colors.red("No weekly limit data available"));
51
+ return;
52
+ }
53
+
54
+ const { utilization, resets_at } = limits.seven_day;
55
+ const { timeElapsedPercent } = calculateTimeProgress(resets_at);
56
+
57
+ const expectedUsage = timeElapsedPercent;
58
+ const delta = utilization - expectedUsage;
59
+
60
+ console.log("");
61
+ console.log(colors.gray("Weekly Limits Overview"));
62
+ console.log(colors.gray("─".repeat(50)));
63
+ console.log("");
64
+
65
+ const usageBar = formatProgressBar({
66
+ percentage: utilization,
67
+ length: 20,
68
+ style: "filled",
69
+ colorMode: "blue",
70
+ background: "none",
71
+ });
72
+ console.log(
73
+ `${colors.blue("Usage")} ${usageBar} ${colors.lightGray(utilization.toFixed(1))}${colors.gray("%")}`,
74
+ );
75
+
76
+ const timeBar = formatProgressBar({
77
+ percentage: timeElapsedPercent,
78
+ length: 20,
79
+ style: "filled",
80
+ colorMode: "purple",
81
+ background: "none",
82
+ });
83
+ console.log(
84
+ `${colors.purple("Time ")} ${timeBar} ${colors.lightGray(timeElapsedPercent.toFixed(1))}${colors.gray("%")} ${colors.gray("Ā·")} ${formatDelta(delta)}`,
85
+ );
86
+
87
+ console.log("");
88
+ console.log(colors.gray("─".repeat(50)));
89
+ console.log("");
90
+
91
+ if (limits.five_hour) {
92
+ console.log(colors.gray("─".repeat(40)));
93
+ console.log("");
94
+ console.log(colors.gray("Session Limit (5h):"));
95
+ console.log(
96
+ ` ${formatProgressBar({
97
+ percentage: limits.five_hour.utilization,
98
+ length: 15,
99
+ style: "filled",
100
+ colorMode: "progressive",
101
+ background: "none",
102
+ })} ${colors.lightGray(limits.five_hour.utilization.toString())}${colors.gray("%")}`,
103
+ );
104
+ console.log("");
105
+ }
106
+ }
107
+
108
+ main();
@@ -0,0 +1,111 @@
1
+ import { existsSync } from "node:fs";
2
+ import { readFile, writeFile } from "node:fs/promises";
3
+ import { join } from "node:path";
4
+ import { getClaudeCodeTokenSafe } from "../../../../../claude-code-ai/helper/credentials";
5
+ import type { CachedUsageLimits, UsageLimits } from "./types";
6
+
7
+ const CACHE_DURATION_MS = 60 * 1000; // 1 minute
8
+
9
+ function getCacheFilePath(): string {
10
+ const projectRoot = join(import.meta.dir, "..", "..", "..", "..");
11
+ return join(projectRoot, "data", "usage-limits-cache.json");
12
+ }
13
+
14
+ export async function getCredentials(): Promise<string | null> {
15
+ return getClaudeCodeTokenSafe();
16
+ }
17
+
18
+ export async function fetchUsageLimits(
19
+ token: string,
20
+ ): Promise<UsageLimits | null> {
21
+ try {
22
+ const response = await fetch("https://api.anthropic.com/api/oauth/usage", {
23
+ method: "GET",
24
+ headers: {
25
+ Accept: "application/json, text/plain, */*",
26
+ "Content-Type": "application/json",
27
+ "User-Agent": "claude-code/2.0.31",
28
+ Authorization: `Bearer ${token}`,
29
+ "anthropic-beta": "oauth-2025-04-20",
30
+ "Accept-Encoding": "gzip, compress, deflate, br",
31
+ },
32
+ });
33
+
34
+ if (!response.ok) {
35
+ return null;
36
+ }
37
+
38
+ const data = await response.json();
39
+
40
+ return {
41
+ five_hour: data.five_hour || null,
42
+ seven_day: data.seven_day || null,
43
+ };
44
+ } catch {
45
+ return null;
46
+ }
47
+ }
48
+
49
+ async function loadCache(): Promise<CachedUsageLimits | null> {
50
+ try {
51
+ const cacheFile = getCacheFilePath();
52
+ if (!existsSync(cacheFile)) {
53
+ return null;
54
+ }
55
+
56
+ const content = await readFile(cacheFile, "utf-8");
57
+ const cached: CachedUsageLimits = JSON.parse(content);
58
+
59
+ const now = Date.now();
60
+ if (now - cached.timestamp < CACHE_DURATION_MS) {
61
+ return cached;
62
+ }
63
+
64
+ return null;
65
+ } catch {
66
+ return null;
67
+ }
68
+ }
69
+
70
+ async function saveCache(data: UsageLimits): Promise<void> {
71
+ try {
72
+ const cacheFile = getCacheFilePath();
73
+ const cached: CachedUsageLimits = {
74
+ data,
75
+ timestamp: Date.now(),
76
+ };
77
+
78
+ await writeFile(cacheFile, JSON.stringify(cached, null, 2));
79
+ } catch {
80
+ // Fail silently
81
+ }
82
+ }
83
+
84
+ export async function getUsageLimits(): Promise<UsageLimits> {
85
+ try {
86
+ const cached = await loadCache();
87
+ if (cached) {
88
+ return cached.data;
89
+ }
90
+
91
+ const token = await getCredentials();
92
+
93
+ if (!token) {
94
+ return { five_hour: null, seven_day: null };
95
+ }
96
+
97
+ const limits = await fetchUsageLimits(token);
98
+
99
+ if (!limits) {
100
+ return { five_hour: null, seven_day: null };
101
+ }
102
+
103
+ await saveCache(limits);
104
+
105
+ return limits;
106
+ } catch {
107
+ return { five_hour: null, seven_day: null };
108
+ }
109
+ }
110
+
111
+ export type { CachedUsageLimits, UsageLimits } from "./types";
@@ -0,0 +1,15 @@
1
+ export interface UsageLimits {
2
+ five_hour: {
3
+ utilization: number;
4
+ resets_at: string | null;
5
+ } | null;
6
+ seven_day: {
7
+ utilization: number;
8
+ resets_at: string | null;
9
+ } | null;
10
+ }
11
+
12
+ export interface CachedUsageLimits {
13
+ data: UsageLimits;
14
+ timestamp: number;
15
+ }
@@ -0,0 +1,136 @@
1
+ #!/usr/bin/env bun
2
+
3
+ import { existsSync } from "node:fs";
4
+ import { readFile } from "node:fs/promises";
5
+ import { join } from "node:path";
6
+ import { getDb, upsertPeriod, upsertSession } from "../database";
7
+
8
+ interface OldSpendSession {
9
+ id: string;
10
+ cost: number;
11
+ date: string;
12
+ duration_ms: number;
13
+ lines_added: number;
14
+ lines_removed: number;
15
+ cwd: string;
16
+ last_resets_at?: string;
17
+ }
18
+
19
+ interface OldSpendData {
20
+ sessions: OldSpendSession[];
21
+ }
22
+
23
+ interface OldDailyUsageEntry {
24
+ date: string;
25
+ utilization: number;
26
+ resets_at: string;
27
+ timestamp: number;
28
+ period_cost: number;
29
+ }
30
+
31
+ async function loadOldSpendData(): Promise<OldSpendData | null> {
32
+ const projectRoot = join(import.meta.dir, "..", "..", "..", "..", "..");
33
+ const spendFile = join(projectRoot, "data", "spend.json");
34
+
35
+ if (!existsSync(spendFile)) {
36
+ console.log("No spend.json found, skipping session migration");
37
+ return null;
38
+ }
39
+
40
+ const content = await readFile(spendFile, "utf-8");
41
+ return JSON.parse(content);
42
+ }
43
+
44
+ async function loadOldDailyUsage(): Promise<OldDailyUsageEntry[]> {
45
+ const projectRoot = join(import.meta.dir, "..", "..", "..", "..", "..");
46
+ const dailyUsageFile = join(projectRoot, "data", "daily-usage.json");
47
+
48
+ if (!existsSync(dailyUsageFile)) {
49
+ console.log("No daily-usage.json found, skipping period migration");
50
+ return [];
51
+ }
52
+
53
+ const content = await readFile(dailyUsageFile, "utf-8");
54
+ return JSON.parse(content);
55
+ }
56
+
57
+ async function migrate() {
58
+ console.log("šŸ”„ Starting migration to SQLite...\n");
59
+
60
+ const db = getDb();
61
+ console.log("āœ… Database initialized\n");
62
+
63
+ const oldSpendData = await loadOldSpendData();
64
+ if (oldSpendData) {
65
+ console.log(`šŸ“¦ Migrating ${oldSpendData.sessions.length} sessions...`);
66
+
67
+ let migrated = 0;
68
+ let skipped = 0;
69
+
70
+ for (const session of oldSpendData.sessions) {
71
+ try {
72
+ upsertSession({
73
+ session_id: session.id,
74
+ total_cost: session.cost,
75
+ cwd: session.cwd,
76
+ date: session.date,
77
+ duration_ms: session.duration_ms,
78
+ lines_added: session.lines_added,
79
+ lines_removed: session.lines_removed,
80
+ last_resets_at: session.last_resets_at ?? null,
81
+ cumulative_counted: 0,
82
+ });
83
+ migrated++;
84
+ } catch (error) {
85
+ console.error(` āŒ Failed to migrate session ${session.id}:`, error);
86
+ skipped++;
87
+ }
88
+ }
89
+
90
+ console.log(` āœ… Migrated: ${migrated}`);
91
+ console.log(` āš ļø Skipped: ${skipped}\n`);
92
+ }
93
+
94
+ const oldDailyUsage = await loadOldDailyUsage();
95
+ if (oldDailyUsage.length > 0) {
96
+ console.log(`šŸ“… Migrating ${oldDailyUsage.length} period entries...`);
97
+
98
+ let migrated = 0;
99
+
100
+ for (const entry of oldDailyUsage) {
101
+ try {
102
+ upsertPeriod({
103
+ period_id: entry.resets_at,
104
+ total_cost: 0,
105
+ utilization: entry.utilization,
106
+ date: entry.date,
107
+ });
108
+ migrated++;
109
+ } catch (error) {
110
+ console.error(
111
+ ` āŒ Failed to migrate period ${entry.resets_at}:`,
112
+ error,
113
+ );
114
+ }
115
+ }
116
+
117
+ console.log(` āœ… Migrated: ${migrated}\n`);
118
+ }
119
+
120
+ const sessionCount = db
121
+ .query<{ count: number }, []>("SELECT COUNT(*) as count FROM sessions")
122
+ .get();
123
+ const periodCount = db
124
+ .query<{ count: number }, []>("SELECT COUNT(*) as count FROM periods")
125
+ .get();
126
+
127
+ console.log("šŸ“Š Migration Summary:");
128
+ console.log(` Sessions in DB: ${sessionCount?.count ?? 0}`);
129
+ console.log(` Periods in DB: ${periodCount?.count ?? 0}`);
130
+ console.log("\nāœ… Migration complete!");
131
+ console.log("\nāš ļø Note: Period costs have been reset to 0.");
132
+ console.log(" They will be recalculated correctly as you use Claude Code.");
133
+ console.log(" The old JSON files are preserved as backup.");
134
+ }
135
+
136
+ migrate().catch(console.error);
@@ -0,0 +1,79 @@
1
+ #!/usr/bin/env bun
2
+
3
+ import { table } from "table";
4
+ import { formatCost, formatDuration } from "../../../formatters";
5
+ import { getSessionsByDate } from "../database";
6
+
7
+ function parseDate(input?: string): string {
8
+ if (!input) {
9
+ return new Date().toISOString().split("T")[0];
10
+ }
11
+
12
+ // Try parsing as YYYY-MM-DD
13
+ if (/^\d{4}-\d{2}-\d{2}$/.test(input)) {
14
+ return input;
15
+ }
16
+
17
+ // Try parsing as MM-DD (current year)
18
+ if (/^\d{2}-\d{2}$/.test(input)) {
19
+ const year = new Date().getFullYear();
20
+ return `${year}-${input}`;
21
+ }
22
+
23
+ // Try parsing as natural date
24
+ const parsed = new Date(input);
25
+ if (!Number.isNaN(parsed.getTime())) {
26
+ return parsed.toISOString().split("T")[0];
27
+ }
28
+
29
+ console.log(`āš ļø Invalid date format: "${input}". Using today.`);
30
+ return new Date().toISOString().split("T")[0];
31
+ }
32
+
33
+ function isToday(date: string): boolean {
34
+ return date === new Date().toISOString().split("T")[0];
35
+ }
36
+
37
+ async function main() {
38
+ const dateArg = process.env.DATE || process.argv[2];
39
+ const targetDate = parseDate(dateArg);
40
+ const sessions = getSessionsByDate(targetDate);
41
+
42
+ const dateLabel = isToday(targetDate) ? "Today" : targetDate;
43
+
44
+ if (sessions.length === 0) {
45
+ console.log(`šŸ“Š No sessions for ${dateLabel}`);
46
+ return;
47
+ }
48
+
49
+ const totalCost = sessions.reduce((sum, s) => sum + s.total_cost, 0);
50
+ const totalDuration = sessions.reduce((sum, s) => sum + s.duration_ms, 0);
51
+
52
+ console.log(`\nšŸ“Š Spend for ${dateLabel}\n`);
53
+ console.log(`Sessions: ${sessions.length}`);
54
+ console.log(`Total Cost: $${formatCost(totalCost)}`);
55
+ console.log(`Total Duration: ${formatDuration(totalDuration)}`);
56
+ console.log("\nšŸ“ Sessions:\n");
57
+
58
+ const filteredSessions = sessions
59
+ .filter((session) => session.total_cost >= 0.1)
60
+ .sort((a, b) => b.total_cost - a.total_cost);
61
+
62
+ const tableData = [
63
+ ["ID", "Cost", "Duration", "Changes", "Directory"],
64
+ ...filteredSessions.map((session) => [
65
+ session.session_id.slice(0, 8),
66
+ `$${formatCost(session.total_cost)}`,
67
+ formatDuration(session.duration_ms),
68
+ `+${session.lines_added} -${session.lines_removed}`,
69
+ session.cwd.replace(
70
+ /^(?:\/Users\/[^/]+\/|[A-Z]:\\Users\\[^\\]+\\)/,
71
+ "~/",
72
+ ),
73
+ ]),
74
+ ];
75
+
76
+ console.log(table(tableData));
77
+ }
78
+
79
+ main();
@@ -0,0 +1,66 @@
1
+ #!/usr/bin/env bun
2
+
3
+ import { table } from "table";
4
+ import { formatCost, formatDuration } from "../../../formatters";
5
+ import { getAllSessions } from "../database";
6
+ import { getMonthStart } from "../index";
7
+ import type { SessionRow } from "../types";
8
+
9
+ async function main() {
10
+ const allSessions = getAllSessions();
11
+ const monthStart = getMonthStart();
12
+ const monthStartStr = monthStart.toISOString().split("T")[0];
13
+
14
+ const monthSessions = allSessions.filter((s) => s.date >= monthStartStr);
15
+
16
+ if (monthSessions.length === 0) {
17
+ console.log("šŸ“Š No sessions this month");
18
+ return;
19
+ }
20
+
21
+ const totalCost = monthSessions.reduce((sum, s) => sum + s.total_cost, 0);
22
+ const totalDuration = monthSessions.reduce(
23
+ (sum, s) => sum + s.duration_ms,
24
+ 0,
25
+ );
26
+
27
+ const sessionsByDate = monthSessions.reduce(
28
+ (acc, session) => {
29
+ if (!acc[session.date]) {
30
+ acc[session.date] = [];
31
+ }
32
+ acc[session.date].push(session);
33
+ return acc;
34
+ },
35
+ {} as Record<string, SessionRow[]>,
36
+ );
37
+
38
+ const monthName = monthStart.toLocaleString("default", { month: "long" });
39
+
40
+ console.log(`\nšŸ“Š ${monthName}'s Spend\n`);
41
+ console.log(`Sessions: ${monthSessions.length}`);
42
+ console.log(`Total Cost: $${formatCost(totalCost)}`);
43
+ console.log(`Total Duration: ${formatDuration(totalDuration)}`);
44
+ console.log("\nšŸ“… By Date:\n");
45
+
46
+ const sortedDates = Object.keys(sessionsByDate).sort();
47
+
48
+ const tableData = [
49
+ ["Date", "Cost", "Duration", "Sessions"],
50
+ ...sortedDates.map((date) => {
51
+ const sessions = sessionsByDate[date];
52
+ const dayCost = sessions.reduce((sum, s) => sum + s.total_cost, 0);
53
+ const dayDuration = sessions.reduce((sum, s) => sum + s.duration_ms, 0);
54
+ return [
55
+ date,
56
+ `$${formatCost(dayCost)}`,
57
+ formatDuration(dayDuration),
58
+ sessions.length.toString(),
59
+ ];
60
+ }),
61
+ ];
62
+
63
+ console.log(table(tableData));
64
+ }
65
+
66
+ main();
@@ -0,0 +1,85 @@
1
+ #!/usr/bin/env bun
2
+
3
+ import { table } from "table";
4
+ import { formatCost, formatDuration } from "../../../formatters";
5
+ import { getAllSessions } from "../database";
6
+
7
+ function main() {
8
+ const projectName = process.argv[2];
9
+
10
+ if (!projectName) {
11
+ console.log("Usage: bun run spend:project <project-name>");
12
+ console.log('Example: bun run spend:project "statusline"');
13
+ process.exit(1);
14
+ }
15
+
16
+ const allSessions = getAllSessions();
17
+ const matchingSessions = allSessions.filter((session) =>
18
+ session.cwd.endsWith(projectName),
19
+ );
20
+
21
+ if (matchingSessions.length === 0) {
22
+ console.log(`\nNo sessions found for project ending with "${projectName}"`);
23
+ console.log("\nAvailable project paths:");
24
+ const uniquePaths = [...new Set(allSessions.map((s) => s.cwd))];
25
+ for (const path of uniquePaths.slice(0, 10)) {
26
+ console.log(
27
+ ` - ${path.replace(/^(?:\/Users\/[^/]+\/|[A-Z]:\\Users\\[^\\]+\\)/, "~/")}`,
28
+ );
29
+ }
30
+ return;
31
+ }
32
+
33
+ const totalCost = matchingSessions.reduce((sum, s) => sum + s.total_cost, 0);
34
+ const totalDuration = matchingSessions.reduce(
35
+ (sum, s) => sum + s.duration_ms,
36
+ 0,
37
+ );
38
+ const totalLinesAdded = matchingSessions.reduce(
39
+ (sum, s) => sum + s.lines_added,
40
+ 0,
41
+ );
42
+ const totalLinesRemoved = matchingSessions.reduce(
43
+ (sum, s) => sum + s.lines_removed,
44
+ 0,
45
+ );
46
+
47
+ console.log(`\nšŸ“Š Spend for project: ${projectName}\n`);
48
+ console.log(`Sessions: ${matchingSessions.length}`);
49
+ console.log(`Total Cost: $${formatCost(totalCost)}`);
50
+ console.log(`Total Duration: ${formatDuration(totalDuration)}`);
51
+ console.log(`Total Changes: +${totalLinesAdded} -${totalLinesRemoved}`);
52
+ console.log("\nšŸ“ Sessions by date:\n");
53
+
54
+ const sessionsByDate = matchingSessions.reduce(
55
+ (acc, session) => {
56
+ const date = session.date;
57
+ if (!acc[date]) {
58
+ acc[date] = { cost: 0, duration: 0, count: 0 };
59
+ }
60
+ acc[date].cost += session.total_cost;
61
+ acc[date].duration += session.duration_ms;
62
+ acc[date].count += 1;
63
+ return acc;
64
+ },
65
+ {} as Record<string, { cost: number; duration: number; count: number }>,
66
+ );
67
+
68
+ const sortedDates = Object.entries(sessionsByDate).sort(([a], [b]) =>
69
+ b.localeCompare(a),
70
+ );
71
+
72
+ const tableData = [
73
+ ["Date", "Sessions", "Cost", "Duration"],
74
+ ...sortedDates.map(([date, data]) => [
75
+ date,
76
+ String(data.count),
77
+ `$${formatCost(data.cost)}`,
78
+ formatDuration(data.duration),
79
+ ]),
80
+ ];
81
+
82
+ console.log(table(tableData));
83
+ }
84
+
85
+ main();