@pugi/cli 0.1.0-beta.3 → 0.1.0-beta.30
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/THIRD_PARTY_NOTICES.md +40 -0
- package/assets/pugi-mascot.ansi +15 -40
- package/bin/run.js +33 -1
- package/dist/commands/jobs-watch.js +201 -0
- package/dist/commands/jobs.js +15 -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/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/auth/env-provider.js +238 -0
- 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/checkpoint/resumer.js +149 -0
- package/dist/core/checkpoint/rewinder.js +291 -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/diff-capture.js +73 -0
- package/dist/core/context/index.js +7 -0
- package/dist/core/context/markdown-traverse.js +255 -0
- package/dist/core/cost/rate-card.js +129 -0
- package/dist/core/cost/tracker.js +221 -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 +86 -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/git.js +65 -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/session.js +74 -0
- package/dist/core/diagnostics/probes/status-snapshot.js +442 -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/dispatch.js +218 -2
- package/dist/core/edits/journal.js +199 -0
- package/dist/core/edits/layer-d-ast.js +557 -14
- package/dist/core/edits/verify-hook.js +273 -0
- package/dist/core/edits/worktree.js +111 -18
- package/dist/core/engine/anvil-client.js +115 -5
- package/dist/core/engine/budgets.js +89 -0
- package/dist/core/engine/context-prefix.js +155 -0
- package/dist/core/engine/intent.js +260 -0
- package/dist/core/engine/native-pugi.js +852 -210
- package/dist/core/engine/prompts.js +89 -6
- package/dist/core/engine/strip-internal-fields.js +124 -0
- package/dist/core/engine/tool-bridge.js +972 -33
- 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/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/init/scaffold.js +195 -0
- package/dist/core/lsp/cache.js +105 -0
- package/dist/core/lsp/client.js +174 -29
- package/dist/core/lsp/language-detect.js +66 -0
- package/dist/core/lsp/post-edit-diagnostics.js +171 -0
- package/dist/core/mcp/client.js +75 -6
- package/dist/core/mcp/http-server.js +553 -0
- package/dist/core/mcp/permission.js +190 -0
- package/dist/core/mcp/registry.js +24 -2
- package/dist/core/mcp/server-tools.js +219 -0
- package/dist/core/mcp/server.js +397 -0
- package/dist/core/memory/dual-write.js +416 -0
- package/dist/core/memory/dual-write.spec.js +297 -0
- package/dist/core/memory/phase1-kinds.js +20 -0
- package/dist/core/memory-sync/queue.js +158 -0
- package/dist/core/memory-sync/queue.spec.js +105 -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/permissions/gate.js +187 -0
- package/dist/core/permissions/index.js +18 -0
- package/dist/core/permissions/mode.js +102 -0
- package/dist/core/permissions/state.js +215 -0
- package/dist/core/permissions/tool-class.js +93 -0
- package/dist/core/prd-check/parser.js +215 -0
- package/dist/core/prd-check/reporter.js +127 -0
- package/dist/core/prd-check/verifiers.js +223 -0
- package/dist/core/pugi-md/context-injector.js +76 -0
- package/dist/core/pugi-md/walk-up.js +207 -0
- package/dist/core/release-notes/parser.js +241 -0
- package/dist/core/release-notes/state.js +116 -0
- package/dist/core/repl/codebase-survey.js +308 -0
- package/dist/core/repl/history.js +11 -1
- package/dist/core/repl/init-interview.js +457 -0
- package/dist/core/repl/model-pricing.js +135 -0
- package/dist/core/repl/onboarding-state.js +297 -0
- package/dist/core/repl/session.js +1486 -30
- package/dist/core/repl/slash-commands.js +345 -9
- package/dist/core/repl/store/session-store.js +31 -2
- package/dist/core/repl/workspace-context.js +22 -0
- 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/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/session.js +44 -0
- package/dist/core/settings.js +80 -0
- 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/subagents/dispatcher-real.js +600 -0
- package/dist/core/subagents/dispatcher.js +113 -24
- package/dist/core/subagents/index.js +18 -5
- 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/transport/version-interceptor.js +166 -0
- package/dist/core/vim/keymap.js +288 -0
- package/dist/core/vim/state.js +92 -0
- package/dist/index.js +28 -0
- package/dist/runtime/bootstrap.js +190 -0
- package/dist/runtime/cli.js +2595 -278
- package/dist/runtime/commands/chain.js +489 -0
- package/dist/runtime/commands/compact.js +297 -0
- package/dist/runtime/commands/cost.js +199 -0
- package/dist/runtime/commands/delegate.js +312 -0
- package/dist/runtime/commands/dispatch.js +126 -0
- package/dist/runtime/commands/doctor.js +390 -0
- package/dist/runtime/commands/feedback.js +184 -0
- package/dist/runtime/commands/hooks.js +184 -0
- package/dist/runtime/commands/lsp.js +212 -28
- package/dist/runtime/commands/mcp.js +824 -0
- package/dist/runtime/commands/memory.js +508 -0
- package/dist/runtime/commands/memory.spec.js +174 -0
- package/dist/runtime/commands/model.js +237 -0
- package/dist/runtime/commands/onboarding.js +275 -0
- package/dist/runtime/commands/patch.js +17 -0
- package/dist/runtime/commands/permissions.js +87 -0
- package/dist/runtime/commands/plan.js +143 -0
- package/dist/runtime/commands/prd-check.js +235 -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 +17 -2
- package/dist/runtime/commands/rewind.js +333 -0
- package/dist/runtime/commands/roster.js +117 -0
- package/dist/runtime/commands/sessions.js +163 -0
- package/dist/runtime/commands/share.js +316 -0
- package/dist/runtime/commands/status.js +178 -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/update.js +289 -0
- package/dist/runtime/commands/vim.js +140 -0
- package/dist/runtime/commands/worktree.js +50 -6
- 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/version.js +65 -0
- package/dist/tools/agent-tool.js +229 -0
- package/dist/tools/apply-patch.js +281 -39
- package/dist/tools/ask-user-question.js +213 -0
- package/dist/tools/ask-user.js +115 -0
- package/dist/tools/file-tools.js +85 -14
- package/dist/tools/mcp-tool.js +260 -0
- package/dist/tools/multi-edit.js +361 -0
- package/dist/tools/registry.js +30 -2
- package/dist/tools/skill-tool.js +96 -0
- package/dist/tools/tasks.js +208 -0
- package/dist/tools/todo-write.js +184 -0
- package/dist/tools/web-fetch.js +147 -2
- package/dist/tools/web-search.js +458 -0
- package/dist/tui/agent-progress-card.js +111 -0
- package/dist/tui/agent-tree.js +10 -0
- package/dist/tui/ask-modal.js +2 -2
- package/dist/tui/ask-user-question-prompt.js +192 -0
- package/dist/tui/compact-banner.js +81 -0
- package/dist/tui/conversation-pane.js +82 -8
- package/dist/tui/cost-table.js +111 -0
- package/dist/tui/doctor-table.js +46 -0
- package/dist/tui/feedback-prompt.js +156 -0
- package/dist/tui/input-box.js +46 -2
- package/dist/tui/markdown-render.js +4 -4
- package/dist/tui/onboarding-wizard.js +240 -0
- package/dist/tui/repl-render.js +293 -35
- package/dist/tui/repl-splash.js +2 -2
- package/dist/tui/repl.js +45 -13
- package/dist/tui/splash.js +1 -1
- package/dist/tui/status-bar.js +94 -16
- 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/tool-stream-pane.js +7 -0
- package/dist/tui/update-banner.js +20 -2
- package/dist/tui/vim-input.js +267 -0
- package/docs/examples/codegraph.mcp.json +10 -0
- package/package.json +9 -6
|
@@ -0,0 +1,213 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Pugi hooks MVP — registry (Leak L12, first pass).
|
|
3
|
+
*
|
|
4
|
+
* Reads `<home>/hooks-mvp.json` and validates its shape with Zod. The
|
|
5
|
+
* file uses the Claude Code-style nested config:
|
|
6
|
+
*
|
|
7
|
+
* {
|
|
8
|
+
* "hooks": {
|
|
9
|
+
* "SessionStart": [{ "command": "echo session-start" }],
|
|
10
|
+
* "PreToolUse": [{ "matcher": "bash", "command": "echo bash-pre", "blocking": true }]
|
|
11
|
+
* }
|
|
12
|
+
* }
|
|
13
|
+
*
|
|
14
|
+
* Schema constraints:
|
|
15
|
+
* - Each hook entry MUST have a `command` (non-empty string).
|
|
16
|
+
* - `matcher` is optional. Defaults to `*` (any tool / any payload).
|
|
17
|
+
* For tool events, `matcher` is compared against the tool name.
|
|
18
|
+
* For non-tool events (SessionStart), `matcher` is ignored.
|
|
19
|
+
* - `timeoutMs` is optional. Defaults to 30 000 ms (per task spec).
|
|
20
|
+
* Capped at 60 000 ms to prevent operator-defined deadlocks.
|
|
21
|
+
* - `blocking` is optional. When true AND the hook exits non-zero,
|
|
22
|
+
* the registry surfaces an `anyBlocked: true` outcome so the
|
|
23
|
+
* caller can refuse the originating action. Only honoured for
|
|
24
|
+
* `PreToolUse` in the MVP — other events log but do not block.
|
|
25
|
+
*
|
|
26
|
+
* Failure modes:
|
|
27
|
+
* - File missing -> the registry is `empty()`. `list()` returns []
|
|
28
|
+
* and `fire()` is a no-op. This matches the Claude Code default
|
|
29
|
+
* (hooks are opt-in).
|
|
30
|
+
* - File present but invalid JSON / fails schema -> `load()` throws.
|
|
31
|
+
* The CLI surface (`pugi hooks doctor`) reports the error
|
|
32
|
+
* verbatim so the operator can fix the config.
|
|
33
|
+
*
|
|
34
|
+
* Brand voice: ASCII only, no emoji, no em-dashes.
|
|
35
|
+
*/
|
|
36
|
+
import { existsSync, readFileSync } from 'node:fs';
|
|
37
|
+
import { homedir } from 'node:os';
|
|
38
|
+
import { resolve } from 'node:path';
|
|
39
|
+
import { z } from 'zod';
|
|
40
|
+
import { ALL_HOOK_EVENTS_V2 } from './events.js';
|
|
41
|
+
/** Default per-hook timeout when the operator does not set `timeoutMs`. */
|
|
42
|
+
export const DEFAULT_HOOK_TIMEOUT_MS = 30_000;
|
|
43
|
+
/** Hard upper bound on `timeoutMs`. Prevents config-defined deadlocks. */
|
|
44
|
+
export const MAX_HOOK_TIMEOUT_MS = 60_000;
|
|
45
|
+
const hookEntrySchema = z
|
|
46
|
+
.object({
|
|
47
|
+
/**
|
|
48
|
+
* Tool-name matcher. `*` matches any tool. Plain strings match
|
|
49
|
+
* exactly (no glob in the MVP — fast-follow widens to glob). Ignored
|
|
50
|
+
* for non-tool events such as `SessionStart`.
|
|
51
|
+
*/
|
|
52
|
+
matcher: z.string().min(1).optional(),
|
|
53
|
+
/** Shell command. Spawned via `/bin/sh -c <command>`. */
|
|
54
|
+
command: z.string().min(1),
|
|
55
|
+
/** Per-hook timeout override. Defaults to 30 000 ms. */
|
|
56
|
+
timeoutMs: z.number().int().positive().max(MAX_HOOK_TIMEOUT_MS).optional(),
|
|
57
|
+
/**
|
|
58
|
+
* When true, a non-zero exit code from this hook blocks the
|
|
59
|
+
* originating action (currently `PreToolUse` only). Other events
|
|
60
|
+
* log the exit but do not block.
|
|
61
|
+
*/
|
|
62
|
+
blocking: z.boolean().optional(),
|
|
63
|
+
})
|
|
64
|
+
.strict();
|
|
65
|
+
const hookEventEnum = z.enum([
|
|
66
|
+
'SessionStart',
|
|
67
|
+
'PreToolUse',
|
|
68
|
+
'PostToolUse',
|
|
69
|
+
'UserPromptSubmit',
|
|
70
|
+
'Stop',
|
|
71
|
+
'SubagentStop',
|
|
72
|
+
'PreCompact',
|
|
73
|
+
'Notification',
|
|
74
|
+
]);
|
|
75
|
+
const hooksFileSchema = z
|
|
76
|
+
.object({
|
|
77
|
+
hooks: z.record(hookEventEnum, z.array(hookEntrySchema)).default({}),
|
|
78
|
+
})
|
|
79
|
+
.strict();
|
|
80
|
+
/** Default config file location — `~/.pugi/hooks-mvp.json`. */
|
|
81
|
+
export function defaultHooksMvpPath(home) {
|
|
82
|
+
const root = home ?? process.env.PUGI_HOME ?? resolve(homedir(), '.pugi');
|
|
83
|
+
return resolve(root, 'hooks-mvp.json');
|
|
84
|
+
}
|
|
85
|
+
/**
|
|
86
|
+
* In-memory snapshot of the operator's `hooks-mvp.json`. Construct
|
|
87
|
+
* via `loadHooksConfig(path)` — `new HooksConfig()` is intentionally
|
|
88
|
+
* not exported so all production code paths go through the loader.
|
|
89
|
+
*/
|
|
90
|
+
export class HooksConfig {
|
|
91
|
+
path;
|
|
92
|
+
entries;
|
|
93
|
+
constructor(path, entries) {
|
|
94
|
+
this.path = path;
|
|
95
|
+
this.entries = entries;
|
|
96
|
+
}
|
|
97
|
+
/** Absolute path of the config file this snapshot was loaded from. */
|
|
98
|
+
configPath() {
|
|
99
|
+
return this.path;
|
|
100
|
+
}
|
|
101
|
+
/** All hooks declared for a given event. Returns [] when none. */
|
|
102
|
+
list(event) {
|
|
103
|
+
return this.entries[event] ?? [];
|
|
104
|
+
}
|
|
105
|
+
/**
|
|
106
|
+
* Hooks that match the (event, toolName?) tuple. For tool events
|
|
107
|
+
* (`PreToolUse`), `matcher` is compared against the tool name with
|
|
108
|
+
* `*` matching any. For non-tool events, all entries are returned
|
|
109
|
+
* regardless of `matcher`.
|
|
110
|
+
*/
|
|
111
|
+
listMatching(event, toolName) {
|
|
112
|
+
const all = this.list(event);
|
|
113
|
+
if (!isToolEvent(event))
|
|
114
|
+
return all;
|
|
115
|
+
return all.filter((entry) => matchesTool(entry.matcher, toolName));
|
|
116
|
+
}
|
|
117
|
+
/** Flat list of (event, entry) pairs across every configured event. */
|
|
118
|
+
flatten() {
|
|
119
|
+
const out = [];
|
|
120
|
+
for (const event of ALL_HOOK_EVENTS_V2) {
|
|
121
|
+
for (const entry of this.list(event)) {
|
|
122
|
+
out.push({ event, entry });
|
|
123
|
+
}
|
|
124
|
+
}
|
|
125
|
+
return out;
|
|
126
|
+
}
|
|
127
|
+
/** True iff at least one hook is registered for any event. */
|
|
128
|
+
isEmpty() {
|
|
129
|
+
return this.flatten().length === 0;
|
|
130
|
+
}
|
|
131
|
+
/** A no-op snapshot used when the config file is absent. */
|
|
132
|
+
static empty(path) {
|
|
133
|
+
return new HooksConfig(path, {});
|
|
134
|
+
}
|
|
135
|
+
}
|
|
136
|
+
/**
|
|
137
|
+
* Load + validate `hooks-mvp.json`. Returns a no-op snapshot when the
|
|
138
|
+
* file is absent. Throws on invalid JSON or schema violations — the
|
|
139
|
+
* caller is expected to surface the error to the operator via
|
|
140
|
+
* `pugi hooks doctor`.
|
|
141
|
+
*
|
|
142
|
+
* Contract (non-null invariant): this function ALWAYS returns a
|
|
143
|
+
* `HooksConfig` instance. It never returns `null` / `undefined`. When
|
|
144
|
+
* the config file is missing, callers receive `HooksConfig.empty(path)`
|
|
145
|
+
* — a truthy snapshot for which `isEmpty()` returns `true` and `list()`
|
|
146
|
+
* returns `[]`. Callers may safely chain `.isEmpty()` without a null
|
|
147
|
+
* guard. Asserted by `registry-empty.spec.ts`.
|
|
148
|
+
*/
|
|
149
|
+
export function loadHooksConfig(pathOverride) {
|
|
150
|
+
const path = pathOverride ?? defaultHooksMvpPath();
|
|
151
|
+
if (!existsSync(path)) {
|
|
152
|
+
return HooksConfig.empty(path);
|
|
153
|
+
}
|
|
154
|
+
let raw;
|
|
155
|
+
try {
|
|
156
|
+
raw = readFileSync(path, 'utf8');
|
|
157
|
+
}
|
|
158
|
+
catch (error) {
|
|
159
|
+
throw new Error(`pugi hooks: cannot read ${path}: ${error.message}`);
|
|
160
|
+
}
|
|
161
|
+
let parsed;
|
|
162
|
+
try {
|
|
163
|
+
parsed = JSON.parse(raw);
|
|
164
|
+
}
|
|
165
|
+
catch (error) {
|
|
166
|
+
throw new Error(`pugi hooks: ${path} is not valid JSON: ${error.message}`);
|
|
167
|
+
}
|
|
168
|
+
const result = hooksFileSchema.safeParse(parsed);
|
|
169
|
+
if (!result.success) {
|
|
170
|
+
const issues = result.error.issues
|
|
171
|
+
.map((issue) => `${issue.path.join('.') || '<root>'} ${issue.message}`)
|
|
172
|
+
.join('; ');
|
|
173
|
+
throw new Error(`pugi hooks: ${path} failed schema validation: ${issues}`);
|
|
174
|
+
}
|
|
175
|
+
// Zod's `z.record(enum, value)` returns `Partial<Record<...>>` shape
|
|
176
|
+
// — keys that the operator did not include are `undefined`. Coerce
|
|
177
|
+
// explicitly into the same shape `HooksConfig` expects.
|
|
178
|
+
const entries = {};
|
|
179
|
+
for (const event of ALL_HOOK_EVENTS_V2) {
|
|
180
|
+
const list = result.data.hooks[event];
|
|
181
|
+
if (list && list.length > 0) {
|
|
182
|
+
entries[event] = list;
|
|
183
|
+
}
|
|
184
|
+
}
|
|
185
|
+
return new HooksConfig(path, entries);
|
|
186
|
+
}
|
|
187
|
+
/**
|
|
188
|
+
* `isToolEvent(event)` -> true for events where `matcher` is compared
|
|
189
|
+
* against the tool name. SessionStart / Stop / Notification / etc. do
|
|
190
|
+
* not have an associated tool so matcher is ignored.
|
|
191
|
+
*/
|
|
192
|
+
export function isToolEvent(event) {
|
|
193
|
+
return event === 'PreToolUse' || event === 'PostToolUse';
|
|
194
|
+
}
|
|
195
|
+
/**
|
|
196
|
+
* Tool-name match grammar for the MVP. Intentionally narrow:
|
|
197
|
+
* - matcher missing or `*` -> matches any tool name (and the
|
|
198
|
+
* `bash`/`read`/... shape).
|
|
199
|
+
* - matcher === toolName -> exact match.
|
|
200
|
+
*
|
|
201
|
+
* Fast-follow widens this to glob via `picomatch` so operators can
|
|
202
|
+
* write `mcp__*` patterns. Deliberately not pulling in a glob lib for
|
|
203
|
+
* the MVP — the narrow grammar is enough to land the surface and the
|
|
204
|
+
* test matrix stays small.
|
|
205
|
+
*/
|
|
206
|
+
export function matchesTool(matcher, toolName) {
|
|
207
|
+
if (!matcher || matcher === '*')
|
|
208
|
+
return true;
|
|
209
|
+
if (!toolName)
|
|
210
|
+
return false;
|
|
211
|
+
return matcher === toolName;
|
|
212
|
+
}
|
|
213
|
+
//# sourceMappingURL=registry.js.map
|
|
@@ -0,0 +1,236 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Pugi hooks MVP — runner (Leak L12, first pass).
|
|
3
|
+
*
|
|
4
|
+
* Spawns the shell command declared in `hooks-mvp.json`, applies the
|
|
5
|
+
* timeout watchdog, captures stdout / stderr, and surfaces a
|
|
6
|
+
* structured result. Two events are wired in the MVP (SessionStart,
|
|
7
|
+
* PreToolUse); the runner itself is event-agnostic so the fast-follow
|
|
8
|
+
* PR can attach the remaining 6 events without changing this file.
|
|
9
|
+
*
|
|
10
|
+
* Safety properties:
|
|
11
|
+
* - 30 s default timeout (per task spec); SIGTERM then SIGKILL with
|
|
12
|
+
* a 2 s grace window.
|
|
13
|
+
* - 1 MiB output cap per stream — a misbehaving hook (`yes`) cannot
|
|
14
|
+
* OOM the parent CLI by buffering unbounded data.
|
|
15
|
+
* - Spawn failures are caught + logged; the session never crashes
|
|
16
|
+
* because of a missing binary or a syntax error in the command.
|
|
17
|
+
* - Hook errors are atomic-appended to `<workspaceRoot>/.pugi/logs/
|
|
18
|
+
* hooks.log`. Multiple sessions can write concurrently without
|
|
19
|
+
* interleaving because `appendFileSync` opens with O_APPEND.
|
|
20
|
+
*
|
|
21
|
+
* Brand voice: ASCII only.
|
|
22
|
+
*/
|
|
23
|
+
import { spawn } from 'node:child_process';
|
|
24
|
+
import { appendFileSync, existsSync, mkdirSync } from 'node:fs';
|
|
25
|
+
import { resolve } from 'node:path';
|
|
26
|
+
import { DEFAULT_HOOK_TIMEOUT_MS, isToolEvent, } from './registry.js';
|
|
27
|
+
const HOOK_STREAM_CAP_BYTES = 1024 * 1024;
|
|
28
|
+
const SIGKILL_GRACE_MS = 2_000;
|
|
29
|
+
/**
|
|
30
|
+
* Fire every matching hook for `event` sequentially. Sequential (not
|
|
31
|
+
* parallel) is the intentional default — operators frequently chain
|
|
32
|
+
* `git add` -> `eslint --fix` style hooks that would race otherwise.
|
|
33
|
+
* Returns a `HookFireOutcome` with the per-invocation results.
|
|
34
|
+
*/
|
|
35
|
+
export async function fireHooks(opts) {
|
|
36
|
+
const { config, event, payload, toolName, workspaceRoot, env } = opts;
|
|
37
|
+
const matching = config.listMatching(event, toolName);
|
|
38
|
+
if (matching.length === 0) {
|
|
39
|
+
return { event, results: [], anyBlocked: false };
|
|
40
|
+
}
|
|
41
|
+
const logger = workspaceRoot ? new HookLogger(workspaceRoot) : undefined;
|
|
42
|
+
const results = [];
|
|
43
|
+
let anyBlocked = false;
|
|
44
|
+
for (const entry of matching) {
|
|
45
|
+
const timeoutMs = entry.timeoutMs ?? DEFAULT_HOOK_TIMEOUT_MS;
|
|
46
|
+
const result = await executeOne(entry.command, payload, timeoutMs, env);
|
|
47
|
+
// Blocking semantics only honored for PreToolUse in the MVP.
|
|
48
|
+
// Other events can declare `blocking: true` but the runner just
|
|
49
|
+
// logs that intent — it does NOT short-circuit. The fast-follow
|
|
50
|
+
// PR threads PostToolUse + UserPromptSubmit blocking through.
|
|
51
|
+
const blockable = entry.blocking === true && event === 'PreToolUse';
|
|
52
|
+
const blocked = blockable && !result.ok;
|
|
53
|
+
if (blocked) {
|
|
54
|
+
anyBlocked = true;
|
|
55
|
+
result.blocked = true;
|
|
56
|
+
result.blockSentinel = `HOOK_BLOCKED: ${truncate(entry.command, 80)} exited ${result.exitCode}`;
|
|
57
|
+
}
|
|
58
|
+
if (logger && !result.ok) {
|
|
59
|
+
logger.recordFailure(event, entry.command, result);
|
|
60
|
+
}
|
|
61
|
+
results.push(result);
|
|
62
|
+
}
|
|
63
|
+
return { event, results, anyBlocked };
|
|
64
|
+
}
|
|
65
|
+
async function executeOne(command, payload, timeoutMs, env) {
|
|
66
|
+
const startedAt = Date.now();
|
|
67
|
+
return new Promise((resolvePromise) => {
|
|
68
|
+
const payloadJson = JSON.stringify(payload);
|
|
69
|
+
const childEnv = {
|
|
70
|
+
...(env ?? process.env),
|
|
71
|
+
PUGI_HOOK_PAYLOAD: payloadJson,
|
|
72
|
+
PUGI_HOOK_EVENT: payload.event,
|
|
73
|
+
PUGI_HOOK_SESSION_ID: payload.sessionId,
|
|
74
|
+
};
|
|
75
|
+
const child = spawn('/bin/sh', ['-c', command], {
|
|
76
|
+
env: childEnv,
|
|
77
|
+
stdio: ['pipe', 'pipe', 'pipe'],
|
|
78
|
+
});
|
|
79
|
+
const state = {
|
|
80
|
+
stdout: '',
|
|
81
|
+
stderr: '',
|
|
82
|
+
killedForTimeout: false,
|
|
83
|
+
killedForStreamCap: false,
|
|
84
|
+
};
|
|
85
|
+
const escalateKill = () => {
|
|
86
|
+
if (state.sigKillTimer)
|
|
87
|
+
return;
|
|
88
|
+
state.sigKillTimer = setTimeout(() => {
|
|
89
|
+
if (!child.killed)
|
|
90
|
+
child.kill('SIGKILL');
|
|
91
|
+
}, SIGKILL_GRACE_MS);
|
|
92
|
+
if (state.sigKillTimer.unref)
|
|
93
|
+
state.sigKillTimer.unref();
|
|
94
|
+
};
|
|
95
|
+
const enforceStreamCap = () => {
|
|
96
|
+
if (state.killedForStreamCap)
|
|
97
|
+
return;
|
|
98
|
+
if (state.stdout.length + state.stderr.length <= HOOK_STREAM_CAP_BYTES)
|
|
99
|
+
return;
|
|
100
|
+
state.killedForStreamCap = true;
|
|
101
|
+
child.kill('SIGTERM');
|
|
102
|
+
escalateKill();
|
|
103
|
+
};
|
|
104
|
+
child.stdout?.on('data', (chunk) => {
|
|
105
|
+
if (state.killedForStreamCap)
|
|
106
|
+
return;
|
|
107
|
+
state.stdout += chunk.toString('utf8');
|
|
108
|
+
enforceStreamCap();
|
|
109
|
+
});
|
|
110
|
+
child.stderr?.on('data', (chunk) => {
|
|
111
|
+
if (state.killedForStreamCap)
|
|
112
|
+
return;
|
|
113
|
+
state.stderr += chunk.toString('utf8');
|
|
114
|
+
enforceStreamCap();
|
|
115
|
+
});
|
|
116
|
+
// Best-effort stdin payload — hook scripts that want to read it can
|
|
117
|
+
// (e.g. `jq .`); scripts that ignore stdin will EPIPE on our write
|
|
118
|
+
// which we swallow because the env var carries the same data.
|
|
119
|
+
if (child.stdin) {
|
|
120
|
+
child.stdin.on('error', () => {
|
|
121
|
+
// EPIPE is benign — see above.
|
|
122
|
+
});
|
|
123
|
+
child.stdin.end(payloadJson);
|
|
124
|
+
}
|
|
125
|
+
const timer = setTimeout(() => {
|
|
126
|
+
state.killedForTimeout = true;
|
|
127
|
+
child.kill('SIGTERM');
|
|
128
|
+
escalateKill();
|
|
129
|
+
}, timeoutMs);
|
|
130
|
+
if (timer.unref)
|
|
131
|
+
timer.unref();
|
|
132
|
+
child.on('error', (error) => {
|
|
133
|
+
clearTimeout(timer);
|
|
134
|
+
if (state.sigKillTimer)
|
|
135
|
+
clearTimeout(state.sigKillTimer);
|
|
136
|
+
resolvePromise({
|
|
137
|
+
command: truncate(command, 200),
|
|
138
|
+
exitCode: -1,
|
|
139
|
+
stdoutBytes: state.stdout.length,
|
|
140
|
+
stderrBytes: state.stderr.length,
|
|
141
|
+
elapsedMs: Date.now() - startedAt,
|
|
142
|
+
ok: false,
|
|
143
|
+
blocked: false,
|
|
144
|
+
timedOut: false,
|
|
145
|
+
// No blockSentinel here — spawn errors are not the same as
|
|
146
|
+
// blocking-failure semantics. The caller logs them generically.
|
|
147
|
+
});
|
|
148
|
+
});
|
|
149
|
+
child.on('close', (code, signal) => {
|
|
150
|
+
clearTimeout(timer);
|
|
151
|
+
if (state.sigKillTimer)
|
|
152
|
+
clearTimeout(state.sigKillTimer);
|
|
153
|
+
let exitCode;
|
|
154
|
+
if (code !== null) {
|
|
155
|
+
exitCode = code;
|
|
156
|
+
}
|
|
157
|
+
else if (signal === 'SIGTERM') {
|
|
158
|
+
exitCode = -15;
|
|
159
|
+
}
|
|
160
|
+
else if (signal === 'SIGKILL') {
|
|
161
|
+
exitCode = -9;
|
|
162
|
+
}
|
|
163
|
+
else {
|
|
164
|
+
exitCode = -1;
|
|
165
|
+
}
|
|
166
|
+
const ok = exitCode === 0 &&
|
|
167
|
+
!state.killedForTimeout &&
|
|
168
|
+
!state.killedForStreamCap;
|
|
169
|
+
resolvePromise({
|
|
170
|
+
command: truncate(command, 200),
|
|
171
|
+
exitCode,
|
|
172
|
+
stdoutBytes: state.stdout.length,
|
|
173
|
+
stderrBytes: state.stderr.length,
|
|
174
|
+
elapsedMs: Date.now() - startedAt,
|
|
175
|
+
ok,
|
|
176
|
+
blocked: false,
|
|
177
|
+
timedOut: state.killedForTimeout,
|
|
178
|
+
});
|
|
179
|
+
});
|
|
180
|
+
});
|
|
181
|
+
}
|
|
182
|
+
/**
|
|
183
|
+
* Append-only failure log at `<workspaceRoot>/.pugi/logs/hooks.log`.
|
|
184
|
+
* Each line is a JSON record so log scrapers can `jq` over it.
|
|
185
|
+
*/
|
|
186
|
+
class HookLogger {
|
|
187
|
+
path;
|
|
188
|
+
prepared = false;
|
|
189
|
+
constructor(workspaceRoot) {
|
|
190
|
+
this.path = resolve(workspaceRoot, '.pugi', 'logs', 'hooks.log');
|
|
191
|
+
}
|
|
192
|
+
recordFailure(event, command, result) {
|
|
193
|
+
this.prepareDir();
|
|
194
|
+
const line = JSON.stringify({
|
|
195
|
+
ts: new Date().toISOString(),
|
|
196
|
+
event,
|
|
197
|
+
command: truncate(command, 200),
|
|
198
|
+
exitCode: result.exitCode,
|
|
199
|
+
timedOut: result.timedOut,
|
|
200
|
+
elapsedMs: result.elapsedMs,
|
|
201
|
+
stdoutBytes: result.stdoutBytes,
|
|
202
|
+
stderrBytes: result.stderrBytes,
|
|
203
|
+
toolEvent: isToolEvent(event),
|
|
204
|
+
});
|
|
205
|
+
try {
|
|
206
|
+
appendFileSync(this.path, `${line}\n`, 'utf8');
|
|
207
|
+
}
|
|
208
|
+
catch {
|
|
209
|
+
// Logging is best-effort — the session must not crash when the
|
|
210
|
+
// disk is full or the directory is read-only. The runner has
|
|
211
|
+
// already returned the result; dropping the log line is the
|
|
212
|
+
// safe fallback.
|
|
213
|
+
}
|
|
214
|
+
}
|
|
215
|
+
prepareDir() {
|
|
216
|
+
if (this.prepared)
|
|
217
|
+
return;
|
|
218
|
+
const dir = resolve(this.path, '..');
|
|
219
|
+
if (!existsSync(dir)) {
|
|
220
|
+
try {
|
|
221
|
+
mkdirSync(dir, { recursive: true });
|
|
222
|
+
}
|
|
223
|
+
catch {
|
|
224
|
+
// ignored — appendFileSync will surface a fresh error on the
|
|
225
|
+
// write path, which we also swallow.
|
|
226
|
+
}
|
|
227
|
+
}
|
|
228
|
+
this.prepared = true;
|
|
229
|
+
}
|
|
230
|
+
}
|
|
231
|
+
function truncate(value, max) {
|
|
232
|
+
if (value.length <= max)
|
|
233
|
+
return value;
|
|
234
|
+
return `${value.slice(0, max - 3)}...`;
|
|
235
|
+
}
|
|
236
|
+
//# sourceMappingURL=runner.js.map
|
|
@@ -0,0 +1,195 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Workspace scaffold — extracted from `pugi init` so the bare REPL boot
|
|
3
|
+
* can call it automatically when the operator launches `pugi` in a
|
|
4
|
+
* fresh directory (CEO directive 2026-05-26).
|
|
5
|
+
*
|
|
6
|
+
* Before this module, `pugi init` was the only path that materialised
|
|
7
|
+
* `.pugi/` + the canonical config files. Launching the REPL in an empty
|
|
8
|
+
* directory printed `workspace: (not bound - run /init OR cd into
|
|
9
|
+
* project)` and instructed the operator to Ctrl+C, run `pugi init`,
|
|
10
|
+
* relaunch. That round trip is hostile on a first-touch install — CEO
|
|
11
|
+
* escalated "auto = решение" on 2026-05-26.
|
|
12
|
+
*
|
|
13
|
+
* The module is intentionally side-effect free at import time: the
|
|
14
|
+
* scaffold runs only when `ensureWorkspaceInitialized` is called. The
|
|
15
|
+
* scaffold is also idempotent — every file write is gated by an
|
|
16
|
+
* `existsSync` check, so re-running against a workspace that already has
|
|
17
|
+
* `.pugi/settings.json` (e.g. a manual `pugi init` followed by auto-init
|
|
18
|
+
* on next REPL launch) is a no-op. The function is safe to call before
|
|
19
|
+
* any other init logic.
|
|
20
|
+
*
|
|
21
|
+
* Two CRITICAL invariants:
|
|
22
|
+
*
|
|
23
|
+
* 1. **Atomic per-file.** Every write uses `existsSync` + `writeFileSync`
|
|
24
|
+
* against the final path. There is no read-modify-write pattern that
|
|
25
|
+
* could lose data on a concurrent `pugi init` race. The one path
|
|
26
|
+
* that DOES mutate an existing file — `.gitignore` (append `.pugi/`
|
|
27
|
+
* marker) — also gates on the marker being absent before appending,
|
|
28
|
+
* so the worst-case race is a duplicate marker line that the next
|
|
29
|
+
* run skips.
|
|
30
|
+
*
|
|
31
|
+
* 2. **Silent by default.** When `opts.silent` is true (the REPL
|
|
32
|
+
* auto-init path) the scaffold writes NOTHING to stderr/stdout.
|
|
33
|
+
* The REPL bootstrap runs before Ink mounts, and a stray
|
|
34
|
+
* stdout/stderr write at that point would land on the operator's
|
|
35
|
+
* shell ABOVE the alt-screen entry — visible until they scroll up,
|
|
36
|
+
* and noisy in a CI tail. The explicit `pugi init` path stays
|
|
37
|
+
* verbose via the standalone command in `runtime/cli.ts`.
|
|
38
|
+
*/
|
|
39
|
+
import { existsSync, mkdirSync, readFileSync, writeFileSync } from 'node:fs';
|
|
40
|
+
import { resolve } from 'node:path';
|
|
41
|
+
import { emptyIndex } from '../index-store.js';
|
|
42
|
+
/**
|
|
43
|
+
* Materialise the canonical `.pugi/` workspace scaffold under `cwd`.
|
|
44
|
+
* Returns a `{created, dir, createdPaths, skippedPaths}` summary so the
|
|
45
|
+
* caller can log a one-shot "initialized" line on the first call without
|
|
46
|
+
* re-checking the filesystem.
|
|
47
|
+
*
|
|
48
|
+
* The scaffold mirrors `pugi init` minus the bundled default-skills
|
|
49
|
+
* install (that is a heavier operation gated on the `--no-defaults`
|
|
50
|
+
* flag, and the standalone `pugi init` command keeps owning it).
|
|
51
|
+
*
|
|
52
|
+
* Idempotent: every file write gates on `existsSync`, so re-running
|
|
53
|
+
* against an existing workspace is a no-op and returns
|
|
54
|
+
* `{created: false}` with every path in `skippedPaths`.
|
|
55
|
+
*/
|
|
56
|
+
export function ensureWorkspaceInitialized(cwd, opts = {}) {
|
|
57
|
+
const silent = opts.silent !== false;
|
|
58
|
+
const pugiDir = resolve(cwd, '.pugi');
|
|
59
|
+
// Local trackers so the existing helpers (mkdirIfMissing /
|
|
60
|
+
// writeJsonIfMissing / writeTextIfMissing) keep their (created, skipped)
|
|
61
|
+
// signature. The explicit `pugi init` command forwards these straight
|
|
62
|
+
// into its JSON payload.
|
|
63
|
+
const created = [];
|
|
64
|
+
const skipped = [];
|
|
65
|
+
mkdirIfMissing(pugiDir, created, skipped);
|
|
66
|
+
mkdirIfMissing(resolve(pugiDir, 'artifacts'), created, skipped);
|
|
67
|
+
mkdirIfMissing(resolve(pugiDir, 'sessions'), created, skipped);
|
|
68
|
+
mkdirIfMissing(resolve(pugiDir, 'skills'), created, skipped);
|
|
69
|
+
writeJsonIfMissing(resolve(pugiDir, 'settings.json'), {
|
|
70
|
+
schema: 1,
|
|
71
|
+
workflow: {
|
|
72
|
+
brand: 'pugi',
|
|
73
|
+
legacyName: 'codeforge',
|
|
74
|
+
approvals: 'auto',
|
|
75
|
+
notAutomatic: [],
|
|
76
|
+
defaultBaseBranch: 'dev',
|
|
77
|
+
branchPrefixes: ['feature', 'fix', 'refactor', 'chore'],
|
|
78
|
+
aiCoAuthorTrailers: false,
|
|
79
|
+
},
|
|
80
|
+
permissions: {
|
|
81
|
+
mode: 'auto',
|
|
82
|
+
allow: [],
|
|
83
|
+
deny: [],
|
|
84
|
+
notAutomatic: [],
|
|
85
|
+
},
|
|
86
|
+
privacy: {
|
|
87
|
+
mode: 'balanced',
|
|
88
|
+
telemetry: 'off',
|
|
89
|
+
},
|
|
90
|
+
artifacts: {
|
|
91
|
+
defaultPath: '.pugi/artifacts',
|
|
92
|
+
promoteExplicitly: true,
|
|
93
|
+
},
|
|
94
|
+
}, created, skipped);
|
|
95
|
+
writeJsonIfMissing(resolve(pugiDir, 'mcp.json'), { schema: 1, servers: [] }, created, skipped);
|
|
96
|
+
writeJsonIfMissing(resolve(pugiDir, 'index.json'), emptyIndex(), created, skipped);
|
|
97
|
+
writeTextIfMissing(resolve(pugiDir, 'PUGI.md'), [
|
|
98
|
+
'# Pugi Project Context',
|
|
99
|
+
'',
|
|
100
|
+
'## Product Workflow',
|
|
101
|
+
'',
|
|
102
|
+
'- Public product name: Pugi',
|
|
103
|
+
'- Default flow: idea -> build -> review',
|
|
104
|
+
'- Approvals are automatic by default until a repo, environment, workflow, or action is marked notAutomatic.',
|
|
105
|
+
'- Do not add AI Co-Authored-By trailers.',
|
|
106
|
+
'- Generated code, comments, commits, PR text, and technical docs default to English.',
|
|
107
|
+
'',
|
|
108
|
+
'## Project Notes',
|
|
109
|
+
'',
|
|
110
|
+
'- Add repo-specific architecture, commands, and business rules here.',
|
|
111
|
+
'- Do not store secrets, real IPs, private key paths, tokens, or credentials here.',
|
|
112
|
+
'',
|
|
113
|
+
].join('\n'), created, skipped);
|
|
114
|
+
writeTextIfMissing(resolve(cwd, '.pugiignore'), [
|
|
115
|
+
'# Pugi ignore rules',
|
|
116
|
+
'.env',
|
|
117
|
+
'.env.*',
|
|
118
|
+
'!.env.example',
|
|
119
|
+
'node_modules/',
|
|
120
|
+
'dist/',
|
|
121
|
+
'.next/',
|
|
122
|
+
'coverage/',
|
|
123
|
+
'*.log',
|
|
124
|
+
'*.pem',
|
|
125
|
+
'*.key',
|
|
126
|
+
'*.crt',
|
|
127
|
+
'*.p12',
|
|
128
|
+
'*.sql',
|
|
129
|
+
'*.dump',
|
|
130
|
+
'',
|
|
131
|
+
].join('\n'), created, skipped);
|
|
132
|
+
ensurePugiGitIgnore(cwd, created, skipped);
|
|
133
|
+
// `silent` is honoured implicitly — this module never writes to
|
|
134
|
+
// stdout/stderr. The flag exists so the standalone `pugi init` command
|
|
135
|
+
// can layer its own logger on top (it does, in runtime/cli.ts), while
|
|
136
|
+
// the auto-init REPL path leaves the boot stream untouched. We
|
|
137
|
+
// reference the flag here to defeat the lint "unused" warning and to
|
|
138
|
+
// document the contract in the source.
|
|
139
|
+
void silent;
|
|
140
|
+
return {
|
|
141
|
+
created: created.length > 0,
|
|
142
|
+
dir: pugiDir,
|
|
143
|
+
createdPaths: created,
|
|
144
|
+
skippedPaths: skipped,
|
|
145
|
+
};
|
|
146
|
+
}
|
|
147
|
+
/* ------------------------------------------------------------------ */
|
|
148
|
+
/* Helpers (mirror the previous in-file implementations in cli.ts) */
|
|
149
|
+
/* ------------------------------------------------------------------ */
|
|
150
|
+
function mkdirIfMissing(path, created, skipped) {
|
|
151
|
+
if (existsSync(path)) {
|
|
152
|
+
skipped.push(path);
|
|
153
|
+
return;
|
|
154
|
+
}
|
|
155
|
+
mkdirSync(path, { recursive: true });
|
|
156
|
+
created.push(path);
|
|
157
|
+
}
|
|
158
|
+
function writeJsonIfMissing(path, value, created, skipped) {
|
|
159
|
+
writeTextIfMissing(path, `${JSON.stringify(value, null, 2)}\n`, created, skipped);
|
|
160
|
+
}
|
|
161
|
+
function writeTextIfMissing(path, value, created, skipped) {
|
|
162
|
+
if (existsSync(path)) {
|
|
163
|
+
skipped.push(path);
|
|
164
|
+
return;
|
|
165
|
+
}
|
|
166
|
+
writeFileSync(path, value, { encoding: 'utf8', mode: 0o600 });
|
|
167
|
+
created.push(path);
|
|
168
|
+
}
|
|
169
|
+
/**
|
|
170
|
+
* Ensure the workspace `.gitignore` ignores `.pugi/`. The function is
|
|
171
|
+
* additive: it leaves an existing `.gitignore` body intact and appends
|
|
172
|
+
* the marker only when none of `.pugi/`, `/.pugi/`, or `.pugi` is
|
|
173
|
+
* already present. On a fresh repo with no `.gitignore` it creates the
|
|
174
|
+
* file with the single marker line. Mode 0o600 matches the rest of the
|
|
175
|
+
* scaffold so a paranoid CI does not surface "world-readable" warnings.
|
|
176
|
+
*/
|
|
177
|
+
function ensurePugiGitIgnore(cwd, created, skipped) {
|
|
178
|
+
const gitignorePath = resolve(cwd, '.gitignore');
|
|
179
|
+
const marker = '.pugi/';
|
|
180
|
+
if (!existsSync(gitignorePath)) {
|
|
181
|
+
writeFileSync(gitignorePath, `${marker}\n`, { encoding: 'utf8', mode: 0o600 });
|
|
182
|
+
created.push(gitignorePath);
|
|
183
|
+
return;
|
|
184
|
+
}
|
|
185
|
+
const current = readFileSync(gitignorePath, 'utf8');
|
|
186
|
+
const lines = current.split('\n').map((line) => line.trim());
|
|
187
|
+
if (lines.includes(marker) || lines.includes('/.pugi/') || lines.includes('.pugi')) {
|
|
188
|
+
skipped.push(gitignorePath);
|
|
189
|
+
return;
|
|
190
|
+
}
|
|
191
|
+
const next = current.endsWith('\n') ? `${current}${marker}\n` : `${current}\n${marker}\n`;
|
|
192
|
+
writeFileSync(gitignorePath, next, { encoding: 'utf8' });
|
|
193
|
+
created.push(`${gitignorePath} (+${marker})`);
|
|
194
|
+
}
|
|
195
|
+
//# sourceMappingURL=scaffold.js.map
|