everything-dev 1.7.2 → 1.8.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/api.cjs +1 -1
- package/dist/api.mjs +1 -1
- package/dist/app.cjs +82 -51
- package/dist/app.cjs.map +1 -1
- package/dist/app.mjs +82 -51
- package/dist/app.mjs.map +1 -1
- package/dist/cli/upgrade.cjs.map +1 -1
- package/dist/cli/upgrade.mjs.map +1 -1
- package/dist/components/dev-view.cjs +6 -3
- package/dist/components/dev-view.cjs.map +1 -1
- package/dist/components/dev-view.mjs +6 -3
- package/dist/components/dev-view.mjs.map +1 -1
- package/dist/components/streaming-view.cjs +5 -2
- package/dist/components/streaming-view.cjs.map +1 -1
- package/dist/components/streaming-view.mjs +5 -2
- package/dist/components/streaming-view.mjs.map +1 -1
- package/dist/config.cjs +28 -5
- package/dist/config.cjs.map +1 -1
- package/dist/config.d.cts.map +1 -1
- package/dist/config.d.mts.map +1 -1
- package/dist/config.mjs +28 -5
- package/dist/config.mjs.map +1 -1
- package/dist/contract.cjs +1 -0
- package/dist/contract.cjs.map +1 -1
- package/dist/contract.d.cts +14 -6
- package/dist/contract.d.cts.map +1 -1
- package/dist/contract.d.mts +14 -6
- package/dist/contract.d.mts.map +1 -1
- package/dist/contract.mjs +1 -0
- package/dist/contract.mjs.map +1 -1
- package/dist/dev-logs.cjs +6 -2
- package/dist/dev-logs.cjs.map +1 -1
- package/dist/dev-logs.mjs +7 -2
- package/dist/dev-logs.mjs.map +1 -1
- package/dist/dev-session.cjs +27 -23
- package/dist/dev-session.cjs.map +1 -1
- package/dist/dev-session.mjs +27 -24
- package/dist/dev-session.mjs.map +1 -1
- package/dist/federation.server.cjs +1 -1
- package/dist/federation.server.mjs +1 -1
- package/dist/host.cjs +4 -3
- package/dist/host.cjs.map +1 -1
- package/dist/host.d.cts.map +1 -1
- package/dist/host.d.mts.map +1 -1
- package/dist/host.mjs +4 -3
- package/dist/host.mjs.map +1 -1
- package/dist/integrity.cjs +68 -2
- package/dist/integrity.cjs.map +1 -1
- package/dist/integrity.d.cts +14 -1
- package/dist/integrity.d.cts.map +1 -1
- package/dist/integrity.d.mts +14 -1
- package/dist/integrity.d.mts.map +1 -1
- package/dist/integrity.mjs +66 -3
- package/dist/integrity.mjs.map +1 -1
- package/dist/mf.cjs +32 -0
- package/dist/mf.cjs.map +1 -1
- package/dist/mf.d.cts +3 -1
- package/dist/mf.d.cts.map +1 -1
- package/dist/mf.d.mts +3 -1
- package/dist/mf.d.mts.map +1 -1
- package/dist/mf.mjs +32 -1
- package/dist/mf.mjs.map +1 -1
- package/dist/orchestrator.cjs +167 -317
- package/dist/orchestrator.cjs.map +1 -1
- package/dist/orchestrator.d.cts +24 -21
- package/dist/orchestrator.d.cts.map +1 -1
- package/dist/orchestrator.d.mts +24 -21
- package/dist/orchestrator.d.mts.map +1 -1
- package/dist/orchestrator.mjs +168 -316
- package/dist/orchestrator.mjs.map +1 -1
- package/dist/plugin.cjs +38 -107
- package/dist/plugin.cjs.map +1 -1
- package/dist/plugin.d.cts +19 -5
- package/dist/plugin.d.cts.map +1 -1
- package/dist/plugin.d.mts +19 -5
- package/dist/plugin.d.mts.map +1 -1
- package/dist/plugin.mjs +39 -108
- package/dist/plugin.mjs.map +1 -1
- package/dist/service-descriptor.cjs +188 -0
- package/dist/service-descriptor.cjs.map +1 -0
- package/dist/service-descriptor.d.cts +107 -0
- package/dist/service-descriptor.d.cts.map +1 -0
- package/dist/service-descriptor.d.mts +107 -0
- package/dist/service-descriptor.d.mts.map +1 -0
- package/dist/service-descriptor.mjs +182 -0
- package/dist/service-descriptor.mjs.map +1 -0
- package/dist/types.cjs +8 -1
- package/dist/types.cjs.map +1 -1
- package/dist/types.d.cts +18 -3
- package/dist/types.d.cts.map +1 -1
- package/dist/types.d.mts +18 -3
- package/dist/types.d.mts.map +1 -1
- package/dist/types.mjs +8 -1
- package/dist/types.mjs.map +1 -1
- package/dist/ui/index.cjs +1 -0
- package/dist/ui/index.d.cts +2 -2
- package/dist/ui/index.d.mts +2 -2
- package/dist/ui/index.mjs +2 -2
- package/dist/ui/runtime.cjs +4 -0
- package/dist/ui/runtime.cjs.map +1 -1
- package/dist/ui/runtime.d.cts +2 -1
- package/dist/ui/runtime.d.cts.map +1 -1
- package/dist/ui/runtime.d.mts +2 -1
- package/dist/ui/runtime.d.mts.map +1 -1
- package/dist/ui/runtime.mjs +4 -1
- package/dist/ui/runtime.mjs.map +1 -1
- package/package.json +12 -4
- package/skills/dev-workflow/SKILL.md +105 -0
- package/skills/publish-sync/SKILL.md +130 -0
- package/src/app.ts +98 -204
- package/src/cli/upgrade.ts +20 -4
- package/src/components/dev-view.tsx +8 -3
- package/src/components/streaming-view.ts +7 -2
- package/src/config.ts +40 -8
- package/src/contract.ts +1 -0
- package/src/dev-logs.ts +8 -1
- package/src/dev-session.ts +56 -79
- package/src/host.ts +4 -3
- package/src/integrity.ts +96 -10
- package/src/mf.ts +42 -0
- package/src/orchestrator.ts +232 -411
- package/src/plugin.ts +48 -136
- package/src/service-descriptor.ts +258 -0
- package/src/types.ts +8 -1
- package/src/ui/runtime.ts +5 -0
- package/dist/process-registry.cjs +0 -120
- package/dist/process-registry.cjs.map +0 -1
- package/dist/process-registry.d.cts +0 -25
- package/dist/process-registry.d.cts.map +0 -1
- package/dist/process-registry.d.mts +0 -25
- package/dist/process-registry.d.mts.map +0 -1
- package/dist/process-registry.mjs +0 -119
- package/dist/process-registry.mjs.map +0 -1
- package/src/process-registry.ts +0 -154
|
@@ -1,120 +0,0 @@
|
|
|
1
|
-
const require_runtime = require('./_virtual/_rolldown/runtime.cjs');
|
|
2
|
-
const require_dev_logs = require('./dev-logs.cjs');
|
|
3
|
-
let node_fs = require("node:fs");
|
|
4
|
-
let node_path = require("node:path");
|
|
5
|
-
let effect = require("effect");
|
|
6
|
-
let node_fs_promises = require("node:fs/promises");
|
|
7
|
-
|
|
8
|
-
//#region src/process-registry.ts
|
|
9
|
-
function getPidFilePath(configDir) {
|
|
10
|
-
return (0, node_path.join)(require_dev_logs.getBosDir(configDir), "pids.json");
|
|
11
|
-
}
|
|
12
|
-
const isProcessAlive = (pid) => {
|
|
13
|
-
try {
|
|
14
|
-
process.kill(pid, 0);
|
|
15
|
-
return true;
|
|
16
|
-
} catch {
|
|
17
|
-
return false;
|
|
18
|
-
}
|
|
19
|
-
};
|
|
20
|
-
const killProcess = (pid, signal) => {
|
|
21
|
-
try {
|
|
22
|
-
try {
|
|
23
|
-
process.kill(-pid, signal);
|
|
24
|
-
} catch {}
|
|
25
|
-
process.kill(pid, signal);
|
|
26
|
-
return true;
|
|
27
|
-
} catch {
|
|
28
|
-
return false;
|
|
29
|
-
}
|
|
30
|
-
};
|
|
31
|
-
const makeProcessRegistry = (configDir) => effect.Effect.gen(function* () {
|
|
32
|
-
const tracked = yield* effect.Ref.make(/* @__PURE__ */ new Map());
|
|
33
|
-
const pidFile = getPidFilePath(configDir);
|
|
34
|
-
const getAll = () => effect.Ref.get(tracked).pipe(effect.Effect.map((m) => Array.from(m.values())));
|
|
35
|
-
const persist = () => effect.Effect.gen(function* () {
|
|
36
|
-
const procs = yield* getAll();
|
|
37
|
-
const dir = (0, node_path.dirname)(pidFile);
|
|
38
|
-
yield* effect.Effect.tryPromise({
|
|
39
|
-
try: async () => {
|
|
40
|
-
if (!(0, node_fs.existsSync)(dir)) await (0, node_fs_promises.mkdir)(dir, { recursive: true });
|
|
41
|
-
await (0, node_fs_promises.writeFile)(pidFile, JSON.stringify(procs, null, 2));
|
|
42
|
-
},
|
|
43
|
-
catch: () => /* @__PURE__ */ new Error("Failed to persist PIDs")
|
|
44
|
-
}).pipe(effect.Effect.catchAll(() => effect.Effect.void));
|
|
45
|
-
});
|
|
46
|
-
const track = (proc) => effect.Effect.gen(function* () {
|
|
47
|
-
yield* effect.Ref.update(tracked, (m) => new Map(m).set(proc.pid, proc));
|
|
48
|
-
yield* persist();
|
|
49
|
-
});
|
|
50
|
-
const untrack = (pid) => effect.Effect.gen(function* () {
|
|
51
|
-
yield* effect.Ref.update(tracked, (m) => {
|
|
52
|
-
const copy = new Map(m);
|
|
53
|
-
copy.delete(pid);
|
|
54
|
-
return copy;
|
|
55
|
-
});
|
|
56
|
-
yield* persist();
|
|
57
|
-
});
|
|
58
|
-
const restore = () => effect.Effect.gen(function* () {
|
|
59
|
-
if (!(0, node_fs.existsSync)(pidFile)) return;
|
|
60
|
-
const content = yield* effect.Effect.tryPromise({
|
|
61
|
-
try: () => (0, node_fs_promises.readFile)(pidFile, "utf8"),
|
|
62
|
-
catch: () => /* @__PURE__ */ new Error("Failed to read PID file")
|
|
63
|
-
}).pipe(effect.Effect.catchAll(() => effect.Effect.succeed("")));
|
|
64
|
-
if (!content) return;
|
|
65
|
-
let procs;
|
|
66
|
-
try {
|
|
67
|
-
procs = JSON.parse(content);
|
|
68
|
-
} catch {
|
|
69
|
-
return;
|
|
70
|
-
}
|
|
71
|
-
const alive = procs.filter((p) => isProcessAlive(p.pid));
|
|
72
|
-
yield* effect.Ref.set(tracked, new Map(alive.map((p) => [p.pid, p])));
|
|
73
|
-
});
|
|
74
|
-
yield* restore();
|
|
75
|
-
const killAll = (force = false) => effect.Effect.gen(function* () {
|
|
76
|
-
const procs = yield* getAll();
|
|
77
|
-
const killed = [];
|
|
78
|
-
const failed = [];
|
|
79
|
-
for (const proc of procs) {
|
|
80
|
-
if (!isProcessAlive(proc.pid)) {
|
|
81
|
-
yield* untrack(proc.pid);
|
|
82
|
-
continue;
|
|
83
|
-
}
|
|
84
|
-
const signal = force ? "SIGKILL" : "SIGTERM";
|
|
85
|
-
if (killProcess(proc.pid, signal)) {
|
|
86
|
-
killed.push(proc.pid);
|
|
87
|
-
yield* untrack(proc.pid);
|
|
88
|
-
} else failed.push(proc.pid);
|
|
89
|
-
}
|
|
90
|
-
if (!force && failed.length > 0) {
|
|
91
|
-
yield* effect.Effect.sleep("500 millis");
|
|
92
|
-
for (const pid of [...failed]) if (killProcess(pid, "SIGKILL")) {
|
|
93
|
-
const idx = failed.indexOf(pid);
|
|
94
|
-
if (idx !== -1) {
|
|
95
|
-
failed.splice(idx, 1);
|
|
96
|
-
killed.push(pid);
|
|
97
|
-
}
|
|
98
|
-
yield* untrack(pid);
|
|
99
|
-
}
|
|
100
|
-
}
|
|
101
|
-
yield* persist();
|
|
102
|
-
return {
|
|
103
|
-
killed,
|
|
104
|
-
failed
|
|
105
|
-
};
|
|
106
|
-
});
|
|
107
|
-
return {
|
|
108
|
-
tracked,
|
|
109
|
-
track,
|
|
110
|
-
untrack,
|
|
111
|
-
getAll,
|
|
112
|
-
killAll,
|
|
113
|
-
persist,
|
|
114
|
-
restore
|
|
115
|
-
};
|
|
116
|
-
});
|
|
117
|
-
|
|
118
|
-
//#endregion
|
|
119
|
-
exports.makeProcessRegistry = makeProcessRegistry;
|
|
120
|
-
//# sourceMappingURL=process-registry.cjs.map
|
|
@@ -1 +0,0 @@
|
|
|
1
|
-
{"version":3,"file":"process-registry.cjs","names":["getBosDir","Effect","Ref"],"sources":["../src/process-registry.ts"],"sourcesContent":["import { existsSync } from \"node:fs\";\nimport { mkdir, readFile, writeFile } from \"node:fs/promises\";\nimport { dirname, join } from \"node:path\";\nimport { Effect, Ref } from \"effect\";\nimport { getBosDir } from \"./dev-logs\";\n\nexport interface TrackedProcess {\n pid: number;\n name: string;\n port: number;\n startedAt: number;\n command: string;\n}\n\nexport interface ProcessRegistry {\n readonly tracked: Ref.Ref<Map<number, TrackedProcess>>;\n track: (proc: TrackedProcess) => Effect.Effect<void>;\n untrack: (pid: number) => Effect.Effect<void>;\n getAll: () => Effect.Effect<TrackedProcess[]>;\n killAll: (force?: boolean) => Effect.Effect<{ killed: number[]; failed: number[] }>;\n persist: () => Effect.Effect<void>;\n restore: () => Effect.Effect<void>;\n}\n\nexport function getPidFilePath(configDir: string): string {\n return join(getBosDir(configDir), \"pids.json\");\n}\n\nconst isProcessAlive = (pid: number): boolean => {\n try {\n process.kill(pid, 0);\n return true;\n } catch {\n return false;\n }\n};\n\nconst killProcess = (pid: number, signal: NodeJS.Signals): boolean => {\n try {\n // Try process group first (covers spawned child trees)\n try {\n process.kill(-pid, signal);\n } catch {}\n process.kill(pid, signal);\n return true;\n } catch {\n return false;\n }\n};\n\nexport const makeProcessRegistry = (configDir: string) =>\n Effect.gen(function* () {\n const tracked = yield* Ref.make(new Map<number, TrackedProcess>());\n const pidFile = getPidFilePath(configDir);\n\n const getAll: ProcessRegistry[\"getAll\"] = () =>\n Ref.get(tracked).pipe(Effect.map((m) => Array.from(m.values())));\n\n const persist: ProcessRegistry[\"persist\"] = () =>\n Effect.gen(function* () {\n const procs = yield* getAll();\n const dir = dirname(pidFile);\n yield* Effect.tryPromise({\n try: async () => {\n if (!existsSync(dir)) {\n await mkdir(dir, { recursive: true });\n }\n await writeFile(pidFile, JSON.stringify(procs, null, 2));\n },\n catch: () => new Error(\"Failed to persist PIDs\"),\n }).pipe(Effect.catchAll(() => Effect.void));\n });\n\n const track: ProcessRegistry[\"track\"] = (proc) =>\n Effect.gen(function* () {\n yield* Ref.update(tracked, (m) => new Map(m).set(proc.pid, proc));\n yield* persist();\n });\n\n const untrack: ProcessRegistry[\"untrack\"] = (pid) =>\n Effect.gen(function* () {\n yield* Ref.update(tracked, (m) => {\n const copy = new Map(m);\n copy.delete(pid);\n return copy;\n });\n yield* persist();\n });\n\n const restore: ProcessRegistry[\"restore\"] = () =>\n Effect.gen(function* () {\n if (!existsSync(pidFile)) return;\n\n const content = yield* Effect.tryPromise({\n try: () => readFile(pidFile, \"utf8\"),\n catch: () => new Error(\"Failed to read PID file\"),\n }).pipe(Effect.catchAll(() => Effect.succeed(\"\")));\n\n if (!content) return;\n\n let procs: TrackedProcess[];\n try {\n procs = JSON.parse(content) as TrackedProcess[];\n } catch {\n return;\n }\n\n const alive = procs.filter((p) => isProcessAlive(p.pid));\n yield* Ref.set(tracked, new Map(alive.map((p) => [p.pid, p])));\n });\n\n yield* restore();\n\n const killAll: ProcessRegistry[\"killAll\"] = (force = false) =>\n Effect.gen(function* () {\n const procs = yield* getAll();\n const killed: number[] = [];\n const failed: number[] = [];\n\n for (const proc of procs) {\n if (!isProcessAlive(proc.pid)) {\n yield* untrack(proc.pid);\n continue;\n }\n\n const signal = force ? \"SIGKILL\" : \"SIGTERM\";\n if (killProcess(proc.pid, signal)) {\n killed.push(proc.pid);\n yield* untrack(proc.pid);\n } else {\n failed.push(proc.pid);\n }\n }\n\n if (!force && failed.length > 0) {\n yield* Effect.sleep(\"500 millis\");\n for (const pid of [...failed]) {\n if (killProcess(pid, \"SIGKILL\")) {\n const idx = failed.indexOf(pid);\n if (idx !== -1) {\n failed.splice(idx, 1);\n killed.push(pid);\n }\n yield* untrack(pid);\n }\n }\n }\n\n yield* persist();\n return { killed, failed };\n });\n\n return { tracked, track, untrack, getAll, killAll, persist, restore };\n });\n"],"mappings":"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"}
|
|
@@ -1,25 +0,0 @@
|
|
|
1
|
-
import { Effect, Ref } from "effect";
|
|
2
|
-
|
|
3
|
-
//#region src/process-registry.d.ts
|
|
4
|
-
interface TrackedProcess {
|
|
5
|
-
pid: number;
|
|
6
|
-
name: string;
|
|
7
|
-
port: number;
|
|
8
|
-
startedAt: number;
|
|
9
|
-
command: string;
|
|
10
|
-
}
|
|
11
|
-
interface ProcessRegistry {
|
|
12
|
-
readonly tracked: Ref.Ref<Map<number, TrackedProcess>>;
|
|
13
|
-
track: (proc: TrackedProcess) => Effect.Effect<void>;
|
|
14
|
-
untrack: (pid: number) => Effect.Effect<void>;
|
|
15
|
-
getAll: () => Effect.Effect<TrackedProcess[]>;
|
|
16
|
-
killAll: (force?: boolean) => Effect.Effect<{
|
|
17
|
-
killed: number[];
|
|
18
|
-
failed: number[];
|
|
19
|
-
}>;
|
|
20
|
-
persist: () => Effect.Effect<void>;
|
|
21
|
-
restore: () => Effect.Effect<void>;
|
|
22
|
-
}
|
|
23
|
-
//#endregion
|
|
24
|
-
export { ProcessRegistry };
|
|
25
|
-
//# sourceMappingURL=process-registry.d.cts.map
|
|
@@ -1 +0,0 @@
|
|
|
1
|
-
{"version":3,"file":"process-registry.d.cts","names":[],"sources":["../src/process-registry.ts"],"mappings":";;;UAMiB,cAAA;EACf,GAAA;EACA,IAAA;EACA,IAAA;EACA,SAAA;EACA,OAAA;AAAA;AAAA,UAGe,eAAA;EAAA,SACN,OAAA,EAAS,GAAA,CAAI,GAAA,CAAI,GAAA,SAAY,cAAA;EACtC,KAAA,GAAQ,IAAA,EAAM,cAAA,KAAmB,MAAA,CAAO,MAAA;EACxC,OAAA,GAAU,GAAA,aAAgB,MAAA,CAAO,MAAA;EACjC,MAAA,QAAc,MAAA,CAAO,MAAA,CAAO,cAAA;EAC5B,OAAA,GAAU,KAAA,eAAoB,MAAA,CAAO,MAAA;IAAS,MAAA;IAAkB,MAAA;EAAA;EAChE,OAAA,QAAe,MAAA,CAAO,MAAA;EACtB,OAAA,QAAe,MAAA,CAAO,MAAA;AAAA"}
|
|
@@ -1,25 +0,0 @@
|
|
|
1
|
-
import { Effect, Ref } from "effect";
|
|
2
|
-
|
|
3
|
-
//#region src/process-registry.d.ts
|
|
4
|
-
interface TrackedProcess {
|
|
5
|
-
pid: number;
|
|
6
|
-
name: string;
|
|
7
|
-
port: number;
|
|
8
|
-
startedAt: number;
|
|
9
|
-
command: string;
|
|
10
|
-
}
|
|
11
|
-
interface ProcessRegistry {
|
|
12
|
-
readonly tracked: Ref.Ref<Map<number, TrackedProcess>>;
|
|
13
|
-
track: (proc: TrackedProcess) => Effect.Effect<void>;
|
|
14
|
-
untrack: (pid: number) => Effect.Effect<void>;
|
|
15
|
-
getAll: () => Effect.Effect<TrackedProcess[]>;
|
|
16
|
-
killAll: (force?: boolean) => Effect.Effect<{
|
|
17
|
-
killed: number[];
|
|
18
|
-
failed: number[];
|
|
19
|
-
}>;
|
|
20
|
-
persist: () => Effect.Effect<void>;
|
|
21
|
-
restore: () => Effect.Effect<void>;
|
|
22
|
-
}
|
|
23
|
-
//#endregion
|
|
24
|
-
export { ProcessRegistry };
|
|
25
|
-
//# sourceMappingURL=process-registry.d.mts.map
|
|
@@ -1 +0,0 @@
|
|
|
1
|
-
{"version":3,"file":"process-registry.d.mts","names":[],"sources":["../src/process-registry.ts"],"mappings":";;;UAMiB,cAAA;EACf,GAAA;EACA,IAAA;EACA,IAAA;EACA,SAAA;EACA,OAAA;AAAA;AAAA,UAGe,eAAA;EAAA,SACN,OAAA,EAAS,GAAA,CAAI,GAAA,CAAI,GAAA,SAAY,cAAA;EACtC,KAAA,GAAQ,IAAA,EAAM,cAAA,KAAmB,MAAA,CAAO,MAAA;EACxC,OAAA,GAAU,GAAA,aAAgB,MAAA,CAAO,MAAA;EACjC,MAAA,QAAc,MAAA,CAAO,MAAA,CAAO,cAAA;EAC5B,OAAA,GAAU,KAAA,eAAoB,MAAA,CAAO,MAAA;IAAS,MAAA;IAAkB,MAAA;EAAA;EAChE,OAAA,QAAe,MAAA,CAAO,MAAA;EACtB,OAAA,QAAe,MAAA,CAAO,MAAA;AAAA"}
|
|
@@ -1,119 +0,0 @@
|
|
|
1
|
-
import { getBosDir } from "./dev-logs.mjs";
|
|
2
|
-
import { existsSync } from "node:fs";
|
|
3
|
-
import { dirname, join } from "node:path";
|
|
4
|
-
import { Effect, Ref } from "effect";
|
|
5
|
-
import { mkdir, readFile, writeFile } from "node:fs/promises";
|
|
6
|
-
|
|
7
|
-
//#region src/process-registry.ts
|
|
8
|
-
function getPidFilePath(configDir) {
|
|
9
|
-
return join(getBosDir(configDir), "pids.json");
|
|
10
|
-
}
|
|
11
|
-
const isProcessAlive = (pid) => {
|
|
12
|
-
try {
|
|
13
|
-
process.kill(pid, 0);
|
|
14
|
-
return true;
|
|
15
|
-
} catch {
|
|
16
|
-
return false;
|
|
17
|
-
}
|
|
18
|
-
};
|
|
19
|
-
const killProcess = (pid, signal) => {
|
|
20
|
-
try {
|
|
21
|
-
try {
|
|
22
|
-
process.kill(-pid, signal);
|
|
23
|
-
} catch {}
|
|
24
|
-
process.kill(pid, signal);
|
|
25
|
-
return true;
|
|
26
|
-
} catch {
|
|
27
|
-
return false;
|
|
28
|
-
}
|
|
29
|
-
};
|
|
30
|
-
const makeProcessRegistry = (configDir) => Effect.gen(function* () {
|
|
31
|
-
const tracked = yield* Ref.make(/* @__PURE__ */ new Map());
|
|
32
|
-
const pidFile = getPidFilePath(configDir);
|
|
33
|
-
const getAll = () => Ref.get(tracked).pipe(Effect.map((m) => Array.from(m.values())));
|
|
34
|
-
const persist = () => Effect.gen(function* () {
|
|
35
|
-
const procs = yield* getAll();
|
|
36
|
-
const dir = dirname(pidFile);
|
|
37
|
-
yield* Effect.tryPromise({
|
|
38
|
-
try: async () => {
|
|
39
|
-
if (!existsSync(dir)) await mkdir(dir, { recursive: true });
|
|
40
|
-
await writeFile(pidFile, JSON.stringify(procs, null, 2));
|
|
41
|
-
},
|
|
42
|
-
catch: () => /* @__PURE__ */ new Error("Failed to persist PIDs")
|
|
43
|
-
}).pipe(Effect.catchAll(() => Effect.void));
|
|
44
|
-
});
|
|
45
|
-
const track = (proc) => Effect.gen(function* () {
|
|
46
|
-
yield* Ref.update(tracked, (m) => new Map(m).set(proc.pid, proc));
|
|
47
|
-
yield* persist();
|
|
48
|
-
});
|
|
49
|
-
const untrack = (pid) => Effect.gen(function* () {
|
|
50
|
-
yield* Ref.update(tracked, (m) => {
|
|
51
|
-
const copy = new Map(m);
|
|
52
|
-
copy.delete(pid);
|
|
53
|
-
return copy;
|
|
54
|
-
});
|
|
55
|
-
yield* persist();
|
|
56
|
-
});
|
|
57
|
-
const restore = () => Effect.gen(function* () {
|
|
58
|
-
if (!existsSync(pidFile)) return;
|
|
59
|
-
const content = yield* Effect.tryPromise({
|
|
60
|
-
try: () => readFile(pidFile, "utf8"),
|
|
61
|
-
catch: () => /* @__PURE__ */ new Error("Failed to read PID file")
|
|
62
|
-
}).pipe(Effect.catchAll(() => Effect.succeed("")));
|
|
63
|
-
if (!content) return;
|
|
64
|
-
let procs;
|
|
65
|
-
try {
|
|
66
|
-
procs = JSON.parse(content);
|
|
67
|
-
} catch {
|
|
68
|
-
return;
|
|
69
|
-
}
|
|
70
|
-
const alive = procs.filter((p) => isProcessAlive(p.pid));
|
|
71
|
-
yield* Ref.set(tracked, new Map(alive.map((p) => [p.pid, p])));
|
|
72
|
-
});
|
|
73
|
-
yield* restore();
|
|
74
|
-
const killAll = (force = false) => Effect.gen(function* () {
|
|
75
|
-
const procs = yield* getAll();
|
|
76
|
-
const killed = [];
|
|
77
|
-
const failed = [];
|
|
78
|
-
for (const proc of procs) {
|
|
79
|
-
if (!isProcessAlive(proc.pid)) {
|
|
80
|
-
yield* untrack(proc.pid);
|
|
81
|
-
continue;
|
|
82
|
-
}
|
|
83
|
-
const signal = force ? "SIGKILL" : "SIGTERM";
|
|
84
|
-
if (killProcess(proc.pid, signal)) {
|
|
85
|
-
killed.push(proc.pid);
|
|
86
|
-
yield* untrack(proc.pid);
|
|
87
|
-
} else failed.push(proc.pid);
|
|
88
|
-
}
|
|
89
|
-
if (!force && failed.length > 0) {
|
|
90
|
-
yield* Effect.sleep("500 millis");
|
|
91
|
-
for (const pid of [...failed]) if (killProcess(pid, "SIGKILL")) {
|
|
92
|
-
const idx = failed.indexOf(pid);
|
|
93
|
-
if (idx !== -1) {
|
|
94
|
-
failed.splice(idx, 1);
|
|
95
|
-
killed.push(pid);
|
|
96
|
-
}
|
|
97
|
-
yield* untrack(pid);
|
|
98
|
-
}
|
|
99
|
-
}
|
|
100
|
-
yield* persist();
|
|
101
|
-
return {
|
|
102
|
-
killed,
|
|
103
|
-
failed
|
|
104
|
-
};
|
|
105
|
-
});
|
|
106
|
-
return {
|
|
107
|
-
tracked,
|
|
108
|
-
track,
|
|
109
|
-
untrack,
|
|
110
|
-
getAll,
|
|
111
|
-
killAll,
|
|
112
|
-
persist,
|
|
113
|
-
restore
|
|
114
|
-
};
|
|
115
|
-
});
|
|
116
|
-
|
|
117
|
-
//#endregion
|
|
118
|
-
export { makeProcessRegistry };
|
|
119
|
-
//# sourceMappingURL=process-registry.mjs.map
|
|
@@ -1 +0,0 @@
|
|
|
1
|
-
{"version":3,"file":"process-registry.mjs","names":[],"sources":["../src/process-registry.ts"],"sourcesContent":["import { existsSync } from \"node:fs\";\nimport { mkdir, readFile, writeFile } from \"node:fs/promises\";\nimport { dirname, join } from \"node:path\";\nimport { Effect, Ref } from \"effect\";\nimport { getBosDir } from \"./dev-logs\";\n\nexport interface TrackedProcess {\n pid: number;\n name: string;\n port: number;\n startedAt: number;\n command: string;\n}\n\nexport interface ProcessRegistry {\n readonly tracked: Ref.Ref<Map<number, TrackedProcess>>;\n track: (proc: TrackedProcess) => Effect.Effect<void>;\n untrack: (pid: number) => Effect.Effect<void>;\n getAll: () => Effect.Effect<TrackedProcess[]>;\n killAll: (force?: boolean) => Effect.Effect<{ killed: number[]; failed: number[] }>;\n persist: () => Effect.Effect<void>;\n restore: () => Effect.Effect<void>;\n}\n\nexport function getPidFilePath(configDir: string): string {\n return join(getBosDir(configDir), \"pids.json\");\n}\n\nconst isProcessAlive = (pid: number): boolean => {\n try {\n process.kill(pid, 0);\n return true;\n } catch {\n return false;\n }\n};\n\nconst killProcess = (pid: number, signal: NodeJS.Signals): boolean => {\n try {\n // Try process group first (covers spawned child trees)\n try {\n process.kill(-pid, signal);\n } catch {}\n process.kill(pid, signal);\n return true;\n } catch {\n return false;\n }\n};\n\nexport const makeProcessRegistry = (configDir: string) =>\n Effect.gen(function* () {\n const tracked = yield* Ref.make(new Map<number, TrackedProcess>());\n const pidFile = getPidFilePath(configDir);\n\n const getAll: ProcessRegistry[\"getAll\"] = () =>\n Ref.get(tracked).pipe(Effect.map((m) => Array.from(m.values())));\n\n const persist: ProcessRegistry[\"persist\"] = () =>\n Effect.gen(function* () {\n const procs = yield* getAll();\n const dir = dirname(pidFile);\n yield* Effect.tryPromise({\n try: async () => {\n if (!existsSync(dir)) {\n await mkdir(dir, { recursive: true });\n }\n await writeFile(pidFile, JSON.stringify(procs, null, 2));\n },\n catch: () => new Error(\"Failed to persist PIDs\"),\n }).pipe(Effect.catchAll(() => Effect.void));\n });\n\n const track: ProcessRegistry[\"track\"] = (proc) =>\n Effect.gen(function* () {\n yield* Ref.update(tracked, (m) => new Map(m).set(proc.pid, proc));\n yield* persist();\n });\n\n const untrack: ProcessRegistry[\"untrack\"] = (pid) =>\n Effect.gen(function* () {\n yield* Ref.update(tracked, (m) => {\n const copy = new Map(m);\n copy.delete(pid);\n return copy;\n });\n yield* persist();\n });\n\n const restore: ProcessRegistry[\"restore\"] = () =>\n Effect.gen(function* () {\n if (!existsSync(pidFile)) return;\n\n const content = yield* Effect.tryPromise({\n try: () => readFile(pidFile, \"utf8\"),\n catch: () => new Error(\"Failed to read PID file\"),\n }).pipe(Effect.catchAll(() => Effect.succeed(\"\")));\n\n if (!content) return;\n\n let procs: TrackedProcess[];\n try {\n procs = JSON.parse(content) as TrackedProcess[];\n } catch {\n return;\n }\n\n const alive = procs.filter((p) => isProcessAlive(p.pid));\n yield* Ref.set(tracked, new Map(alive.map((p) => [p.pid, p])));\n });\n\n yield* restore();\n\n const killAll: ProcessRegistry[\"killAll\"] = (force = false) =>\n Effect.gen(function* () {\n const procs = yield* getAll();\n const killed: number[] = [];\n const failed: number[] = [];\n\n for (const proc of procs) {\n if (!isProcessAlive(proc.pid)) {\n yield* untrack(proc.pid);\n continue;\n }\n\n const signal = force ? \"SIGKILL\" : \"SIGTERM\";\n if (killProcess(proc.pid, signal)) {\n killed.push(proc.pid);\n yield* untrack(proc.pid);\n } else {\n failed.push(proc.pid);\n }\n }\n\n if (!force && failed.length > 0) {\n yield* Effect.sleep(\"500 millis\");\n for (const pid of [...failed]) {\n if (killProcess(pid, \"SIGKILL\")) {\n const idx = failed.indexOf(pid);\n if (idx !== -1) {\n failed.splice(idx, 1);\n killed.push(pid);\n }\n yield* untrack(pid);\n }\n }\n }\n\n yield* persist();\n return { killed, failed };\n });\n\n return { tracked, track, untrack, getAll, killAll, persist, restore };\n });\n"],"mappings":"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"}
|
package/src/process-registry.ts
DELETED
|
@@ -1,154 +0,0 @@
|
|
|
1
|
-
import { existsSync } from "node:fs";
|
|
2
|
-
import { mkdir, readFile, writeFile } from "node:fs/promises";
|
|
3
|
-
import { dirname, join } from "node:path";
|
|
4
|
-
import { Effect, Ref } from "effect";
|
|
5
|
-
import { getBosDir } from "./dev-logs";
|
|
6
|
-
|
|
7
|
-
export interface TrackedProcess {
|
|
8
|
-
pid: number;
|
|
9
|
-
name: string;
|
|
10
|
-
port: number;
|
|
11
|
-
startedAt: number;
|
|
12
|
-
command: string;
|
|
13
|
-
}
|
|
14
|
-
|
|
15
|
-
export interface ProcessRegistry {
|
|
16
|
-
readonly tracked: Ref.Ref<Map<number, TrackedProcess>>;
|
|
17
|
-
track: (proc: TrackedProcess) => Effect.Effect<void>;
|
|
18
|
-
untrack: (pid: number) => Effect.Effect<void>;
|
|
19
|
-
getAll: () => Effect.Effect<TrackedProcess[]>;
|
|
20
|
-
killAll: (force?: boolean) => Effect.Effect<{ killed: number[]; failed: number[] }>;
|
|
21
|
-
persist: () => Effect.Effect<void>;
|
|
22
|
-
restore: () => Effect.Effect<void>;
|
|
23
|
-
}
|
|
24
|
-
|
|
25
|
-
export function getPidFilePath(configDir: string): string {
|
|
26
|
-
return join(getBosDir(configDir), "pids.json");
|
|
27
|
-
}
|
|
28
|
-
|
|
29
|
-
const isProcessAlive = (pid: number): boolean => {
|
|
30
|
-
try {
|
|
31
|
-
process.kill(pid, 0);
|
|
32
|
-
return true;
|
|
33
|
-
} catch {
|
|
34
|
-
return false;
|
|
35
|
-
}
|
|
36
|
-
};
|
|
37
|
-
|
|
38
|
-
const killProcess = (pid: number, signal: NodeJS.Signals): boolean => {
|
|
39
|
-
try {
|
|
40
|
-
// Try process group first (covers spawned child trees)
|
|
41
|
-
try {
|
|
42
|
-
process.kill(-pid, signal);
|
|
43
|
-
} catch {}
|
|
44
|
-
process.kill(pid, signal);
|
|
45
|
-
return true;
|
|
46
|
-
} catch {
|
|
47
|
-
return false;
|
|
48
|
-
}
|
|
49
|
-
};
|
|
50
|
-
|
|
51
|
-
export const makeProcessRegistry = (configDir: string) =>
|
|
52
|
-
Effect.gen(function* () {
|
|
53
|
-
const tracked = yield* Ref.make(new Map<number, TrackedProcess>());
|
|
54
|
-
const pidFile = getPidFilePath(configDir);
|
|
55
|
-
|
|
56
|
-
const getAll: ProcessRegistry["getAll"] = () =>
|
|
57
|
-
Ref.get(tracked).pipe(Effect.map((m) => Array.from(m.values())));
|
|
58
|
-
|
|
59
|
-
const persist: ProcessRegistry["persist"] = () =>
|
|
60
|
-
Effect.gen(function* () {
|
|
61
|
-
const procs = yield* getAll();
|
|
62
|
-
const dir = dirname(pidFile);
|
|
63
|
-
yield* Effect.tryPromise({
|
|
64
|
-
try: async () => {
|
|
65
|
-
if (!existsSync(dir)) {
|
|
66
|
-
await mkdir(dir, { recursive: true });
|
|
67
|
-
}
|
|
68
|
-
await writeFile(pidFile, JSON.stringify(procs, null, 2));
|
|
69
|
-
},
|
|
70
|
-
catch: () => new Error("Failed to persist PIDs"),
|
|
71
|
-
}).pipe(Effect.catchAll(() => Effect.void));
|
|
72
|
-
});
|
|
73
|
-
|
|
74
|
-
const track: ProcessRegistry["track"] = (proc) =>
|
|
75
|
-
Effect.gen(function* () {
|
|
76
|
-
yield* Ref.update(tracked, (m) => new Map(m).set(proc.pid, proc));
|
|
77
|
-
yield* persist();
|
|
78
|
-
});
|
|
79
|
-
|
|
80
|
-
const untrack: ProcessRegistry["untrack"] = (pid) =>
|
|
81
|
-
Effect.gen(function* () {
|
|
82
|
-
yield* Ref.update(tracked, (m) => {
|
|
83
|
-
const copy = new Map(m);
|
|
84
|
-
copy.delete(pid);
|
|
85
|
-
return copy;
|
|
86
|
-
});
|
|
87
|
-
yield* persist();
|
|
88
|
-
});
|
|
89
|
-
|
|
90
|
-
const restore: ProcessRegistry["restore"] = () =>
|
|
91
|
-
Effect.gen(function* () {
|
|
92
|
-
if (!existsSync(pidFile)) return;
|
|
93
|
-
|
|
94
|
-
const content = yield* Effect.tryPromise({
|
|
95
|
-
try: () => readFile(pidFile, "utf8"),
|
|
96
|
-
catch: () => new Error("Failed to read PID file"),
|
|
97
|
-
}).pipe(Effect.catchAll(() => Effect.succeed("")));
|
|
98
|
-
|
|
99
|
-
if (!content) return;
|
|
100
|
-
|
|
101
|
-
let procs: TrackedProcess[];
|
|
102
|
-
try {
|
|
103
|
-
procs = JSON.parse(content) as TrackedProcess[];
|
|
104
|
-
} catch {
|
|
105
|
-
return;
|
|
106
|
-
}
|
|
107
|
-
|
|
108
|
-
const alive = procs.filter((p) => isProcessAlive(p.pid));
|
|
109
|
-
yield* Ref.set(tracked, new Map(alive.map((p) => [p.pid, p])));
|
|
110
|
-
});
|
|
111
|
-
|
|
112
|
-
yield* restore();
|
|
113
|
-
|
|
114
|
-
const killAll: ProcessRegistry["killAll"] = (force = false) =>
|
|
115
|
-
Effect.gen(function* () {
|
|
116
|
-
const procs = yield* getAll();
|
|
117
|
-
const killed: number[] = [];
|
|
118
|
-
const failed: number[] = [];
|
|
119
|
-
|
|
120
|
-
for (const proc of procs) {
|
|
121
|
-
if (!isProcessAlive(proc.pid)) {
|
|
122
|
-
yield* untrack(proc.pid);
|
|
123
|
-
continue;
|
|
124
|
-
}
|
|
125
|
-
|
|
126
|
-
const signal = force ? "SIGKILL" : "SIGTERM";
|
|
127
|
-
if (killProcess(proc.pid, signal)) {
|
|
128
|
-
killed.push(proc.pid);
|
|
129
|
-
yield* untrack(proc.pid);
|
|
130
|
-
} else {
|
|
131
|
-
failed.push(proc.pid);
|
|
132
|
-
}
|
|
133
|
-
}
|
|
134
|
-
|
|
135
|
-
if (!force && failed.length > 0) {
|
|
136
|
-
yield* Effect.sleep("500 millis");
|
|
137
|
-
for (const pid of [...failed]) {
|
|
138
|
-
if (killProcess(pid, "SIGKILL")) {
|
|
139
|
-
const idx = failed.indexOf(pid);
|
|
140
|
-
if (idx !== -1) {
|
|
141
|
-
failed.splice(idx, 1);
|
|
142
|
-
killed.push(pid);
|
|
143
|
-
}
|
|
144
|
-
yield* untrack(pid);
|
|
145
|
-
}
|
|
146
|
-
}
|
|
147
|
-
}
|
|
148
|
-
|
|
149
|
-
yield* persist();
|
|
150
|
-
return { killed, failed };
|
|
151
|
-
});
|
|
152
|
-
|
|
153
|
-
return { tracked, track, untrack, getAll, killAll, persist, restore };
|
|
154
|
-
});
|