@love-moon/conductor-cli 0.2.8 → 0.2.9
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/package.json +3 -3
- package/src/daemon.js +56 -12
package/package.json
CHANGED
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
{
|
|
2
2
|
"name": "@love-moon/conductor-cli",
|
|
3
|
-
"version": "0.2.
|
|
3
|
+
"version": "0.2.9",
|
|
4
4
|
"type": "module",
|
|
5
5
|
"bin": {
|
|
6
6
|
"conductor": "bin/conductor.js"
|
|
@@ -16,8 +16,8 @@
|
|
|
16
16
|
"test": "node --test"
|
|
17
17
|
},
|
|
18
18
|
"dependencies": {
|
|
19
|
-
"@love-moon/tui-driver": "0.2.
|
|
20
|
-
"@love-moon/conductor-sdk": "0.2.
|
|
19
|
+
"@love-moon/tui-driver": "0.2.9",
|
|
20
|
+
"@love-moon/conductor-sdk": "0.2.9",
|
|
21
21
|
"dotenv": "^16.4.5",
|
|
22
22
|
"enquirer": "^2.4.1",
|
|
23
23
|
"js-yaml": "^4.1.1",
|
package/src/daemon.js
CHANGED
|
@@ -61,6 +61,7 @@ export function startDaemon(config = {}, deps = {}) {
|
|
|
61
61
|
const killFn = deps.kill || process.kill;
|
|
62
62
|
let requestShutdown = async () => {};
|
|
63
63
|
let shutdownSignalHandled = false;
|
|
64
|
+
let forcedSignalExitHandled = false;
|
|
64
65
|
|
|
65
66
|
const exitAndReturn = (code) => {
|
|
66
67
|
exitFn(code);
|
|
@@ -145,6 +146,14 @@ export function startDaemon(config = {}, deps = {}) {
|
|
|
145
146
|
process.env.CONDUCTOR_STOP_FORCE_KILL_TIMEOUT_MS,
|
|
146
147
|
5000,
|
|
147
148
|
);
|
|
149
|
+
const SHUTDOWN_STATUS_REPORT_TIMEOUT_MS = parsePositiveInt(
|
|
150
|
+
process.env.CONDUCTOR_SHUTDOWN_STATUS_REPORT_TIMEOUT_MS,
|
|
151
|
+
1000,
|
|
152
|
+
);
|
|
153
|
+
const SHUTDOWN_DISCONNECT_TIMEOUT_MS = parsePositiveInt(
|
|
154
|
+
process.env.CONDUCTOR_SHUTDOWN_DISCONNECT_TIMEOUT_MS,
|
|
155
|
+
1000,
|
|
156
|
+
);
|
|
148
157
|
|
|
149
158
|
try {
|
|
150
159
|
mkdirSyncFn(WORKSPACE_ROOT, { recursive: true });
|
|
@@ -216,8 +225,16 @@ export function startDaemon(config = {}, deps = {}) {
|
|
|
216
225
|
};
|
|
217
226
|
|
|
218
227
|
process.on("exit", cleanupLock);
|
|
228
|
+
const signalExitCode = (signal) => (signal === "SIGINT" ? 130 : 143);
|
|
219
229
|
const handleSignal = (signal) => {
|
|
220
|
-
if (shutdownSignalHandled)
|
|
230
|
+
if (shutdownSignalHandled) {
|
|
231
|
+
if (forcedSignalExitHandled) return;
|
|
232
|
+
forcedSignalExitHandled = true;
|
|
233
|
+
log(`Received ${signal} again, forcing exit now`);
|
|
234
|
+
cleanupLock();
|
|
235
|
+
exitFn(signalExitCode(signal));
|
|
236
|
+
return;
|
|
237
|
+
}
|
|
221
238
|
shutdownSignalHandled = true;
|
|
222
239
|
void (async () => {
|
|
223
240
|
try {
|
|
@@ -227,7 +244,7 @@ export function startDaemon(config = {}, deps = {}) {
|
|
|
227
244
|
logError(`Graceful shutdown failed on ${signal}: ${err?.message || err}`);
|
|
228
245
|
} finally {
|
|
229
246
|
cleanupLock();
|
|
230
|
-
exitFn(
|
|
247
|
+
exitFn(signalExitCode(signal));
|
|
231
248
|
}
|
|
232
249
|
})();
|
|
233
250
|
};
|
|
@@ -892,15 +909,19 @@ export function startDaemon(config = {}, deps = {}) {
|
|
|
892
909
|
activeEntries.map(async ([taskId, record]) => {
|
|
893
910
|
suppressedExitStatusReports.add(taskId);
|
|
894
911
|
try {
|
|
895
|
-
await
|
|
896
|
-
|
|
897
|
-
|
|
898
|
-
|
|
899
|
-
|
|
900
|
-
|
|
901
|
-
|
|
902
|
-
|
|
903
|
-
|
|
912
|
+
await withTimeout(
|
|
913
|
+
client.sendJson({
|
|
914
|
+
type: "task_status_update",
|
|
915
|
+
payload: {
|
|
916
|
+
task_id: taskId,
|
|
917
|
+
project_id: record.projectId,
|
|
918
|
+
status: "KILLED",
|
|
919
|
+
summary: `daemon shutdown (${reason})`,
|
|
920
|
+
},
|
|
921
|
+
}),
|
|
922
|
+
SHUTDOWN_STATUS_REPORT_TIMEOUT_MS,
|
|
923
|
+
`report shutdown status for ${taskId}`,
|
|
924
|
+
);
|
|
904
925
|
} catch (err) {
|
|
905
926
|
logError(`Failed to report shutdown status (KILLED) for ${taskId}: ${err?.message || err}`);
|
|
906
927
|
}
|
|
@@ -923,7 +944,11 @@ export function startDaemon(config = {}, deps = {}) {
|
|
|
923
944
|
activeTaskProcesses.clear();
|
|
924
945
|
|
|
925
946
|
try {
|
|
926
|
-
await
|
|
947
|
+
await withTimeout(
|
|
948
|
+
Promise.resolve(client.disconnect()),
|
|
949
|
+
SHUTDOWN_DISCONNECT_TIMEOUT_MS,
|
|
950
|
+
"disconnect daemon websocket",
|
|
951
|
+
);
|
|
927
952
|
} catch (error) {
|
|
928
953
|
logError(`Failed to disconnect client on daemon close: ${error?.message || error}`);
|
|
929
954
|
}
|
|
@@ -989,6 +1014,25 @@ function parsePositiveInt(value, fallback) {
|
|
|
989
1014
|
return fallback;
|
|
990
1015
|
}
|
|
991
1016
|
|
|
1017
|
+
async function withTimeout(promise, timeoutMs, label) {
|
|
1018
|
+
let timer = null;
|
|
1019
|
+
const timeoutPromise = new Promise((_, reject) => {
|
|
1020
|
+
timer = setTimeout(() => {
|
|
1021
|
+
reject(new Error(`${label} timed out after ${timeoutMs}ms`));
|
|
1022
|
+
}, timeoutMs);
|
|
1023
|
+
if (typeof timer?.unref === "function") {
|
|
1024
|
+
timer.unref();
|
|
1025
|
+
}
|
|
1026
|
+
});
|
|
1027
|
+
try {
|
|
1028
|
+
return await Promise.race([promise, timeoutPromise]);
|
|
1029
|
+
} finally {
|
|
1030
|
+
if (timer) {
|
|
1031
|
+
clearTimeout(timer);
|
|
1032
|
+
}
|
|
1033
|
+
}
|
|
1034
|
+
}
|
|
1035
|
+
|
|
992
1036
|
function expandHomePath(inputPath, homeDir) {
|
|
993
1037
|
if (typeof inputPath !== "string" || !inputPath) {
|
|
994
1038
|
return inputPath;
|