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.
- package/LICENSE +21 -0
- package/README.md +186 -0
- package/dist/cli.js +6870 -0
- package/package.json +48 -0
- package/src/config/claude-code-config/.claude/commands/setup-tmux.md +0 -0
- package/src/config/claude-code-config/CHANGELOG.md +261 -0
- package/src/config/claude-code-config/README.md +1 -0
- package/src/config/claude-code-config/agents/action.md +36 -0
- package/src/config/claude-code-config/agents/clean-code-runner.md +61 -0
- package/src/config/claude-code-config/agents/code-reviewer.md +172 -0
- package/src/config/claude-code-config/agents/code-simplifier.md +49 -0
- package/src/config/claude-code-config/agents/explore-codebase.md +65 -0
- package/src/config/claude-code-config/agents/explore-docs.md +65 -0
- package/src/config/claude-code-config/agents/explore-fast.md +22 -0
- package/src/config/claude-code-config/agents/implementer.md +181 -0
- package/src/config/claude-code-config/agents/snipper.md +36 -0
- package/src/config/claude-code-config/agents/websearch.md +42 -0
- package/src/config/claude-code-config/commands/prompts/create-vitejs-app.md +272 -0
- package/src/config/claude-code-config/commands/prompts/nextjs-add-prisma-db.md +136 -0
- package/src/config/claude-code-config/commands/prompts/nextjs-setup-better-auth.md +173 -0
- package/src/config/claude-code-config/commands/prompts/nextjs-setup-project.md +200 -0
- package/src/config/claude-code-config/commands/prompts/prompt.md +55 -0
- package/src/config/claude-code-config/commands/prompts/saas-challenge-idea.md +135 -0
- package/src/config/claude-code-config/commands/prompts/saas-create-architecture.md +242 -0
- package/src/config/claude-code-config/commands/prompts/saas-create-headline.md +132 -0
- package/src/config/claude-code-config/commands/prompts/saas-create-landing-copywritting.md +267 -0
- package/src/config/claude-code-config/commands/prompts/saas-create-legals-docs.md +176 -0
- package/src/config/claude-code-config/commands/prompts/saas-create-logos.md +240 -0
- package/src/config/claude-code-config/commands/prompts/saas-create-prd.md +195 -0
- package/src/config/claude-code-config/commands/prompts/saas-create-tasks.md +240 -0
- package/src/config/claude-code-config/commands/prompts/saas-define-pricing.md +293 -0
- package/src/config/claude-code-config/commands/prompts/saas-find-domain-name.md +190 -0
- package/src/config/claude-code-config/commands/prompts/saas-implement-landing-page.md +257 -0
- package/src/config/claude-code-config/commands/prompts/setup-tmux.md +160 -0
- package/src/config/claude-code-config/commands/prompts/tools.md +148 -0
- package/src/config/claude-code-config/hooks/hook-post-file.ts +162 -0
- package/src/config/claude-code-config/scripts/.claude/skills/fix-on-my-computer/SKILL.md +81 -0
- package/src/config/claude-code-config/scripts/CLAUDE.md +50 -0
- package/src/config/claude-code-config/scripts/auto-rename-session/CLAUDE.md +59 -0
- package/src/config/claude-code-config/scripts/auto-rename-session/__tests__/shared.test.ts +185 -0
- package/src/config/claude-code-config/scripts/auto-rename-session/__tests__/title-generation.test.ts +44 -0
- package/src/config/claude-code-config/scripts/auto-rename-session/fixtures/sample-transcript.jsonl +3 -0
- package/src/config/claude-code-config/scripts/auto-rename-session/fixtures/transcript-short-message.jsonl +2 -0
- package/src/config/claude-code-config/scripts/auto-rename-session/fixtures/transcript-with-title.jsonl +3 -0
- package/src/config/claude-code-config/scripts/auto-rename-session/src/index.ts +113 -0
- package/src/config/claude-code-config/scripts/auto-rename-session/src/rename-all.ts +148 -0
- package/src/config/claude-code-config/scripts/auto-rename-session/src/shared.ts +157 -0
- package/src/config/claude-code-config/scripts/auto-rename-session/src/worker.ts +58 -0
- package/src/config/claude-code-config/scripts/biome.json +37 -0
- package/src/config/claude-code-config/scripts/claude-code-ai/CLAUDE.md +84 -0
- package/src/config/claude-code-config/scripts/claude-code-ai/__tests__/claude.test.ts +19 -0
- package/src/config/claude-code-config/scripts/claude-code-ai/__tests__/paths.test.ts +68 -0
- package/src/config/claude-code-config/scripts/claude-code-ai/claude.ts +56 -0
- package/src/config/claude-code-config/scripts/claude-code-ai/cli.ts +46 -0
- package/src/config/claude-code-config/scripts/claude-code-ai/helper/credentials.ts +76 -0
- package/src/config/claude-code-config/scripts/claude-code-ai/helper/index.ts +9 -0
- package/src/config/claude-code-config/scripts/claude-code-ai/helper/paths.ts +51 -0
- package/src/config/claude-code-config/scripts/command-validator/CLAUDE.md +112 -0
- package/src/config/claude-code-config/scripts/command-validator/README.md +147 -0
- package/src/config/claude-code-config/scripts/command-validator/src/__tests__/validator.test.ts +99 -0
- package/src/config/claude-code-config/scripts/command-validator/src/cli.ts +121 -0
- package/src/config/claude-code-config/scripts/command-validator/src/lib/security-rules.ts +174 -0
- package/src/config/claude-code-config/scripts/command-validator/src/lib/types.ts +34 -0
- package/src/config/claude-code-config/scripts/command-validator/src/lib/validator.ts +90 -0
- package/src/config/claude-code-config/scripts/package.json +43 -0
- package/src/config/claude-code-config/scripts/statusline/CLAUDE.md +205 -0
- package/src/config/claude-code-config/scripts/statusline/README.md +193 -0
- package/src/config/claude-code-config/scripts/statusline/__tests__/context.test.ts +229 -0
- package/src/config/claude-code-config/scripts/statusline/__tests__/formatters.test.ts +108 -0
- package/src/config/claude-code-config/scripts/statusline/__tests__/statusline.test.ts +309 -0
- package/src/config/claude-code-config/scripts/statusline/data/.gitkeep +0 -0
- package/src/config/claude-code-config/scripts/statusline/defaults.json +82 -0
- package/src/config/claude-code-config/scripts/statusline/docs/ARCHITECTURE.md +166 -0
- package/src/config/claude-code-config/scripts/statusline/fixtures/mock-transcript.jsonl +4 -0
- package/src/config/claude-code-config/scripts/statusline/fixtures/test-input.json +35 -0
- package/src/config/claude-code-config/scripts/statusline/src/analyze-daily-usage.ts +151 -0
- package/src/config/claude-code-config/scripts/statusline/src/commands/interactive-config.ts +515 -0
- package/src/config/claude-code-config/scripts/statusline/src/debug-payloads.ts +195 -0
- package/src/config/claude-code-config/scripts/statusline/src/index.ts +224 -0
- package/src/config/claude-code-config/scripts/statusline/src/lib/config-types.ts +110 -0
- package/src/config/claude-code-config/scripts/statusline/src/lib/config.ts +21 -0
- package/src/config/claude-code-config/scripts/statusline/src/lib/context.ts +103 -0
- package/src/config/claude-code-config/scripts/statusline/src/lib/features/limits/commands/weekly-analysis.ts +108 -0
- package/src/config/claude-code-config/scripts/statusline/src/lib/features/limits/index.ts +111 -0
- package/src/config/claude-code-config/scripts/statusline/src/lib/features/limits/types.ts +15 -0
- package/src/config/claude-code-config/scripts/statusline/src/lib/features/spend/commands/migrate-to-sqlite.ts +136 -0
- package/src/config/claude-code-config/scripts/statusline/src/lib/features/spend/commands/spend-day.ts +79 -0
- package/src/config/claude-code-config/scripts/statusline/src/lib/features/spend/commands/spend-month.ts +66 -0
- package/src/config/claude-code-config/scripts/statusline/src/lib/features/spend/commands/spend-project.ts +85 -0
- package/src/config/claude-code-config/scripts/statusline/src/lib/features/spend/database.ts +395 -0
- package/src/config/claude-code-config/scripts/statusline/src/lib/features/spend/index.ts +178 -0
- package/src/config/claude-code-config/scripts/statusline/src/lib/features/spend/payload-logger.ts +161 -0
- package/src/config/claude-code-config/scripts/statusline/src/lib/features/spend/types.ts +37 -0
- package/src/config/claude-code-config/scripts/statusline/src/lib/formatters.ts +426 -0
- package/src/config/claude-code-config/scripts/statusline/src/lib/git.ts +100 -0
- package/src/config/claude-code-config/scripts/statusline/src/lib/menu-factories.ts +224 -0
- package/src/config/claude-code-config/scripts/statusline/src/lib/presets.ts +177 -0
- package/src/config/claude-code-config/scripts/statusline/src/lib/render-pure.ts +516 -0
- package/src/config/claude-code-config/scripts/statusline/src/lib/types.ts +36 -0
- package/src/config/claude-code-config/scripts/statusline/src/lib/utils.ts +15 -0
- package/src/config/claude-code-config/scripts/statusline/src/stats.ts +119 -0
- package/src/config/claude-code-config/scripts/statusline/src/tests/spend-v2.test.ts +377 -0
- package/src/config/claude-code-config/scripts/statusline/src/tools/fixed-cost-calc.ts +317 -0
- package/src/config/claude-code-config/scripts/statusline/statusline.config.free.json +79 -0
- package/src/config/claude-code-config/scripts/statusline/statusline.config.json +100 -0
- package/src/config/claude-code-config/scripts/statusline/test-with-fixtures.ts +37 -0
- package/src/config/claude-code-config/scripts/statusline/test.ts +20 -0
- package/src/config/claude-code-config/scripts/statusline/tsconfig.json +27 -0
- package/src/config/claude-code-config/scripts/tsconfig.json +27 -0
- package/src/config/claude-code-config/settings.json +92 -0
- package/src/config/claude-code-config/skills/aibuilder-create-saas/SKILL.md +176 -0
- package/src/config/claude-code-config/skills/aibuilder-create-saas/references/architecture-template.md +436 -0
- package/src/config/claude-code-config/skills/aibuilder-create-saas/references/challenge-framework.md +289 -0
- package/src/config/claude-code-config/skills/aibuilder-create-saas/references/discovery-framework.md +338 -0
- package/src/config/claude-code-config/skills/aibuilder-create-saas/references/prd-template.md +452 -0
- package/src/config/claude-code-config/skills/aibuilder-create-saas/references/task-template.md +263 -0
- package/src/config/claude-code-config/skills/aibuilder-create-saas/references/tools.md +132 -0
- package/src/config/claude-code-config/skills/aibuilder-create-saas/scripts/rename-project.sh +42 -0
- package/src/config/claude-code-config/skills/aibuilder-create-saas/scripts/setup.sh +104 -0
- package/src/config/claude-code-config/skills/aibuilder-create-saas/steps/step-00-init.md +174 -0
- package/src/config/claude-code-config/skills/aibuilder-create-saas/steps/step-01-discovery.md +342 -0
- package/src/config/claude-code-config/skills/aibuilder-create-saas/steps/step-02-brainstorm.md +230 -0
- package/src/config/claude-code-config/skills/aibuilder-create-saas/steps/step-03-validate.md +372 -0
- package/src/config/claude-code-config/skills/aibuilder-create-saas/steps/step-04-prd.md +364 -0
- package/src/config/claude-code-config/skills/aibuilder-create-saas/steps/step-05-architecture.md +904 -0
- package/src/config/claude-code-config/skills/aibuilder-create-saas/steps/step-06-tasks.md +446 -0
- package/src/config/claude-code-config/skills/explore/SKILL.md +119 -0
- package/src/config/claude-code-config/skills/frontend-design/LICENSE.txt +177 -0
- package/src/config/claude-code-config/skills/frontend-design/SKILL.md +42 -0
- package/src/config/claude-code-config/skills/git-commit/SKILL.md +42 -0
- package/src/config/claude-code-config/skills/git-create-pr/SKILL.md +53 -0
- package/src/config/claude-code-config/skills/git-fix-pr-comments/SKILL.md +51 -0
- package/src/config/claude-code-config/skills/git-merge/SKILL.md +68 -0
- package/src/config/claude-code-config/skills/marketing-copywriting/SKILL.md +72 -0
- package/src/config/claude-code-config/skills/meta-claude-memory/SKILL.md +694 -0
- package/src/config/claude-code-config/skills/meta-claude-memory/references/comprehensive-example.md +175 -0
- package/src/config/claude-code-config/skills/meta-claude-memory/references/project-patterns.md +334 -0
- package/src/config/claude-code-config/skills/meta-claude-memory/references/prompting-techniques.md +411 -0
- package/src/config/claude-code-config/skills/meta-claude-memory/references/rules-directory-guide.md +298 -0
- package/src/config/claude-code-config/skills/meta-claude-memory/references/section-templates.md +347 -0
- package/src/config/claude-code-config/skills/meta-hooks-creator/SKILL.md +357 -0
- package/src/config/claude-code-config/skills/meta-hooks-creator/references/command-vs-prompt.md +287 -0
- package/src/config/claude-code-config/skills/meta-hooks-creator/references/examples.md +689 -0
- package/src/config/claude-code-config/skills/meta-hooks-creator/references/hook-types.md +495 -0
- package/src/config/claude-code-config/skills/meta-hooks-creator/references/input-output-schemas.md +503 -0
- package/src/config/claude-code-config/skills/meta-hooks-creator/references/matchers.md +517 -0
- package/src/config/claude-code-config/skills/meta-hooks-creator/references/troubleshooting.md +653 -0
- package/src/config/claude-code-config/skills/meta-prompt-creator/SKILL.md +285 -0
- package/src/config/claude-code-config/skills/meta-prompt-creator/references/anthropic-best-practices.md +126 -0
- package/src/config/claude-code-config/skills/meta-prompt-creator/references/anti-patterns.md +57 -0
- package/src/config/claude-code-config/skills/meta-prompt-creator/references/clarity-principles.md +54 -0
- package/src/config/claude-code-config/skills/meta-prompt-creator/references/context-management.md +389 -0
- package/src/config/claude-code-config/skills/meta-prompt-creator/references/few-shot-patterns.md +47 -0
- package/src/config/claude-code-config/skills/meta-prompt-creator/references/openai-best-practices.md +50 -0
- package/src/config/claude-code-config/skills/meta-prompt-creator/references/prompt-templates.md +110 -0
- package/src/config/claude-code-config/skills/meta-prompt-creator/references/reasoning-techniques.md +52 -0
- package/src/config/claude-code-config/skills/meta-prompt-creator/references/system-prompt-patterns.md +48 -0
- package/src/config/claude-code-config/skills/meta-prompt-creator/references/xml-structure.md +36 -0
- package/src/config/claude-code-config/skills/meta-skill-creator/LICENSE.txt +202 -0
- package/src/config/claude-code-config/skills/meta-skill-creator/SKILL.md +421 -0
- package/src/config/claude-code-config/skills/meta-skill-creator/package.json +5 -0
- package/src/config/claude-code-config/skills/meta-skill-creator/references/output-patterns.md +82 -0
- package/src/config/claude-code-config/skills/meta-skill-creator/references/progressive-disclosure-patterns.md +374 -0
- package/src/config/claude-code-config/skills/meta-skill-creator/references/prompting-integration.md +363 -0
- package/src/config/claude-code-config/skills/meta-skill-creator/references/real-world-examples.md +513 -0
- package/src/config/claude-code-config/skills/meta-skill-creator/references/script-patterns.md +385 -0
- package/src/config/claude-code-config/skills/meta-skill-creator/references/workflows.md +28 -0
- package/src/config/claude-code-config/skills/meta-skill-creator/references/xml-tag-guide.md +606 -0
- package/src/config/claude-code-config/skills/meta-skill-creator/scripts/init-skill.ts +214 -0
- package/src/config/claude-code-config/skills/meta-skill-creator/scripts/package-skill.ts +146 -0
- package/src/config/claude-code-config/skills/meta-skill-creator/scripts/validate.ts +138 -0
- package/src/config/claude-code-config/skills/meta-skill-workflow-creator/SKILL.md +390 -0
- package/src/config/claude-code-config/skills/meta-skill-workflow-creator/references/ask-patterns.md +225 -0
- package/src/config/claude-code-config/skills/meta-skill-workflow-creator/references/prompt-engineering.md +377 -0
- package/src/config/claude-code-config/skills/meta-skill-workflow-creator/references/state-management.md +275 -0
- package/src/config/claude-code-config/skills/meta-skill-workflow-creator/references/step-template.md +334 -0
- package/src/config/claude-code-config/skills/meta-skill-workflow-creator/references/workflow-patterns.md +300 -0
- package/src/config/claude-code-config/skills/meta-subagent-creator/SKILL.md +451 -0
- package/src/config/claude-code-config/skills/meta-subagent-creator/references/context-management.md +567 -0
- package/src/config/claude-code-config/skills/meta-subagent-creator/references/debugging-agents.md +714 -0
- package/src/config/claude-code-config/skills/meta-subagent-creator/references/error-handling-and-recovery.md +502 -0
- package/src/config/claude-code-config/skills/meta-subagent-creator/references/evaluation-and-testing.md +374 -0
- package/src/config/claude-code-config/skills/meta-subagent-creator/references/orchestration-patterns.md +591 -0
- package/src/config/claude-code-config/skills/meta-subagent-creator/references/subagents.md +599 -0
- package/src/config/claude-code-config/skills/meta-subagent-creator/references/writing-subagent-prompts.md +513 -0
- package/src/config/claude-code-config/skills/ralph-loop/SKILL.md +117 -0
- package/src/config/claude-code-config/skills/ralph-loop/scripts/setup.sh +278 -0
- package/src/config/claude-code-config/skills/ralph-loop/steps/step-00-init.md +215 -0
- package/src/config/claude-code-config/skills/ralph-loop/steps/step-01-interactive-prd.md +366 -0
- package/src/config/claude-code-config/skills/ralph-loop/steps/step-02-create-stories.md +273 -0
- package/src/config/claude-code-config/skills/ralph-loop/steps/step-03-finish.md +245 -0
- package/src/config/claude-code-config/skills/ralph-tasks/SKILL.md +88 -0
- package/src/config/claude-code-config/skills/ralph-tasks/scripts/add-task.sh +36 -0
- package/src/config/claude-code-config/skills/ralph-tasks/scripts/list-tasks.sh +66 -0
- package/src/config/claude-code-config/skills/ralph-tasks/scripts/remove-task.sh +47 -0
- package/src/config/claude-code-config/skills/ralph-tasks/scripts/setup.sh +201 -0
- package/src/config/claude-code-config/skills/ralph-tasks/steps/action-add-search.md +131 -0
- package/src/config/claude-code-config/skills/ralph-tasks/steps/action-add.md +46 -0
- package/src/config/claude-code-config/skills/ralph-tasks/steps/action-init.md +123 -0
- package/src/config/claude-code-config/skills/ralph-tasks/steps/action-list.md +58 -0
- package/src/config/claude-code-config/skills/ralph-tasks/steps/action-remove.md +48 -0
- package/src/config/claude-code-config/skills/ralph-tasks/steps/action-setup.md +46 -0
- package/src/config/claude-code-config/skills/ralph-tasks/steps/action-wake-up.md +62 -0
- package/src/config/claude-code-config/skills/utils-fix-errors/SKILL.md +61 -0
- package/src/config/claude-code-config/skills/utils-fix-grammar/SKILL.md +59 -0
- package/src/config/claude-code-config/skills/utils-oneshot/SKILL.md +56 -0
- package/src/config/claude-code-config/skills/utils-refactor/SKILL.md +89 -0
- package/src/config/claude-code-config/skills/utils-save-docs/SKILL.md +74 -0
- package/src/config/claude-code-config/skills/utils-ultrathink/SKILL.md +42 -0
- package/src/config/claude-code-config/skills/workflow-apex/SKILL.md +116 -0
- package/src/config/claude-code-config/skills/workflow-apex/scripts/setup-templates.sh +144 -0
- package/src/config/claude-code-config/skills/workflow-apex/scripts/update-progress.sh +80 -0
- package/src/config/claude-code-config/skills/workflow-apex/steps/step-00-init.md +273 -0
- package/src/config/claude-code-config/skills/workflow-apex/steps/step-00b-branch.md +126 -0
- package/src/config/claude-code-config/skills/workflow-apex/steps/step-00b-economy.md +244 -0
- package/src/config/claude-code-config/skills/workflow-apex/steps/step-00b-interactive.md +165 -0
- package/src/config/claude-code-config/skills/workflow-apex/steps/step-00b-save.md +123 -0
- package/src/config/claude-code-config/skills/workflow-apex/steps/step-01-analyze.md +361 -0
- package/src/config/claude-code-config/skills/workflow-apex/steps/step-02-plan.md +422 -0
- package/src/config/claude-code-config/skills/workflow-apex/steps/step-02b-tasks.md +301 -0
- package/src/config/claude-code-config/skills/workflow-apex/steps/step-03-execute-teams.md +297 -0
- package/src/config/claude-code-config/skills/workflow-apex/steps/step-03-execute.md +239 -0
- package/src/config/claude-code-config/skills/workflow-apex/steps/step-04-validate.md +264 -0
- package/src/config/claude-code-config/skills/workflow-apex/steps/step-05-examine.md +294 -0
- package/src/config/claude-code-config/skills/workflow-apex/steps/step-06-resolve.md +237 -0
- package/src/config/claude-code-config/skills/workflow-apex/steps/step-07-tests.md +250 -0
- package/src/config/claude-code-config/skills/workflow-apex/steps/step-08-run-tests.md +308 -0
- package/src/config/claude-code-config/skills/workflow-apex/steps/step-09-finish.md +193 -0
- package/src/config/claude-code-config/skills/workflow-apex/templates/00-context.md +53 -0
- package/src/config/claude-code-config/skills/workflow-apex/templates/01-analyze.md +10 -0
- package/src/config/claude-code-config/skills/workflow-apex/templates/02-plan.md +10 -0
- package/src/config/claude-code-config/skills/workflow-apex/templates/03-execute.md +10 -0
- package/src/config/claude-code-config/skills/workflow-apex/templates/04-validate.md +10 -0
- package/src/config/claude-code-config/skills/workflow-apex/templates/05-examine.md +10 -0
- package/src/config/claude-code-config/skills/workflow-apex/templates/06-resolve.md +10 -0
- package/src/config/claude-code-config/skills/workflow-apex/templates/07-tests.md +10 -0
- package/src/config/claude-code-config/skills/workflow-apex/templates/08-run-tests.md +10 -0
- package/src/config/claude-code-config/skills/workflow-apex/templates/09-finish.md +10 -0
- package/src/config/claude-code-config/skills/workflow-apex/templates/README.md +195 -0
- package/src/config/claude-code-config/skills/workflow-apex/templates/step-complete.md +7 -0
- package/src/config/claude-code-config/skills/workflow-brainstorm/SKILL.md +127 -0
- package/src/config/claude-code-config/skills/workflow-brainstorm/steps/step-01-explore.md +230 -0
- package/src/config/claude-code-config/skills/workflow-brainstorm/steps/step-02-challenge.md +238 -0
- package/src/config/claude-code-config/skills/workflow-brainstorm/steps/step-03-synthesize.md +337 -0
- package/src/config/claude-code-config/skills/workflow-brainstorm/steps/step-04-action.md +285 -0
- package/src/config/claude-code-config/skills/workflow-ci-fixer/SKILL.md +150 -0
- package/src/config/claude-code-config/skills/workflow-ci-fixer/references/cli-commands.md +122 -0
- package/src/config/claude-code-config/skills/workflow-ci-fixer/references/github-cli.md +243 -0
- package/src/config/claude-code-config/skills/workflow-ci-fixer/references/troubleshooting.md +362 -0
- package/src/config/claude-code-config/skills/workflow-ci-fixer/references/vercel-cli.md +192 -0
- package/src/config/claude-code-config/skills/workflow-ci-fixer/steps/step-00-init.md +157 -0
- package/src/config/claude-code-config/skills/workflow-ci-fixer/steps/step-01-watch-ci.md +192 -0
- package/src/config/claude-code-config/skills/workflow-ci-fixer/steps/step-02-analyze-errors.md +263 -0
- package/src/config/claude-code-config/skills/workflow-ci-fixer/steps/step-03-fix-locally.md +312 -0
- package/src/config/claude-code-config/skills/workflow-ci-fixer/steps/step-04-commit-push.md +206 -0
- package/src/config/claude-code-config/skills/workflow-ci-fixer/steps/step-05-cleanup.md +196 -0
- package/src/config/claude-code-config/skills/workflow-clean-code/SKILL.md +129 -0
- package/src/config/claude-code-config/skills/workflow-clean-code/references/general-clean-code.md +347 -0
- package/src/config/claude-code-config/skills/workflow-clean-code/references/nextjs-clean-code.md +458 -0
- package/src/config/claude-code-config/skills/workflow-clean-code/references/react-clean-code.md +406 -0
- package/src/config/claude-code-config/skills/workflow-clean-code/references/tanstack-query-best-practices.md +516 -0
- package/src/config/claude-code-config/skills/workflow-clean-code/references/zustand-best-practices.md +491 -0
- package/src/config/claude-code-config/skills/workflow-clean-code/steps/step-01-scan.md +137 -0
- package/src/config/claude-code-config/skills/workflow-clean-code/steps/step-02-apply.md +152 -0
- package/src/config/claude-code-config/skills/workflow-clean-code/steps/step-03-verify.md +143 -0
- package/src/config/claude-code-config/skills/workflow-debug/SKILL.md +126 -0
- package/src/config/claude-code-config/skills/workflow-debug/references/log-technique.md +285 -0
- package/src/config/claude-code-config/skills/workflow-debug/steps/step-00-init.md +94 -0
- package/src/config/claude-code-config/skills/workflow-debug/steps/step-01-analyze.md +180 -0
- package/src/config/claude-code-config/skills/workflow-debug/steps/step-01b-log-instrumentation.md +271 -0
- package/src/config/claude-code-config/skills/workflow-debug/steps/step-02-find-solutions.md +141 -0
- package/src/config/claude-code-config/skills/workflow-debug/steps/step-03-propose.md +142 -0
- package/src/config/claude-code-config/skills/workflow-debug/steps/step-04-fix.md +176 -0
- package/src/config/claude-code-config/skills/workflow-debug/steps/step-05-verify.md +279 -0
- package/src/config/claude-code-config/skills/workflow-review-code/SKILL.md +219 -0
- package/src/config/claude-code-config/skills/workflow-review-code/references/clean-code-principles.md +140 -0
- package/src/config/claude-code-config/skills/workflow-review-code/references/code-quality-metrics.md +174 -0
- package/src/config/claude-code-config/skills/workflow-review-code/references/feedback-patterns.md +149 -0
- package/src/config/claude-code-config/skills/workflow-review-code/references/security-checklist.md +127 -0
- package/src/config/claude-code-config/song/finish.mp3 +0 -0
- 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();
|