@pugi/cli 0.1.0-beta.8 → 0.1.0-beta.87
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/CHANGELOG.md +96 -0
- package/THIRD_PARTY_NOTICES.md +40 -0
- package/assets/pugi-prozr2-mascot.ansi +9 -0
- package/bin/run.js +33 -1
- package/dist/commands/deploy.js +40 -40
- package/dist/commands/flatten.js +191 -0
- package/dist/commands/jobs-watch.js +201 -0
- package/dist/commands/jobs.js +42 -27
- package/dist/commands/smoke.js +133 -0
- package/dist/core/agent-progress/cleanup.js +134 -0
- package/dist/core/agent-progress/schema.js +144 -0
- package/dist/core/agent-progress/writer.js +101 -0
- package/dist/core/agents/adaptive-router.js +330 -0
- package/dist/core/agents/query-decomposer.js +297 -0
- package/dist/core/agents/registry.js +2 -2
- package/dist/core/approvals/shortcut-resolver.js +98 -0
- package/dist/core/artifact-chain/dispatcher.js +148 -0
- package/dist/core/artifact-chain/exporter.js +164 -0
- package/dist/core/artifact-chain/state.js +243 -0
- package/dist/core/artifact-chain/steps.js +169 -0
- package/dist/core/ask-user/question.js +92 -0
- package/dist/core/audit/audit-trail.js +275 -0
- package/dist/core/auth/ensure-authenticated.js +129 -0
- package/dist/core/auth/env-provider.js +238 -0
- package/dist/core/auto-open-browser.js +4 -4
- package/dist/core/auto-update/channels.js +122 -0
- package/dist/core/auto-update/checker.js +241 -0
- package/dist/core/auto-update/state.js +235 -0
- package/dist/core/bare-mode/index.js +107 -0
- package/dist/core/bash/redirect.js +281 -0
- package/dist/core/bash-classifier.js +436 -40
- package/dist/core/checkpoint/resumer.js +149 -0
- package/dist/core/checkpoint/rewinder.js +291 -0
- package/dist/core/checkpoints/shadow-git.js +670 -0
- package/dist/core/citations/parser.js +109 -0
- package/dist/core/classifier/yolo-classifier.js +88 -0
- package/dist/core/codegraph/decision-store.js +248 -0
- package/dist/core/codegraph/detect-repo.js +459 -0
- package/dist/core/codegraph/install.js +134 -0
- package/dist/core/codegraph/offer-hook.js +220 -0
- package/dist/core/compact/auto-trigger.js +96 -0
- package/dist/core/compact/buffer-rewriter.js +115 -0
- package/dist/core/compact/summarizer.js +208 -0
- package/dist/core/compact/token-counter.js +108 -0
- package/dist/core/consensus/anvil-fanout.js +25 -25
- package/dist/core/consensus/diff-capture.js +121 -12
- package/dist/core/consensus/rubric.js +21 -21
- package/dist/core/context/builder.js +6 -6
- package/dist/core/context/compaction-events.js +8 -8
- package/dist/core/context/compaction.js +31 -31
- package/dist/core/context/index.js +15 -8
- package/dist/core/context/invariants.js +51 -51
- package/dist/core/context/markdown-loader.js +28 -10
- package/dist/core/context/markdown-traverse.js +255 -0
- package/dist/core/context/pugiignore.js +41 -41
- package/dist/core/context/repo-skeleton.js +37 -37
- package/dist/core/context/tool-eviction.js +55 -0
- package/dist/core/context/watcher.js +32 -32
- package/dist/core/context/working-set.js +23 -23
- package/dist/core/coordinator/agent-tools.js +77 -0
- package/dist/core/coordinator/agent-toolset.js +65 -0
- package/dist/core/coordinator/fsm.js +73 -0
- package/dist/core/coordinator/mode-fsm.js +70 -0
- package/dist/core/cost/rate-card.js +129 -0
- package/dist/core/cost/tracker.js +221 -0
- package/dist/core/credentials.js +12 -12
- package/dist/core/cron/scheduler.js +138 -0
- package/dist/core/denial-tracking/index.js +8 -0
- package/dist/core/denial-tracking/state.js +264 -0
- package/dist/core/diagnostics/probe-runner.js +93 -0
- package/dist/core/diagnostics/probes/api.js +46 -0
- package/dist/core/diagnostics/probes/auth.js +93 -0
- package/dist/core/diagnostics/probes/bare-mode.js +42 -0
- package/dist/core/diagnostics/probes/cli-version.js +127 -0
- package/dist/core/diagnostics/probes/config.js +72 -0
- package/dist/core/diagnostics/probes/denial-tracking.js +57 -0
- package/dist/core/diagnostics/probes/disk.js +81 -0
- package/dist/core/diagnostics/probes/engine-live.js +46 -0
- package/dist/core/diagnostics/probes/git.js +65 -0
- package/dist/core/diagnostics/probes/hooks.js +118 -0
- package/dist/core/diagnostics/probes/mcp.js +75 -0
- package/dist/core/diagnostics/probes/node.js +59 -0
- package/dist/core/diagnostics/probes/pnpm.js +36 -0
- package/dist/core/diagnostics/probes/pugi-md.js +89 -0
- package/dist/core/diagnostics/probes/sandbox.js +40 -0
- package/dist/core/diagnostics/probes/session.js +74 -0
- package/dist/core/diagnostics/probes/status-snapshot.js +488 -0
- package/dist/core/diagnostics/probes/workspace.js +63 -0
- package/dist/core/diagnostics/types.js +70 -0
- package/dist/core/dispatch/cache-cleanup.js +197 -0
- package/dist/core/dispatch/cache-handoff.js +295 -0
- package/dist/core/edits/apply-patch-layer-e.js +189 -0
- package/dist/core/edits/dispatch.js +293 -7
- package/dist/core/edits/format-matrix.js +26 -0
- package/dist/core/edits/fuzzy-ladder.js +650 -0
- package/dist/core/edits/index.js +3 -1
- package/dist/core/edits/journal.js +199 -0
- package/dist/core/edits/layer-a-apply.js +15 -15
- package/dist/core/edits/layer-a-fuzzy-apply.js +198 -0
- package/dist/core/edits/layer-b-apply.js +9 -9
- package/dist/core/edits/layer-c-apply.js +6 -6
- package/dist/core/edits/layer-d-ast.js +557 -14
- package/dist/core/edits/marker-parser.js +12 -12
- package/dist/core/edits/security-gate.js +27 -27
- package/dist/core/edits/verify-hook.js +273 -0
- package/dist/core/edits/worktree.js +322 -0
- package/dist/core/engine/anvil-client.js +140 -26
- package/dist/core/engine/auto-compact.js +179 -0
- package/dist/core/engine/budgets.js +186 -0
- package/dist/core/engine/context-prefix.js +155 -0
- package/dist/core/engine/index.js +1 -1
- package/dist/core/engine/intensity.js +158 -0
- package/dist/core/engine/intent.js +260 -0
- package/dist/core/engine/native-pugi.js +1295 -227
- package/dist/core/engine/prompts.js +134 -16
- package/dist/core/engine/strip-internal-fields.js +124 -0
- package/dist/core/engine/tool-bridge.js +1295 -59
- package/dist/core/evaluation/golden-dataset.js +293 -0
- package/dist/core/feedback/queue.js +177 -0
- package/dist/core/feedback/submitter.js +145 -0
- package/dist/core/file-cache.js +113 -1
- package/dist/core/flatten/flatten-repo.js +439 -0
- package/dist/core/format/osc8-link.js +28 -0
- package/dist/core/hook-chains.js +392 -0
- package/dist/core/hooks/citation-verify-hook.js +138 -0
- package/dist/core/hooks/citation-verify.js +112 -0
- package/dist/core/hooks/events.js +44 -0
- package/dist/core/hooks/index.js +15 -0
- package/dist/core/hooks/registry.js +213 -0
- package/dist/core/hooks/runner.js +236 -0
- package/dist/core/hooks/v2/event-emitter.js +115 -0
- package/dist/core/hooks/v2/executor.js +282 -0
- package/dist/core/hooks/v2/index.js +25 -0
- package/dist/core/hooks/v2/lifecycle.js +104 -0
- package/dist/core/hooks/v2/loader.js +216 -0
- package/dist/core/hooks/v2/matcher.js +125 -0
- package/dist/core/hooks/v2/trust.js +143 -0
- package/dist/core/hooks/v2/types.js +86 -0
- package/dist/core/image/renderer.js +71 -0
- package/dist/core/init/detector.js +582 -0
- package/dist/core/init/template-renderer.js +242 -0
- package/dist/core/jobs/registry.js +18 -18
- package/dist/core/ledger/results-tsv.js +142 -0
- package/dist/core/log-discipline/stdout-redirect.js +51 -0
- package/dist/core/lsp/cache.js +105 -0
- package/dist/core/lsp/client.js +776 -0
- package/dist/core/lsp/language-detect.js +66 -0
- package/dist/core/lsp/post-edit-diagnostics.js +171 -0
- package/dist/core/lsp/symbol-tools.js +372 -0
- package/dist/core/mcp/client.js +97 -28
- package/dist/core/mcp/http-server.js +553 -0
- package/dist/core/mcp/orchestrator-tools.js +662 -0
- package/dist/core/mcp/permission.js +190 -0
- package/dist/core/mcp/registry.js +39 -17
- package/dist/core/mcp/server-tools.js +219 -0
- package/dist/core/mcp/server.js +397 -0
- package/dist/core/mcp/trust.js +10 -10
- package/dist/core/memory/dual-write.js +416 -0
- package/dist/core/memory/passive-extract.js +130 -0
- package/dist/core/memory/phase1-kinds.js +20 -0
- package/dist/core/memory/secret-scanner.js +304 -0
- package/dist/core/memory-sync/queue.js +170 -0
- package/dist/core/metrics/extract.js +113 -0
- package/dist/core/modes/roo-modes.js +68 -0
- package/dist/core/onboarding/ensure-initialized.js +133 -0
- package/dist/core/onboarding/marker.js +111 -0
- package/dist/core/onboarding/telemetry-state.js +108 -0
- package/dist/core/output-style/presets.js +176 -0
- package/dist/core/output-style/state.js +185 -0
- package/dist/core/path-security.js +287 -5
- package/dist/core/permission.js +82 -22
- package/dist/core/permissions/auto-classifier.js +124 -0
- package/dist/core/permissions/bash-parser.js +371 -0
- package/dist/core/permissions/circuit-breaker.js +83 -0
- package/dist/core/permissions/constrained-edit.js +91 -0
- package/dist/core/permissions/gate.js +278 -0
- package/dist/core/permissions/index.js +20 -0
- package/dist/core/permissions/mode.js +174 -0
- package/dist/core/permissions/network-egress.js +137 -0
- package/dist/core/permissions/state.js +241 -0
- package/dist/core/permissions/tool-class.js +93 -0
- package/dist/core/plan-mode/ui-state.js +51 -0
- package/dist/core/plans/plan-artifact.js +721 -0
- package/dist/core/policy-limits/etag-store.js +122 -0
- package/dist/core/prd-check/parser.js +215 -0
- package/dist/core/prd-check/reporter.js +127 -0
- package/dist/core/prd-check/session-review.js +557 -0
- package/dist/core/prd-check/verifiers.js +223 -0
- package/dist/core/prompt-cache/client-cache.js +99 -0
- package/dist/core/prompts/assembly.js +29 -0
- package/dist/core/prompts/registry.js +364 -0
- package/dist/core/pugi-md/cc-compat-rules.js +735 -0
- package/dist/core/pugi-md/context-injector.js +76 -0
- package/dist/core/pugi-md/walk-up.js +207 -0
- package/dist/core/python/uv-installer.js +270 -0
- package/dist/core/python/uv-resolver.js +83 -0
- package/dist/core/rate-limit/narrator.js +146 -0
- package/dist/core/recipes/cli-types.js +20 -0
- package/dist/core/recipes/loader.js +103 -0
- package/dist/core/recipes/runner.js +345 -0
- package/dist/core/recipes/schema.js +587 -0
- package/dist/core/release-notes/parser.js +241 -0
- package/dist/core/release-notes/state.js +116 -0
- package/dist/core/repl/ask.js +37 -37
- package/dist/core/repl/cancellation.js +26 -26
- package/dist/core/repl/cap-warning.js +4 -4
- package/dist/core/repl/clipboard-read.js +11 -11
- package/dist/core/repl/dispatch-fsm.js +12 -12
- package/dist/core/repl/history-search.js +15 -15
- package/dist/core/repl/history.js +28 -18
- package/dist/core/repl/kill-ring.js +5 -5
- package/dist/core/repl/model-pricing.js +135 -0
- package/dist/core/repl/privacy-banner.js +22 -22
- package/dist/core/repl/session.js +2157 -214
- package/dist/core/repl/slash-commands.js +533 -40
- package/dist/core/repl/store/index.js +1 -1
- package/dist/core/repl/store/jsonl-log.js +22 -22
- package/dist/core/repl/store/lockfile.js +10 -10
- package/dist/core/repl/store/session-store.js +136 -107
- package/dist/core/repl/store/types.js +15 -15
- package/dist/core/repl/store/uuid-v7.js +12 -12
- package/dist/core/repl/workspace-context.js +43 -21
- package/dist/core/repo-map/build.js +125 -0
- package/dist/core/repo-map/cache.js +185 -0
- package/dist/core/repo-map/extractor.js +254 -0
- package/dist/core/repo-map/formatter.js +145 -0
- package/dist/core/repo-map/page-rank.js +105 -0
- package/dist/core/repo-map/scanner.js +211 -0
- package/dist/core/retry-budget/budget.js +284 -0
- package/dist/core/retry-budget/index.js +5 -0
- package/dist/core/retry-budget/retry-cap.js +74 -0
- package/dist/core/routing/lead-worker.js +43 -0
- package/dist/core/routing/pre-flight-estimator.js +108 -0
- package/dist/core/runs/run-tree.js +103 -0
- package/dist/core/security/injection-scanner.js +367 -0
- package/dist/core/security/output-filter.js +418 -0
- package/dist/core/session/env-file.js +105 -0
- package/dist/core/session/section-budgets.js +140 -0
- package/dist/core/session.js +92 -0
- package/dist/core/settings.js +286 -5
- package/dist/core/share/formatter.js +271 -0
- package/dist/core/share/redactor.js +221 -0
- package/dist/core/share/uploader.js +267 -0
- package/dist/core/skills/defaults.js +457 -0
- package/dist/core/skills/loader.js +22 -22
- package/dist/core/skills/sources.js +27 -27
- package/dist/core/smoke/headless-driver.js +174 -0
- package/dist/core/smoke/orchestrator.js +194 -0
- package/dist/core/smoke/runner.js +238 -0
- package/dist/core/smoke/scenario-parser.js +316 -0
- package/dist/core/statusline.js +99 -0
- package/dist/core/subagents/dispatcher-real.js +600 -0
- package/dist/core/subagents/dispatcher.js +132 -43
- package/dist/core/subagents/index.js +19 -6
- package/dist/core/subagents/isolation-matrix.js +213 -0
- package/dist/core/subagents/spawn.js +19 -4
- package/dist/core/telemetry/emitter.js +229 -0
- package/dist/core/telemetry/queue.js +251 -0
- package/dist/core/theme/context.js +91 -0
- package/dist/core/theme/presets.js +228 -0
- package/dist/core/theme/state.js +181 -0
- package/dist/core/todos/invariant.js +10 -0
- package/dist/core/todos/state.js +177 -0
- package/dist/core/tool-schema/compressor.js +89 -0
- package/dist/core/transport/version-interceptor.js +166 -0
- package/dist/core/trust.js +2 -2
- package/dist/core/tui/thinking-block.js +64 -0
- package/dist/core/vim/keymap.js +288 -0
- package/dist/core/vim/state.js +92 -0
- package/dist/core/watch-markers/marker-watcher.js +133 -0
- package/dist/core/worktree-manager/cleanup.js +123 -0
- package/dist/core/worktree-manager/manager.js +303 -0
- package/dist/index.js +28 -0
- package/dist/runtime/bootstrap.js +190 -0
- package/dist/runtime/cli.js +4151 -489
- package/dist/runtime/commands/agents.js +30 -30
- package/dist/runtime/commands/budget.js +5 -5
- package/dist/runtime/commands/cancel.js +231 -0
- package/dist/runtime/commands/chain.js +489 -0
- package/dist/runtime/commands/codegraph-status.js +227 -0
- package/dist/runtime/commands/compact.js +297 -0
- package/dist/runtime/commands/config.js +32 -32
- package/dist/runtime/commands/cost.js +199 -0
- package/dist/runtime/commands/delegate.js +244 -13
- package/dist/runtime/commands/dispatch.js +126 -0
- package/dist/runtime/commands/doctor.js +579 -0
- package/dist/runtime/commands/feedback.js +184 -0
- package/dist/runtime/commands/hooks.js +184 -0
- package/dist/runtime/commands/init.js +254 -0
- package/dist/runtime/commands/lsp.js +368 -0
- package/dist/runtime/commands/mcp.js +879 -0
- package/dist/runtime/commands/memory.js +582 -0
- package/dist/runtime/commands/model.js +237 -0
- package/dist/runtime/commands/onboarding.js +275 -0
- package/dist/runtime/commands/patch.js +128 -0
- package/dist/runtime/commands/permissions.js +112 -0
- package/dist/runtime/commands/plan.js +143 -0
- package/dist/runtime/commands/prd-check.js +285 -0
- package/dist/runtime/commands/privacy.js +17 -17
- package/dist/runtime/commands/recipe.js +325 -0
- package/dist/runtime/commands/redo-blob-store.js +92 -0
- package/dist/runtime/commands/redo.js +361 -0
- package/dist/runtime/commands/release-notes.js +229 -0
- package/dist/runtime/commands/repo-map.js +95 -0
- package/dist/runtime/commands/report.js +299 -0
- package/dist/runtime/commands/resume.js +118 -0
- package/dist/runtime/commands/review-consensus.js +68 -53
- package/dist/runtime/commands/rewind.js +333 -0
- package/dist/runtime/commands/roster.js +14 -14
- package/dist/runtime/commands/sessions.js +163 -0
- package/dist/runtime/commands/share.js +316 -0
- package/dist/runtime/commands/skills.js +31 -31
- package/dist/runtime/commands/status.js +186 -0
- package/dist/runtime/commands/stickers.js +82 -0
- package/dist/runtime/commands/style.js +194 -0
- package/dist/runtime/commands/theme.js +196 -0
- package/dist/runtime/commands/undo.js +54 -22
- package/dist/runtime/commands/update.js +289 -0
- package/dist/runtime/commands/vim.js +140 -0
- package/dist/runtime/commands/worktree.js +177 -0
- package/dist/runtime/commands/worktrees.js +155 -0
- package/dist/runtime/headless-repl.js +195 -0
- package/dist/runtime/headless.js +543 -0
- package/dist/runtime/load-hooks-or-exit.js +71 -0
- package/dist/runtime/plan-decompose.js +531 -0
- package/dist/runtime/update-check.js +28 -28
- package/dist/runtime/version.js +65 -0
- package/dist/skills/bundled/batch.js +617 -0
- package/dist/skills/bundled/index.js +45 -0
- package/dist/skills/bundled/loop.js +358 -0
- package/dist/skills/bundled/remember.js +383 -0
- package/dist/skills/bundled/simplify.js +289 -0
- package/dist/skills/bundled/skillify.js +373 -0
- package/dist/skills/bundled/stuck.js +558 -0
- package/dist/skills/bundled/verify.js +439 -0
- package/dist/testing/vcr.js +486 -0
- package/dist/tools/agent-tool.js +229 -0
- package/dist/tools/apply-patch.js +556 -0
- package/dist/tools/ask-user-question.js +222 -0
- package/dist/tools/ask-user.js +115 -0
- package/dist/tools/bash.js +623 -45
- package/dist/tools/brief.js +224 -0
- package/dist/tools/enter-worktree.js +250 -0
- package/dist/tools/exit-worktree.js +147 -0
- package/dist/tools/file-tools.js +161 -44
- package/dist/tools/lsp-tools.js +189 -0
- package/dist/tools/mcp-tool.js +260 -0
- package/dist/tools/multi-edit.js +361 -0
- package/dist/tools/powershell.js +268 -0
- package/dist/tools/registry.js +85 -0
- package/dist/tools/skill-tool.js +96 -0
- package/dist/tools/sleep.js +99 -0
- package/dist/tools/synthetic-output.js +133 -0
- package/dist/tools/tasks.js +208 -0
- package/dist/tools/todo-write.js +184 -0
- package/dist/tools/verify-plan-execution.js +295 -0
- package/dist/tools/web-fetch-injection-scanner.js +207 -0
- package/dist/tools/web-fetch.js +195 -10
- package/dist/tools/web-search.js +458 -0
- package/dist/tui/agent-progress-card.js +111 -0
- package/dist/tui/agent-tree.js +11 -1
- package/dist/tui/ask-modal.js +14 -14
- package/dist/tui/ask-user-question-prompt.js +203 -0
- package/dist/tui/compact-banner.js +81 -0
- package/dist/tui/conversation-pane.js +85 -11
- package/dist/tui/cost-table.js +111 -0
- package/dist/tui/device-flow.js +2 -2
- package/dist/tui/doctor-table.js +46 -0
- package/dist/tui/feedback-prompt.js +156 -0
- package/dist/tui/input-box.js +247 -32
- package/dist/tui/login-picker.js +3 -3
- package/dist/tui/markdown-render.js +6 -6
- package/dist/tui/onboarding-wizard.js +240 -0
- package/dist/tui/permissions-picker.js +86 -0
- package/dist/tui/render.js +35 -0
- package/dist/tui/repl-render.js +332 -54
- package/dist/tui/repl-splash-art.js +16 -16
- package/dist/tui/repl-splash-mascot.js +48 -24
- package/dist/tui/repl-splash.js +22 -22
- package/dist/tui/repl.js +124 -44
- package/dist/tui/slash-palette.js +6 -6
- package/dist/tui/splash.js +2 -2
- package/dist/tui/status-bar.js +109 -31
- package/dist/tui/status-table.js +7 -0
- package/dist/tui/stickers-art.js +136 -0
- package/dist/tui/style-table.js +28 -0
- package/dist/tui/theme-table.js +29 -0
- package/dist/tui/thinking-spinner.js +123 -0
- package/dist/tui/tool-stream-pane.js +53 -4
- package/dist/tui/update-banner.js +27 -2
- package/dist/tui/vim-input.js +267 -0
- package/dist/tui/welcome-banner.js +107 -0
- package/dist/tui/welcome-data.js +293 -0
- package/dist/tui/workspace-context.js +2 -2
- package/docs/examples/codegraph.mcp.json +10 -0
- package/package.json +23 -6
- package/test/scenarios/codegen-create-file.scenario.txt +13 -0
- package/test/scenarios/compact-force.scenario.txt +11 -0
- package/test/scenarios/identity.scenario.txt +11 -0
- package/test/scenarios/persona-handoff.scenario.txt +11 -0
- package/test/scenarios/walkback.scenario.txt +12 -0
- package/dist/core/engine/compaction-hook.js +0 -154
|
@@ -0,0 +1,457 @@
|
|
|
1
|
+
import { existsSync, mkdtempSync, rmSync, writeFileSync } from 'node:fs';
|
|
2
|
+
import { tmpdir } from 'node:os';
|
|
3
|
+
import { join } from 'node:path';
|
|
4
|
+
import { installSkill, workspaceSkillDir, } from './loader.js';
|
|
5
|
+
import { hashSkillDir, recordTrust } from './trust.js';
|
|
6
|
+
/**
|
|
7
|
+
* Static catalog of the three bundled defaults. Exported so the tests
|
|
8
|
+
* (and any future `pugi skills install --bundled` opt-in path) can
|
|
9
|
+
* iterate over the canonical list without re-listing skill names.
|
|
10
|
+
*/
|
|
11
|
+
export const DEFAULT_SKILLS = [
|
|
12
|
+
{
|
|
13
|
+
name: 'brand-voice',
|
|
14
|
+
description: 'Pugi brand voice: English in code/commits, terse in chat, no AI attribution, no emoji slop.',
|
|
15
|
+
version: '1.0.0',
|
|
16
|
+
source: 'pugi-original',
|
|
17
|
+
sourceUrl: 'https://github.com/pugi-io/pugi/blob/main/apps/pugi-cli/src/core/skills/defaults.ts',
|
|
18
|
+
license: 'MIT',
|
|
19
|
+
body: [
|
|
20
|
+
'# Brand voice',
|
|
21
|
+
'',
|
|
22
|
+
'Apply this rule set to every commit message, PR body, code comment,',
|
|
23
|
+
'generated docstring, and chat reply Pugi produces on this workspace.',
|
|
24
|
+
'',
|
|
25
|
+
'## Language split',
|
|
26
|
+
'',
|
|
27
|
+
'- Code, comments, commit messages, PR titles + bodies, technical docs,',
|
|
28
|
+
" generated tests, log strings: **English**. Do not switch mid-line.",
|
|
29
|
+
'- Operator-facing chat replies follow the workspace `settings.json`',
|
|
30
|
+
' `workflow.chatLang` (defaults to English when unset).',
|
|
31
|
+
'',
|
|
32
|
+
'## No AI attribution',
|
|
33
|
+
'',
|
|
34
|
+
'- Never add `Co-Authored-By: Claude` (or Codex, Gemini, Cursor) trailers',
|
|
35
|
+
' to commits or PR bodies.',
|
|
36
|
+
'- Never write "Generated with the upstream tool", "Made with AI", or any',
|
|
37
|
+
' variant in PR descriptions, README content, or release notes.',
|
|
38
|
+
'- Pugi itself is the byline; the underlying model is implementation',
|
|
39
|
+
' detail and stays out of customer-visible text.',
|
|
40
|
+
'',
|
|
41
|
+
'## No emoji decoration',
|
|
42
|
+
'',
|
|
43
|
+
'- Skip decorative emoji in commit messages, PR bodies, code comments,',
|
|
44
|
+
' log lines, and chat replies.',
|
|
45
|
+
'- Functional unicode that carries meaning (math symbols, currency,',
|
|
46
|
+
' language scripts) is fine.',
|
|
47
|
+
'',
|
|
48
|
+
'## No em dashes',
|
|
49
|
+
'',
|
|
50
|
+
'- Prefer regular hyphens or restructure the sentence. The em dash',
|
|
51
|
+
' character reads as ChatGPT-flavoured and we keep our voice distinct.',
|
|
52
|
+
'',
|
|
53
|
+
'## Verb grammar',
|
|
54
|
+
'',
|
|
55
|
+
'Pugi exposes four customer verbs: `idea`, `plan`, `build`, `review`.',
|
|
56
|
+
'When generating help text, READMEs, or onboarding docs reference these',
|
|
57
|
+
'exact verbs in this exact order. Do not invent synonyms like "scaffold",',
|
|
58
|
+
'"draft", "ship", or "execute" for the same surface.',
|
|
59
|
+
'',
|
|
60
|
+
].join('\n'),
|
|
61
|
+
},
|
|
62
|
+
{
|
|
63
|
+
name: 'endpoint-probe',
|
|
64
|
+
description: 'Verify every URL with a HEAD probe before codegen writes it so Pugi never ships a fabricated endpoint.',
|
|
65
|
+
version: '1.0.0',
|
|
66
|
+
source: 'pugi-original',
|
|
67
|
+
sourceUrl: 'https://github.com/pugi-io/pugi/blob/main/apps/pugi-cli/src/core/skills/defaults.ts',
|
|
68
|
+
license: 'MIT',
|
|
69
|
+
body: [
|
|
70
|
+
'# Endpoint probe',
|
|
71
|
+
'',
|
|
72
|
+
'When the model is about to write a URL into source code, configuration,',
|
|
73
|
+
'documentation, or a generated README, treat that URL as unverified',
|
|
74
|
+
'until a real HTTP probe confirms it resolves.',
|
|
75
|
+
'',
|
|
76
|
+
'## Rule',
|
|
77
|
+
'',
|
|
78
|
+
'1. Before any tool call that writes a URL (`write`, `edit`, `patch`),',
|
|
79
|
+
' enumerate the URLs the diff introduces.',
|
|
80
|
+
'2. For each URL whose host is reachable from the workspace, issue a',
|
|
81
|
+
' `HEAD` request (or `GET` when the host blocks `HEAD`).',
|
|
82
|
+
'3. Treat the URL as **valid** only when the response is 2xx, 3xx, or',
|
|
83
|
+
' a documented 401/403 (auth-gated endpoint we expect).',
|
|
84
|
+
'4. Treat the URL as **invalid** on `ENOTFOUND`, `ECONNREFUSED`,',
|
|
85
|
+
' timeout, or 4xx/5xx other than the documented auth codes. In that',
|
|
86
|
+
' case do not write the URL; ask the operator for the correct one or',
|
|
87
|
+
' pull it from `.pugi/settings.json` / `.env`.',
|
|
88
|
+
'',
|
|
89
|
+
'## Why this matters',
|
|
90
|
+
'',
|
|
91
|
+
'Pugi has historically shipped READMEs with installer URLs that 404,',
|
|
92
|
+
'API base URLs that point at deprecated tunnels, and "see docs at X"',
|
|
93
|
+
'links to pages that never existed. Each is a trust event: the next',
|
|
94
|
+
'operator pastes the URL, hits the 404, and assumes the whole CLI is',
|
|
95
|
+
'broken. A HEAD probe costs one round-trip and closes that class.',
|
|
96
|
+
'',
|
|
97
|
+
'## Skip cases',
|
|
98
|
+
'',
|
|
99
|
+
'- Localhost URLs (`http://127.0.0.1:*`, `http://localhost:*`): skip',
|
|
100
|
+
' the probe; verify visually that the port matches the workspace dev',
|
|
101
|
+
' server config instead.',
|
|
102
|
+
'- Example URLs in comments explicitly marked `// example:` or',
|
|
103
|
+
' `# example:`: skip the probe; the operator has flagged the URL as',
|
|
104
|
+
' illustrative.',
|
|
105
|
+
'',
|
|
106
|
+
].join('\n'),
|
|
107
|
+
},
|
|
108
|
+
{
|
|
109
|
+
name: 'readme-sync',
|
|
110
|
+
description: 'When package.json bin/name/scripts change, cross-check README install + usage so shipped docs never drift from the binary.',
|
|
111
|
+
version: '1.0.0',
|
|
112
|
+
source: 'pugi-original',
|
|
113
|
+
sourceUrl: 'https://github.com/pugi-io/pugi/blob/main/apps/pugi-cli/src/core/skills/defaults.ts',
|
|
114
|
+
license: 'MIT',
|
|
115
|
+
body: [
|
|
116
|
+
'# README sync',
|
|
117
|
+
'',
|
|
118
|
+
'Whenever a `build` task renames a package or its binary, the README',
|
|
119
|
+
'must move with it. Stale install snippets are the most common Pugi',
|
|
120
|
+
'regression: the binary ships as `@pugi/cli` but the README still',
|
|
121
|
+
'tells the operator to `npm i -g @pugi/cli`, and the install',
|
|
122
|
+
"fails before they ever see `pugi --version`.",
|
|
123
|
+
'',
|
|
124
|
+
'## Rule',
|
|
125
|
+
'',
|
|
126
|
+
'Whenever a diff touches `package.json` and changes any of:',
|
|
127
|
+
'',
|
|
128
|
+
' - `name`',
|
|
129
|
+
' - `bin`',
|
|
130
|
+
' - `scripts.start` / `scripts.dev`',
|
|
131
|
+
' - top-level `homepage` / `repository.url`',
|
|
132
|
+
'',
|
|
133
|
+
'then the same diff MUST also update the corresponding sections of the',
|
|
134
|
+
"package's `README.md` (and, when the package has a docs landing page,",
|
|
135
|
+
'the install snippet on that landing page too).',
|
|
136
|
+
'',
|
|
137
|
+
'## Verification snippet',
|
|
138
|
+
'',
|
|
139
|
+
'Before marking a `build` step done, run this check on every touched',
|
|
140
|
+
'package directory:',
|
|
141
|
+
'',
|
|
142
|
+
'```',
|
|
143
|
+
'pkg_name=$(jq -r .name package.json)',
|
|
144
|
+
'bin_name=$(jq -r ".bin | if type==\\"string\\" then \\"unknown\\" else keys[0] end" package.json)',
|
|
145
|
+
'grep -F "npm i -g $pkg_name" README.md || echo "MISMATCH: README missing install for $pkg_name"',
|
|
146
|
+
'grep -F "$bin_name " README.md || echo "MISMATCH: README missing usage for $bin_name"',
|
|
147
|
+
'```',
|
|
148
|
+
'',
|
|
149
|
+
'When either grep prints `MISMATCH`, fix the README before the commit',
|
|
150
|
+
'is considered ready for `review`.',
|
|
151
|
+
'',
|
|
152
|
+
'## Idempotence',
|
|
153
|
+
'',
|
|
154
|
+
'This rule applies to every package in the workspace, not just the',
|
|
155
|
+
"first one the operator mentions. A monorepo-wide rename (`@pugi/*`",
|
|
156
|
+
"to `@pugi/*`) has to sync every README, not just the customer-facing",
|
|
157
|
+
'CLI README.',
|
|
158
|
+
'',
|
|
159
|
+
].join('\n'),
|
|
160
|
+
},
|
|
161
|
+
{
|
|
162
|
+
name: 'zoom-out',
|
|
163
|
+
description: 'Ask Pugi to zoom out and give broader context or a higher-level perspective. Use when the model is unfamiliar with a section of code or needs to understand how it fits into the bigger picture.',
|
|
164
|
+
version: '1.0.0',
|
|
165
|
+
source: 'mattpocock/skills',
|
|
166
|
+
sourceUrl: 'https://github.com/mattpocock/skills/blob/main/skills/engineering/zoom-out/SKILL.md',
|
|
167
|
+
license: 'MIT',
|
|
168
|
+
body: [
|
|
169
|
+
'# Zoom out',
|
|
170
|
+
'',
|
|
171
|
+
'**Source:** [mattpocock/skills](https://github.com/mattpocock/skills/blob/main/skills/engineering/zoom-out/SKILL.md) - MIT licensed, authored by Matt Pocock. Ported verbatim into the Pugi bundle on.',
|
|
172
|
+
'',
|
|
173
|
+
'I do not know this area of code well. Go up a layer of abstraction.',
|
|
174
|
+
'Give me a map of all the relevant modules and callers, using the',
|
|
175
|
+
"project's domain glossary vocabulary.",
|
|
176
|
+
'',
|
|
177
|
+
].join('\n'),
|
|
178
|
+
},
|
|
179
|
+
{
|
|
180
|
+
name: 'diagnose',
|
|
181
|
+
description: 'Disciplined six-phase diagnosis loop for hard bugs and performance regressions: build feedback loop, reproduce, hypothesise, instrument, fix + regression-test, cleanup + post-mortem. Use when the operator reports a bug, a flake, or a perf regression.',
|
|
182
|
+
version: '1.0.0',
|
|
183
|
+
source: 'mattpocock/skills',
|
|
184
|
+
sourceUrl: 'https://github.com/mattpocock/skills/blob/main/skills/engineering/diagnose/SKILL.md',
|
|
185
|
+
license: 'MIT',
|
|
186
|
+
body: [
|
|
187
|
+
'# Diagnose',
|
|
188
|
+
'',
|
|
189
|
+
'**Source:** [mattpocock/skills](https://github.com/mattpocock/skills/blob/main/skills/engineering/diagnose/SKILL.md) - MIT licensed, authored by Matt Pocock. Ported into the Pugi bundle on with minor de-namespacing (cross-skill links rewritten, project-glossary references generalised).',
|
|
190
|
+
'',
|
|
191
|
+
'A discipline for hard bugs. Skip phases only when explicitly justified.',
|
|
192
|
+
'',
|
|
193
|
+
'When exploring the codebase, use the project domain glossary (if one',
|
|
194
|
+
'exists) to build a clear mental model of the relevant modules, and',
|
|
195
|
+
'check ADRs in the area you are touching.',
|
|
196
|
+
'',
|
|
197
|
+
'## Phase 1 - Build a feedback loop',
|
|
198
|
+
'',
|
|
199
|
+
'**This is the skill.** Everything else is mechanical. If you have a',
|
|
200
|
+
'fast, deterministic, agent-runnable pass/fail signal for the bug, you',
|
|
201
|
+
'will find the cause - bisection, hypothesis-testing, and instrumentation',
|
|
202
|
+
'all just consume that signal. If you do not have one, no amount of',
|
|
203
|
+
'staring at code will save you.',
|
|
204
|
+
'',
|
|
205
|
+
'Spend disproportionate effort here. **Be aggressive. Be creative.',
|
|
206
|
+
'Refuse to give up.**',
|
|
207
|
+
'',
|
|
208
|
+
'### Ways to construct one, in roughly this order',
|
|
209
|
+
'',
|
|
210
|
+
'1. **Failing test** at whatever seam reaches the bug - unit, integration, e2e.',
|
|
211
|
+
'2. **Curl / HTTP script** against a running dev server.',
|
|
212
|
+
'3. **CLI invocation** with a fixture input, diffing stdout against a known-good snapshot.',
|
|
213
|
+
'4. **Headless browser script** (Playwright / Puppeteer) - drives the UI, asserts on DOM/console/network.',
|
|
214
|
+
'5. **Replay a captured trace.** Save a real network request / payload / event log to disk; replay it through the code path in isolation.',
|
|
215
|
+
'6. **Throwaway harness.** Spin up a minimal subset of the system (one service, mocked deps) that exercises the bug code path with a single function call.',
|
|
216
|
+
'7. **Property / fuzz loop.** If the bug is "sometimes wrong output", run 1000 random inputs and look for the failure mode.',
|
|
217
|
+
'8. **Bisection harness.** If the bug appeared between two known states (commit, dataset, version), automate "boot at state X, check, repeat" so you can `git bisect run` it.',
|
|
218
|
+
'9. **Differential loop.** Run the same input through old-version vs new-version (or two configs) and diff outputs.',
|
|
219
|
+
'10. **HITL bash script.** Last resort. If a human must click, drive _them_ with a templated loop script so the loop is still structured. Captured output feeds back to you.',
|
|
220
|
+
'',
|
|
221
|
+
'Build the right feedback loop, and the bug is 90% fixed.',
|
|
222
|
+
'',
|
|
223
|
+
'### Iterate on the loop itself',
|
|
224
|
+
'',
|
|
225
|
+
'Treat the loop as a product. Once you have _a_ loop, ask:',
|
|
226
|
+
'',
|
|
227
|
+
'- Can I make it faster? (Cache setup, skip unrelated init, narrow the test scope.)',
|
|
228
|
+
'- Can I make the signal sharper? (Assert on the specific symptom, not "did not crash".)',
|
|
229
|
+
'- Can I make it more deterministic? (Pin time, seed RNG, isolate filesystem, freeze network.)',
|
|
230
|
+
'',
|
|
231
|
+
'A 30-second flaky loop is barely better than no loop. A 2-second',
|
|
232
|
+
'deterministic loop is a debugging superpower.',
|
|
233
|
+
'',
|
|
234
|
+
'### Non-deterministic bugs',
|
|
235
|
+
'',
|
|
236
|
+
'The goal is not a clean repro but a **higher reproduction rate**. Loop',
|
|
237
|
+
'the trigger 100x, parallelise, add stress, narrow timing windows, inject',
|
|
238
|
+
'sleeps. A 50% flake is debuggable; 1% is not - keep raising the rate',
|
|
239
|
+
'until it is debuggable.',
|
|
240
|
+
'',
|
|
241
|
+
'### When you genuinely cannot build a loop',
|
|
242
|
+
'',
|
|
243
|
+
'Stop and say so explicitly. List what you tried. Ask the operator for:',
|
|
244
|
+
'(a) access to whatever environment reproduces it, (b) a captured artifact',
|
|
245
|
+
'(HAR file, log dump, core dump, screen recording with timestamps), or',
|
|
246
|
+
'(c) permission to add temporary production instrumentation. Do **not**',
|
|
247
|
+
'proceed to hypothesise without a loop.',
|
|
248
|
+
'',
|
|
249
|
+
'Do not proceed to Phase 2 until you have a loop you believe in.',
|
|
250
|
+
'',
|
|
251
|
+
'## Phase 2 - Reproduce',
|
|
252
|
+
'',
|
|
253
|
+
'Run the loop. Watch the bug appear.',
|
|
254
|
+
'',
|
|
255
|
+
'Confirm:',
|
|
256
|
+
'',
|
|
257
|
+
'- The loop produces the failure mode the **operator** described - not a different failure that happens to be nearby. Wrong bug = wrong fix.',
|
|
258
|
+
'- The failure is reproducible across multiple runs (or, for non-deterministic bugs, reproducible at a high enough rate to debug against).',
|
|
259
|
+
'- You have captured the exact symptom (error message, wrong output, slow timing) so later phases can verify the fix actually addresses it.',
|
|
260
|
+
'',
|
|
261
|
+
'Do not proceed until you reproduce the bug.',
|
|
262
|
+
'',
|
|
263
|
+
'## Phase 3 - Hypothesise',
|
|
264
|
+
'',
|
|
265
|
+
'Generate **3-5 ranked hypotheses** before testing any of them.',
|
|
266
|
+
'Single-hypothesis generation anchors on the first plausible idea.',
|
|
267
|
+
'',
|
|
268
|
+
'Each hypothesis must be **falsifiable**: state the prediction it makes.',
|
|
269
|
+
'',
|
|
270
|
+
'> Format: "If <X> is the cause, then <changing Y> will make the bug disappear / <changing Z> will make it worse."',
|
|
271
|
+
'',
|
|
272
|
+
'If you cannot state the prediction, the hypothesis is a vibe - discard',
|
|
273
|
+
'or sharpen it.',
|
|
274
|
+
'',
|
|
275
|
+
'**Show the ranked list to the operator before testing.** They often',
|
|
276
|
+
'have domain knowledge that re-ranks instantly ("we just deployed a',
|
|
277
|
+
'change to #3"), or know hypotheses they have already ruled out. Cheap',
|
|
278
|
+
'checkpoint, big time saver. Do not block on it - proceed with your',
|
|
279
|
+
'ranking if the operator is AFK.',
|
|
280
|
+
'',
|
|
281
|
+
'## Phase 4 - Instrument',
|
|
282
|
+
'',
|
|
283
|
+
'Each probe must map to a specific prediction from Phase 3. **Change',
|
|
284
|
+
'one variable at a time.**',
|
|
285
|
+
'',
|
|
286
|
+
'Tool preference:',
|
|
287
|
+
'',
|
|
288
|
+
'1. **Debugger / REPL inspection** if the env supports it. One breakpoint beats ten logs.',
|
|
289
|
+
'2. **Targeted logs** at the boundaries that distinguish hypotheses.',
|
|
290
|
+
'3. Never "log everything and grep".',
|
|
291
|
+
'',
|
|
292
|
+
'**Tag every debug log** with a unique prefix, e.g. `[DEBUG-a4f2]`.',
|
|
293
|
+
'Cleanup at the end becomes a single grep. Untagged logs survive;',
|
|
294
|
+
'tagged logs die.',
|
|
295
|
+
'',
|
|
296
|
+
'**Perf branch.** For performance regressions, logs are usually wrong.',
|
|
297
|
+
'Instead: establish a baseline measurement (timing harness,',
|
|
298
|
+
'`performance.now()`, profiler, query plan), then bisect. Measure first,',
|
|
299
|
+
'fix second.',
|
|
300
|
+
'',
|
|
301
|
+
'## Phase 5 - Fix + regression test',
|
|
302
|
+
'',
|
|
303
|
+
'Write the regression test **before the fix** - but only if there is a',
|
|
304
|
+
'**correct seam** for it.',
|
|
305
|
+
'',
|
|
306
|
+
'A correct seam is one where the test exercises the **real bug pattern**',
|
|
307
|
+
'as it occurs at the call site. If the only available seam is too',
|
|
308
|
+
'shallow (single-caller test when the bug needs multiple callers, unit',
|
|
309
|
+
'test that cannot replicate the chain that triggered the bug), a',
|
|
310
|
+
'regression test there gives false confidence.',
|
|
311
|
+
'',
|
|
312
|
+
'**If no correct seam exists, that itself is the finding.** Note it. The',
|
|
313
|
+
'codebase architecture is preventing the bug from being locked down.',
|
|
314
|
+
'Flag this for the next phase.',
|
|
315
|
+
'',
|
|
316
|
+
'If a correct seam exists:',
|
|
317
|
+
'',
|
|
318
|
+
'1. Turn the minimised repro into a failing test at that seam.',
|
|
319
|
+
'2. Watch it fail.',
|
|
320
|
+
'3. Apply the fix.',
|
|
321
|
+
'4. Watch it pass.',
|
|
322
|
+
'5. Re-run the Phase 1 feedback loop against the original (un-minimised) scenario.',
|
|
323
|
+
'',
|
|
324
|
+
'## Phase 6 - Cleanup + post-mortem',
|
|
325
|
+
'',
|
|
326
|
+
'Required before declaring done:',
|
|
327
|
+
'',
|
|
328
|
+
'- Original repro no longer reproduces (re-run the Phase 1 loop)',
|
|
329
|
+
'- Regression test passes (or absence of seam is documented)',
|
|
330
|
+
'- All `[DEBUG-...]` instrumentation removed (`grep` the prefix)',
|
|
331
|
+
'- Throwaway prototypes deleted (or moved to a clearly-marked debug location)',
|
|
332
|
+
'- The hypothesis that turned out correct is stated in the commit / PR message - so the next debugger learns',
|
|
333
|
+
'',
|
|
334
|
+
'**Then ask: what would have prevented this bug?** If the answer',
|
|
335
|
+
'involves architectural change (no good test seam, tangled callers,',
|
|
336
|
+
'hidden coupling) note the specifics for a follow-up refactor. Make the',
|
|
337
|
+
'recommendation **after** the fix is in, not before - you have more',
|
|
338
|
+
'information now than when you started.',
|
|
339
|
+
'',
|
|
340
|
+
].join('\n'),
|
|
341
|
+
},
|
|
342
|
+
{
|
|
343
|
+
name: 'grill-me',
|
|
344
|
+
description: 'Interview the operator relentlessly about a plan or design until reaching shared understanding, resolving each branch of the decision tree. Use when the operator wants to stress-test a plan or says "grill me".',
|
|
345
|
+
version: '1.0.0',
|
|
346
|
+
source: 'mattpocock/skills',
|
|
347
|
+
sourceUrl: 'https://github.com/mattpocock/skills/blob/main/skills/productivity/grill-me/SKILL.md',
|
|
348
|
+
license: 'MIT',
|
|
349
|
+
body: [
|
|
350
|
+
'# Grill me',
|
|
351
|
+
'',
|
|
352
|
+
'**Source:** [mattpocock/skills](https://github.com/mattpocock/skills/blob/main/skills/productivity/grill-me/SKILL.md) - MIT licensed, authored by Matt Pocock. Ported verbatim into the Pugi bundle on (terminology swap: "user" -> "operator" to match Pugi vocabulary).',
|
|
353
|
+
'',
|
|
354
|
+
'Interview the operator relentlessly about every aspect of this plan',
|
|
355
|
+
'until we reach a shared understanding. Walk down each branch of the',
|
|
356
|
+
'design tree, resolving dependencies between decisions one-by-one. For',
|
|
357
|
+
'each question, provide your recommended answer.',
|
|
358
|
+
'',
|
|
359
|
+
'Ask the questions one at a time.',
|
|
360
|
+
'',
|
|
361
|
+
'If a question can be answered by exploring the codebase, explore the',
|
|
362
|
+
'codebase instead.',
|
|
363
|
+
'',
|
|
364
|
+
].join('\n'),
|
|
365
|
+
},
|
|
366
|
+
];
|
|
367
|
+
/**
|
|
368
|
+
* Build the canonical `SKILL.md` text for a bundled default. We hand-roll
|
|
369
|
+
* the frontmatter (instead of going through a YAML library) so the file
|
|
370
|
+
* round-trips byte-identical for the same `DEFAULT_SKILLS` entry. Stable
|
|
371
|
+
* bytes means a stable sha256: re-running `pugi init` against a clean
|
|
372
|
+
* workspace produces the same trust signature.
|
|
373
|
+
*/
|
|
374
|
+
function renderSkillMarkdown(spec) {
|
|
375
|
+
const frontmatter = [
|
|
376
|
+
'---',
|
|
377
|
+
`name: ${spec.name}`,
|
|
378
|
+
`description: ${spec.description}`,
|
|
379
|
+
'metadata:',
|
|
380
|
+
' type: skill',
|
|
381
|
+
` version: ${spec.version}`,
|
|
382
|
+
'---',
|
|
383
|
+
'',
|
|
384
|
+
].join('\n');
|
|
385
|
+
return `${frontmatter}${spec.body}`;
|
|
386
|
+
}
|
|
387
|
+
/**
|
|
388
|
+
* Install every bundled default skill into the workspace `.pugi/skills/`
|
|
389
|
+
* directory. Idempotent: a skill whose target directory already exists
|
|
390
|
+
* (i.e. operator has customised it) is left untouched.
|
|
391
|
+
*
|
|
392
|
+
* Trust records are written under the workspace scope with
|
|
393
|
+
* `signedBy: pugi-bundled` so `pugi skills list` reports them as
|
|
394
|
+
* `[trusted]` without the operator having to run `pugi skills trust`
|
|
395
|
+
* manually after init.
|
|
396
|
+
*/
|
|
397
|
+
export async function installDefaultSkills(input) {
|
|
398
|
+
const summaries = [];
|
|
399
|
+
const scope = 'workspace';
|
|
400
|
+
for (const spec of DEFAULT_SKILLS) {
|
|
401
|
+
const targetDir = workspaceSkillDir(input.workspaceRoot, spec.name);
|
|
402
|
+
if (existsSync(targetDir)) {
|
|
403
|
+
// Idempotent: never overwrite a skill the operator has already
|
|
404
|
+
// edited. The `init` command is re-runnable by design.
|
|
405
|
+
summaries.push({ name: spec.name, status: 'skipped-existing', dir: targetDir });
|
|
406
|
+
input.log?.(`[pugi init] skill "${spec.name}" already present, leaving as-is.\n`);
|
|
407
|
+
continue;
|
|
408
|
+
}
|
|
409
|
+
// Materialise the bundled body in a tmp dir so `installSkill` can copy
|
|
410
|
+
// it through its standard payload path (includes its own SKILL.md
|
|
411
|
+
// existence check + symlink hardening via verbatimSymlinks).
|
|
412
|
+
const payloadDir = mkdtempSync(join(tmpdir(), 'pugi-default-skill-'));
|
|
413
|
+
try {
|
|
414
|
+
writeFileSync(join(payloadDir, 'SKILL.md'), renderSkillMarkdown(spec), {
|
|
415
|
+
encoding: 'utf8',
|
|
416
|
+
mode: 0o600,
|
|
417
|
+
});
|
|
418
|
+
const installedDir = installSkill({
|
|
419
|
+
payloadDir,
|
|
420
|
+
name: spec.name,
|
|
421
|
+
scope,
|
|
422
|
+
workspaceRoot: input.workspaceRoot,
|
|
423
|
+
});
|
|
424
|
+
const sha256 = hashSkillDir(installedDir);
|
|
425
|
+
await recordTrust({
|
|
426
|
+
kind: 'skill',
|
|
427
|
+
scope,
|
|
428
|
+
name: spec.name,
|
|
429
|
+
sha256,
|
|
430
|
+
// Source records the provenance for the trust ledger so a future
|
|
431
|
+
// `pugi skills info brand-voice` shows where it came from without
|
|
432
|
+
// pointing at a URL that never existed.
|
|
433
|
+
source: 'pugi-bundled-default',
|
|
434
|
+
signedBy: 'pugi-bundled',
|
|
435
|
+
});
|
|
436
|
+
summaries.push({ name: spec.name, status: 'installed', dir: installedDir });
|
|
437
|
+
input.log?.(`[pugi init] installed default skill "${spec.name}" -> ${installedDir}\n`);
|
|
438
|
+
}
|
|
439
|
+
finally {
|
|
440
|
+
rmSync(payloadDir, { recursive: true, force: true });
|
|
441
|
+
}
|
|
442
|
+
}
|
|
443
|
+
return summaries;
|
|
444
|
+
}
|
|
445
|
+
/**
|
|
446
|
+
* Internal helper used by tests to keep parity with the runtime. Exposed
|
|
447
|
+
* so a test fixture can write a SKILL.md byte-for-byte matching what the
|
|
448
|
+
* installer would have produced.
|
|
449
|
+
*/
|
|
450
|
+
export function renderDefaultSkillMarkdown(name) {
|
|
451
|
+
const spec = DEFAULT_SKILLS.find((entry) => entry.name === name);
|
|
452
|
+
if (!spec) {
|
|
453
|
+
throw new Error(`renderDefaultSkillMarkdown: no bundled default named "${name}"`);
|
|
454
|
+
}
|
|
455
|
+
return renderSkillMarkdown(spec);
|
|
456
|
+
}
|
|
457
|
+
//# sourceMappingURL=defaults.js.map
|
|
@@ -6,12 +6,12 @@ const FRONTMATTER_DELIM = '---';
|
|
|
6
6
|
* Parse a markdown file with YAML frontmatter into a structured form.
|
|
7
7
|
*
|
|
8
8
|
* Throws when:
|
|
9
|
-
*
|
|
10
|
-
*
|
|
11
|
-
*
|
|
9
|
+
* - no frontmatter delimiter pair is present
|
|
10
|
+
* - frontmatter does not parse as the minimal YAML grammar
|
|
11
|
+
* - required keys `name`, `description`, `metadata.type` are missing
|
|
12
12
|
*
|
|
13
13
|
* Frontmatter `tools` accepts either flow array (`[a, b, c]`) or block
|
|
14
|
-
* array (newline + `
|
|
14
|
+
* array (newline + ` - a` lines). Strings are unquoted, single, or
|
|
15
15
|
* double quoted. Multi-line scalar values are NOT supported (no `>` or
|
|
16
16
|
* `|` folding) — Skills do not need them.
|
|
17
17
|
*/
|
|
@@ -146,7 +146,7 @@ function parseBlockObject(lines, from) {
|
|
|
146
146
|
consumed += arr.consumed + 1;
|
|
147
147
|
continue;
|
|
148
148
|
}
|
|
149
|
-
// Deeper objects are out of scope for
|
|
149
|
+
// Deeper objects are out of scope for — keep frontmatter
|
|
150
150
|
// shape predictable. Throw with a clear pointer.
|
|
151
151
|
throw new Error(`SKILL_PARSE: nested objects deeper than 1 level are not supported (key "${key}")`);
|
|
152
152
|
}
|
|
@@ -212,12 +212,12 @@ function validateFrontmatter(raw) {
|
|
|
212
212
|
// Two frontmatter dialects coexist in the wild:
|
|
213
213
|
//
|
|
214
214
|
// 1. Pugi-native — explicit `metadata: { type: skill|agent, ... }`.
|
|
215
|
-
//
|
|
216
|
-
//
|
|
215
|
+
// Future Pugi-published skills + agents use this so the kind is
|
|
216
|
+
// self-declared at parse time.
|
|
217
217
|
//
|
|
218
218
|
// 2. Anthropic Skills — flat top-level keys, no `metadata` block, no
|
|
219
|
-
//
|
|
220
|
-
//
|
|
219
|
+
// `type` field. Every file inside `github.com/anthropics/skills`
|
|
220
|
+
// follows this shape (e.g. `algorithmic-art`, `pdf`, `pptx`).
|
|
221
221
|
//
|
|
222
222
|
// We accept both. When `metadata` is absent we synthesize one with
|
|
223
223
|
// `type` defaulted to `skill` — the on-disk layout (`SKILL.md` in a
|
|
@@ -267,7 +267,7 @@ function validateFrontmatter(raw) {
|
|
|
267
267
|
if (node.kind === 'scalar') {
|
|
268
268
|
fm[key] = node.value;
|
|
269
269
|
// Anthropic Skills flat dialect — also surface select top-level
|
|
270
|
-
// keys inside metadata so downstream consumers (
|
|
270
|
+
// keys inside metadata so downstream consumers (Pugi system
|
|
271
271
|
// prompt builder, `skills list` table) have one consistent path.
|
|
272
272
|
if (key === 'license' || key === 'version' || key === 'model' || key === 'allowed-tools') {
|
|
273
273
|
const metaKey = key === 'allowed-tools' ? 'tools' : key;
|
|
@@ -317,20 +317,20 @@ function expectScalar(obj, key) {
|
|
|
317
317
|
/* ----------------------------- Slug validation ----------------------------- */
|
|
318
318
|
/**
|
|
319
319
|
* Slug grammar for skill + agent names. Allowed characters:
|
|
320
|
-
*
|
|
321
|
-
*
|
|
320
|
+
* - lowercase ASCII letters, digits, hyphen, underscore, dot
|
|
321
|
+
* - 1–64 characters total, must START with [a-z0-9]
|
|
322
322
|
*
|
|
323
323
|
* Anything outside this grammar is rejected BEFORE the name reaches a
|
|
324
324
|
* `path.join` call. Attacker payloads we close here:
|
|
325
|
-
*
|
|
326
|
-
*
|
|
327
|
-
*
|
|
328
|
-
*
|
|
329
|
-
*
|
|
330
|
-
*
|
|
331
|
-
*
|
|
332
|
-
*
|
|
333
|
-
*
|
|
325
|
+
* - `../../foo` (path traversal segment)
|
|
326
|
+
* - `/etc/passwd` (absolute path)
|
|
327
|
+
* - `..\\..\\foo` (Windows traversal)
|
|
328
|
+
* - `\0name` (null-byte truncation)
|
|
329
|
+
* - `name\\with\\backslash` (Windows separator)
|
|
330
|
+
* - `..` or `.` (dot-only segments)
|
|
331
|
+
* - empty string (matches existing skill on globbed list)
|
|
332
|
+
* - 65+ char strings (DoS via huge directory names)
|
|
333
|
+
* - uppercase / unicode (collision on case-insensitive filesystems)
|
|
334
334
|
*
|
|
335
335
|
* The CLI commands also accept `--as <slug>` so this guard runs on the
|
|
336
336
|
* operator-supplied override AND on the parsed frontmatter `name`. Both
|
|
@@ -415,7 +415,7 @@ function loadSkill(skillDir, scope) {
|
|
|
415
415
|
* responsible for trust-prompting + hashing before this is invoked.
|
|
416
416
|
*
|
|
417
417
|
* Refuses to install when the payload does not contain a `SKILL.md` at
|
|
418
|
-
* its root: that file is the contract every consumer (
|
|
418
|
+
* its root: that file is the contract every consumer (Pugi system
|
|
419
419
|
* prompt, `skills list`, `skills info`) reads from.
|
|
420
420
|
*/
|
|
421
421
|
export function installSkill(input) {
|