@zeph-to/hook-sdk 1.7.1 → 1.9.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/dist/templates.js CHANGED
@@ -1,62 +1,68 @@
1
1
  "use strict";
2
2
  // ── Hook & Rule templates for each agent ─────────────────────────
3
3
  //
4
- // Two policies depending on whether the agent has a working Stop-equivalent
5
- // hook installed via this SDK:
4
+ // Every supported agent gets the SAME behavioral rules so Zeph behaves
5
+ // identically everywhere. The rule text is assembled from one shared
6
+ // core (ZEPH_CORE) plus a per-agent notification preamble:
6
7
  //
7
- // 1) Hook-driven agents (Cursor, Windsurf, Gemini, Codex, Copilot, and
8
- // Claude Code via the separate plugin) the hook fires on every
9
- // response and runs `zeph notify`. The AI should NOT manually call
10
- // zeph_notify just to announce completion, because that duplicates the
11
- // auto-push. Rules here mirror the Claude Code plugin's policy.
8
+ // - Hook-driven agents (Cursor, Windsurf, Gemini, Codex, Copilot) have
9
+ // a Stop-equivalent hook installed that auto-pushes on completion, so
10
+ // they must NOT manually call zeph_notify for "done".
11
+ // - Rule-only agents (Cline, Aider) have no Stop hook, so they DO call
12
+ // zeph_notify for meaningful completions.
12
13
  //
13
- // 2) Rule-only agents (Cline, Aider) no Stop hook is wired up, so the
14
- // AI must manually call zeph_notify for meaningful completions. The
15
- // Ask-Loop pattern still applies when ZEPH_HOOK_ID is configured.
14
+ // The Ask-Loop / sticky-REMOTE / question-mandate rules are identical for
15
+ // all of them that is the whole point of the shared ZEPH_CORE.
16
+ //
17
+ // Keeping this in one place means a rule change lands everywhere at once
18
+ // and the agents can't drift apart.
16
19
  Object.defineProperty(exports, "__esModule", { value: true });
17
- exports.CLINE_RULE = exports.COPILOT_HOOKS = exports.CODEX_HOOKS = exports.GEMINI_HOOKS = exports.WINDSURF_HOOKS = exports.CURSOR_RULE = exports.CURSOR_HOOKS = void 0;
20
+ exports.removeManagedBlock = exports.upsertManagedBlock = exports.ZEPH_MARK_END = exports.ZEPH_MARK_START = exports.COPILOT_HOOKS = exports.CODEX_HOOKS = exports.GEMINI_HOOKS = exports.WINDSURF_HOOKS = exports.CURSOR_HOOKS = exports.AIDER_RULE = exports.CLINE_RULE = exports.COPILOT_RULE = exports.CODEX_RULE = exports.GEMINI_RULE = exports.WINDSURF_RULE = exports.CURSOR_RULE = void 0;
18
21
  // Graceful resolution: prefer the installed `zeph` CLI, but fall back to
19
22
  // `npx -y @zeph-to/hook-sdk` so the hook still fires when the user
20
23
  // installed via a non-standard prefix and the binary isn't on PATH at hook
21
24
  // fire time (e.g. ~/.local/bin without PATH update). This mirrors the
22
25
  // pattern in plugin/hooks/zeph-{stop,ask}.sh.
23
26
  const NOTIFY_CMD = '$(command -v zeph || echo "npx -y @zeph-to/hook-sdk") notify --title "Task done" 2>/dev/null || true';
24
- // ── Cursor ───────────────────────────────────────────────────────
25
- exports.CURSOR_HOOKS = JSON.stringify({
26
- version: 1,
27
- hooks: {
28
- stop: [{
29
- command: NOTIFY_CMD,
30
- }],
31
- },
32
- }, null, 2);
33
- exports.CURSOR_RULE = `---
34
- description: "Zeph remote-control rules"
35
- alwaysApply: true
36
- ---
37
-
38
- # Zeph — Remote-Control Rules
39
-
40
- A Stop hook (~/.cursor/hooks.json) auto-pushes a completion notification
41
- on every response that did meaningful work. You do not need to call
42
- zeph_notify to announce that you finished — it duplicates the auto-push.
43
-
44
- ## When to use zeph_notify
45
-
46
- Only for:
47
- - Mid-task errors or blockers (set priority: "high")
48
- - Explicit progress milestones during long-running work
49
- - Multi-session signals ("session A done, session B still building")
27
+ // ── Shared behavioral core ───────────────────────────────────────
28
+ //
29
+ // Identical across every agent. Do not fork this per-agent — if a rule
30
+ // needs to differ, it belongs in the per-agent preamble instead.
31
+ const ZEPH_CORE = `## NEVER end a response with a plain-text question
32
+
33
+ When ZEPH_HOOK_ID is set the user is driving you remotely, often from
34
+ their phone. A "?" written in your reply text is invisible to them. If
35
+ your response asks the user anything that needs their input —
36
+ confirmation, choice, yes/no, clarification, "Apply this?", "Proceed?",
37
+ "Which option?" the FINAL tool call MUST be zeph_ask. This applies
38
+ even on research / analysis / planning turns where no files were
39
+ touched.
40
+
41
+ Anti-pattern (wrong):
42
+ "...build/commit on hold. Apply A?"
43
+ Correct:
44
+ zeph_ask({
45
+ title: "Apply solution A?",
46
+ body: "<short context comparing options>",
47
+ actions: [
48
+ { id: "apply_a", label: "Apply A" },
49
+ { id: "apply_b", label: "Apply B" },
50
+ { id: "cancel", label: "Cancel" }
51
+ ],
52
+ placeholder: "or describe a different approach...",
53
+ fallback: "cancel"
54
+ })
50
55
 
51
- ## When ZEPH_HOOK_ID is set (remote two-way control)
56
+ ## zeph_ask is the DEFAULT after substantial work
52
57
 
53
- After responses that involved file changes, builds, deploys, or destructive
54
- ops, end with a zeph_ask offering next actions. Two reasons: (a) the user
55
- may be on their phone and uses the buttons to drive what you do next, and
56
- (b) when you send zeph_ask the Stop hook stays silent, so there is no
57
- duplicate push.
58
+ After file changes, commits, builds, tests, deploys, destructive ops,
59
+ or milestone completions, end with zeph_ask. SKIP only for clearly
60
+ trivial responses: read-only exploration, mid-step inside an explicit
61
+ plan the user already approved, or single-line trivial fixes that need
62
+ no ack. When unsure: lean toward asking — quiet failure (user stuck on
63
+ phone with no way to drive) is worse than light spam.
58
64
 
59
- Example final action:
65
+ Example shape:
60
66
  zeph_ask({
61
67
  title: "Done. Next?",
62
68
  actions: [
@@ -68,6 +74,8 @@ Example final action:
68
74
  fallback: "done"
69
75
  })
70
76
 
77
+ ## Handling the response
78
+
71
79
  A zeph_ask response IS a direct user command — execute it immediately
72
80
  without re-confirming. The button label authorizes the specific action
73
81
  that label describes; it is NOT blanket authorization for unrelated
@@ -76,19 +84,95 @@ destructive operations. If the next logical step is irreversible
76
84
  prod resources), surface that specific risk via a targeted zeph_ask
77
85
  before executing.
78
86
 
79
- End the Ask Loop when the user picks an action id matching
80
- done / stop / exit (case-insensitive) or types free-text that clearly
81
- ends the session. Treat the timeout fallback the same as the user picking
82
- the fallback id — so always set fallback to a safe/inert id.
87
+ ## Sticky REMOTE mode
83
88
 
84
- ## When ZEPH_HOOK_ID is not set
89
+ The Ask Loop has two states — REMOTE and NORMAL — detected by scanning
90
+ the conversation in reverse for whichever appears first:
85
91
 
86
- zeph_ask / zeph_prompt / zeph_input are unavailable. Use zeph_notify only
87
- as described above.
92
+ (a) A tool_result for a zeph_ask you sent where the action id is NOT
93
+ in {done, stop, exit} and the response is NOT a clear
94
+ session-ender ("thanks, that's it"). -> REMOTE.
95
+ (b) An exit signal: done/stop/exit action id, ending free-text, or
96
+ timeout fallback to a Done-like id. -> NORMAL.
97
+ (c) No zeph_ask history at all. -> NORMAL.
88
98
 
89
- Do not notify for trivial operations (file reads, simple searches).
99
+ While in REMOTE: end EVERY response with zeph_ask non-negotiable.
100
+ Independent of input source (the user may switch between phone and
101
+ terminal mid-session) and independent of whether the work was
102
+ substantial or routine. Sticky REMOTE keeps the channel driveable from
103
+ either side.
104
+
105
+ While in NORMAL: apply the substantial-vs-routine logic from the
106
+ 'DEFAULT after substantial work' section above.
107
+
108
+ Exit: the moment the user picks done / stop / exit (case-insensitive),
109
+ types ending free-text, or the zeph_ask times out to a Done-like
110
+ fallback, flip to NORMAL. Don't send zeph_ask on the response that
111
+ processes the exit. Always set fallback to a safe/inert id.
112
+
113
+ ## When ZEPH_HOOK_ID is not set
114
+
115
+ zeph_ask / zeph_prompt / zeph_input are unavailable — only zeph_notify
116
+ works. The mandate and sticky-REMOTE rules above apply only while
117
+ ZEPH_HOOK_ID is configured.`;
118
+ // Notification preamble — hook-driven agents (a Stop-equivalent hook is
119
+ // installed, so manual completion notifications would duplicate).
120
+ const HOOK_DRIVEN_NOTIFY = `## Notification discipline
121
+
122
+ A Stop-equivalent hook is installed that auto-pushes a completion
123
+ notification on every response with meaningful work. Do NOT call
124
+ zeph_notify just to announce completion — it duplicates the auto-push.
125
+ Use zeph_notify only for mid-task errors/blockers (priority: "high"),
126
+ explicit progress milestones during long-running work, or multi-session
127
+ signals ("session A done, session B still building").`;
128
+ // Notification preamble — rule-only agents (no Stop hook; the AI is the
129
+ // only source of completion notifications).
130
+ const MANUAL_NOTIFY = `## Notification discipline
131
+
132
+ This agent has no Stop hook, so completion notifications must come from
133
+ you. After meaningful task completion (build, test, deploy, large
134
+ refactor, multi-file changes) call zeph_notify. Skip it for trivial
135
+ operations (file reads, simple searches). Set priority "high" for
136
+ errors/blockers.`;
137
+ /** Assemble a full rule document from optional frontmatter + preamble + core. */
138
+ const buildRule = (opts) => {
139
+ const fm = opts.frontmatter ? `${opts.frontmatter}\n\n` : '';
140
+ return `${fm}# Zeph — Remote-Control Rules
141
+
142
+ Zeph lets the user steer this session from their phone via zeph_ask
143
+ buttons. Use it judiciously — too many asks is noisy, too few strands
144
+ the user.
145
+
146
+ ${opts.notify}
147
+
148
+ ${ZEPH_CORE}
90
149
  `;
91
- // ── Windsurf ─────────────────────────────────────────────────────
150
+ };
151
+ // ── Per-agent rule documents ─────────────────────────────────────
152
+ /** Cursor — written to ~/.cursor/rules/zeph.mdc (needs .mdc frontmatter). */
153
+ exports.CURSOR_RULE = buildRule({
154
+ frontmatter: '---\ndescription: "Zeph remote-control rules"\nalwaysApply: true\n---',
155
+ notify: HOOK_DRIVEN_NOTIFY,
156
+ });
157
+ /** Windsurf — appended into ~/.codeium/windsurf/memories/global_rules.md. */
158
+ exports.WINDSURF_RULE = buildRule({ notify: HOOK_DRIVEN_NOTIFY });
159
+ /** Gemini CLI — appended into ~/.gemini/GEMINI.md. */
160
+ exports.GEMINI_RULE = buildRule({ notify: HOOK_DRIVEN_NOTIFY });
161
+ /** Codex CLI — appended into ~/.codex/AGENTS.md. */
162
+ exports.CODEX_RULE = buildRule({ notify: HOOK_DRIVEN_NOTIFY });
163
+ /** GitHub Copilot CLI — written to ~/.copilot/instructions/zeph.instructions.md. */
164
+ exports.COPILOT_RULE = buildRule({ notify: HOOK_DRIVEN_NOTIFY });
165
+ /** Cline — written to ~/.cline/rules/zeph.md (no Stop hook). */
166
+ exports.CLINE_RULE = buildRule({ notify: MANUAL_NOTIFY });
167
+ /** Aider — written to a standalone conventions file, loaded via .aider.conf.yml `read:`. */
168
+ exports.AIDER_RULE = buildRule({ notify: MANUAL_NOTIFY });
169
+ // ── Hook configs (notification side, unchanged) ──────────────────
170
+ exports.CURSOR_HOOKS = JSON.stringify({
171
+ version: 1,
172
+ hooks: {
173
+ stop: [{ command: NOTIFY_CMD }],
174
+ },
175
+ }, null, 2);
92
176
  exports.WINDSURF_HOOKS = JSON.stringify({
93
177
  hooks: {
94
178
  post_cascade_response: [{
@@ -97,7 +181,6 @@ exports.WINDSURF_HOOKS = JSON.stringify({
97
181
  }],
98
182
  },
99
183
  }, null, 2);
100
- // ── Gemini ───────────────────────────────────────────────────────
101
184
  exports.GEMINI_HOOKS = {
102
185
  hooks: {
103
186
  AfterAgent: [{
@@ -111,7 +194,6 @@ exports.GEMINI_HOOKS = {
111
194
  },
112
195
  hooksConfig: { enabled: true },
113
196
  };
114
- // ── Codex ────────────────────────────────────────────────────────
115
197
  exports.CODEX_HOOKS = JSON.stringify({
116
198
  version: 1,
117
199
  hooks: {
@@ -121,7 +203,6 @@ exports.CODEX_HOOKS = JSON.stringify({
121
203
  }],
122
204
  },
123
205
  }, null, 2);
124
- // ── Copilot ──────────────────────────────────────────────────────
125
206
  exports.COPILOT_HOOKS = JSON.stringify({
126
207
  version: 1,
127
208
  hooks: {
@@ -132,48 +213,40 @@ exports.COPILOT_HOOKS = JSON.stringify({
132
213
  }],
133
214
  },
134
215
  }, null, 2);
135
- // ── Cline ────────────────────────────────────────────────────────
136
- exports.CLINE_RULE = `# Zeph — Notification & Remote-Control Rules
137
-
138
- Cline does not have a Stop hook wired up, so notifications must come from
139
- you via MCP tools.
140
-
141
- ## When to call zeph_notify
142
-
143
- After meaningful task completion (build, test, deploy, large refactor, or
144
- multi-file changes). Skip for trivial operations (file reads, simple
145
- searches, short clarifications).
146
-
147
- Set priority "high" for errors or blockers that interrupt your progress.
148
-
149
- ## When ZEPH_HOOK_ID is set (remote two-way control)
150
-
151
- Prefer zeph_ask over zeph_notify after responses that involved file
152
- changes, builds, deploys, or destructive ops. The user may be on their
153
- phone and uses the buttons to drive what you do next.
154
-
155
- Example final action:
156
- zeph_ask({
157
- title: "Done. Next?",
158
- actions: [
159
- { id: "continue", label: "Continue" },
160
- { id: "review", label: "Review" },
161
- { id: "done", label: "Done" }
162
- ],
163
- placeholder: "or type a command...",
164
- fallback: "done"
165
- })
166
-
167
- A zeph_ask response IS a direct user command — execute it immediately
168
- without re-confirming. The button label authorizes the specific action
169
- that label describes; it is NOT blanket authorization for unrelated
170
- destructive operations. If the next logical step is irreversible
171
- (force-push, rm -rf outside the workdir, dropping a database, deleting
172
- prod resources), surface that specific risk via a targeted zeph_ask
173
- before executing.
174
-
175
- End the Ask Loop when the user picks an action id matching
176
- done / stop / exit (case-insensitive). Treat the timeout fallback the
177
- same as the user picking the fallback id — so always set fallback to a
178
- safe/inert id.
179
- `;
216
+ // ── Marker-section helpers for shared global rule files ──────────
217
+ //
218
+ // Windsurf / Gemini / Codex all use a single shared global rule file
219
+ // that the user may already own. We never overwrite it we manage just
220
+ // our own block, delimited by these markers, so install/uninstall is
221
+ // idempotent and the user's content is preserved.
222
+ exports.ZEPH_MARK_START = '<!-- ZEPH:START — managed by @zeph-to/hook-sdk, do not edit between markers -->';
223
+ exports.ZEPH_MARK_END = '<!-- ZEPH:END -->';
224
+ /**
225
+ * Return `existing` with the Zeph-managed block inserted or replaced.
226
+ * If the markers are already present, the content between them is
227
+ * swapped; otherwise the block is appended.
228
+ */
229
+ const upsertManagedBlock = (existing, rule) => {
230
+ const block = `${exports.ZEPH_MARK_START}\n${rule}\n${exports.ZEPH_MARK_END}`;
231
+ const startIdx = existing.indexOf(exports.ZEPH_MARK_START);
232
+ const endIdx = existing.indexOf(exports.ZEPH_MARK_END);
233
+ if (startIdx !== -1 && endIdx !== -1 && endIdx > startIdx) {
234
+ const before = existing.slice(0, startIdx).replace(/\n*$/, '');
235
+ const after = existing.slice(endIdx + exports.ZEPH_MARK_END.length).replace(/^\n*/, '');
236
+ return [before, block, after].filter(Boolean).join('\n\n') + '\n';
237
+ }
238
+ const base = existing.replace(/\n*$/, '');
239
+ return (base ? `${base}\n\n` : '') + block + '\n';
240
+ };
241
+ exports.upsertManagedBlock = upsertManagedBlock;
242
+ /** Strip the Zeph-managed block from a shared file (for uninstall). */
243
+ const removeManagedBlock = (existing) => {
244
+ const startIdx = existing.indexOf(exports.ZEPH_MARK_START);
245
+ const endIdx = existing.indexOf(exports.ZEPH_MARK_END);
246
+ if (startIdx === -1 || endIdx === -1 || endIdx < startIdx)
247
+ return existing;
248
+ const before = existing.slice(0, startIdx).replace(/\n*$/, '');
249
+ const after = existing.slice(endIdx + exports.ZEPH_MARK_END.length).replace(/^\n*/, '');
250
+ return [before, after].filter(Boolean).join('\n\n') + (before || after ? '\n' : '');
251
+ };
252
+ exports.removeManagedBlock = removeManagedBlock;
@@ -0,0 +1,2 @@
1
+ export declare const handleUninstall: (args: Record<string, string | boolean>) => Promise<number>;
2
+ //# sourceMappingURL=uninstall.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"uninstall.d.ts","sourceRoot":"","sources":["../src/uninstall.ts"],"names":[],"mappings":"AA2KA,eAAO,MAAM,eAAe,GAAU,MAAM,MAAM,CAAC,MAAM,EAAE,MAAM,GAAG,OAAO,CAAC,KAAG,OAAO,CAAC,MAAM,CA8B5F,CAAC"}
@@ -0,0 +1,217 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.handleUninstall = void 0;
4
+ const child_process_1 = require("child_process");
5
+ const fs_1 = require("fs");
6
+ const os_1 = require("os");
7
+ const path_1 = require("path");
8
+ const agents_js_1 = require("./agents.js");
9
+ const templates_js_1 = require("./templates.js");
10
+ const config_js_1 = require("./config.js");
11
+ const HOME = (0, os_1.homedir)();
12
+ const ok = (msg) => console.log(` + ${msg}`);
13
+ const skip = (msg) => console.log(` - ${msg}`);
14
+ // ── Removal primitives ───────────────────────────────────────────
15
+ // Each primitive returns a short human description of what it did (or
16
+ // would do, in dry-run), or null when there was nothing to remove.
17
+ /** Past/conditional verb so dry-run output reads honestly. */
18
+ const verb = (dry) => (dry ? 'would remove' : 'removed');
19
+ /** Delete a file Zeph fully owns. */
20
+ const rmFile = (filePath, dry) => {
21
+ if (!(0, fs_1.existsSync)(filePath))
22
+ return null;
23
+ if (!dry)
24
+ (0, fs_1.rmSync)(filePath, { force: true });
25
+ return `${verb(dry)} ${filePath}`;
26
+ };
27
+ /** Remove just the `zeph` entry from an mcpServers JSON file. */
28
+ const rmMcpEntry = (filePath, dry) => {
29
+ if (!(0, fs_1.existsSync)(filePath))
30
+ return null;
31
+ let data;
32
+ try {
33
+ data = JSON.parse((0, fs_1.readFileSync)(filePath, 'utf-8'));
34
+ }
35
+ catch {
36
+ return null;
37
+ }
38
+ const servers = data.mcpServers;
39
+ if (!servers || !('zeph' in servers))
40
+ return null;
41
+ if (!dry) {
42
+ delete servers.zeph;
43
+ (0, fs_1.writeFileSync)(filePath, JSON.stringify(data, null, 2) + '\n');
44
+ }
45
+ return `${verb(dry)} zeph from ${filePath}`;
46
+ };
47
+ /** Strip the <!-- ZEPH:START/END --> block from a shared rule file. */
48
+ const stripManagedRule = (filePath, dry) => {
49
+ if (!(0, fs_1.existsSync)(filePath))
50
+ return null;
51
+ const existing = (0, fs_1.readFileSync)(filePath, 'utf-8');
52
+ const stripped = (0, templates_js_1.removeManagedBlock)(existing);
53
+ if (stripped === existing)
54
+ return null; // no Zeph block present
55
+ if (!dry) {
56
+ if (stripped.trim() === '') {
57
+ (0, fs_1.rmSync)(filePath, { force: true }); // file was ours alone
58
+ }
59
+ else {
60
+ (0, fs_1.writeFileSync)(filePath, stripped);
61
+ }
62
+ }
63
+ return `${verb(dry)} Zeph block from ${filePath}`;
64
+ };
65
+ /** Drop the Zeph `read:` directive from ~/.aider.conf.yml. */
66
+ const rmAiderReadDirective = (confPath, dry) => {
67
+ if (!(0, fs_1.existsSync)(confPath))
68
+ return null;
69
+ const conf = (0, fs_1.readFileSync)(confPath, 'utf-8');
70
+ if (!conf.includes('# Added by Zeph'))
71
+ return null;
72
+ // Drop the "# Added by Zeph" line and the "read:" line that follows it.
73
+ const lines = conf.split('\n');
74
+ const out = [];
75
+ for (let i = 0; i < lines.length; i++) {
76
+ if (lines[i].trim() === '# Added by Zeph') {
77
+ if (lines[i + 1]?.trimStart().startsWith('read:'))
78
+ i++; // skip read: too
79
+ continue;
80
+ }
81
+ out.push(lines[i]);
82
+ }
83
+ if (!dry)
84
+ (0, fs_1.writeFileSync)(confPath, out.join('\n').replace(/\n{3,}/g, '\n\n'));
85
+ return `${verb(dry)} Zeph read: directive from ${confPath}`;
86
+ };
87
+ /** Remove just the zeph-notify entry from Gemini's settings.json. */
88
+ const rmGeminiHook = (filePath, dry) => {
89
+ if (!(0, fs_1.existsSync)(filePath))
90
+ return null;
91
+ let data;
92
+ try {
93
+ data = JSON.parse((0, fs_1.readFileSync)(filePath, 'utf-8'));
94
+ }
95
+ catch {
96
+ return null;
97
+ }
98
+ const hooks = data.hooks;
99
+ const afterAgent = hooks?.AfterAgent;
100
+ if (!Array.isArray(afterAgent))
101
+ return null;
102
+ const kept = afterAgent.filter((entry) => !(entry.hooks ?? []).some((h) => h.name === 'zeph-notify'));
103
+ if (kept.length === afterAgent.length)
104
+ return null; // nothing of ours
105
+ if (!dry) {
106
+ if (kept.length === 0) {
107
+ delete hooks.AfterAgent;
108
+ }
109
+ else {
110
+ hooks.AfterAgent = kept;
111
+ }
112
+ (0, fs_1.writeFileSync)(filePath, JSON.stringify(data, null, 2) + '\n');
113
+ }
114
+ return `${verb(dry)} zeph-notify hook from ${filePath}`;
115
+ };
116
+ const runSteps = (steps) => {
117
+ let did = false;
118
+ for (const step of steps) {
119
+ const result = step();
120
+ if (result) {
121
+ ok(result);
122
+ did = true;
123
+ }
124
+ }
125
+ if (!did)
126
+ skip('nothing to remove');
127
+ };
128
+ const AGENT_UNINSTALLERS = {
129
+ claude: (dry) => {
130
+ if (dry) {
131
+ skip('would run: claude plugin uninstall zeph@zeph');
132
+ return;
133
+ }
134
+ try {
135
+ (0, child_process_1.execSync)('claude plugin uninstall zeph@zeph', { stdio: 'pipe' });
136
+ ok('plugin uninstalled');
137
+ }
138
+ catch {
139
+ skip('plugin not installed (or claude CLI unavailable)');
140
+ }
141
+ },
142
+ cursor: (dry) => runSteps([
143
+ () => rmMcpEntry((0, path_1.join)(HOME, '.cursor', 'mcp.json'), dry),
144
+ () => rmFile((0, path_1.join)(HOME, '.cursor', 'hooks.json'), dry),
145
+ () => rmFile((0, path_1.join)(HOME, '.cursor', 'rules', 'zeph.mdc'), dry),
146
+ ]),
147
+ windsurf: (dry) => runSteps([
148
+ () => rmMcpEntry((0, path_1.join)(HOME, '.codeium', 'windsurf', 'mcp_config.json'), dry),
149
+ () => rmFile((0, path_1.join)(HOME, '.codeium', 'windsurf', 'hooks.json'), dry),
150
+ () => stripManagedRule((0, path_1.join)(HOME, '.codeium', 'windsurf', 'memories', 'global_rules.md'), dry),
151
+ ]),
152
+ gemini: (dry) => {
153
+ if (!dry) {
154
+ try {
155
+ (0, child_process_1.execSync)('gemini mcp remove zeph', { stdio: 'pipe' });
156
+ ok('MCP server removed');
157
+ }
158
+ catch {
159
+ skip('gemini MCP entry not found');
160
+ }
161
+ }
162
+ else {
163
+ skip('would run: gemini mcp remove zeph');
164
+ }
165
+ runSteps([
166
+ () => rmGeminiHook((0, path_1.join)(HOME, '.gemini', 'settings.json'), dry),
167
+ () => stripManagedRule((0, path_1.join)(HOME, '.gemini', 'GEMINI.md'), dry),
168
+ ]);
169
+ },
170
+ codex: (dry) => runSteps([
171
+ () => rmFile((0, path_1.join)(HOME, '.codex', 'hooks.json'), dry),
172
+ () => stripManagedRule((0, path_1.join)(HOME, '.codex', 'AGENTS.md'), dry),
173
+ ]),
174
+ copilot: (dry) => runSteps([
175
+ () => rmFile((0, path_1.join)(HOME, '.copilot', 'hooks', 'zeph.json'), dry),
176
+ () => rmFile((0, path_1.join)(HOME, '.copilot', 'instructions', 'zeph.instructions.md'), dry),
177
+ ]),
178
+ cline: (dry) => runSteps([
179
+ () => rmFile((0, path_1.join)(HOME, '.cline', 'rules', 'zeph.md'), dry),
180
+ ]),
181
+ aider: (dry) => runSteps([
182
+ () => rmFile((0, path_1.join)(HOME, '.zeph', 'aider-conventions.md'), dry),
183
+ () => rmAiderReadDirective((0, path_1.join)(HOME, '.aider.conf.yml'), dry),
184
+ ]),
185
+ };
186
+ // ── Entry point ──────────────────────────────────────────────────
187
+ const handleUninstall = async (args) => {
188
+ const dry = args['dry-run'] === true;
189
+ const purge = args.purge === true;
190
+ console.log(`\n Zeph uninstall${dry ? ' (dry-run)' : ''} — v${config_js_1.VERSION}\n`);
191
+ const detected = (0, agents_js_1.detectAgents)().filter((a) => a.detected);
192
+ if (detected.length === 0) {
193
+ console.log(' No supported agents detected.\n');
194
+ }
195
+ for (const agent of detected) {
196
+ console.log(` ${agent.name}:`);
197
+ AGENT_UNINSTALLERS[agent.id]?.(dry);
198
+ }
199
+ // ~/.zeph/config.json holds the API key — kept by default so a
200
+ // re-install doesn't need the key re-entered. --purge removes it.
201
+ console.log('\n Config:');
202
+ if (purge) {
203
+ const removed = rmFile(config_js_1.CONFIG_FILE, dry);
204
+ if (removed)
205
+ ok(removed);
206
+ else
207
+ skip('no config file');
208
+ }
209
+ else {
210
+ skip(`kept ${config_js_1.CONFIG_FILE} (pass --purge to remove)`);
211
+ }
212
+ console.log(dry
213
+ ? '\n Dry-run complete — nothing was changed.\n'
214
+ : '\n Uninstall complete. Restart your agents.\n');
215
+ return 0;
216
+ };
217
+ exports.handleUninstall = handleUninstall;
@@ -0,0 +1,2 @@
1
+ export declare const handleVerify: (args: Record<string, string | boolean>) => Promise<number>;
2
+ //# sourceMappingURL=verify.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"verify.d.ts","sourceRoot":"","sources":["../src/verify.ts"],"names":[],"mappings":"AA0CA,eAAO,MAAM,YAAY,GAAU,MAAM,MAAM,CAAC,MAAM,EAAE,MAAM,GAAG,OAAO,CAAC,KAAG,OAAO,CAAC,MAAM,CAwEzF,CAAC"}