metaharness 0.1.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/LICENSE +21 -0
- package/README.md +73 -0
- package/dist/analyze-repo.d.ts +74 -0
- package/dist/analyze-repo.d.ts.map +1 -0
- package/dist/analyze-repo.js +329 -0
- package/dist/analyze-repo.js.map +1 -0
- package/dist/audit-cmd.d.ts +6 -0
- package/dist/audit-cmd.d.ts.map +1 -0
- package/dist/audit-cmd.js +139 -0
- package/dist/audit-cmd.js.map +1 -0
- package/dist/bin.d.ts +3 -0
- package/dist/bin.d.ts.map +1 -0
- package/dist/bin.js +8 -0
- package/dist/bin.js.map +1 -0
- package/dist/compare-cmd.d.ts +56 -0
- package/dist/compare-cmd.d.ts.map +1 -0
- package/dist/compare-cmd.js +173 -0
- package/dist/compare-cmd.js.map +1 -0
- package/dist/completions-cmd.d.ts +6 -0
- package/dist/completions-cmd.d.ts.map +1 -0
- package/dist/completions-cmd.js +171 -0
- package/dist/completions-cmd.js.map +1 -0
- package/dist/diag.d.ts +76 -0
- package/dist/diag.d.ts.map +1 -0
- package/dist/diag.js +349 -0
- package/dist/diag.js.map +1 -0
- package/dist/eject.d.ts +33 -0
- package/dist/eject.d.ts.map +1 -0
- package/dist/eject.js +147 -0
- package/dist/eject.js.map +1 -0
- package/dist/export-config.d.ts +25 -0
- package/dist/export-config.d.ts.map +1 -0
- package/dist/export-config.js +134 -0
- package/dist/export-config.js.map +1 -0
- package/dist/external-template.d.ts +19 -0
- package/dist/external-template.d.ts.map +1 -0
- package/dist/external-template.js +37 -0
- package/dist/external-template.js.map +1 -0
- package/dist/federate.d.ts +37 -0
- package/dist/federate.d.ts.map +1 -0
- package/dist/federate.js +167 -0
- package/dist/federate.js.map +1 -0
- package/dist/genome-scorers.d.ts +33 -0
- package/dist/genome-scorers.d.ts.map +1 -0
- package/dist/genome-scorers.js +115 -0
- package/dist/genome-scorers.js.map +1 -0
- package/dist/genome.d.ts +28 -0
- package/dist/genome.d.ts.map +1 -0
- package/dist/genome.js +219 -0
- package/dist/genome.js.map +1 -0
- package/dist/harness-bin.d.ts +3 -0
- package/dist/harness-bin.d.ts.map +1 -0
- package/dist/harness-bin.js +17 -0
- package/dist/harness-bin.js.map +1 -0
- package/dist/index.d.ts +79 -0
- package/dist/index.d.ts.map +1 -0
- package/dist/index.js +409 -0
- package/dist/index.js.map +1 -0
- package/dist/manifest.d.ts +55 -0
- package/dist/manifest.d.ts.map +1 -0
- package/dist/manifest.js +62 -0
- package/dist/manifest.js.map +1 -0
- package/dist/mcp-cmd.d.ts +19 -0
- package/dist/mcp-cmd.d.ts.map +1 -0
- package/dist/mcp-cmd.js +179 -0
- package/dist/mcp-cmd.js.map +1 -0
- package/dist/mcp-scan.d.ts +25 -0
- package/dist/mcp-scan.d.ts.map +1 -0
- package/dist/mcp-scan.js +132 -0
- package/dist/mcp-scan.js.map +1 -0
- package/dist/oia-manifest.d.ts +97 -0
- package/dist/oia-manifest.d.ts.map +1 -0
- package/dist/oia-manifest.js +241 -0
- package/dist/oia-manifest.js.map +1 -0
- package/dist/publish-cmd.d.ts +9 -0
- package/dist/publish-cmd.d.ts.map +1 -0
- package/dist/publish-cmd.js +56 -0
- package/dist/publish-cmd.js.map +1 -0
- package/dist/publish.d.ts +55 -0
- package/dist/publish.d.ts.map +1 -0
- package/dist/publish.js +104 -0
- package/dist/publish.js.map +1 -0
- package/dist/registry.d.ts +60 -0
- package/dist/registry.d.ts.map +1 -0
- package/dist/registry.js +67 -0
- package/dist/registry.js.map +1 -0
- package/dist/rename.d.ts +26 -0
- package/dist/rename.d.ts.map +1 -0
- package/dist/rename.js +86 -0
- package/dist/rename.js.map +1 -0
- package/dist/renderer.d.ts +39 -0
- package/dist/renderer.d.ts.map +1 -0
- package/dist/renderer.js +75 -0
- package/dist/renderer.js.map +1 -0
- package/dist/sbom-cmd.d.ts +6 -0
- package/dist/sbom-cmd.d.ts.map +1 -0
- package/dist/sbom-cmd.js +125 -0
- package/dist/sbom-cmd.js.map +1 -0
- package/dist/score.d.ts +32 -0
- package/dist/score.d.ts.map +1 -0
- package/dist/score.js +381 -0
- package/dist/score.js.map +1 -0
- package/dist/secrets.d.ts +27 -0
- package/dist/secrets.d.ts.map +1 -0
- package/dist/secrets.js +228 -0
- package/dist/secrets.js.map +1 -0
- package/dist/subcommands.d.ts +32 -0
- package/dist/subcommands.d.ts.map +1 -0
- package/dist/subcommands.js +335 -0
- package/dist/subcommands.js.map +1 -0
- package/dist/tarball.d.ts +14 -0
- package/dist/tarball.d.ts.map +1 -0
- package/dist/tarball.js +121 -0
- package/dist/tarball.js.map +1 -0
- package/dist/threat-model.d.ts +32 -0
- package/dist/threat-model.d.ts.map +1 -0
- package/dist/threat-model.js +236 -0
- package/dist/threat-model.js.map +1 -0
- package/dist/upgrade-cmd.d.ts +9 -0
- package/dist/upgrade-cmd.d.ts.map +1 -0
- package/dist/upgrade-cmd.js +90 -0
- package/dist/upgrade-cmd.js.map +1 -0
- package/dist/upgrade.d.ts +37 -0
- package/dist/upgrade.d.ts.map +1 -0
- package/dist/upgrade.js +124 -0
- package/dist/upgrade.js.map +1 -0
- package/dist/validate.d.ts +7 -0
- package/dist/validate.d.ts.map +1 -0
- package/dist/validate.js +240 -0
- package/dist/validate.js.map +1 -0
- package/dist/walker.d.ts +24 -0
- package/dist/walker.d.ts.map +1 -0
- package/dist/walker.js +76 -0
- package/dist/walker.js.map +1 -0
- package/dist/witness-client.d.ts +40 -0
- package/dist/witness-client.d.ts.map +1 -0
- package/dist/witness-client.js +99 -0
- package/dist/witness-client.js.map +1 -0
- package/dist/wizard.d.ts +37 -0
- package/dist/wizard.d.ts.map +1 -0
- package/dist/wizard.js +137 -0
- package/dist/wizard.js.map +1 -0
- package/dist/writer.d.ts +13 -0
- package/dist/writer.d.ts.map +1 -0
- package/dist/writer.js +43 -0
- package/dist/writer.js.map +1 -0
- package/package.json +114 -0
- package/templates/catalog.def.mjs +542 -0
- package/templates/catalog.json +1274 -0
- package/templates/minimal/.claude/settings.json.tmpl +18 -0
- package/templates/minimal/CLAUDE.md.tmpl +23 -0
- package/templates/minimal/README.md.tmpl +20 -0
- package/templates/minimal/manifest.json +16 -0
- package/templates/minimal/package.json.tmpl +30 -0
- package/templates/minimal/src/init.ts.tmpl +21 -0
- package/templates/vertical_advertising/.claude/commands/doctor.md.tmpl +12 -0
- package/templates/vertical_advertising/.claude/settings.json.tmpl +33 -0
- package/templates/vertical_advertising/.claude/skills/media-plan/SKILL.md.tmpl +13 -0
- package/templates/vertical_advertising/CLAUDE.md.tmpl +30 -0
- package/templates/vertical_advertising/README.md.tmpl +29 -0
- package/templates/vertical_advertising/manifest.json +81 -0
- package/templates/vertical_advertising/package.json.tmpl +30 -0
- package/templates/vertical_advertising/src/agents/copywriter.ts.tmpl +7 -0
- package/templates/vertical_advertising/src/agents/media-planner.ts.tmpl +7 -0
- package/templates/vertical_advertising/src/agents/performance-analyst.ts.tmpl +7 -0
- package/templates/vertical_advertising/src/init.ts.tmpl +21 -0
- package/templates/vertical_agentics/.claude/commands/doctor.md.tmpl +12 -0
- package/templates/vertical_agentics/.claude/settings.json.tmpl +33 -0
- package/templates/vertical_agentics/.claude/skills/memory-inspect/SKILL.md.tmpl +14 -0
- package/templates/vertical_agentics/.claude/skills/run-swarm/SKILL.md.tmpl +14 -0
- package/templates/vertical_agentics/CLAUDE.md.tmpl +32 -0
- package/templates/vertical_agentics/README.md.tmpl +30 -0
- package/templates/vertical_agentics/manifest.json +91 -0
- package/templates/vertical_agentics/package.json.tmpl +30 -0
- package/templates/vertical_agentics/src/agents/critic.ts.tmpl +7 -0
- package/templates/vertical_agentics/src/agents/orchestrator.ts.tmpl +7 -0
- package/templates/vertical_agentics/src/agents/planner.ts.tmpl +7 -0
- package/templates/vertical_agentics/src/agents/worker.ts.tmpl +7 -0
- package/templates/vertical_agentics/src/init.ts.tmpl +21 -0
- package/templates/vertical_ai/.claude/commands/doctor.md.tmpl +12 -0
- package/templates/vertical_ai/.claude/settings.json.tmpl +35 -0
- package/templates/vertical_ai/.claude/skills/eval-report/SKILL.md.tmpl +13 -0
- package/templates/vertical_ai/CLAUDE.md.tmpl +31 -0
- package/templates/vertical_ai/README.md.tmpl +30 -0
- package/templates/vertical_ai/manifest.json +86 -0
- package/templates/vertical_ai/package.json.tmpl +30 -0
- package/templates/vertical_ai/src/agents/data-curator.ts.tmpl +7 -0
- package/templates/vertical_ai/src/agents/deployer.ts.tmpl +7 -0
- package/templates/vertical_ai/src/agents/evaluator.ts.tmpl +7 -0
- package/templates/vertical_ai/src/agents/trainer.ts.tmpl +7 -0
- package/templates/vertical_ai/src/init.ts.tmpl +21 -0
- package/templates/vertical_business/.claude/commands/doctor.md.tmpl +12 -0
- package/templates/vertical_business/.claude/settings.json.tmpl +33 -0
- package/templates/vertical_business/.claude/skills/quarterly-plan/SKILL.md.tmpl +13 -0
- package/templates/vertical_business/CLAUDE.md.tmpl +30 -0
- package/templates/vertical_business/README.md.tmpl +29 -0
- package/templates/vertical_business/manifest.json +81 -0
- package/templates/vertical_business/package.json.tmpl +30 -0
- package/templates/vertical_business/src/agents/analyst.ts.tmpl +7 -0
- package/templates/vertical_business/src/agents/ops-coordinator.ts.tmpl +7 -0
- package/templates/vertical_business/src/agents/strategist.ts.tmpl +7 -0
- package/templates/vertical_business/src/init.ts.tmpl +21 -0
- package/templates/vertical_coding/.claude/commands/doctor.md.tmpl +12 -0
- package/templates/vertical_coding/.claude/commands/review-diff.md.tmpl +10 -0
- package/templates/vertical_coding/.claude/settings.json.tmpl +40 -0
- package/templates/vertical_coding/.claude/skills/plan-change/SKILL.md.tmpl +15 -0
- package/templates/vertical_coding/CLAUDE.md.tmpl +32 -0
- package/templates/vertical_coding/README.md.tmpl +30 -0
- package/templates/vertical_coding/manifest.json +91 -0
- package/templates/vertical_coding/package.json.tmpl +30 -0
- package/templates/vertical_coding/src/agents/architect.ts.tmpl +7 -0
- package/templates/vertical_coding/src/agents/implementer.ts.tmpl +7 -0
- package/templates/vertical_coding/src/agents/reviewer.ts.tmpl +7 -0
- package/templates/vertical_coding/src/agents/test-writer.ts.tmpl +7 -0
- package/templates/vertical_coding/src/init.ts.tmpl +21 -0
- package/templates/vertical_crm/.claude/commands/doctor.md.tmpl +12 -0
- package/templates/vertical_crm/.claude/settings.json.tmpl +33 -0
- package/templates/vertical_crm/.claude/skills/memory-inspect/SKILL.md.tmpl +14 -0
- package/templates/vertical_crm/CLAUDE.md.tmpl +30 -0
- package/templates/vertical_crm/README.md.tmpl +29 -0
- package/templates/vertical_crm/manifest.json +81 -0
- package/templates/vertical_crm/package.json.tmpl +30 -0
- package/templates/vertical_crm/src/agents/account-manager.ts.tmpl +7 -0
- package/templates/vertical_crm/src/agents/churn-watcher.ts.tmpl +7 -0
- package/templates/vertical_crm/src/agents/lead-qualifier.ts.tmpl +7 -0
- package/templates/vertical_crm/src/init.ts.tmpl +21 -0
- package/templates/vertical_devops/.claude/settings.json.tmpl +36 -0
- package/templates/vertical_devops/CLAUDE.md.tmpl +34 -0
- package/templates/vertical_devops/README.md.tmpl +27 -0
- package/templates/vertical_devops/manifest.json +22 -0
- package/templates/vertical_devops/package.json.tmpl +26 -0
- package/templates/vertical_devops/runbooks/README.md.tmpl +32 -0
- package/templates/vertical_devops/src/agents/escalator.ts.tmpl +19 -0
- package/templates/vertical_devops/src/agents/postmortem.ts.tmpl +23 -0
- package/templates/vertical_devops/src/agents/responder.ts.tmpl +14 -0
- package/templates/vertical_devops/src/agents/runbook-runner.ts.tmpl +17 -0
- package/templates/vertical_devops/src/init.ts.tmpl +23 -0
- package/templates/vertical_education/.claude/commands/doctor.md.tmpl +12 -0
- package/templates/vertical_education/.claude/commands/mastery-report.md.tmpl +12 -0
- package/templates/vertical_education/.claude/settings.json.tmpl +45 -0
- package/templates/vertical_education/.claude/skills/memory-inspect/SKILL.md.tmpl +14 -0
- package/templates/vertical_education/.claude/skills/teach-next/SKILL.md.tmpl +16 -0
- package/templates/vertical_education/CLAUDE.md.tmpl +33 -0
- package/templates/vertical_education/README.md.tmpl +30 -0
- package/templates/vertical_education/manifest.json +96 -0
- package/templates/vertical_education/package.json.tmpl +30 -0
- package/templates/vertical_education/src/agents/explainer.ts.tmpl +7 -0
- package/templates/vertical_education/src/agents/grader.ts.tmpl +7 -0
- package/templates/vertical_education/src/agents/quiz-master.ts.tmpl +7 -0
- package/templates/vertical_education/src/agents/tutor.ts.tmpl +7 -0
- package/templates/vertical_education/src/init.ts.tmpl +21 -0
- package/templates/vertical_exotic/.claude/commands/doctor.md.tmpl +12 -0
- package/templates/vertical_exotic/.claude/settings.json.tmpl +44 -0
- package/templates/vertical_exotic/.claude/skills/evolve/SKILL.md.tmpl +14 -0
- package/templates/vertical_exotic/.claude/skills/memory-inspect/SKILL.md.tmpl +14 -0
- package/templates/vertical_exotic/CLAUDE.md.tmpl +31 -0
- package/templates/vertical_exotic/README.md.tmpl +29 -0
- package/templates/vertical_exotic/manifest.json +86 -0
- package/templates/vertical_exotic/package.json.tmpl +30 -0
- package/templates/vertical_exotic/src/agents/experimenter.ts.tmpl +7 -0
- package/templates/vertical_exotic/src/agents/federator.ts.tmpl +7 -0
- package/templates/vertical_exotic/src/agents/hypothesizer.ts.tmpl +7 -0
- package/templates/vertical_exotic/src/init.ts.tmpl +21 -0
- package/templates/vertical_gaming/.claude/commands/design-doc-diff.md.tmpl +13 -0
- package/templates/vertical_gaming/.claude/commands/doctor.md.tmpl +12 -0
- package/templates/vertical_gaming/.claude/settings.json.tmpl +45 -0
- package/templates/vertical_gaming/.claude/skills/memory-inspect/SKILL.md.tmpl +14 -0
- package/templates/vertical_gaming/.claude/skills/playtest-recap/SKILL.md.tmpl +16 -0
- package/templates/vertical_gaming/CLAUDE.md.tmpl +33 -0
- package/templates/vertical_gaming/README.md.tmpl +30 -0
- package/templates/vertical_gaming/manifest.json +96 -0
- package/templates/vertical_gaming/package.json.tmpl +30 -0
- package/templates/vertical_gaming/src/agents/balance-critic.ts.tmpl +7 -0
- package/templates/vertical_gaming/src/agents/economy-modeler.ts.tmpl +7 -0
- package/templates/vertical_gaming/src/agents/narrative-keeper.ts.tmpl +7 -0
- package/templates/vertical_gaming/src/agents/playtest-reader.ts.tmpl +7 -0
- package/templates/vertical_gaming/src/init.ts.tmpl +21 -0
- package/templates/vertical_health/.claude/commands/doctor.md.tmpl +12 -0
- package/templates/vertical_health/.claude/settings.json.tmpl +33 -0
- package/templates/vertical_health/.claude/skills/wellness-intake/SKILL.md.tmpl +15 -0
- package/templates/vertical_health/CLAUDE.md.tmpl +30 -0
- package/templates/vertical_health/README.md.tmpl +29 -0
- package/templates/vertical_health/manifest.json +81 -0
- package/templates/vertical_health/package.json.tmpl +30 -0
- package/templates/vertical_health/src/agents/care-coordinator.ts.tmpl +7 -0
- package/templates/vertical_health/src/agents/intake.ts.tmpl +7 -0
- package/templates/vertical_health/src/agents/triage.ts.tmpl +7 -0
- package/templates/vertical_health/src/init.ts.tmpl +21 -0
- package/templates/vertical_legal/.claude/settings.json.tmpl +10 -0
- package/templates/vertical_legal/CLAUDE.md.tmpl +24 -0
- package/templates/vertical_legal/README.md.tmpl +21 -0
- package/templates/vertical_legal/manifest.json +20 -0
- package/templates/vertical_legal/package.json.tmpl +13 -0
- package/templates/vertical_legal/src/agents/citation-checker.ts.tmpl +7 -0
- package/templates/vertical_legal/src/agents/redline.ts.tmpl +7 -0
- package/templates/vertical_legal/src/agents/risk-rater.ts.tmpl +8 -0
- package/templates/vertical_legal/src/init.ts.tmpl +13 -0
- package/templates/vertical_marketing/.claude/commands/doctor.md.tmpl +12 -0
- package/templates/vertical_marketing/.claude/settings.json.tmpl +33 -0
- package/templates/vertical_marketing/.claude/skills/campaign-brief/SKILL.md.tmpl +13 -0
- package/templates/vertical_marketing/CLAUDE.md.tmpl +30 -0
- package/templates/vertical_marketing/README.md.tmpl +29 -0
- package/templates/vertical_marketing/manifest.json +81 -0
- package/templates/vertical_marketing/package.json.tmpl +30 -0
- package/templates/vertical_marketing/src/agents/content-creator.ts.tmpl +7 -0
- package/templates/vertical_marketing/src/agents/seo-analyst.ts.tmpl +7 -0
- package/templates/vertical_marketing/src/agents/strategist.ts.tmpl +7 -0
- package/templates/vertical_marketing/src/init.ts.tmpl +21 -0
- package/templates/vertical_repo-maintainer/.claude/commands/doctor.md.tmpl +12 -0
- package/templates/vertical_repo-maintainer/.claude/commands/release-check.md.tmpl +13 -0
- package/templates/vertical_repo-maintainer/.claude/commands/repo-triage.md.tmpl +16 -0
- package/templates/vertical_repo-maintainer/.claude/settings.json.tmpl +44 -0
- package/templates/vertical_repo-maintainer/.claude/skills/memory-inspect/SKILL.md.tmpl +14 -0
- package/templates/vertical_repo-maintainer/.claude/skills/plan-change/SKILL.md.tmpl +15 -0
- package/templates/vertical_repo-maintainer/CLAUDE.md.tmpl +34 -0
- package/templates/vertical_repo-maintainer/README.md.tmpl +30 -0
- package/templates/vertical_repo-maintainer/manifest.json +101 -0
- package/templates/vertical_repo-maintainer/package.json.tmpl +30 -0
- package/templates/vertical_repo-maintainer/src/agents/benchmarker.ts.tmpl +7 -0
- package/templates/vertical_repo-maintainer/src/agents/maintainer.ts.tmpl +7 -0
- package/templates/vertical_repo-maintainer/src/agents/release.ts.tmpl +7 -0
- package/templates/vertical_repo-maintainer/src/agents/security.ts.tmpl +7 -0
- package/templates/vertical_repo-maintainer/src/init.ts.tmpl +21 -0
- package/templates/vertical_research/.claude/settings.json.tmpl +9 -0
- package/templates/vertical_research/CLAUDE.md.tmpl +33 -0
- package/templates/vertical_research/README.md.tmpl +17 -0
- package/templates/vertical_research/manifest.json +23 -0
- package/templates/vertical_research/package.json.tmpl +13 -0
- package/templates/vertical_research/src/agents/citer.ts.tmpl +8 -0
- package/templates/vertical_research/src/agents/fact-checker.ts.tmpl +8 -0
- package/templates/vertical_research/src/agents/scout.ts.tmpl +7 -0
- package/templates/vertical_research/src/agents/source-grader.ts.tmpl +10 -0
- package/templates/vertical_research/src/agents/synthesizer.ts.tmpl +8 -0
- package/templates/vertical_research/src/agents/web-searcher.ts.tmpl +6 -0
- package/templates/vertical_research/src/init.ts.tmpl +13 -0
- package/templates/vertical_ruview/.claude/commands/doctor.md.tmpl +12 -0
- package/templates/vertical_ruview/.claude/settings.json.tmpl +33 -0
- package/templates/vertical_ruview/.claude/skills/index-and-ask/SKILL.md.tmpl +15 -0
- package/templates/vertical_ruview/.claude/skills/memory-inspect/SKILL.md.tmpl +14 -0
- package/templates/vertical_ruview/CLAUDE.md.tmpl +31 -0
- package/templates/vertical_ruview/README.md.tmpl +29 -0
- package/templates/vertical_ruview/manifest.json +86 -0
- package/templates/vertical_ruview/package.json.tmpl +30 -0
- package/templates/vertical_ruview/src/agents/indexer.ts.tmpl +7 -0
- package/templates/vertical_ruview/src/agents/retriever.ts.tmpl +7 -0
- package/templates/vertical_ruview/src/agents/reviewer.ts.tmpl +7 -0
- package/templates/vertical_ruview/src/init.ts.tmpl +21 -0
- package/templates/vertical_sales/.claude/commands/doctor.md.tmpl +12 -0
- package/templates/vertical_sales/.claude/commands/pipeline-report.md.tmpl +13 -0
- package/templates/vertical_sales/.claude/settings.json.tmpl +45 -0
- package/templates/vertical_sales/.claude/skills/memory-inspect/SKILL.md.tmpl +14 -0
- package/templates/vertical_sales/.claude/skills/qualify-lead/SKILL.md.tmpl +15 -0
- package/templates/vertical_sales/CLAUDE.md.tmpl +33 -0
- package/templates/vertical_sales/README.md.tmpl +30 -0
- package/templates/vertical_sales/manifest.json +96 -0
- package/templates/vertical_sales/package.json.tmpl +30 -0
- package/templates/vertical_sales/src/agents/closer.ts.tmpl +7 -0
- package/templates/vertical_sales/src/agents/demo-coach.ts.tmpl +7 -0
- package/templates/vertical_sales/src/agents/prospector.ts.tmpl +7 -0
- package/templates/vertical_sales/src/agents/qualifier.ts.tmpl +7 -0
- package/templates/vertical_sales/src/init.ts.tmpl +21 -0
- package/templates/vertical_support/.claude/settings.json.tmpl +25 -0
- package/templates/vertical_support/CLAUDE.md.tmpl +34 -0
- package/templates/vertical_support/README.md.tmpl +16 -0
- package/templates/vertical_support/kb/README.md.tmpl +23 -0
- package/templates/vertical_support/manifest.json +22 -0
- package/templates/vertical_support/package.json.tmpl +24 -0
- package/templates/vertical_support/src/agents/escalator.ts.tmpl +6 -0
- package/templates/vertical_support/src/agents/kb-searcher.ts.tmpl +6 -0
- package/templates/vertical_support/src/agents/responder.ts.tmpl +7 -0
- package/templates/vertical_support/src/agents/triager.ts.tmpl +6 -0
- package/templates/vertical_support/src/init.ts.tmpl +16 -0
- package/templates/vertical_trading/.claude/settings.json.tmpl +20 -0
- package/templates/vertical_trading/CLAUDE.md.tmpl +24 -0
- package/templates/vertical_trading/README.md.tmpl +17 -0
- package/templates/vertical_trading/manifest.json +22 -0
- package/templates/vertical_trading/package.json.tmpl +13 -0
- package/templates/vertical_trading/src/agents/executor.ts.tmpl +8 -0
- package/templates/vertical_trading/src/agents/market-watcher.ts.tmpl +7 -0
- package/templates/vertical_trading/src/agents/postmortem.ts.tmpl +8 -0
- package/templates/vertical_trading/src/agents/risk-checker.ts.tmpl +11 -0
- package/templates/vertical_trading/src/agents/signal-gen.ts.tmpl +8 -0
- package/templates/vertical_trading/src/init.ts.tmpl +20 -0
package/dist/score.js
ADDED
|
@@ -0,0 +1,381 @@
|
|
|
1
|
+
// SPDX-License-Identifier: MIT
|
|
2
|
+
//
|
|
3
|
+
// `harness score <path>` — 19th subcommand (iter 111). Priority 2 from the
|
|
4
|
+
// user's roadmap. Aggregates 5 dimensions into a single 0–100 score plus a
|
|
5
|
+
// per-dimension breakdown.
|
|
6
|
+
//
|
|
7
|
+
// Dimensions + weights (from the user's spec):
|
|
8
|
+
// - Repo understanding (25%) — meta.surface + meta.kernel_version + diag verdict
|
|
9
|
+
// - Agent usefulness (25%) — count of agents + skills + commands
|
|
10
|
+
// - MCP safety (20%) — mcp-policy.json default-deny + audit + no risky perms
|
|
11
|
+
// - Test coverage (15%) — presence of __tests__/ or tests/ + ci wiring
|
|
12
|
+
// - Publish readiness (15%) — witness present + sbom emit-able + npm pack ready
|
|
13
|
+
//
|
|
14
|
+
// Modes: text (default) · --json · --bundle (ADR-031 schema-1 envelope) · --out <file>.
|
|
15
|
+
//
|
|
16
|
+
// Verdict + exit code:
|
|
17
|
+
// - score ≥ 85 → 'A' (exit 0, the target the user named)
|
|
18
|
+
// - score ≥ 70 → 'B' (exit 0)
|
|
19
|
+
// - score ≥ 50 → 'C' (exit 1, needs work)
|
|
20
|
+
// - score < 50 → 'F' (exit 2, blocked)
|
|
21
|
+
import { existsSync, statSync, writeFileSync, readFileSync, readdirSync } from 'node:fs';
|
|
22
|
+
import { resolve, join } from 'node:path';
|
|
23
|
+
// --- helpers ---------------------------------------------------------------
|
|
24
|
+
function safeReadJson(path) {
|
|
25
|
+
try {
|
|
26
|
+
if (!existsSync(path))
|
|
27
|
+
return null;
|
|
28
|
+
return JSON.parse(readFileSync(path, 'utf-8'));
|
|
29
|
+
}
|
|
30
|
+
catch {
|
|
31
|
+
return null;
|
|
32
|
+
}
|
|
33
|
+
}
|
|
34
|
+
function dirExists(dir, sub) {
|
|
35
|
+
try {
|
|
36
|
+
return existsSync(join(dir, sub)) && statSync(join(dir, sub)).isDirectory();
|
|
37
|
+
}
|
|
38
|
+
catch {
|
|
39
|
+
return false;
|
|
40
|
+
}
|
|
41
|
+
}
|
|
42
|
+
function fileExists(dir, sub) {
|
|
43
|
+
try {
|
|
44
|
+
return existsSync(join(dir, sub)) && statSync(join(dir, sub)).isFile();
|
|
45
|
+
}
|
|
46
|
+
catch {
|
|
47
|
+
return false;
|
|
48
|
+
}
|
|
49
|
+
}
|
|
50
|
+
function countDir(dir, sub) {
|
|
51
|
+
try {
|
|
52
|
+
const target = join(dir, sub);
|
|
53
|
+
if (!existsSync(target))
|
|
54
|
+
return 0;
|
|
55
|
+
return readdirSync(target).filter((n) => !n.startsWith('.')).length;
|
|
56
|
+
}
|
|
57
|
+
catch {
|
|
58
|
+
return 0;
|
|
59
|
+
}
|
|
60
|
+
}
|
|
61
|
+
// --- dimension scorers (pure-ish — they read disk but no exec) -------------
|
|
62
|
+
function scoreRepoUnderstanding(dir, manifest) {
|
|
63
|
+
const signals = [];
|
|
64
|
+
let s = 0;
|
|
65
|
+
if (manifest) {
|
|
66
|
+
s += 35;
|
|
67
|
+
signals.push('manifest.json present');
|
|
68
|
+
if (manifest.meta?.surface) {
|
|
69
|
+
s += 20;
|
|
70
|
+
signals.push(`surface=${manifest.meta.surface}`);
|
|
71
|
+
}
|
|
72
|
+
if (manifest.meta?.kernel_version) {
|
|
73
|
+
s += 25;
|
|
74
|
+
signals.push(`kernel=${manifest.meta.kernel_version}`);
|
|
75
|
+
}
|
|
76
|
+
if (manifest.host || manifest.hosts) {
|
|
77
|
+
s += 20;
|
|
78
|
+
signals.push(`host(s)=${manifest.hosts ?? manifest.host}`);
|
|
79
|
+
}
|
|
80
|
+
}
|
|
81
|
+
else {
|
|
82
|
+
signals.push('no manifest — cannot identify what this harness is');
|
|
83
|
+
}
|
|
84
|
+
return { name: 'Repo understanding', weight: 0.25, score: Math.min(100, s), signals };
|
|
85
|
+
}
|
|
86
|
+
function scoreAgentUsefulness(dir) {
|
|
87
|
+
const signals = [];
|
|
88
|
+
let s = 0;
|
|
89
|
+
const agents = countDir(dir, 'src/agents');
|
|
90
|
+
const skills = countDir(dir, '.claude/skills');
|
|
91
|
+
const commands = countDir(dir, '.claude/commands');
|
|
92
|
+
if (agents > 0) {
|
|
93
|
+
s += Math.min(40, agents * 10);
|
|
94
|
+
signals.push(`${agents} agent(s) in src/agents/`);
|
|
95
|
+
}
|
|
96
|
+
if (skills > 0) {
|
|
97
|
+
s += Math.min(30, skills * 15);
|
|
98
|
+
signals.push(`${skills} skill(s) in .claude/skills/`);
|
|
99
|
+
}
|
|
100
|
+
if (commands > 0) {
|
|
101
|
+
s += Math.min(30, commands * 15);
|
|
102
|
+
signals.push(`${commands} command(s) in .claude/commands/`);
|
|
103
|
+
}
|
|
104
|
+
if (s === 0)
|
|
105
|
+
signals.push('no agents / skills / commands found');
|
|
106
|
+
return { name: 'Agent usefulness', weight: 0.25, score: Math.min(100, s), signals };
|
|
107
|
+
}
|
|
108
|
+
function scoreMcpSafety(dir) {
|
|
109
|
+
const signals = [];
|
|
110
|
+
const policy = safeReadJson(join(dir, '.harness', 'mcp-policy.json'));
|
|
111
|
+
// No policy + no .mcp.json at all = MCP not in use, which is the safest possible.
|
|
112
|
+
const hasMcp = policy != null || fileExists(dir, '.mcp.json');
|
|
113
|
+
if (!hasMcp) {
|
|
114
|
+
signals.push('MCP not in use (mode=off — safest)');
|
|
115
|
+
return { name: 'MCP safety', weight: 0.2, score: 100, signals, mcpRisk: 'None' };
|
|
116
|
+
}
|
|
117
|
+
let s = 0;
|
|
118
|
+
let risk = 'Low';
|
|
119
|
+
if (policy?.defaultDeny === true) {
|
|
120
|
+
s += 40;
|
|
121
|
+
signals.push('default-deny ON');
|
|
122
|
+
}
|
|
123
|
+
else {
|
|
124
|
+
risk = 'High';
|
|
125
|
+
signals.push('default-deny OFF — dangerous');
|
|
126
|
+
}
|
|
127
|
+
if (policy?.auditLog === true) {
|
|
128
|
+
s += 15;
|
|
129
|
+
signals.push('audit-log ON');
|
|
130
|
+
}
|
|
131
|
+
if (policy?.allowShell === false || policy?.allowShell == null) {
|
|
132
|
+
s += 15;
|
|
133
|
+
signals.push('shell gated');
|
|
134
|
+
}
|
|
135
|
+
else {
|
|
136
|
+
risk = 'High';
|
|
137
|
+
signals.push('shell ALLOWED — risky');
|
|
138
|
+
}
|
|
139
|
+
if (policy?.allowNetwork === false || policy?.allowNetwork == null) {
|
|
140
|
+
s += 15;
|
|
141
|
+
signals.push('network gated');
|
|
142
|
+
}
|
|
143
|
+
else {
|
|
144
|
+
if (risk !== 'High')
|
|
145
|
+
risk = 'Medium';
|
|
146
|
+
signals.push('network ALLOWED');
|
|
147
|
+
}
|
|
148
|
+
if (policy?.allowFileWrite === false || policy?.allowFileWrite == null) {
|
|
149
|
+
s += 15;
|
|
150
|
+
signals.push('file-write gated');
|
|
151
|
+
}
|
|
152
|
+
else {
|
|
153
|
+
if (risk !== 'High')
|
|
154
|
+
risk = 'Medium';
|
|
155
|
+
signals.push('file-write ALLOWED');
|
|
156
|
+
}
|
|
157
|
+
return { name: 'MCP safety', weight: 0.2, score: Math.min(100, s), signals, mcpRisk: risk };
|
|
158
|
+
}
|
|
159
|
+
function scoreTestCoverage(dir) {
|
|
160
|
+
const signals = [];
|
|
161
|
+
let s = 0;
|
|
162
|
+
const hasTests = dirExists(dir, '__tests__') || dirExists(dir, 'tests') || dirExists(dir, 'test');
|
|
163
|
+
if (hasTests) {
|
|
164
|
+
s += 50;
|
|
165
|
+
signals.push('test directory present');
|
|
166
|
+
}
|
|
167
|
+
const pkg = safeReadJson(join(dir, 'package.json'));
|
|
168
|
+
if (pkg?.scripts?.test) {
|
|
169
|
+
s += 25;
|
|
170
|
+
signals.push(`npm test → ${pkg.scripts.test}`);
|
|
171
|
+
}
|
|
172
|
+
if (dirExists(dir, '.github/workflows')) {
|
|
173
|
+
s += 25;
|
|
174
|
+
signals.push('CI workflow present');
|
|
175
|
+
}
|
|
176
|
+
if (s === 0)
|
|
177
|
+
signals.push('no test signals detected');
|
|
178
|
+
return { name: 'Test coverage', weight: 0.15, score: Math.min(100, s), signals };
|
|
179
|
+
}
|
|
180
|
+
function scorePublishReadiness(dir, manifest) {
|
|
181
|
+
const signals = [];
|
|
182
|
+
let s = 0;
|
|
183
|
+
const witnessSigned = fileExists(dir, '.harness/witness.json');
|
|
184
|
+
if (witnessSigned) {
|
|
185
|
+
s += 40;
|
|
186
|
+
signals.push('witness.json present');
|
|
187
|
+
}
|
|
188
|
+
// sbom.json is emitted by `harness sbom`; presence indicates the operator
|
|
189
|
+
// has at least run it once. Even if it's not committed, presence is the
|
|
190
|
+
// strong publish-readiness signal.
|
|
191
|
+
const sbom = fileExists(dir, 'sbom.json') || fileExists(dir, '.harness/sbom.json');
|
|
192
|
+
if (sbom) {
|
|
193
|
+
s += 20;
|
|
194
|
+
signals.push('SBOM present');
|
|
195
|
+
}
|
|
196
|
+
const pkg = safeReadJson(join(dir, 'package.json'));
|
|
197
|
+
if (pkg?.name && pkg?.version) {
|
|
198
|
+
s += 20;
|
|
199
|
+
signals.push(`pkg=${pkg.name}@${pkg.version}`);
|
|
200
|
+
}
|
|
201
|
+
if (manifest?.host && pkg?.bin) {
|
|
202
|
+
s += 20;
|
|
203
|
+
signals.push('bin entry present (npx-runnable)');
|
|
204
|
+
}
|
|
205
|
+
else if (pkg?.bin) {
|
|
206
|
+
s += 10;
|
|
207
|
+
signals.push('bin entry present');
|
|
208
|
+
}
|
|
209
|
+
if (s === 0)
|
|
210
|
+
signals.push('no publish signals — run harness validate first');
|
|
211
|
+
const releaseReady = s >= 70;
|
|
212
|
+
return { name: 'Publish readiness', weight: 0.15, score: Math.min(100, s), signals, witnessSigned, sbom, releaseReady };
|
|
213
|
+
}
|
|
214
|
+
// --- overall ---------------------------------------------------------------
|
|
215
|
+
export function buildScorecard(dir, generatedAt = new Date().toISOString()) {
|
|
216
|
+
const manifest = safeReadJson(join(dir, '.harness', 'manifest.json'));
|
|
217
|
+
const repoUnderstanding = scoreRepoUnderstanding(dir, manifest);
|
|
218
|
+
const agentUsefulness = scoreAgentUsefulness(dir);
|
|
219
|
+
const mcpSafety = scoreMcpSafety(dir);
|
|
220
|
+
const testCoverage = scoreTestCoverage(dir);
|
|
221
|
+
const publishReadiness = scorePublishReadiness(dir, manifest);
|
|
222
|
+
const dims = [repoUnderstanding, agentUsefulness, mcpSafety, testCoverage, publishReadiness];
|
|
223
|
+
const overall = Math.round(dims.reduce((sum, d) => sum + d.weight * d.score, 0));
|
|
224
|
+
let grade;
|
|
225
|
+
let exitCode;
|
|
226
|
+
if (overall >= 85) {
|
|
227
|
+
grade = 'A';
|
|
228
|
+
exitCode = 0;
|
|
229
|
+
}
|
|
230
|
+
else if (overall >= 70) {
|
|
231
|
+
grade = 'B';
|
|
232
|
+
exitCode = 0;
|
|
233
|
+
}
|
|
234
|
+
else if (overall >= 50) {
|
|
235
|
+
grade = 'C';
|
|
236
|
+
exitCode = 1;
|
|
237
|
+
}
|
|
238
|
+
else {
|
|
239
|
+
grade = 'F';
|
|
240
|
+
exitCode = 2;
|
|
241
|
+
}
|
|
242
|
+
// Detect tests: any of __tests__/ tests/ test/.
|
|
243
|
+
const testsDetected = testCoverage.score > 0;
|
|
244
|
+
return {
|
|
245
|
+
schema: 1,
|
|
246
|
+
generatedAt,
|
|
247
|
+
dir,
|
|
248
|
+
overall,
|
|
249
|
+
grade,
|
|
250
|
+
dimensions: dims,
|
|
251
|
+
badges: {
|
|
252
|
+
score: overall,
|
|
253
|
+
mcpRisk: mcpSafety.mcpRisk,
|
|
254
|
+
releaseReady: publishReadiness.releaseReady,
|
|
255
|
+
testsDetected,
|
|
256
|
+
sbom: publishReadiness.sbom,
|
|
257
|
+
witnessSigned: publishReadiness.witnessSigned,
|
|
258
|
+
},
|
|
259
|
+
exitCode,
|
|
260
|
+
};
|
|
261
|
+
}
|
|
262
|
+
// --- formatting ------------------------------------------------------------
|
|
263
|
+
function bar(score) {
|
|
264
|
+
const width = 20;
|
|
265
|
+
const filled = Math.round((score / 100) * width);
|
|
266
|
+
return '█'.repeat(filled) + '░'.repeat(width - filled);
|
|
267
|
+
}
|
|
268
|
+
function yn(b) {
|
|
269
|
+
return b ? 'Yes' : 'No';
|
|
270
|
+
}
|
|
271
|
+
export function formatScorecard(sc) {
|
|
272
|
+
const out = [];
|
|
273
|
+
out.push(`harness score — ${sc.dir}`);
|
|
274
|
+
out.push('');
|
|
275
|
+
out.push(`Overall: ${sc.overall}/100 Grade: ${sc.grade}`);
|
|
276
|
+
out.push('');
|
|
277
|
+
for (const d of sc.dimensions) {
|
|
278
|
+
out.push(` ${d.name.padEnd(22)} ${String(d.score).padStart(3)} ${bar(d.score)} (${Math.round(d.weight * 100)}%)`);
|
|
279
|
+
for (const s of d.signals)
|
|
280
|
+
out.push(` · ${s}`);
|
|
281
|
+
}
|
|
282
|
+
out.push('');
|
|
283
|
+
out.push('Badges:');
|
|
284
|
+
out.push(` Harness Score: ${sc.badges.score}`);
|
|
285
|
+
out.push(` MCP Risk: ${sc.badges.mcpRisk}`);
|
|
286
|
+
out.push(` Release Ready: ${yn(sc.badges.releaseReady)}`);
|
|
287
|
+
out.push(` Tests Detected:${yn(sc.badges.testsDetected)}`);
|
|
288
|
+
out.push(` SBOM: ${yn(sc.badges.sbom)}`);
|
|
289
|
+
out.push(` Witness Signed:${yn(sc.badges.witnessSigned)}`);
|
|
290
|
+
out.push('');
|
|
291
|
+
out.push(`Verdict: ${sc.grade} (exit ${sc.exitCode})`);
|
|
292
|
+
return out;
|
|
293
|
+
}
|
|
294
|
+
// --- sanitisation (ADR-031) ------------------------------------------------
|
|
295
|
+
const SECRET_RE = /(secret|token|key|password|passphrase)/i;
|
|
296
|
+
function sanitise(v) {
|
|
297
|
+
if (v == null)
|
|
298
|
+
return v;
|
|
299
|
+
if (Array.isArray(v))
|
|
300
|
+
return v.map(sanitise);
|
|
301
|
+
if (typeof v === 'object') {
|
|
302
|
+
const out = {};
|
|
303
|
+
for (const [k, val] of Object.entries(v)) {
|
|
304
|
+
out[k] = SECRET_RE.test(k) ? '[REDACTED]' : sanitise(val);
|
|
305
|
+
}
|
|
306
|
+
return out;
|
|
307
|
+
}
|
|
308
|
+
return v;
|
|
309
|
+
}
|
|
310
|
+
// --- dispatch --------------------------------------------------------------
|
|
311
|
+
function usage() {
|
|
312
|
+
return [
|
|
313
|
+
'Usage: harness score <path> [--out <file>] [--json] [--bundle]',
|
|
314
|
+
'',
|
|
315
|
+
' 5-dimension scorecard for a generated harness. 0–100, grade A/B/C/F.',
|
|
316
|
+
' Dimensions: Repo understanding (25%) · Agent usefulness (25%) ·',
|
|
317
|
+
' MCP safety (20%) · Test coverage (15%) · Publish readiness (15%).',
|
|
318
|
+
'',
|
|
319
|
+
' --out <file> Write the badges JSON to <file>.',
|
|
320
|
+
' --json Emit the badges JSON to stdout instead of text.',
|
|
321
|
+
' --bundle Emit the full scorecard as an ADR-031 schema-1 envelope.',
|
|
322
|
+
];
|
|
323
|
+
}
|
|
324
|
+
export async function scoreCmd(args) {
|
|
325
|
+
const bundle = args.includes('--bundle');
|
|
326
|
+
const json = args.includes('--json');
|
|
327
|
+
let outPath = null;
|
|
328
|
+
const positional = [];
|
|
329
|
+
for (let i = 0; i < args.length; i++) {
|
|
330
|
+
const a = args[i];
|
|
331
|
+
if (a === '--out') {
|
|
332
|
+
outPath = args[++i] ?? null;
|
|
333
|
+
if (!outPath) {
|
|
334
|
+
const err = { schema: 1, error: 'missing-out-path', exitCode: 2 };
|
|
335
|
+
return { code: 2, lines: [bundle || json ? JSON.stringify(err, null, 2) : '--out requires a file path'] };
|
|
336
|
+
}
|
|
337
|
+
}
|
|
338
|
+
else if (a === '--bundle' || a === '--json') {
|
|
339
|
+
/* handled */
|
|
340
|
+
}
|
|
341
|
+
else if (a === '--help' || a === '-h') {
|
|
342
|
+
return { code: 0, lines: usage() };
|
|
343
|
+
}
|
|
344
|
+
else if (a && !a.startsWith('--')) {
|
|
345
|
+
positional.push(a);
|
|
346
|
+
}
|
|
347
|
+
else if (a) {
|
|
348
|
+
const err = { schema: 1, error: `unknown-flag-${a.replace(/^--?/, '')}`, exitCode: 2 };
|
|
349
|
+
return { code: 2, lines: [bundle || json ? JSON.stringify(err, null, 2) : `Unknown flag: ${a}`] };
|
|
350
|
+
}
|
|
351
|
+
}
|
|
352
|
+
if (positional.length === 0) {
|
|
353
|
+
return { code: 2, lines: usage() };
|
|
354
|
+
}
|
|
355
|
+
const dir = resolve(positional[0]);
|
|
356
|
+
if (!existsSync(dir) || !statSync(dir).isDirectory()) {
|
|
357
|
+
const generatedAt = new Date().toISOString();
|
|
358
|
+
const err = { schema: 1, generatedAt, error: 'not-a-directory', dir, exitCode: 2 };
|
|
359
|
+
if (bundle || json)
|
|
360
|
+
return { code: 2, lines: [JSON.stringify(err, null, 2)] };
|
|
361
|
+
return { code: 2, lines: [`harness score: not a directory: ${dir}`] };
|
|
362
|
+
}
|
|
363
|
+
const sc = buildScorecard(dir);
|
|
364
|
+
if (outPath) {
|
|
365
|
+
try {
|
|
366
|
+
writeFileSync(resolve(outPath), JSON.stringify(sc.badges, null, 2) + '\n', 'utf-8');
|
|
367
|
+
}
|
|
368
|
+
catch (e) {
|
|
369
|
+
const err = { schema: 1, error: 'out-write-failed', detail: String(e), exitCode: 2 };
|
|
370
|
+
return { code: 2, lines: [bundle || json ? JSON.stringify(err, null, 2) : `harness score: failed to write --out: ${String(e)}`] };
|
|
371
|
+
}
|
|
372
|
+
}
|
|
373
|
+
if (bundle) {
|
|
374
|
+
return { code: sc.exitCode, lines: [JSON.stringify(sanitise(sc), null, 2)] };
|
|
375
|
+
}
|
|
376
|
+
if (json) {
|
|
377
|
+
return { code: sc.exitCode, lines: [JSON.stringify(sc.badges, null, 2)] };
|
|
378
|
+
}
|
|
379
|
+
return { code: sc.exitCode, lines: formatScorecard(sc) };
|
|
380
|
+
}
|
|
381
|
+
//# sourceMappingURL=score.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"score.js","sourceRoot":"","sources":["../src/score.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,27 @@
|
|
|
1
|
+
export type SubcommandResult = {
|
|
2
|
+
code: number;
|
|
3
|
+
lines: string[];
|
|
4
|
+
};
|
|
5
|
+
export interface GcloudRunner {
|
|
6
|
+
run(args: string[], opts?: {
|
|
7
|
+
input?: string;
|
|
8
|
+
}): Promise<{
|
|
9
|
+
code: number;
|
|
10
|
+
stdout: string;
|
|
11
|
+
stderr: string;
|
|
12
|
+
}>;
|
|
13
|
+
}
|
|
14
|
+
/** Default runner — wraps gcloud subprocess. */
|
|
15
|
+
export declare const defaultRunner: GcloudRunner;
|
|
16
|
+
/** Validate full GCP setup for publish-time secret fetch. */
|
|
17
|
+
export declare function check(args: string[], runner?: GcloudRunner): Promise<SubcommandResult>;
|
|
18
|
+
/** Fetch a secret's latest value and print to stdout. */
|
|
19
|
+
export declare function fetch(args: string[], runner?: GcloudRunner): Promise<SubcommandResult>;
|
|
20
|
+
/**
|
|
21
|
+
* Fetch NPM_TOKEN and run `npm whoami --registry=https://registry.npmjs.org/`.
|
|
22
|
+
* No publish — just confirms the token is valid + non-revoked.
|
|
23
|
+
*/
|
|
24
|
+
export declare function validateToken(args: string[], runner?: GcloudRunner): Promise<SubcommandResult>;
|
|
25
|
+
/** Top-level dispatcher for `harness secrets <subsub> ...`. */
|
|
26
|
+
export declare function secretsDispatch(args: string[], runner?: GcloudRunner): Promise<SubcommandResult>;
|
|
27
|
+
//# sourceMappingURL=secrets.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"secrets.d.ts","sourceRoot":"","sources":["../src/secrets.ts"],"names":[],"mappings":"AAsBA,MAAM,MAAM,gBAAgB,GAAG;IAAE,IAAI,EAAE,MAAM,CAAC;IAAC,KAAK,EAAE,MAAM,EAAE,CAAA;CAAE,CAAC;AAEjE,MAAM,WAAW,YAAY;IAC3B,GAAG,CAAC,IAAI,EAAE,MAAM,EAAE,EAAE,IAAI,CAAC,EAAE;QAAE,KAAK,CAAC,EAAE,MAAM,CAAA;KAAE,GAAG,OAAO,CAAC;QAAE,IAAI,EAAE,MAAM,CAAC;QAAC,MAAM,EAAE,MAAM,CAAC;QAAC,MAAM,EAAE,MAAM,CAAA;KAAE,CAAC,CAAC;CAC3G;AAED,gDAAgD;AAChD,eAAO,MAAM,aAAa,EAAE,YAiB3B,CAAC;AAWF,6DAA6D;AAC7D,wBAAsB,KAAK,CAAC,IAAI,EAAE,MAAM,EAAE,EAAE,MAAM,GAAE,YAA4B,GAAG,OAAO,CAAC,gBAAgB,CAAC,CAuE3G;AAED,yDAAyD;AACzD,wBAAsB,KAAK,CAAC,IAAI,EAAE,MAAM,EAAE,EAAE,MAAM,GAAE,YAA4B,GAAG,OAAO,CAAC,gBAAgB,CAAC,CAyB3G;AAED;;;GAGG;AACH,wBAAsB,aAAa,CAAC,IAAI,EAAE,MAAM,EAAE,EAAE,MAAM,GAAE,YAA4B,GAAG,OAAO,CAAC,gBAAgB,CAAC,CAwCnH;AAED,+DAA+D;AAC/D,wBAAsB,eAAe,CAAC,IAAI,EAAE,MAAM,EAAE,EAAE,MAAM,GAAE,YAA4B,GAAG,OAAO,CAAC,gBAAgB,CAAC,CAmCrH"}
|
package/dist/secrets.js
ADDED
|
@@ -0,0 +1,228 @@
|
|
|
1
|
+
// SPDX-License-Identifier: MIT
|
|
2
|
+
//
|
|
3
|
+
// `harness secrets` — GCP Secret Manager integration for publish-time
|
|
4
|
+
// credential validation. Uses `gcloud` shell-outs (already a documented
|
|
5
|
+
// prereq) rather than the @google-cloud/secret-manager SDK to avoid
|
|
6
|
+
// pulling 12 MB of GCP client deps into every install.
|
|
7
|
+
//
|
|
8
|
+
// Commands:
|
|
9
|
+
// harness secrets check validate full GCP setup
|
|
10
|
+
// harness secrets fetch <name> fetch a secret value (stdout)
|
|
11
|
+
// harness secrets validate-token fetch NPM_TOKEN + npm whoami
|
|
12
|
+
//
|
|
13
|
+
// Auth path:
|
|
14
|
+
// - Local dev: ADC via `gcloud auth application-default login`
|
|
15
|
+
// - CI: Workload Identity Federation (no long-lived keys)
|
|
16
|
+
import { spawn } from 'node:child_process';
|
|
17
|
+
import { promisify } from 'node:util';
|
|
18
|
+
import { execFile as execFileCb } from 'node:child_process';
|
|
19
|
+
const execFile = promisify(execFileCb);
|
|
20
|
+
/** Default runner — wraps gcloud subprocess. */
|
|
21
|
+
export const defaultRunner = {
|
|
22
|
+
async run(args, _opts) {
|
|
23
|
+
try {
|
|
24
|
+
const { stdout, stderr } = await execFile('gcloud', args, {
|
|
25
|
+
maxBuffer: 1024 * 1024,
|
|
26
|
+
windowsHide: true,
|
|
27
|
+
});
|
|
28
|
+
return { code: 0, stdout, stderr };
|
|
29
|
+
}
|
|
30
|
+
catch (err) {
|
|
31
|
+
const e = err;
|
|
32
|
+
return {
|
|
33
|
+
code: typeof e.code === 'number' ? e.code : 1,
|
|
34
|
+
stdout: e.stdout ?? '',
|
|
35
|
+
stderr: e.stderr ?? (e.message ?? ''),
|
|
36
|
+
};
|
|
37
|
+
}
|
|
38
|
+
},
|
|
39
|
+
};
|
|
40
|
+
function isGcloudOnPath() {
|
|
41
|
+
return new Promise(resolve => {
|
|
42
|
+
const cmd = process.platform === 'win32' ? 'where' : 'which';
|
|
43
|
+
const p = spawn(cmd, ['gcloud'], { stdio: 'ignore', windowsHide: true });
|
|
44
|
+
p.on('error', () => resolve(false));
|
|
45
|
+
p.on('exit', code => resolve(code === 0));
|
|
46
|
+
});
|
|
47
|
+
}
|
|
48
|
+
/** Validate full GCP setup for publish-time secret fetch. */
|
|
49
|
+
export async function check(args, runner = defaultRunner) {
|
|
50
|
+
const lines = ['harness secrets check'];
|
|
51
|
+
let problems = 0;
|
|
52
|
+
const projectFromFlag = args.find(a => a.startsWith('--project='))?.slice('--project='.length);
|
|
53
|
+
const secretName = args.find(a => a.startsWith('--secret='))?.slice('--secret='.length) ?? 'NPM_TOKEN';
|
|
54
|
+
if (!(await isGcloudOnPath())) {
|
|
55
|
+
lines.push(' FAIL gcloud CLI not on PATH');
|
|
56
|
+
lines.push(' install: https://cloud.google.com/sdk/docs/install');
|
|
57
|
+
return { code: 1, lines };
|
|
58
|
+
}
|
|
59
|
+
lines.push(' PASS gcloud is on PATH');
|
|
60
|
+
// 1. Project
|
|
61
|
+
let project = projectFromFlag;
|
|
62
|
+
if (!project) {
|
|
63
|
+
const r = await runner.run(['config', 'get-value', 'project']);
|
|
64
|
+
project = r.stdout.trim();
|
|
65
|
+
if (!project || project === '(unset)') {
|
|
66
|
+
lines.push(' FAIL no active gcloud project (try --project=<id> or gcloud config set project)');
|
|
67
|
+
problems++;
|
|
68
|
+
}
|
|
69
|
+
else {
|
|
70
|
+
lines.push(` PASS active project: ${project}`);
|
|
71
|
+
}
|
|
72
|
+
}
|
|
73
|
+
else {
|
|
74
|
+
lines.push(` PASS project flag: ${project}`);
|
|
75
|
+
}
|
|
76
|
+
if (!project || problems > 0)
|
|
77
|
+
return { code: 1, lines };
|
|
78
|
+
// 2. Auth
|
|
79
|
+
const auth = await runner.run(['auth', 'list', '--filter=status:ACTIVE', '--format=value(account)']);
|
|
80
|
+
if (auth.code !== 0 || !auth.stdout.trim()) {
|
|
81
|
+
lines.push(' FAIL no active gcloud auth (run: gcloud auth application-default login)');
|
|
82
|
+
problems++;
|
|
83
|
+
}
|
|
84
|
+
else {
|
|
85
|
+
lines.push(` PASS active account: ${auth.stdout.trim()}`);
|
|
86
|
+
}
|
|
87
|
+
// 3. Secret exists
|
|
88
|
+
const sec = await runner.run([
|
|
89
|
+
'secrets', 'describe', secretName,
|
|
90
|
+
`--project=${project}`,
|
|
91
|
+
'--format=value(name)',
|
|
92
|
+
]);
|
|
93
|
+
if (sec.code !== 0) {
|
|
94
|
+
lines.push(` FAIL secret '${secretName}' not found in project ${project}`);
|
|
95
|
+
lines.push(' create with: gcloud secrets create ' + secretName + ' --replication-policy=automatic');
|
|
96
|
+
problems++;
|
|
97
|
+
}
|
|
98
|
+
else {
|
|
99
|
+
lines.push(` PASS secret '${secretName}' exists`);
|
|
100
|
+
}
|
|
101
|
+
// 4. WIF pool (best-effort — skip if list errors)
|
|
102
|
+
const pool = await runner.run([
|
|
103
|
+
'iam', 'workload-identity-pools', 'list',
|
|
104
|
+
'--location=global', `--project=${project}`,
|
|
105
|
+
'--format=value(displayName)', '--filter=state:ACTIVE',
|
|
106
|
+
]);
|
|
107
|
+
if (pool.code === 0 && pool.stdout.trim()) {
|
|
108
|
+
lines.push(` PASS workload-identity-pool present (${pool.stdout.trim().split('\n')[0]})`);
|
|
109
|
+
}
|
|
110
|
+
else {
|
|
111
|
+
lines.push(' WARN no active workload-identity-pool found');
|
|
112
|
+
lines.push(' CI cannot fetch this secret without WIF — see docs/setup/gcp-secrets.md');
|
|
113
|
+
}
|
|
114
|
+
if (problems === 0) {
|
|
115
|
+
lines.push('', `Result: HEALTHY (project=${project}, secret=${secretName})`);
|
|
116
|
+
return { code: 0, lines };
|
|
117
|
+
}
|
|
118
|
+
lines.push('', `Result: ${problems} issue${problems === 1 ? '' : 's'}`);
|
|
119
|
+
return { code: 1, lines };
|
|
120
|
+
}
|
|
121
|
+
/** Fetch a secret's latest value and print to stdout. */
|
|
122
|
+
export async function fetch(args, runner = defaultRunner) {
|
|
123
|
+
const lines = [];
|
|
124
|
+
const name = args.find(a => !a.startsWith('--'));
|
|
125
|
+
if (!name) {
|
|
126
|
+
return { code: 2, lines: ['Usage: harness secrets fetch <secret-name> [--project=<id>] [--version=latest]'] };
|
|
127
|
+
}
|
|
128
|
+
const project = args.find(a => a.startsWith('--project='))?.slice('--project='.length);
|
|
129
|
+
const version = args.find(a => a.startsWith('--version='))?.slice('--version='.length) ?? 'latest';
|
|
130
|
+
const cmd = ['secrets', 'versions', 'access', version, `--secret=${name}`];
|
|
131
|
+
if (project)
|
|
132
|
+
cmd.push(`--project=${project}`);
|
|
133
|
+
const r = await runner.run(cmd);
|
|
134
|
+
if (r.code !== 0) {
|
|
135
|
+
return {
|
|
136
|
+
code: 1,
|
|
137
|
+
lines: [
|
|
138
|
+
`Fetch failed for secret '${name}': ${r.stderr.trim() || 'unknown error'}`,
|
|
139
|
+
'Try: harness secrets check --secret=' + name,
|
|
140
|
+
],
|
|
141
|
+
};
|
|
142
|
+
}
|
|
143
|
+
// Print raw value to stdout for piping (no trailing newline added).
|
|
144
|
+
process.stdout.write(r.stdout);
|
|
145
|
+
return { code: 0, lines };
|
|
146
|
+
}
|
|
147
|
+
/**
|
|
148
|
+
* Fetch NPM_TOKEN and run `npm whoami --registry=https://registry.npmjs.org/`.
|
|
149
|
+
* No publish — just confirms the token is valid + non-revoked.
|
|
150
|
+
*/
|
|
151
|
+
export async function validateToken(args, runner = defaultRunner) {
|
|
152
|
+
const lines = ['harness secrets validate-token'];
|
|
153
|
+
const project = args.find(a => a.startsWith('--project='))?.slice('--project='.length);
|
|
154
|
+
const secret = args.find(a => a.startsWith('--secret='))?.slice('--secret='.length) ?? 'NPM_TOKEN';
|
|
155
|
+
const cmd = ['secrets', 'versions', 'access', 'latest', `--secret=${secret}`];
|
|
156
|
+
if (project)
|
|
157
|
+
cmd.push(`--project=${project}`);
|
|
158
|
+
const r = await runner.run(cmd);
|
|
159
|
+
if (r.code !== 0) {
|
|
160
|
+
lines.push(` FAIL could not fetch '${secret}' — ${r.stderr.trim() || 'unknown error'}`);
|
|
161
|
+
return { code: 1, lines };
|
|
162
|
+
}
|
|
163
|
+
const token = r.stdout.trim();
|
|
164
|
+
if (!token) {
|
|
165
|
+
lines.push(` FAIL secret '${secret}' returned an empty value`);
|
|
166
|
+
return { code: 1, lines };
|
|
167
|
+
}
|
|
168
|
+
lines.push(` PASS fetched '${secret}' (${token.length} chars)`);
|
|
169
|
+
// Run npm whoami with token injected via env. No persistent .npmrc rewrite.
|
|
170
|
+
try {
|
|
171
|
+
const { stdout, stderr } = await execFile('npm', ['whoami', '--registry=https://registry.npmjs.org/'], {
|
|
172
|
+
env: { ...process.env, npm_config__authToken: token },
|
|
173
|
+
windowsHide: true,
|
|
174
|
+
});
|
|
175
|
+
const who = stdout.trim();
|
|
176
|
+
if (!who) {
|
|
177
|
+
lines.push(` FAIL npm whoami returned empty (stderr: ${stderr.trim()})`);
|
|
178
|
+
return { code: 1, lines };
|
|
179
|
+
}
|
|
180
|
+
lines.push(` PASS npm whoami: ${who}`);
|
|
181
|
+
lines.push('', `Result: HEALTHY (secret=${secret}, npm_user=${who})`);
|
|
182
|
+
return { code: 0, lines };
|
|
183
|
+
}
|
|
184
|
+
catch (err) {
|
|
185
|
+
const e = err;
|
|
186
|
+
lines.push(` FAIL npm whoami failed: ${e.stderr?.trim() || e.message || 'unknown'}`);
|
|
187
|
+
lines.push(' token is set but npm rejected it — may be expired or revoked');
|
|
188
|
+
return { code: 1, lines };
|
|
189
|
+
}
|
|
190
|
+
}
|
|
191
|
+
/** Top-level dispatcher for `harness secrets <subsub> ...`. */
|
|
192
|
+
export async function secretsDispatch(args, runner = defaultRunner) {
|
|
193
|
+
const [subsub = 'help', ...rest] = args;
|
|
194
|
+
switch (subsub) {
|
|
195
|
+
case 'check':
|
|
196
|
+
return check(rest, runner);
|
|
197
|
+
case 'fetch':
|
|
198
|
+
return fetch(rest, runner);
|
|
199
|
+
case 'validate-token':
|
|
200
|
+
return validateToken(rest, runner);
|
|
201
|
+
case 'help':
|
|
202
|
+
return {
|
|
203
|
+
code: 0,
|
|
204
|
+
lines: [
|
|
205
|
+
'Usage: harness secrets <subcommand> [options]',
|
|
206
|
+
'',
|
|
207
|
+
'Subcommands:',
|
|
208
|
+
' check — validate full GCP Secret Manager setup',
|
|
209
|
+
' fetch <name> — fetch a secret value (prints to stdout)',
|
|
210
|
+
' validate-token — fetch NPM_TOKEN and run `npm whoami`',
|
|
211
|
+
'',
|
|
212
|
+
'Common options:',
|
|
213
|
+
' --project=<id> override gcloud active project',
|
|
214
|
+
' --secret=<name> override default secret name (NPM_TOKEN)',
|
|
215
|
+
' --version=latest fetch a specific secret version',
|
|
216
|
+
'',
|
|
217
|
+
'CI usage (Workload Identity Federation):',
|
|
218
|
+
' see docs/setup/gcp-secrets.md',
|
|
219
|
+
],
|
|
220
|
+
};
|
|
221
|
+
default:
|
|
222
|
+
return {
|
|
223
|
+
code: 2,
|
|
224
|
+
lines: [`Unknown secrets subcommand: ${subsub}`, `Run 'harness secrets help' for usage.`],
|
|
225
|
+
};
|
|
226
|
+
}
|
|
227
|
+
}
|
|
228
|
+
//# sourceMappingURL=secrets.js.map
|