@a5c-ai/genty-runtime 5.1.1-staging.0c6199708314
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/README.md +69 -0
- package/dist/apiResult.d.ts +19 -0
- package/dist/apiResult.d.ts.map +1 -0
- package/dist/apiResult.js +16 -0
- package/dist/background/state.d.ts +20 -0
- package/dist/background/state.d.ts.map +1 -0
- package/dist/background/state.js +52 -0
- package/dist/backgroundProcessRegistry.d.ts +124 -0
- package/dist/backgroundProcessRegistry.d.ts.map +1 -0
- package/dist/backgroundProcessRegistry.js +427 -0
- package/dist/cost/claudeCodeParser.d.ts +81 -0
- package/dist/cost/claudeCodeParser.d.ts.map +1 -0
- package/dist/cost/claudeCodeParser.js +232 -0
- package/dist/cost/collector.d.ts +42 -0
- package/dist/cost/collector.d.ts.map +1 -0
- package/dist/cost/collector.js +105 -0
- package/dist/cost/effectCost.d.ts +23 -0
- package/dist/cost/effectCost.d.ts.map +1 -0
- package/dist/cost/effectCost.js +26 -0
- package/dist/cost/index.d.ts +19 -0
- package/dist/cost/index.d.ts.map +1 -0
- package/dist/cost/index.js +39 -0
- package/dist/cost/journal.d.ts +40 -0
- package/dist/cost/journal.d.ts.map +1 -0
- package/dist/cost/journal.js +137 -0
- package/dist/cost/types.d.ts +164 -0
- package/dist/cost/types.d.ts.map +1 -0
- package/dist/cost/types.js +228 -0
- package/dist/daemon/automationExecutor.d.ts +16 -0
- package/dist/daemon/automationExecutor.d.ts.map +1 -0
- package/dist/daemon/automationExecutor.js +222 -0
- package/dist/daemon/config.d.ts +8 -0
- package/dist/daemon/config.d.ts.map +1 -0
- package/dist/daemon/config.js +245 -0
- package/dist/daemon/daemonLog.d.ts +30 -0
- package/dist/daemon/daemonLog.d.ts.map +1 -0
- package/dist/daemon/daemonLog.js +140 -0
- package/dist/daemon/durableQueue.d.ts +41 -0
- package/dist/daemon/durableQueue.d.ts.map +1 -0
- package/dist/daemon/durableQueue.js +183 -0
- package/dist/daemon/fileWatcher.d.ts +9 -0
- package/dist/daemon/fileWatcher.d.ts.map +1 -0
- package/dist/daemon/fileWatcher.js +144 -0
- package/dist/daemon/index.d.ts +15 -0
- package/dist/daemon/index.d.ts.map +1 -0
- package/dist/daemon/index.js +25 -0
- package/dist/daemon/lifecycle.d.ts +13 -0
- package/dist/daemon/lifecycle.d.ts.map +1 -0
- package/dist/daemon/lifecycle.js +320 -0
- package/dist/daemon/loop.d.ts +27 -0
- package/dist/daemon/loop.d.ts.map +1 -0
- package/dist/daemon/loop.js +387 -0
- package/dist/daemon/timerScheduler.d.ts +13 -0
- package/dist/daemon/timerScheduler.d.ts.map +1 -0
- package/dist/daemon/timerScheduler.js +212 -0
- package/dist/daemon/types.d.ts +122 -0
- package/dist/daemon/types.d.ts.map +1 -0
- package/dist/daemon/types.js +25 -0
- package/dist/daemon/webhookListener.d.ts +6 -0
- package/dist/daemon/webhookListener.d.ts.map +1 -0
- package/dist/daemon/webhookListener.js +132 -0
- package/dist/execution/index.d.ts +10 -0
- package/dist/execution/index.d.ts.map +1 -0
- package/dist/execution/index.js +20 -0
- package/dist/execution/modes/docker.d.ts +26 -0
- package/dist/execution/modes/docker.d.ts.map +1 -0
- package/dist/execution/modes/docker.js +183 -0
- package/dist/execution/modes/index.d.ts +10 -0
- package/dist/execution/modes/index.d.ts.map +1 -0
- package/dist/execution/modes/index.js +14 -0
- package/dist/execution/modes/kubernetes.d.ts +46 -0
- package/dist/execution/modes/kubernetes.d.ts.map +1 -0
- package/dist/execution/modes/kubernetes.js +334 -0
- package/dist/execution/modes/local.d.ts +23 -0
- package/dist/execution/modes/local.d.ts.map +1 -0
- package/dist/execution/modes/local.js +117 -0
- package/dist/execution/modes/ssh.d.ts +23 -0
- package/dist/execution/modes/ssh.d.ts.map +1 -0
- package/dist/execution/modes/ssh.js +144 -0
- package/dist/execution/policy.d.ts +15 -0
- package/dist/execution/policy.d.ts.map +1 -0
- package/dist/execution/policy.js +121 -0
- package/dist/execution/provider.d.ts +32 -0
- package/dist/execution/provider.d.ts.map +1 -0
- package/dist/execution/provider.js +90 -0
- package/dist/execution/types.d.ts +189 -0
- package/dist/execution/types.d.ts.map +1 -0
- package/dist/execution/types.js +9 -0
- package/dist/index.d.ts +12 -0
- package/dist/index.d.ts.map +1 -0
- package/dist/index.js +44 -0
- package/dist/observability/diagnostics.d.ts +25 -0
- package/dist/observability/diagnostics.d.ts.map +1 -0
- package/dist/observability/diagnostics.js +98 -0
- package/dist/observability/health.d.ts +19 -0
- package/dist/observability/health.d.ts.map +1 -0
- package/dist/observability/health.js +145 -0
- package/dist/observability/index.d.ts +7 -0
- package/dist/observability/index.d.ts.map +1 -0
- package/dist/observability/index.js +25 -0
- package/dist/observability/runStatus.d.ts +44 -0
- package/dist/observability/runStatus.d.ts.map +1 -0
- package/dist/observability/runStatus.js +169 -0
- package/dist/observability/timeline.d.ts +11 -0
- package/dist/observability/timeline.d.ts.map +1 -0
- package/dist/observability/timeline.js +176 -0
- package/dist/observability/types.d.ts +65 -0
- package/dist/observability/types.d.ts.map +1 -0
- package/dist/observability/types.js +8 -0
- package/dist/observability/webhooks.d.ts +68 -0
- package/dist/observability/webhooks.d.ts.map +1 -0
- package/dist/observability/webhooks.js +132 -0
- package/dist/resources/budget-tracker.d.ts +56 -0
- package/dist/resources/budget-tracker.d.ts.map +1 -0
- package/dist/resources/budget-tracker.js +131 -0
- package/dist/resources/concurrency-guard.d.ts +55 -0
- package/dist/resources/concurrency-guard.d.ts.map +1 -0
- package/dist/resources/concurrency-guard.js +132 -0
- package/dist/resources/index.d.ts +12 -0
- package/dist/resources/index.d.ts.map +1 -0
- package/dist/resources/index.js +20 -0
- package/dist/resources/manager.d.ts +52 -0
- package/dist/resources/manager.d.ts.map +1 -0
- package/dist/resources/manager.js +150 -0
- package/dist/resources/timeout-cascade.d.ts +56 -0
- package/dist/resources/timeout-cascade.d.ts.map +1 -0
- package/dist/resources/timeout-cascade.js +145 -0
- package/dist/resources/types.d.ts +130 -0
- package/dist/resources/types.d.ts.map +1 -0
- package/dist/resources/types.js +9 -0
- package/dist/rpc/index.d.ts +5 -0
- package/dist/rpc/index.d.ts.map +1 -0
- package/dist/rpc/index.js +7 -0
- package/dist/rpc/server.d.ts +13 -0
- package/dist/rpc/server.d.ts.map +1 -0
- package/dist/rpc/server.js +64 -0
- package/dist/rpc/server.test.d.ts +2 -0
- package/dist/rpc/server.test.d.ts.map +1 -0
- package/dist/rpc/server.test.js +35 -0
- package/dist/rpc/types.d.ts +23 -0
- package/dist/rpc/types.d.ts.map +1 -0
- package/dist/rpc/types.js +20 -0
- package/dist/session/context.d.ts +22 -0
- package/dist/session/context.d.ts.map +1 -0
- package/dist/session/context.js +113 -0
- package/dist/session/continuityState.d.ts +39 -0
- package/dist/session/continuityState.d.ts.map +1 -0
- package/dist/session/continuityState.js +164 -0
- package/dist/session/cost.d.ts +63 -0
- package/dist/session/cost.d.ts.map +1 -0
- package/dist/session/cost.js +194 -0
- package/dist/session/discovery.d.ts +22 -0
- package/dist/session/discovery.d.ts.map +1 -0
- package/dist/session/discovery.js +35 -0
- package/dist/session/export.d.ts +4 -0
- package/dist/session/export.d.ts.map +1 -0
- package/dist/session/export.js +56 -0
- package/dist/session/export.test.d.ts +2 -0
- package/dist/session/export.test.d.ts.map +1 -0
- package/dist/session/export.test.js +42 -0
- package/dist/session/history.d.ts +30 -0
- package/dist/session/history.d.ts.map +1 -0
- package/dist/session/history.js +143 -0
- package/dist/session/index.d.ts +20 -0
- package/dist/session/index.d.ts.map +1 -0
- package/dist/session/index.js +78 -0
- package/dist/session/memoryExtraction.d.ts +65 -0
- package/dist/session/memoryExtraction.d.ts.map +1 -0
- package/dist/session/memoryExtraction.js +201 -0
- package/dist/session/parse.d.ts +45 -0
- package/dist/session/parse.d.ts.map +1 -0
- package/dist/session/parse.js +170 -0
- package/dist/session/persistence.d.ts +46 -0
- package/dist/session/persistence.d.ts.map +1 -0
- package/dist/session/persistence.js +180 -0
- package/dist/session/tree.d.ts +29 -0
- package/dist/session/tree.d.ts.map +1 -0
- package/dist/session/tree.js +115 -0
- package/dist/session/tree.test.d.ts +2 -0
- package/dist/session/tree.test.d.ts.map +1 -0
- package/dist/session/tree.test.js +75 -0
- package/dist/session/types.d.ts +267 -0
- package/dist/session/types.d.ts.map +1 -0
- package/dist/session/types.js +45 -0
- package/dist/session/write.d.ts +61 -0
- package/dist/session/write.d.ts.map +1 -0
- package/dist/session/write.js +213 -0
- package/dist/shellInvocation.d.ts +6 -0
- package/dist/shellInvocation.d.ts.map +1 -0
- package/dist/shellInvocation.js +8 -0
- package/dist/shellInvocation.test.d.ts +2 -0
- package/dist/shellInvocation.test.d.ts.map +1 -0
- package/dist/shellInvocation.test.js +18 -0
- package/dist/telemetry/audit-log.d.ts +56 -0
- package/dist/telemetry/audit-log.d.ts.map +1 -0
- package/dist/telemetry/audit-log.js +59 -0
- package/dist/telemetry/exporters.d.ts +35 -0
- package/dist/telemetry/exporters.d.ts.map +1 -0
- package/dist/telemetry/exporters.js +141 -0
- package/dist/telemetry/index.d.ts +12 -0
- package/dist/telemetry/index.d.ts.map +1 -0
- package/dist/telemetry/index.js +25 -0
- package/dist/telemetry/provider.d.ts +57 -0
- package/dist/telemetry/provider.d.ts.map +1 -0
- package/dist/telemetry/provider.js +261 -0
- package/dist/telemetry/span-tree.d.ts +46 -0
- package/dist/telemetry/span-tree.d.ts.map +1 -0
- package/dist/telemetry/span-tree.js +93 -0
- package/dist/telemetry/traceContext.d.ts +10 -0
- package/dist/telemetry/traceContext.d.ts.map +1 -0
- package/dist/telemetry/traceContext.js +43 -0
- package/dist/telemetry/types.d.ts +109 -0
- package/dist/telemetry/types.d.ts.map +1 -0
- package/dist/telemetry/types.js +21 -0
- package/package.json +137 -0
|
@@ -0,0 +1,320 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
/**
|
|
3
|
+
* GAP-REMOTE-001: Daemon Lifecycle — start/stop/status management.
|
|
4
|
+
*
|
|
5
|
+
* In foreground mode, the daemon runs in the current process (for dev/debug).
|
|
6
|
+
* In background mode, it spawns a detached child process.
|
|
7
|
+
*/
|
|
8
|
+
var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) {
|
|
9
|
+
if (k2 === undefined) k2 = k;
|
|
10
|
+
var desc = Object.getOwnPropertyDescriptor(m, k);
|
|
11
|
+
if (!desc || ("get" in desc ? !m.__esModule : desc.writable || desc.configurable)) {
|
|
12
|
+
desc = { enumerable: true, get: function() { return m[k]; } };
|
|
13
|
+
}
|
|
14
|
+
Object.defineProperty(o, k2, desc);
|
|
15
|
+
}) : (function(o, m, k, k2) {
|
|
16
|
+
if (k2 === undefined) k2 = k;
|
|
17
|
+
o[k2] = m[k];
|
|
18
|
+
}));
|
|
19
|
+
var __setModuleDefault = (this && this.__setModuleDefault) || (Object.create ? (function(o, v) {
|
|
20
|
+
Object.defineProperty(o, "default", { enumerable: true, value: v });
|
|
21
|
+
}) : function(o, v) {
|
|
22
|
+
o["default"] = v;
|
|
23
|
+
});
|
|
24
|
+
var __importStar = (this && this.__importStar) || (function () {
|
|
25
|
+
var ownKeys = function(o) {
|
|
26
|
+
ownKeys = Object.getOwnPropertyNames || function (o) {
|
|
27
|
+
var ar = [];
|
|
28
|
+
for (var k in o) if (Object.prototype.hasOwnProperty.call(o, k)) ar[ar.length] = k;
|
|
29
|
+
return ar;
|
|
30
|
+
};
|
|
31
|
+
return ownKeys(o);
|
|
32
|
+
};
|
|
33
|
+
return function (mod) {
|
|
34
|
+
if (mod && mod.__esModule) return mod;
|
|
35
|
+
var result = {};
|
|
36
|
+
if (mod != null) for (var k = ownKeys(mod), i = 0; i < k.length; i++) if (k[i] !== "default") __createBinding(result, mod, k[i]);
|
|
37
|
+
__setModuleDefault(result, mod);
|
|
38
|
+
return result;
|
|
39
|
+
};
|
|
40
|
+
})();
|
|
41
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
42
|
+
exports.startDaemon = startDaemon;
|
|
43
|
+
exports.stopDaemon = stopDaemon;
|
|
44
|
+
exports.getDaemonStatus = getDaemonStatus;
|
|
45
|
+
exports.watchDaemon = watchDaemon;
|
|
46
|
+
const node_fs_1 = require("node:fs");
|
|
47
|
+
const node_child_process_1 = require("node:child_process");
|
|
48
|
+
const path = __importStar(require("node:path"));
|
|
49
|
+
const apiResult_1 = require("../apiResult");
|
|
50
|
+
const loop_1 = require("./loop");
|
|
51
|
+
const daemonLog_1 = require("./daemonLog");
|
|
52
|
+
const DEFAULT_GRACE_PERIOD_MS = 10_000;
|
|
53
|
+
const DEFAULT_WATCHDOG_POLL_MS = 5_000;
|
|
54
|
+
// ── Atomic write helper ─────────────────────────────────────────────────────
|
|
55
|
+
async function atomicWrite(filePath, content) {
|
|
56
|
+
const dir = path.dirname(filePath);
|
|
57
|
+
await node_fs_1.promises.mkdir(dir, { recursive: true });
|
|
58
|
+
const tmpPath = `${filePath}.tmp-${process.pid}-${Date.now()}`;
|
|
59
|
+
await node_fs_1.promises.writeFile(tmpPath, content, "utf-8");
|
|
60
|
+
await node_fs_1.promises.rename(tmpPath, filePath);
|
|
61
|
+
}
|
|
62
|
+
// ── PID checking ────────────────────────────────────────────────────────────
|
|
63
|
+
function isProcessAlive(pid) {
|
|
64
|
+
try {
|
|
65
|
+
process.kill(pid, 0);
|
|
66
|
+
return true;
|
|
67
|
+
}
|
|
68
|
+
catch {
|
|
69
|
+
return false;
|
|
70
|
+
}
|
|
71
|
+
}
|
|
72
|
+
async function readPidFile(daemonDir) {
|
|
73
|
+
try {
|
|
74
|
+
const content = await node_fs_1.promises.readFile(path.join(daemonDir, "daemon.pid"), "utf-8");
|
|
75
|
+
const pid = parseInt(content.trim(), 10);
|
|
76
|
+
return isNaN(pid) ? null : pid;
|
|
77
|
+
}
|
|
78
|
+
catch {
|
|
79
|
+
return null;
|
|
80
|
+
}
|
|
81
|
+
}
|
|
82
|
+
async function readDaemonMetadata(daemonDir) {
|
|
83
|
+
try {
|
|
84
|
+
const content = await node_fs_1.promises.readFile(path.join(daemonDir, "daemon.json"), "utf-8");
|
|
85
|
+
return JSON.parse(content);
|
|
86
|
+
}
|
|
87
|
+
catch {
|
|
88
|
+
return null;
|
|
89
|
+
}
|
|
90
|
+
}
|
|
91
|
+
async function cleanupDaemonFiles(daemonDir) {
|
|
92
|
+
await node_fs_1.promises.unlink(path.join(daemonDir, "daemon.pid")).catch(() => { });
|
|
93
|
+
await node_fs_1.promises.unlink(path.join(daemonDir, "daemon.json")).catch(() => { });
|
|
94
|
+
}
|
|
95
|
+
function sleep(ms) {
|
|
96
|
+
return new Promise((resolve) => setTimeout(resolve, ms));
|
|
97
|
+
}
|
|
98
|
+
// ── startDaemon ─────────────────────────────────────────────────────────────
|
|
99
|
+
async function startDaemon(options) {
|
|
100
|
+
try {
|
|
101
|
+
const { daemonDir, workspace, foreground = false, config } = options;
|
|
102
|
+
// Check for existing daemon
|
|
103
|
+
const existingPid = await readPidFile(daemonDir);
|
|
104
|
+
if (existingPid !== null) {
|
|
105
|
+
if (isProcessAlive(existingPid)) {
|
|
106
|
+
return (0, apiResult_1.fail)("DAEMON_RUNNING", `Daemon already running with PID ${existingPid}`);
|
|
107
|
+
}
|
|
108
|
+
// Stale PID — clean up both files
|
|
109
|
+
await cleanupDaemonFiles(daemonDir);
|
|
110
|
+
}
|
|
111
|
+
await node_fs_1.promises.mkdir(daemonDir, { recursive: true });
|
|
112
|
+
let pid;
|
|
113
|
+
const startedAt = new Date().toISOString();
|
|
114
|
+
const maxConcurrentRuns = config?.maxConcurrentRuns ?? 4;
|
|
115
|
+
if (foreground) {
|
|
116
|
+
pid = process.pid;
|
|
117
|
+
}
|
|
118
|
+
else {
|
|
119
|
+
// Spawn detached child process running daemon:run
|
|
120
|
+
const child = (0, node_child_process_1.spawn)(process.execPath, [process.argv[1], "daemon:run", "--daemon-dir", daemonDir], {
|
|
121
|
+
detached: true,
|
|
122
|
+
stdio: "ignore",
|
|
123
|
+
cwd: workspace,
|
|
124
|
+
});
|
|
125
|
+
child.unref();
|
|
126
|
+
if (child.pid == null) {
|
|
127
|
+
return (0, apiResult_1.fail)("SPAWN_FAILED", "Failed to spawn daemon process — no PID returned");
|
|
128
|
+
}
|
|
129
|
+
pid = child.pid;
|
|
130
|
+
}
|
|
131
|
+
// Write PID file atomically
|
|
132
|
+
await atomicWrite(path.join(daemonDir, "daemon.pid"), String(pid));
|
|
133
|
+
// Write daemon.json metadata atomically
|
|
134
|
+
const metadata = {
|
|
135
|
+
workspace,
|
|
136
|
+
startedAt,
|
|
137
|
+
triggers: config?.triggers ?? [],
|
|
138
|
+
maxConcurrentRuns,
|
|
139
|
+
pid,
|
|
140
|
+
};
|
|
141
|
+
await atomicWrite(path.join(daemonDir, "daemon.json"), JSON.stringify(metadata, null, 2));
|
|
142
|
+
// Log daemon start
|
|
143
|
+
await (0, daemonLog_1.appendDaemonLog)(daemonDir, {
|
|
144
|
+
timestamp: startedAt,
|
|
145
|
+
event: "DAEMON_STARTED",
|
|
146
|
+
data: { pid, workspace, foreground },
|
|
147
|
+
});
|
|
148
|
+
return (0, apiResult_1.ok)({
|
|
149
|
+
pid,
|
|
150
|
+
daemonDir,
|
|
151
|
+
startedAt,
|
|
152
|
+
});
|
|
153
|
+
}
|
|
154
|
+
catch (error) {
|
|
155
|
+
const msg = error instanceof Error ? error.message : String(error);
|
|
156
|
+
return (0, apiResult_1.fail)("INTERNAL_ERROR", msg);
|
|
157
|
+
}
|
|
158
|
+
}
|
|
159
|
+
// ── stopDaemon ──────────────────────────────────────────────────────────────
|
|
160
|
+
async function stopDaemon(options) {
|
|
161
|
+
try {
|
|
162
|
+
const { daemonDir, gracePeriodMs = DEFAULT_GRACE_PERIOD_MS } = options;
|
|
163
|
+
const pid = await readPidFile(daemonDir);
|
|
164
|
+
if (pid === null) {
|
|
165
|
+
return (0, apiResult_1.fail)("DAEMON_NOT_RUNNING", "No daemon PID file found");
|
|
166
|
+
}
|
|
167
|
+
const stoppedAt = new Date().toISOString();
|
|
168
|
+
if (!isProcessAlive(pid)) {
|
|
169
|
+
// PID exists but process is dead — clean up stale files
|
|
170
|
+
await cleanupDaemonFiles(daemonDir);
|
|
171
|
+
return (0, apiResult_1.ok)({ pid, stoppedAt });
|
|
172
|
+
}
|
|
173
|
+
// For foreground mode (same process), just clean up files
|
|
174
|
+
if (pid === process.pid) {
|
|
175
|
+
await cleanupDaemonFiles(daemonDir);
|
|
176
|
+
await (0, daemonLog_1.appendDaemonLog)(daemonDir, {
|
|
177
|
+
timestamp: stoppedAt,
|
|
178
|
+
event: "DAEMON_STOPPED",
|
|
179
|
+
data: { pid, reason: "foreground-stop" },
|
|
180
|
+
}).catch(() => { });
|
|
181
|
+
return (0, apiResult_1.ok)({ pid, stoppedAt });
|
|
182
|
+
}
|
|
183
|
+
// Send SIGTERM first
|
|
184
|
+
try {
|
|
185
|
+
process.kill(pid, "SIGTERM");
|
|
186
|
+
}
|
|
187
|
+
catch {
|
|
188
|
+
// Process may have exited between check and kill
|
|
189
|
+
await cleanupDaemonFiles(daemonDir);
|
|
190
|
+
return (0, apiResult_1.ok)({ pid, stoppedAt });
|
|
191
|
+
}
|
|
192
|
+
// Wait for graceful shutdown with polling
|
|
193
|
+
const pollInterval = 100;
|
|
194
|
+
const maxPolls = Math.ceil(gracePeriodMs / pollInterval);
|
|
195
|
+
for (let i = 0; i < maxPolls; i++) {
|
|
196
|
+
await sleep(pollInterval);
|
|
197
|
+
if (!isProcessAlive(pid)) {
|
|
198
|
+
await cleanupDaemonFiles(daemonDir);
|
|
199
|
+
await (0, daemonLog_1.appendDaemonLog)(daemonDir, {
|
|
200
|
+
timestamp: new Date().toISOString(),
|
|
201
|
+
event: "DAEMON_STOPPED",
|
|
202
|
+
data: { pid, reason: "sigterm" },
|
|
203
|
+
}).catch(() => { });
|
|
204
|
+
return (0, apiResult_1.ok)({ pid, stoppedAt: new Date().toISOString() });
|
|
205
|
+
}
|
|
206
|
+
}
|
|
207
|
+
// Escalate to SIGKILL after grace period
|
|
208
|
+
try {
|
|
209
|
+
process.kill(pid, "SIGKILL");
|
|
210
|
+
}
|
|
211
|
+
catch {
|
|
212
|
+
// Already exited
|
|
213
|
+
}
|
|
214
|
+
await cleanupDaemonFiles(daemonDir);
|
|
215
|
+
await (0, daemonLog_1.appendDaemonLog)(daemonDir, {
|
|
216
|
+
timestamp: new Date().toISOString(),
|
|
217
|
+
event: "DAEMON_KILLED",
|
|
218
|
+
data: { pid, reason: "sigkill-after-grace-period", gracePeriodMs },
|
|
219
|
+
}).catch(() => { });
|
|
220
|
+
return (0, apiResult_1.ok)({ pid, stoppedAt: new Date().toISOString() });
|
|
221
|
+
}
|
|
222
|
+
catch (error) {
|
|
223
|
+
const msg = error instanceof Error ? error.message : String(error);
|
|
224
|
+
return (0, apiResult_1.fail)("INTERNAL_ERROR", msg);
|
|
225
|
+
}
|
|
226
|
+
}
|
|
227
|
+
// ── getDaemonStatus ─────────────────────────────────────────────────────────
|
|
228
|
+
async function getDaemonStatus(options) {
|
|
229
|
+
try {
|
|
230
|
+
const { daemonDir } = options;
|
|
231
|
+
const pid = await readPidFile(daemonDir);
|
|
232
|
+
if (pid === null) {
|
|
233
|
+
return (0, apiResult_1.ok)({ running: false });
|
|
234
|
+
}
|
|
235
|
+
if (!isProcessAlive(pid)) {
|
|
236
|
+
// Stale PID — clean up
|
|
237
|
+
await node_fs_1.promises.unlink(path.join(daemonDir, "daemon.pid")).catch(() => { });
|
|
238
|
+
return (0, apiResult_1.ok)({ running: false });
|
|
239
|
+
}
|
|
240
|
+
const metadata = await readDaemonMetadata(daemonDir);
|
|
241
|
+
const startedAt = metadata?.startedAt;
|
|
242
|
+
const uptime = startedAt
|
|
243
|
+
? Math.floor((Date.now() - new Date(startedAt).getTime()) / 1000)
|
|
244
|
+
: 0;
|
|
245
|
+
const loopStatus = await (0, loop_1.readDaemonLoopStatus)(daemonDir);
|
|
246
|
+
return (0, apiResult_1.ok)({
|
|
247
|
+
running: true,
|
|
248
|
+
pid,
|
|
249
|
+
uptime,
|
|
250
|
+
startedAt,
|
|
251
|
+
activeTriggers: metadata?.triggers?.length ?? 0,
|
|
252
|
+
pendingRuns: loopStatus?.pendingRuns ?? 0,
|
|
253
|
+
deadLetterRuns: loopStatus?.deadLetterRuns ?? 0,
|
|
254
|
+
});
|
|
255
|
+
}
|
|
256
|
+
catch (error) {
|
|
257
|
+
const msg = error instanceof Error ? error.message : String(error);
|
|
258
|
+
return (0, apiResult_1.fail)("INTERNAL_ERROR", msg);
|
|
259
|
+
}
|
|
260
|
+
}
|
|
261
|
+
async function watchDaemon(options) {
|
|
262
|
+
try {
|
|
263
|
+
const { daemonDir, pollIntervalMs = DEFAULT_WATCHDOG_POLL_MS, maxRestarts = Number.POSITIVE_INFINITY, signal, } = options;
|
|
264
|
+
let restarts = 0;
|
|
265
|
+
while (!signal?.aborted) {
|
|
266
|
+
const metadata = await readDaemonMetadata(daemonDir);
|
|
267
|
+
if (!metadata) {
|
|
268
|
+
return (0, apiResult_1.fail)("DAEMON_NOT_CONFIGURED", "No daemon metadata found for watchdog restart");
|
|
269
|
+
}
|
|
270
|
+
const pid = await readPidFile(daemonDir);
|
|
271
|
+
if (pid === null || !isProcessAlive(pid)) {
|
|
272
|
+
if (restarts >= maxRestarts) {
|
|
273
|
+
return (0, apiResult_1.fail)("WATCHDOG_RESTART_LIMIT", `Daemon restart limit reached: ${maxRestarts}`);
|
|
274
|
+
}
|
|
275
|
+
await (0, daemonLog_1.appendDaemonLog)(daemonDir, {
|
|
276
|
+
timestamp: new Date().toISOString(),
|
|
277
|
+
event: "DAEMON_WATCHDOG_RESTART",
|
|
278
|
+
data: { previousPid: pid, restarts },
|
|
279
|
+
}).catch(() => { });
|
|
280
|
+
const result = await startDaemon({
|
|
281
|
+
daemonDir,
|
|
282
|
+
workspace: metadata.workspace,
|
|
283
|
+
foreground: false,
|
|
284
|
+
config: {
|
|
285
|
+
workspace: metadata.workspace,
|
|
286
|
+
triggers: metadata.triggers,
|
|
287
|
+
maxConcurrentRuns: metadata.maxConcurrentRuns,
|
|
288
|
+
},
|
|
289
|
+
});
|
|
290
|
+
if (!result.ok) {
|
|
291
|
+
return (0, apiResult_1.fail)(result.error.code, result.error.message);
|
|
292
|
+
}
|
|
293
|
+
restarts += 1;
|
|
294
|
+
}
|
|
295
|
+
await sleepOrAbort(pollIntervalMs, signal);
|
|
296
|
+
}
|
|
297
|
+
return (0, apiResult_1.ok)({ restarts, stoppedAt: new Date().toISOString() });
|
|
298
|
+
}
|
|
299
|
+
catch (error) {
|
|
300
|
+
const msg = error instanceof Error ? error.message : String(error);
|
|
301
|
+
return (0, apiResult_1.fail)("INTERNAL_ERROR", msg);
|
|
302
|
+
}
|
|
303
|
+
}
|
|
304
|
+
async function sleepOrAbort(ms, signal) {
|
|
305
|
+
if (!signal) {
|
|
306
|
+
await sleep(ms);
|
|
307
|
+
return;
|
|
308
|
+
}
|
|
309
|
+
await new Promise((resolve) => {
|
|
310
|
+
if (signal.aborted) {
|
|
311
|
+
resolve();
|
|
312
|
+
return;
|
|
313
|
+
}
|
|
314
|
+
const timeout = setTimeout(resolve, ms);
|
|
315
|
+
signal.addEventListener("abort", () => {
|
|
316
|
+
clearTimeout(timeout);
|
|
317
|
+
resolve();
|
|
318
|
+
}, { once: true });
|
|
319
|
+
});
|
|
320
|
+
}
|
|
@@ -0,0 +1,27 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* GAP-REMOTE-001: Daemon Loop — main event loop with concurrent run pool.
|
|
3
|
+
*/
|
|
4
|
+
import type { DaemonConfig, TriggerCallback } from "./types";
|
|
5
|
+
import type { DurableTriggerQueueOptions } from "./durableQueue";
|
|
6
|
+
export interface DaemonLoopOptions {
|
|
7
|
+
onTrigger?: TriggerCallback;
|
|
8
|
+
signal?: AbortSignal;
|
|
9
|
+
logDir?: string;
|
|
10
|
+
queue?: DurableTriggerQueueOptions;
|
|
11
|
+
}
|
|
12
|
+
export interface DaemonLoopStatus {
|
|
13
|
+
activeRuns: number;
|
|
14
|
+
pendingRuns: number;
|
|
15
|
+
deadLetterRuns?: number;
|
|
16
|
+
rejectedTriggers?: number;
|
|
17
|
+
duplicateTriggers?: number;
|
|
18
|
+
rateLimitedTriggers?: number;
|
|
19
|
+
updatedAt: string;
|
|
20
|
+
}
|
|
21
|
+
export declare function runDaemonLoop(config: DaemonConfig, options?: DaemonLoopOptions): Promise<void>;
|
|
22
|
+
/**
|
|
23
|
+
* Read the daemon loop's runtime status from its status file.
|
|
24
|
+
* Returns null if the file doesn't exist or is unreadable.
|
|
25
|
+
*/
|
|
26
|
+
export declare function readDaemonLoopStatus(daemonDir: string): Promise<DaemonLoopStatus | null>;
|
|
27
|
+
//# sourceMappingURL=loop.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"loop.d.ts","sourceRoot":"","sources":["../../src/daemon/loop.ts"],"names":[],"mappings":"AAAA;;GAEG;AAIH,OAAO,KAAK,EACV,YAAY,EAGZ,eAAe,EAEhB,MAAM,SAAS,CAAC;AAYjB,OAAO,KAAK,EAAE,0BAA0B,EAAE,MAAM,gBAAgB,CAAC;AAEjE,MAAM,WAAW,iBAAiB;IAChC,SAAS,CAAC,EAAE,eAAe,CAAC;IAC5B,MAAM,CAAC,EAAE,WAAW,CAAC;IACrB,MAAM,CAAC,EAAE,MAAM,CAAC;IAChB,KAAK,CAAC,EAAE,0BAA0B,CAAC;CACpC;AAED,MAAM,WAAW,gBAAgB;IAC/B,UAAU,EAAE,MAAM,CAAC;IACnB,WAAW,EAAE,MAAM,CAAC;IACpB,cAAc,CAAC,EAAE,MAAM,CAAC;IACxB,gBAAgB,CAAC,EAAE,MAAM,CAAC;IAC1B,iBAAiB,CAAC,EAAE,MAAM,CAAC;IAC3B,mBAAmB,CAAC,EAAE,MAAM,CAAC;IAC7B,SAAS,EAAE,MAAM,CAAC;CACnB;AAED,wBAAsB,aAAa,CACjC,MAAM,EAAE,YAAY,EACpB,OAAO,CAAC,EAAE,iBAAiB,GAC1B,OAAO,CAAC,IAAI,CAAC,CAwPf;AAsHD;;;GAGG;AACH,wBAAsB,oBAAoB,CAAC,SAAS,EAAE,MAAM,GAAG,OAAO,CAAC,gBAAgB,GAAG,IAAI,CAAC,CAQ9F"}
|