@event4u/agent-config 3.3.0 → 4.2.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/.agent-src/README.md +2 -2
- package/.agent-src/commands/agent-handoff.md +31 -2
- package/.agent-src/commands/agent-status.md +5 -5
- package/.agent-src/commands/agents/audit.md +8 -8
- package/.agent-src/commands/agents/init.md +25 -1
- package/.agent-src/commands/agents/optimize.md +3 -3
- package/.agent-src/commands/agents/user.md +1 -1
- package/.agent-src/commands/agents.md +1 -1
- package/.agent-src/commands/analyze-reference-repo.md +1 -1
- package/.agent-src/commands/check-current-md.md +8 -8
- package/.agent-src/commands/{compress.md → condense.md} +55 -55
- package/.agent-src/commands/context/create.md +7 -4
- package/.agent-src/commands/context/refactor.md +3 -1
- package/.agent-src/commands/feature/dev.md +1 -1
- package/.agent-src/commands/feature/explore.md +1 -1
- package/.agent-src/commands/feature/plan.md +10 -8
- package/.agent-src/commands/feature/refactor.md +3 -1
- package/.agent-src/commands/feature/roadmap.md +7 -4
- package/.agent-src/commands/fix/portability.md +3 -3
- package/.agent-src/commands/fix/refs.md +4 -4
- package/.agent-src/commands/ghostwriter.md +2 -2
- package/.agent-src/commands/memory/learn-low-impact.md +3 -3
- package/.agent-src/commands/module/explore.md +34 -8
- package/.agent-src/commands/optimize/agents-dir.md +9 -7
- package/.agent-src/commands/optimize/augmentignore.md +2 -2
- package/.agent-src/commands/optimize/skills.md +9 -9
- package/.agent-src/commands/post-as.md +1 -1
- package/.agent-src/commands/project-analyze.md +2 -2
- package/.agent-src/commands/project-health.md +3 -2
- package/.agent-src/commands/research/deep.md +1 -1
- package/.agent-src/commands/research/report.md +1 -1
- package/.agent-src/commands/research.md +1 -1
- package/.agent-src/commands/roadmap/ai-council.md +1 -1
- package/.agent-src/commands/roadmap/create.md +9 -4
- package/.agent-src/commands/rule-compliance-audit.md +1 -1
- package/.agent-src/commands/upstream-contribute.md +14 -14
- package/.agent-src/commands/video/from-script.md +1 -1
- package/.agent-src/commands/video/scene.md +1 -1
- package/.agent-src/commands/video/stitch.md +1 -1
- package/.agent-src/commands/video/storyboard.md +1 -1
- package/.agent-src/commands/video.md +1 -1
- package/.agent-src/contexts/augment-infrastructure.md +1 -1
- package/.agent-src/contexts/authority/commit-mechanics.md +15 -0
- package/.agent-src/contexts/authority/kernel-rule-edits.md +3 -3
- package/.agent-src/contexts/authority/scope-mechanics.md +1 -1
- package/.agent-src/contexts/communication/rules-auto/augment-source-of-truth-mechanics.md +28 -28
- package/.agent-src/contexts/communication/rules-auto/skill-quality-mechanics.md +4 -4
- package/.agent-src/contexts/communication/rules-auto/think-before-action-mechanics.md +2 -2
- package/.agent-src/contexts/contracts/artifact-engagement-flow.md +6 -6
- package/.agent-src/contexts/contracts/command-suggestion-flow.md +3 -3
- package/.agent-src/contexts/contracts/emergency-triage-block.md +4 -4
- package/.agent-src/contexts/contracts/frugality-charter.md +3 -3
- package/.agent-src/contexts/documentation-hierarchy.md +14 -7
- package/.agent-src/contexts/execution/autonomy-examples.md +1 -1
- package/.agent-src/contexts/execution/cheap-question-mechanics.md +39 -2
- package/.agent-src/contexts/execution/roadmap-process-loop.md +28 -5
- package/.agent-src/contexts/override-system.md +5 -5
- package/.agent-src/ghostwriter/fictional-fixture-v1.md +1 -1
- package/.agent-src/personas/advisors/first-principles.md +1 -1
- package/.agent-src/personas/hollywood-director.md +1 -1
- package/.agent-src/rules/architecture.md +5 -1
- package/.agent-src/rules/augment-edit-discipline.md +5 -5
- package/.agent-src/rules/augment-source-of-truth.md +15 -15
- package/.agent-src/rules/commit-conventions.md +1 -1
- package/.agent-src/rules/commit-policy.md +10 -0
- package/.agent-src/rules/domain-adoption-policy.md +3 -3
- package/.agent-src/rules/fast-path-marker-visibility.md +3 -3
- package/.agent-src/rules/finance-safety-floor.md +1 -1
- package/.agent-src/rules/framework-neutrality-in-generic-skills.md +8 -8
- package/.agent-src/rules/git-history-discipline.md +1 -1
- package/.agent-src/rules/improve-before-implement.md +2 -2
- package/.agent-src/rules/language-and-tone.md +2 -2
- package/.agent-src/rules/media-governance-routing.md +5 -5
- package/.agent-src/rules/no-attribution-footers.md +1 -0
- package/.agent-src/rules/no-cheap-questions.md +3 -0
- package/.agent-src/rules/no-decorative-emojis-in-git-surfaces.md +111 -0
- package/.agent-src/rules/no-pr-progress-comments.md +118 -0
- package/.agent-src/rules/no-roadmap-references.md +3 -3
- package/.agent-src/rules/non-destructive-by-default.md +1 -1
- package/.agent-src/rules/persona-governance.md +3 -3
- package/.agent-src/rules/preservation-guard.md +15 -15
- package/.agent-src/rules/roadmap-ci-steps-policy.md +7 -3
- package/.agent-src/rules/rule-type-governance.md +1 -1
- package/.agent-src/rules/skill-quality.md +1 -1
- package/.agent-src/rules/{caveman-speak.md → telegraph-speak.md} +15 -15
- package/.agent-src/rules/token-optimizer-maintenance.md +6 -6
- package/.agent-src/skills/agent-docs-writing/SKILL.md +17 -11
- package/.agent-src/skills/agents-md-thin-root/SKILL.md +9 -9
- package/.agent-src/skills/check-refs/SKILL.md +2 -2
- package/.agent-src/skills/code-refactoring/SKILL.md +2 -2
- package/.agent-src/skills/command-writing/SKILL.md +19 -19
- package/.agent-src/skills/comp-banding/SKILL.md +1 -1
- package/.agent-src/skills/condense-memory/SKILL.md +131 -0
- package/.agent-src/skills/context-authoring/SKILL.md +2 -2
- package/.agent-src/skills/context-document/SKILL.md +5 -3
- package/.agent-src/skills/copilot-agents-optimization/SKILL.md +3 -3
- package/.agent-src/skills/description-assist/SKILL.md +2 -2
- package/.agent-src/skills/git-workflow/SKILL.md +1 -1
- package/.agent-src/skills/guideline-writing/SKILL.md +5 -5
- package/.agent-src/skills/learning-to-rule-or-skill/SKILL.md +4 -4
- package/.agent-src/skills/lint-skills/SKILL.md +3 -3
- package/.agent-src/skills/md-language-check/SKILL.md +2 -2
- package/.agent-src/skills/module-detect-on-the-fly/SKILL.md +138 -0
- package/.agent-src/skills/module-management/SKILL.md +166 -94
- package/.agent-src/skills/override-management/SKILL.md +1 -1
- package/.agent-src/skills/persona-writing/SKILL.md +5 -5
- package/.agent-src/skills/positioning-strategy/SKILL.md +1 -1
- package/.agent-src/skills/project-docs/SKILL.md +6 -4
- package/.agent-src/skills/readme-reviewer/SKILL.md +2 -2
- package/.agent-src/skills/roadmap-management/SKILL.md +13 -1
- package/.agent-src/skills/roadmap-writing/SKILL.md +4 -2
- package/.agent-src/skills/rule-refactor/SKILL.md +5 -5
- package/.agent-src/skills/rule-writing/SKILL.md +18 -18
- package/.agent-src/skills/script-writing/SKILL.md +1 -1
- package/.agent-src/skills/skill-improvement-pipeline/SKILL.md +6 -6
- package/.agent-src/skills/skill-management/SKILL.md +21 -21
- package/.agent-src/skills/skill-reviewer/SKILL.md +2 -2
- package/.agent-src/skills/skill-writing/SKILL.md +8 -8
- package/.agent-src/skills/skill-writing/evals/triggers.json +1 -1
- package/.agent-src/skills/token-optimizer/SKILL.md +4 -4
- package/.agent-src/skills/unit-economics-modeling/SKILL.md +1 -1
- package/.agent-src/skills/upstream-contribute/SKILL.md +17 -17
- package/.agent-src/templates/AGENTS.md +1 -0
- package/.agent-src/templates/agent-settings.md +24 -13
- package/.agent-src/templates/agents/agent-project-settings.example.yml +61 -2
- package/.agent-src/templates/command.md +5 -5
- package/.agent-src/templates/contexts.md +1 -1
- package/.agent-src/templates/copilot-instructions.md +8 -8
- package/.agent-src/templates/features.md +1 -1
- package/.agent-src/templates/hooks/pre-commit-frontmatter +2 -2
- package/.agent-src/templates/hooks/pre-commit-roadmap-progress +3 -3
- package/.agent-src/templates/persona.md +2 -2
- package/.agent-src/templates/roadmaps.md +1 -1
- package/.agent-src/templates/rule.md +13 -13
- package/.agent-src/templates/scripts/memory_lookup.py +1 -1
- package/.agent-src/templates/scripts/memory_status.py +2 -2
- package/.agent-src/templates/scripts/work_engine/_lib/agent_settings.py +195 -1
- package/.agent-src/templates/scripts/work_engine/orchestration.py +1 -1
- package/.agent-src/templates/skill-archive-note.md +5 -5
- package/.agent-src/templates/skill.md +1 -1
- package/.claude-plugin/marketplace.json +4 -4
- package/AGENTS.md +16 -17
- package/CHANGELOG.md +216 -3
- package/CONTRIBUTING.md +31 -12
- package/README.md +21 -12
- package/config/agent-settings.template.yml +22 -2
- package/config/discovery/unassigned-artefacts.yml +24 -24
- package/config/profiles/full.ini +1 -1
- package/dist/cli/agent-config.js +52 -3
- package/dist/cli/agent-config.js.map +1 -1
- package/dist/cli/commands/uiServe.js +9 -0
- package/dist/cli/commands/uiServe.js.map +1 -1
- package/dist/cli/registry.js +2 -1
- package/dist/cli/registry.js.map +1 -1
- package/dist/discovery/deprecation-report.md +1 -1
- package/dist/discovery/discovery-manifest.json +649 -606
- package/dist/discovery/discovery-manifest.json.sha256 +1 -1
- package/dist/discovery/discovery-manifest.summary.md +4 -4
- package/dist/discovery/orphan-report.md +1 -1
- package/dist/discovery/packs.json +439 -437
- package/dist/discovery/trust-report.md +5 -5
- package/dist/discovery/workspaces.json +450 -448
- package/dist/install/atomic.js +92 -0
- package/dist/install/atomic.js.map +1 -0
- package/dist/install/conflict.js +196 -0
- package/dist/install/conflict.js.map +1 -0
- package/dist/install/detect.js +218 -0
- package/dist/install/detect.js.map +1 -0
- package/dist/install/paths.js +82 -0
- package/dist/install/paths.js.map +1 -0
- package/dist/install/plan.js +157 -0
- package/dist/install/plan.js.map +1 -0
- package/dist/install/txlog.js +140 -0
- package/dist/install/txlog.js.map +1 -0
- package/dist/install/types.js +19 -0
- package/dist/install/types.js.map +1 -0
- package/dist/install/wizard-plan.js +184 -0
- package/dist/install/wizard-plan.js.map +1 -0
- package/dist/mcp/registry-manifest.json +4 -4
- package/dist/router.json +67 -19
- package/dist/server/app.js +6 -0
- package/dist/server/app.js.map +1 -1
- package/dist/server/routes/install.js +358 -0
- package/dist/server/routes/install.js.map +1 -0
- package/dist/server/routes/wizard.js +468 -32
- package/dist/server/routes/wizard.js.map +1 -1
- package/dist/server/routes/workspace.js +396 -0
- package/dist/server/routes/workspace.js.map +1 -0
- package/dist/server/schemas/settings.js +5 -3
- package/dist/server/schemas/settings.js.map +1 -1
- package/dist/ui/assets/index-BDAhhpDV.js +40 -0
- package/dist/ui/assets/index-BDAhhpDV.js.map +1 -0
- package/dist/ui/assets/index-BXZILUxe.css +1 -0
- package/dist/ui/index.html +2 -2
- package/docs/MIGRATION.md +1 -1
- package/docs/adrs/cost/0001-hard-stop-hook.md +1 -1
- package/docs/adrs/router/0001-three-tier-routing.md +4 -4
- package/docs/adrs/schema/0001-json-schema-frontmatter.md +1 -1
- package/docs/adrs/smoke/0001-per-tier-smoke-scripts.md +4 -4
- package/docs/adrs/{caveman → telegraph}/0001-default-off-until-bench.md +9 -9
- package/docs/adrs/telegraph/README.md +9 -0
- package/docs/architecture/augment-projection.md +4 -4
- package/docs/architecture/claude-bundle.md +1 -1
- package/docs/architecture/current-onboard-baseline.md +3 -3
- package/docs/architecture/multi-tool-projection.md +10 -10
- package/docs/architecture/source-projection.md +27 -27
- package/docs/architecture.md +19 -15
- package/docs/archive/CHANGELOG-pre-2.11.0.md +2 -2
- package/docs/archive/CHANGELOG-pre-2.15.0.md +3 -3
- package/docs/archive/CHANGELOG-pre-2.16.0.md +1 -1
- package/docs/archive/CHANGELOG-pre-2.2.0.md +70 -70
- package/docs/archive/CHANGELOG-pre-2.20.0.md +2 -2
- package/docs/archive/CHANGELOG-pre-2.25.0.md +15 -15
- package/docs/archive/CHANGELOG-pre-3.0.0.md +4 -4
- package/docs/archive/CHANGELOG-pre-3.1.0.md +2 -2
- package/docs/archive/CHANGELOG-pre-3.2.0.md +3 -3
- package/docs/benchmark.md +65 -0
- package/docs/benchmarks.md +16 -16
- package/docs/catalog.md +17 -15
- package/docs/contracts/CHANGELOG-conventions.md +1 -1
- package/docs/contracts/STABILITY.md +2 -2
- package/docs/contracts/adoption-signal-floor.md +110 -0
- package/docs/contracts/adr-chat-history-split.md +4 -4
- package/docs/contracts/adr-command-suggestion.md +4 -4
- package/docs/contracts/adr-gtm-context-spine.md +1 -1
- package/docs/contracts/adr-implement-ticket-runtime.md +4 -4
- package/docs/contracts/adr-install-user-type-axis.md +1 -1
- package/docs/contracts/adr-layout.md +2 -2
- package/docs/contracts/adr-product-ui-track.md +10 -10
- package/docs/contracts/adr-user-types-axis.md +3 -3
- package/docs/contracts/adr-wing4-context-spine.md +1 -1
- package/docs/contracts/agent-memory-contract.md +3 -3
- package/docs/contracts/agents-md-tech-stack.md +2 -2
- package/docs/contracts/ai-council-config.md +2 -2
- package/docs/contracts/at-rest-encryption.md +4 -0
- package/docs/contracts/audit-log-v1.md +1 -1
- package/docs/contracts/benchmark-ab-contract.md +101 -0
- package/docs/contracts/benchmark-corpus-spec.md +1 -1
- package/docs/contracts/branch-protection-policy.md +98 -0
- package/docs/contracts/ci-cost-budget.md +106 -0
- package/docs/contracts/ci-green-floor.md +141 -0
- package/docs/contracts/command-clusters.md +6 -6
- package/docs/contracts/command-surface-tiers.md +2 -2
- package/docs/contracts/command-taxonomy.md +2 -2
- package/docs/contracts/{compression-default-kill-criterion.md → condensation-default-kill-criterion.md} +29 -29
- package/docs/contracts/config-presets.md +1 -1
- package/docs/contracts/context-paths.md +3 -3
- package/docs/contracts/context-spine.md +1 -1
- package/docs/contracts/cost-summary-schema.md +12 -12
- package/docs/contracts/cross-wing-handoff.md +4 -4
- package/docs/contracts/daily-workspace.md +4 -0
- package/docs/contracts/decision-trace-v1.md +2 -2
- package/docs/contracts/discovery-manifest.md +4 -4
- package/docs/contracts/explain-modes.md +4 -0
- package/docs/contracts/file-ownership-matrix.json +3493 -3318
- package/docs/contracts/file-ownership-matrix.md +3 -3
- package/docs/contracts/frontmatter-contract.md +4 -4
- package/docs/contracts/ghostwriter-schema.md +3 -3
- package/docs/contracts/gui-wizard.md +110 -97
- package/docs/contracts/harness-expectations.md +123 -0
- package/docs/contracts/host-agent-protocol.md +4 -0
- package/docs/contracts/implement-ticket-flow.md +9 -9
- package/docs/contracts/install-scopes.md +77 -0
- package/docs/contracts/iron-law-overrides.txt +1 -1
- package/docs/contracts/kernel-membership.md +26 -26
- package/docs/contracts/linear-ai-rules-inclusion.md +1 -1
- package/docs/contracts/linter-structural-model.md +2 -2
- package/docs/contracts/load-context-budget-model.md +4 -4
- package/docs/contracts/load-context-schema.md +13 -13
- package/docs/contracts/local-analytics.md +4 -0
- package/docs/contracts/local-knowledge-ingestion.md +1 -1
- package/docs/contracts/mcp-cloud-scope.md +2 -2
- package/docs/contracts/mcp-phase-1-scope.md +3 -3
- package/docs/contracts/measurement-baseline.md +5 -5
- package/docs/contracts/mental-models.md +30 -30
- package/docs/contracts/multi-tool-projection-fidelity.md +4 -4
- package/docs/contracts/namespace.md +4 -4
- package/docs/contracts/orchestration-dsl-v1.md +7 -7
- package/docs/contracts/package-self-orientation.md +12 -12
- package/docs/contracts/persona-schema.md +6 -6
- package/docs/contracts/pilot/language-and-tone.md +1 -1
- package/docs/contracts/plain-language-surface.md +117 -0
- package/docs/contracts/profile-system.md +3 -3
- package/docs/contracts/release-pr-gating.md +103 -0
- package/docs/contracts/role-experience.md +3 -3
- package/docs/contracts/rule-classification.md +13 -13
- package/docs/contracts/rule-interactions.md +4 -4
- package/docs/contracts/rule-interactions.yml +30 -30
- package/docs/contracts/rule-priority-hierarchy.md +13 -13
- package/docs/contracts/rule-router.md +2 -2
- package/docs/contracts/safety-model.md +1 -1
- package/docs/contracts/skill-distribution-channels.md +61 -0
- package/docs/contracts/skill-domains.md +2 -2
- package/docs/contracts/smoke-contracts.md +5 -5
- package/docs/contracts/telegraph-telemetry.md +83 -0
- package/docs/contracts/trust-and-safety.md +5 -5
- package/docs/contracts/ui-stack-extension.md +7 -7
- package/docs/contracts/ui-track-flow.md +9 -9
- package/docs/contracts/user-type-schema.md +4 -4
- package/docs/contracts/workflow-packs.md +4 -4
- package/docs/contracts/workspace-documents.md +4 -0
- package/docs/customization.md +28 -8
- package/docs/decisions/ADR-001-kernel-swap-deferred.md +6 -6
- package/docs/decisions/ADR-002-kernel-bucket-overrides.md +11 -11
- package/docs/decisions/ADR-003-flat-cluster-subs-and-colon-syntax.md +2 -2
- package/docs/decisions/ADR-004-rule-governance-pruning.md +4 -4
- package/docs/decisions/ADR-005-subagent-worktrees.md +7 -7
- package/docs/decisions/ADR-011-domain-pack-readiness.md +6 -6
- package/docs/decisions/ADR-013-discovery-frontmatter-contract.md +3 -3
- package/docs/decisions/ADR-015-discovery-manifest-contract.md +3 -3
- package/docs/decisions/ADR-017-monorepo-physical-layout.md +10 -10
- package/docs/decisions/ADR-018-trust-and-safety-layer.md +6 -6
- package/docs/decisions/ADR-019-router-json-dist-location.md +2 -2
- package/docs/decisions/ADR-020-global-only-consumer-scope.md +2 -2
- package/docs/decisions/ADR-021-deployment-shape.md +3 -3
- package/docs/decisions/ADR-022-daily-workspace-decomposition.md +1 -1
- package/docs/decisions/ADR-027-changelog-machine-vs-manual.md +2 -2
- package/docs/decisions/ADR-028-root-layout.md +7 -7
- package/docs/decisions/ADR-029-multi-workspace-deferred.md +2 -2
- package/docs/decisions/ADR-rule-kernel-and-router.md +5 -5
- package/docs/deploy/connector-setup.md +2 -2
- package/docs/deploy/policy-cookbook.md +2 -2
- package/docs/deploy/team-deployment-posture.md +20 -0
- package/docs/development.md +17 -17
- package/docs/distribution/registries.md +32 -0
- package/docs/distribution/registry-submissions.md +85 -0
- package/docs/distribution/telemetry-schema.md +1 -1
- package/docs/getting-started-by-role.md +45 -3
- package/docs/getting-started.md +2 -2
- package/docs/guidelines/agent-infra/5w2h-analysis.md +3 -3
- package/docs/guidelines/agent-infra/ask-when-uncertain-demos.md +1 -1
- package/docs/guidelines/agent-infra/asking-and-brevity-examples.md +3 -3
- package/docs/guidelines/agent-infra/carve-out-predicates.md +3 -3
- package/docs/guidelines/agent-infra/critical-thinking.md +4 -4
- package/docs/guidelines/agent-infra/direct-answers-demos.md +1 -1
- package/docs/guidelines/agent-infra/first-principles.md +2 -2
- package/docs/guidelines/agent-infra/inversion-thinking.md +5 -5
- package/docs/guidelines/agent-infra/layered-settings.md +56 -2
- package/docs/guidelines/agent-infra/mental-models.md +3 -3
- package/docs/guidelines/agent-infra/roadmap-progress-mechanics.md +2 -2
- package/docs/guidelines/agent-infra/rule-type-governance.md +1 -1
- package/docs/guidelines/agent-infra/scqa-framework.md +5 -5
- package/docs/guidelines/agent-infra/self-improvement-pipeline.md +2 -2
- package/docs/guidelines/agent-infra/six-hats.md +3 -3
- package/docs/guidelines/agent-infra/skill-quality-checklist.md +5 -5
- package/docs/guidelines/agent-infra/systems-thinking.md +1 -1
- package/docs/guidelines/agent-infra/verify-before-complete-demos.md +1 -1
- package/docs/guidelines/augment-portability-patterns.md +4 -4
- package/docs/guidelines/cross-role-handoff.md +2 -2
- package/docs/guidelines/php/php-coding-patterns.md +1 -1
- package/docs/guidelines/prompt-templates.md +6 -6
- package/docs/maintainers/dev-mode.md +1 -1
- package/docs/mcp.md +1 -1
- package/docs/parity/bench.json +3 -3
- package/docs/parity/ruflo.md +2 -2
- package/docs/profiles.md +11 -11
- package/docs/quality.md +11 -11
- package/docs/safety.md +3 -3
- package/docs/setup/mcp-client-config.md +1 -1
- package/docs/setup/mcp-r2-bootstrap.md +1 -1
- package/docs/setup/mcp-server-docker.md +3 -3
- package/docs/setup/per-ide/windsurf.md +1 -1
- package/docs/skills-catalog.md +8 -7
- package/docs/troubleshooting.md +1 -1
- package/docs/walkthroughs/daily-workspace-a11y.md +87 -0
- package/llms.txt +7 -6
- package/package.json +1 -1
- package/scripts/__pycache__/validate_frontmatter.cpython-312.pyc +0 -0
- package/scripts/_archive/README.md +2 -2
- package/scripts/_archive/_backfill_skill_domains.py +3 -3
- package/scripts/_archive/_bootstrap_tier_frontmatter.py +3 -3
- package/scripts/_archive/_p43_bodies.py +10 -10
- package/scripts/_archive/{_p43_compress.py → _p43_condense.py} +5 -5
- package/scripts/_archive/_p4_migrate.py +7 -7
- package/scripts/_archive/_phase2_shim_helper.py +1 -1
- package/scripts/_archive/_pilot_council_question.py +5 -5
- package/scripts/_cli/explain_last/inputs.py +1 -1
- package/scripts/_lib/__pycache__/__init__.cpython-312.pyc +0 -0
- package/scripts/_lib/__pycache__/agent_src.cpython-312.pyc +0 -0
- package/scripts/_lib/agent_settings.py +195 -1
- package/scripts/_lib/agent_src.py +19 -19
- package/scripts/_lib/bench_ab_cache.py +162 -0
- package/scripts/_lib/bench_ab_scoring.py +209 -0
- package/scripts/_lib/{bench_caveman.py → bench_telegraph.py} +21 -21
- package/scripts/_lib/{bench_caveman_report.py → bench_telegraph_report.py} +21 -21
- package/scripts/_lib/claude_desktop_bundler.py +5 -5
- package/scripts/_lib/module_detection.py +223 -0
- package/scripts/_lib/scope_guard.sh +162 -0
- package/scripts/_phase4_bucket.py +3 -3
- package/scripts/_pilot_measure.py +4 -4
- package/scripts/_tmp_scan_framework_leakage.py +1 -1
- package/scripts/adoption_report.py +195 -0
- package/scripts/adoption_snapshot.py +219 -0
- package/scripts/adoption_status.py +166 -0
- package/scripts/ai-video/lib/parse-blueprint.sh +1 -1
- package/scripts/ai_council/advisors.py +5 -5
- package/scripts/ai_council/compile_corpus.py +1 -1
- package/scripts/ai_council/one_off_archive/2026-05/_one_off_budget_v2_audit.py +3 -3
- package/scripts/ai_council/one_off_archive/2026-05/_one_off_context_layer_v1_review.py +2 -2
- package/scripts/ai_council/one_off_archive/2026-05/_one_off_inject_quiet_flag.py +1 -1
- package/scripts/ai_council/one_off_archive/2026-05/_one_off_measure_v2.sh +1 -1
- package/scripts/ai_council/one_off_archive/2026-05/_one_off_measure_verbosity.sh +1 -1
- package/scripts/ai_council/one_off_archive/2026-05/_one_off_nondestructive_inline_audit.py +3 -3
- package/scripts/ai_council/one_off_archive/2026-05/_one_off_per_task.sh +1 -1
- package/scripts/ai_council/one_off_archive/2026-05/_one_off_phase6_trigger_jaccard.py +1 -1
- package/scripts/ai_council/one_off_archive/2026-05/_one_off_phase_2a_budget_rebalance.py +6 -6
- package/scripts/ai_council/one_off_archive/2026-05/_one_off_rebalancing_audit.py +1 -1
- package/scripts/ai_council/one_off_archive/2026-05/_one_off_tier_retrofit.py +6 -6
- package/scripts/annotate_discovery.py +13 -13
- package/scripts/apply_modules_config.py +290 -0
- package/scripts/audit_adr_coverage.py +2 -2
- package/scripts/audit_auto_rules.py +2 -2
- package/scripts/audit_cloud_compatibility.py +3 -3
- package/scripts/audit_command_surface.py +9 -9
- package/scripts/audit_likelihood.py +2 -2
- package/scripts/audit_user_type_axis.py +2 -2
- package/scripts/bench_ab_cache_dispatch.py +68 -0
- package/scripts/bench_ab_clone.py +170 -0
- package/scripts/bench_ab_diff.py +227 -0
- package/scripts/bench_ab_integrity.py +143 -0
- package/scripts/bench_ab_run.py +235 -0
- package/scripts/bench_ab_task_runner.py +369 -0
- package/scripts/bench_ab_tracka_run.py +202 -0
- package/scripts/{bench_compress_memory.py → bench_condense_memory.py} +16 -16
- package/scripts/bench_run.py +33 -33
- package/scripts/bench_runner.py +2 -2
- package/scripts/bootstrap.sh +99 -0
- package/scripts/build_cloud_bundle.py +6 -6
- package/scripts/build_discovery_manifest.py +7 -7
- package/scripts/build_linear_digest.py +3 -3
- package/scripts/build_rule_trigger_matrix.py +8 -8
- package/scripts/chat_history.py +5 -5
- package/scripts/check_always_budget.py +11 -5
- package/scripts/check_augment_description_cap.py +3 -3
- package/scripts/check_cluster_patterns.py +2 -2
- package/scripts/check_command_count_messaging.py +3 -3
- package/scripts/{check_compression.py → check_condensation.py} +34 -34
- package/scripts/{check_compressed_paths.py → check_condensed_paths.py} +8 -8
- package/scripts/check_context_paths.py +7 -7
- package/scripts/check_council_layout.py +2 -2
- package/scripts/check_council_references.py +9 -9
- package/scripts/check_iron_law_prominence.py +2 -2
- package/scripts/check_kernel_rule_bundle.py +2 -2
- package/scripts/check_module_management_neutral.py +149 -0
- package/scripts/check_no_roadmap_refs.py +9 -9
- package/scripts/check_portability.py +3 -3
- package/scripts/check_public_catalog_links.py +4 -4
- package/scripts/check_references.py +7 -6
- package/scripts/check_release_pr_shape.py +112 -0
- package/scripts/check_reply_consistency.py +3 -3
- package/scripts/check_safety_floor_untouched.py +1 -1
- package/scripts/check_template_pin_drift.py +5 -5
- package/scripts/check_token_optimizer_freshness.py +3 -3
- package/scripts/ci_status.py +301 -0
- package/scripts/ci_time_ratio.py +1 -1
- package/scripts/cleanup_other_scope.sh +146 -0
- package/scripts/compile_router.py +10 -10
- package/scripts/{compress.py → condense.py} +64 -64
- package/scripts/condense.sh +18 -0
- package/scripts/{compress_memory.py → condense_memory.py} +33 -33
- package/scripts/config/presets.py +2 -2
- package/scripts/config/profiles.py +1 -1
- package/scripts/cost_by_conversation.py +3 -3
- package/scripts/cost_summary.py +7 -7
- package/scripts/count_token_optimizer_usage.sh +1 -1
- package/scripts/gen_discovery_baseline.py +5 -5
- package/scripts/generate_index.py +6 -6
- package/scripts/generate_ownership_matrix.py +10 -10
- package/scripts/generate_pack_manifests.py +1 -1
- package/scripts/ghostwriter_fixture_allowlist.txt +1 -1
- package/scripts/install +3 -3
- package/scripts/install-hooks.sh +6 -6
- package/scripts/install.py +273 -45
- package/scripts/install.sh +187 -1
- package/scripts/inventory_frontmatter.py +2 -2
- package/scripts/iron_law_sha.py +3 -3
- package/scripts/lint_agents_layout.py +14 -7
- package/scripts/lint_agents_md.py +4 -4
- package/scripts/lint_archived_skills.py +3 -3
- package/scripts/lint_artefact_frontmatter.py +2 -2
- package/scripts/lint_bench_ab.py +172 -0
- package/scripts/lint_bench_corpus.py +1 -1
- package/scripts/lint_command_tiers.py +5 -5
- package/scripts/lint_context_spine_usage.py +1 -1
- package/scripts/lint_framework_leakage.py +7 -7
- package/scripts/lint_framework_leakage_allowlist.json +152 -84
- package/scripts/lint_ghostwriter_source.py +3 -3
- package/scripts/lint_handoffs.py +1 -1
- package/scripts/lint_load_context.py +11 -11
- package/scripts/lint_media_policy_linkage.py +5 -5
- package/scripts/lint_namespace.py +1 -1
- package/scripts/lint_no_new_atomic_commands.py +2 -2
- package/scripts/lint_orchestration_dsl.py +1 -1
- package/scripts/lint_pack_boundaries.py +2 -2
- package/scripts/lint_persona_governance.py +4 -4
- package/scripts/lint_role_experiences.py +237 -0
- package/scripts/lint_rule_interactions.py +2 -2
- package/scripts/lint_rule_tiers.py +1 -1
- package/scripts/lint_trust_coherence.py +2 -2
- package/scripts/mcp_registry_submit.sh +187 -0
- package/scripts/mcp_server/tools.py +1 -1
- package/scripts/measure_frugality_savings.py +10 -10
- package/scripts/measure_patterns.py +1 -1
- package/scripts/measure_projection_bytes.py +5 -5
- package/scripts/measure_rule_budget.py +3 -3
- package/scripts/measure_skill_reduction.py +1 -1
- package/scripts/memory_lookup.py +1 -1
- package/scripts/memory_status.py +2 -2
- package/scripts/migrate_command_suggestions.py +3 -3
- package/scripts/mine_session.py +1 -1
- package/scripts/move_artefact.py +3 -3
- package/scripts/new_skill.py +2 -2
- package/scripts/pack_mcp_content.py +9 -9
- package/scripts/plan_physical_move.py +6 -6
- package/scripts/print_required_checks.py +196 -0
- package/scripts/probe_skill_registration.py +413 -0
- package/scripts/propose_modules_config.py +145 -0
- package/scripts/prototype_lint_contradictions.py +1 -1
- package/scripts/recruit_preflight.sh +152 -0
- package/scripts/refine_ticket_detect.py +3 -3
- package/scripts/release.py +20 -0
- package/scripts/render_benchmark_md.py +308 -0
- package/scripts/roadmap_progress_hook.py +1 -1
- package/scripts/run_skill_evals.py +2 -2
- package/scripts/runtime_registry.py +4 -4
- package/scripts/schemas/command.schema.json +4 -4
- package/scripts/schemas/rule.schema.json +5 -5
- package/scripts/schemas/skill.schema.json +3 -3
- package/scripts/schemas/user-type.schema.json +1 -1
- package/scripts/score_skill_selection.py +1 -1
- package/scripts/skill_collision_clusters.py +2 -2
- package/scripts/skill_linter.py +81 -81
- package/scripts/skill_overlap.py +5 -5
- package/scripts/skill_tools/audit_persona_coverage.py +2 -2
- package/scripts/skill_tools/audit_user_type_coverage.py +2 -2
- package/scripts/skill_tools/run_block_d_eval.py +1 -1
- package/scripts/skill_tools/score_skill_relevance.py +1 -1
- package/scripts/skill_tools/suggest_skill_for_task.py +1 -1
- package/scripts/skill_trigger_eval.py +3 -3
- package/scripts/smoke/kernel.sh +7 -1
- package/scripts/smoke/router.sh +5 -5
- package/scripts/smoke/skills.sh +1 -1
- package/scripts/smoke_quickstart.py +1 -1
- package/scripts/snapshot_agent_outputs.py +3 -3
- package/scripts/spotcheck_thin_root.py +1 -1
- package/scripts/{caveman_stats.py → telegraph_stats.py} +18 -18
- package/scripts/update_counts.py +1 -1
- package/scripts/validate_decision_engine.py +1 -1
- package/scripts/validate_frontmatter.py +1 -1
- package/scripts/validate_safe_paths.py +3 -3
- package/scripts/{validate_caveman_carveouts.py → validate_telegraph_carveouts.py} +7 -7
- package/scripts/verify_roadmap_closure.py +6 -6
- package/templates/consumer-settings/ONBOARDING.md +41 -0
- package/.agent-src/commands/install-via-agent.md +0 -129
- package/.agent-src/skills/compress-memory/SKILL.md +0 -131
- package/dist/ui/assets/index-D-DY1ywI.js +0 -35
- package/dist/ui/assets/index-D-DY1ywI.js.map +0 -1
- package/dist/ui/assets/index-Dqfhmg-d.css +0 -1
- package/docs/adrs/caveman/README.md +0 -9
- package/docs/contracts/caveman-telemetry.md +0 -83
- package/scripts/compress.sh +0 -18
|
@@ -0,0 +1,196 @@
|
|
|
1
|
+
#!/usr/bin/env python3
|
|
2
|
+
"""Print the expected required-check set for the current branch.
|
|
3
|
+
|
|
4
|
+
Contract: `docs/contracts/branch-protection-policy.md`. Per-PR-shape
|
|
5
|
+
required-check floor — feature PR vs release PR vs docs-only PR. The
|
|
6
|
+
script resolves the PR shape locally so the maintainer can sanity-check
|
|
7
|
+
before pushing, without round-tripping through the GitHub UI.
|
|
8
|
+
|
|
9
|
+
Resolution order:
|
|
10
|
+
|
|
11
|
+
1. `--branch <name>` flag — explicit override.
|
|
12
|
+
2. Current git branch — `git rev-parse --abbrev-ref HEAD`.
|
|
13
|
+
3. Fail with exit 2 (usage error).
|
|
14
|
+
|
|
15
|
+
PR-shape classification:
|
|
16
|
+
|
|
17
|
+
release — branch matches `^release/\\d+\\.\\d+\\.\\d+$` AND
|
|
18
|
+
`check_release_pr_shape.py` reports SHAPE-CLEAN for the
|
|
19
|
+
local diff against `--base` (default `origin/main`).
|
|
20
|
+
docs-only — diff vs base is entirely inside `docs/**` or top-level
|
|
21
|
+
`.md` files (`README.md`, `CHANGELOG.md`,
|
|
22
|
+
`CONTRIBUTING.md`, `AGENTS.md`).
|
|
23
|
+
feature — everything else (default).
|
|
24
|
+
|
|
25
|
+
The script never invokes `gh` and never touches the network — it works
|
|
26
|
+
offline against the local git index so pre-push previews stay fast.
|
|
27
|
+
|
|
28
|
+
Exit codes:
|
|
29
|
+
|
|
30
|
+
0 — printed the expected required-check set.
|
|
31
|
+
1 — release-PR shape detector reported OUT-OF-SHAPE; falls back to
|
|
32
|
+
the feature-PR set, which is also printed, plus a warning.
|
|
33
|
+
2 — usage / environment error.
|
|
34
|
+
"""
|
|
35
|
+
|
|
36
|
+
from __future__ import annotations
|
|
37
|
+
|
|
38
|
+
import argparse
|
|
39
|
+
import re
|
|
40
|
+
import subprocess
|
|
41
|
+
import sys
|
|
42
|
+
from pathlib import Path
|
|
43
|
+
|
|
44
|
+
REPO_ROOT = Path(__file__).resolve().parent.parent
|
|
45
|
+
sys.path.insert(0, str(REPO_ROOT / "scripts"))
|
|
46
|
+
|
|
47
|
+
import check_release_pr_shape as shape # noqa: E402
|
|
48
|
+
|
|
49
|
+
RELEASE_BRANCH_RE = re.compile(r"^release/\d+\.\d+\.\d+$")
|
|
50
|
+
|
|
51
|
+
DOCS_ONLY_ALLOWED_TOP = {
|
|
52
|
+
"README.md",
|
|
53
|
+
"CHANGELOG.md",
|
|
54
|
+
"CONTRIBUTING.md",
|
|
55
|
+
"AGENTS.md",
|
|
56
|
+
"LICENSE",
|
|
57
|
+
"llms.txt",
|
|
58
|
+
}
|
|
59
|
+
|
|
60
|
+
FEATURE_CHECKS = (
|
|
61
|
+
"Consistency",
|
|
62
|
+
"Smoke Contracts",
|
|
63
|
+
"Skill Lint",
|
|
64
|
+
"Tests / install-tests",
|
|
65
|
+
"Tests / install-aux-tests",
|
|
66
|
+
"Tests / python-tests",
|
|
67
|
+
"Tests / node-tests",
|
|
68
|
+
"Public Install Smoke / smoke",
|
|
69
|
+
)
|
|
70
|
+
|
|
71
|
+
RELEASE_CHECKS = (
|
|
72
|
+
"Consistency",
|
|
73
|
+
"Smoke Contracts",
|
|
74
|
+
"Migration Dry-Run",
|
|
75
|
+
"Release Validation / release-shape",
|
|
76
|
+
"Release Validation / changelog-entry",
|
|
77
|
+
"Release Validation / version-consistency",
|
|
78
|
+
"Release Guard (post-tag)",
|
|
79
|
+
)
|
|
80
|
+
|
|
81
|
+
DOCS_ONLY_CHECKS = (
|
|
82
|
+
"Consistency",
|
|
83
|
+
"Smoke Contracts",
|
|
84
|
+
)
|
|
85
|
+
|
|
86
|
+
|
|
87
|
+
def _git(*args: str) -> str:
|
|
88
|
+
return subprocess.run(
|
|
89
|
+
["git", *args], check=True, capture_output=True, text=True
|
|
90
|
+
).stdout.strip()
|
|
91
|
+
|
|
92
|
+
|
|
93
|
+
def current_branch() -> str:
|
|
94
|
+
return _git("rev-parse", "--abbrev-ref", "HEAD")
|
|
95
|
+
|
|
96
|
+
|
|
97
|
+
def diff_files(base: str) -> list[str]:
|
|
98
|
+
out = subprocess.run(
|
|
99
|
+
["git", "diff", "--name-only", f"{base}...HEAD"],
|
|
100
|
+
capture_output=True,
|
|
101
|
+
text=True,
|
|
102
|
+
)
|
|
103
|
+
if out.returncode != 0:
|
|
104
|
+
# Base ref unknown locally — fall back to staged + tracked
|
|
105
|
+
# changes vs HEAD so the dry-run still tells the maintainer
|
|
106
|
+
# something useful instead of crashing.
|
|
107
|
+
out = subprocess.run(
|
|
108
|
+
["git", "diff", "--name-only", "HEAD"],
|
|
109
|
+
check=True,
|
|
110
|
+
capture_output=True,
|
|
111
|
+
text=True,
|
|
112
|
+
)
|
|
113
|
+
return [line for line in out.stdout.splitlines() if line.strip()]
|
|
114
|
+
|
|
115
|
+
|
|
116
|
+
def is_docs_only(files: list[str]) -> bool:
|
|
117
|
+
if not files:
|
|
118
|
+
return False
|
|
119
|
+
for f in files:
|
|
120
|
+
if f in DOCS_ONLY_ALLOWED_TOP:
|
|
121
|
+
continue
|
|
122
|
+
if f.startswith("docs/"):
|
|
123
|
+
continue
|
|
124
|
+
return False
|
|
125
|
+
return True
|
|
126
|
+
|
|
127
|
+
|
|
128
|
+
def classify(branch: str, files: list[str]) -> tuple[str, int]:
|
|
129
|
+
"""Return (shape, exit_code)."""
|
|
130
|
+
if RELEASE_BRANCH_RE.match(branch):
|
|
131
|
+
# Release-shape predicate — re-uses the fail-closed allowlist.
|
|
132
|
+
bad = [f for f in files if not shape._matches(f)]
|
|
133
|
+
if bad:
|
|
134
|
+
print(
|
|
135
|
+
"WARNING: branch matches release/X.Y.Z but diff contains "
|
|
136
|
+
f"{len(bad)} out-of-allowlist file(s):",
|
|
137
|
+
file=sys.stderr,
|
|
138
|
+
)
|
|
139
|
+
for f in bad:
|
|
140
|
+
print(f" - {f}", file=sys.stderr)
|
|
141
|
+
print(
|
|
142
|
+
"Falling back to feature-PR required-check set.",
|
|
143
|
+
file=sys.stderr,
|
|
144
|
+
)
|
|
145
|
+
return "feature", 1
|
|
146
|
+
return "release", 0
|
|
147
|
+
if is_docs_only(files):
|
|
148
|
+
return "docs-only", 0
|
|
149
|
+
return "feature", 0
|
|
150
|
+
|
|
151
|
+
|
|
152
|
+
def print_set(shape_label: str, files: list[str]) -> None:
|
|
153
|
+
table = {
|
|
154
|
+
"feature": FEATURE_CHECKS,
|
|
155
|
+
"release": RELEASE_CHECKS,
|
|
156
|
+
"docs-only": DOCS_ONLY_CHECKS,
|
|
157
|
+
}[shape_label]
|
|
158
|
+
print(f"PR shape: {shape_label} ({len(files)} file(s) in diff)")
|
|
159
|
+
print(f"Required checks ({len(table)}):")
|
|
160
|
+
for name in table:
|
|
161
|
+
print(f" - {name}")
|
|
162
|
+
print()
|
|
163
|
+
print(
|
|
164
|
+
"Contract: docs/contracts/branch-protection-policy.md "
|
|
165
|
+
"(per-PR-shape matrix)"
|
|
166
|
+
)
|
|
167
|
+
|
|
168
|
+
|
|
169
|
+
def main(argv: list[str] | None = None) -> int:
|
|
170
|
+
parser = argparse.ArgumentParser(
|
|
171
|
+
description=(
|
|
172
|
+
"Print the expected required-check set for the current branch."
|
|
173
|
+
),
|
|
174
|
+
)
|
|
175
|
+
parser.add_argument(
|
|
176
|
+
"--branch",
|
|
177
|
+
help="Branch name override (default: current git branch).",
|
|
178
|
+
)
|
|
179
|
+
parser.add_argument(
|
|
180
|
+
"--base",
|
|
181
|
+
default="origin/main",
|
|
182
|
+
help="Base ref for the diff (default: origin/main).",
|
|
183
|
+
)
|
|
184
|
+
args = parser.parse_args(argv)
|
|
185
|
+
|
|
186
|
+
branch = args.branch or current_branch()
|
|
187
|
+
files = diff_files(args.base)
|
|
188
|
+
shape_label, exit_code = classify(branch, files)
|
|
189
|
+
print(f"Branch: {branch}")
|
|
190
|
+
print(f"Base: {args.base}")
|
|
191
|
+
print_set(shape_label, files)
|
|
192
|
+
return exit_code
|
|
193
|
+
|
|
194
|
+
|
|
195
|
+
if __name__ == "__main__":
|
|
196
|
+
sys.exit(main())
|
|
@@ -0,0 +1,413 @@
|
|
|
1
|
+
#!/usr/bin/env python3
|
|
2
|
+
"""Tool-agnostic skill-registration probe.
|
|
3
|
+
|
|
4
|
+
Roadmap: road-to-clean-skill-distribution-channels.md § Phase C.
|
|
5
|
+
Contract: docs/contracts/skill-distribution-channels.md.
|
|
6
|
+
|
|
7
|
+
Surfaces every skill registered for any of the six supported AI tools
|
|
8
|
+
across user-global, project-local, and plugin-manifest sources. Flags
|
|
9
|
+
``DUPLICATE`` (same skill name registered in ≥ 2 sources) and ``DRIFT``
|
|
10
|
+
(same name, different description-hash or version). Both shapes are the
|
|
11
|
+
class of bug that opened road-to-clean-skill-distribution-channels.md
|
|
12
|
+
on 2026-05-25.
|
|
13
|
+
|
|
14
|
+
CLI:
|
|
15
|
+
|
|
16
|
+
python3 scripts/probe_skill_registration.py
|
|
17
|
+
python3 scripts/probe_skill_registration.py --tool=claude --format=json
|
|
18
|
+
python3 scripts/probe_skill_registration.py --strict
|
|
19
|
+
|
|
20
|
+
``--strict`` flips the exit code: 0 if no DUPLICATE / DRIFT findings,
|
|
21
|
+
non-zero otherwise. Without ``--strict`` the script is informational
|
|
22
|
+
(always exits 0).
|
|
23
|
+
"""
|
|
24
|
+
from __future__ import annotations
|
|
25
|
+
|
|
26
|
+
import argparse
|
|
27
|
+
import hashlib
|
|
28
|
+
import json
|
|
29
|
+
import os
|
|
30
|
+
import sys
|
|
31
|
+
from dataclasses import dataclass, field, asdict
|
|
32
|
+
from pathlib import Path
|
|
33
|
+
from typing import Iterable
|
|
34
|
+
|
|
35
|
+
TOOL_IDS = ("claude", "augment", "cursor", "cline", "windsurf", "copilot")
|
|
36
|
+
SCOPE_IDS = ("user", "project")
|
|
37
|
+
|
|
38
|
+
|
|
39
|
+
@dataclass(frozen=True)
|
|
40
|
+
class Registration:
|
|
41
|
+
"""One row in the probe table — a single (skill_id, scope, source) tuple."""
|
|
42
|
+
skill_id: str
|
|
43
|
+
tool: str
|
|
44
|
+
scope: str
|
|
45
|
+
source_path: str
|
|
46
|
+
version: str
|
|
47
|
+
description_snippet: str
|
|
48
|
+
description_hash: str
|
|
49
|
+
|
|
50
|
+
def to_dict(self) -> dict[str, str]:
|
|
51
|
+
return asdict(self)
|
|
52
|
+
|
|
53
|
+
|
|
54
|
+
@dataclass
|
|
55
|
+
class ProbeResult:
|
|
56
|
+
registrations: list[Registration] = field(default_factory=list)
|
|
57
|
+
duplicates: dict[str, list[Registration]] = field(default_factory=dict)
|
|
58
|
+
drift: dict[str, list[Registration]] = field(default_factory=dict)
|
|
59
|
+
|
|
60
|
+
def to_dict(self) -> dict[str, object]:
|
|
61
|
+
return {
|
|
62
|
+
"registrations": [r.to_dict() for r in self.registrations],
|
|
63
|
+
"duplicates": {k: [r.to_dict() for r in v] for k, v in self.duplicates.items()},
|
|
64
|
+
"drift": {k: [r.to_dict() for r in v] for k, v in self.drift.items()},
|
|
65
|
+
}
|
|
66
|
+
|
|
67
|
+
|
|
68
|
+
# ---------------------------------------------------------------------------
|
|
69
|
+
# Frontmatter + version helpers
|
|
70
|
+
# ---------------------------------------------------------------------------
|
|
71
|
+
|
|
72
|
+
def _read_frontmatter(skill_md: Path) -> dict[str, str]:
|
|
73
|
+
"""Minimal YAML frontmatter extractor — no PyYAML dependency."""
|
|
74
|
+
try:
|
|
75
|
+
text = skill_md.read_text(encoding="utf-8", errors="replace")
|
|
76
|
+
except OSError:
|
|
77
|
+
return {}
|
|
78
|
+
if not text.startswith("---\n") and not text.startswith("---\r\n"):
|
|
79
|
+
return {}
|
|
80
|
+
rest = text.split("---", 2)
|
|
81
|
+
if len(rest) < 3:
|
|
82
|
+
return {}
|
|
83
|
+
body = rest[1]
|
|
84
|
+
out: dict[str, str] = {}
|
|
85
|
+
for line in body.splitlines():
|
|
86
|
+
if ":" not in line or line.lstrip().startswith("#"):
|
|
87
|
+
continue
|
|
88
|
+
key, _, value = line.partition(":")
|
|
89
|
+
out[key.strip()] = value.strip().strip('"').strip("'")
|
|
90
|
+
return out
|
|
91
|
+
|
|
92
|
+
|
|
93
|
+
def _hash_desc(desc: str) -> str:
|
|
94
|
+
return hashlib.sha256(desc.encode("utf-8", "replace")).hexdigest()[:12]
|
|
95
|
+
|
|
96
|
+
|
|
97
|
+
def _snippet(desc: str, n: int = 80) -> str:
|
|
98
|
+
desc = desc.strip()
|
|
99
|
+
return desc if len(desc) <= n else desc[: n - 1] + "…"
|
|
100
|
+
|
|
101
|
+
|
|
102
|
+
def _version_at(root: Path) -> str:
|
|
103
|
+
"""Read a 'this install's version' value from package.json / plugin.json."""
|
|
104
|
+
for candidate in (root / "package.json", root / ".augment-plugin" / "plugin.json"):
|
|
105
|
+
if not candidate.is_file():
|
|
106
|
+
continue
|
|
107
|
+
try:
|
|
108
|
+
data = json.loads(candidate.read_text(encoding="utf-8"))
|
|
109
|
+
v = data.get("version")
|
|
110
|
+
if isinstance(v, str) and v:
|
|
111
|
+
return v
|
|
112
|
+
except (OSError, json.JSONDecodeError):
|
|
113
|
+
continue
|
|
114
|
+
return "unknown"
|
|
115
|
+
|
|
116
|
+
|
|
117
|
+
# ---------------------------------------------------------------------------
|
|
118
|
+
# Per-tool readers
|
|
119
|
+
# ---------------------------------------------------------------------------
|
|
120
|
+
|
|
121
|
+
def _iter_skill_md(skills_root: Path) -> Iterable[Path]:
|
|
122
|
+
if not skills_root.is_dir():
|
|
123
|
+
return
|
|
124
|
+
for entry in sorted(skills_root.iterdir()):
|
|
125
|
+
if not entry.is_dir():
|
|
126
|
+
continue
|
|
127
|
+
skill_md = entry / "SKILL.md"
|
|
128
|
+
if skill_md.is_file():
|
|
129
|
+
yield skill_md
|
|
130
|
+
|
|
131
|
+
|
|
132
|
+
def _read_claude(scope: str, root: Path) -> Iterable[Registration]:
|
|
133
|
+
skills = root / ".claude" / "skills"
|
|
134
|
+
for skill_md in _iter_skill_md(skills):
|
|
135
|
+
fm = _read_frontmatter(skill_md)
|
|
136
|
+
name = fm.get("name") or skill_md.parent.name
|
|
137
|
+
desc = fm.get("description", "")
|
|
138
|
+
yield Registration(
|
|
139
|
+
skill_id=name,
|
|
140
|
+
tool="claude",
|
|
141
|
+
scope=scope,
|
|
142
|
+
source_path=str(skill_md),
|
|
143
|
+
version=_version_at(root),
|
|
144
|
+
description_snippet=_snippet(desc),
|
|
145
|
+
description_hash=_hash_desc(desc),
|
|
146
|
+
)
|
|
147
|
+
# Plugin manifest at the same scope. Each entry is a path string into
|
|
148
|
+
# the same .claude/skills/ tree; emit it as a separate "manifest" row so
|
|
149
|
+
# duplicate-detection catches manifest-vs-filesystem double-counting.
|
|
150
|
+
manifest = root / ".claude-plugin" / "marketplace.json"
|
|
151
|
+
if manifest.is_file():
|
|
152
|
+
try:
|
|
153
|
+
data = json.loads(manifest.read_text(encoding="utf-8"))
|
|
154
|
+
plugins = data.get("plugins") or []
|
|
155
|
+
entries: list[str] = []
|
|
156
|
+
for plug in plugins:
|
|
157
|
+
entries.extend(plug.get("skills") or [])
|
|
158
|
+
for entry in entries:
|
|
159
|
+
# The entry is a path string; resolve the name from the path.
|
|
160
|
+
tail = Path(entry).name
|
|
161
|
+
yield Registration(
|
|
162
|
+
skill_id=tail,
|
|
163
|
+
tool="claude",
|
|
164
|
+
scope=f"{scope}-plugin",
|
|
165
|
+
source_path=str(manifest),
|
|
166
|
+
version=_version_at(root),
|
|
167
|
+
description_snippet="(plugin manifest entry)",
|
|
168
|
+
description_hash="manifest",
|
|
169
|
+
)
|
|
170
|
+
except (OSError, json.JSONDecodeError):
|
|
171
|
+
pass
|
|
172
|
+
|
|
173
|
+
|
|
174
|
+
def _read_augment(scope: str, root: Path) -> Iterable[Registration]:
|
|
175
|
+
skills = root / ".augment" / "skills"
|
|
176
|
+
for skill_md in _iter_skill_md(skills):
|
|
177
|
+
fm = _read_frontmatter(skill_md)
|
|
178
|
+
name = fm.get("name") or skill_md.parent.name
|
|
179
|
+
desc = fm.get("description", "")
|
|
180
|
+
yield Registration(
|
|
181
|
+
skill_id=name,
|
|
182
|
+
tool="augment",
|
|
183
|
+
scope=scope,
|
|
184
|
+
source_path=str(skill_md),
|
|
185
|
+
version=_version_at(root),
|
|
186
|
+
description_snippet=_snippet(desc),
|
|
187
|
+
description_hash=_hash_desc(desc),
|
|
188
|
+
)
|
|
189
|
+
|
|
190
|
+
|
|
191
|
+
def _read_cursor(scope: str, root: Path) -> Iterable[Registration]:
|
|
192
|
+
rules = root / ".cursor" / "rules"
|
|
193
|
+
if not rules.is_dir():
|
|
194
|
+
return
|
|
195
|
+
for rule in sorted(rules.glob("*.mdc")):
|
|
196
|
+
fm = _read_frontmatter(rule)
|
|
197
|
+
name = fm.get("name") or rule.stem
|
|
198
|
+
desc = fm.get("description", "")
|
|
199
|
+
yield Registration(
|
|
200
|
+
skill_id=name,
|
|
201
|
+
tool="cursor",
|
|
202
|
+
scope=scope,
|
|
203
|
+
source_path=str(rule),
|
|
204
|
+
version=_version_at(root),
|
|
205
|
+
description_snippet=_snippet(desc),
|
|
206
|
+
description_hash=_hash_desc(desc),
|
|
207
|
+
)
|
|
208
|
+
|
|
209
|
+
|
|
210
|
+
def _read_cline(scope: str, root: Path) -> Iterable[Registration]:
|
|
211
|
+
rules = root / ".clinerules"
|
|
212
|
+
if not rules.is_dir():
|
|
213
|
+
return
|
|
214
|
+
for rule in sorted(rules.glob("*.md")):
|
|
215
|
+
fm = _read_frontmatter(rule)
|
|
216
|
+
name = fm.get("name") or rule.stem
|
|
217
|
+
desc = fm.get("description", "")
|
|
218
|
+
yield Registration(
|
|
219
|
+
skill_id=name,
|
|
220
|
+
tool="cline",
|
|
221
|
+
scope=scope,
|
|
222
|
+
source_path=str(rule),
|
|
223
|
+
version=_version_at(root),
|
|
224
|
+
description_snippet=_snippet(desc),
|
|
225
|
+
description_hash=_hash_desc(desc),
|
|
226
|
+
)
|
|
227
|
+
|
|
228
|
+
|
|
229
|
+
def _read_windsurf(scope: str, root: Path) -> Iterable[Registration]:
|
|
230
|
+
rules = root / ".windsurf" / "rules"
|
|
231
|
+
if not rules.is_dir():
|
|
232
|
+
return
|
|
233
|
+
for rule in sorted(rules.glob("*.md")):
|
|
234
|
+
fm = _read_frontmatter(rule)
|
|
235
|
+
name = fm.get("name") or rule.stem
|
|
236
|
+
desc = fm.get("description", "")
|
|
237
|
+
yield Registration(
|
|
238
|
+
skill_id=name,
|
|
239
|
+
tool="windsurf",
|
|
240
|
+
scope=scope,
|
|
241
|
+
source_path=str(rule),
|
|
242
|
+
version=_version_at(root),
|
|
243
|
+
description_snippet=_snippet(desc),
|
|
244
|
+
description_hash=_hash_desc(desc),
|
|
245
|
+
)
|
|
246
|
+
|
|
247
|
+
|
|
248
|
+
def _read_copilot(scope: str, root: Path) -> Iterable[Registration]:
|
|
249
|
+
# Copilot ships a single file; check both common locations.
|
|
250
|
+
for candidate in (
|
|
251
|
+
root / ".github" / "copilot-instructions.md",
|
|
252
|
+
root / "copilot-instructions.md",
|
|
253
|
+
):
|
|
254
|
+
if not candidate.is_file():
|
|
255
|
+
continue
|
|
256
|
+
try:
|
|
257
|
+
text = candidate.read_text(encoding="utf-8", errors="replace")
|
|
258
|
+
except OSError:
|
|
259
|
+
continue
|
|
260
|
+
snippet = _snippet(text.split("\n", 1)[0] if text else "")
|
|
261
|
+
yield Registration(
|
|
262
|
+
skill_id="copilot-instructions",
|
|
263
|
+
tool="copilot",
|
|
264
|
+
scope=scope,
|
|
265
|
+
source_path=str(candidate),
|
|
266
|
+
version=_version_at(root),
|
|
267
|
+
description_snippet=snippet,
|
|
268
|
+
description_hash=_hash_desc(text),
|
|
269
|
+
)
|
|
270
|
+
|
|
271
|
+
|
|
272
|
+
TOOL_READERS = {
|
|
273
|
+
"claude": _read_claude,
|
|
274
|
+
"augment": _read_augment,
|
|
275
|
+
"cursor": _read_cursor,
|
|
276
|
+
"cline": _read_cline,
|
|
277
|
+
"windsurf": _read_windsurf,
|
|
278
|
+
"copilot": _read_copilot,
|
|
279
|
+
}
|
|
280
|
+
|
|
281
|
+
|
|
282
|
+
# ---------------------------------------------------------------------------
|
|
283
|
+
# Probe
|
|
284
|
+
# ---------------------------------------------------------------------------
|
|
285
|
+
|
|
286
|
+
def run_probe(
|
|
287
|
+
*,
|
|
288
|
+
tool_filter: str = "all",
|
|
289
|
+
scope_filter: str = "all",
|
|
290
|
+
home: Path | None = None,
|
|
291
|
+
project: Path | None = None,
|
|
292
|
+
) -> ProbeResult:
|
|
293
|
+
home = home or Path(os.environ.get("HOME", "/tmp"))
|
|
294
|
+
project = project or Path.cwd()
|
|
295
|
+
|
|
296
|
+
scopes: dict[str, Path] = {}
|
|
297
|
+
if scope_filter in ("all", "user"):
|
|
298
|
+
scopes["user"] = home
|
|
299
|
+
if scope_filter in ("all", "project"):
|
|
300
|
+
scopes["project"] = project
|
|
301
|
+
|
|
302
|
+
tools = TOOL_IDS if tool_filter == "all" else (tool_filter,)
|
|
303
|
+
|
|
304
|
+
result = ProbeResult()
|
|
305
|
+
for scope, root in scopes.items():
|
|
306
|
+
for tool in tools:
|
|
307
|
+
reader = TOOL_READERS.get(tool)
|
|
308
|
+
if reader is None:
|
|
309
|
+
continue
|
|
310
|
+
for reg in reader(scope, root):
|
|
311
|
+
result.registrations.append(reg)
|
|
312
|
+
|
|
313
|
+
# Group by (tool, skill_id). Anything with ≥ 2 entries is DUPLICATE.
|
|
314
|
+
# If their description hashes diverge, also flag DRIFT.
|
|
315
|
+
by_key: dict[tuple[str, str], list[Registration]] = {}
|
|
316
|
+
for reg in result.registrations:
|
|
317
|
+
by_key.setdefault((reg.tool, reg.skill_id), []).append(reg)
|
|
318
|
+
|
|
319
|
+
for (tool, skill_id), regs in by_key.items():
|
|
320
|
+
if len(regs) < 2:
|
|
321
|
+
continue
|
|
322
|
+
key = f"{tool}:{skill_id}"
|
|
323
|
+
result.duplicates[key] = regs
|
|
324
|
+
hashes = {r.description_hash for r in regs if r.description_hash != "manifest"}
|
|
325
|
+
versions = {r.version for r in regs if r.version != "unknown"}
|
|
326
|
+
if len(hashes) > 1 or len(versions) > 1:
|
|
327
|
+
result.drift[key] = regs
|
|
328
|
+
|
|
329
|
+
return result
|
|
330
|
+
|
|
331
|
+
|
|
332
|
+
# ---------------------------------------------------------------------------
|
|
333
|
+
# Rendering
|
|
334
|
+
# ---------------------------------------------------------------------------
|
|
335
|
+
|
|
336
|
+
def render_text(result: ProbeResult) -> str:
|
|
337
|
+
lines: list[str] = []
|
|
338
|
+
lines.append("Skill-registration probe")
|
|
339
|
+
lines.append("=" * 64)
|
|
340
|
+
lines.append("")
|
|
341
|
+
if not result.registrations:
|
|
342
|
+
lines.append("(no registrations found)")
|
|
343
|
+
return "\n".join(lines)
|
|
344
|
+
|
|
345
|
+
lines.append(f"{'TOOL':<10} {'SCOPE':<14} {'SKILL':<32} {'VER':<10} SOURCE")
|
|
346
|
+
lines.append("-" * 100)
|
|
347
|
+
for reg in result.registrations:
|
|
348
|
+
lines.append(
|
|
349
|
+
f"{reg.tool:<10} {reg.scope:<14} {reg.skill_id[:32]:<32} {reg.version:<10} {reg.source_path}"
|
|
350
|
+
)
|
|
351
|
+
|
|
352
|
+
if result.duplicates:
|
|
353
|
+
lines.append("")
|
|
354
|
+
lines.append("DUPLICATE — same skill registered in ≥ 2 sources")
|
|
355
|
+
lines.append("-" * 64)
|
|
356
|
+
for key, regs in result.duplicates.items():
|
|
357
|
+
lines.append(f" {key}")
|
|
358
|
+
for r in regs:
|
|
359
|
+
lines.append(f" - [{r.scope}/{r.version}] {r.source_path}")
|
|
360
|
+
if result.drift:
|
|
361
|
+
lines.append("")
|
|
362
|
+
lines.append("DRIFT — same skill registered with DIFFERENT description / version")
|
|
363
|
+
lines.append("-" * 64)
|
|
364
|
+
for key, regs in result.drift.items():
|
|
365
|
+
lines.append(f" {key}")
|
|
366
|
+
for r in regs:
|
|
367
|
+
lines.append(f" - [{r.scope}/{r.version}] hash={r.description_hash} desc={r.description_snippet!r}")
|
|
368
|
+
lines.append(f" source: {r.source_path}")
|
|
369
|
+
|
|
370
|
+
return "\n".join(lines)
|
|
371
|
+
|
|
372
|
+
|
|
373
|
+
def render_json(result: ProbeResult) -> str:
|
|
374
|
+
return json.dumps(result.to_dict(), indent=2)
|
|
375
|
+
|
|
376
|
+
|
|
377
|
+
# ---------------------------------------------------------------------------
|
|
378
|
+
# CLI
|
|
379
|
+
# ---------------------------------------------------------------------------
|
|
380
|
+
|
|
381
|
+
def main(argv: list[str] | None = None) -> int:
|
|
382
|
+
parser = argparse.ArgumentParser(
|
|
383
|
+
prog="probe_skill_registration",
|
|
384
|
+
description="Detect duplicate / drifting skill registrations across AI tools.",
|
|
385
|
+
)
|
|
386
|
+
parser.add_argument("--tool", choices=("all", *TOOL_IDS), default="all")
|
|
387
|
+
parser.add_argument("--scope", choices=("all", *SCOPE_IDS), default="all")
|
|
388
|
+
parser.add_argument("--format", choices=("text", "json"), default="text")
|
|
389
|
+
parser.add_argument(
|
|
390
|
+
"--strict", action="store_true",
|
|
391
|
+
help="Exit non-zero if any DUPLICATE / DRIFT finding is present.",
|
|
392
|
+
)
|
|
393
|
+
parser.add_argument("--home", type=Path, default=None, help="Override the user-global root (testing).")
|
|
394
|
+
parser.add_argument("--project", type=Path, default=None, help="Override the project root (testing).")
|
|
395
|
+
args = parser.parse_args(argv)
|
|
396
|
+
|
|
397
|
+
result = run_probe(
|
|
398
|
+
tool_filter=args.tool,
|
|
399
|
+
scope_filter=args.scope,
|
|
400
|
+
home=args.home,
|
|
401
|
+
project=args.project,
|
|
402
|
+
)
|
|
403
|
+
|
|
404
|
+
out = render_json(result) if args.format == "json" else render_text(result)
|
|
405
|
+
print(out)
|
|
406
|
+
|
|
407
|
+
if args.strict and (result.duplicates or result.drift):
|
|
408
|
+
return 2
|
|
409
|
+
return 0
|
|
410
|
+
|
|
411
|
+
|
|
412
|
+
if __name__ == "__main__":
|
|
413
|
+
sys.exit(main())
|