@ijfw/install 1.1.1 → 1.1.2
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 +57 -0
- package/dist/install.js +9 -2
- package/dist/uninstall.js +99 -10
- package/docs/GUIDE.md +1 -1
- package/package.json +1 -1
package/CHANGELOG.md
CHANGED
|
@@ -1,5 +1,62 @@
|
|
|
1
1
|
# Changelog -- @ijfw/install
|
|
2
2
|
|
|
3
|
+
## [1.1.2] -- 2026-04-21
|
|
4
|
+
|
|
5
|
+
Reach + bug fixes. Two new platforms (Hermes + Wayland), deep installer repairs uncovered by live-platform testing, cross-platform sync of new behavioral rules. Ships with a full end-to-end smoke harness at `scripts/e2e-smoke.sh` that now has to pass before any future release.
|
|
6
|
+
|
|
7
|
+
### New platforms
|
|
8
|
+
|
|
9
|
+
- **Hermes** (`hermes/`): MCP registration in `~/.hermes/config.yaml`, `HERMES.md` context file, 19 IJFW skills dropped into `~/.hermes/skills/ijfw-*` in agentskills.io format. Python CLI (`hermes` command).
|
|
10
|
+
- **Wayland** (`wayland/`): same shape. MCP registration in `~/.wayland/config.yaml`, `WAYLAND.md` context file, skills bundle in `~/.wayland/skills/ijfw-*`. Python CLI (`wayland` command).
|
|
11
|
+
- `scripts/install.sh` gains a `merge_yaml_mcp` helper (prefers python3+PyYAML for parser-safe merge; sentinel-anchored fallback if PyYAML isn't available).
|
|
12
|
+
- Default target list expands to 8: `claude codex gemini cursor windsurf copilot hermes wayland`. `is_live` and `pretty_name` updated to match.
|
|
13
|
+
|
|
14
|
+
### Installer repairs (high -- announcement blockers)
|
|
15
|
+
|
|
16
|
+
- **Bug A: platform-config writes now respect `IJFW_CUSTOM_DIR`.** The 1.1.2 prep pass only guarded sibling links and bin wiring. The `~/.claude/settings.json`, `~/.codex/`, `~/.gemini/`, `~/.codeium/windsurf/` merges were still running during scratch installs and clobbering real user configs with scratch paths. Every platform case block now short-circuits early when `IJFW_CUSTOM_DIR=1`, prints a single "real platform config left untouched" line, and still classifies the platform as live/standby for the summary banner.
|
|
17
|
+
- **Bug B: Codex `hooks.json` schema migrated to the current nested format.** Codex CLI 0.120+ rejects both the legacy `{"hooks":[...]}` object-wrapper and the bare-array shape this release started with. Authoritative schema (per `codex-rs/hooks/src/engine/config.rs`): `{"hooks": {EventName: [MatcherGroup]}}` where each MatcherGroup is `{matcher?, hooks: [{type: "command", command, timeout?, ...}]}`. Installer writer now emits this shape, absorbs either legacy shape on read, drops the non-existent `AfterAgent` event, renames `script` to `command`, and adds the `"type": "command"` discriminator.
|
|
18
|
+
- **Bug C: `suppress_unstable_features_warning = true` is now written to `~/.codex/config.toml`.** Stops the "under-development features enabled: codex_hooks" banner on every Codex startup.
|
|
19
|
+
- **Self-loop guards now canonicalize `$HOME`.** On macOS `/var/folders` is a symlink to `/private/var/folders`; the `cd -P` used for `REPO_ROOT` resolved that, but `$HOME` did not, so the `PLUGIN_SRC == PLUGIN_DST` and `MCP_SRC == MCP_DST` comparisons missed the equal case and created recursive self-symlinks ("too many levels of symbolic links" on next access). Installer now computes `HOME_REAL="$(cd -P "$HOME" && pwd)"` once and uses it for all self-loop comparisons.
|
|
20
|
+
- **`C_RED` variable declared.** Previously only initialized on interactive TTYs; a failing post-install gate in a non-TTY context (CI, harness, `npx` capture) would crash the installer with `C_RED: unbound variable` under `set -u`. Declared in both branches of the color-init block.
|
|
21
|
+
|
|
22
|
+
### Behavioral additions (synced across Claude / Codex / Gemini / Cursor / Windsurf / Copilot / Hermes / Wayland / universal where relevant)
|
|
23
|
+
|
|
24
|
+
- `ijfw-core` and platform rules-files: explicit banned openers ("Great question", "You're absolutely right", "Excellent idea", "I'd be happy to") and a sharpened two-strikes session-reset rule that asks the user to start a fresh session with a tighter prompt rather than burning context on a third failed attempt.
|
|
25
|
+
- `ijfw-debug`: new Step 6 templating the two-strikes reset with a memory-store call so lessons inherit forward without context noise.
|
|
26
|
+
- `ijfw-verify`: opens with "Plausibility is not correctness." Every claim must trace to a command output, test pass, or manual verification.
|
|
27
|
+
- `ijfw-workflow` Quick FRAME: five concrete goal-rewrite examples ("Add validation" -> "Write tests for invalid inputs..."). Vague asks must surface the gap rather than silently proceed.
|
|
28
|
+
- `ijfw-memory-audit`: pruning question added ("Would removing this rule cause the agent to make a mistake?") so memory stays sharp instead of bloated.
|
|
29
|
+
- `ijfw-critique`: refactor reframe ("Knowing everything I know now, what would the elegant solution look like?") for breaking frame on non-trivial decisions.
|
|
30
|
+
|
|
31
|
+
### End-to-end smoke harness
|
|
32
|
+
|
|
33
|
+
- New `scripts/e2e-smoke.sh`. Two modes, both must pass:
|
|
34
|
+
1. **Scratch-guard check** -- runs installer with `IJFW_CUSTOM_DIR=1` pointed at a throwaway dir, verifies zero drift across 10 real-home config paths (hashes before and after). Catches any future Bug A regression.
|
|
35
|
+
2. **Canonical isolated-HOME install** -- runs installer with `HOME=$(mktemp -d)`, parses every platform's written config against its expected schema (Codex nested hooks, Gemini JSON, YAML for Hermes/Wayland, etc.), completes the MCP `initialize + tools/list` handshake, and fails loudly on any mismatch.
|
|
36
|
+
- 13 gates total. Harness must be green before any future `npm publish`.
|
|
37
|
+
|
|
38
|
+
### Uninstaller
|
|
39
|
+
|
|
40
|
+
- `installer/src/uninstall.js:removeCodexHooks` now handles all three hook-file shapes we have ever shipped (bare array, legacy `{hooks:[...]}` object-wrapper, current nested-map). Uninstall works regardless of which version the user last installed.
|
|
41
|
+
- New `removeYamlMcpEntry` helper (python3+PyYAML preferred, regex fallback). Cleans `~/.hermes/config.yaml` and `~/.wayland/config.yaml`, removes skill dirs and context files for both new platforms.
|
|
42
|
+
- `cleanPlatforms()` comment updated: "all 8 platforms".
|
|
43
|
+
|
|
44
|
+
### Installer scope-leak fixes (carried from 1.1.2 prep)
|
|
45
|
+
|
|
46
|
+
- `scripts/install.sh` now respects `IJFW_CUSTOM_DIR` from `install.js`. Custom-dir installs (`--dir <scratch>`) skip user-home mutations: no sibling links into `~/.ijfw/`, no bin symlinks into `~/.local/bin/`, no `.mcp.json` patching of the real plugin, no `~/.claude/plugins/cache/ijfw` invalidation. Default canonical install behavior unchanged.
|
|
47
|
+
- Self-loop guard: when `PLUGIN_SRC == PLUGIN_DST` (install dir is the canonical home and source happens to live there), the symlink step is skipped instead of creating a recursive `~/.ijfw/claude -> ~/.ijfw/claude` loop.
|
|
48
|
+
- `installer/src/uninstall.js`: `uninstall --dir <scratch>` now leaves `~/.codex/`, `~/.gemini/`, `~/.codeium/windsurf/` configs and skill dirs alone. Only canonical uninstalls (`~/.ijfw`) clean platform configs.
|
|
49
|
+
|
|
50
|
+
### Dynamic version strings
|
|
51
|
+
|
|
52
|
+
- `mcp-server/src/server.js` and `mcp-server/src/dashboard-server.js` now read version from `mcp-server/package.json` at module load instead of hardcoding. MCP `serverInfo` and `/api/health` always match the shipped version.
|
|
53
|
+
|
|
54
|
+
### Internal
|
|
55
|
+
|
|
56
|
+
- `mcp-server/package.json` bumped from 1.1.0 to 1.1.2 (was lagging two minor cycles).
|
|
57
|
+
- `.gitattributes`: added LF normalization rules (carried from 1.1.1).
|
|
58
|
+
- Banner on successful install now says "8 platforms" instead of "7".
|
|
59
|
+
|
|
3
60
|
## [1.1.1] -- 2026-04-19
|
|
4
61
|
|
|
5
62
|
Docs and discoverability.
|
package/dist/install.js
CHANGED
|
@@ -216,7 +216,14 @@ function cloneOrPull(dir, branch) {
|
|
|
216
216
|
function runInstallScript(dir) {
|
|
217
217
|
const script = join2(dir, "scripts", "install.sh");
|
|
218
218
|
if (!existsSync2(script)) throw new Error(`IJFW install script not found at ${script} -- re-run the installer to restore it.`);
|
|
219
|
-
const
|
|
219
|
+
const canonicalDir = join2(homedir2(), ".ijfw");
|
|
220
|
+
const isCustomDir = resolve(dir) !== canonicalDir ? "1" : "0";
|
|
221
|
+
const env = {
|
|
222
|
+
...process.env,
|
|
223
|
+
IJFW_NONINTERACTIVE: process.env.CI ? "1" : process.env.IJFW_NONINTERACTIVE ?? "",
|
|
224
|
+
IJFW_HOME: dir,
|
|
225
|
+
IJFW_CUSTOM_DIR: isCustomDir
|
|
226
|
+
};
|
|
220
227
|
const r = spawnSync("bash", ["scripts/install.sh"], { cwd: dir, stdio: "inherit", env });
|
|
221
228
|
if (r.status !== 0) throw new Error(`IJFW platform config step did not complete (exit ${r.status}) -- run ijfw doctor to see what to fix.`);
|
|
222
229
|
}
|
|
@@ -253,7 +260,7 @@ async function main() {
|
|
|
253
260
|
console.log(` marketplace registered in ${settingsPath}`);
|
|
254
261
|
}
|
|
255
262
|
console.log("");
|
|
256
|
-
console.log("IJFW now active across
|
|
263
|
+
console.log("IJFW now active across 8 platforms -- one memory layer, all your AIs, zero config.");
|
|
257
264
|
console.log(" Run `ijfw demo` to see the Trident in action.");
|
|
258
265
|
console.log(" Run `ijfw doctor` to confirm which auditors are reachable.");
|
|
259
266
|
console.log(" Privacy: everything stays local. See NO_TELEMETRY.md.");
|
package/dist/uninstall.js
CHANGED
|
@@ -4,6 +4,7 @@
|
|
|
4
4
|
import { existsSync as existsSync2, rmSync, cpSync, mkdtempSync, readFileSync as readFileSync2, writeFileSync as writeFileSync2, readdirSync } from "node:fs";
|
|
5
5
|
import { resolve, join as join2 } from "node:path";
|
|
6
6
|
import { homedir as homedir2, tmpdir } from "node:os";
|
|
7
|
+
import { spawnSync } from "node:child_process";
|
|
7
8
|
|
|
8
9
|
// src/marketplace.js
|
|
9
10
|
import { readFileSync, writeFileSync, renameSync, existsSync, mkdirSync } from "node:fs";
|
|
@@ -156,12 +157,74 @@ function removeCodexHooks(p) {
|
|
|
156
157
|
} catch {
|
|
157
158
|
return false;
|
|
158
159
|
}
|
|
159
|
-
if (
|
|
160
|
-
|
|
161
|
-
|
|
162
|
-
|
|
160
|
+
if (Array.isArray(doc)) {
|
|
161
|
+
const before = doc.length;
|
|
162
|
+
const after = doc.filter((h) => !(h && h._ijfw));
|
|
163
|
+
if (after.length === before) return false;
|
|
164
|
+
backupFile(p);
|
|
165
|
+
writeFileSync2(p, JSON.stringify(after, null, 2) + "\n");
|
|
166
|
+
return true;
|
|
167
|
+
}
|
|
168
|
+
if (!doc || typeof doc !== "object" || !doc.hooks) return false;
|
|
169
|
+
if (doc.hooks && typeof doc.hooks === "object" && !Array.isArray(doc.hooks)) {
|
|
170
|
+
let changed = false;
|
|
171
|
+
for (const ev of Object.keys(doc.hooks)) {
|
|
172
|
+
const groups = doc.hooks[ev];
|
|
173
|
+
if (!Array.isArray(groups)) continue;
|
|
174
|
+
const before = groups.length;
|
|
175
|
+
doc.hooks[ev] = groups.filter((g) => {
|
|
176
|
+
if (!g || !Array.isArray(g.hooks)) return true;
|
|
177
|
+
return !g.hooks.some((h) => h && h._ijfw);
|
|
178
|
+
});
|
|
179
|
+
if (doc.hooks[ev].length !== before) changed = true;
|
|
180
|
+
}
|
|
181
|
+
if (!changed) return false;
|
|
182
|
+
backupFile(p);
|
|
183
|
+
writeFileSync2(p, JSON.stringify(doc, null, 2) + "\n");
|
|
184
|
+
return true;
|
|
185
|
+
}
|
|
186
|
+
if (Array.isArray(doc.hooks)) {
|
|
187
|
+
const before = doc.hooks.length;
|
|
188
|
+
doc.hooks = doc.hooks.filter((h) => !(h && h._ijfw));
|
|
189
|
+
if (doc.hooks.length === before) return false;
|
|
190
|
+
backupFile(p);
|
|
191
|
+
writeFileSync2(p, JSON.stringify(doc, null, 2) + "\n");
|
|
192
|
+
return true;
|
|
193
|
+
}
|
|
194
|
+
return false;
|
|
195
|
+
}
|
|
196
|
+
function removeYamlMcpEntry(p) {
|
|
197
|
+
if (!existsSync2(p)) return false;
|
|
198
|
+
const raw = readFileSync2(p, "utf8");
|
|
199
|
+
if (!/\bijfw-memory\b/.test(raw)) return false;
|
|
200
|
+
const py = spawnSync("python3", ["-c", `
|
|
201
|
+
import sys, yaml
|
|
202
|
+
p = sys.argv[1]
|
|
203
|
+
with open(p) as f: raw = f.read()
|
|
204
|
+
doc = yaml.safe_load(raw) if raw.strip() else {}
|
|
205
|
+
if not isinstance(doc, dict): sys.exit(2)
|
|
206
|
+
srv = doc.get("mcp_servers")
|
|
207
|
+
if not isinstance(srv, dict) or "ijfw-memory" not in srv: sys.exit(3)
|
|
208
|
+
del srv["ijfw-memory"]
|
|
209
|
+
if not srv: del doc["mcp_servers"]
|
|
210
|
+
with open(p + ".tmp", "w") as f:
|
|
211
|
+
yaml.safe_dump(doc, f, sort_keys=False, default_flow_style=False)
|
|
212
|
+
import os; os.replace(p + ".tmp", p)
|
|
213
|
+
`, p], { encoding: "utf8" });
|
|
214
|
+
if (py.status === 0) {
|
|
215
|
+
backupFile(p);
|
|
216
|
+
return true;
|
|
217
|
+
}
|
|
218
|
+
const stripped = raw.replace(
|
|
219
|
+
/^ ijfw-memory:\n(?: .*\n)*(?:# IJFW-MCP-END ijfw-memory\n)?/m,
|
|
220
|
+
""
|
|
221
|
+
).replace(
|
|
222
|
+
/# IJFW-MCP-BEGIN ijfw-memory\n(?:.*\n)*?# IJFW-MCP-END ijfw-memory\n/,
|
|
223
|
+
""
|
|
224
|
+
);
|
|
225
|
+
if (stripped === raw) return false;
|
|
163
226
|
backupFile(p);
|
|
164
|
-
writeFileSync2(p,
|
|
227
|
+
writeFileSync2(p, stripped);
|
|
165
228
|
return true;
|
|
166
229
|
}
|
|
167
230
|
function removeIjfwSkills(dir) {
|
|
@@ -205,6 +268,26 @@ function cleanPlatforms() {
|
|
|
205
268
|
}
|
|
206
269
|
const vscodeMcp = join2(".vscode", "mcp.json");
|
|
207
270
|
if (removeJsonMcpEntry(vscodeMcp)) removed.push(".vscode/mcp.json (removed ijfw-memory)");
|
|
271
|
+
if (removeYamlMcpEntry(join2(HOME, ".hermes", "config.yaml"))) {
|
|
272
|
+
removed.push("~/.hermes/config.yaml (removed ijfw-memory)");
|
|
273
|
+
}
|
|
274
|
+
const hermesSkills = removeIjfwSkills(join2(HOME, ".hermes", "skills"));
|
|
275
|
+
if (hermesSkills > 0) removed.push(`~/.hermes/skills/ijfw-* (removed ${hermesSkills} skill dirs)`);
|
|
276
|
+
const hermesMd = join2(HOME, ".hermes", "HERMES.md");
|
|
277
|
+
if (existsSync2(hermesMd)) {
|
|
278
|
+
rmSync(hermesMd, { force: true });
|
|
279
|
+
removed.push("~/.hermes/HERMES.md");
|
|
280
|
+
}
|
|
281
|
+
if (removeYamlMcpEntry(join2(HOME, ".wayland", "config.yaml"))) {
|
|
282
|
+
removed.push("~/.wayland/config.yaml (removed ijfw-memory)");
|
|
283
|
+
}
|
|
284
|
+
const waylandSkills = removeIjfwSkills(join2(HOME, ".wayland", "skills"));
|
|
285
|
+
if (waylandSkills > 0) removed.push(`~/.wayland/skills/ijfw-* (removed ${waylandSkills} skill dirs)`);
|
|
286
|
+
const waylandMd = join2(HOME, ".wayland", "WAYLAND.md");
|
|
287
|
+
if (existsSync2(waylandMd)) {
|
|
288
|
+
rmSync(waylandMd, { force: true });
|
|
289
|
+
removed.push("~/.wayland/WAYLAND.md");
|
|
290
|
+
}
|
|
208
291
|
return removed;
|
|
209
292
|
}
|
|
210
293
|
function resolveTarget(opt) {
|
|
@@ -238,17 +321,23 @@ async function main() {
|
|
|
238
321
|
console.log(" memory/ was not present; nothing to preserve");
|
|
239
322
|
}
|
|
240
323
|
}
|
|
241
|
-
|
|
324
|
+
const canonicalDir = join2(HOME, ".ijfw");
|
|
325
|
+
const isCanonical = target === canonicalDir;
|
|
326
|
+
if (isCanonical && !opts.noMarketplace) {
|
|
242
327
|
const settingsPath = claudeSettingsPath();
|
|
243
328
|
if (existsSync2(settingsPath)) {
|
|
244
329
|
unmergeMarketplace(settingsPath);
|
|
245
330
|
console.log(` marketplace removed from ${settingsPath}`);
|
|
246
331
|
}
|
|
247
332
|
}
|
|
248
|
-
|
|
249
|
-
|
|
250
|
-
|
|
251
|
-
|
|
333
|
+
if (isCanonical) {
|
|
334
|
+
const cleaned = cleanPlatforms();
|
|
335
|
+
if (cleaned.length > 0) {
|
|
336
|
+
console.log(" platform configs cleaned:");
|
|
337
|
+
for (const line of cleaned) console.log(` ${line}`);
|
|
338
|
+
}
|
|
339
|
+
} else {
|
|
340
|
+
console.log(` custom-dir uninstall (${target}) -- platform configs in your real home left untouched.`);
|
|
252
341
|
}
|
|
253
342
|
console.log("\nIJFW uninstalled. Thanks for trying it.");
|
|
254
343
|
process.exit(0);
|
package/docs/GUIDE.md
CHANGED
|
@@ -1,5 +1,5 @@
|
|
|
1
1
|
<p align="center">
|
|
2
|
-
<img src="https://github.com/TheRealSeanDonahoe/ijfw/releases/download/v1.1.
|
|
2
|
+
<img src="https://github.com/TheRealSeanDonahoe/ijfw/releases/download/v1.1.1/ijfw-hero.png" alt="IJFW" width="100%"/>
|
|
3
3
|
</p>
|
|
4
4
|
|
|
5
5
|
# The IJFW Guide
|