@caoruhua/open-claude-remote 0.1.7 → 0.2.4
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 +534 -11
- package/dist/backend/src/api/config-routes.d.ts +2 -2
- package/dist/backend/src/api/config-routes.d.ts.map +1 -1
- package/dist/backend/src/api/config-routes.js +24 -6
- package/dist/backend/src/api/config-routes.js.map +1 -1
- package/dist/backend/src/api/health-routes.d.ts +4 -1
- package/dist/backend/src/api/health-routes.d.ts.map +1 -1
- package/dist/backend/src/api/health-routes.js +20 -2
- package/dist/backend/src/api/health-routes.js.map +1 -1
- package/dist/backend/src/api/hook-routes.d.ts +2 -2
- package/dist/backend/src/api/hook-routes.d.ts.map +1 -1
- package/dist/backend/src/api/hook-routes.js +10 -3
- package/dist/backend/src/api/hook-routes.js.map +1 -1
- package/dist/backend/src/api/instance-routes.d.ts +5 -3
- package/dist/backend/src/api/instance-routes.d.ts.map +1 -1
- package/dist/backend/src/api/instance-routes.js +48 -45
- package/dist/backend/src/api/instance-routes.js.map +1 -1
- package/dist/backend/src/api/router.d.ts +4 -12
- package/dist/backend/src/api/router.d.ts.map +1 -1
- package/dist/backend/src/api/router.js +28 -20
- package/dist/backend/src/api/router.js.map +1 -1
- package/dist/backend/src/api/status-routes.d.ts +2 -2
- package/dist/backend/src/api/status-routes.d.ts.map +1 -1
- package/dist/backend/src/api/status-routes.js +11 -6
- package/dist/backend/src/api/status-routes.js.map +1 -1
- package/dist/backend/src/attach.d.ts +1 -1
- package/dist/backend/src/attach.d.ts.map +1 -1
- package/dist/backend/src/attach.js +25 -71
- package/dist/backend/src/attach.js.map +1 -1
- package/dist/backend/src/cli-utils.d.ts +24 -1
- package/dist/backend/src/cli-utils.d.ts.map +1 -1
- package/dist/backend/src/cli-utils.js +73 -37
- package/dist/backend/src/cli-utils.js.map +1 -1
- package/dist/backend/src/cli.d.ts +23 -2
- package/dist/backend/src/cli.d.ts.map +1 -1
- package/dist/backend/src/cli.js +359 -22
- package/dist/backend/src/cli.js.map +1 -1
- package/dist/backend/src/config.d.ts +48 -15
- package/dist/backend/src/config.d.ts.map +1 -1
- package/dist/backend/src/config.js +124 -24
- package/dist/backend/src/config.js.map +1 -1
- package/dist/backend/src/daemon/daemon-client.d.ts +91 -0
- package/dist/backend/src/daemon/daemon-client.d.ts.map +1 -0
- package/dist/backend/src/daemon/daemon-client.js +258 -0
- package/dist/backend/src/daemon/daemon-client.js.map +1 -0
- package/dist/backend/src/daemon/daemon-entry.d.ts +8 -0
- package/dist/backend/src/daemon/daemon-entry.d.ts.map +1 -0
- package/dist/backend/src/daemon/daemon-entry.js +37 -0
- package/dist/backend/src/daemon/daemon-entry.js.map +1 -0
- package/dist/backend/src/daemon/daemon-launcher.d.ts +13 -0
- package/dist/backend/src/daemon/daemon-launcher.d.ts.map +1 -0
- package/dist/backend/src/daemon/daemon-launcher.js +62 -0
- package/dist/backend/src/daemon/daemon-launcher.js.map +1 -0
- package/dist/backend/src/index.d.ts.map +1 -1
- package/dist/backend/src/index.js +115 -251
- package/dist/backend/src/index.js.map +1 -1
- package/dist/backend/src/instance/index.d.ts +4 -0
- package/dist/backend/src/instance/index.d.ts.map +1 -0
- package/dist/backend/src/instance/index.js +3 -0
- package/dist/backend/src/instance/index.js.map +1 -0
- package/dist/backend/src/instance/instance-manager.d.ts +62 -0
- package/dist/backend/src/instance/instance-manager.d.ts.map +1 -0
- package/dist/backend/src/instance/instance-manager.js +194 -0
- package/dist/backend/src/instance/instance-manager.js.map +1 -0
- package/dist/backend/src/instance/instance-session.d.ts +87 -0
- package/dist/backend/src/instance/instance-session.d.ts.map +1 -0
- package/dist/backend/src/instance/instance-session.js +359 -0
- package/dist/backend/src/instance/instance-session.js.map +1 -0
- package/dist/backend/src/instance/types.d.ts +39 -0
- package/dist/backend/src/instance/types.d.ts.map +1 -0
- package/dist/backend/src/instance/types.js +2 -0
- package/dist/backend/src/instance/types.js.map +1 -0
- package/dist/backend/src/notification/notification-manager.js +1 -1
- package/dist/backend/src/notification/notification-manager.js.map +1 -1
- package/dist/backend/src/notification/notification-service-factory.js +1 -1
- package/dist/backend/src/notification/notification-service-factory.js.map +1 -1
- package/dist/backend/src/pty/virtual-pty.d.ts.map +1 -1
- package/dist/backend/src/pty/virtual-pty.js +6 -0
- package/dist/backend/src/pty/virtual-pty.js.map +1 -1
- package/dist/backend/src/registry/shared-token.d.ts +2 -2
- package/dist/backend/src/registry/shared-token.js +11 -11
- package/dist/backend/src/registry/shared-token.js.map +1 -1
- package/dist/backend/src/registry/stop-instances.d.ts +0 -25
- package/dist/backend/src/registry/stop-instances.d.ts.map +1 -1
- package/dist/backend/src/registry/stop-instances.js +6 -206
- package/dist/backend/src/registry/stop-instances.js.map +1 -1
- package/dist/backend/src/skills/index.d.ts +4 -0
- package/dist/backend/src/skills/index.d.ts.map +1 -0
- package/dist/backend/src/skills/index.js +4 -0
- package/dist/backend/src/skills/index.js.map +1 -0
- package/dist/backend/src/skills/skill-command-merger.d.ts +32 -0
- package/dist/backend/src/skills/skill-command-merger.d.ts.map +1 -0
- package/dist/backend/src/skills/skill-command-merger.js +76 -0
- package/dist/backend/src/skills/skill-command-merger.js.map +1 -0
- package/dist/backend/src/skills/skill-commands.d.ts +20 -0
- package/dist/backend/src/skills/skill-commands.d.ts.map +1 -0
- package/dist/backend/src/skills/skill-commands.js +34 -0
- package/dist/backend/src/skills/skill-commands.js.map +1 -0
- package/dist/backend/src/skills/skill-scanner.d.ts +40 -0
- package/dist/backend/src/skills/skill-scanner.d.ts.map +1 -0
- package/dist/backend/src/skills/skill-scanner.js +160 -0
- package/dist/backend/src/skills/skill-scanner.js.map +1 -0
- package/dist/backend/src/terminal/terminal-relay.d.ts +1 -0
- package/dist/backend/src/terminal/terminal-relay.d.ts.map +1 -1
- package/dist/backend/src/terminal/terminal-relay.js +6 -0
- package/dist/backend/src/terminal/terminal-relay.js.map +1 -1
- package/dist/backend/src/update.d.ts.map +1 -1
- package/dist/backend/src/update.js +46 -1
- package/dist/backend/src/update.js.map +1 -1
- package/dist/backend/src/utils/banner.d.ts +15 -0
- package/dist/backend/src/utils/banner.d.ts.map +1 -0
- package/dist/backend/src/utils/banner.js +95 -0
- package/dist/backend/src/utils/banner.js.map +1 -0
- package/dist/backend/src/ws/ws-server.d.ts +13 -54
- package/dist/backend/src/ws/ws-server.d.ts.map +1 -1
- package/dist/backend/src/ws/ws-server.js +57 -155
- package/dist/backend/src/ws/ws-server.js.map +1 -1
- package/dist/shared/constants.d.ts +2 -1
- package/dist/shared/constants.d.ts.map +1 -1
- package/dist/shared/constants.js +2 -1
- package/dist/shared/constants.js.map +1 -1
- package/dist/shared/defaults.d.ts +1 -0
- package/dist/shared/defaults.d.ts.map +1 -1
- package/dist/shared/defaults.js +1 -5
- package/dist/shared/defaults.js.map +1 -1
- package/dist/shared/instance.d.ts +4 -9
- package/dist/shared/instance.d.ts.map +1 -1
- package/dist/shared/instance.js +0 -2
- package/dist/shared/instance.js.map +1 -1
- package/frontend-dist/assets/index-CM2xfmS8.js +152 -0
- package/frontend-dist/index.html +1 -1
- package/package.json +2 -2
- package/frontend-dist/assets/index-Cfhr3h3e.js +0 -152
|
@@ -0,0 +1,194 @@
|
|
|
1
|
+
import { EventEmitter } from 'node:events';
|
|
2
|
+
import { randomUUID } from 'node:crypto';
|
|
3
|
+
import { basename, resolve } from 'node:path';
|
|
4
|
+
import { homedir } from 'node:os';
|
|
5
|
+
import { CLAUDE_REMOTE_DIR, DEFAULT_PORT } from '../../../shared/index.js';
|
|
6
|
+
import { InstanceSession } from './instance-session.js';
|
|
7
|
+
import { createClaudeSettings, extractSettingsFromArgs, saveClaudeSettings, loadWorkdirConfig, mergeConfigs, loadUserConfig } from '../config.js';
|
|
8
|
+
import { logger } from '../logger/logger.js';
|
|
9
|
+
/**
|
|
10
|
+
* InstanceManager: 管理所有 InstanceSession 的中央协调器
|
|
11
|
+
* 单进程内通过 Map 管理,无需文件注册表
|
|
12
|
+
*/
|
|
13
|
+
export class InstanceManager extends EventEmitter {
|
|
14
|
+
instances = new Map();
|
|
15
|
+
pushService = null;
|
|
16
|
+
notificationManager = null;
|
|
17
|
+
notificationServiceFactory = null;
|
|
18
|
+
displayIp = '127.0.0.1';
|
|
19
|
+
/**
|
|
20
|
+
* 注入全局共享的通知服务
|
|
21
|
+
*/
|
|
22
|
+
setSharedServices(options) {
|
|
23
|
+
if (options.pushService)
|
|
24
|
+
this.pushService = options.pushService;
|
|
25
|
+
if (options.notificationManager)
|
|
26
|
+
this.notificationManager = options.notificationManager;
|
|
27
|
+
if (options.notificationServiceFactory)
|
|
28
|
+
this.notificationServiceFactory = options.notificationServiceFactory;
|
|
29
|
+
if (options.displayIp)
|
|
30
|
+
this.displayIp = options.displayIp;
|
|
31
|
+
}
|
|
32
|
+
/**
|
|
33
|
+
* 创建新实例(进程内创建 PTY,不再 spawn 子进程)
|
|
34
|
+
*/
|
|
35
|
+
createInstance(options) {
|
|
36
|
+
const instanceId = randomUUID();
|
|
37
|
+
const name = options.name || basename(options.cwd);
|
|
38
|
+
// 加载工作路径配置
|
|
39
|
+
const globalConfig = loadUserConfig();
|
|
40
|
+
const workdirConfig = loadWorkdirConfig(options.cwd);
|
|
41
|
+
const mergedConfig = mergeConfigs(globalConfig, workdirConfig);
|
|
42
|
+
const maxBufferLines = options.maxBufferLines ?? mergedConfig.maxBufferLines ?? 10000;
|
|
43
|
+
const claudeCommand = options.claudeCommand ?? mergedConfig.claudeCommand ?? 'claude';
|
|
44
|
+
// 合并 claudeArgs
|
|
45
|
+
const mergedArgs = mergedConfig.claudeArgs ?? [];
|
|
46
|
+
const inputArgs = options.claudeArgs ?? [];
|
|
47
|
+
const finalClaudeArgs = inputArgs.length === 0 && options.claudeArgs !== undefined
|
|
48
|
+
? [] // 显式传空数组 = 清空
|
|
49
|
+
: Array.from(new Set([...mergedArgs, ...inputArgs]));
|
|
50
|
+
const sessionOptions = {
|
|
51
|
+
instanceId,
|
|
52
|
+
name,
|
|
53
|
+
cwd: options.cwd,
|
|
54
|
+
maxBufferLines,
|
|
55
|
+
headless: options.headless ?? false,
|
|
56
|
+
claudeArgs: finalClaudeArgs,
|
|
57
|
+
};
|
|
58
|
+
const session = new InstanceSession(sessionOptions);
|
|
59
|
+
// 注入共享的通知服务
|
|
60
|
+
if (this.pushService)
|
|
61
|
+
session.setPushService(this.pushService);
|
|
62
|
+
if (this.notificationManager)
|
|
63
|
+
session.setNotificationManager(this.notificationManager);
|
|
64
|
+
if (this.notificationServiceFactory)
|
|
65
|
+
session.setNotificationServiceFactory(this.notificationServiceFactory);
|
|
66
|
+
// 设置实例 URL
|
|
67
|
+
const instanceUrl = `http://${this.displayIp}:${DEFAULT_PORT}`;
|
|
68
|
+
session.setInstanceUrl(instanceUrl);
|
|
69
|
+
// PTY 退出时自动移除
|
|
70
|
+
session.on('exit', () => {
|
|
71
|
+
this.removeInstance(instanceId, 'pty_exit');
|
|
72
|
+
});
|
|
73
|
+
// 生成 Claude settings 并 spawn PTY
|
|
74
|
+
const sharedConfigDir = resolve(homedir(), CLAUDE_REMOTE_DIR);
|
|
75
|
+
const extracted = extractSettingsFromArgs(finalClaudeArgs);
|
|
76
|
+
const finalSettings = createClaudeSettings(instanceId, extracted?.settingsValue);
|
|
77
|
+
const settingsPath = saveClaudeSettings(finalSettings, instanceId, sharedConfigDir);
|
|
78
|
+
let spawnArgs;
|
|
79
|
+
if (extracted) {
|
|
80
|
+
spawnArgs = [...extracted.otherArgs, '--settings', settingsPath];
|
|
81
|
+
}
|
|
82
|
+
else {
|
|
83
|
+
spawnArgs = [...finalClaudeArgs, '--settings', settingsPath];
|
|
84
|
+
}
|
|
85
|
+
session.ptyManager.spawn({
|
|
86
|
+
command: claudeCommand,
|
|
87
|
+
args: spawnArgs,
|
|
88
|
+
cwd: options.cwd,
|
|
89
|
+
});
|
|
90
|
+
session.setStatus('running');
|
|
91
|
+
// 注册到 Map
|
|
92
|
+
this.instances.set(instanceId, session);
|
|
93
|
+
logger.info({
|
|
94
|
+
instanceId,
|
|
95
|
+
name,
|
|
96
|
+
cwd: options.cwd,
|
|
97
|
+
headless: options.headless,
|
|
98
|
+
}, 'Instance created');
|
|
99
|
+
this.emit('instance_created', this.getInstanceInfo(session));
|
|
100
|
+
return session;
|
|
101
|
+
}
|
|
102
|
+
/**
|
|
103
|
+
* 获取实例
|
|
104
|
+
*/
|
|
105
|
+
getInstance(instanceId) {
|
|
106
|
+
return this.instances.get(instanceId);
|
|
107
|
+
}
|
|
108
|
+
/**
|
|
109
|
+
* 列出所有实例信息
|
|
110
|
+
*/
|
|
111
|
+
listInstances() {
|
|
112
|
+
const result = [];
|
|
113
|
+
for (const session of this.instances.values()) {
|
|
114
|
+
result.push(this.getInstanceInfo(session));
|
|
115
|
+
}
|
|
116
|
+
return result;
|
|
117
|
+
}
|
|
118
|
+
/**
|
|
119
|
+
* 销毁特定实例
|
|
120
|
+
*/
|
|
121
|
+
destroyInstance(instanceId) {
|
|
122
|
+
const session = this.instances.get(instanceId);
|
|
123
|
+
if (!session)
|
|
124
|
+
return false;
|
|
125
|
+
session.destroy();
|
|
126
|
+
this.removeInstance(instanceId, 'manual_destroy');
|
|
127
|
+
return true;
|
|
128
|
+
}
|
|
129
|
+
/**
|
|
130
|
+
* 销毁所有实例
|
|
131
|
+
*/
|
|
132
|
+
destroyAll() {
|
|
133
|
+
for (const [id, session] of this.instances) {
|
|
134
|
+
session.destroy();
|
|
135
|
+
logger.info({ instanceId: id }, 'Instance destroyed during shutdown');
|
|
136
|
+
}
|
|
137
|
+
this.instances.clear();
|
|
138
|
+
}
|
|
139
|
+
/**
|
|
140
|
+
* Ping 所有实例的客户端(由全局心跳定时器调用)
|
|
141
|
+
*/
|
|
142
|
+
pingAllClients() {
|
|
143
|
+
for (const session of this.instances.values()) {
|
|
144
|
+
session.pingClients();
|
|
145
|
+
}
|
|
146
|
+
}
|
|
147
|
+
/**
|
|
148
|
+
* 向所有实例的所有客户端广播消息
|
|
149
|
+
*/
|
|
150
|
+
broadcastAll(message) {
|
|
151
|
+
for (const session of this.instances.values()) {
|
|
152
|
+
session.broadcast(message);
|
|
153
|
+
}
|
|
154
|
+
}
|
|
155
|
+
/**
|
|
156
|
+
* 更新所有实例的 IP 地址
|
|
157
|
+
*/
|
|
158
|
+
updateDisplayIp(newIp) {
|
|
159
|
+
const oldIp = this.displayIp;
|
|
160
|
+
this.displayIp = newIp;
|
|
161
|
+
const newUrl = `http://${newIp}:${DEFAULT_PORT}`;
|
|
162
|
+
for (const session of this.instances.values()) {
|
|
163
|
+
session.setInstanceUrl(newUrl);
|
|
164
|
+
session.broadcast({
|
|
165
|
+
type: 'ip_changed',
|
|
166
|
+
oldIp,
|
|
167
|
+
newIp,
|
|
168
|
+
newUrl,
|
|
169
|
+
});
|
|
170
|
+
}
|
|
171
|
+
}
|
|
172
|
+
get size() {
|
|
173
|
+
return this.instances.size;
|
|
174
|
+
}
|
|
175
|
+
getInstanceInfo(session) {
|
|
176
|
+
return {
|
|
177
|
+
instanceId: session.instanceId,
|
|
178
|
+
name: session.name,
|
|
179
|
+
cwd: session.cwd,
|
|
180
|
+
status: session.status,
|
|
181
|
+
startedAt: session.startedAt,
|
|
182
|
+
headless: session.headless,
|
|
183
|
+
clientCount: session.clientCount,
|
|
184
|
+
claudeArgs: session.claudeArgs.length > 0 ? session.claudeArgs : undefined,
|
|
185
|
+
};
|
|
186
|
+
}
|
|
187
|
+
removeInstance(instanceId, reason) {
|
|
188
|
+
if (this.instances.delete(instanceId)) {
|
|
189
|
+
logger.info({ instanceId, reason }, 'Instance removed');
|
|
190
|
+
this.emit('instance_removed', instanceId, reason);
|
|
191
|
+
}
|
|
192
|
+
}
|
|
193
|
+
}
|
|
194
|
+
//# sourceMappingURL=instance-manager.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"instance-manager.js","sourceRoot":"","sources":["../../../../backend/src/instance/instance-manager.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,87 @@
|
|
|
1
|
+
import { EventEmitter } from 'node:events';
|
|
2
|
+
import { WebSocket } from 'ws';
|
|
3
|
+
import type { SessionStatus, ServerMessage } from '../../../shared/index.js';
|
|
4
|
+
import { PtyManager } from '../pty/pty-manager.js';
|
|
5
|
+
import { HookReceiver } from '../hooks/hook-receiver.js';
|
|
6
|
+
import type { PushService } from '../push/push-service.js';
|
|
7
|
+
import type { NotificationManager } from '../notification/notification-manager.js';
|
|
8
|
+
import type { NotificationServiceFactory } from '../notification/notification-service-factory.js';
|
|
9
|
+
/** 客户端类型:attach(从控端)或 webapp(主控端) */
|
|
10
|
+
export type ClientType = 'attach' | 'webapp';
|
|
11
|
+
export interface InstanceSessionOptions {
|
|
12
|
+
instanceId: string;
|
|
13
|
+
name: string;
|
|
14
|
+
cwd: string;
|
|
15
|
+
maxBufferLines: number;
|
|
16
|
+
headless: boolean;
|
|
17
|
+
claudeArgs?: string[];
|
|
18
|
+
/** 可选:注入自定义 PtyManager(用于测试) */
|
|
19
|
+
ptyManager?: PtyManager;
|
|
20
|
+
}
|
|
21
|
+
/**
|
|
22
|
+
* InstanceSession: 单个 Claude 实例的会话协调器
|
|
23
|
+
* 持有: PtyManager, OutputBuffer, HookReceiver, WS 客户端集合
|
|
24
|
+
* 复用 SessionController 的核心逻辑
|
|
25
|
+
*/
|
|
26
|
+
export declare class InstanceSession extends EventEmitter {
|
|
27
|
+
readonly instanceId: string;
|
|
28
|
+
readonly name: string;
|
|
29
|
+
readonly cwd: string;
|
|
30
|
+
readonly headless: boolean;
|
|
31
|
+
readonly claudeArgs: string[];
|
|
32
|
+
readonly startedAt: string;
|
|
33
|
+
readonly ptyManager: PtyManager;
|
|
34
|
+
readonly hookReceiver: HookReceiver;
|
|
35
|
+
private _status;
|
|
36
|
+
private buffer;
|
|
37
|
+
private clients;
|
|
38
|
+
private pushService;
|
|
39
|
+
private notificationServiceFactory;
|
|
40
|
+
private notificationManager;
|
|
41
|
+
private instanceUrl;
|
|
42
|
+
private wsPendingChunks;
|
|
43
|
+
private wsPendingBytes;
|
|
44
|
+
private wsFlushTimer;
|
|
45
|
+
constructor(options: InstanceSessionOptions);
|
|
46
|
+
get status(): SessionStatus;
|
|
47
|
+
get clientCount(): number;
|
|
48
|
+
setStatus(status: SessionStatus): void;
|
|
49
|
+
setPushService(pushService: PushService): void;
|
|
50
|
+
setNotificationServiceFactory(factory: NotificationServiceFactory): void;
|
|
51
|
+
setNotificationManager(manager: NotificationManager): void;
|
|
52
|
+
setInstanceUrl(url: string): void;
|
|
53
|
+
/**
|
|
54
|
+
* 注册一个 WS 客户端到此实例
|
|
55
|
+
*/
|
|
56
|
+
addClient(ws: WebSocket, clientType: ClientType): void;
|
|
57
|
+
/**
|
|
58
|
+
* 向所有客户端广播消息
|
|
59
|
+
*/
|
|
60
|
+
broadcast(message: ServerMessage): void;
|
|
61
|
+
/**
|
|
62
|
+
* 向特定客户端发送消息
|
|
63
|
+
*/
|
|
64
|
+
sendTo(ws: WebSocket, message: ServerMessage): void;
|
|
65
|
+
/**
|
|
66
|
+
* 获取客户端计数
|
|
67
|
+
*/
|
|
68
|
+
getClientCounts(): {
|
|
69
|
+
attach: number;
|
|
70
|
+
webapp: number;
|
|
71
|
+
};
|
|
72
|
+
/**
|
|
73
|
+
* Ping 所有客户端,终止无响应的
|
|
74
|
+
*/
|
|
75
|
+
pingClients(): void;
|
|
76
|
+
private setupPtyHandlers;
|
|
77
|
+
private enqueueWsOutput;
|
|
78
|
+
private flushPendingWsOutput;
|
|
79
|
+
private handleWsMessage;
|
|
80
|
+
private setupHookHandlers;
|
|
81
|
+
private sendNotificationByChannel;
|
|
82
|
+
/**
|
|
83
|
+
* 销毁实例:终止 PTY,断开所有客户端
|
|
84
|
+
*/
|
|
85
|
+
destroy(): void;
|
|
86
|
+
}
|
|
87
|
+
//# sourceMappingURL=instance-session.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"instance-session.d.ts","sourceRoot":"","sources":["../../../../backend/src/instance/instance-session.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,359 @@
|
|
|
1
|
+
import { EventEmitter } from 'node:events';
|
|
2
|
+
import { WebSocket } from 'ws';
|
|
3
|
+
import { PtyManager } from '../pty/pty-manager.js';
|
|
4
|
+
import { OutputBuffer } from '../pty/output-buffer.js';
|
|
5
|
+
import { HookReceiver } from '../hooks/hook-receiver.js';
|
|
6
|
+
import { logger } from '../logger/logger.js';
|
|
7
|
+
const WS_FLUSH_INTERVAL_MS = 16;
|
|
8
|
+
const WS_MAX_CHUNK_BYTES = 32 * 1024;
|
|
9
|
+
/**
|
|
10
|
+
* InstanceSession: 单个 Claude 实例的会话协调器
|
|
11
|
+
* 持有: PtyManager, OutputBuffer, HookReceiver, WS 客户端集合
|
|
12
|
+
* 复用 SessionController 的核心逻辑
|
|
13
|
+
*/
|
|
14
|
+
export class InstanceSession extends EventEmitter {
|
|
15
|
+
instanceId;
|
|
16
|
+
name;
|
|
17
|
+
cwd;
|
|
18
|
+
headless;
|
|
19
|
+
claudeArgs;
|
|
20
|
+
startedAt;
|
|
21
|
+
ptyManager;
|
|
22
|
+
hookReceiver;
|
|
23
|
+
_status = 'idle';
|
|
24
|
+
buffer;
|
|
25
|
+
clients = new Set();
|
|
26
|
+
pushService = null;
|
|
27
|
+
notificationServiceFactory = null;
|
|
28
|
+
notificationManager = null;
|
|
29
|
+
instanceUrl = null;
|
|
30
|
+
wsPendingChunks = [];
|
|
31
|
+
wsPendingBytes = 0;
|
|
32
|
+
wsFlushTimer = null;
|
|
33
|
+
constructor(options) {
|
|
34
|
+
super();
|
|
35
|
+
this.instanceId = options.instanceId;
|
|
36
|
+
this.name = options.name;
|
|
37
|
+
this.cwd = options.cwd;
|
|
38
|
+
this.headless = options.headless;
|
|
39
|
+
this.claudeArgs = options.claudeArgs ?? [];
|
|
40
|
+
this.startedAt = new Date().toISOString();
|
|
41
|
+
this.ptyManager = options.ptyManager ?? new PtyManager();
|
|
42
|
+
this.hookReceiver = new HookReceiver();
|
|
43
|
+
this.buffer = new OutputBuffer(options.maxBufferLines);
|
|
44
|
+
this.setupPtyHandlers();
|
|
45
|
+
this.setupHookHandlers();
|
|
46
|
+
}
|
|
47
|
+
get status() {
|
|
48
|
+
return this._status;
|
|
49
|
+
}
|
|
50
|
+
get clientCount() {
|
|
51
|
+
return this.clients.size;
|
|
52
|
+
}
|
|
53
|
+
setStatus(status) {
|
|
54
|
+
this._status = status;
|
|
55
|
+
this.broadcast({ type: 'status_update', status });
|
|
56
|
+
}
|
|
57
|
+
setPushService(pushService) {
|
|
58
|
+
this.pushService = pushService;
|
|
59
|
+
}
|
|
60
|
+
setNotificationServiceFactory(factory) {
|
|
61
|
+
this.notificationServiceFactory = factory;
|
|
62
|
+
}
|
|
63
|
+
setNotificationManager(manager) {
|
|
64
|
+
this.notificationManager = manager;
|
|
65
|
+
}
|
|
66
|
+
setInstanceUrl(url) {
|
|
67
|
+
this.instanceUrl = url;
|
|
68
|
+
logger.info({ instanceId: this.instanceId, instanceUrl: url }, 'Instance URL updated');
|
|
69
|
+
}
|
|
70
|
+
// ─── Client Management ────────────────────────────────────
|
|
71
|
+
/**
|
|
72
|
+
* 注册一个 WS 客户端到此实例
|
|
73
|
+
*/
|
|
74
|
+
addClient(ws, clientType) {
|
|
75
|
+
const clientInfo = { ws, alive: true, clientType };
|
|
76
|
+
this.clients.add(clientInfo);
|
|
77
|
+
logger.info({
|
|
78
|
+
instanceId: this.instanceId,
|
|
79
|
+
clientCount: this.clients.size,
|
|
80
|
+
clientType,
|
|
81
|
+
}, 'Client connected to instance');
|
|
82
|
+
ws.on('pong', () => {
|
|
83
|
+
clientInfo.alive = true;
|
|
84
|
+
});
|
|
85
|
+
ws.on('message', (rawData) => {
|
|
86
|
+
const data = rawData.toString();
|
|
87
|
+
this.handleWsMessage(ws, data, clientType);
|
|
88
|
+
});
|
|
89
|
+
ws.on('close', () => {
|
|
90
|
+
this.clients.delete(clientInfo);
|
|
91
|
+
logger.info({
|
|
92
|
+
instanceId: this.instanceId,
|
|
93
|
+
clientCount: this.clients.size,
|
|
94
|
+
}, 'Client disconnected from instance');
|
|
95
|
+
});
|
|
96
|
+
ws.on('error', (err) => {
|
|
97
|
+
logger.error({ err, instanceId: this.instanceId, clientType }, 'Client error');
|
|
98
|
+
this.clients.delete(clientInfo);
|
|
99
|
+
});
|
|
100
|
+
// 发送历史数据
|
|
101
|
+
this.sendTo(ws, {
|
|
102
|
+
type: 'history_sync',
|
|
103
|
+
data: this.buffer.getFullContent(),
|
|
104
|
+
seq: this.buffer.sequenceNumber,
|
|
105
|
+
status: this._status,
|
|
106
|
+
cols: this.ptyManager.cols,
|
|
107
|
+
rows: this.ptyManager.rows,
|
|
108
|
+
});
|
|
109
|
+
}
|
|
110
|
+
/**
|
|
111
|
+
* 向所有客户端广播消息
|
|
112
|
+
*/
|
|
113
|
+
broadcast(message) {
|
|
114
|
+
const data = JSON.stringify(message);
|
|
115
|
+
for (const client of this.clients) {
|
|
116
|
+
if (client.ws.readyState === WebSocket.OPEN) {
|
|
117
|
+
client.ws.send(data);
|
|
118
|
+
}
|
|
119
|
+
}
|
|
120
|
+
}
|
|
121
|
+
/**
|
|
122
|
+
* 向特定客户端发送消息
|
|
123
|
+
*/
|
|
124
|
+
sendTo(ws, message) {
|
|
125
|
+
if (ws.readyState === WebSocket.OPEN) {
|
|
126
|
+
ws.send(JSON.stringify(message));
|
|
127
|
+
}
|
|
128
|
+
}
|
|
129
|
+
/**
|
|
130
|
+
* 获取客户端计数
|
|
131
|
+
*/
|
|
132
|
+
getClientCounts() {
|
|
133
|
+
let attach = 0;
|
|
134
|
+
let webapp = 0;
|
|
135
|
+
for (const client of this.clients) {
|
|
136
|
+
if (client.clientType === 'attach')
|
|
137
|
+
attach++;
|
|
138
|
+
else
|
|
139
|
+
webapp++;
|
|
140
|
+
}
|
|
141
|
+
return { attach, webapp };
|
|
142
|
+
}
|
|
143
|
+
/**
|
|
144
|
+
* Ping 所有客户端,终止无响应的
|
|
145
|
+
*/
|
|
146
|
+
pingClients() {
|
|
147
|
+
for (const client of this.clients) {
|
|
148
|
+
if (!client.alive) {
|
|
149
|
+
logger.info({ instanceId: this.instanceId }, 'Terminating unresponsive client');
|
|
150
|
+
client.ws.terminate();
|
|
151
|
+
continue;
|
|
152
|
+
}
|
|
153
|
+
client.alive = false;
|
|
154
|
+
client.ws.ping();
|
|
155
|
+
}
|
|
156
|
+
}
|
|
157
|
+
// ─── PTY Handlers ─────────────────────────────────────────
|
|
158
|
+
setupPtyHandlers() {
|
|
159
|
+
this.ptyManager.on('data', (data) => {
|
|
160
|
+
// Buffer raw data for reconnection history
|
|
161
|
+
this.buffer.append(data);
|
|
162
|
+
// Broadcast raw PTY output to web clients (batched)
|
|
163
|
+
this.enqueueWsOutput(data);
|
|
164
|
+
});
|
|
165
|
+
this.ptyManager.on('exit', (exitCode) => {
|
|
166
|
+
this.flushPendingWsOutput();
|
|
167
|
+
this._status = 'idle';
|
|
168
|
+
this.broadcast({
|
|
169
|
+
type: 'session_ended',
|
|
170
|
+
exitCode,
|
|
171
|
+
reason: exitCode === 0 ? 'Process exited normally' : `Process exited with code ${exitCode}`,
|
|
172
|
+
});
|
|
173
|
+
// Gracefully close all WS connections after sending session_ended.
|
|
174
|
+
// ws.close() queues a close frame AFTER the data frame, ensuring
|
|
175
|
+
// the client receives session_ended before the connection drops.
|
|
176
|
+
for (const client of this.clients) {
|
|
177
|
+
client.ws.close(1000, 'PTY exited');
|
|
178
|
+
}
|
|
179
|
+
logger.info({ instanceId: this.instanceId, exitCode }, 'PTY exited');
|
|
180
|
+
// Emit exit event so InstanceManager can auto-remove
|
|
181
|
+
this.emit('exit', exitCode);
|
|
182
|
+
});
|
|
183
|
+
this.ptyManager.on('error', (err) => {
|
|
184
|
+
logger.error({ instanceId: this.instanceId, err }, 'PTY error');
|
|
185
|
+
this.broadcast({
|
|
186
|
+
type: 'error',
|
|
187
|
+
code: 'pty_error',
|
|
188
|
+
message: err.message,
|
|
189
|
+
});
|
|
190
|
+
});
|
|
191
|
+
this.ptyManager.on('resize', (cols, rows) => {
|
|
192
|
+
this.broadcast({ type: 'terminal_resize', cols, rows });
|
|
193
|
+
});
|
|
194
|
+
}
|
|
195
|
+
enqueueWsOutput(data) {
|
|
196
|
+
this.wsPendingChunks.push(data);
|
|
197
|
+
this.wsPendingBytes += Buffer.byteLength(data, 'utf8');
|
|
198
|
+
// 当累积数据达到最大块大小时立即刷新
|
|
199
|
+
if (this.wsPendingBytes >= WS_MAX_CHUNK_BYTES) {
|
|
200
|
+
this.flushPendingWsOutput();
|
|
201
|
+
return;
|
|
202
|
+
}
|
|
203
|
+
if (!this.wsFlushTimer) {
|
|
204
|
+
this.wsFlushTimer = setTimeout(() => {
|
|
205
|
+
this.wsFlushTimer = null;
|
|
206
|
+
this.flushPendingWsOutput();
|
|
207
|
+
}, WS_FLUSH_INTERVAL_MS);
|
|
208
|
+
}
|
|
209
|
+
}
|
|
210
|
+
flushPendingWsOutput() {
|
|
211
|
+
if (this.wsFlushTimer) {
|
|
212
|
+
clearTimeout(this.wsFlushTimer);
|
|
213
|
+
this.wsFlushTimer = null;
|
|
214
|
+
}
|
|
215
|
+
if (this.wsPendingChunks.length === 0)
|
|
216
|
+
return;
|
|
217
|
+
const output = this.wsPendingChunks.join('');
|
|
218
|
+
this.wsPendingChunks = [];
|
|
219
|
+
this.wsPendingBytes = 0;
|
|
220
|
+
this.broadcast({
|
|
221
|
+
type: 'terminal_output',
|
|
222
|
+
data: output,
|
|
223
|
+
seq: this.buffer.sequenceNumber,
|
|
224
|
+
});
|
|
225
|
+
}
|
|
226
|
+
// ─── WS Message Handler ───────────────────────────────────
|
|
227
|
+
handleWsMessage(ws, data, clientType) {
|
|
228
|
+
try {
|
|
229
|
+
const msg = JSON.parse(data);
|
|
230
|
+
switch (msg.type) {
|
|
231
|
+
case 'user_input':
|
|
232
|
+
if (typeof msg.data === 'string') {
|
|
233
|
+
this.ptyManager.write(msg.data);
|
|
234
|
+
}
|
|
235
|
+
break;
|
|
236
|
+
case 'resize':
|
|
237
|
+
if (typeof msg.cols === 'number' && typeof msg.rows === 'number') {
|
|
238
|
+
// webapp 在线时忽略 attach 的 resize
|
|
239
|
+
if (clientType === 'attach') {
|
|
240
|
+
const counts = this.getClientCounts();
|
|
241
|
+
if (counts.webapp > 0)
|
|
242
|
+
return;
|
|
243
|
+
}
|
|
244
|
+
this.ptyManager.resize(msg.cols, msg.rows);
|
|
245
|
+
}
|
|
246
|
+
break;
|
|
247
|
+
case 'heartbeat':
|
|
248
|
+
// Reply with server heartbeat
|
|
249
|
+
this.sendTo(ws, { type: 'heartbeat', timestamp: Date.now() });
|
|
250
|
+
break;
|
|
251
|
+
default:
|
|
252
|
+
logger.warn({ instanceId: this.instanceId, type: msg.type }, 'Unknown WS message type');
|
|
253
|
+
}
|
|
254
|
+
}
|
|
255
|
+
catch (err) {
|
|
256
|
+
logger.warn({ instanceId: this.instanceId, err }, 'Failed to parse WS message');
|
|
257
|
+
}
|
|
258
|
+
}
|
|
259
|
+
// ─── Hook Handlers ────────────────────────────────────────
|
|
260
|
+
setupHookHandlers() {
|
|
261
|
+
this.hookReceiver.on('notification', (notification) => {
|
|
262
|
+
this._status = 'waiting_input';
|
|
263
|
+
for (const channel of notification.channels) {
|
|
264
|
+
this.sendNotificationByChannel(channel, notification);
|
|
265
|
+
}
|
|
266
|
+
logger.info({
|
|
267
|
+
instanceId: this.instanceId,
|
|
268
|
+
eventType: notification.eventType,
|
|
269
|
+
tool: notification.tool,
|
|
270
|
+
channels: notification.channels,
|
|
271
|
+
}, 'Hook notification processed');
|
|
272
|
+
});
|
|
273
|
+
this.hookReceiver.on('task_completed', (_data) => {
|
|
274
|
+
if (this._status === 'waiting_input') {
|
|
275
|
+
this._status = 'running';
|
|
276
|
+
this.broadcast({
|
|
277
|
+
type: 'status_update',
|
|
278
|
+
status: 'running',
|
|
279
|
+
detail: 'Task resumed',
|
|
280
|
+
});
|
|
281
|
+
}
|
|
282
|
+
});
|
|
283
|
+
}
|
|
284
|
+
sendNotificationByChannel(channel, notification) {
|
|
285
|
+
if (channel !== 'websocket' && channel !== 'push') {
|
|
286
|
+
if (this.notificationManager && !this.notificationManager.isEnabled(channel))
|
|
287
|
+
return;
|
|
288
|
+
}
|
|
289
|
+
const urlHint = this.instanceUrl ? `\n\nInstance: ${this.instanceUrl}` : '';
|
|
290
|
+
switch (channel) {
|
|
291
|
+
case 'websocket':
|
|
292
|
+
this.broadcast({
|
|
293
|
+
type: 'status_update',
|
|
294
|
+
status: 'waiting_input',
|
|
295
|
+
detail: notification.message + urlHint,
|
|
296
|
+
});
|
|
297
|
+
break;
|
|
298
|
+
case 'push':
|
|
299
|
+
if (this.pushService) {
|
|
300
|
+
this.pushService
|
|
301
|
+
.notifyAll({
|
|
302
|
+
title: notification.title,
|
|
303
|
+
body: notification.message + urlHint,
|
|
304
|
+
tag: `claude-${notification.eventType}`,
|
|
305
|
+
renotify: true,
|
|
306
|
+
})
|
|
307
|
+
.catch((err) => {
|
|
308
|
+
logger.error({ err, channel: 'push' }, 'Failed to send push notification');
|
|
309
|
+
});
|
|
310
|
+
}
|
|
311
|
+
break;
|
|
312
|
+
case 'dingtalk':
|
|
313
|
+
if (this.notificationServiceFactory) {
|
|
314
|
+
const service = this.notificationServiceFactory.getDingtalkService();
|
|
315
|
+
if (service) {
|
|
316
|
+
const body = notification.detail
|
|
317
|
+
? `${notification.message}\n${notification.detail}${urlHint}`
|
|
318
|
+
: notification.message + urlHint;
|
|
319
|
+
service.sendNotification(notification.title, notification.tool, body).catch((err) => {
|
|
320
|
+
logger.error({ err, channel: 'dingtalk' }, 'Failed to send dingtalk notification');
|
|
321
|
+
});
|
|
322
|
+
}
|
|
323
|
+
}
|
|
324
|
+
break;
|
|
325
|
+
case 'wechat_work':
|
|
326
|
+
if (this.notificationServiceFactory) {
|
|
327
|
+
const service = this.notificationServiceFactory.getWechatWorkService();
|
|
328
|
+
if (service) {
|
|
329
|
+
const body = notification.detail
|
|
330
|
+
? `${notification.message}\n${notification.detail}${urlHint}`
|
|
331
|
+
: notification.message + urlHint;
|
|
332
|
+
service.sendNotification(notification.title, notification.tool, body).catch((err) => {
|
|
333
|
+
logger.error({ err, channel: 'wechat_work' }, 'Failed to send WeChat notification');
|
|
334
|
+
});
|
|
335
|
+
}
|
|
336
|
+
}
|
|
337
|
+
break;
|
|
338
|
+
}
|
|
339
|
+
}
|
|
340
|
+
// ─── Lifecycle ─────────────────────────────────────────────
|
|
341
|
+
/**
|
|
342
|
+
* 销毁实例:终止 PTY,断开所有客户端
|
|
343
|
+
*/
|
|
344
|
+
destroy() {
|
|
345
|
+
if (this.wsFlushTimer) {
|
|
346
|
+
clearTimeout(this.wsFlushTimer);
|
|
347
|
+
this.wsFlushTimer = null;
|
|
348
|
+
}
|
|
349
|
+
// 终止所有客户端连接
|
|
350
|
+
for (const client of this.clients) {
|
|
351
|
+
client.ws.terminate();
|
|
352
|
+
}
|
|
353
|
+
this.clients.clear();
|
|
354
|
+
// 销毁 PTY
|
|
355
|
+
this.ptyManager.destroy();
|
|
356
|
+
logger.info({ instanceId: this.instanceId }, 'Instance session destroyed');
|
|
357
|
+
}
|
|
358
|
+
}
|
|
359
|
+
//# sourceMappingURL=instance-session.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"instance-session.js","sourceRoot":"","sources":["../../../../backend/src/instance/instance-session.ts"],"names":[],"mappings":"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"}
|