@ngxtm/devkit 2.1.0 → 3.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/cli/detect.js +292 -0
- package/cli/index.js +204 -92
- package/cli/init.js +245 -0
- package/cli/update.js +243 -0
- package/cli/utils.js +195 -0
- package/package.json +16 -20
- package/rules-index.json +76 -0
- package/scripts/generate-index.js +223 -0
- package/scripts/merge-commands.js +290 -0
- package/scripts/organize-rules.js +226 -0
- package/templates/base/hooks/lib/ck-config-utils.cjs +769 -0
- package/templates/base/hooks/lib/ck-paths.cjs +110 -0
- package/templates/base/hooks/lib/colors.cjs +95 -0
- package/templates/base/hooks/lib/config-counter.cjs +103 -0
- package/templates/base/hooks/lib/context-builder.cjs +600 -0
- package/templates/base/hooks/lib/context-tracker.cjs +335 -0
- package/templates/base/hooks/lib/privacy-checker.cjs +297 -0
- package/templates/base/hooks/lib/project-detector.cjs +430 -0
- package/templates/base/hooks/lib/scout-checker.cjs +172 -0
- package/templates/base/hooks/lib/transcript-parser.cjs +164 -0
- package/templates/base/hooks/privacy-block.cjs +145 -0
- package/agents/backend-engineer.md +0 -154
- package/agents/brainstormer.md +0 -169
- package/agents/business-analyst.md +0 -166
- package/agents/database-architect.md +0 -159
- package/agents/debugger.md +0 -155
- package/agents/designer.md +0 -150
- package/agents/devops-engineer.md +0 -155
- package/agents/docs-manager.md +0 -171
- package/agents/frontend-engineer.md +0 -159
- package/agents/game-engineer.md +0 -148
- package/agents/mobile-engineer.md +0 -149
- package/agents/performance-engineer.md +0 -152
- package/agents/planner.md +0 -161
- package/agents/project-manager.md +0 -160
- package/agents/researcher.md +0 -146
- package/agents/reviewer.md +0 -155
- package/agents/scouter.md +0 -157
- package/agents/security-engineer.md +0 -154
- package/agents/tech-lead.md +0 -159
- package/agents/tester.md +0 -157
- package/agents-claudekit/brainstormer.md +0 -113
- package/agents-claudekit/code-reviewer.md +0 -157
- package/agents-claudekit/code-simplifier.md +0 -42
- package/agents-claudekit/copywriter.md +0 -110
- package/agents-claudekit/database-admin.md +0 -92
- package/agents-claudekit/debugger.md +0 -137
- package/agents-claudekit/docs-manager.md +0 -208
- package/agents-claudekit/fullstack-developer.md +0 -95
- package/agents-claudekit/git-manager.md +0 -394
- package/agents-claudekit/journal-writer.md +0 -113
- package/agents-claudekit/mcp-manager.md +0 -93
- package/agents-claudekit/planner.md +0 -108
- package/agents-claudekit/project-manager.md +0 -125
- package/agents-claudekit/researcher.md +0 -38
- package/agents-claudekit/scout-external.md +0 -141
- package/agents-claudekit/scout.md +0 -107
- package/agents-claudekit/tester.md +0 -105
- package/agents-claudekit/ui-ux-designer.md +0 -236
- package/commands/ask.md +0 -64
- package/commands/brainstorm.md +0 -64
- package/commands/code.md +0 -64
- package/commands/cook.md +0 -64
- package/commands/debug.md +0 -64
- package/commands/design/fast.md +0 -134
- package/commands/fix/fast.md +0 -84
- package/commands/fix/hard.md +0 -116
- package/commands/fix.md +0 -64
- package/commands/plan/fast.md +0 -78
- package/commands/plan/hard.md +0 -131
- package/commands/plan.md +0 -64
- package/commands/test.md +0 -64
- package/matrix-skills/_index.yaml +0 -275
- package/matrix-skills/ai-ml.yaml +0 -353
- package/matrix-skills/architecture.yaml +0 -93
- package/matrix-skills/backend.yaml +0 -280
- package/matrix-skills/cloud.yaml +0 -112
- package/matrix-skills/data.yaml +0 -74
- package/matrix-skills/design.yaml +0 -98
- package/matrix-skills/devops.yaml +0 -200
- package/matrix-skills/frontend.yaml +0 -200
- package/matrix-skills/gaming.yaml +0 -39
- package/matrix-skills/languages.yaml +0 -160
- package/matrix-skills/management.yaml +0 -50
- package/matrix-skills/mcp.yaml +0 -82
- package/matrix-skills/mobile.yaml +0 -85
- package/matrix-skills/performance.yaml +0 -23
- package/matrix-skills/planning.yaml +0 -117
- package/matrix-skills/quality.yaml +0 -195
- package/matrix-skills/research.yaml +0 -106
- package/matrix-skills/security.yaml +0 -293
- package/matrix-skills/tools.yaml +0 -352
- package/output-styles/coding-level-0-eli5.md +0 -103
- package/output-styles/coding-level-1-junior.md +0 -124
- package/output-styles/coding-level-2-mid.md +0 -146
- package/output-styles/coding-level-3-senior.md +0 -148
- package/output-styles/coding-level-4-lead.md +0 -159
- package/output-styles/coding-level-5-god.md +0 -91
- package/rules/README.md +0 -141
- package/rules/metadata.json +0 -54
- package/settings.json +0 -3
- package/statusline.cjs +0 -500
- package/statusline.ps1 +0 -307
- package/statusline.sh +0 -237
- package/workflows/development-rules.md +0 -42
- package/workflows/documentation-management.md +0 -121
- package/workflows/orchestration-protocol.md +0 -16
- package/workflows/primary-workflow.md +0 -45
- /package/{commands → merged-commands}/ask/fast.md +0 -0
- /package/{commands → merged-commands}/ask/hard.md +0 -0
- /package/{commands-claudekit → merged-commands}/ask.md +0 -0
- /package/{commands → merged-commands}/auto.md +0 -0
- /package/{commands-claudekit → merged-commands}/bootstrap/auto/fast.md +0 -0
- /package/{commands-claudekit → merged-commands}/bootstrap/auto/parallel.md +0 -0
- /package/{commands-claudekit → merged-commands}/bootstrap/auto.md +0 -0
- /package/{commands-claudekit → merged-commands}/bootstrap.md +0 -0
- /package/{commands → merged-commands}/brainstorm/fast.md +0 -0
- /package/{commands → merged-commands}/brainstorm/hard.md +0 -0
- /package/{commands-claudekit → merged-commands}/brainstorm.md +0 -0
- /package/{commands-claudekit → merged-commands}/ck-help.md +0 -0
- /package/{commands-claudekit → merged-commands}/code/auto.md +0 -0
- /package/{commands → merged-commands}/code/fast.md +0 -0
- /package/{commands → merged-commands}/code/hard.md +0 -0
- /package/{commands-claudekit → merged-commands}/code/no-test.md +0 -0
- /package/{commands-claudekit → merged-commands}/code/parallel.md +0 -0
- /package/{commands-claudekit → merged-commands}/code.md +0 -0
- /package/{commands-claudekit → merged-commands}/coding-level.md +0 -0
- /package/{commands-claudekit → merged-commands}/content/cro.md +0 -0
- /package/{commands-claudekit → merged-commands}/content/enhance.md +0 -0
- /package/{commands-claudekit → merged-commands}/content/fast.md +0 -0
- /package/{commands-claudekit → merged-commands}/content/good.md +0 -0
- /package/{commands-claudekit → merged-commands}/cook/auto/fast.md +0 -0
- /package/{commands-claudekit → merged-commands}/cook/auto/parallel.md +0 -0
- /package/{commands-claudekit → merged-commands}/cook/auto.md +0 -0
- /package/{commands → merged-commands}/cook/fast.md +0 -0
- /package/{commands → merged-commands}/cook/hard.md +0 -0
- /package/{commands-claudekit → merged-commands}/cook.md +0 -0
- /package/{commands → merged-commands}/debug/fast.md +0 -0
- /package/{commands → merged-commands}/debug/hard.md +0 -0
- /package/{commands-claudekit → merged-commands}/debug.md +0 -0
- /package/{commands → merged-commands}/deploy/check.md +0 -0
- /package/{commands → merged-commands}/deploy/preview.md +0 -0
- /package/{commands → merged-commands}/deploy/production.md +0 -0
- /package/{commands → merged-commands}/deploy/rollback.md +0 -0
- /package/{commands → merged-commands}/deploy.md +0 -0
- /package/{commands-claudekit → merged-commands}/design/3d.md +0 -0
- /package/{commands-claudekit → merged-commands}/design/describe.md +0 -0
- /package/{commands-claudekit → merged-commands}/design/fast.md +0 -0
- /package/{commands-claudekit → merged-commands}/design/good.md +0 -0
- /package/{commands → merged-commands}/design/hard.md +0 -0
- /package/{commands-claudekit → merged-commands}/design/screenshot.md +0 -0
- /package/{commands-claudekit → merged-commands}/design/video.md +0 -0
- /package/{commands → merged-commands}/design.md +0 -0
- /package/{commands → merged-commands}/docs/audit.md +0 -0
- /package/{commands → merged-commands}/docs/business.md +0 -0
- /package/{commands → merged-commands}/docs/core.md +0 -0
- /package/{commands-claudekit → merged-commands}/docs/init.md +0 -0
- /package/{commands-claudekit → merged-commands}/docs/summarize.md +0 -0
- /package/{commands-claudekit → merged-commands}/docs/update.md +0 -0
- /package/{commands → merged-commands}/docs.md +0 -0
- /package/{commands-claudekit → merged-commands}/fix/ci.md +0 -0
- /package/{commands-claudekit → merged-commands}/fix/fast.md +0 -0
- /package/{commands-claudekit → merged-commands}/fix/hard.md +0 -0
- /package/{commands-claudekit → merged-commands}/fix/logs.md +0 -0
- /package/{commands-claudekit → merged-commands}/fix/parallel.md +0 -0
- /package/{commands-claudekit → merged-commands}/fix/test.md +0 -0
- /package/{commands-claudekit → merged-commands}/fix/types.md +0 -0
- /package/{commands-claudekit → merged-commands}/fix/ui.md +0 -0
- /package/{commands-claudekit → merged-commands}/fix.md +0 -0
- /package/{commands-claudekit → merged-commands}/git/cm.md +0 -0
- /package/{commands-claudekit → merged-commands}/git/cp.md +0 -0
- /package/{commands-claudekit → merged-commands}/git/merge.md +0 -0
- /package/{commands-claudekit → merged-commands}/git/pr.md +0 -0
- /package/{commands-claudekit → merged-commands}/integrate/polar.md +0 -0
- /package/{commands-claudekit → merged-commands}/integrate/sepay.md +0 -0
- /package/{commands-claudekit → merged-commands}/journal.md +0 -0
- /package/{commands-claudekit → merged-commands}/kanban.md +0 -0
- /package/{commands-claudekit → merged-commands}/plan/archive.md +0 -0
- /package/{commands-claudekit → merged-commands}/plan/ci.md +0 -0
- /package/{commands-claudekit → merged-commands}/plan/cro.md +0 -0
- /package/{commands-claudekit → merged-commands}/plan/fast.md +0 -0
- /package/{commands-claudekit → merged-commands}/plan/hard.md +0 -0
- /package/{commands-claudekit → merged-commands}/plan/parallel.md +0 -0
- /package/{commands-claudekit → merged-commands}/plan/two.md +0 -0
- /package/{commands-claudekit → merged-commands}/plan/validate.md +0 -0
- /package/{commands-claudekit → merged-commands}/plan.md +0 -0
- /package/{commands-claudekit → merged-commands}/preview.md +0 -0
- /package/{commands-claudekit → merged-commands}/review/codebase/parallel.md +0 -0
- /package/{commands-claudekit → merged-commands}/review/codebase.md +0 -0
- /package/{commands → merged-commands}/review/fast.md +0 -0
- /package/{commands → merged-commands}/review/hard.md +0 -0
- /package/{commands → merged-commands}/review.md +0 -0
- /package/{commands-claudekit → merged-commands}/scout/ext.md +0 -0
- /package/{commands-claudekit → merged-commands}/scout.md +0 -0
- /package/{commands-claudekit → merged-commands}/skill/add.md +0 -0
- /package/{commands-claudekit → merged-commands}/skill/create.md +0 -0
- /package/{commands-claudekit → merged-commands}/skill/fix-logs.md +0 -0
- /package/{commands-claudekit → merged-commands}/skill/optimize/auto.md +0 -0
- /package/{commands-claudekit → merged-commands}/skill/optimize.md +0 -0
- /package/{commands-claudekit → merged-commands}/skill/plan.md +0 -0
- /package/{commands-claudekit → merged-commands}/skill/update.md +0 -0
- /package/{commands → merged-commands}/test/fast.md +0 -0
- /package/{commands → merged-commands}/test/hard.md +0 -0
- /package/{commands-claudekit → merged-commands}/test/ui.md +0 -0
- /package/{commands-claudekit → merged-commands}/test.md +0 -0
- /package/{commands-claudekit → merged-commands}/use-mcp.md +0 -0
- /package/{commands-claudekit → merged-commands}/watzup.md +0 -0
- /package/{commands-claudekit → merged-commands}/worktree.md +0 -0
- /package/{rules → templates/dart/rules}/dart/best-practices/SKILL.md +0 -0
- /package/{rules → templates/dart/rules}/dart/language/SKILL.md +0 -0
- /package/{rules → templates/dart/rules}/dart/tooling/SKILL.md +0 -0
- /package/{rules → templates/flutter/rules}/flutter/auto-route-navigation/SKILL.md +0 -0
- /package/{rules → templates/flutter/rules}/flutter/auto-route-navigation/references/REFERENCE.md +0 -0
- /package/{rules → templates/flutter/rules}/flutter/auto-route-navigation/references/router-config.md +0 -0
- /package/{rules → templates/flutter/rules}/flutter/bloc-state-management/SKILL.md +0 -0
- /package/{rules → templates/flutter/rules}/flutter/bloc-state-management/references/REFERENCE.md +0 -0
- /package/{rules → templates/flutter/rules}/flutter/bloc-state-management/references/auth-bloc-example.md +0 -0
- /package/{rules → templates/flutter/rules}/flutter/bloc-state-management/references/equatable-usage.md +0 -0
- /package/{rules → templates/flutter/rules}/flutter/bloc-state-management/references/property-based-state.md +0 -0
- /package/{rules → templates/flutter/rules}/flutter/bloc.rule.md +0 -0
- /package/{rules → templates/flutter/rules}/flutter/cicd/SKILL.md +0 -0
- /package/{rules → templates/flutter/rules}/flutter/cicd/references/advanced-workflow.md +0 -0
- /package/{rules → templates/flutter/rules}/flutter/cicd/references/fastlane.md +0 -0
- /package/{rules → templates/flutter/rules}/flutter/cicd/references/github-actions.md +0 -0
- /package/{rules → templates/flutter/rules}/flutter/dependency-injection/SKILL.md +0 -0
- /package/{rules → templates/flutter/rules}/flutter/dependency-injection/references/REFERENCE.md +0 -0
- /package/{rules → templates/flutter/rules}/flutter/dependency-injection/references/modules.md +0 -0
- /package/{rules → templates/flutter/rules}/flutter/error-handling/SKILL.md +0 -0
- /package/{rules → templates/flutter/rules}/flutter/error-handling/references/REFERENCE.md +0 -0
- /package/{rules → templates/flutter/rules}/flutter/error-handling/references/error-mapping.md +0 -0
- /package/{rules → templates/flutter/rules}/flutter/feature-based-clean-architecture/SKILL.md +0 -0
- /package/{rules → templates/flutter/rules}/flutter/feature-based-clean-architecture/references/REFERENCE.md +0 -0
- /package/{rules → templates/flutter/rules}/flutter/feature-based-clean-architecture/references/folder-structure.md +0 -0
- /package/{rules → templates/flutter/rules}/flutter/getx-navigation/SKILL.md +0 -0
- /package/{rules → templates/flutter/rules}/flutter/getx-navigation/references/app-pages.md +0 -0
- /package/{rules → templates/flutter/rules}/flutter/getx-navigation/references/middleware-example.md +0 -0
- /package/{rules → templates/flutter/rules}/flutter/getx-state-management/SKILL.md +0 -0
- /package/{rules → templates/flutter/rules}/flutter/getx-state-management/references/binding-example.md +0 -0
- /package/{rules → templates/flutter/rules}/flutter/getx-state-management/references/reactive-vs-simple.md +0 -0
- /package/{rules → templates/flutter/rules}/flutter/go-router-navigation/SKILL.md +0 -0
- /package/{rules → templates/flutter/rules}/flutter/idiomatic-flutter/SKILL.md +0 -0
- /package/{rules → templates/flutter/rules}/flutter/layer-based-clean-architecture/SKILL.md +0 -0
- /package/{rules → templates/flutter/rules}/flutter/layer-based-clean-architecture/references/REFERENCE.md +0 -0
- /package/{rules → templates/flutter/rules}/flutter/layer-based-clean-architecture/references/repository-mapping.md +0 -0
- /package/{rules → templates/flutter/rules}/flutter/localization/SKILL.md +0 -0
- /package/{rules → templates/flutter/rules}/flutter/localization/references/REFERENCE.md +0 -0
- /package/{rules → templates/flutter/rules}/flutter/localization/references/sheet-loader.md +0 -0
- /package/{rules → templates/flutter/rules}/flutter/navigator-v1-navigation/SKILL.md +0 -0
- /package/{rules → templates/flutter/rules}/flutter/navigator-v1-navigation/references/on-generate-route.md +0 -0
- /package/{rules → templates/flutter/rules}/flutter/performance/SKILL.md +0 -0
- /package/{rules → templates/flutter/rules}/flutter/retrofit-networking/SKILL.md +0 -0
- /package/{rules → templates/flutter/rules}/flutter/retrofit-networking/references/REFERENCE.md +0 -0
- /package/{rules → templates/flutter/rules}/flutter/retrofit-networking/references/token-refresh.md +0 -0
- /package/{rules → templates/flutter/rules}/flutter/riverpod-state-management/SKILL.md +0 -0
- /package/{rules → templates/flutter/rules}/flutter/riverpod-state-management/references/architecture.md +0 -0
- /package/{rules → templates/flutter/rules}/flutter/riverpod-state-management/references/best-practices.md +0 -0
- /package/{rules → templates/flutter/rules}/flutter/riverpod-state-management/references/testing.md +0 -0
- /package/{rules → templates/flutter/rules}/flutter/riverpod.rule.md +0 -0
- /package/{rules → templates/flutter/rules}/flutter/security/SKILL.md +0 -0
- /package/{rules → templates/flutter/rules}/flutter/security/references/REFERENCE.md +0 -0
- /package/{rules → templates/flutter/rules}/flutter/security/references/network-security.md +0 -0
- /package/{rules → templates/flutter/rules}/flutter/testing/SKILL.md +0 -0
- /package/{rules → templates/flutter/rules}/flutter/testing/references/REFERENCE.md +0 -0
- /package/{rules → templates/flutter/rules}/flutter/testing/references/bloc-testing.md +0 -0
- /package/{rules → templates/flutter/rules}/flutter/testing/references/integration-testing.md +0 -0
- /package/{rules → templates/flutter/rules}/flutter/testing/references/robot-pattern.md +0 -0
- /package/{rules → templates/flutter/rules}/flutter/testing/references/unit-testing.md +0 -0
- /package/{rules → templates/flutter/rules}/flutter/testing/references/widget-testing.md +0 -0
- /package/{rules → templates/flutter/rules}/flutter/widgets/SKILL.md +0 -0
- /package/{rules → templates/golang/rules}/golang/chi-router/SKILL.md +0 -0
- /package/{rules → templates/golang/rules}/golang/chi-router/references/REFERENCE.md +0 -0
- /package/{rules → templates/golang/rules}/golang/chi-router/references/routing-patterns.md +0 -0
- /package/{rules → templates/golang/rules}/golang/cobra-cli/SKILL.md +0 -0
- /package/{rules → templates/golang/rules}/golang/cobra-cli/references/REFERENCE.md +0 -0
- /package/{rules → templates/golang/rules}/golang/cobra-cli/references/command-patterns.md +0 -0
- /package/{rules → templates/golang/rules}/golang/core/SKILL.md +0 -0
- /package/{rules → templates/golang/rules}/golang/core/references/REFERENCE.md +0 -0
- /package/{rules → templates/golang/rules}/golang/core/references/concurrency-patterns.md +0 -0
- /package/{rules → templates/golang/rules}/golang/core/references/error-handling.md +0 -0
- /package/{rules → templates/golang/rules}/golang/echo-framework/SKILL.md +0 -0
- /package/{rules → templates/golang/rules}/golang/echo-framework/references/REFERENCE.md +0 -0
- /package/{rules → templates/golang/rules}/golang/echo-framework/references/middleware-patterns.md +0 -0
- /package/{rules → templates/golang/rules}/golang/echo-framework/references/routing-patterns.md +0 -0
- /package/{rules → templates/golang/rules}/golang/ent-orm/SKILL.md +0 -0
- /package/{rules → templates/golang/rules}/golang/ent-orm/references/REFERENCE.md +0 -0
- /package/{rules → templates/golang/rules}/golang/ent-orm/references/schema-patterns.md +0 -0
- /package/{rules → templates/golang/rules}/golang/fiber-framework/SKILL.md +0 -0
- /package/{rules → templates/golang/rules}/golang/fiber-framework/references/REFERENCE.md +0 -0
- /package/{rules → templates/golang/rules}/golang/fiber-framework/references/routing-patterns.md +0 -0
- /package/{rules → templates/golang/rules}/golang/gin-framework/SKILL.md +0 -0
- /package/{rules → templates/golang/rules}/golang/gin-framework/references/REFERENCE.md +0 -0
- /package/{rules → templates/golang/rules}/golang/gin-framework/references/middleware-patterns.md +0 -0
- /package/{rules → templates/golang/rules}/golang/gorm-orm/SKILL.md +0 -0
- /package/{rules → templates/golang/rules}/golang/gorm-orm/references/REFERENCE.md +0 -0
- /package/{rules → templates/golang/rules}/golang/gorm-orm/references/model-definitions.md +0 -0
- /package/{rules → templates/golang/rules}/golang/gorm-orm/references/query-patterns.md +0 -0
- /package/{rules → templates/golang/rules}/golang/grpc/SKILL.md +0 -0
- /package/{rules → templates/golang/rules}/golang/grpc/references/REFERENCE.md +0 -0
- /package/{rules → templates/golang/rules}/golang/grpc/references/service-patterns.md +0 -0
- /package/{rules → templates/golang/rules}/golang/testify/SKILL.md +0 -0
- /package/{rules → templates/golang/rules}/golang/testify/references/REFERENCE.md +0 -0
- /package/{rules → templates/golang/rules}/golang/testify/references/assert-patterns.md +0 -0
- /package/{rules → templates/golang/rules}/golang/validator/SKILL.md +0 -0
- /package/{rules → templates/golang/rules}/golang/validator/references/REFERENCE.md +0 -0
- /package/{rules → templates/golang/rules}/golang/validator/references/validation-tags.md +0 -0
- /package/{rules → templates/golang/rules}/golang/viper-config/SKILL.md +0 -0
- /package/{rules → templates/golang/rules}/golang/viper-config/references/REFERENCE.md +0 -0
- /package/{rules → templates/golang/rules}/golang/viper-config/references/config-loading.md +0 -0
- /package/{rules → templates/golang/rules}/golang/wire-di/SKILL.md +0 -0
- /package/{rules → templates/golang/rules}/golang/wire-di/references/REFERENCE.md +0 -0
- /package/{rules → templates/golang/rules}/golang/wire-di/references/provider-patterns.md +0 -0
- /package/{rules → templates/golang/rules}/golang/zap-logging/SKILL.md +0 -0
- /package/{rules → templates/golang/rules}/golang/zap-logging/references/REFERENCE.md +0 -0
- /package/{rules → templates/golang/rules}/golang/zap-logging/references/logger-config.md +0 -0
- /package/{rules → templates/java/rules}/java/build-gradle/SKILL.md +0 -0
- /package/{rules → templates/java/rules}/java/build-gradle/references/REFERENCE.md +0 -0
- /package/{rules → templates/java/rules}/java/build-gradle/references/kotlin-dsl.md +0 -0
- /package/{rules → templates/java/rules}/java/build-gradle/references/task-configuration.md +0 -0
- /package/{rules → templates/java/rules}/java/build-maven/SKILL.md +0 -0
- /package/{rules → templates/java/rules}/java/build-maven/references/REFERENCE.md +0 -0
- /package/{rules → templates/java/rules}/java/build-maven/references/dependency-management.md +0 -0
- /package/{rules → templates/java/rules}/java/build-maven/references/lifecycle-phases.md +0 -0
- /package/{rules → templates/java/rules}/java/graalvm-native/SKILL.md +0 -0
- /package/{rules → templates/java/rules}/java/graalvm-native/references/REFERENCE.md +0 -0
- /package/{rules → templates/java/rules}/java/java-collections-streams/SKILL.md +0 -0
- /package/{rules → templates/java/rules}/java/java-collections-streams/references/REFERENCE.md +0 -0
- /package/{rules → templates/java/rules}/java/java-collections-streams/references/collectors-patterns.md +0 -0
- /package/{rules → templates/java/rules}/java/java-collections-streams/references/stream-pipelines.md +0 -0
- /package/{rules → templates/java/rules}/java/java-concurrency/SKILL.md +0 -0
- /package/{rules → templates/java/rules}/java/java-concurrency/references/REFERENCE.md +0 -0
- /package/{rules → templates/java/rules}/java/java-concurrency/references/completable-future.md +0 -0
- /package/{rules → templates/java/rules}/java/java-concurrency/references/executor-patterns.md +0 -0
- /package/{rules → templates/java/rules}/java/java-concurrency/references/virtual-threads.md +0 -0
- /package/{rules → templates/java/rules}/java/java-core-language/SKILL.md +0 -0
- /package/{rules → templates/java/rules}/java/java-core-language/references/REFERENCE.md +0 -0
- /package/{rules → templates/java/rules}/java/java-core-language/references/jvm-memory-model.md +0 -0
- /package/{rules → templates/java/rules}/java/java-core-language/references/modern-java-features.md +0 -0
- /package/{rules → templates/java/rules}/java/java-project-structure/SKILL.md +0 -0
- /package/{rules → templates/java/rules}/java/java-project-structure/references/REFERENCE.md +0 -0
- /package/{rules → templates/java/rules}/java/java-project-structure/references/maven-project-layout.md +0 -0
- /package/{rules → templates/java/rules}/java/java-project-structure/references/module-system.md +0 -0
- /package/{rules → templates/java/rules}/java/micronaut-core/SKILL.md +0 -0
- /package/{rules → templates/java/rules}/java/micronaut-core/references/REFERENCE.md +0 -0
- /package/{rules → templates/java/rules}/java/micronaut-reactive/SKILL.md +0 -0
- /package/{rules → templates/java/rules}/java/micronaut-reactive/references/REFERENCE.md +0 -0
- /package/{rules → templates/java/rules}/java/quarkus-core/SKILL.md +0 -0
- /package/{rules → templates/java/rules}/java/quarkus-core/references/REFERENCE.md +0 -0
- /package/{rules → templates/java/rules}/java/quarkus-reactive/SKILL.md +0 -0
- /package/{rules → templates/java/rules}/java/quarkus-reactive/references/REFERENCE.md +0 -0
- /package/{rules → templates/java/rules}/java/spring-batch/SKILL.md +0 -0
- /package/{rules → templates/java/rules}/java/spring-batch/references/REFERENCE.md +0 -0
- /package/{rules → templates/java/rules}/java/spring-boot-architecture/SKILL.md +0 -0
- /package/{rules → templates/java/rules}/java/spring-boot-architecture/references/REFERENCE.md +0 -0
- /package/{rules → templates/java/rules}/java/spring-boot-architecture/references/auto-configuration.md +0 -0
- /package/{rules → templates/java/rules}/java/spring-boot-architecture/references/configuration-properties.md +0 -0
- /package/{rules → templates/java/rules}/java/spring-boot-web/SKILL.md +0 -0
- /package/{rules → templates/java/rules}/java/spring-boot-web/references/REFERENCE.md +0 -0
- /package/{rules → templates/java/rules}/java/spring-cloud/SKILL.md +0 -0
- /package/{rules → templates/java/rules}/java/spring-cloud/references/REFERENCE.md +0 -0
- /package/{rules → templates/java/rules}/java/spring-data-jpa/SKILL.md +0 -0
- /package/{rules → templates/java/rules}/java/spring-data-jpa/references/REFERENCE.md +0 -0
- /package/{rules → templates/java/rules}/java/spring-security/SKILL.md +0 -0
- /package/{rules → templates/java/rules}/java/spring-security/references/REFERENCE.md +0 -0
- /package/{rules → templates/java/rules}/java/spring-security/references/jwt-auth-flow.md +0 -0
- /package/{rules → templates/java/rules}/java/testing-junit-mockito/SKILL.md +0 -0
- /package/{rules → templates/java/rules}/java/testing-junit-mockito/references/REFERENCE.md +0 -0
- /package/{rules → templates/java/rules}/java/testing-junit-mockito/references/junit5-patterns.md +0 -0
- /package/{rules → templates/java/rules}/java/testing-junit-mockito/references/mockito-patterns.md +0 -0
- /package/{rules → templates/java/rules}/java/testing-junit-mockito/references/spring-boot-testing.md +0 -0
- /package/{rules → templates/javascript/rules}/javascript/best-practices/SKILL.md +0 -0
- /package/{rules → templates/javascript/rules}/javascript/best-practices/references/REFERENCE.md +0 -0
- /package/{rules → templates/javascript/rules}/javascript/language/SKILL.md +0 -0
- /package/{rules → templates/javascript/rules}/javascript/language/references/REFERENCE.md +0 -0
- /package/{rules → templates/javascript/rules}/javascript/tooling/SKILL.md +0 -0
- /package/{rules → templates/javascript/rules}/javascript/tooling/references/REFERENCE.md +0 -0
- /package/{rules → templates/nestjs/rules}/nestjs/api-standards/SKILL.md +0 -0
- /package/{rules → templates/nestjs/rules}/nestjs/api-standards/references/pagination-wrapper.md +0 -0
- /package/{rules → templates/nestjs/rules}/nestjs/architecture/SKILL.md +0 -0
- /package/{rules → templates/nestjs/rules}/nestjs/architecture/references/dynamic-module.md +0 -0
- /package/{rules → templates/nestjs/rules}/nestjs/caching/SKILL.md +0 -0
- /package/{rules → templates/nestjs/rules}/nestjs/caching/references/REFERENCE.md +0 -0
- /package/{rules → templates/nestjs/rules}/nestjs/caching/references/cache-patterns.md +0 -0
- /package/{rules → templates/nestjs/rules}/nestjs/configuration/SKILL.md +0 -0
- /package/{rules → templates/nestjs/rules}/nestjs/configuration/references/REFERENCE.md +0 -0
- /package/{rules → templates/nestjs/rules}/nestjs/configuration/references/config-patterns.md +0 -0
- /package/{rules → templates/nestjs/rules}/nestjs/controllers-services/SKILL.md +0 -0
- /package/{rules → templates/nestjs/rules}/nestjs/controllers-services/references/REFERENCE.md +0 -0
- /package/{rules → templates/nestjs/rules}/nestjs/controllers-services/references/controller-patterns.md +0 -0
- /package/{rules → templates/nestjs/rules}/nestjs/controllers-services/references/service-patterns.md +0 -0
- /package/{rules → templates/nestjs/rules}/nestjs/database/SKILL.md +0 -0
- /package/{rules → templates/nestjs/rules}/nestjs/database/references/REFERENCE.md +0 -0
- /package/{rules → templates/nestjs/rules}/nestjs/database/references/typeorm-patterns.md +0 -0
- /package/{rules → templates/nestjs/rules}/nestjs/deployment/SKILL.md +0 -0
- /package/{rules → templates/nestjs/rules}/nestjs/deployment/references/REFERENCE.md +0 -0
- /package/{rules → templates/nestjs/rules}/nestjs/deployment/references/deployment-patterns.md +0 -0
- /package/{rules → templates/nestjs/rules}/nestjs/documentation/SKILL.md +0 -0
- /package/{rules → templates/nestjs/rules}/nestjs/documentation/references/REFERENCE.md +0 -0
- /package/{rules → templates/nestjs/rules}/nestjs/documentation/references/swagger-patterns.md +0 -0
- /package/{rules → templates/nestjs/rules}/nestjs/error-handling/SKILL.md +0 -0
- /package/{rules → templates/nestjs/rules}/nestjs/error-handling/references/REFERENCE.md +0 -0
- /package/{rules → templates/nestjs/rules}/nestjs/error-handling/references/exception-filters.md +0 -0
- /package/{rules → templates/nestjs/rules}/nestjs/file-uploads/SKILL.md +0 -0
- /package/{rules → templates/nestjs/rules}/nestjs/file-uploads/references/REFERENCE.md +0 -0
- /package/{rules → templates/nestjs/rules}/nestjs/file-uploads/references/upload-patterns.md +0 -0
- /package/{rules → templates/nestjs/rules}/nestjs/observability/SKILL.md +0 -0
- /package/{rules → templates/nestjs/rules}/nestjs/observability/references/REFERENCE.md +0 -0
- /package/{rules → templates/nestjs/rules}/nestjs/observability/references/logging-metrics.md +0 -0
- /package/{rules → templates/nestjs/rules}/nestjs/performance/SKILL.md +0 -0
- /package/{rules → templates/nestjs/rules}/nestjs/performance/references/REFERENCE.md +0 -0
- /package/{rules → templates/nestjs/rules}/nestjs/performance/references/performance-patterns.md +0 -0
- /package/{rules → templates/nestjs/rules}/nestjs/real-time/SKILL.md +0 -0
- /package/{rules → templates/nestjs/rules}/nestjs/real-time/references/REFERENCE.md +0 -0
- /package/{rules → templates/nestjs/rules}/nestjs/real-time/references/websocket-patterns.md +0 -0
- /package/{rules → templates/nestjs/rules}/nestjs/scheduling/SKILL.md +0 -0
- /package/{rules → templates/nestjs/rules}/nestjs/scheduling/references/REFERENCE.md +0 -0
- /package/{rules → templates/nestjs/rules}/nestjs/scheduling/references/scheduling-patterns.md +0 -0
- /package/{rules → templates/nestjs/rules}/nestjs/search/SKILL.md +0 -0
- /package/{rules → templates/nestjs/rules}/nestjs/search/references/REFERENCE.md +0 -0
- /package/{rules → templates/nestjs/rules}/nestjs/search/references/search-patterns.md +0 -0
- /package/{rules → templates/nestjs/rules}/nestjs/security/SKILL.md +0 -0
- /package/{rules → templates/nestjs/rules}/nestjs/security/references/REFERENCE.md +0 -0
- /package/{rules → templates/nestjs/rules}/nestjs/security/references/authentication.md +0 -0
- /package/{rules → templates/nestjs/rules}/nestjs/testing/SKILL.md +0 -0
- /package/{rules → templates/nestjs/rules}/nestjs/testing/references/REFERENCE.md +0 -0
- /package/{rules → templates/nestjs/rules}/nestjs/testing/references/unit-testing.md +0 -0
- /package/{rules → templates/nestjs/rules}/nestjs/transport/SKILL.md +0 -0
- /package/{rules → templates/nestjs/rules}/nestjs/transport/references/REFERENCE.md +0 -0
- /package/{rules → templates/nestjs/rules}/nestjs/transport/references/microservices-patterns.md +0 -0
- /package/{rules → templates/nextjs/rules}/nextjs/app-router/SKILL.md +0 -0
- /package/{rules → templates/nextjs/rules}/nextjs/app-router/references/REFERENCE.md +0 -0
- /package/{rules → templates/nextjs/rules}/nextjs/app-router/references/routing-patterns.md +0 -0
- /package/{rules → templates/nextjs/rules}/nextjs/architecture/SKILL.md +0 -0
- /package/{rules → templates/nextjs/rules}/nextjs/architecture/references/fsd-structure.md +0 -0
- /package/{rules → templates/nextjs/rules}/nextjs/authentication/SKILL.md +0 -0
- /package/{rules → templates/nextjs/rules}/nextjs/authentication/references/auth-implementation.md +0 -0
- /package/{rules → templates/nextjs/rules}/nextjs/caching/SKILL.md +0 -0
- /package/{rules → templates/nextjs/rules}/nextjs/caching/references/REFERENCE.md +0 -0
- /package/{rules → templates/nextjs/rules}/nextjs/caching/references/cache-strategies.md +0 -0
- /package/{rules → templates/nextjs/rules}/nextjs/data-access-layer/SKILL.md +0 -0
- /package/{rules → templates/nextjs/rules}/nextjs/data-access-layer/references/patterns.md +0 -0
- /package/{rules → templates/nextjs/rules}/nextjs/data-fetching/SKILL.md +0 -0
- /package/{rules → templates/nextjs/rules}/nextjs/data-fetching/references/REFERENCE.md +0 -0
- /package/{rules → templates/nextjs/rules}/nextjs/data-fetching/references/fetch-patterns.md +0 -0
- /package/{rules → templates/nextjs/rules}/nextjs/internationalization/SKILL.md +0 -0
- /package/{rules → templates/nextjs/rules}/nextjs/internationalization/references/REFERENCE.md +0 -0
- /package/{rules → templates/nextjs/rules}/nextjs/internationalization/references/i18n-patterns.md +0 -0
- /package/{rules → templates/nextjs/rules}/nextjs/optimization/SKILL.md +0 -0
- /package/{rules → templates/nextjs/rules}/nextjs/optimization/references/REFERENCE.md +0 -0
- /package/{rules → templates/nextjs/rules}/nextjs/optimization/references/optimization-patterns.md +0 -0
- /package/{rules → templates/nextjs/rules}/nextjs/rendering/SKILL.md +0 -0
- /package/{rules → templates/nextjs/rules}/nextjs/rendering/references/REFERENCE.md +0 -0
- /package/{rules → templates/nextjs/rules}/nextjs/rendering/references/rendering-modes.md +0 -0
- /package/{rules → templates/nextjs/rules}/nextjs/server-actions/SKILL.md +0 -0
- /package/{rules → templates/nextjs/rules}/nextjs/server-actions/references/REFERENCE.md +0 -0
- /package/{rules → templates/nextjs/rules}/nextjs/server-actions/references/action-patterns.md +0 -0
- /package/{rules → templates/nextjs/rules}/nextjs/server-components/SKILL.md +0 -0
- /package/{rules → templates/nextjs/rules}/nextjs/server-components/references/REFERENCE.md +0 -0
- /package/{rules → templates/nextjs/rules}/nextjs/server-components/references/component-patterns.md +0 -0
- /package/{rules → templates/nextjs/rules}/nextjs/state-management/SKILL.md +0 -0
- /package/{rules → templates/nextjs/rules}/nextjs/state-management/references/REFERENCE.md +0 -0
- /package/{rules → templates/nextjs/rules}/nextjs/state-management/references/state-patterns.md +0 -0
- /package/{rules → templates/nextjs/rules}/nextjs/styling/SKILL.md +0 -0
- /package/{rules → templates/nextjs/rules}/nextjs/styling/references/implementation.md +0 -0
- /package/{rules → templates/react/rules}/react/component-patterns/SKILL.md +0 -0
- /package/{rules → templates/react/rules}/react/component-patterns/references/REFERENCE.md +0 -0
- /package/{rules → templates/react/rules}/react/hooks/SKILL.md +0 -0
- /package/{rules → templates/react/rules}/react/hooks/references/REFERENCE.md +0 -0
- /package/{rules → templates/react/rules}/react/hooks.rule.md +0 -0
- /package/{rules → templates/react/rules}/react/performance/SKILL.md +0 -0
- /package/{rules → templates/react/rules}/react/performance/references/REFERENCE.md +0 -0
- /package/{rules → templates/react/rules}/react/security/SKILL.md +0 -0
- /package/{rules → templates/react/rules}/react/security/references/REFERENCE.md +0 -0
- /package/{rules → templates/react/rules}/react/state-management/SKILL.md +0 -0
- /package/{rules → templates/react/rules}/react/state-management/references/REFERENCE.md +0 -0
- /package/{rules → templates/react/rules}/react/testing/SKILL.md +0 -0
- /package/{rules → templates/react/rules}/react/testing/references/REFERENCE.md +0 -0
- /package/{rules → templates/react/rules}/react/tooling/SKILL.md +0 -0
- /package/{rules → templates/react/rules}/react/typescript/SKILL.md +0 -0
- /package/{rules → templates/rust/rules}/rust/actix-web/SKILL.md +0 -0
- /package/{rules → templates/rust/rules}/rust/actix-web/references/REFERENCE.md +0 -0
- /package/{rules → templates/rust/rules}/rust/actix-web/references/handler-patterns.md +0 -0
- /package/{rules → templates/rust/rules}/rust/async-graphql/SKILL.md +0 -0
- /package/{rules → templates/rust/rules}/rust/async-graphql/references/REFERENCE.md +0 -0
- /package/{rules → templates/rust/rules}/rust/async-graphql/references/schema-patterns.md +0 -0
- /package/{rules → templates/rust/rules}/rust/axum/SKILL.md +0 -0
- /package/{rules → templates/rust/rules}/rust/axum/references/REFERENCE.md +0 -0
- /package/{rules → templates/rust/rules}/rust/axum/references/handler-patterns.md +0 -0
- /package/{rules → templates/rust/rules}/rust/bevy/SKILL.md +0 -0
- /package/{rules → templates/rust/rules}/rust/bevy/references/REFERENCE.md +0 -0
- /package/{rules → templates/rust/rules}/rust/bevy/references/ecs-patterns.md +0 -0
- /package/{rules → templates/rust/rules}/rust/clap/SKILL.md +0 -0
- /package/{rules → templates/rust/rules}/rust/clap/references/REFERENCE.md +0 -0
- /package/{rules → templates/rust/rules}/rust/clap/references/derive-patterns.md +0 -0
- /package/{rules → templates/rust/rules}/rust/core/SKILL.md +0 -0
- /package/{rules → templates/rust/rules}/rust/core/references/REFERENCE.md +0 -0
- /package/{rules → templates/rust/rules}/rust/core/references/error-handling.md +0 -0
- /package/{rules → templates/rust/rules}/rust/diesel-orm/SKILL.md +0 -0
- /package/{rules → templates/rust/rules}/rust/diesel-orm/references/REFERENCE.md +0 -0
- /package/{rules → templates/rust/rules}/rust/diesel-orm/references/schema-patterns.md +0 -0
- /package/{rules → templates/rust/rules}/rust/rocket/SKILL.md +0 -0
- /package/{rules → templates/rust/rules}/rust/rocket/references/REFERENCE.md +0 -0
- /package/{rules → templates/rust/rules}/rust/rocket/references/handler-patterns.md +0 -0
- /package/{rules → templates/rust/rules}/rust/sea-orm/SKILL.md +0 -0
- /package/{rules → templates/rust/rules}/rust/sea-orm/references/REFERENCE.md +0 -0
- /package/{rules → templates/rust/rules}/rust/sea-orm/references/entity-patterns.md +0 -0
- /package/{rules → templates/rust/rules}/rust/serde-serialization/SKILL.md +0 -0
- /package/{rules → templates/rust/rules}/rust/serde-serialization/references/REFERENCE.md +0 -0
- /package/{rules → templates/rust/rules}/rust/serde-serialization/references/serialization-patterns.md +0 -0
- /package/{rules → templates/rust/rules}/rust/sqlx-database/SKILL.md +0 -0
- /package/{rules → templates/rust/rules}/rust/sqlx-database/references/REFERENCE.md +0 -0
- /package/{rules → templates/rust/rules}/rust/sqlx-database/references/query-patterns.md +0 -0
- /package/{rules → templates/rust/rules}/rust/tauri/SKILL.md +0 -0
- /package/{rules → templates/rust/rules}/rust/tauri/references/REFERENCE.md +0 -0
- /package/{rules → templates/rust/rules}/rust/tauri/references/command-patterns.md +0 -0
- /package/{rules → templates/rust/rules}/rust/tokio-runtime/SKILL.md +0 -0
- /package/{rules → templates/rust/rules}/rust/tokio-runtime/references/REFERENCE.md +0 -0
- /package/{rules → templates/rust/rules}/rust/tokio-runtime/references/async-patterns.md +0 -0
- /package/{rules → templates/rust/rules}/rust/tokio-runtime/references/synchronization.md +0 -0
- /package/{rules → templates/rust/rules}/rust/tonic/SKILL.md +0 -0
- /package/{rules → templates/rust/rules}/rust/tonic/references/REFERENCE.md +0 -0
- /package/{rules → templates/rust/rules}/rust/tonic/references/service-patterns.md +0 -0
- /package/{rules → templates/rust/rules}/rust/tracing/SKILL.md +0 -0
- /package/{rules → templates/rust/rules}/rust/tracing/references/REFERENCE.md +0 -0
- /package/{rules → templates/rust/rules}/rust/tracing/references/instrumentation.md +0 -0
- /package/{rules → templates/typescript/rules}/typescript/best-practices/SKILL.md +0 -0
- /package/{rules → templates/typescript/rules}/typescript/best-practices/references/REFERENCE.md +0 -0
- /package/{rules → templates/typescript/rules}/typescript/language/SKILL.md +0 -0
- /package/{rules → templates/typescript/rules}/typescript/language/references/REFERENCE.md +0 -0
- /package/{rules → templates/typescript/rules}/typescript/patterns.rule.md +0 -0
- /package/{rules → templates/typescript/rules}/typescript/security/SKILL.md +0 -0
- /package/{rules → templates/typescript/rules}/typescript/security/references/REFERENCE.md +0 -0
- /package/{rules → templates/typescript/rules}/typescript/tooling/SKILL.md +0 -0
- /package/{rules → templates/typescript/rules}/typescript/tooling/references/REFERENCE.md +0 -0
package/cli/init.js
ADDED
|
@@ -0,0 +1,245 @@
|
|
|
1
|
+
#!/usr/bin/env node
|
|
2
|
+
|
|
3
|
+
/**
|
|
4
|
+
* Devkit Init - Per-Project Installation
|
|
5
|
+
*
|
|
6
|
+
* Installs devkit to the current project's .claude/ directory with:
|
|
7
|
+
* - Merged commands (from both agent-assistant and claudekit)
|
|
8
|
+
* - Tech-specific rules (based on project detection)
|
|
9
|
+
* - Skills index (for on-demand loading)
|
|
10
|
+
* - Essential hooks
|
|
11
|
+
*/
|
|
12
|
+
|
|
13
|
+
const fs = require('fs');
|
|
14
|
+
const path = require('path');
|
|
15
|
+
|
|
16
|
+
const { detectProjectType, getRulesForTypes, printDetectionResults } = require('./detect');
|
|
17
|
+
const { copyDir, getDirSize, validatePath } = require('./utils');
|
|
18
|
+
|
|
19
|
+
const VERSION = require('../package.json').version;
|
|
20
|
+
const PACKAGE_ROOT = path.join(__dirname, '..');
|
|
21
|
+
|
|
22
|
+
/**
|
|
23
|
+
* Initialize devkit in a project directory
|
|
24
|
+
*/
|
|
25
|
+
function initProject(options = {}) {
|
|
26
|
+
const projectDir = options.path || process.cwd();
|
|
27
|
+
const claudeDir = path.join(projectDir, '.claude');
|
|
28
|
+
const isUpdate = options.update || false;
|
|
29
|
+
|
|
30
|
+
console.log('\n' + '='.repeat(60));
|
|
31
|
+
console.log(' DEVKIT v' + VERSION + (isUpdate ? ' - UPDATE' : ' - INIT'));
|
|
32
|
+
console.log('='.repeat(60));
|
|
33
|
+
|
|
34
|
+
// Check if .claude already exists
|
|
35
|
+
if (fs.existsSync(claudeDir) && !isUpdate && !options.force) {
|
|
36
|
+
console.log(`\n .claude/ folder already exists.`);
|
|
37
|
+
console.log(' Use --force to overwrite or --update to update.\n');
|
|
38
|
+
return { success: false, reason: 'exists' };
|
|
39
|
+
}
|
|
40
|
+
|
|
41
|
+
// 1. Detect project type
|
|
42
|
+
console.log('\n Detecting project type...');
|
|
43
|
+
const detectedTypes = detectProjectType(projectDir);
|
|
44
|
+
const rulesToInstall = getRulesForTypes(detectedTypes);
|
|
45
|
+
|
|
46
|
+
if (detectedTypes.length > 0) {
|
|
47
|
+
console.log(` Detected: ${detectedTypes.join(', ')}`);
|
|
48
|
+
console.log(` Rules: ${rulesToInstall.join(', ')}`);
|
|
49
|
+
} else {
|
|
50
|
+
console.log(' No specific technology detected.');
|
|
51
|
+
console.log(' Installing base commands only.');
|
|
52
|
+
}
|
|
53
|
+
|
|
54
|
+
// Create .claude directory
|
|
55
|
+
fs.mkdirSync(claudeDir, { recursive: true });
|
|
56
|
+
|
|
57
|
+
let totalFiles = 0;
|
|
58
|
+
const stats = {};
|
|
59
|
+
|
|
60
|
+
// 2. Install merged commands
|
|
61
|
+
console.log('\n Installing components...');
|
|
62
|
+
const mergedCommandsDir = path.join(PACKAGE_ROOT, 'merged-commands');
|
|
63
|
+
const commandsDir = path.join(claudeDir, 'commands');
|
|
64
|
+
|
|
65
|
+
if (fs.existsSync(mergedCommandsDir)) {
|
|
66
|
+
const count = copyDir(mergedCommandsDir, commandsDir);
|
|
67
|
+
stats.commands = count;
|
|
68
|
+
totalFiles += count;
|
|
69
|
+
console.log(` Commands: ${count} files`);
|
|
70
|
+
} else {
|
|
71
|
+
// Fallback to commands-claudekit if merged not available
|
|
72
|
+
const fallbackDir = path.join(PACKAGE_ROOT, 'commands-claudekit');
|
|
73
|
+
if (fs.existsSync(fallbackDir)) {
|
|
74
|
+
const count = copyDir(fallbackDir, commandsDir);
|
|
75
|
+
stats.commands = count;
|
|
76
|
+
totalFiles += count;
|
|
77
|
+
console.log(` Commands (claudekit): ${count} files`);
|
|
78
|
+
}
|
|
79
|
+
}
|
|
80
|
+
|
|
81
|
+
// 3. Install tech-specific rules
|
|
82
|
+
const rulesDir = path.join(claudeDir, 'rules');
|
|
83
|
+
let rulesCount = 0;
|
|
84
|
+
|
|
85
|
+
for (const ruleType of rulesToInstall) {
|
|
86
|
+
const srcRulesDir = path.join(PACKAGE_ROOT, 'templates', ruleType, 'rules');
|
|
87
|
+
if (fs.existsSync(srcRulesDir)) {
|
|
88
|
+
const destRulesDir = path.join(rulesDir, ruleType);
|
|
89
|
+
const count = copyDir(srcRulesDir, destRulesDir);
|
|
90
|
+
rulesCount += count;
|
|
91
|
+
}
|
|
92
|
+
}
|
|
93
|
+
|
|
94
|
+
if (rulesCount > 0) {
|
|
95
|
+
stats.rules = rulesCount;
|
|
96
|
+
totalFiles += rulesCount;
|
|
97
|
+
console.log(` Rules: ${rulesCount} files (${rulesToInstall.join(', ')})`);
|
|
98
|
+
}
|
|
99
|
+
|
|
100
|
+
// 4. Install essential hooks
|
|
101
|
+
const srcHooksDir = path.join(PACKAGE_ROOT, 'templates', 'base', 'hooks');
|
|
102
|
+
const hooksDir = path.join(claudeDir, 'hooks');
|
|
103
|
+
|
|
104
|
+
if (fs.existsSync(srcHooksDir)) {
|
|
105
|
+
const count = copyDir(srcHooksDir, hooksDir);
|
|
106
|
+
stats.hooks = count;
|
|
107
|
+
totalFiles += count;
|
|
108
|
+
console.log(` Hooks: ${count} files`);
|
|
109
|
+
} else {
|
|
110
|
+
// Fallback to main hooks directory (essential only)
|
|
111
|
+
const fallbackHooks = path.join(PACKAGE_ROOT, 'hooks');
|
|
112
|
+
if (fs.existsSync(fallbackHooks)) {
|
|
113
|
+
const count = copyDir(fallbackHooks, hooksDir);
|
|
114
|
+
stats.hooks = count;
|
|
115
|
+
totalFiles += count;
|
|
116
|
+
console.log(` Hooks: ${count} files`);
|
|
117
|
+
}
|
|
118
|
+
}
|
|
119
|
+
|
|
120
|
+
// 5. Install skills index
|
|
121
|
+
const skillsIndexSrc = path.join(PACKAGE_ROOT, 'skills-index.json');
|
|
122
|
+
const skillsIndexDest = path.join(claudeDir, 'skills-index.json');
|
|
123
|
+
|
|
124
|
+
if (fs.existsSync(skillsIndexSrc)) {
|
|
125
|
+
fs.copyFileSync(skillsIndexSrc, skillsIndexDest);
|
|
126
|
+
totalFiles++;
|
|
127
|
+
console.log(` Skills Index: 1 file`);
|
|
128
|
+
}
|
|
129
|
+
|
|
130
|
+
// 6. Create devkit.json tracking file
|
|
131
|
+
const devkitConfig = {
|
|
132
|
+
version: VERSION,
|
|
133
|
+
detectedTypes: detectedTypes,
|
|
134
|
+
installedRules: rulesToInstall,
|
|
135
|
+
installedAt: new Date().toISOString(),
|
|
136
|
+
updatedAt: isUpdate ? new Date().toISOString() : null,
|
|
137
|
+
stats: {
|
|
138
|
+
totalFiles: totalFiles,
|
|
139
|
+
sizeKB: Math.round(getDirSize(claudeDir) / 1024)
|
|
140
|
+
}
|
|
141
|
+
};
|
|
142
|
+
|
|
143
|
+
fs.writeFileSync(
|
|
144
|
+
path.join(claudeDir, 'devkit.json'),
|
|
145
|
+
JSON.stringify(devkitConfig, null, 2)
|
|
146
|
+
);
|
|
147
|
+
totalFiles++;
|
|
148
|
+
|
|
149
|
+
// 7. Create settings.json if not exists
|
|
150
|
+
const settingsPath = path.join(claudeDir, 'settings.json');
|
|
151
|
+
if (!fs.existsSync(settingsPath)) {
|
|
152
|
+
const settings = {
|
|
153
|
+
includeCoAuthoredBy: false
|
|
154
|
+
};
|
|
155
|
+
fs.writeFileSync(settingsPath, JSON.stringify(settings, null, 2));
|
|
156
|
+
totalFiles++;
|
|
157
|
+
}
|
|
158
|
+
|
|
159
|
+
// Calculate total size
|
|
160
|
+
const totalSizeKB = Math.round(getDirSize(claudeDir) / 1024);
|
|
161
|
+
|
|
162
|
+
console.log('\n' + '='.repeat(60));
|
|
163
|
+
console.log(' INSTALLATION COMPLETE');
|
|
164
|
+
console.log('='.repeat(60));
|
|
165
|
+
console.log(`\n Total: ${totalFiles} files (${totalSizeKB} KB)`);
|
|
166
|
+
console.log(` Location: ${claudeDir}`);
|
|
167
|
+
|
|
168
|
+
console.log('\n Available commands:');
|
|
169
|
+
console.log(' /plan - Plan implementation');
|
|
170
|
+
console.log(' /cook - Build a feature');
|
|
171
|
+
console.log(' /fix - Fix issues');
|
|
172
|
+
console.log(' /code - Start coding');
|
|
173
|
+
console.log(' /brainstorm - Brainstorm ideas');
|
|
174
|
+
|
|
175
|
+
if (detectedTypes.length > 0) {
|
|
176
|
+
console.log(`\n Tech-specific rules loaded for: ${detectedTypes.join(', ')}`);
|
|
177
|
+
}
|
|
178
|
+
|
|
179
|
+
console.log('\n Restart Claude Code to use the new skills.\n');
|
|
180
|
+
|
|
181
|
+
return {
|
|
182
|
+
success: true,
|
|
183
|
+
detected: detectedTypes,
|
|
184
|
+
rules: rulesToInstall,
|
|
185
|
+
stats: {
|
|
186
|
+
files: totalFiles,
|
|
187
|
+
sizeKB: totalSizeKB
|
|
188
|
+
}
|
|
189
|
+
};
|
|
190
|
+
}
|
|
191
|
+
|
|
192
|
+
/**
|
|
193
|
+
* Uninstall devkit from project
|
|
194
|
+
*/
|
|
195
|
+
function uninstallProject(options = {}) {
|
|
196
|
+
const projectDir = options.path || process.cwd();
|
|
197
|
+
const claudeDir = path.join(projectDir, '.claude');
|
|
198
|
+
|
|
199
|
+
console.log('\n' + '='.repeat(60));
|
|
200
|
+
console.log(' DEVKIT - UNINSTALL');
|
|
201
|
+
console.log('='.repeat(60));
|
|
202
|
+
|
|
203
|
+
if (!fs.existsSync(claudeDir)) {
|
|
204
|
+
console.log('\n No .claude/ folder found.\n');
|
|
205
|
+
return { success: false, reason: 'not_found' };
|
|
206
|
+
}
|
|
207
|
+
|
|
208
|
+
// Check if it's a devkit installation
|
|
209
|
+
const devkitConfig = path.join(claudeDir, 'devkit.json');
|
|
210
|
+
if (!fs.existsSync(devkitConfig)) {
|
|
211
|
+
console.log('\n .claude/ exists but is not a devkit installation.');
|
|
212
|
+
console.log(' Remove manually if needed.\n');
|
|
213
|
+
return { success: false, reason: 'not_devkit' };
|
|
214
|
+
}
|
|
215
|
+
|
|
216
|
+
// Remove the directory
|
|
217
|
+
fs.rmSync(claudeDir, { recursive: true, force: true });
|
|
218
|
+
console.log(`\n Removed: ${claudeDir}`);
|
|
219
|
+
console.log(' Devkit uninstalled successfully.\n');
|
|
220
|
+
|
|
221
|
+
return { success: true };
|
|
222
|
+
}
|
|
223
|
+
|
|
224
|
+
module.exports = {
|
|
225
|
+
initProject,
|
|
226
|
+
uninstallProject,
|
|
227
|
+
copyDir,
|
|
228
|
+
getDirSize
|
|
229
|
+
};
|
|
230
|
+
|
|
231
|
+
// Run if called directly
|
|
232
|
+
if (require.main === module) {
|
|
233
|
+
const args = process.argv.slice(2);
|
|
234
|
+
const options = {
|
|
235
|
+
force: args.includes('--force') || args.includes('-f'),
|
|
236
|
+
update: args.includes('--update') || args.includes('-u'),
|
|
237
|
+
path: args.find(a => !a.startsWith('-')) || process.cwd()
|
|
238
|
+
};
|
|
239
|
+
|
|
240
|
+
if (args.includes('--uninstall')) {
|
|
241
|
+
uninstallProject(options);
|
|
242
|
+
} else {
|
|
243
|
+
initProject(options);
|
|
244
|
+
}
|
|
245
|
+
}
|
package/cli/update.js
ADDED
|
@@ -0,0 +1,243 @@
|
|
|
1
|
+
#!/usr/bin/env node
|
|
2
|
+
|
|
3
|
+
/**
|
|
4
|
+
* Devkit Update - Update existing installation
|
|
5
|
+
*
|
|
6
|
+
* Re-detects project type and updates rules accordingly.
|
|
7
|
+
* Preserves user customizations where possible.
|
|
8
|
+
*/
|
|
9
|
+
|
|
10
|
+
const fs = require('fs');
|
|
11
|
+
const path = require('path');
|
|
12
|
+
const { execSync } = require('child_process');
|
|
13
|
+
|
|
14
|
+
const { detectProjectType, getRulesForTypes } = require('./detect');
|
|
15
|
+
const { initProject } = require('./init');
|
|
16
|
+
const { copyDir, getDirSize, parseJsonFile } = require('./utils');
|
|
17
|
+
|
|
18
|
+
const VERSION = require('../package.json').version;
|
|
19
|
+
const PACKAGE_ROOT = path.join(__dirname, '..');
|
|
20
|
+
|
|
21
|
+
/**
|
|
22
|
+
* Check for npm package updates
|
|
23
|
+
*/
|
|
24
|
+
function checkForUpdates() {
|
|
25
|
+
try {
|
|
26
|
+
const result = execSync('npm view @ngxtm/devkit version', {
|
|
27
|
+
encoding: 'utf-8',
|
|
28
|
+
stdio: ['pipe', 'pipe', 'pipe']
|
|
29
|
+
});
|
|
30
|
+
return result.trim();
|
|
31
|
+
} catch (e) {
|
|
32
|
+
return null;
|
|
33
|
+
}
|
|
34
|
+
}
|
|
35
|
+
|
|
36
|
+
/**
|
|
37
|
+
* Update devkit installation
|
|
38
|
+
*/
|
|
39
|
+
function updateProject(options = {}) {
|
|
40
|
+
const projectDir = options.path || process.cwd();
|
|
41
|
+
const claudeDir = path.join(projectDir, '.claude');
|
|
42
|
+
const configPath = path.join(claudeDir, 'devkit.json');
|
|
43
|
+
|
|
44
|
+
console.log('\n' + '='.repeat(60));
|
|
45
|
+
console.log(' DEVKIT - UPDATE');
|
|
46
|
+
console.log('='.repeat(60));
|
|
47
|
+
|
|
48
|
+
// Check if devkit is installed
|
|
49
|
+
if (!fs.existsSync(configPath)) {
|
|
50
|
+
console.log('\n No devkit installation found.');
|
|
51
|
+
console.log(' Run: devkit init\n');
|
|
52
|
+
return { success: false, reason: 'not_installed' };
|
|
53
|
+
}
|
|
54
|
+
|
|
55
|
+
// Read current config with error handling
|
|
56
|
+
const config = parseJsonFile(configPath);
|
|
57
|
+
if (!config) {
|
|
58
|
+
console.log('\n Error: devkit.json is corrupted.');
|
|
59
|
+
console.log(' Try reinitializing with: devkit init --force\n');
|
|
60
|
+
return { success: false, reason: 'corrupted_config' };
|
|
61
|
+
}
|
|
62
|
+
console.log(`\n Current version: ${config.version}`);
|
|
63
|
+
console.log(` Installed: ${new Date(config.installedAt).toLocaleDateString()}`);
|
|
64
|
+
|
|
65
|
+
// Check for package updates
|
|
66
|
+
console.log('\n Checking for updates...');
|
|
67
|
+
const latestVersion = checkForUpdates();
|
|
68
|
+
|
|
69
|
+
if (latestVersion && latestVersion !== VERSION) {
|
|
70
|
+
console.log(` New version available: ${latestVersion}`);
|
|
71
|
+
console.log(' Run: npm update -g @ngxtm/devkit');
|
|
72
|
+
console.log(' Then: devkit update\n');
|
|
73
|
+
} else if (latestVersion) {
|
|
74
|
+
console.log(` Package is up to date (${VERSION})`);
|
|
75
|
+
} else {
|
|
76
|
+
console.log(' Could not check for updates (offline?)');
|
|
77
|
+
}
|
|
78
|
+
|
|
79
|
+
// Re-detect project type
|
|
80
|
+
console.log('\n Re-detecting project type...');
|
|
81
|
+
const newDetected = detectProjectType(projectDir);
|
|
82
|
+
const oldDetected = config.detectedTypes || [];
|
|
83
|
+
|
|
84
|
+
// Find differences
|
|
85
|
+
const added = newDetected.filter(t => !oldDetected.includes(t));
|
|
86
|
+
const removed = oldDetected.filter(t => !newDetected.includes(t));
|
|
87
|
+
const unchanged = newDetected.filter(t => oldDetected.includes(t));
|
|
88
|
+
|
|
89
|
+
if (added.length > 0) {
|
|
90
|
+
console.log(` New tech detected: ${added.join(', ')}`);
|
|
91
|
+
}
|
|
92
|
+
if (removed.length > 0) {
|
|
93
|
+
console.log(` Removed: ${removed.join(', ')}`);
|
|
94
|
+
}
|
|
95
|
+
if (unchanged.length > 0) {
|
|
96
|
+
console.log(` Unchanged: ${unchanged.join(', ')}`);
|
|
97
|
+
}
|
|
98
|
+
if (added.length === 0 && removed.length === 0) {
|
|
99
|
+
console.log(' No changes in detected technologies.');
|
|
100
|
+
}
|
|
101
|
+
|
|
102
|
+
// Update rules if needed
|
|
103
|
+
if (added.length > 0 || options.force) {
|
|
104
|
+
const newRules = getRulesForTypes(added);
|
|
105
|
+
const rulesDir = path.join(claudeDir, 'rules');
|
|
106
|
+
|
|
107
|
+
console.log('\n Installing new rules...');
|
|
108
|
+
|
|
109
|
+
for (const ruleType of newRules) {
|
|
110
|
+
const srcRulesDir = path.join(PACKAGE_ROOT, 'templates', ruleType, 'rules');
|
|
111
|
+
if (fs.existsSync(srcRulesDir)) {
|
|
112
|
+
const destRulesDir = path.join(rulesDir, ruleType);
|
|
113
|
+
const count = copyDir(srcRulesDir, destRulesDir);
|
|
114
|
+
console.log(` ${ruleType}: ${count} files`);
|
|
115
|
+
}
|
|
116
|
+
}
|
|
117
|
+
}
|
|
118
|
+
|
|
119
|
+
// Remove old rules if needed and user agrees
|
|
120
|
+
if (removed.length > 0 && options.clean) {
|
|
121
|
+
const rulesDir = path.join(claudeDir, 'rules');
|
|
122
|
+
|
|
123
|
+
console.log('\n Removing old rules...');
|
|
124
|
+
|
|
125
|
+
for (const ruleType of removed) {
|
|
126
|
+
const ruleDir = path.join(rulesDir, ruleType);
|
|
127
|
+
if (fs.existsSync(ruleDir)) {
|
|
128
|
+
fs.rmSync(ruleDir, { recursive: true, force: true });
|
|
129
|
+
console.log(` Removed: ${ruleType}`);
|
|
130
|
+
}
|
|
131
|
+
}
|
|
132
|
+
}
|
|
133
|
+
|
|
134
|
+
// Update commands if --force
|
|
135
|
+
if (options.force) {
|
|
136
|
+
console.log('\n Updating commands...');
|
|
137
|
+
const mergedCommandsDir = path.join(PACKAGE_ROOT, 'merged-commands');
|
|
138
|
+
const commandsDir = path.join(claudeDir, 'commands');
|
|
139
|
+
|
|
140
|
+
if (fs.existsSync(mergedCommandsDir)) {
|
|
141
|
+
// Backup existing
|
|
142
|
+
const backupDir = path.join(claudeDir, 'commands.backup');
|
|
143
|
+
if (fs.existsSync(commandsDir)) {
|
|
144
|
+
fs.renameSync(commandsDir, backupDir);
|
|
145
|
+
}
|
|
146
|
+
|
|
147
|
+
const count = copyDir(mergedCommandsDir, commandsDir);
|
|
148
|
+
console.log(` Commands: ${count} files`);
|
|
149
|
+
|
|
150
|
+
// Remove backup
|
|
151
|
+
if (fs.existsSync(backupDir)) {
|
|
152
|
+
fs.rmSync(backupDir, { recursive: true, force: true });
|
|
153
|
+
}
|
|
154
|
+
}
|
|
155
|
+
}
|
|
156
|
+
|
|
157
|
+
// Update config
|
|
158
|
+
const newConfig = {
|
|
159
|
+
...config,
|
|
160
|
+
version: VERSION,
|
|
161
|
+
detectedTypes: newDetected,
|
|
162
|
+
installedRules: getRulesForTypes(newDetected),
|
|
163
|
+
updatedAt: new Date().toISOString(),
|
|
164
|
+
stats: {
|
|
165
|
+
totalFiles: config.stats?.totalFiles || 0,
|
|
166
|
+
sizeKB: Math.round(getDirSize(claudeDir) / 1024)
|
|
167
|
+
}
|
|
168
|
+
};
|
|
169
|
+
|
|
170
|
+
fs.writeFileSync(configPath, JSON.stringify(newConfig, null, 2));
|
|
171
|
+
|
|
172
|
+
console.log('\n' + '='.repeat(60));
|
|
173
|
+
console.log(' UPDATE COMPLETE');
|
|
174
|
+
console.log('='.repeat(60));
|
|
175
|
+
console.log(`\n Size: ${newConfig.stats.sizeKB} KB`);
|
|
176
|
+
console.log(` Detected: ${newDetected.join(', ') || 'generic'}\n`);
|
|
177
|
+
|
|
178
|
+
return {
|
|
179
|
+
success: true,
|
|
180
|
+
added,
|
|
181
|
+
removed,
|
|
182
|
+
detected: newDetected
|
|
183
|
+
};
|
|
184
|
+
}
|
|
185
|
+
|
|
186
|
+
/**
|
|
187
|
+
* Show project status
|
|
188
|
+
*/
|
|
189
|
+
function showStatus(options = {}) {
|
|
190
|
+
const projectDir = options.path || process.cwd();
|
|
191
|
+
const claudeDir = path.join(projectDir, '.claude');
|
|
192
|
+
const configPath = path.join(claudeDir, 'devkit.json');
|
|
193
|
+
|
|
194
|
+
console.log('\n' + '='.repeat(60));
|
|
195
|
+
console.log(' DEVKIT - STATUS');
|
|
196
|
+
console.log('='.repeat(60));
|
|
197
|
+
|
|
198
|
+
if (!fs.existsSync(configPath)) {
|
|
199
|
+
console.log('\n No devkit installation found.');
|
|
200
|
+
console.log(' Run: devkit init\n');
|
|
201
|
+
return { installed: false };
|
|
202
|
+
}
|
|
203
|
+
|
|
204
|
+
const config = parseJsonFile(configPath);
|
|
205
|
+
if (!config) {
|
|
206
|
+
console.log('\n Error: devkit.json is corrupted.');
|
|
207
|
+
console.log(' Try reinitializing with: devkit init --force\n');
|
|
208
|
+
return { installed: false, corrupted: true };
|
|
209
|
+
}
|
|
210
|
+
|
|
211
|
+
console.log(`\n Version: ${config.version}`);
|
|
212
|
+
console.log(` Installed: ${new Date(config.installedAt).toLocaleDateString()}`);
|
|
213
|
+
if (config.updatedAt) {
|
|
214
|
+
console.log(` Updated: ${new Date(config.updatedAt).toLocaleDateString()}`);
|
|
215
|
+
}
|
|
216
|
+
console.log(`\n Detected: ${config.detectedTypes?.join(', ') || 'none'}`);
|
|
217
|
+
console.log(` Rules: ${config.installedRules?.join(', ') || 'none'}`);
|
|
218
|
+
console.log(` Size: ${config.stats?.sizeKB || 'unknown'} KB\n`);
|
|
219
|
+
|
|
220
|
+
return { installed: true, config };
|
|
221
|
+
}
|
|
222
|
+
|
|
223
|
+
module.exports = {
|
|
224
|
+
updateProject,
|
|
225
|
+
showStatus,
|
|
226
|
+
checkForUpdates
|
|
227
|
+
};
|
|
228
|
+
|
|
229
|
+
// Run if called directly
|
|
230
|
+
if (require.main === module) {
|
|
231
|
+
const args = process.argv.slice(2);
|
|
232
|
+
const options = {
|
|
233
|
+
force: args.includes('--force') || args.includes('-f'),
|
|
234
|
+
clean: args.includes('--clean') || args.includes('-c'),
|
|
235
|
+
path: args.find(a => !a.startsWith('-')) || process.cwd()
|
|
236
|
+
};
|
|
237
|
+
|
|
238
|
+
if (args.includes('--status')) {
|
|
239
|
+
showStatus(options);
|
|
240
|
+
} else {
|
|
241
|
+
updateProject(options);
|
|
242
|
+
}
|
|
243
|
+
}
|
package/cli/utils.js
ADDED
|
@@ -0,0 +1,195 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Shared Utilities for Devkit CLI
|
|
3
|
+
*
|
|
4
|
+
* Common functions used across CLI modules.
|
|
5
|
+
*/
|
|
6
|
+
|
|
7
|
+
const fs = require('fs');
|
|
8
|
+
const path = require('path');
|
|
9
|
+
|
|
10
|
+
/**
|
|
11
|
+
* Copy directory recursively
|
|
12
|
+
* @param {string} src - Source directory
|
|
13
|
+
* @param {string} dest - Destination directory
|
|
14
|
+
* @param {Object} options - Options
|
|
15
|
+
* @param {string[]} options.exclude - Patterns to exclude
|
|
16
|
+
* @param {Set} options.visited - Visited directories (for cycle detection)
|
|
17
|
+
* @param {number} options.depth - Current depth (max 50)
|
|
18
|
+
* @returns {number} - Number of files copied
|
|
19
|
+
*/
|
|
20
|
+
function copyDir(src, dest, options = {}) {
|
|
21
|
+
const { exclude = [], visited = new Set(), depth = 0 } = options;
|
|
22
|
+
|
|
23
|
+
// Prevent infinite recursion
|
|
24
|
+
if (depth > 50) return 0;
|
|
25
|
+
|
|
26
|
+
if (!fs.existsSync(src)) return 0;
|
|
27
|
+
|
|
28
|
+
// Detect symlink cycles
|
|
29
|
+
try {
|
|
30
|
+
const realSrc = fs.realpathSync(src);
|
|
31
|
+
if (visited.has(realSrc)) return 0;
|
|
32
|
+
visited.add(realSrc);
|
|
33
|
+
} catch (e) {
|
|
34
|
+
return 0;
|
|
35
|
+
}
|
|
36
|
+
|
|
37
|
+
fs.mkdirSync(dest, { recursive: true });
|
|
38
|
+
let count = 0;
|
|
39
|
+
|
|
40
|
+
const entries = fs.readdirSync(src, { withFileTypes: true });
|
|
41
|
+
|
|
42
|
+
for (const entry of entries) {
|
|
43
|
+
// Skip hidden files, tests, node_modules by default
|
|
44
|
+
if (entry.name.startsWith('.') ||
|
|
45
|
+
entry.name === '__tests__' ||
|
|
46
|
+
entry.name === 'tests' ||
|
|
47
|
+
entry.name === 'node_modules') {
|
|
48
|
+
continue;
|
|
49
|
+
}
|
|
50
|
+
|
|
51
|
+
// Check custom exclusions
|
|
52
|
+
if (exclude.some(pattern => entry.name.includes(pattern))) {
|
|
53
|
+
continue;
|
|
54
|
+
}
|
|
55
|
+
|
|
56
|
+
const srcPath = path.join(src, entry.name);
|
|
57
|
+
const destPath = path.join(dest, entry.name);
|
|
58
|
+
|
|
59
|
+
if (entry.isDirectory()) {
|
|
60
|
+
count += copyDir(srcPath, destPath, { exclude, visited, depth: depth + 1 });
|
|
61
|
+
} else {
|
|
62
|
+
fs.copyFileSync(srcPath, destPath);
|
|
63
|
+
count++;
|
|
64
|
+
}
|
|
65
|
+
}
|
|
66
|
+
|
|
67
|
+
return count;
|
|
68
|
+
}
|
|
69
|
+
|
|
70
|
+
/**
|
|
71
|
+
* Get directory size in KB
|
|
72
|
+
* @param {string} dir - Directory path
|
|
73
|
+
* @param {Set} visited - Visited directories (for cycle detection)
|
|
74
|
+
* @param {number} depth - Current depth (max 50)
|
|
75
|
+
* @returns {number} - Size in bytes
|
|
76
|
+
*/
|
|
77
|
+
function getDirSize(dir, visited = new Set(), depth = 0) {
|
|
78
|
+
// Prevent infinite recursion
|
|
79
|
+
if (depth > 50) return 0;
|
|
80
|
+
|
|
81
|
+
if (!fs.existsSync(dir)) return 0;
|
|
82
|
+
|
|
83
|
+
// Detect symlink cycles
|
|
84
|
+
try {
|
|
85
|
+
const realDir = fs.realpathSync(dir);
|
|
86
|
+
if (visited.has(realDir)) return 0;
|
|
87
|
+
visited.add(realDir);
|
|
88
|
+
} catch (e) {
|
|
89
|
+
return 0;
|
|
90
|
+
}
|
|
91
|
+
|
|
92
|
+
let size = 0;
|
|
93
|
+
const entries = fs.readdirSync(dir, { withFileTypes: true });
|
|
94
|
+
|
|
95
|
+
for (const entry of entries) {
|
|
96
|
+
const entryPath = path.join(dir, entry.name);
|
|
97
|
+
try {
|
|
98
|
+
if (entry.isDirectory()) {
|
|
99
|
+
size += getDirSize(entryPath, visited, depth + 1);
|
|
100
|
+
} else {
|
|
101
|
+
size += fs.statSync(entryPath).size;
|
|
102
|
+
}
|
|
103
|
+
} catch (e) {
|
|
104
|
+
// Skip unreadable files
|
|
105
|
+
}
|
|
106
|
+
}
|
|
107
|
+
|
|
108
|
+
return size;
|
|
109
|
+
}
|
|
110
|
+
|
|
111
|
+
/**
|
|
112
|
+
* Safely parse JSON file
|
|
113
|
+
* @param {string} filePath - Path to JSON file
|
|
114
|
+
* @param {*} defaultValue - Default value if parsing fails
|
|
115
|
+
* @returns {*} - Parsed JSON or default value
|
|
116
|
+
*/
|
|
117
|
+
function parseJsonFile(filePath, defaultValue = null) {
|
|
118
|
+
try {
|
|
119
|
+
if (!fs.existsSync(filePath)) {
|
|
120
|
+
return defaultValue;
|
|
121
|
+
}
|
|
122
|
+
const content = fs.readFileSync(filePath, 'utf-8');
|
|
123
|
+
return JSON.parse(content);
|
|
124
|
+
} catch (e) {
|
|
125
|
+
console.error(`Warning: Failed to parse ${path.basename(filePath)}: ${e.message}`);
|
|
126
|
+
return defaultValue;
|
|
127
|
+
}
|
|
128
|
+
}
|
|
129
|
+
|
|
130
|
+
/**
|
|
131
|
+
* Validate and resolve path
|
|
132
|
+
* @param {string} inputPath - User-provided path
|
|
133
|
+
* @param {boolean} mustExist - Whether path must exist
|
|
134
|
+
* @returns {string|null} - Resolved absolute path or null if invalid
|
|
135
|
+
*/
|
|
136
|
+
function validatePath(inputPath, mustExist = false) {
|
|
137
|
+
if (!inputPath) return null;
|
|
138
|
+
|
|
139
|
+
try {
|
|
140
|
+
const resolved = path.resolve(inputPath);
|
|
141
|
+
|
|
142
|
+
if (mustExist && !fs.existsSync(resolved)) {
|
|
143
|
+
console.error(`Error: Path does not exist: ${resolved}`);
|
|
144
|
+
return null;
|
|
145
|
+
}
|
|
146
|
+
|
|
147
|
+
// Check if it's a directory when it exists
|
|
148
|
+
if (fs.existsSync(resolved) && !fs.statSync(resolved).isDirectory()) {
|
|
149
|
+
console.error(`Error: Path is not a directory: ${resolved}`);
|
|
150
|
+
return null;
|
|
151
|
+
}
|
|
152
|
+
|
|
153
|
+
return resolved;
|
|
154
|
+
} catch (e) {
|
|
155
|
+
console.error(`Error: Invalid path: ${inputPath}`);
|
|
156
|
+
return null;
|
|
157
|
+
}
|
|
158
|
+
}
|
|
159
|
+
|
|
160
|
+
/**
|
|
161
|
+
* Get all files and directories from a path recursively
|
|
162
|
+
* @param {string} dir - Directory to scan
|
|
163
|
+
* @param {string} basePath - Base path for relative paths
|
|
164
|
+
* @returns {{files: string[], dirs: string[]}} - Files and directories
|
|
165
|
+
*/
|
|
166
|
+
function getAllEntries(dir, basePath = '') {
|
|
167
|
+
const entries = { files: [], dirs: [] };
|
|
168
|
+
|
|
169
|
+
if (!fs.existsSync(dir)) return entries;
|
|
170
|
+
|
|
171
|
+
const items = fs.readdirSync(dir, { withFileTypes: true });
|
|
172
|
+
|
|
173
|
+
for (const item of items) {
|
|
174
|
+
const relativePath = path.join(basePath, item.name);
|
|
175
|
+
|
|
176
|
+
if (item.isDirectory()) {
|
|
177
|
+
entries.dirs.push(relativePath);
|
|
178
|
+
const subEntries = getAllEntries(path.join(dir, item.name), relativePath);
|
|
179
|
+
entries.files.push(...subEntries.files);
|
|
180
|
+
entries.dirs.push(...subEntries.dirs);
|
|
181
|
+
} else {
|
|
182
|
+
entries.files.push(relativePath);
|
|
183
|
+
}
|
|
184
|
+
}
|
|
185
|
+
|
|
186
|
+
return entries;
|
|
187
|
+
}
|
|
188
|
+
|
|
189
|
+
module.exports = {
|
|
190
|
+
copyDir,
|
|
191
|
+
getDirSize,
|
|
192
|
+
parseJsonFile,
|
|
193
|
+
validatePath,
|
|
194
|
+
getAllEntries
|
|
195
|
+
};
|