opencode-bridge 2.9.0-beta
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/.env.example +131 -0
- package/LICENSE +674 -0
- package/README.md +1195 -0
- package/bin/opencode-bridge.js +31 -0
- package/dist/commands/effort.d.ts +9 -0
- package/dist/commands/effort.d.ts.map +1 -0
- package/dist/commands/effort.js +56 -0
- package/dist/commands/effort.js.map +1 -0
- package/dist/commands/parser.d.ts +37 -0
- package/dist/commands/parser.d.ts.map +1 -0
- package/dist/commands/parser.js +355 -0
- package/dist/commands/parser.js.map +1 -0
- package/dist/config.d.ts +91 -0
- package/dist/config.d.ts.map +1 -0
- package/dist/config.js +340 -0
- package/dist/config.js.map +1 -0
- package/dist/feishu/cards-stream.d.ts +65 -0
- package/dist/feishu/cards-stream.d.ts.map +1 -0
- package/dist/feishu/cards-stream.js +448 -0
- package/dist/feishu/cards-stream.js.map +1 -0
- package/dist/feishu/cards.d.ts +81 -0
- package/dist/feishu/cards.d.ts.map +1 -0
- package/dist/feishu/cards.js +560 -0
- package/dist/feishu/cards.js.map +1 -0
- package/dist/feishu/client.d.ts +132 -0
- package/dist/feishu/client.d.ts.map +1 -0
- package/dist/feishu/client.js +952 -0
- package/dist/feishu/client.js.map +1 -0
- package/dist/feishu/streamer.d.ts +30 -0
- package/dist/feishu/streamer.d.ts.map +1 -0
- package/dist/feishu/streamer.js +95 -0
- package/dist/feishu/streamer.js.map +1 -0
- package/dist/handlers/card-action.d.ts +12 -0
- package/dist/handlers/card-action.d.ts.map +1 -0
- package/dist/handlers/card-action.js +154 -0
- package/dist/handlers/card-action.js.map +1 -0
- package/dist/handlers/command.d.ts +76 -0
- package/dist/handlers/command.d.ts.map +1 -0
- package/dist/handlers/command.js +1773 -0
- package/dist/handlers/command.js.map +1 -0
- package/dist/handlers/discord.d.ts +78 -0
- package/dist/handlers/discord.d.ts.map +1 -0
- package/dist/handlers/discord.js +1832 -0
- package/dist/handlers/discord.js.map +1 -0
- package/dist/handlers/file-sender.d.ts +22 -0
- package/dist/handlers/file-sender.d.ts.map +1 -0
- package/dist/handlers/file-sender.js +183 -0
- package/dist/handlers/file-sender.js.map +1 -0
- package/dist/handlers/group.d.ts +21 -0
- package/dist/handlers/group.d.ts.map +1 -0
- package/dist/handlers/group.js +414 -0
- package/dist/handlers/group.js.map +1 -0
- package/dist/handlers/lifecycle.d.ts +17 -0
- package/dist/handlers/lifecycle.d.ts.map +1 -0
- package/dist/handlers/lifecycle.js +129 -0
- package/dist/handlers/lifecycle.js.map +1 -0
- package/dist/handlers/p2p.d.ts +44 -0
- package/dist/handlers/p2p.d.ts.map +1 -0
- package/dist/handlers/p2p.js +625 -0
- package/dist/handlers/p2p.js.map +1 -0
- package/dist/index.d.ts +33 -0
- package/dist/index.d.ts.map +1 -0
- package/dist/index.js +1562 -0
- package/dist/index.js.map +1 -0
- package/dist/opencode/client.d.ts +176 -0
- package/dist/opencode/client.d.ts.map +1 -0
- package/dist/opencode/client.js +1126 -0
- package/dist/opencode/client.js.map +1 -0
- package/dist/opencode/delayed-handler.d.ts +33 -0
- package/dist/opencode/delayed-handler.d.ts.map +1 -0
- package/dist/opencode/delayed-handler.js +74 -0
- package/dist/opencode/delayed-handler.js.map +1 -0
- package/dist/opencode/output-buffer.d.ts +56 -0
- package/dist/opencode/output-buffer.d.ts.map +1 -0
- package/dist/opencode/output-buffer.js +202 -0
- package/dist/opencode/output-buffer.js.map +1 -0
- package/dist/opencode/question-handler.d.ts +61 -0
- package/dist/opencode/question-handler.d.ts.map +1 -0
- package/dist/opencode/question-handler.js +183 -0
- package/dist/opencode/question-handler.js.map +1 -0
- package/dist/opencode/question-parser.d.ts +9 -0
- package/dist/opencode/question-parser.d.ts.map +1 -0
- package/dist/opencode/question-parser.js +69 -0
- package/dist/opencode/question-parser.js.map +1 -0
- package/dist/opencode/session-queue.d.ts +16 -0
- package/dist/opencode/session-queue.d.ts.map +1 -0
- package/dist/opencode/session-queue.js +41 -0
- package/dist/opencode/session-queue.js.map +1 -0
- package/dist/permissions/handler.d.ts +36 -0
- package/dist/permissions/handler.d.ts.map +1 -0
- package/dist/permissions/handler.js +141 -0
- package/dist/permissions/handler.js.map +1 -0
- package/dist/platform/adapters/discord-adapter.d.ts +45 -0
- package/dist/platform/adapters/discord-adapter.d.ts.map +1 -0
- package/dist/platform/adapters/discord-adapter.js +497 -0
- package/dist/platform/adapters/discord-adapter.js.map +1 -0
- package/dist/platform/adapters/feishu-adapter.d.ts +29 -0
- package/dist/platform/adapters/feishu-adapter.d.ts.map +1 -0
- package/dist/platform/adapters/feishu-adapter.js +150 -0
- package/dist/platform/adapters/feishu-adapter.js.map +1 -0
- package/dist/platform/registry.d.ts +41 -0
- package/dist/platform/registry.d.ts.map +1 -0
- package/dist/platform/registry.js +87 -0
- package/dist/platform/registry.js.map +1 -0
- package/dist/platform/types.d.ts +92 -0
- package/dist/platform/types.d.ts.map +1 -0
- package/dist/platform/types.js +4 -0
- package/dist/platform/types.js.map +1 -0
- package/dist/reliability/audit-log.d.ts +93 -0
- package/dist/reliability/audit-log.d.ts.map +1 -0
- package/dist/reliability/audit-log.js +248 -0
- package/dist/reliability/audit-log.js.map +1 -0
- package/dist/reliability/config-guard.d.ts +42 -0
- package/dist/reliability/config-guard.d.ts.map +1 -0
- package/dist/reliability/config-guard.js +264 -0
- package/dist/reliability/config-guard.js.map +1 -0
- package/dist/reliability/conversation-heartbeat.d.ts +37 -0
- package/dist/reliability/conversation-heartbeat.d.ts.map +1 -0
- package/dist/reliability/conversation-heartbeat.js +179 -0
- package/dist/reliability/conversation-heartbeat.js.map +1 -0
- package/dist/reliability/cron-api-server.d.ts +13 -0
- package/dist/reliability/cron-api-server.d.ts.map +1 -0
- package/dist/reliability/cron-api-server.js +247 -0
- package/dist/reliability/cron-api-server.js.map +1 -0
- package/dist/reliability/cron-control.d.ts +34 -0
- package/dist/reliability/cron-control.d.ts.map +1 -0
- package/dist/reliability/cron-control.js +864 -0
- package/dist/reliability/cron-control.js.map +1 -0
- package/dist/reliability/cron-semantic.d.ts +9 -0
- package/dist/reliability/cron-semantic.d.ts.map +1 -0
- package/dist/reliability/cron-semantic.js +208 -0
- package/dist/reliability/cron-semantic.js.map +1 -0
- package/dist/reliability/environment-doctor.d.ts +56 -0
- package/dist/reliability/environment-doctor.d.ts.map +1 -0
- package/dist/reliability/environment-doctor.js +213 -0
- package/dist/reliability/environment-doctor.js.map +1 -0
- package/dist/reliability/job-registry.d.ts +26 -0
- package/dist/reliability/job-registry.d.ts.map +1 -0
- package/dist/reliability/job-registry.js +77 -0
- package/dist/reliability/job-registry.js.map +1 -0
- package/dist/reliability/opencode-probe.d.ts +37 -0
- package/dist/reliability/opencode-probe.d.ts.map +1 -0
- package/dist/reliability/opencode-probe.js +195 -0
- package/dist/reliability/opencode-probe.js.map +1 -0
- package/dist/reliability/opencode-restart.d.ts +42 -0
- package/dist/reliability/opencode-restart.d.ts.map +1 -0
- package/dist/reliability/opencode-restart.js +155 -0
- package/dist/reliability/opencode-restart.js.map +1 -0
- package/dist/reliability/proactive-heartbeat.d.ts +39 -0
- package/dist/reliability/proactive-heartbeat.d.ts.map +1 -0
- package/dist/reliability/proactive-heartbeat.js +147 -0
- package/dist/reliability/proactive-heartbeat.js.map +1 -0
- package/dist/reliability/process-check-job.d.ts +73 -0
- package/dist/reliability/process-check-job.d.ts.map +1 -0
- package/dist/reliability/process-check-job.js +254 -0
- package/dist/reliability/process-check-job.js.map +1 -0
- package/dist/reliability/process-guard.d.ts +53 -0
- package/dist/reliability/process-guard.d.ts.map +1 -0
- package/dist/reliability/process-guard.js +344 -0
- package/dist/reliability/process-guard.js.map +1 -0
- package/dist/reliability/recovery-reporter.d.ts +37 -0
- package/dist/reliability/recovery-reporter.d.ts.map +1 -0
- package/dist/reliability/recovery-reporter.js +161 -0
- package/dist/reliability/recovery-reporter.js.map +1 -0
- package/dist/reliability/rescue-executor.d.ts +52 -0
- package/dist/reliability/rescue-executor.d.ts.map +1 -0
- package/dist/reliability/rescue-executor.js +244 -0
- package/dist/reliability/rescue-executor.js.map +1 -0
- package/dist/reliability/rescue-policy.d.ts +39 -0
- package/dist/reliability/rescue-policy.d.ts.map +1 -0
- package/dist/reliability/rescue-policy.js +85 -0
- package/dist/reliability/rescue-policy.js.map +1 -0
- package/dist/reliability/runtime-cron-dispatcher.d.ts +30 -0
- package/dist/reliability/runtime-cron-dispatcher.d.ts.map +1 -0
- package/dist/reliability/runtime-cron-dispatcher.js +100 -0
- package/dist/reliability/runtime-cron-dispatcher.js.map +1 -0
- package/dist/reliability/runtime-cron-orphan.d.ts +18 -0
- package/dist/reliability/runtime-cron-orphan.d.ts.map +1 -0
- package/dist/reliability/runtime-cron-orphan.js +87 -0
- package/dist/reliability/runtime-cron-orphan.js.map +1 -0
- package/dist/reliability/runtime-cron-registry.d.ts +4 -0
- package/dist/reliability/runtime-cron-registry.d.ts.map +1 -0
- package/dist/reliability/runtime-cron-registry.js +8 -0
- package/dist/reliability/runtime-cron-registry.js.map +1 -0
- package/dist/reliability/runtime-cron.d.ts +75 -0
- package/dist/reliability/runtime-cron.d.ts.map +1 -0
- package/dist/reliability/runtime-cron.js +309 -0
- package/dist/reliability/runtime-cron.js.map +1 -0
- package/dist/reliability/scheduler.d.ts +38 -0
- package/dist/reliability/scheduler.d.ts.map +1 -0
- package/dist/reliability/scheduler.js +174 -0
- package/dist/reliability/scheduler.js.map +1 -0
- package/dist/reliability/types.d.ts +151 -0
- package/dist/reliability/types.d.ts.map +1 -0
- package/dist/reliability/types.js +178 -0
- package/dist/reliability/types.js.map +1 -0
- package/dist/router/action-handlers.d.ts +27 -0
- package/dist/router/action-handlers.d.ts.map +1 -0
- package/dist/router/action-handlers.js +226 -0
- package/dist/router/action-handlers.js.map +1 -0
- package/dist/router/opencode-event-hub.d.ts +159 -0
- package/dist/router/opencode-event-hub.d.ts.map +1 -0
- package/dist/router/opencode-event-hub.js +589 -0
- package/dist/router/opencode-event-hub.js.map +1 -0
- package/dist/router/root-router.d.ts +94 -0
- package/dist/router/root-router.d.ts.map +1 -0
- package/dist/router/root-router.js +214 -0
- package/dist/router/root-router.js.map +1 -0
- package/dist/store/chat-session.d.ts +150 -0
- package/dist/store/chat-session.d.ts.map +1 -0
- package/dist/store/chat-session.js +640 -0
- package/dist/store/chat-session.js.map +1 -0
- package/dist/store/session-directory.d.ts +12 -0
- package/dist/store/session-directory.d.ts.map +1 -0
- package/dist/store/session-directory.js +47 -0
- package/dist/store/session-directory.js.map +1 -0
- package/dist/store/session-group.d.ts +19 -0
- package/dist/store/session-group.d.ts.map +1 -0
- package/dist/store/session-group.js +92 -0
- package/dist/store/session-group.js.map +1 -0
- package/dist/store/user-session.d.ts +19 -0
- package/dist/store/user-session.d.ts.map +1 -0
- package/dist/store/user-session.js +112 -0
- package/dist/store/user-session.js.map +1 -0
- package/dist/utils/async-queue.d.ts +12 -0
- package/dist/utils/async-queue.d.ts.map +1 -0
- package/dist/utils/async-queue.js +51 -0
- package/dist/utils/async-queue.js.map +1 -0
- package/dist/utils/directory-policy.d.ts +50 -0
- package/dist/utils/directory-policy.d.ts.map +1 -0
- package/dist/utils/directory-policy.js +379 -0
- package/dist/utils/directory-policy.js.map +1 -0
- package/dist/utils/session-title.d.ts +2 -0
- package/dist/utils/session-title.d.ts.map +1 -0
- package/dist/utils/session-title.js +10 -0
- package/dist/utils/session-title.js.map +1 -0
- package/package.json +73 -0
|
@@ -0,0 +1,147 @@
|
|
|
1
|
+
import fs from 'node:fs';
|
|
2
|
+
import path from 'node:path';
|
|
3
|
+
const DEFAULT_PROMPT = 'Read HEARTBEAT.md if it exists (workspace context). Follow it strictly. Do not infer or repeat old tasks from prior chats. If nothing needs attention, reply HEARTBEAT_OK.';
|
|
4
|
+
export function createProactiveHeartbeatRunner(options) {
|
|
5
|
+
const logger = options.logger ?? console;
|
|
6
|
+
const checklistPath = options.checklistPath ?? path.join(process.cwd(), 'HEARTBEAT.md');
|
|
7
|
+
const sessionStatePath = options.sessionStatePath ?? path.join(process.cwd(), 'memory', 'heartbeat-session.json');
|
|
8
|
+
const sessionTitle = options.sessionTitle?.trim() || 'Bridge Heartbeat Monitor';
|
|
9
|
+
const prompt = options.prompt?.trim() || DEFAULT_PROMPT;
|
|
10
|
+
const intervalMs = options.intervalMs > 0 ? options.intervalMs : 30 * 60 * 1000;
|
|
11
|
+
const agent = options.agent?.trim();
|
|
12
|
+
const directory = options.directory?.trim();
|
|
13
|
+
let timer = null;
|
|
14
|
+
let running = false;
|
|
15
|
+
let closed = false;
|
|
16
|
+
const runNow = async () => {
|
|
17
|
+
if (!options.enabled || closed) {
|
|
18
|
+
return;
|
|
19
|
+
}
|
|
20
|
+
if (running) {
|
|
21
|
+
logger.info('[Heartbeat] proactive tick skipped: previous run still active');
|
|
22
|
+
return;
|
|
23
|
+
}
|
|
24
|
+
running = true;
|
|
25
|
+
try {
|
|
26
|
+
if (!fs.existsSync(checklistPath)) {
|
|
27
|
+
logger.warn(`[Heartbeat] checklist not found, continue with prompt contract: ${checklistPath}`);
|
|
28
|
+
}
|
|
29
|
+
const sessionId = await ensureSessionId();
|
|
30
|
+
const response = await options.client.sendMessage(sessionId, prompt, {
|
|
31
|
+
...(agent ? { agent } : {}),
|
|
32
|
+
...(directory ? { directory } : {}),
|
|
33
|
+
});
|
|
34
|
+
const text = extractText(response.parts).trim();
|
|
35
|
+
if (isHeartbeatOk(text)) {
|
|
36
|
+
logger.info('[Heartbeat] proactive heartbeat ack: HEARTBEAT_OK');
|
|
37
|
+
return;
|
|
38
|
+
}
|
|
39
|
+
const alert = text || 'Heartbeat run completed without HEARTBEAT_OK marker.';
|
|
40
|
+
logger.warn(`[Heartbeat] proactive alert: ${alert.slice(0, 300)}`);
|
|
41
|
+
if (options.notifyAlert) {
|
|
42
|
+
await options.notifyAlert(alert);
|
|
43
|
+
}
|
|
44
|
+
}
|
|
45
|
+
catch (error) {
|
|
46
|
+
logger.error('[Heartbeat] proactive heartbeat failed:', error);
|
|
47
|
+
if (options.notifyAlert) {
|
|
48
|
+
const message = error instanceof Error ? error.message : String(error);
|
|
49
|
+
await options.notifyAlert(`Heartbeat proactive run failed: ${message}`);
|
|
50
|
+
}
|
|
51
|
+
}
|
|
52
|
+
finally {
|
|
53
|
+
running = false;
|
|
54
|
+
}
|
|
55
|
+
};
|
|
56
|
+
return {
|
|
57
|
+
start: () => {
|
|
58
|
+
if (!options.enabled || closed || timer) {
|
|
59
|
+
return;
|
|
60
|
+
}
|
|
61
|
+
timer = setInterval(() => {
|
|
62
|
+
void runNow();
|
|
63
|
+
}, intervalMs);
|
|
64
|
+
logger.info(`[Heartbeat] proactive runner started (intervalMs=${intervalMs})`);
|
|
65
|
+
},
|
|
66
|
+
stop: async () => {
|
|
67
|
+
closed = true;
|
|
68
|
+
if (timer) {
|
|
69
|
+
clearInterval(timer);
|
|
70
|
+
timer = null;
|
|
71
|
+
}
|
|
72
|
+
while (running) {
|
|
73
|
+
await wait(50);
|
|
74
|
+
}
|
|
75
|
+
logger.info('[Heartbeat] proactive runner stopped');
|
|
76
|
+
},
|
|
77
|
+
runNow,
|
|
78
|
+
};
|
|
79
|
+
async function ensureSessionId() {
|
|
80
|
+
const state = readSessionState(sessionStatePath);
|
|
81
|
+
if (state?.sessionId) {
|
|
82
|
+
const exists = await options.client.getSessionById(state.sessionId, directory ? { directory } : undefined);
|
|
83
|
+
if (exists) {
|
|
84
|
+
return state.sessionId;
|
|
85
|
+
}
|
|
86
|
+
}
|
|
87
|
+
const created = await options.client.createSession(sessionTitle, directory);
|
|
88
|
+
const nextState = {
|
|
89
|
+
sessionId: created.id,
|
|
90
|
+
updatedAtMs: Date.now(),
|
|
91
|
+
};
|
|
92
|
+
writeSessionState(sessionStatePath, nextState);
|
|
93
|
+
return created.id;
|
|
94
|
+
}
|
|
95
|
+
}
|
|
96
|
+
function extractText(parts) {
|
|
97
|
+
const fragments = [];
|
|
98
|
+
for (const part of parts) {
|
|
99
|
+
if (!part || typeof part !== 'object') {
|
|
100
|
+
continue;
|
|
101
|
+
}
|
|
102
|
+
const record = part;
|
|
103
|
+
if (typeof record.text === 'string' && record.text.trim()) {
|
|
104
|
+
fragments.push(record.text.trim());
|
|
105
|
+
}
|
|
106
|
+
}
|
|
107
|
+
return fragments.join('\n').trim();
|
|
108
|
+
}
|
|
109
|
+
function isHeartbeatOk(text) {
|
|
110
|
+
return /^HEARTBEAT_OK(\b|$)/i.test(text.trim());
|
|
111
|
+
}
|
|
112
|
+
function readSessionState(filePath) {
|
|
113
|
+
if (!fs.existsSync(filePath)) {
|
|
114
|
+
return null;
|
|
115
|
+
}
|
|
116
|
+
try {
|
|
117
|
+
const raw = fs.readFileSync(filePath, 'utf-8');
|
|
118
|
+
const parsed = JSON.parse(raw);
|
|
119
|
+
const sessionId = typeof parsed.sessionId === 'string' ? parsed.sessionId.trim() : '';
|
|
120
|
+
if (!sessionId) {
|
|
121
|
+
return null;
|
|
122
|
+
}
|
|
123
|
+
return {
|
|
124
|
+
sessionId,
|
|
125
|
+
updatedAtMs: typeof parsed.updatedAtMs === 'number' ? parsed.updatedAtMs : 0,
|
|
126
|
+
};
|
|
127
|
+
}
|
|
128
|
+
catch (error) {
|
|
129
|
+
console.error('[proactive-heartbeat] readSessionState failed:', error instanceof Error ? error.message : String(error));
|
|
130
|
+
return null;
|
|
131
|
+
}
|
|
132
|
+
}
|
|
133
|
+
function writeSessionState(filePath, state) {
|
|
134
|
+
const dir = path.dirname(filePath);
|
|
135
|
+
if (!fs.existsSync(dir)) {
|
|
136
|
+
fs.mkdirSync(dir, { recursive: true });
|
|
137
|
+
}
|
|
138
|
+
const tmpPath = `${filePath}.${process.pid}.tmp`;
|
|
139
|
+
fs.writeFileSync(tmpPath, `${JSON.stringify(state, null, 2)}\n`, 'utf-8');
|
|
140
|
+
fs.renameSync(tmpPath, filePath);
|
|
141
|
+
}
|
|
142
|
+
async function wait(ms) {
|
|
143
|
+
await new Promise(resolve => {
|
|
144
|
+
setTimeout(resolve, ms);
|
|
145
|
+
});
|
|
146
|
+
}
|
|
147
|
+
//# sourceMappingURL=proactive-heartbeat.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"proactive-heartbeat.js","sourceRoot":"","sources":["../../src/reliability/proactive-heartbeat.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,73 @@
|
|
|
1
|
+
import { type ProcessGuardResult, type ProcessGuardStatus } from './process-guard.js';
|
|
2
|
+
import { type CronScheduler } from './scheduler.js';
|
|
3
|
+
type ProcessAliveChecker = (pid: number) => Promise<boolean>;
|
|
4
|
+
type OpenCodeInstanceChecker = (options: {
|
|
5
|
+
pidFilePath: string;
|
|
6
|
+
host: string;
|
|
7
|
+
port: number;
|
|
8
|
+
timeoutMs?: number;
|
|
9
|
+
processKeywords?: string[];
|
|
10
|
+
}) => Promise<ProcessGuardResult>;
|
|
11
|
+
export type BridgeProcessStatus = 'ok' | 'not-running' | 'stale-pid';
|
|
12
|
+
export interface ProcessConsistencyStatus {
|
|
13
|
+
opencodeStatus: ProcessGuardStatus;
|
|
14
|
+
bridgeStatus: BridgeProcessStatus;
|
|
15
|
+
stalePidCleaned: boolean;
|
|
16
|
+
details: string;
|
|
17
|
+
}
|
|
18
|
+
export interface StaleLockCleanupResult {
|
|
19
|
+
cleanedPaths: string[];
|
|
20
|
+
skippedPaths: string[];
|
|
21
|
+
}
|
|
22
|
+
export interface BudgetResetResult {
|
|
23
|
+
previous: number;
|
|
24
|
+
current: number;
|
|
25
|
+
maxBudget: number;
|
|
26
|
+
}
|
|
27
|
+
export interface RepairBudgetState {
|
|
28
|
+
maxBudget: number;
|
|
29
|
+
remaining: number;
|
|
30
|
+
lastUpdatedAtMs: number;
|
|
31
|
+
lastResetAtMs: number;
|
|
32
|
+
}
|
|
33
|
+
export interface ProcessCheckJobRunner {
|
|
34
|
+
checkProcessConsistency: () => Promise<ProcessConsistencyStatus>;
|
|
35
|
+
cleanupStaleLocks: () => Promise<StaleLockCleanupResult>;
|
|
36
|
+
resetBudget: () => Promise<BudgetResetResult>;
|
|
37
|
+
}
|
|
38
|
+
interface ProcessCheckJobAuditEvent {
|
|
39
|
+
action: string;
|
|
40
|
+
result: 'success' | 'failed';
|
|
41
|
+
metadata?: Record<string, unknown>;
|
|
42
|
+
}
|
|
43
|
+
type ProcessCheckJobAuditLogger = (event: ProcessCheckJobAuditEvent) => Promise<void>;
|
|
44
|
+
export interface ProcessCheckJobRunnerOptions {
|
|
45
|
+
bridgePidFilePath: string;
|
|
46
|
+
opencodePidFilePath: string;
|
|
47
|
+
opencodeHost: string;
|
|
48
|
+
opencodePort: number;
|
|
49
|
+
repairBudgetState: RepairBudgetState;
|
|
50
|
+
staleLockPaths: string[];
|
|
51
|
+
staleLockMs?: number;
|
|
52
|
+
opencodeProbeTimeoutMs?: number;
|
|
53
|
+
processKeywords?: string[];
|
|
54
|
+
now?: () => number;
|
|
55
|
+
isProcessAlive?: ProcessAliveChecker;
|
|
56
|
+
checkOpenCodeSingleInstance?: OpenCodeInstanceChecker;
|
|
57
|
+
auditLog?: ProcessCheckJobAuditLogger;
|
|
58
|
+
}
|
|
59
|
+
export interface ProcessCheckJobCronExpressions {
|
|
60
|
+
processConsistencyCheck: string;
|
|
61
|
+
staleLockCleanup: string;
|
|
62
|
+
budgetReset: string;
|
|
63
|
+
}
|
|
64
|
+
export interface RegisterProcessCheckJobsOptions {
|
|
65
|
+
runner: ProcessCheckJobRunner;
|
|
66
|
+
cronExpressions?: Partial<ProcessCheckJobCronExpressions>;
|
|
67
|
+
timezone?: string;
|
|
68
|
+
}
|
|
69
|
+
export declare function createRepairBudgetState(maxBudget: number, now?: () => number): RepairBudgetState;
|
|
70
|
+
export declare function createProcessCheckJobRunner(options: ProcessCheckJobRunnerOptions): ProcessCheckJobRunner;
|
|
71
|
+
export declare function registerProcessCheckJobs(scheduler: CronScheduler, options: RegisterProcessCheckJobsOptions): void;
|
|
72
|
+
export {};
|
|
73
|
+
//# sourceMappingURL=process-check-job.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"process-check-job.d.ts","sourceRoot":"","sources":["../../src/reliability/process-check-job.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,254 @@
|
|
|
1
|
+
import fs from 'node:fs/promises';
|
|
2
|
+
import path from 'node:path';
|
|
3
|
+
import { reliabilityConfig } from '../config.js';
|
|
4
|
+
import { auditLogger, generateIncidentId } from './audit-log.js';
|
|
5
|
+
import { checkOpenCodeSingleInstance, } from './process-guard.js';
|
|
6
|
+
const DEFAULT_CRON_EXPRESSIONS = {
|
|
7
|
+
processConsistencyCheck: '*/30 * * * * *',
|
|
8
|
+
staleLockCleanup: '0 */5 * * * *',
|
|
9
|
+
budgetReset: '0 0 * * *',
|
|
10
|
+
};
|
|
11
|
+
const DEFAULT_STALE_LOCK_MS = 10 * 60 * 1000;
|
|
12
|
+
const DEFAULT_AUDIT_PATH = path.resolve(process.cwd(), 'logs', 'reliability-audit.jsonl');
|
|
13
|
+
const missingPidNoticePaths = new Set();
|
|
14
|
+
function isErrnoException(error) {
|
|
15
|
+
return typeof error === 'object' && error !== null && 'code' in error;
|
|
16
|
+
}
|
|
17
|
+
export function createRepairBudgetState(maxBudget, now = Date.now) {
|
|
18
|
+
const safeMaxBudget = Number.isFinite(maxBudget) && maxBudget > 0
|
|
19
|
+
? Math.floor(maxBudget)
|
|
20
|
+
: reliabilityConfig.repairBudget;
|
|
21
|
+
const ts = now();
|
|
22
|
+
return {
|
|
23
|
+
maxBudget: safeMaxBudget,
|
|
24
|
+
remaining: safeMaxBudget,
|
|
25
|
+
lastUpdatedAtMs: ts,
|
|
26
|
+
lastResetAtMs: ts,
|
|
27
|
+
};
|
|
28
|
+
}
|
|
29
|
+
const createAuditLogger = () => {
|
|
30
|
+
const logger = auditLogger(DEFAULT_AUDIT_PATH, true);
|
|
31
|
+
return async (event) => {
|
|
32
|
+
const auditEvent = {
|
|
33
|
+
incidentId: generateIncidentId(),
|
|
34
|
+
classification: 'system',
|
|
35
|
+
decision: 'update',
|
|
36
|
+
action: event.action,
|
|
37
|
+
result: event.result,
|
|
38
|
+
timestamp: new Date().toISOString(),
|
|
39
|
+
...(event.metadata ? { metadata: event.metadata } : {}),
|
|
40
|
+
};
|
|
41
|
+
await logger.log(auditEvent);
|
|
42
|
+
};
|
|
43
|
+
};
|
|
44
|
+
async function readPidFile(pidFilePath) {
|
|
45
|
+
try {
|
|
46
|
+
const raw = (await fs.readFile(pidFilePath, 'utf-8')).trim();
|
|
47
|
+
const pid = Number.parseInt(raw, 10);
|
|
48
|
+
return Number.isNaN(pid) ? null : pid;
|
|
49
|
+
}
|
|
50
|
+
catch (error) {
|
|
51
|
+
if (isErrnoException(error) && error.code === 'ENOENT') {
|
|
52
|
+
if (!missingPidNoticePaths.has(pidFilePath)) {
|
|
53
|
+
missingPidNoticePaths.add(pidFilePath);
|
|
54
|
+
console.info(`[process-check-job] bridge pid file not found, fallback to foreground mode detection: ${pidFilePath}`);
|
|
55
|
+
}
|
|
56
|
+
return null;
|
|
57
|
+
}
|
|
58
|
+
console.error('[process-check-job] readPidFile failed:', error instanceof Error ? error.message : String(error));
|
|
59
|
+
return null;
|
|
60
|
+
}
|
|
61
|
+
}
|
|
62
|
+
async function removeFileIfExists(filePath) {
|
|
63
|
+
try {
|
|
64
|
+
await fs.rm(filePath, { force: true });
|
|
65
|
+
return true;
|
|
66
|
+
}
|
|
67
|
+
catch {
|
|
68
|
+
return false;
|
|
69
|
+
}
|
|
70
|
+
}
|
|
71
|
+
function resolveLockPath(targetPath) {
|
|
72
|
+
return targetPath.endsWith('.lock') ? targetPath : `${targetPath}.lock`;
|
|
73
|
+
}
|
|
74
|
+
export function createProcessCheckJobRunner(options) {
|
|
75
|
+
const getNow = options.now ?? Date.now;
|
|
76
|
+
const staleLockMs = options.staleLockMs ?? DEFAULT_STALE_LOCK_MS;
|
|
77
|
+
const isProcessAlive = options.isProcessAlive ?? (async (pid) => {
|
|
78
|
+
try {
|
|
79
|
+
process.kill(pid, 0);
|
|
80
|
+
return true;
|
|
81
|
+
}
|
|
82
|
+
catch {
|
|
83
|
+
return false;
|
|
84
|
+
}
|
|
85
|
+
});
|
|
86
|
+
const checkSingleInstance = options.checkOpenCodeSingleInstance ?? checkOpenCodeSingleInstance;
|
|
87
|
+
const auditLog = options.auditLog ?? createAuditLogger();
|
|
88
|
+
return {
|
|
89
|
+
checkProcessConsistency: async () => {
|
|
90
|
+
const opencode = await checkSingleInstance({
|
|
91
|
+
pidFilePath: options.opencodePidFilePath,
|
|
92
|
+
host: options.opencodeHost,
|
|
93
|
+
port: options.opencodePort,
|
|
94
|
+
timeoutMs: options.opencodeProbeTimeoutMs,
|
|
95
|
+
processKeywords: options.processKeywords,
|
|
96
|
+
});
|
|
97
|
+
let bridgeStatus = 'not-running';
|
|
98
|
+
let stalePidCleaned = false;
|
|
99
|
+
const bridgePid = await readPidFile(options.bridgePidFilePath);
|
|
100
|
+
if (bridgePid === null) {
|
|
101
|
+
// 前台运行模式通常不会写 bridge.pid,此时当前进程即为 bridge 本体。
|
|
102
|
+
bridgeStatus = 'ok';
|
|
103
|
+
}
|
|
104
|
+
else {
|
|
105
|
+
const alive = await isProcessAlive(bridgePid);
|
|
106
|
+
if (alive) {
|
|
107
|
+
bridgeStatus = 'ok';
|
|
108
|
+
}
|
|
109
|
+
else {
|
|
110
|
+
bridgeStatus = 'stale-pid';
|
|
111
|
+
stalePidCleaned = await removeFileIfExists(options.bridgePidFilePath);
|
|
112
|
+
await auditLog({
|
|
113
|
+
action: 'bridge.pid.cleaned',
|
|
114
|
+
result: stalePidCleaned ? 'success' : 'failed',
|
|
115
|
+
metadata: {
|
|
116
|
+
pidFilePath: options.bridgePidFilePath,
|
|
117
|
+
pid: bridgePid,
|
|
118
|
+
},
|
|
119
|
+
});
|
|
120
|
+
}
|
|
121
|
+
}
|
|
122
|
+
const details = [
|
|
123
|
+
`opencode=${opencode.status}`,
|
|
124
|
+
`bridge=${bridgeStatus}`,
|
|
125
|
+
`runningPids=${opencode.runningPids.join(',') || 'none'}`,
|
|
126
|
+
].join(' ');
|
|
127
|
+
await auditLog({
|
|
128
|
+
action: 'process.consistency.check',
|
|
129
|
+
result: 'success',
|
|
130
|
+
metadata: {
|
|
131
|
+
opencodeStatus: opencode.status,
|
|
132
|
+
bridgeStatus,
|
|
133
|
+
stalePidCleaned,
|
|
134
|
+
opencodePidFromFile: opencode.pidFromFile,
|
|
135
|
+
opencodeConflictPids: opencode.conflictPids,
|
|
136
|
+
opencodeProbeReason: opencode.probeReason,
|
|
137
|
+
},
|
|
138
|
+
});
|
|
139
|
+
if (opencode.status === 'single-instance-violation') {
|
|
140
|
+
await auditLog({
|
|
141
|
+
action: 'opencode.single_instance_violation',
|
|
142
|
+
result: 'failed',
|
|
143
|
+
metadata: {
|
|
144
|
+
pidFromFile: opencode.pidFromFile,
|
|
145
|
+
runningPids: opencode.runningPids,
|
|
146
|
+
conflictPids: opencode.conflictPids,
|
|
147
|
+
},
|
|
148
|
+
});
|
|
149
|
+
}
|
|
150
|
+
return {
|
|
151
|
+
opencodeStatus: opencode.status,
|
|
152
|
+
bridgeStatus,
|
|
153
|
+
stalePidCleaned,
|
|
154
|
+
details,
|
|
155
|
+
};
|
|
156
|
+
},
|
|
157
|
+
cleanupStaleLocks: async () => {
|
|
158
|
+
const cleanedPaths = [];
|
|
159
|
+
const skippedPaths = [];
|
|
160
|
+
for (const lockTargetPath of options.staleLockPaths) {
|
|
161
|
+
const lockPath = resolveLockPath(lockTargetPath);
|
|
162
|
+
try {
|
|
163
|
+
const stat = await fs.stat(lockPath);
|
|
164
|
+
const ageMs = getNow() - stat.mtimeMs;
|
|
165
|
+
if (ageMs <= staleLockMs) {
|
|
166
|
+
skippedPaths.push(lockPath);
|
|
167
|
+
continue;
|
|
168
|
+
}
|
|
169
|
+
await fs.rm(lockPath, { recursive: true, force: true });
|
|
170
|
+
cleanedPaths.push(lockPath);
|
|
171
|
+
await auditLog({
|
|
172
|
+
action: 'lock.stale.cleaned',
|
|
173
|
+
result: 'success',
|
|
174
|
+
metadata: {
|
|
175
|
+
lockPath,
|
|
176
|
+
ageMs,
|
|
177
|
+
staleLockMs,
|
|
178
|
+
},
|
|
179
|
+
});
|
|
180
|
+
}
|
|
181
|
+
catch (error) {
|
|
182
|
+
console.error('[process-check-job] cleanupStaleLocks failed:', error instanceof Error ? error.message : String(error));
|
|
183
|
+
skippedPaths.push(lockPath);
|
|
184
|
+
}
|
|
185
|
+
}
|
|
186
|
+
await auditLog({
|
|
187
|
+
action: 'lock.cleanup.completed',
|
|
188
|
+
result: 'success',
|
|
189
|
+
metadata: {
|
|
190
|
+
cleanedCount: cleanedPaths.length,
|
|
191
|
+
skippedCount: skippedPaths.length,
|
|
192
|
+
},
|
|
193
|
+
});
|
|
194
|
+
return {
|
|
195
|
+
cleanedPaths,
|
|
196
|
+
skippedPaths,
|
|
197
|
+
};
|
|
198
|
+
},
|
|
199
|
+
resetBudget: async () => {
|
|
200
|
+
const previous = options.repairBudgetState.remaining;
|
|
201
|
+
options.repairBudgetState.remaining = options.repairBudgetState.maxBudget;
|
|
202
|
+
options.repairBudgetState.lastResetAtMs = getNow();
|
|
203
|
+
options.repairBudgetState.lastUpdatedAtMs = options.repairBudgetState.lastResetAtMs;
|
|
204
|
+
await auditLog({
|
|
205
|
+
action: 'rescue.budget.reset',
|
|
206
|
+
result: 'success',
|
|
207
|
+
metadata: {
|
|
208
|
+
previous,
|
|
209
|
+
current: options.repairBudgetState.remaining,
|
|
210
|
+
maxBudget: options.repairBudgetState.maxBudget,
|
|
211
|
+
},
|
|
212
|
+
});
|
|
213
|
+
return {
|
|
214
|
+
previous,
|
|
215
|
+
current: options.repairBudgetState.remaining,
|
|
216
|
+
maxBudget: options.repairBudgetState.maxBudget,
|
|
217
|
+
};
|
|
218
|
+
},
|
|
219
|
+
};
|
|
220
|
+
}
|
|
221
|
+
export function registerProcessCheckJobs(scheduler, options) {
|
|
222
|
+
const cronExpressions = {
|
|
223
|
+
...DEFAULT_CRON_EXPRESSIONS,
|
|
224
|
+
...options.cronExpressions,
|
|
225
|
+
};
|
|
226
|
+
scheduler.registerJob({
|
|
227
|
+
id: 'process-consistency-check',
|
|
228
|
+
cronExpression: cronExpressions.processConsistencyCheck,
|
|
229
|
+
timezone: options.timezone,
|
|
230
|
+
waitForCompletion: true,
|
|
231
|
+
run: async () => {
|
|
232
|
+
await options.runner.checkProcessConsistency();
|
|
233
|
+
},
|
|
234
|
+
});
|
|
235
|
+
scheduler.registerJob({
|
|
236
|
+
id: 'stale-lock-cleanup',
|
|
237
|
+
cronExpression: cronExpressions.staleLockCleanup,
|
|
238
|
+
timezone: options.timezone,
|
|
239
|
+
waitForCompletion: true,
|
|
240
|
+
run: async () => {
|
|
241
|
+
await options.runner.cleanupStaleLocks();
|
|
242
|
+
},
|
|
243
|
+
});
|
|
244
|
+
scheduler.registerJob({
|
|
245
|
+
id: 'budget-reset',
|
|
246
|
+
cronExpression: cronExpressions.budgetReset,
|
|
247
|
+
timezone: options.timezone,
|
|
248
|
+
waitForCompletion: true,
|
|
249
|
+
run: async () => {
|
|
250
|
+
await options.runner.resetBudget();
|
|
251
|
+
},
|
|
252
|
+
});
|
|
253
|
+
}
|
|
254
|
+
//# sourceMappingURL=process-check-job.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"process-check-job.js","sourceRoot":"","sources":["../../src/reliability/process-check-job.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,MAAM,kBAAkB,CAAC;AAClC,OAAO,IAAI,MAAM,WAAW,CAAC;AAC7B,OAAO,EAAE,iBAAiB,EAAE,MAAM,cAAc,CAAC;AACjD,OAAO,EAAE,WAAW,EAAE,kBAAkB,EAAmB,MAAM,gBAAgB,CAAC;AAClF,OAAO,EACL,2BAA2B,GAG5B,MAAM,oBAAoB,CAAC;AAiF5B,MAAM,wBAAwB,GAAmC;IAC/D,uBAAuB,EAAE,gBAAgB;IACzC,gBAAgB,EAAE,eAAe;IACjC,WAAW,EAAE,WAAW;CACzB,CAAC;AAEF,MAAM,qBAAqB,GAAG,EAAE,GAAG,EAAE,GAAG,IAAI,CAAC;AAC7C,MAAM,kBAAkB,GAAG,IAAI,CAAC,OAAO,CAAC,OAAO,CAAC,GAAG,EAAE,EAAE,MAAM,EAAE,yBAAyB,CAAC,CAAC;AAC1F,MAAM,qBAAqB,GAAG,IAAI,GAAG,EAAU,CAAC;AAEhD,SAAS,gBAAgB,CAAC,KAAc;IACtC,OAAO,OAAO,KAAK,KAAK,QAAQ,IAAI,KAAK,KAAK,IAAI,IAAI,MAAM,IAAI,KAAK,CAAC;AACxE,CAAC;AAED,MAAM,UAAU,uBAAuB,CAAC,SAAiB,EAAE,MAAoB,IAAI,CAAC,GAAG;IACrF,MAAM,aAAa,GAAG,MAAM,CAAC,QAAQ,CAAC,SAAS,CAAC,IAAI,SAAS,GAAG,CAAC;QAC/D,CAAC,CAAC,IAAI,CAAC,KAAK,CAAC,SAAS,CAAC;QACvB,CAAC,CAAC,iBAAiB,CAAC,YAAY,CAAC;IACnC,MAAM,EAAE,GAAG,GAAG,EAAE,CAAC;IACjB,OAAO;QACL,SAAS,EAAE,aAAa;QACxB,SAAS,EAAE,aAAa;QACxB,eAAe,EAAE,EAAE;QACnB,aAAa,EAAE,EAAE;KAClB,CAAC;AACJ,CAAC;AAED,MAAM,iBAAiB,GAAG,GAA+B,EAAE;IACzD,MAAM,MAAM,GAAG,WAAW,CAAC,kBAAkB,EAAE,IAAI,CAAC,CAAC;IACrD,OAAO,KAAK,EAAC,KAAK,EAAC,EAAE;QACnB,MAAM,UAAU,GAAe;YAC7B,UAAU,EAAE,kBAAkB,EAAE;YAChC,cAAc,EAAE,QAAQ;YACxB,QAAQ,EAAE,QAAQ;YAClB,MAAM,EAAE,KAAK,CAAC,MAAM;YACpB,MAAM,EAAE,KAAK,CAAC,MAAM;YACpB,SAAS,EAAE,IAAI,IAAI,EAAE,CAAC,WAAW,EAAE;YACnC,GAAG,CAAC,KAAK,CAAC,QAAQ,CAAC,CAAC,CAAC,EAAE,QAAQ,EAAE,KAAK,CAAC,QAAQ,EAAE,CAAC,CAAC,CAAC,EAAE,CAAC;SACxD,CAAC;QACF,MAAM,MAAM,CAAC,GAAG,CAAC,UAAU,CAAC,CAAC;IAC/B,CAAC,CAAC;AACJ,CAAC,CAAC;AAEF,KAAK,UAAU,WAAW,CAAC,WAAmB;IAC5C,IAAI,CAAC;QACH,MAAM,GAAG,GAAG,CAAC,MAAM,EAAE,CAAC,QAAQ,CAAC,WAAW,EAAE,OAAO,CAAC,CAAC,CAAC,IAAI,EAAE,CAAC;QAC7D,MAAM,GAAG,GAAG,MAAM,CAAC,QAAQ,CAAC,GAAG,EAAE,EAAE,CAAC,CAAC;QACrC,OAAO,MAAM,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC,CAAC,CAAC,IAAI,CAAC,CAAC,CAAC,GAAG,CAAC;IACxC,CAAC;IAAC,OAAO,KAAK,EAAE,CAAC;QACf,IAAI,gBAAgB,CAAC,KAAK,CAAC,IAAI,KAAK,CAAC,IAAI,KAAK,QAAQ,EAAE,CAAC;YACvD,IAAI,CAAC,qBAAqB,CAAC,GAAG,CAAC,WAAW,CAAC,EAAE,CAAC;gBAC5C,qBAAqB,CAAC,GAAG,CAAC,WAAW,CAAC,CAAC;gBACvC,OAAO,CAAC,IAAI,CAAC,yFAAyF,WAAW,EAAE,CAAC,CAAC;YACvH,CAAC;YACD,OAAO,IAAI,CAAC;QACd,CAAC;QAED,OAAO,CAAC,KAAK,CAAC,yCAAyC,EAAE,KAAK,YAAY,KAAK,CAAC,CAAC,CAAC,KAAK,CAAC,OAAO,CAAC,CAAC,CAAC,MAAM,CAAC,KAAK,CAAC,CAAC,CAAC;QACjH,OAAO,IAAI,CAAC;IACd,CAAC;AACH,CAAC;AAED,KAAK,UAAU,kBAAkB,CAAC,QAAgB;IAChD,IAAI,CAAC;QACH,MAAM,EAAE,CAAC,EAAE,CAAC,QAAQ,EAAE,EAAE,KAAK,EAAE,IAAI,EAAE,CAAC,CAAC;QACvC,OAAO,IAAI,CAAC;IACd,CAAC;IAAC,MAAM,CAAC;QACP,OAAO,KAAK,CAAC;IACf,CAAC;AACH,CAAC;AAED,SAAS,eAAe,CAAC,UAAkB;IACzC,OAAO,UAAU,CAAC,QAAQ,CAAC,OAAO,CAAC,CAAC,CAAC,CAAC,UAAU,CAAC,CAAC,CAAC,GAAG,UAAU,OAAO,CAAC;AAC1E,CAAC;AAED,MAAM,UAAU,2BAA2B,CAAC,OAAqC;IAC/E,MAAM,MAAM,GAAG,OAAO,CAAC,GAAG,IAAI,IAAI,CAAC,GAAG,CAAC;IACvC,MAAM,WAAW,GAAG,OAAO,CAAC,WAAW,IAAI,qBAAqB,CAAC;IACjE,MAAM,cAAc,GAAG,OAAO,CAAC,cAAc,IAAI,CAAC,KAAK,EAAC,GAAG,EAAC,EAAE;QAC5D,IAAI,CAAC;YACH,OAAO,CAAC,IAAI,CAAC,GAAG,EAAE,CAAC,CAAC,CAAC;YACrB,OAAO,IAAI,CAAC;QACd,CAAC;QAAC,MAAM,CAAC;YACP,OAAO,KAAK,CAAC;QACf,CAAC;IACH,CAAC,CAAC,CAAC;IACH,MAAM,mBAAmB,GAAG,OAAO,CAAC,2BAA2B,IAAI,2BAA2B,CAAC;IAC/F,MAAM,QAAQ,GAAG,OAAO,CAAC,QAAQ,IAAI,iBAAiB,EAAE,CAAC;IAEzD,OAAO;QACL,uBAAuB,EAAE,KAAK,IAAuC,EAAE;YACrE,MAAM,QAAQ,GAAG,MAAM,mBAAmB,CAAC;gBACzC,WAAW,EAAE,OAAO,CAAC,mBAAmB;gBACxC,IAAI,EAAE,OAAO,CAAC,YAAY;gBAC1B,IAAI,EAAE,OAAO,CAAC,YAAY;gBAC1B,SAAS,EAAE,OAAO,CAAC,sBAAsB;gBACzC,eAAe,EAAE,OAAO,CAAC,eAAe;aACzC,CAAC,CAAC;YAEH,IAAI,YAAY,GAAwB,aAAa,CAAC;YACtD,IAAI,eAAe,GAAG,KAAK,CAAC;YAC5B,MAAM,SAAS,GAAG,MAAM,WAAW,CAAC,OAAO,CAAC,iBAAiB,CAAC,CAAC;YAE/D,IAAI,SAAS,KAAK,IAAI,EAAE,CAAC;gBACvB,6CAA6C;gBAC7C,YAAY,GAAG,IAAI,CAAC;YACtB,CAAC;iBAAM,CAAC;gBACN,MAAM,KAAK,GAAG,MAAM,cAAc,CAAC,SAAS,CAAC,CAAC;gBAC9C,IAAI,KAAK,EAAE,CAAC;oBACV,YAAY,GAAG,IAAI,CAAC;gBACtB,CAAC;qBAAM,CAAC;oBACN,YAAY,GAAG,WAAW,CAAC;oBAC3B,eAAe,GAAG,MAAM,kBAAkB,CAAC,OAAO,CAAC,iBAAiB,CAAC,CAAC;oBACtE,MAAM,QAAQ,CAAC;wBACb,MAAM,EAAE,oBAAoB;wBAC5B,MAAM,EAAE,eAAe,CAAC,CAAC,CAAC,SAAS,CAAC,CAAC,CAAC,QAAQ;wBAC9C,QAAQ,EAAE;4BACR,WAAW,EAAE,OAAO,CAAC,iBAAiB;4BACtC,GAAG,EAAE,SAAS;yBACf;qBACF,CAAC,CAAC;gBACL,CAAC;YACH,CAAC;YAED,MAAM,OAAO,GAAG;gBACd,YAAY,QAAQ,CAAC,MAAM,EAAE;gBAC7B,UAAU,YAAY,EAAE;gBACxB,eAAe,QAAQ,CAAC,WAAW,CAAC,IAAI,CAAC,GAAG,CAAC,IAAI,MAAM,EAAE;aAC1D,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC;YAEZ,MAAM,QAAQ,CAAC;gBACb,MAAM,EAAE,2BAA2B;gBACnC,MAAM,EAAE,SAAS;gBACjB,QAAQ,EAAE;oBACR,cAAc,EAAE,QAAQ,CAAC,MAAM;oBAC/B,YAAY;oBACZ,eAAe;oBACf,mBAAmB,EAAE,QAAQ,CAAC,WAAW;oBACzC,oBAAoB,EAAE,QAAQ,CAAC,YAAY;oBAC3C,mBAAmB,EAAE,QAAQ,CAAC,WAAW;iBAC1C;aACF,CAAC,CAAC;YAEH,IAAI,QAAQ,CAAC,MAAM,KAAK,2BAA2B,EAAE,CAAC;gBACpD,MAAM,QAAQ,CAAC;oBACb,MAAM,EAAE,oCAAoC;oBAC5C,MAAM,EAAE,QAAQ;oBAChB,QAAQ,EAAE;wBACR,WAAW,EAAE,QAAQ,CAAC,WAAW;wBACjC,WAAW,EAAE,QAAQ,CAAC,WAAW;wBACjC,YAAY,EAAE,QAAQ,CAAC,YAAY;qBACpC;iBACF,CAAC,CAAC;YACL,CAAC;YAED,OAAO;gBACL,cAAc,EAAE,QAAQ,CAAC,MAAM;gBAC/B,YAAY;gBACZ,eAAe;gBACf,OAAO;aACR,CAAC;QACJ,CAAC;QAED,iBAAiB,EAAE,KAAK,IAAqC,EAAE;YAC7D,MAAM,YAAY,GAAa,EAAE,CAAC;YAClC,MAAM,YAAY,GAAa,EAAE,CAAC;YAElC,KAAK,MAAM,cAAc,IAAI,OAAO,CAAC,cAAc,EAAE,CAAC;gBACpD,MAAM,QAAQ,GAAG,eAAe,CAAC,cAAc,CAAC,CAAC;gBACjD,IAAI,CAAC;oBACH,MAAM,IAAI,GAAG,MAAM,EAAE,CAAC,IAAI,CAAC,QAAQ,CAAC,CAAC;oBACrC,MAAM,KAAK,GAAG,MAAM,EAAE,GAAG,IAAI,CAAC,OAAO,CAAC;oBACtC,IAAI,KAAK,IAAI,WAAW,EAAE,CAAC;wBACzB,YAAY,CAAC,IAAI,CAAC,QAAQ,CAAC,CAAC;wBAC5B,SAAS;oBACX,CAAC;oBAED,MAAM,EAAE,CAAC,EAAE,CAAC,QAAQ,EAAE,EAAE,SAAS,EAAE,IAAI,EAAE,KAAK,EAAE,IAAI,EAAE,CAAC,CAAC;oBACxD,YAAY,CAAC,IAAI,CAAC,QAAQ,CAAC,CAAC;oBAC5B,MAAM,QAAQ,CAAC;wBACb,MAAM,EAAE,oBAAoB;wBAC5B,MAAM,EAAE,SAAS;wBACjB,QAAQ,EAAE;4BACR,QAAQ;4BACR,KAAK;4BACL,WAAW;yBACZ;qBACF,CAAC,CAAC;gBACL,CAAC;gBAAC,OAAO,KAAK,EAAE,CAAC;oBACf,OAAO,CAAC,KAAK,CAAC,+CAA+C,EAAE,KAAK,YAAY,KAAK,CAAC,CAAC,CAAC,KAAK,CAAC,OAAO,CAAC,CAAC,CAAC,MAAM,CAAC,KAAK,CAAC,CAAC,CAAC;oBACvH,YAAY,CAAC,IAAI,CAAC,QAAQ,CAAC,CAAC;gBAC9B,CAAC;YACH,CAAC;YAED,MAAM,QAAQ,CAAC;gBACb,MAAM,EAAE,wBAAwB;gBAChC,MAAM,EAAE,SAAS;gBACjB,QAAQ,EAAE;oBACR,YAAY,EAAE,YAAY,CAAC,MAAM;oBACjC,YAAY,EAAE,YAAY,CAAC,MAAM;iBAClC;aACF,CAAC,CAAC;YAEH,OAAO;gBACL,YAAY;gBACZ,YAAY;aACb,CAAC;QACJ,CAAC;QAED,WAAW,EAAE,KAAK,IAAgC,EAAE;YAClD,MAAM,QAAQ,GAAG,OAAO,CAAC,iBAAiB,CAAC,SAAS,CAAC;YACrD,OAAO,CAAC,iBAAiB,CAAC,SAAS,GAAG,OAAO,CAAC,iBAAiB,CAAC,SAAS,CAAC;YAC1E,OAAO,CAAC,iBAAiB,CAAC,aAAa,GAAG,MAAM,EAAE,CAAC;YACnD,OAAO,CAAC,iBAAiB,CAAC,eAAe,GAAG,OAAO,CAAC,iBAAiB,CAAC,aAAa,CAAC;YAEpF,MAAM,QAAQ,CAAC;gBACb,MAAM,EAAE,qBAAqB;gBAC7B,MAAM,EAAE,SAAS;gBACjB,QAAQ,EAAE;oBACR,QAAQ;oBACR,OAAO,EAAE,OAAO,CAAC,iBAAiB,CAAC,SAAS;oBAC5C,SAAS,EAAE,OAAO,CAAC,iBAAiB,CAAC,SAAS;iBAC/C;aACF,CAAC,CAAC;YAEH,OAAO;gBACL,QAAQ;gBACR,OAAO,EAAE,OAAO,CAAC,iBAAiB,CAAC,SAAS;gBAC5C,SAAS,EAAE,OAAO,CAAC,iBAAiB,CAAC,SAAS;aAC/C,CAAC;QACJ,CAAC;KACF,CAAC;AACJ,CAAC;AAED,MAAM,UAAU,wBAAwB,CACtC,SAAwB,EACxB,OAAwC;IAExC,MAAM,eAAe,GAAmC;QACtD,GAAG,wBAAwB;QAC3B,GAAG,OAAO,CAAC,eAAe;KAC3B,CAAC;IAEF,SAAS,CAAC,WAAW,CAAC;QACpB,EAAE,EAAE,2BAA2B;QAC/B,cAAc,EAAE,eAAe,CAAC,uBAAuB;QACvD,QAAQ,EAAE,OAAO,CAAC,QAAQ;QAC1B,iBAAiB,EAAE,IAAI;QACvB,GAAG,EAAE,KAAK,IAAI,EAAE;YACd,MAAM,OAAO,CAAC,MAAM,CAAC,uBAAuB,EAAE,CAAC;QACjD,CAAC;KACF,CAAC,CAAC;IAEH,SAAS,CAAC,WAAW,CAAC;QACpB,EAAE,EAAE,oBAAoB;QACxB,cAAc,EAAE,eAAe,CAAC,gBAAgB;QAChD,QAAQ,EAAE,OAAO,CAAC,QAAQ;QAC1B,iBAAiB,EAAE,IAAI;QACvB,GAAG,EAAE,KAAK,IAAI,EAAE;YACd,MAAM,OAAO,CAAC,MAAM,CAAC,iBAAiB,EAAE,CAAC;QAC3C,CAAC;KACF,CAAC,CAAC;IAEH,SAAS,CAAC,WAAW,CAAC;QACpB,EAAE,EAAE,cAAc;QAClB,cAAc,EAAE,eAAe,CAAC,WAAW;QAC3C,QAAQ,EAAE,OAAO,CAAC,QAAQ;QAC1B,iBAAiB,EAAE,IAAI;QACvB,GAAG,EAAE,KAAK,IAAI,EAAE;YACd,MAAM,OAAO,CAAC,MAAM,CAAC,WAAW,EAAE,CAAC;QACrC,CAAC;KACF,CAAC,CAAC;AACL,CAAC"}
|
|
@@ -0,0 +1,53 @@
|
|
|
1
|
+
export interface OpenCodeProcessInfo {
|
|
2
|
+
pid: number;
|
|
3
|
+
command: string;
|
|
4
|
+
}
|
|
5
|
+
type ProcessAliveChecker = (pid: number) => Promise<boolean>;
|
|
6
|
+
type ProcessListProvider = () => Promise<OpenCodeProcessInfo[]>;
|
|
7
|
+
type PortProbe = (host: string, port: number, timeoutMs: number) => Promise<PortProbeResult>;
|
|
8
|
+
export interface PortProbeResult {
|
|
9
|
+
isOpen: boolean;
|
|
10
|
+
reason: string;
|
|
11
|
+
}
|
|
12
|
+
export interface ProcessGuardOptions {
|
|
13
|
+
pidFilePath: string;
|
|
14
|
+
host: string;
|
|
15
|
+
port: number;
|
|
16
|
+
timeoutMs?: number;
|
|
17
|
+
processKeywords?: string[];
|
|
18
|
+
processAliveChecker?: ProcessAliveChecker;
|
|
19
|
+
processListProvider?: ProcessListProvider;
|
|
20
|
+
portProbe?: PortProbe;
|
|
21
|
+
}
|
|
22
|
+
export type ProcessGuardStatus = 'ok' | 'not-running' | 'single-instance-violation';
|
|
23
|
+
export interface ProcessGuardResult {
|
|
24
|
+
status: ProcessGuardStatus;
|
|
25
|
+
pidFromFile: number | null;
|
|
26
|
+
pidAlive: boolean;
|
|
27
|
+
portOpen: boolean;
|
|
28
|
+
probeReason: string;
|
|
29
|
+
runningPids: number[];
|
|
30
|
+
conflictPids: number[];
|
|
31
|
+
}
|
|
32
|
+
export interface RescueLockOptions {
|
|
33
|
+
lockTargetPath: string;
|
|
34
|
+
staleMs?: number;
|
|
35
|
+
updateMs?: number;
|
|
36
|
+
now?: () => number;
|
|
37
|
+
}
|
|
38
|
+
export interface RescueLockBusyResult {
|
|
39
|
+
ok: false;
|
|
40
|
+
code: 'lock-busy';
|
|
41
|
+
lockPath: string;
|
|
42
|
+
}
|
|
43
|
+
export interface RescueLockSuccessResult {
|
|
44
|
+
ok: true;
|
|
45
|
+
lockPath: string;
|
|
46
|
+
release: () => Promise<void>;
|
|
47
|
+
}
|
|
48
|
+
export type RescueLockAcquireResult = RescueLockBusyResult | RescueLockSuccessResult;
|
|
49
|
+
export declare function checkOpenCodeSingleInstance(options: ProcessGuardOptions): Promise<ProcessGuardResult>;
|
|
50
|
+
export declare function acquireRescueLock(options: RescueLockOptions): Promise<RescueLockAcquireResult>;
|
|
51
|
+
export declare function probeTcpPort(host: string, port: number, timeoutMs?: number): Promise<PortProbeResult>;
|
|
52
|
+
export {};
|
|
53
|
+
//# sourceMappingURL=process-guard.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"process-guard.d.ts","sourceRoot":"","sources":["../../src/reliability/process-guard.ts"],"names":[],"mappings":"AASA,MAAM,WAAW,mBAAmB;IAClC,GAAG,EAAE,MAAM,CAAC;IACZ,OAAO,EAAE,MAAM,CAAC;CACjB;AAED,KAAK,mBAAmB,GAAG,CAAC,GAAG,EAAE,MAAM,KAAK,OAAO,CAAC,OAAO,CAAC,CAAC;AAC7D,KAAK,mBAAmB,GAAG,MAAM,OAAO,CAAC,mBAAmB,EAAE,CAAC,CAAC;AAChE,KAAK,SAAS,GAAG,CAAC,IAAI,EAAE,MAAM,EAAE,IAAI,EAAE,MAAM,EAAE,SAAS,EAAE,MAAM,KAAK,OAAO,CAAC,eAAe,CAAC,CAAC;AAE7F,MAAM,WAAW,eAAe;IAC9B,MAAM,EAAE,OAAO,CAAC;IAChB,MAAM,EAAE,MAAM,CAAC;CAChB;AAED,MAAM,WAAW,mBAAmB;IAClC,WAAW,EAAE,MAAM,CAAC;IACpB,IAAI,EAAE,MAAM,CAAC;IACb,IAAI,EAAE,MAAM,CAAC;IACb,SAAS,CAAC,EAAE,MAAM,CAAC;IACnB,eAAe,CAAC,EAAE,MAAM,EAAE,CAAC;IAC3B,mBAAmB,CAAC,EAAE,mBAAmB,CAAC;IAC1C,mBAAmB,CAAC,EAAE,mBAAmB,CAAC;IAC1C,SAAS,CAAC,EAAE,SAAS,CAAC;CACvB;AAED,MAAM,MAAM,kBAAkB,GAAG,IAAI,GAAG,aAAa,GAAG,2BAA2B,CAAC;AAEpF,MAAM,WAAW,kBAAkB;IACjC,MAAM,EAAE,kBAAkB,CAAC;IAC3B,WAAW,EAAE,MAAM,GAAG,IAAI,CAAC;IAC3B,QAAQ,EAAE,OAAO,CAAC;IAClB,QAAQ,EAAE,OAAO,CAAC;IAClB,WAAW,EAAE,MAAM,CAAC;IACpB,WAAW,EAAE,MAAM,EAAE,CAAC;IACtB,YAAY,EAAE,MAAM,EAAE,CAAC;CACxB;AAED,MAAM,WAAW,iBAAiB;IAChC,cAAc,EAAE,MAAM,CAAC;IACvB,OAAO,CAAC,EAAE,MAAM,CAAC;IACjB,QAAQ,CAAC,EAAE,MAAM,CAAC;IAClB,GAAG,CAAC,EAAE,MAAM,MAAM,CAAC;CACpB;AAED,MAAM,WAAW,oBAAoB;IACnC,EAAE,EAAE,KAAK,CAAC;IACV,IAAI,EAAE,WAAW,CAAC;IAClB,QAAQ,EAAE,MAAM,CAAC;CAClB;AAED,MAAM,WAAW,uBAAuB;IACtC,EAAE,EAAE,IAAI,CAAC;IACT,QAAQ,EAAE,MAAM,CAAC;IACjB,OAAO,EAAE,MAAM,OAAO,CAAC,IAAI,CAAC,CAAC;CAC9B;AAED,MAAM,MAAM,uBAAuB,GAAG,oBAAoB,GAAG,uBAAuB,CAAC;AAErF,wBAAsB,2BAA2B,CAC/C,OAAO,EAAE,mBAAmB,GAC3B,OAAO,CAAC,kBAAkB,CAAC,CAyD7B;AAED,wBAAsB,iBAAiB,CAAC,OAAO,EAAE,iBAAiB,GAAG,OAAO,CAAC,uBAAuB,CAAC,CAiDpG;AAwED,wBAAsB,YAAY,CAChC,IAAI,EAAE,MAAM,EACZ,IAAI,EAAE,MAAM,EACZ,SAAS,SAAO,GACf,OAAO,CAAC,eAAe,CAAC,CA6B1B"}
|