@therocketcode/gsd-core 1.4.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/.claude-plugin/plugin.json +23 -0
- package/GEMINI.md +53 -0
- package/LICENSE +21 -0
- package/README.ja-JP.md +125 -0
- package/README.ko-KR.md +125 -0
- package/README.md +144 -0
- package/README.pt-BR.md +125 -0
- package/README.zh-CN.md +125 -0
- package/agents/gsd-advisor-researcher.md +108 -0
- package/agents/gsd-ai-researcher.md +114 -0
- package/agents/gsd-assumptions-analyzer.md +105 -0
- package/agents/gsd-code-fixer.md +668 -0
- package/agents/gsd-code-reviewer.md +387 -0
- package/agents/gsd-codebase-mapper.md +853 -0
- package/agents/gsd-debug-session-manager.md +314 -0
- package/agents/gsd-debugger.md +1452 -0
- package/agents/gsd-doc-classifier.md +168 -0
- package/agents/gsd-doc-synthesizer.md +204 -0
- package/agents/gsd-doc-verifier.md +217 -0
- package/agents/gsd-doc-writer.md +616 -0
- package/agents/gsd-domain-researcher.md +147 -0
- package/agents/gsd-eval-auditor.md +191 -0
- package/agents/gsd-eval-planner.md +154 -0
- package/agents/gsd-executor.md +785 -0
- package/agents/gsd-framework-selector.md +160 -0
- package/agents/gsd-integration-checker.md +470 -0
- package/agents/gsd-intel-updater.md +342 -0
- package/agents/gsd-nyquist-auditor.md +203 -0
- package/agents/gsd-pattern-mapper.md +335 -0
- package/agents/gsd-phase-researcher.md +867 -0
- package/agents/gsd-plan-checker.md +978 -0
- package/agents/gsd-planner.md +1204 -0
- package/agents/gsd-project-researcher.md +611 -0
- package/agents/gsd-research-synthesizer.md +259 -0
- package/agents/gsd-roadmapper.md +688 -0
- package/agents/gsd-security-auditor.md +155 -0
- package/agents/gsd-ui-auditor.md +495 -0
- package/agents/gsd-ui-checker.md +309 -0
- package/agents/gsd-ui-researcher.md +374 -0
- package/agents/gsd-user-profiler.md +171 -0
- package/agents/gsd-verifier.md +923 -0
- package/assets/gsd-logo-2000-transparent.png +0 -0
- package/assets/gsd-logo-2000-transparent.svg +17 -0
- package/assets/gsd-logo-2000.png +0 -0
- package/assets/gsd-logo-2000.svg +21 -0
- package/assets/terminal.svg +68 -0
- package/bin/install.js +12726 -0
- package/bin/lib/ui-safety-gate.cjs +107 -0
- package/commands/gsd/add-tests.md +42 -0
- package/commands/gsd/ai-integration-phase.md +37 -0
- package/commands/gsd/audit-fix.md +34 -0
- package/commands/gsd/audit-milestone.md +37 -0
- package/commands/gsd/audit-uat.md +24 -0
- package/commands/gsd/autonomous.md +48 -0
- package/commands/gsd/capture.md +62 -0
- package/commands/gsd/cleanup.md +24 -0
- package/commands/gsd/code-review.md +59 -0
- package/commands/gsd/complete-milestone.md +143 -0
- package/commands/gsd/config.md +56 -0
- package/commands/gsd/debug.md +52 -0
- package/commands/gsd/discover-product.md +65 -0
- package/commands/gsd/discuss-phase.md +77 -0
- package/commands/gsd/docs-update.md +49 -0
- package/commands/gsd/eval-review.md +33 -0
- package/commands/gsd/execute-phase.md +66 -0
- package/commands/gsd/explore.md +27 -0
- package/commands/gsd/extract-learnings.md +23 -0
- package/commands/gsd/fast.md +31 -0
- package/commands/gsd/forensics.md +57 -0
- package/commands/gsd/graphify.md +204 -0
- package/commands/gsd/health.md +31 -0
- package/commands/gsd/help.md +28 -0
- package/commands/gsd/import.md +45 -0
- package/commands/gsd/inbox.md +39 -0
- package/commands/gsd/ingest-docs.md +42 -0
- package/commands/gsd/manager.md +45 -0
- package/commands/gsd/map-codebase.md +83 -0
- package/commands/gsd/milestone-summary.md +51 -0
- package/commands/gsd/model-domain.md +65 -0
- package/commands/gsd/mvp-phase.md +45 -0
- package/commands/gsd/new-milestone.md +45 -0
- package/commands/gsd/new-project.md +47 -0
- package/commands/gsd/ns-context.md +23 -0
- package/commands/gsd/ns-ideate.md +24 -0
- package/commands/gsd/ns-manage.md +29 -0
- package/commands/gsd/ns-project.md +22 -0
- package/commands/gsd/ns-review.md +26 -0
- package/commands/gsd/ns-workflow.md +28 -0
- package/commands/gsd/pause-work.md +43 -0
- package/commands/gsd/phase.md +56 -0
- package/commands/gsd/plan-phase.md +64 -0
- package/commands/gsd/plan-review-convergence.md +59 -0
- package/commands/gsd/pr-branch.md +26 -0
- package/commands/gsd/profile-user.md +46 -0
- package/commands/gsd/progress.md +48 -0
- package/commands/gsd/quick.md +174 -0
- package/commands/gsd/recommend-architecture.md +64 -0
- package/commands/gsd/resume-work.md +30 -0
- package/commands/gsd/review-backlog.md +63 -0
- package/commands/gsd/review.md +42 -0
- package/commands/gsd/secure-phase.md +36 -0
- package/commands/gsd/settings.md +29 -0
- package/commands/gsd/ship.md +24 -0
- package/commands/gsd/sketch.md +60 -0
- package/commands/gsd/spec-phase.md +63 -0
- package/commands/gsd/spike.md +57 -0
- package/commands/gsd/stats.md +20 -0
- package/commands/gsd/surface.md +155 -0
- package/commands/gsd/testing-strategy.md +65 -0
- package/commands/gsd/thread.md +24 -0
- package/commands/gsd/ui-phase.md +35 -0
- package/commands/gsd/ui-review.md +33 -0
- package/commands/gsd/ultraplan-phase.md +34 -0
- package/commands/gsd/undo.md +35 -0
- package/commands/gsd/update.md +49 -0
- package/commands/gsd/validate-phase.md +36 -0
- package/commands/gsd/verify-work.md +39 -0
- package/commands/gsd/workspace.md +52 -0
- package/commands/gsd/workstreams.md +70 -0
- package/gemini-extension.json +6 -0
- package/gsd-core/bin/check-latest-version.cjs +161 -0
- package/gsd-core/bin/gsd-tools.cjs +1928 -0
- package/gsd-core/bin/lib/active-workstream-store.cjs +291 -0
- package/gsd-core/bin/lib/adr-parser.cjs +399 -0
- package/gsd-core/bin/lib/agent-command-router.cjs +68 -0
- package/gsd-core/bin/lib/artifacts.cjs +51 -0
- package/gsd-core/bin/lib/audit.cjs +743 -0
- package/gsd-core/bin/lib/check-command-router.cjs +343 -0
- package/gsd-core/bin/lib/cjs-command-router-adapter.cjs +81 -0
- package/gsd-core/bin/lib/cli-exit.cjs +42 -0
- package/gsd-core/bin/lib/clock.cjs +95 -0
- package/gsd-core/bin/lib/clusters.cjs +132 -0
- package/gsd-core/bin/lib/code-review-flags.cjs +59 -0
- package/gsd-core/bin/lib/command-aliases.cjs +809 -0
- package/gsd-core/bin/lib/command-arg-projection.cjs +55 -0
- package/gsd-core/bin/lib/command-routing-hub.cjs +300 -0
- package/gsd-core/bin/lib/commands.cjs +1203 -0
- package/gsd-core/bin/lib/config-schema.cjs +29 -0
- package/gsd-core/bin/lib/config-types.cjs +19 -0
- package/gsd-core/bin/lib/config.cjs +738 -0
- package/gsd-core/bin/lib/configuration.cjs +239 -0
- package/gsd-core/bin/lib/context-utilization.cjs +48 -0
- package/gsd-core/bin/lib/core.cjs +2051 -0
- package/gsd-core/bin/lib/decisions.cjs +118 -0
- package/gsd-core/bin/lib/docs.cjs +252 -0
- package/gsd-core/bin/lib/drift.cjs +364 -0
- package/gsd-core/bin/lib/fallow-runner.cjs +115 -0
- package/gsd-core/bin/lib/frontmatter.cjs +442 -0
- package/gsd-core/bin/lib/gap-checker.cjs +257 -0
- package/gsd-core/bin/lib/graphify.cjs +496 -0
- package/gsd-core/bin/lib/gsd2-import.cjs +456 -0
- package/gsd-core/bin/lib/init-command-router.cjs +62 -0
- package/gsd-core/bin/lib/init.cjs +1815 -0
- package/gsd-core/bin/lib/install-profiles.cjs +584 -0
- package/gsd-core/bin/lib/installer-migration-authoring.cjs +122 -0
- package/gsd-core/bin/lib/installer-migration-report.cjs +350 -0
- package/gsd-core/bin/lib/installer-migrations/000-first-time-baseline.cjs +218 -0
- package/gsd-core/bin/lib/installer-migrations/001-legacy-orphan-files.cjs +48 -0
- package/gsd-core/bin/lib/installer-migrations/002-codex-legacy-hooks-json.cjs +94 -0
- package/gsd-core/bin/lib/installer-migrations/003-rename-get-shit-done-to-gsd-core.cjs +108 -0
- package/gsd-core/bin/lib/installer-migrations.cjs +823 -0
- package/gsd-core/bin/lib/intel.cjs +590 -0
- package/gsd-core/bin/lib/learnings.cjs +270 -0
- package/gsd-core/bin/lib/legacy-cleanup.cjs +253 -0
- package/gsd-core/bin/lib/milestone.cjs +373 -0
- package/gsd-core/bin/lib/model-catalog.cjs +154 -0
- package/gsd-core/bin/lib/model-profiles.cjs +24 -0
- package/gsd-core/bin/lib/observability/event.cjs +51 -0
- package/gsd-core/bin/lib/observability/logger.cjs +146 -0
- package/gsd-core/bin/lib/observability/redaction.cjs +48 -0
- package/gsd-core/bin/lib/package-identity.cjs +35 -0
- package/gsd-core/bin/lib/package-legitimacy.cjs +368 -0
- package/gsd-core/bin/lib/phase-command-router.cjs +189 -0
- package/gsd-core/bin/lib/phase-lifecycle.cjs +74 -0
- package/gsd-core/bin/lib/phase.cjs +1307 -0
- package/gsd-core/bin/lib/phases-command-router.cjs +43 -0
- package/gsd-core/bin/lib/plan-scan.cjs +91 -0
- package/gsd-core/bin/lib/planning-workspace.cjs +245 -0
- package/gsd-core/bin/lib/profile-output.cjs +1120 -0
- package/gsd-core/bin/lib/profile-pipeline.cjs +517 -0
- package/gsd-core/bin/lib/project-root.cjs +119 -0
- package/gsd-core/bin/lib/prompt-budget.cjs +305 -0
- package/gsd-core/bin/lib/research-provider.cjs +137 -0
- package/gsd-core/bin/lib/research-store.cjs +167 -0
- package/gsd-core/bin/lib/review-reviewer-selection.cjs +121 -0
- package/gsd-core/bin/lib/roadmap-command-router.cjs +166 -0
- package/gsd-core/bin/lib/roadmap-upgrade.cjs +476 -0
- package/gsd-core/bin/lib/roadmap.cjs +600 -0
- package/gsd-core/bin/lib/runtime-artifact-layout.cjs +312 -0
- package/gsd-core/bin/lib/runtime-config-adapter-registry.cjs +56 -0
- package/gsd-core/bin/lib/runtime-homes.cjs +190 -0
- package/gsd-core/bin/lib/runtime-name-policy.cjs +96 -0
- package/gsd-core/bin/lib/runtime-slash.cjs +119 -0
- package/gsd-core/bin/lib/schema-detect.cjs +159 -0
- package/gsd-core/bin/lib/secrets.cjs +34 -0
- package/gsd-core/bin/lib/security.cjs +480 -0
- package/gsd-core/bin/lib/semver-compare.cjs +42 -0
- package/gsd-core/bin/lib/shell-command-projection.cjs +533 -0
- package/gsd-core/bin/lib/state-command-router.cjs +160 -0
- package/gsd-core/bin/lib/state-document.cjs +259 -0
- package/gsd-core/bin/lib/state.cjs +2010 -0
- package/gsd-core/bin/lib/surface.cjs +449 -0
- package/gsd-core/bin/lib/task-command-router.cjs +85 -0
- package/gsd-core/bin/lib/template.cjs +237 -0
- package/gsd-core/bin/lib/uat.cjs +297 -0
- package/gsd-core/bin/lib/ui-safety-gate.cjs +98 -0
- package/gsd-core/bin/lib/update-context.cjs +218 -0
- package/gsd-core/bin/lib/validate-command-router.cjs +91 -0
- package/gsd-core/bin/lib/validate.cjs +112 -0
- package/gsd-core/bin/lib/verification-command-router.cjs +31 -0
- package/gsd-core/bin/lib/verification.cjs +193 -0
- package/gsd-core/bin/lib/verify-command-router.cjs +44 -0
- package/gsd-core/bin/lib/verify.cjs +1451 -0
- package/gsd-core/bin/lib/workstream-inventory-builder.cjs +81 -0
- package/gsd-core/bin/lib/workstream-inventory.cjs +147 -0
- package/gsd-core/bin/lib/workstream-name-policy.cjs +91 -0
- package/gsd-core/bin/lib/workstream.cjs +380 -0
- package/gsd-core/bin/lib/worktree-base-ref.cjs +325 -0
- package/gsd-core/bin/lib/worktree-safety.cjs +943 -0
- package/gsd-core/bin/shared/config-defaults.manifest.json +98 -0
- package/gsd-core/bin/shared/config-schema.manifest.json +192 -0
- package/gsd-core/bin/shared/model-catalog.json +149 -0
- package/gsd-core/bin/shared/runtime-aliases.manifest.json +75 -0
- package/gsd-core/bin/verify-reapply-patches.cjs +349 -0
- package/gsd-core/contexts/dev.md +21 -0
- package/gsd-core/contexts/research.md +22 -0
- package/gsd-core/contexts/review.md +23 -0
- package/gsd-core/references/agent-contracts.md +79 -0
- package/gsd-core/references/ai-evals.md +156 -0
- package/gsd-core/references/ai-frameworks.md +186 -0
- package/gsd-core/references/architecture-decision.md +74 -0
- package/gsd-core/references/artifact-types.md +131 -0
- package/gsd-core/references/auth-in-tests.md +91 -0
- package/gsd-core/references/autonomous-smart-discuss.md +277 -0
- package/gsd-core/references/checkpoints.md +814 -0
- package/gsd-core/references/common-bug-patterns.md +114 -0
- package/gsd-core/references/context-budget.md +85 -0
- package/gsd-core/references/continuation-format.md +253 -0
- package/gsd-core/references/db-test-isolation.md +54 -0
- package/gsd-core/references/debugger-philosophy.md +76 -0
- package/gsd-core/references/decimal-phase-calculation.md +64 -0
- package/gsd-core/references/doc-conflict-engine.md +91 -0
- package/gsd-core/references/domain-modeling.md +80 -0
- package/gsd-core/references/domain-probes.md +125 -0
- package/gsd-core/references/e2e-tiering.md +35 -0
- package/gsd-core/references/execute-mvp-tdd.md +81 -0
- package/gsd-core/references/executor-examples.md +110 -0
- package/gsd-core/references/few-shot-examples/plan-checker.md +73 -0
- package/gsd-core/references/few-shot-examples/verifier.md +109 -0
- package/gsd-core/references/flaky-test-checklist.md +22 -0
- package/gsd-core/references/gate-prompts.md +100 -0
- package/gsd-core/references/gates.md +70 -0
- package/gsd-core/references/git-integration.md +298 -0
- package/gsd-core/references/git-planning-commit.md +40 -0
- package/gsd-core/references/ios-scaffold.md +123 -0
- package/gsd-core/references/mandatory-initial-read.md +2 -0
- package/gsd-core/references/model-profile-resolution.md +38 -0
- package/gsd-core/references/model-profiles.md +245 -0
- package/gsd-core/references/mvp-concepts.md +49 -0
- package/gsd-core/references/phase-argument-parsing.md +61 -0
- package/gsd-core/references/planner-antipatterns.md +89 -0
- package/gsd-core/references/planner-chunked.md +49 -0
- package/gsd-core/references/planner-gap-closure.md +62 -0
- package/gsd-core/references/planner-graphify-auto-update.md +67 -0
- package/gsd-core/references/planner-human-verify-mode.md +57 -0
- package/gsd-core/references/planner-interface-context.md +62 -0
- package/gsd-core/references/planner-load-graph-context.md +36 -0
- package/gsd-core/references/planner-mvp-mode.md +53 -0
- package/gsd-core/references/planner-reviews.md +39 -0
- package/gsd-core/references/planner-revision.md +87 -0
- package/gsd-core/references/planner-source-audit.md +73 -0
- package/gsd-core/references/planning-config.md +473 -0
- package/gsd-core/references/product-discovery.md +49 -0
- package/gsd-core/references/project-skills-discovery.md +19 -0
- package/gsd-core/references/questioning.md +162 -0
- package/gsd-core/references/realistic-test-data.md +44 -0
- package/gsd-core/references/research-documentation-lookup.md +29 -0
- package/gsd-core/references/research-philosophy.md +29 -0
- package/gsd-core/references/research-verification-protocol.md +27 -0
- package/gsd-core/references/revision-loop.md +97 -0
- package/gsd-core/references/scout-codebase.md +51 -0
- package/gsd-core/references/skeleton-template.md +48 -0
- package/gsd-core/references/sketch-interactivity.md +41 -0
- package/gsd-core/references/sketch-theme-system.md +94 -0
- package/gsd-core/references/sketch-tooling.md +45 -0
- package/gsd-core/references/sketch-variant-patterns.md +81 -0
- package/gsd-core/references/spidr-splitting.md +69 -0
- package/gsd-core/references/tdd.md +330 -0
- package/gsd-core/references/test-containers.md +55 -0
- package/gsd-core/references/test-strategy.md +75 -0
- package/gsd-core/references/thinking-models-debug.md +44 -0
- package/gsd-core/references/thinking-models-execution.md +50 -0
- package/gsd-core/references/thinking-models-planning.md +62 -0
- package/gsd-core/references/thinking-models-research.md +50 -0
- package/gsd-core/references/thinking-models-verification.md +55 -0
- package/gsd-core/references/thinking-partner.md +96 -0
- package/gsd-core/references/ui-brand.md +162 -0
- package/gsd-core/references/universal-anti-patterns.md +63 -0
- package/gsd-core/references/user-profiling.md +681 -0
- package/gsd-core/references/user-story-template.md +58 -0
- package/gsd-core/references/verification-overrides.md +227 -0
- package/gsd-core/references/verification-patterns.md +612 -0
- package/gsd-core/references/verify-mvp-mode.md +85 -0
- package/gsd-core/references/workstream-flag.md +111 -0
- package/gsd-core/references/worktree-branch-check.md +38 -0
- package/gsd-core/references/worktree-path-safety.md +67 -0
- package/gsd-core/templates/AI-SPEC.md +246 -0
- package/gsd-core/templates/DEBUG.md +169 -0
- package/gsd-core/templates/README.md +77 -0
- package/gsd-core/templates/SECURITY.md +61 -0
- package/gsd-core/templates/UAT.md +265 -0
- package/gsd-core/templates/UI-SPEC.md +100 -0
- package/gsd-core/templates/VALIDATION.md +76 -0
- package/gsd-core/templates/adr.md +58 -0
- package/gsd-core/templates/claude-md.md +145 -0
- package/gsd-core/templates/codebase/architecture.md +255 -0
- package/gsd-core/templates/codebase/concerns.md +310 -0
- package/gsd-core/templates/codebase/conventions.md +307 -0
- package/gsd-core/templates/codebase/integrations.md +280 -0
- package/gsd-core/templates/codebase/stack.md +186 -0
- package/gsd-core/templates/codebase/structure.md +285 -0
- package/gsd-core/templates/codebase/testing.md +480 -0
- package/gsd-core/templates/config.json +62 -0
- package/gsd-core/templates/context.md +352 -0
- package/gsd-core/templates/continue-here.md +78 -0
- package/gsd-core/templates/copilot-instructions.md +7 -0
- package/gsd-core/templates/debug-subagent-prompt.md +91 -0
- package/gsd-core/templates/dev-preferences.md +21 -0
- package/gsd-core/templates/discovery.md +146 -0
- package/gsd-core/templates/discussion-log.md +63 -0
- package/gsd-core/templates/domain-model.md +54 -0
- package/gsd-core/templates/milestone-archive.md +123 -0
- package/gsd-core/templates/milestone.md +115 -0
- package/gsd-core/templates/phase-prompt.md +610 -0
- package/gsd-core/templates/planner-subagent-prompt.md +117 -0
- package/gsd-core/templates/product-brief.md +55 -0
- package/gsd-core/templates/project.md +186 -0
- package/gsd-core/templates/requirements.md +231 -0
- package/gsd-core/templates/research-project/ARCHITECTURE.md +204 -0
- package/gsd-core/templates/research-project/FEATURES.md +147 -0
- package/gsd-core/templates/research-project/PITFALLS.md +200 -0
- package/gsd-core/templates/research-project/STACK.md +120 -0
- package/gsd-core/templates/research-project/SUMMARY.md +170 -0
- package/gsd-core/templates/research.md +592 -0
- package/gsd-core/templates/retrospective.md +54 -0
- package/gsd-core/templates/roadmap.md +202 -0
- package/gsd-core/templates/spec.md +307 -0
- package/gsd-core/templates/state.md +195 -0
- package/gsd-core/templates/summary-complex.md +59 -0
- package/gsd-core/templates/summary-minimal.md +41 -0
- package/gsd-core/templates/summary-standard.md +48 -0
- package/gsd-core/templates/summary.md +248 -0
- package/gsd-core/templates/test-strategy.md +50 -0
- package/gsd-core/templates/user-profile.md +146 -0
- package/gsd-core/templates/user-setup.md +311 -0
- package/gsd-core/templates/verification-report.md +322 -0
- package/gsd-core/workflows/_runtime-launcher.snippet.sh +1 -0
- package/gsd-core/workflows/add-backlog.md +91 -0
- package/gsd-core/workflows/add-phase.md +113 -0
- package/gsd-core/workflows/add-tests.md +355 -0
- package/gsd-core/workflows/add-todo.md +161 -0
- package/gsd-core/workflows/ai-integration-phase.md +295 -0
- package/gsd-core/workflows/analyze-dependencies.md +96 -0
- package/gsd-core/workflows/audit-fix.md +178 -0
- package/gsd-core/workflows/audit-milestone.md +360 -0
- package/gsd-core/workflows/audit-uat.md +110 -0
- package/gsd-core/workflows/autonomous.md +797 -0
- package/gsd-core/workflows/check-todos.md +180 -0
- package/gsd-core/workflows/cleanup.md +195 -0
- package/gsd-core/workflows/code-review-fix.md +502 -0
- package/gsd-core/workflows/code-review.md +658 -0
- package/gsd-core/workflows/complete-milestone.md +855 -0
- package/gsd-core/workflows/debug.md +237 -0
- package/gsd-core/workflows/diagnose-issues.md +245 -0
- package/gsd-core/workflows/discover-product.md +112 -0
- package/gsd-core/workflows/discovery-phase.md +291 -0
- package/gsd-core/workflows/discuss-phase/modes/advisor.md +176 -0
- package/gsd-core/workflows/discuss-phase/modes/all.md +28 -0
- package/gsd-core/workflows/discuss-phase/modes/analyze.md +44 -0
- package/gsd-core/workflows/discuss-phase/modes/auto.md +57 -0
- package/gsd-core/workflows/discuss-phase/modes/batch.md +52 -0
- package/gsd-core/workflows/discuss-phase/modes/chain.md +98 -0
- package/gsd-core/workflows/discuss-phase/modes/default.md +141 -0
- package/gsd-core/workflows/discuss-phase/modes/power.md +44 -0
- package/gsd-core/workflows/discuss-phase/modes/text.md +55 -0
- package/gsd-core/workflows/discuss-phase/templates/checkpoint.json +18 -0
- package/gsd-core/workflows/discuss-phase/templates/context.md +136 -0
- package/gsd-core/workflows/discuss-phase/templates/discussion-log.md +50 -0
- package/gsd-core/workflows/discuss-phase-assumptions.md +675 -0
- package/gsd-core/workflows/discuss-phase-power.md +291 -0
- package/gsd-core/workflows/discuss-phase.md +499 -0
- package/gsd-core/workflows/do.md +111 -0
- package/gsd-core/workflows/docs-update.md +1176 -0
- package/gsd-core/workflows/edit-phase.md +295 -0
- package/gsd-core/workflows/eval-review.md +156 -0
- package/gsd-core/workflows/execute-phase/steps/codebase-drift-gate.md +95 -0
- package/gsd-core/workflows/execute-phase/steps/per-plan-worktree-gate.md +94 -0
- package/gsd-core/workflows/execute-phase/steps/post-merge-gate.md +117 -0
- package/gsd-core/workflows/execute-phase.md +1752 -0
- package/gsd-core/workflows/execute-plan.md +526 -0
- package/gsd-core/workflows/explore.md +146 -0
- package/gsd-core/workflows/extract-learnings.md +243 -0
- package/gsd-core/workflows/fast.md +124 -0
- package/gsd-core/workflows/forensics.md +279 -0
- package/gsd-core/workflows/graduation.md +196 -0
- package/gsd-core/workflows/health.md +224 -0
- package/gsd-core/workflows/help/modes/brief.md +22 -0
- package/gsd-core/workflows/help/modes/default.md +50 -0
- package/gsd-core/workflows/help/modes/full.md +789 -0
- package/gsd-core/workflows/help/modes/topic.md +74 -0
- package/gsd-core/workflows/help.md +24 -0
- package/gsd-core/workflows/import.md +256 -0
- package/gsd-core/workflows/inbox.md +387 -0
- package/gsd-core/workflows/ingest-docs.md +340 -0
- package/gsd-core/workflows/insert-phase.md +152 -0
- package/gsd-core/workflows/list-phase-assumptions.md +178 -0
- package/gsd-core/workflows/list-workspaces.md +57 -0
- package/gsd-core/workflows/manager.md +393 -0
- package/gsd-core/workflows/map-codebase.md +446 -0
- package/gsd-core/workflows/milestone-summary.md +224 -0
- package/gsd-core/workflows/model-domain.md +162 -0
- package/gsd-core/workflows/mvp-phase.md +222 -0
- package/gsd-core/workflows/new-milestone.md +635 -0
- package/gsd-core/workflows/new-project.md +1555 -0
- package/gsd-core/workflows/new-workspace.md +240 -0
- package/gsd-core/workflows/next.md +299 -0
- package/gsd-core/workflows/node-repair.md +92 -0
- package/gsd-core/workflows/note.md +158 -0
- package/gsd-core/workflows/pause-work.md +244 -0
- package/gsd-core/workflows/plan-milestone-gaps.md +281 -0
- package/gsd-core/workflows/plan-phase.md +1814 -0
- package/gsd-core/workflows/plan-review-convergence.md +346 -0
- package/gsd-core/workflows/plant-seed.md +230 -0
- package/gsd-core/workflows/pr-branch.md +157 -0
- package/gsd-core/workflows/profile-user.md +453 -0
- package/gsd-core/workflows/progress.md +699 -0
- package/gsd-core/workflows/quick.md +1017 -0
- package/gsd-core/workflows/reapply-patches.md +426 -0
- package/gsd-core/workflows/recommend-architecture.md +135 -0
- package/gsd-core/workflows/remove-phase.md +156 -0
- package/gsd-core/workflows/remove-workspace.md +108 -0
- package/gsd-core/workflows/resume-project.md +332 -0
- package/gsd-core/workflows/review.md +748 -0
- package/gsd-core/workflows/scan.md +107 -0
- package/gsd-core/workflows/secure-phase.md +182 -0
- package/gsd-core/workflows/session-report.md +146 -0
- package/gsd-core/workflows/settings-advanced.md +810 -0
- package/gsd-core/workflows/settings-integrations.md +312 -0
- package/gsd-core/workflows/settings.md +566 -0
- package/gsd-core/workflows/ship.md +405 -0
- package/gsd-core/workflows/sketch-wrap-up.md +286 -0
- package/gsd-core/workflows/sketch.md +361 -0
- package/gsd-core/workflows/spec-phase.md +263 -0
- package/gsd-core/workflows/spike-wrap-up.md +307 -0
- package/gsd-core/workflows/spike.md +453 -0
- package/gsd-core/workflows/stats.md +80 -0
- package/gsd-core/workflows/sync-skills.md +182 -0
- package/gsd-core/workflows/testing-strategy.md +122 -0
- package/gsd-core/workflows/thread.md +222 -0
- package/gsd-core/workflows/transition.md +694 -0
- package/gsd-core/workflows/ui-phase.md +328 -0
- package/gsd-core/workflows/ui-review.md +193 -0
- package/gsd-core/workflows/ultraplan-phase.md +199 -0
- package/gsd-core/workflows/undo.md +314 -0
- package/gsd-core/workflows/update.md +496 -0
- package/gsd-core/workflows/validate-phase.md +181 -0
- package/gsd-core/workflows/verify-phase.md +544 -0
- package/gsd-core/workflows/verify-work.md +781 -0
- package/hooks/dist/gsd-check-update-worker.js +108 -0
- package/hooks/dist/gsd-check-update.js +66 -0
- package/hooks/dist/gsd-config-reload.js +133 -0
- package/hooks/dist/gsd-context-monitor.js +195 -0
- package/hooks/dist/gsd-cursor-post-tool.js +75 -0
- package/hooks/dist/gsd-cursor-session-start.js +52 -0
- package/hooks/dist/gsd-graphify-update.sh +158 -0
- package/hooks/dist/gsd-phase-boundary.sh +47 -0
- package/hooks/dist/gsd-prompt-guard.js +97 -0
- package/hooks/dist/gsd-read-guard.js +101 -0
- package/hooks/dist/gsd-read-injection-scanner.js +203 -0
- package/hooks/dist/gsd-session-state.sh +59 -0
- package/hooks/dist/gsd-statusline.js +566 -0
- package/hooks/dist/gsd-update-banner.js +138 -0
- package/hooks/dist/gsd-validate-commit.sh +57 -0
- package/hooks/dist/gsd-workflow-guard.js +167 -0
- package/hooks/dist/gsd-worktree-path-guard.js +169 -0
- package/hooks/dist/lib/git-cmd.js +150 -0
- package/hooks/dist/lib/gsd-graphify-rebuild.sh +65 -0
- package/hooks/dist/managed-hooks-registry.cjs +38 -0
- package/hooks/gsd-check-update-worker.js +108 -0
- package/hooks/gsd-check-update.js +66 -0
- package/hooks/gsd-config-reload.js +133 -0
- package/hooks/gsd-context-monitor.js +195 -0
- package/hooks/gsd-cursor-post-tool.js +75 -0
- package/hooks/gsd-cursor-session-start.js +52 -0
- package/hooks/gsd-graphify-update.sh +158 -0
- package/hooks/gsd-phase-boundary.sh +47 -0
- package/hooks/gsd-prompt-guard.js +97 -0
- package/hooks/gsd-read-guard.js +101 -0
- package/hooks/gsd-read-injection-scanner.js +203 -0
- package/hooks/gsd-session-state.sh +59 -0
- package/hooks/gsd-statusline.js +566 -0
- package/hooks/gsd-update-banner.js +138 -0
- package/hooks/gsd-validate-commit.sh +57 -0
- package/hooks/gsd-workflow-guard.js +167 -0
- package/hooks/gsd-worktree-path-guard.js +169 -0
- package/hooks/hooks.json +69 -0
- package/hooks/lib/git-cmd.js +150 -0
- package/hooks/lib/gsd-graphify-rebuild.sh +65 -0
- package/hooks/managed-hooks-registry.cjs +38 -0
- package/package.json +115 -0
- package/scripts/affected-tests-lib.cjs +542 -0
- package/scripts/audit-workflow-script-paths.cjs +73 -0
- package/scripts/base64-scan.sh +351 -0
- package/scripts/build-hooks.js +247 -0
- package/scripts/changeset/README.md +129 -0
- package/scripts/changeset/cli.cjs +590 -0
- package/scripts/changeset/github-release-notes.cjs +199 -0
- package/scripts/changeset/lint.cjs +111 -0
- package/scripts/changeset/new.cjs +137 -0
- package/scripts/changeset/parse.cjs +114 -0
- package/scripts/changeset/render.cjs +34 -0
- package/scripts/changeset/serialize.cjs +130 -0
- package/scripts/check-alias-drift.cjs +114 -0
- package/scripts/check-env.cjs +312 -0
- package/scripts/check-npm-integrity.cjs +215 -0
- package/scripts/ci-guard-runner.cjs +22 -0
- package/scripts/ci-prepare-test-scope.cjs +51 -0
- package/scripts/ci-rebase-check.cjs +86 -0
- package/scripts/ci-test-scope.cjs +431 -0
- package/scripts/command-contract-helpers.cjs +64 -0
- package/scripts/diff-touches-shipped-paths.cjs +155 -0
- package/scripts/fix-slash-commands.cjs +147 -0
- package/scripts/gen-inventory-manifest.cjs +115 -0
- package/scripts/gen-research-agents.cjs +276 -0
- package/scripts/generate-package-identity.cjs +125 -0
- package/scripts/issue-dedupe.cjs +278 -0
- package/scripts/lib/allowlist-ratchet.cjs +136 -0
- package/scripts/lib/cli-exit.cjs +56 -0
- package/scripts/lint-command-contract.cjs +114 -0
- package/scripts/lint-descriptions.cjs +87 -0
- package/scripts/lint-docs-required.cjs +222 -0
- package/scripts/lint-legacy-dir-name.cjs +160 -0
- package/scripts/lint-package-identity-drift.cjs +141 -0
- package/scripts/lint-pr-check-project-dir.cjs +99 -0
- package/scripts/lint-shell-command-projection-drift.cjs +62 -0
- package/scripts/lint-skill-deps.cjs +185 -0
- package/scripts/lint-test-file-count.allowlist.json +135 -0
- package/scripts/lint-test-file-count.cjs +246 -0
- package/scripts/mutation-matrix.cjs +222 -0
- package/scripts/pr-template-policy.cjs +268 -0
- package/scripts/prompt-injection-scan.sh +207 -0
- package/scripts/release-notes/discord-release-summary.cjs +373 -0
- package/scripts/release-notes/format-github-release-notes.cjs +261 -0
- package/scripts/release-tarball-smoke.cjs +629 -0
- package/scripts/research-profiles.cjs +149 -0
- package/scripts/run-affected-tests.cjs +7 -0
- package/scripts/run-cross-platform-tests.cjs +67 -0
- package/scripts/run-tests.cjs +315 -0
- package/scripts/secret-scan-lint.sh +231 -0
- package/scripts/secret-scan.sh +358 -0
- package/scripts/setup-branch-protection.sh +236 -0
- package/scripts/strip-prose-atrefs.cjs +106 -0
- package/scripts/sync-manifest-versions.cjs +119 -0
- package/scripts/sync-rulesets.sh +34 -0
- package/scripts/sync-runtime-launcher.cjs +399 -0
- package/scripts/test-failure-reasons.cjs +34 -0
- package/scripts/verify-npm-publish.cjs +240 -0
- package/scripts/workflow-policy.cjs +450 -0
|
@@ -0,0 +1,278 @@
|
|
|
1
|
+
#!/usr/bin/env node
|
|
2
|
+
'use strict';
|
|
3
|
+
|
|
4
|
+
// ---------------------------------------------------------------------------
|
|
5
|
+
// Constants
|
|
6
|
+
// ---------------------------------------------------------------------------
|
|
7
|
+
|
|
8
|
+
const POSSIBLE_DUPLICATE_LABEL = 'possible-duplicate';
|
|
9
|
+
const HUMAN_REVIEW_LABEL = 'needs-maintainer-review';
|
|
10
|
+
const CHALLENGE_MARKER = '<!-- gsd-dedupe-challenge -->';
|
|
11
|
+
const DEFAULT_WINDOW_HOURS = 24;
|
|
12
|
+
const DEFAULT_THRESHOLD = 0.6;
|
|
13
|
+
const DEFAULT_MAX_CANDIDATES = 5;
|
|
14
|
+
const MIN_TOKEN_LENGTH = 3;
|
|
15
|
+
|
|
16
|
+
const EXEMPT_LABELS = [
|
|
17
|
+
'priority: critical',
|
|
18
|
+
'pinned',
|
|
19
|
+
'confirmed-bug',
|
|
20
|
+
'confirmed',
|
|
21
|
+
'fix-pending',
|
|
22
|
+
'needs-maintainer-review',
|
|
23
|
+
];
|
|
24
|
+
|
|
25
|
+
const STOPWORDS = new Set([
|
|
26
|
+
'the', 'a', 'an', 'and', 'or', 'but', 'if', 'then', 'is', 'are', 'was',
|
|
27
|
+
'be', 'to', 'of', 'in', 'on', 'for', 'with', 'as', 'at', 'by', 'from',
|
|
28
|
+
'this', 'that', 'it', 'its', 'not', 'no', 'when', 'what', 'why', 'how',
|
|
29
|
+
'does', 'do', 'doing', 'did', 'can', 'will', 'would', 'should',
|
|
30
|
+
'i', 'we', 'you', 'your', 'my', 'me',
|
|
31
|
+
'issue', 'bug', 'error', 'problem', 'feature', 'request',
|
|
32
|
+
'help', 'support', 'please', 'question',
|
|
33
|
+
'after', 'before', 'into', 'only', 'then', 'than', 'them', 'they',
|
|
34
|
+
'use', 'used', 'using',
|
|
35
|
+
]);
|
|
36
|
+
|
|
37
|
+
// ---------------------------------------------------------------------------
|
|
38
|
+
// tokenize(title) -> string[]
|
|
39
|
+
//
|
|
40
|
+
// Lowercase the title, replace any non-[a-z0-9] run with a space, split on
|
|
41
|
+
// whitespace, drop tokens shorter than MIN_TOKEN_LENGTH, drop STOPWORDS, and
|
|
42
|
+
// dedupe while preserving stable first-occurrence order.
|
|
43
|
+
//
|
|
44
|
+
// Non-string, null, or empty input returns []. Must not throw on any input.
|
|
45
|
+
// ---------------------------------------------------------------------------
|
|
46
|
+
|
|
47
|
+
function tokenize(title) {
|
|
48
|
+
if (typeof title !== 'string' || !title) return [];
|
|
49
|
+
|
|
50
|
+
const normalized = title.toLowerCase().replace(/[^a-z0-9]+/g, ' ').trim();
|
|
51
|
+
if (!normalized) return [];
|
|
52
|
+
|
|
53
|
+
const seen = new Set();
|
|
54
|
+
const result = [];
|
|
55
|
+
|
|
56
|
+
for (const token of normalized.split(' ')) {
|
|
57
|
+
if (!token || token.length < MIN_TOKEN_LENGTH) continue;
|
|
58
|
+
if (STOPWORDS.has(token)) continue;
|
|
59
|
+
if (seen.has(token)) continue;
|
|
60
|
+
seen.add(token);
|
|
61
|
+
result.push(token);
|
|
62
|
+
}
|
|
63
|
+
|
|
64
|
+
return result;
|
|
65
|
+
}
|
|
66
|
+
|
|
67
|
+
// ---------------------------------------------------------------------------
|
|
68
|
+
// diceSimilarity(aTokens, bTokens) -> number 0..1
|
|
69
|
+
//
|
|
70
|
+
// Sørensen–Dice over the token sets: 2 * |A ∩ B| / (|A| + |B|).
|
|
71
|
+
// Both inputs are treated as sets (duplicates ignored). Empty either side -> 0.
|
|
72
|
+
// Identical sets -> 1.
|
|
73
|
+
// ---------------------------------------------------------------------------
|
|
74
|
+
|
|
75
|
+
function diceSimilarity(aTokens, bTokens) {
|
|
76
|
+
const setA = new Set(aTokens);
|
|
77
|
+
const setB = new Set(bTokens);
|
|
78
|
+
|
|
79
|
+
if (setA.size === 0 || setB.size === 0) return 0;
|
|
80
|
+
|
|
81
|
+
let intersection = 0;
|
|
82
|
+
for (const token of setA) {
|
|
83
|
+
if (setB.has(token)) intersection += 1;
|
|
84
|
+
}
|
|
85
|
+
|
|
86
|
+
return (2 * intersection) / (setA.size + setB.size);
|
|
87
|
+
}
|
|
88
|
+
|
|
89
|
+
// ---------------------------------------------------------------------------
|
|
90
|
+
// scoreCandidates(newTitle, candidates, opts) -> [{number, title, score}]
|
|
91
|
+
//
|
|
92
|
+
// opts: { threshold=DEFAULT_THRESHOLD, limit=DEFAULT_MAX_CANDIDATES, excludeNumber }
|
|
93
|
+
//
|
|
94
|
+
// Tokenizes newTitle once. If no tokens -> []. Filters out null/garbage
|
|
95
|
+
// candidates, those missing a number, and the excluded number. Scores each
|
|
96
|
+
// using diceSimilarity. Keeps score >= threshold. Sorts DESC by score,
|
|
97
|
+
// tie-break ASC by number. Caps to limit.
|
|
98
|
+
// ---------------------------------------------------------------------------
|
|
99
|
+
|
|
100
|
+
function scoreCandidates(newTitle, candidates, opts) {
|
|
101
|
+
const threshold = (opts && opts.threshold != null) ? opts.threshold : DEFAULT_THRESHOLD;
|
|
102
|
+
const limit = (opts && opts.limit != null) ? opts.limit : DEFAULT_MAX_CANDIDATES;
|
|
103
|
+
const excludeNumber = opts && opts.excludeNumber;
|
|
104
|
+
|
|
105
|
+
const newTokens = tokenize(newTitle);
|
|
106
|
+
if (newTokens.length === 0) return [];
|
|
107
|
+
|
|
108
|
+
const scored = [];
|
|
109
|
+
|
|
110
|
+
const safeCandidates = Array.isArray(candidates) ? candidates : [];
|
|
111
|
+
for (const candidate of safeCandidates) {
|
|
112
|
+
if (!candidate || typeof candidate !== 'object') continue;
|
|
113
|
+
if (!(typeof candidate.number === 'number' && Number.isFinite(candidate.number))) continue;
|
|
114
|
+
if (candidate.number === excludeNumber) continue;
|
|
115
|
+
|
|
116
|
+
const score = diceSimilarity(newTokens, tokenize(candidate.title));
|
|
117
|
+
if (score < threshold) continue;
|
|
118
|
+
|
|
119
|
+
scored.push({ number: candidate.number, title: candidate.title, score });
|
|
120
|
+
}
|
|
121
|
+
|
|
122
|
+
scored.sort((a, b) => {
|
|
123
|
+
if (Math.abs(a.score - b.score) > 1e-9) return b.score - a.score;
|
|
124
|
+
return a.number - b.number;
|
|
125
|
+
});
|
|
126
|
+
|
|
127
|
+
return scored.slice(0, limit);
|
|
128
|
+
}
|
|
129
|
+
|
|
130
|
+
// ---------------------------------------------------------------------------
|
|
131
|
+
// renderChallengeComment(candidates, opts) -> string
|
|
132
|
+
//
|
|
133
|
+
// opts: { windowHours=DEFAULT_WINDOW_HOURS }
|
|
134
|
+
//
|
|
135
|
+
// Deterministic. Must start with CHALLENGE_MARKER on its own line. Must list
|
|
136
|
+
// each candidate as a line with #<number>, title, and percentage similarity.
|
|
137
|
+
// Must mention windowHours and the 👎 veto.
|
|
138
|
+
// ---------------------------------------------------------------------------
|
|
139
|
+
|
|
140
|
+
function renderChallengeComment(candidates, opts) {
|
|
141
|
+
const windowHours = (opts && opts.windowHours != null) ? opts.windowHours : DEFAULT_WINDOW_HOURS;
|
|
142
|
+
|
|
143
|
+
const lines = [CHALLENGE_MARKER, ''];
|
|
144
|
+
|
|
145
|
+
lines.push('**Possible duplicate detected.** This issue may already be reported:');
|
|
146
|
+
lines.push('');
|
|
147
|
+
|
|
148
|
+
for (const candidate of candidates) {
|
|
149
|
+
const pct = Math.round(candidate.score * 100);
|
|
150
|
+
lines.push(`- #${candidate.number} — ${candidate.title} (similarity ${pct}%)`);
|
|
151
|
+
}
|
|
152
|
+
|
|
153
|
+
lines.push('');
|
|
154
|
+
lines.push(
|
|
155
|
+
`If this is **not** a duplicate, react with 👎 on this comment to veto and keep the issue open. ` +
|
|
156
|
+
`If no response is received within ${windowHours} hours, this issue may be closed as a duplicate.`,
|
|
157
|
+
);
|
|
158
|
+
|
|
159
|
+
return lines.join('\n');
|
|
160
|
+
}
|
|
161
|
+
|
|
162
|
+
// ---------------------------------------------------------------------------
|
|
163
|
+
// isChallengeComment(body) -> boolean
|
|
164
|
+
//
|
|
165
|
+
// True iff body is a string containing CHALLENGE_MARKER.
|
|
166
|
+
// ---------------------------------------------------------------------------
|
|
167
|
+
|
|
168
|
+
function isChallengeComment(body) {
|
|
169
|
+
if (typeof body !== 'string') return false;
|
|
170
|
+
return body.includes(CHALLENGE_MARKER);
|
|
171
|
+
}
|
|
172
|
+
|
|
173
|
+
// ---------------------------------------------------------------------------
|
|
174
|
+
// hasExemptLabel(labels) -> boolean
|
|
175
|
+
//
|
|
176
|
+
// labels may be an array of strings or array of {name}. True if any name is
|
|
177
|
+
// in EXEMPT_LABELS.
|
|
178
|
+
// ---------------------------------------------------------------------------
|
|
179
|
+
|
|
180
|
+
function hasExemptLabel(labels) {
|
|
181
|
+
if (!Array.isArray(labels)) return false;
|
|
182
|
+
const exemptSet = new Set(EXEMPT_LABELS);
|
|
183
|
+
for (const label of labels) {
|
|
184
|
+
const name = typeof label === 'string' ? label : (label && label.name);
|
|
185
|
+
if (name && exemptSet.has(name)) return true;
|
|
186
|
+
}
|
|
187
|
+
return false;
|
|
188
|
+
}
|
|
189
|
+
|
|
190
|
+
// ---------------------------------------------------------------------------
|
|
191
|
+
// toMs(value) -> number
|
|
192
|
+
//
|
|
193
|
+
// Coerce a Date, ISO string, or ms-number to milliseconds since epoch.
|
|
194
|
+
// ---------------------------------------------------------------------------
|
|
195
|
+
|
|
196
|
+
function toMs(value) {
|
|
197
|
+
if (value instanceof Date) return value.getTime();
|
|
198
|
+
if (typeof value === 'string') return new Date(value).getTime();
|
|
199
|
+
return Number(value);
|
|
200
|
+
}
|
|
201
|
+
|
|
202
|
+
// ---------------------------------------------------------------------------
|
|
203
|
+
// shouldClose(input) -> {close: boolean, reason: string}
|
|
204
|
+
//
|
|
205
|
+
// input: { now, labels, challengeComment, laterUserComments, windowHours=DEFAULT_WINDOW_HOURS }
|
|
206
|
+
//
|
|
207
|
+
// Decision order (returns first match):
|
|
208
|
+
// 1. hasExemptLabel(labels) -> {close:false, reason:'exempt-label'}
|
|
209
|
+
// 2. !challengeComment -> {close:false, reason:'no-challenge-comment'}
|
|
210
|
+
// 3. challengeComment.downvoted -> {close:false, reason:'vetoed'}
|
|
211
|
+
// 4. laterUserComments > 0 -> {close:false, reason:'reporter-responded'}
|
|
212
|
+
// 5. ageHours < windowHours -> {close:false, reason:'within-window'}
|
|
213
|
+
// 6. else -> {close:true, reason:'duplicate-no-response'}
|
|
214
|
+
// ---------------------------------------------------------------------------
|
|
215
|
+
|
|
216
|
+
function shouldClose(input) {
|
|
217
|
+
const {
|
|
218
|
+
now,
|
|
219
|
+
labels = [],
|
|
220
|
+
challengeComment,
|
|
221
|
+
laterUserComments = 0,
|
|
222
|
+
} = input;
|
|
223
|
+
const windowHours = (input.windowHours != null) ? input.windowHours : DEFAULT_WINDOW_HOURS;
|
|
224
|
+
|
|
225
|
+
if (hasExemptLabel(labels)) {
|
|
226
|
+
return { close: false, reason: 'exempt-label' };
|
|
227
|
+
}
|
|
228
|
+
|
|
229
|
+
if (!challengeComment) {
|
|
230
|
+
return { close: false, reason: 'no-challenge-comment' };
|
|
231
|
+
}
|
|
232
|
+
|
|
233
|
+
if (challengeComment.downvoted) {
|
|
234
|
+
return { close: false, reason: 'vetoed' };
|
|
235
|
+
}
|
|
236
|
+
|
|
237
|
+
if (laterUserComments > 0) {
|
|
238
|
+
return { close: false, reason: 'reporter-responded' };
|
|
239
|
+
}
|
|
240
|
+
|
|
241
|
+
const nowMs = toMs(now);
|
|
242
|
+
const createdMs = toMs(challengeComment.createdAt);
|
|
243
|
+
|
|
244
|
+
if (!Number.isFinite(nowMs) || !Number.isFinite(createdMs)) {
|
|
245
|
+
return { close: false, reason: 'invalid-timestamp' };
|
|
246
|
+
}
|
|
247
|
+
|
|
248
|
+
const ageHours = (nowMs - createdMs) / 3600000;
|
|
249
|
+
|
|
250
|
+
if (ageHours < windowHours) {
|
|
251
|
+
return { close: false, reason: 'within-window' };
|
|
252
|
+
}
|
|
253
|
+
|
|
254
|
+
return { close: true, reason: 'duplicate-no-response' };
|
|
255
|
+
}
|
|
256
|
+
|
|
257
|
+
// ---------------------------------------------------------------------------
|
|
258
|
+
// Exports
|
|
259
|
+
// ---------------------------------------------------------------------------
|
|
260
|
+
|
|
261
|
+
module.exports = {
|
|
262
|
+
POSSIBLE_DUPLICATE_LABEL,
|
|
263
|
+
HUMAN_REVIEW_LABEL,
|
|
264
|
+
CHALLENGE_MARKER,
|
|
265
|
+
DEFAULT_WINDOW_HOURS,
|
|
266
|
+
DEFAULT_THRESHOLD,
|
|
267
|
+
DEFAULT_MAX_CANDIDATES,
|
|
268
|
+
MIN_TOKEN_LENGTH,
|
|
269
|
+
EXEMPT_LABELS,
|
|
270
|
+
STOPWORDS,
|
|
271
|
+
tokenize,
|
|
272
|
+
diceSimilarity,
|
|
273
|
+
scoreCandidates,
|
|
274
|
+
renderChallengeComment,
|
|
275
|
+
isChallengeComment,
|
|
276
|
+
hasExemptLabel,
|
|
277
|
+
shouldClose,
|
|
278
|
+
};
|
|
@@ -0,0 +1,136 @@
|
|
|
1
|
+
'use strict';
|
|
2
|
+
|
|
3
|
+
/**
|
|
4
|
+
* @file allowlist-ratchet.cjs
|
|
5
|
+
*
|
|
6
|
+
* Reusable "better than a count ratchet" primitives for CI guards.
|
|
7
|
+
*
|
|
8
|
+
* ## Motivation (issue #597)
|
|
9
|
+
*
|
|
10
|
+
* A count ratchet (`assert(offenders.length <= N)`) has a masking blind spot:
|
|
11
|
+
* fixing one offender and introducing a new one keeps the count constant, so a
|
|
12
|
+
* novel defect slips through green. These helpers enforce on IDENTITY instead,
|
|
13
|
+
* making every individual offender visible and requiring monotonic progress
|
|
14
|
+
* toward zero.
|
|
15
|
+
*
|
|
16
|
+
* ## Design
|
|
17
|
+
*
|
|
18
|
+
* Both functions are pure (no I/O, no global state). The `fail` callback is
|
|
19
|
+
* injected by the caller so the same logic can be used with `node:assert.fail`,
|
|
20
|
+
* a custom throw, or a message-collector in unit tests.
|
|
21
|
+
*/
|
|
22
|
+
|
|
23
|
+
/**
|
|
24
|
+
* Assert that `current` offenders are all within the known allowlist, and that
|
|
25
|
+
* every entry in the allowlist still offends (forcing the allowlist to shrink
|
|
26
|
+
* as defects are fixed).
|
|
27
|
+
*
|
|
28
|
+
* Fails when:
|
|
29
|
+
* - Any id in `current` is NOT in `known` → novel offender introduced.
|
|
30
|
+
* - Any id in `known` is NOT in `current` → stale allowlist entry must be
|
|
31
|
+
* pruned so the guard ratchets toward zero (the ratchet-DOWN direction).
|
|
32
|
+
*
|
|
33
|
+
* ## Masking blind spot this prevents (issue #597)
|
|
34
|
+
*
|
|
35
|
+
* A count ratchet (`assert(count <= N)`) allows one offender to be silently
|
|
36
|
+
* replaced by another while the count stays at N. By asserting on identity
|
|
37
|
+
* instead, every new offender is caught by name, and every fixed offender
|
|
38
|
+
* forces the allowlist to shrink.
|
|
39
|
+
*
|
|
40
|
+
* @param {object} opts
|
|
41
|
+
* @param {string} opts.label - Human-readable name for the guard (used
|
|
42
|
+
* in failure messages).
|
|
43
|
+
* @param {Iterable<string>} opts.current - The offending ids found in the
|
|
44
|
+
* current run.
|
|
45
|
+
* @param {Iterable<string>} opts.known - The allowlisted ids (baseline).
|
|
46
|
+
* @param {function(string): void} opts.fail - Callback invoked with a
|
|
47
|
+
* descriptive message on any violation.
|
|
48
|
+
* Pass `require('node:assert').fail`, a
|
|
49
|
+
* custom thrower, or a collector. The
|
|
50
|
+
* function is NOT imported here so callers
|
|
51
|
+
* control the failure mode.
|
|
52
|
+
* @param {string} [opts.pruneHint] - Optional hint appended to the stale-
|
|
53
|
+
* entry failure message (e.g. the name of
|
|
54
|
+
* the allowlist file to edit).
|
|
55
|
+
* @returns {{ novel: string[], stale: string[] }} Sorted arrays of novel ids
|
|
56
|
+
* (in current but not known) and stale ids (in known but not current).
|
|
57
|
+
*/
|
|
58
|
+
function assertWithinAllowlist({ label, current, known, fail, pruneHint }) {
|
|
59
|
+
const currentSet = new Set(current);
|
|
60
|
+
const knownSet = new Set(known);
|
|
61
|
+
|
|
62
|
+
const novel = [...currentSet].filter((id) => !knownSet.has(id)).sort();
|
|
63
|
+
const stale = [...knownSet].filter((id) => !currentSet.has(id)).sort();
|
|
64
|
+
|
|
65
|
+
if (novel.length > 0) {
|
|
66
|
+
const list = novel.map((id) => ` - ${id}`).join('\n');
|
|
67
|
+
fail(
|
|
68
|
+
`[${label}] ${novel.length} NEW offender(s) introduced — fix at the source; do not just add to the allowlist.\n${list}`
|
|
69
|
+
);
|
|
70
|
+
}
|
|
71
|
+
|
|
72
|
+
if (stale.length > 0) {
|
|
73
|
+
const list = stale.map((id) => ` - ${id}`).join('\n');
|
|
74
|
+
const hint = pruneHint ? `\n(${pruneHint})` : '';
|
|
75
|
+
fail(
|
|
76
|
+
`[${label}] ${stale.length} allowlisted id(s) no longer offend and MUST be pruned so the guard ratchets toward zero.${hint}\n${list}`
|
|
77
|
+
);
|
|
78
|
+
}
|
|
79
|
+
|
|
80
|
+
return { novel, stale };
|
|
81
|
+
}
|
|
82
|
+
|
|
83
|
+
/**
|
|
84
|
+
* Assert that an artifact's measured maximum stays within a declared ceiling,
|
|
85
|
+
* and that the ceiling itself does not creep above the high-water mark (budgets
|
|
86
|
+
* may only decrease, not increase over time).
|
|
87
|
+
*
|
|
88
|
+
* Fails when:
|
|
89
|
+
* - `actualMax > ceiling` → regression: artifact exceeds budget.
|
|
90
|
+
* - `ceiling - actualMax > grace` → ceiling sits too far above the measured
|
|
91
|
+
* value; tighten it toward `actualMax`.
|
|
92
|
+
*
|
|
93
|
+
* ## Masking blind spot this prevents (issue #597)
|
|
94
|
+
*
|
|
95
|
+
* A plain `assert(size <= ceiling)` with a ceiling set generously high allows
|
|
96
|
+
* the artifact to grow unchecked as long as it stays under the ceiling. The
|
|
97
|
+
* `grace` band forces the ceiling to stay close to the high-water mark,
|
|
98
|
+
* ensuring that any upward creep is immediately visible.
|
|
99
|
+
*
|
|
100
|
+
* @param {object} opts
|
|
101
|
+
* @param {string} opts.label - Human-readable name for the guard (used
|
|
102
|
+
* in failure messages).
|
|
103
|
+
* @param {number} opts.actualMax - The measured value (e.g. bundle size in
|
|
104
|
+
* bytes, line count).
|
|
105
|
+
* @param {number} opts.ceiling - The declared budget ceiling.
|
|
106
|
+
* @param {number} opts.grace - Maximum allowed slack (`ceiling -
|
|
107
|
+
* actualMax`) before the ceiling is
|
|
108
|
+
* considered too loose.
|
|
109
|
+
* @param {function(string): void} opts.fail - Callback invoked with a
|
|
110
|
+
* descriptive message on any violation.
|
|
111
|
+
* @returns {{ ok: boolean, slack: number }} Whether both checks passed and the
|
|
112
|
+
* current slack value.
|
|
113
|
+
*/
|
|
114
|
+
function assertTightCeiling({ label, actualMax, ceiling, grace, fail }) {
|
|
115
|
+
const slack = ceiling - actualMax;
|
|
116
|
+
let ok = true;
|
|
117
|
+
|
|
118
|
+
if (actualMax > ceiling) {
|
|
119
|
+
ok = false;
|
|
120
|
+
fail(
|
|
121
|
+
`[${label}] Regression: artifact value ${actualMax} exceeds budget ceiling ${ceiling}. ` +
|
|
122
|
+
`Raise the ceiling to at most ${actualMax} only if the increase is justified.`
|
|
123
|
+
);
|
|
124
|
+
} else if (slack > grace) {
|
|
125
|
+
ok = false;
|
|
126
|
+
fail(
|
|
127
|
+
`[${label}] Ceiling ${ceiling} sits too far above the high-water mark ${actualMax} ` +
|
|
128
|
+
`(slack ${slack} > grace ${grace}). Tighten the ceiling toward ${actualMax}. ` +
|
|
129
|
+
`Budgets may only decrease.`
|
|
130
|
+
);
|
|
131
|
+
}
|
|
132
|
+
|
|
133
|
+
return { ok, slack };
|
|
134
|
+
}
|
|
135
|
+
|
|
136
|
+
module.exports = { assertWithinAllowlist, assertTightCeiling };
|
|
@@ -0,0 +1,56 @@
|
|
|
1
|
+
'use strict';
|
|
2
|
+
|
|
3
|
+
/**
|
|
4
|
+
* Error that carries a process exit code. CLI logic throws this instead of
|
|
5
|
+
* calling process.exit() (banned by n/no-process-exit); runMain() translates it
|
|
6
|
+
* into process.exitCode at the entrypoint.
|
|
7
|
+
*
|
|
8
|
+
* @param {number} code exit code (default 1)
|
|
9
|
+
* @param {string} [message] optional human message; when set and code != 0 it is
|
|
10
|
+
* written to stderr by runMain before the process exits.
|
|
11
|
+
*/
|
|
12
|
+
class ExitError extends Error {
|
|
13
|
+
constructor(code = 1, message) {
|
|
14
|
+
super(message === undefined ? `process exit ${code}` : message);
|
|
15
|
+
this.name = 'ExitError';
|
|
16
|
+
this.code = code;
|
|
17
|
+
// Whether runMain should print this.message to stderr (only when a real
|
|
18
|
+
// message was provided, not the synthetic default).
|
|
19
|
+
this.hasUserMessage = message !== undefined;
|
|
20
|
+
}
|
|
21
|
+
}
|
|
22
|
+
|
|
23
|
+
/**
|
|
24
|
+
* Run a CLI main function and translate its outcome into process.exitCode
|
|
25
|
+
* (never process.exit(), so n/no-process-exit stays satisfied). Supports sync or
|
|
26
|
+
* async main.
|
|
27
|
+
* - main returns a number -> process.exitCode = that number
|
|
28
|
+
* - main throws/rejects ExitError -> process.exitCode = err.code, and if
|
|
29
|
+
* err.hasUserMessage && err.code !== 0, err.message is written to stderr
|
|
30
|
+
* - main throws/rejects anything else -> the stack is written to stderr and
|
|
31
|
+
* process.exitCode = 1
|
|
32
|
+
* Letting the event loop drain (vs process.exit) means buffered stdout/stderr is
|
|
33
|
+
* flushed and process.on('exit') cleanup handlers still fire.
|
|
34
|
+
*
|
|
35
|
+
* @param {() => (number|void|Promise<number|void>)} main
|
|
36
|
+
*/
|
|
37
|
+
function runMain(main) {
|
|
38
|
+
Promise.resolve()
|
|
39
|
+
.then(() => main())
|
|
40
|
+
.then((code) => {
|
|
41
|
+
if (typeof code === 'number') process.exitCode = code;
|
|
42
|
+
})
|
|
43
|
+
.catch((err) => {
|
|
44
|
+
if (err instanceof ExitError) {
|
|
45
|
+
if (err.hasUserMessage && err.code !== 0) {
|
|
46
|
+
process.stderr.write(`${err.message}\n`);
|
|
47
|
+
}
|
|
48
|
+
process.exitCode = err.code;
|
|
49
|
+
return;
|
|
50
|
+
}
|
|
51
|
+
process.stderr.write(`${err && err.stack ? err.stack : String(err)}\n`);
|
|
52
|
+
process.exitCode = 1;
|
|
53
|
+
});
|
|
54
|
+
}
|
|
55
|
+
|
|
56
|
+
module.exports = { ExitError, runMain };
|
|
@@ -0,0 +1,114 @@
|
|
|
1
|
+
#!/usr/bin/env node
|
|
2
|
+
/**
|
|
3
|
+
* lint-command-contract.cjs (ADR-0002)
|
|
4
|
+
*
|
|
5
|
+
* Enforces the commands/gsd/*.md contract across all 65 command files:
|
|
6
|
+
*
|
|
7
|
+
* 1. name: present, non-empty, matches gsd: or gsd- prefix
|
|
8
|
+
* 2. description: present, non-empty
|
|
9
|
+
* 3. allowed-tools: block present, non-empty, all entries from CANONICAL_TOOLS
|
|
10
|
+
* 4. execution_context @-refs: every @-reference resolves to an existing file on disk
|
|
11
|
+
* 5. execution_context @-refs: each appears on its own line (no trailing prose)
|
|
12
|
+
*
|
|
13
|
+
* Exit 0 = clean. Exit 1 = violations (with diagnostics).
|
|
14
|
+
*/
|
|
15
|
+
|
|
16
|
+
'use strict';
|
|
17
|
+
|
|
18
|
+
const fs = require('fs');
|
|
19
|
+
const path = require('path');
|
|
20
|
+
|
|
21
|
+
const ROOT = path.join(__dirname, '..');
|
|
22
|
+
const COMMANDS_DIR = path.join(ROOT, 'commands', 'gsd');
|
|
23
|
+
const GSD_ROOT = path.join(ROOT, 'gsd-core');
|
|
24
|
+
|
|
25
|
+
const {
|
|
26
|
+
CANONICAL_TOOLS,
|
|
27
|
+
parseFrontmatter,
|
|
28
|
+
executionContextRefs: extractExecutionContextRefs,
|
|
29
|
+
} = require('./command-contract-helpers.cjs');
|
|
30
|
+
|
|
31
|
+
const { runMain } = require('./lib/cli-exit.cjs');
|
|
32
|
+
|
|
33
|
+
// ─── check one file ───────────────────────────────────────────────────────────
|
|
34
|
+
|
|
35
|
+
function check(filePath) {
|
|
36
|
+
const content = fs.readFileSync(filePath, 'utf-8');
|
|
37
|
+
const rel = path.relative(ROOT, filePath);
|
|
38
|
+
const fm = parseFrontmatter(content);
|
|
39
|
+
const violations = [];
|
|
40
|
+
|
|
41
|
+
// 1. name: present + gsd: / gsd- prefix
|
|
42
|
+
if (!fm.name || !fm.name.trim()) {
|
|
43
|
+
violations.push('name: field missing or empty');
|
|
44
|
+
} else if (!/^gsd[:-]/.test(fm.name.trim())) {
|
|
45
|
+
violations.push(`name: must start with "gsd:" or "gsd-", got "${fm.name.trim()}"`);
|
|
46
|
+
}
|
|
47
|
+
|
|
48
|
+
// 2. description: present + non-empty
|
|
49
|
+
if (!fm.description || !fm.description.trim()) {
|
|
50
|
+
violations.push('description: field missing or empty');
|
|
51
|
+
}
|
|
52
|
+
|
|
53
|
+
// 3. allowed-tools: present + non-empty + all entries canonical
|
|
54
|
+
if (!fm['allowed-tools'] || !fm['allowed-tools'].trim()) {
|
|
55
|
+
violations.push('allowed-tools: block missing or empty');
|
|
56
|
+
} else {
|
|
57
|
+
const tools = fm['allowed-tools'].split('\n').map(t => t.trim()).filter(Boolean);
|
|
58
|
+
for (const tool of tools) {
|
|
59
|
+
const valid =
|
|
60
|
+
CANONICAL_TOOLS.has(tool) ||
|
|
61
|
+
(tool.startsWith('mcp__context7__') && CANONICAL_TOOLS.has('mcp__context7__*'));
|
|
62
|
+
if (!valid) violations.push(`allowed-tools: unknown tool "${tool}"`);
|
|
63
|
+
}
|
|
64
|
+
}
|
|
65
|
+
|
|
66
|
+
// 4+5. execution_context @-refs resolve + no trailing prose
|
|
67
|
+
const refs = extractExecutionContextRefs(content);
|
|
68
|
+
for (const { token, normalized, trailingProse } of refs) {
|
|
69
|
+
const absPath = path.join(GSD_ROOT, normalized);
|
|
70
|
+
if (!fs.existsSync(absPath)) {
|
|
71
|
+
violations.push(`execution_context: @-ref "${normalized}" does not exist on disk`);
|
|
72
|
+
}
|
|
73
|
+
if (trailingProse) {
|
|
74
|
+
violations.push(`execution_context: @-ref "${token}" has trailing prose on the same line`);
|
|
75
|
+
}
|
|
76
|
+
}
|
|
77
|
+
|
|
78
|
+
if (violations.length === 0) return null;
|
|
79
|
+
return { file: rel, violations };
|
|
80
|
+
}
|
|
81
|
+
|
|
82
|
+
// ─── run ─────────────────────────────────────────────────────────────────────
|
|
83
|
+
|
|
84
|
+
function main() {
|
|
85
|
+
const commandFiles = fs
|
|
86
|
+
.readdirSync(COMMANDS_DIR)
|
|
87
|
+
.filter(f => f.endsWith('.md'))
|
|
88
|
+
.map(f => path.join(COMMANDS_DIR, f));
|
|
89
|
+
|
|
90
|
+
const results = commandFiles.map(check).filter(Boolean);
|
|
91
|
+
|
|
92
|
+
if (results.length === 0) {
|
|
93
|
+
console.log(
|
|
94
|
+
`ok lint-command-contract: ${commandFiles.length} command files checked, 0 violations`,
|
|
95
|
+
);
|
|
96
|
+
return 0;
|
|
97
|
+
}
|
|
98
|
+
|
|
99
|
+
const total = results.reduce((n, r) => n + r.violations.length, 0);
|
|
100
|
+
process.stderr.write(
|
|
101
|
+
`\nERROR lint-command-contract: ${total} violation(s) across ${results.length} file(s)\n\n`,
|
|
102
|
+
);
|
|
103
|
+
for (const r of results) {
|
|
104
|
+
process.stderr.write(` ${r.file}\n`);
|
|
105
|
+
for (const v of r.violations) {
|
|
106
|
+
process.stderr.write(` - ${v}\n`);
|
|
107
|
+
}
|
|
108
|
+
process.stderr.write('\n');
|
|
109
|
+
}
|
|
110
|
+
process.stderr.write('See docs/adr/0002-command-contract-validation-module.md for the contract spec.\n\n');
|
|
111
|
+
return 1;
|
|
112
|
+
}
|
|
113
|
+
|
|
114
|
+
runMain(main);
|
|
@@ -0,0 +1,87 @@
|
|
|
1
|
+
#!/usr/bin/env node
|
|
2
|
+
/**
|
|
3
|
+
* lint-descriptions.cjs
|
|
4
|
+
*
|
|
5
|
+
* Enforces the 100-char description budget for commands/gsd/*.md files.
|
|
6
|
+
*
|
|
7
|
+
* Usage:
|
|
8
|
+
* node scripts/lint-descriptions.cjs [file.md ...]
|
|
9
|
+
*
|
|
10
|
+
* If no args are given, scans commands/gsd/ automatically.
|
|
11
|
+
* Exits 1 if any description exceeds 100 chars; exits 0 if all pass.
|
|
12
|
+
*/
|
|
13
|
+
|
|
14
|
+
'use strict';
|
|
15
|
+
|
|
16
|
+
const fs = require('fs');
|
|
17
|
+
const path = require('path');
|
|
18
|
+
const { ExitError, runMain } = require('./lib/cli-exit.cjs');
|
|
19
|
+
|
|
20
|
+
const MAX_LENGTH = 100;
|
|
21
|
+
const COMMANDS_DIR = path.join(__dirname, '..', 'commands', 'gsd');
|
|
22
|
+
|
|
23
|
+
/**
|
|
24
|
+
* Parse the description field from frontmatter in a .md file.
|
|
25
|
+
* Returns null if no description is found.
|
|
26
|
+
*/
|
|
27
|
+
function parseDescription(content) {
|
|
28
|
+
const fmMatch = content.match(/^---\r?\n([\s\S]*?)\r?\n---/);
|
|
29
|
+
if (!fmMatch) return null;
|
|
30
|
+
const fm = fmMatch[1];
|
|
31
|
+
|
|
32
|
+
const quoted = fm.match(/^description:\s+"((?:[^"\\]|\\.)*)"\s*$/m);
|
|
33
|
+
if (quoted) return quoted[1];
|
|
34
|
+
|
|
35
|
+
const plain = fm.match(/^description:\s+(.+)$/m);
|
|
36
|
+
if (plain) return plain[1].trim();
|
|
37
|
+
|
|
38
|
+
return null;
|
|
39
|
+
}
|
|
40
|
+
|
|
41
|
+
function getFiles() {
|
|
42
|
+
if (process.argv.length > 2) {
|
|
43
|
+
return process.argv.slice(2);
|
|
44
|
+
}
|
|
45
|
+
return fs.readdirSync(COMMANDS_DIR)
|
|
46
|
+
.filter(f => f.endsWith('.md'))
|
|
47
|
+
.map(f => path.join(COMMANDS_DIR, f));
|
|
48
|
+
}
|
|
49
|
+
|
|
50
|
+
function main() {
|
|
51
|
+
const files = getFiles();
|
|
52
|
+
const violations = [];
|
|
53
|
+
|
|
54
|
+
for (const filePath of files) {
|
|
55
|
+
let content;
|
|
56
|
+
try {
|
|
57
|
+
content = fs.readFileSync(filePath, 'utf-8');
|
|
58
|
+
} catch (err) {
|
|
59
|
+
throw new ExitError(1, `ERROR: Cannot read file: ${filePath}\n ${err.message}`);
|
|
60
|
+
}
|
|
61
|
+
|
|
62
|
+
const description = parseDescription(content);
|
|
63
|
+
if (description === null) continue;
|
|
64
|
+
|
|
65
|
+
if (description.length > MAX_LENGTH) {
|
|
66
|
+
violations.push({ filePath, length: description.length, description });
|
|
67
|
+
}
|
|
68
|
+
}
|
|
69
|
+
|
|
70
|
+
if (violations.length === 0) {
|
|
71
|
+
const checked = files.length;
|
|
72
|
+
process.stdout.write(`ok lint-descriptions: ${checked} file(s) checked, 0 violations\n`);
|
|
73
|
+
return 0;
|
|
74
|
+
}
|
|
75
|
+
|
|
76
|
+
process.stderr.write(`\nERROR lint-descriptions: ${violations.length} violation(s) found\n\n`);
|
|
77
|
+
for (const v of violations) {
|
|
78
|
+
const preview = v.description.length > 120 ? v.description.slice(0, 117) + '...' : v.description;
|
|
79
|
+
process.stderr.write(` ${v.filePath}\n`);
|
|
80
|
+
process.stderr.write(` Length : ${v.length} (max ${MAX_LENGTH})\n`);
|
|
81
|
+
process.stderr.write(` Desc : ${preview}\n\n`);
|
|
82
|
+
}
|
|
83
|
+
process.stderr.write(`Trim descriptions to <= ${MAX_LENGTH} chars. Flag docs belong in argument-hint:.\n\n`);
|
|
84
|
+
return 1;
|
|
85
|
+
}
|
|
86
|
+
|
|
87
|
+
runMain(main);
|