@cjwddz/browser-server 0.1.0-alpha

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.
Files changed (58) hide show
  1. package/README.md +148 -0
  2. package/dist/cli.d.ts +3 -0
  3. package/dist/cli.d.ts.map +1 -0
  4. package/dist/cli.js +161 -0
  5. package/dist/cli.js.map +1 -0
  6. package/dist/daemon.d.ts +2 -0
  7. package/dist/daemon.d.ts.map +1 -0
  8. package/dist/daemon.js +57 -0
  9. package/dist/daemon.js.map +1 -0
  10. package/dist/index.d.ts +8 -0
  11. package/dist/index.d.ts.map +1 -0
  12. package/dist/index.js +7 -0
  13. package/dist/index.js.map +1 -0
  14. package/dist/mcp/connection-pool.d.ts +14 -0
  15. package/dist/mcp/connection-pool.d.ts.map +1 -0
  16. package/dist/mcp/connection-pool.js +65 -0
  17. package/dist/mcp/connection-pool.js.map +1 -0
  18. package/dist/mcp/tools.d.ts +230 -0
  19. package/dist/mcp/tools.d.ts.map +1 -0
  20. package/dist/mcp/tools.js +219 -0
  21. package/dist/mcp/tools.js.map +1 -0
  22. package/dist/server/mcp-server.d.ts +15 -0
  23. package/dist/server/mcp-server.d.ts.map +1 -0
  24. package/dist/server/mcp-server.js +96 -0
  25. package/dist/server/mcp-server.js.map +1 -0
  26. package/dist/server/vnc-proxy.d.ts +7 -0
  27. package/dist/server/vnc-proxy.d.ts.map +1 -0
  28. package/dist/server/vnc-proxy.js +38 -0
  29. package/dist/server/vnc-proxy.js.map +1 -0
  30. package/dist/server/web-pages.d.ts +4 -0
  31. package/dist/server/web-pages.d.ts.map +1 -0
  32. package/dist/server/web-pages.js +157 -0
  33. package/dist/server/web-pages.js.map +1 -0
  34. package/dist/server/web-server.d.ts +18 -0
  35. package/dist/server/web-server.d.ts.map +1 -0
  36. package/dist/server/web-server.js +161 -0
  37. package/dist/server/web-server.js.map +1 -0
  38. package/dist/session/process-group.d.ts +33 -0
  39. package/dist/session/process-group.d.ts.map +1 -0
  40. package/dist/session/process-group.js +185 -0
  41. package/dist/session/process-group.js.map +1 -0
  42. package/dist/session/session-manager.d.ts +26 -0
  43. package/dist/session/session-manager.d.ts.map +1 -0
  44. package/dist/session/session-manager.js +133 -0
  45. package/dist/session/session-manager.js.map +1 -0
  46. package/dist/utils/config.d.ts +14 -0
  47. package/dist/utils/config.d.ts.map +1 -0
  48. package/dist/utils/config.js +72 -0
  49. package/dist/utils/config.js.map +1 -0
  50. package/dist/utils/logger.d.ts +9 -0
  51. package/dist/utils/logger.d.ts.map +1 -0
  52. package/dist/utils/logger.js +33 -0
  53. package/dist/utils/logger.js.map +1 -0
  54. package/dist/utils/types.d.ts +26 -0
  55. package/dist/utils/types.d.ts.map +1 -0
  56. package/dist/utils/types.js +2 -0
  57. package/dist/utils/types.js.map +1 -0
  58. package/package.json +42 -0
@@ -0,0 +1,157 @@
1
+ function formatUptime(ms) {
2
+ if (ms < 60_000)
3
+ return '刚启动';
4
+ const mins = Math.floor(ms / 60_000);
5
+ if (mins < 60)
6
+ return `${mins} 分钟`;
7
+ const hours = Math.floor(mins / 60);
8
+ if (hours < 24)
9
+ return `${hours} 小时`;
10
+ const days = Math.floor(hours / 24);
11
+ return `${days} 天`;
12
+ }
13
+ function escapeHtml(s) {
14
+ return s.replace(/&/g, '&amp;').replace(/</g, '&lt;').replace(/>/g, '&gt;').replace(/"/g, '&quot;');
15
+ }
16
+ function layout(title, body) {
17
+ return `<!DOCTYPE html>
18
+ <html lang="zh-CN">
19
+ <head>
20
+ <meta charset="utf-8">
21
+ <meta name="viewport" content="width=device-width, initial-scale=1">
22
+ <title>${escapeHtml(title)}</title>
23
+ <style>
24
+ * { box-sizing: border-box; margin: 0; padding: 0; }
25
+ body { font-family: -apple-system, BlinkMacSystemFont, 'Segoe UI', Roboto, sans-serif; background: #f5f5f5; color: #333; }
26
+ .container { max-width: 960px; margin: 0 auto; padding: 24px; }
27
+ h1 { font-size: 24px; margin-bottom: 24px; color: #1a1a1a; }
28
+ .card { background: #fff; border-radius: 8px; box-shadow: 0 1px 3px rgba(0,0,0,0.1); padding: 16px 20px; margin-bottom: 12px; }
29
+ .card-header { display: flex; align-items: center; justify-content: space-between; }
30
+ .card-name { font-size: 16px; font-weight: 600; }
31
+ .card-meta { font-size: 13px; color: #888; margin-top: 4px; }
32
+ .card-actions { display: flex; gap: 8px; }
33
+ .btn { display: inline-block; padding: 6px 14px; border-radius: 6px; font-size: 13px; text-decoration: none; border: none; cursor: pointer; }
34
+ .btn-primary { background: #2563eb; color: #fff; }
35
+ .btn-primary:hover { background: #1d4ed8; }
36
+ .btn-danger { background: #ef4444; color: #fff; }
37
+ .btn-danger:hover { background: #dc2626; }
38
+ .btn-outline { background: #fff; color: #2563eb; border: 1px solid #2563eb; }
39
+ .btn-outline:hover { background: #eff6ff; }
40
+ .create-form { display: flex; gap: 8px; align-items: center; margin-bottom: 24px; }
41
+ .create-form input { padding: 8px 12px; border: 1px solid #d1d5db; border-radius: 6px; font-size: 14px; flex: 1; max-width: 300px; }
42
+ .mcp-config { background: #f8fafc; border: 1px solid #e2e8f0; border-radius: 6px; padding: 12px; margin-top: 8px; font-family: monospace; font-size: 12px; white-space: pre; overflow-x: auto; position: relative; }
43
+ .mcp-config .copy-btn { position: absolute; top: 8px; right: 8px; }
44
+ .status-dot { display: inline-block; width: 8px; height: 8px; border-radius: 50%; background: #22c55e; margin-right: 6px; }
45
+ .back-link { display: inline-block; margin-bottom: 16px; color: #2563eb; text-decoration: none; }
46
+ .viewer-container { width: 100%; height: calc(100vh - 160px); background: #000; border-radius: 8px; overflow: hidden; }
47
+ .viewer-container canvas { width: 100%; height: 100%; }
48
+ .empty { text-align: center; padding: 48px; color: #888; }
49
+ </style>
50
+ </head>
51
+ <body>
52
+ <div class="container">
53
+ ${body}
54
+ </div>
55
+ </body>
56
+ </html>`;
57
+ }
58
+ export function renderHomePage(sessions, config) {
59
+ const sessionCards = sessions.map(s => {
60
+ const createdDate = new Date(s.createdAt).toLocaleString('zh-CN');
61
+ const mcpConfig = JSON.stringify({
62
+ mcpServers: {
63
+ browser: { url: `http://HOST:${config.mcpPort}/mcp/${s.id}` }
64
+ }
65
+ }, null, 2);
66
+ return `
67
+ <div class="card">
68
+ <div class="card-header">
69
+ <div>
70
+ <div class="card-name"><span class="status-dot"></span>${escapeHtml(s.name)}</div>
71
+ <div class="card-meta">ID: ${s.id} &middot; 创建于 ${createdDate}</div>
72
+ </div>
73
+ <div class="card-actions">
74
+ <a href="/viewer/${s.id}" class="btn btn-primary">查看浏览器</a>
75
+ <button class="btn btn-outline" onclick="toggleMcp('${s.id}')">MCP 配置</button>
76
+ <form method="POST" action="/delete/${s.id}" style="display:inline" onsubmit="return confirm('确定要删除会话 ${escapeHtml(s.name)} 吗?所有浏览器数据将被清除。')">
77
+ <button type="submit" class="btn btn-danger">删除</button>
78
+ </form>
79
+ </div>
80
+ </div>
81
+ <div id="mcp-${s.id}" class="mcp-config" style="display:none">
82
+ <button class="btn btn-outline copy-btn" onclick="copyMcp('${s.id}')">复制</button>
83
+ <span id="mcp-text-${s.id}">${escapeHtml(mcpConfig)}</span>
84
+ </div>
85
+ </div>`;
86
+ }).join('\n');
87
+ const body = `
88
+ <h1>Browser Server</h1>
89
+ <form class="create-form" method="POST" action="/create">
90
+ <input type="text" name="name" placeholder="输入会话名称" required>
91
+ <button type="submit" class="btn btn-primary">新建会话</button>
92
+ </form>
93
+ ${sessions.length === 0 ? '<div class="empty">暂无会话,创建一个开始使用</div>' : sessionCards}
94
+ <script>
95
+ function toggleMcp(id) {
96
+ var el = document.getElementById('mcp-' + id);
97
+ el.style.display = el.style.display === 'none' ? 'block' : 'none';
98
+ }
99
+ function copyMcp(id) {
100
+ var text = document.getElementById('mcp-text-' + id).textContent;
101
+ text = text.replace(/HOST/g, location.hostname);
102
+ navigator.clipboard.writeText(text).then(function() {
103
+ alert('MCP 配置已复制到剪贴板');
104
+ });
105
+ }
106
+ </script>`;
107
+ return layout('Browser Server', body);
108
+ }
109
+ export function renderViewerPage(session, config) {
110
+ const mcpUrl = `http://HOST:${config.mcpPort}/mcp/${session.id}`;
111
+ const body = `
112
+ <a href="/" class="back-link">&larr; 返回会话列表</a>
113
+ <h1>${escapeHtml(session.name)}</h1>
114
+ <div id="viewer-container" class="viewer-container">
115
+ <div class="empty" id="loading">正在连接浏览器...</div>
116
+ </div>
117
+ <div style="margin-top: 12px; display: flex; gap: 8px; align-items: center;">
118
+ <button class="btn btn-outline" onclick="toggleMcpPanel()">MCP 配置</button>
119
+ </div>
120
+ <div id="mcp-panel" class="mcp-config" style="display:none; margin-top:12px;">
121
+ <button class="btn btn-outline copy-btn" onclick="copyConfig()">复制</button>
122
+ <span id="mcp-text">${escapeHtml(JSON.stringify({ mcpServers: { browser: { url: mcpUrl } } }, null, 2))}</span>
123
+ </div>
124
+
125
+ <script type="module">
126
+ import RFB from 'https://cdn.jsdelivr.net/npm/@novnc/novnc@1.5.0/core/rfb.js';
127
+
128
+ const container = document.getElementById('viewer-container');
129
+ const loading = document.getElementById('loading');
130
+ const wsProto = location.protocol === 'https:' ? 'wss:' : 'ws:';
131
+ const wsUrl = wsProto + '//' + location.host + '/vnc/${session.id}';
132
+
133
+ try {
134
+ const rfb = new RFB(container, wsUrl, { wsProtocols: ['binary'] });
135
+ rfb.scaleViewport = true;
136
+ rfb.resizeSession = false;
137
+ rfb.addEventListener('connect', () => { loading.style.display = 'none'; });
138
+ rfb.addEventListener('disconnect', () => {
139
+ loading.textContent = '连接已断开,刷新页面重试';
140
+ loading.style.display = 'block';
141
+ });
142
+ } catch (e) {
143
+ loading.textContent = '连接失败: ' + e.message;
144
+ }
145
+
146
+ window.toggleMcpPanel = () => {
147
+ const el = document.getElementById('mcp-panel');
148
+ el.style.display = el.style.display === 'none' ? 'block' : 'none';
149
+ };
150
+ window.copyConfig = () => {
151
+ const text = document.getElementById('mcp-text').textContent.replace(/HOST/g, location.hostname);
152
+ navigator.clipboard.writeText(text).then(() => alert('已复制'));
153
+ };
154
+ </script>`;
155
+ return layout(`${session.name} - Browser Server`, body);
156
+ }
157
+ //# sourceMappingURL=web-pages.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"web-pages.js","sourceRoot":"","sources":["../../src/server/web-pages.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,18 @@
1
+ import type { SessionManager } from '../session/session-manager.js';
2
+ import type { ServerConfig } from '../utils/types.js';
3
+ export declare class WebServer {
4
+ private config;
5
+ private sessionManager;
6
+ private server;
7
+ private wss;
8
+ constructor(config: ServerConfig, sessionManager: SessionManager);
9
+ start(): Promise<void>;
10
+ stop(): Promise<void>;
11
+ private handleRequest;
12
+ private handleCreate;
13
+ private handleDelete;
14
+ private checkAuth;
15
+ private servePage;
16
+ private serve404;
17
+ }
18
+ //# sourceMappingURL=web-server.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"web-server.d.ts","sourceRoot":"","sources":["../../src/server/web-server.ts"],"names":[],"mappings":"AAKA,OAAO,KAAK,EAAE,cAAc,EAAE,MAAM,+BAA+B,CAAC;AACpE,OAAO,KAAK,EAAE,YAAY,EAAE,MAAM,mBAAmB,CAAC;AAEtD,qBAAa,SAAS;IACpB,OAAO,CAAC,MAAM,CAAe;IAC7B,OAAO,CAAC,cAAc,CAAiB;IACvC,OAAO,CAAC,MAAM,CAA4B;IAC1C,OAAO,CAAC,GAAG,CAAgC;gBAE/B,MAAM,EAAE,YAAY,EAAE,cAAc,EAAE,cAAc;IAK1D,KAAK,IAAI,OAAO,CAAC,IAAI,CAAC;IAiDtB,IAAI,IAAI,OAAO,CAAC,IAAI,CAAC;YAeb,aAAa;YAkCb,YAAY;YAsBZ,YAAY;IAiB1B,OAAO,CAAC,SAAS;IAQjB,OAAO,CAAC,SAAS;IAKjB,OAAO,CAAC,QAAQ;CAIjB"}
@@ -0,0 +1,161 @@
1
+ import * as http from 'node:http';
2
+ import { WebSocketServer } from 'ws';
3
+ import { log } from '../utils/logger.js';
4
+ import { proxyVncWebSocket } from './vnc-proxy.js';
5
+ import { renderHomePage, renderViewerPage } from './web-pages.js';
6
+ export class WebServer {
7
+ config;
8
+ sessionManager;
9
+ server = null;
10
+ wss = null;
11
+ constructor(config, sessionManager) {
12
+ this.config = config;
13
+ this.sessionManager = sessionManager;
14
+ }
15
+ async start() {
16
+ this.server = http.createServer((req, res) => {
17
+ this.handleRequest(req, res).catch((err) => {
18
+ log.error('Web 请求处理异常:', err instanceof Error ? err.message : err);
19
+ if (!res.headersSent) {
20
+ res.writeHead(500);
21
+ res.end('Internal Server Error');
22
+ }
23
+ });
24
+ });
25
+ this.wss = new WebSocketServer({ noServer: true });
26
+ this.server.on('upgrade', (req, socket, head) => {
27
+ if (!this.checkAuth(req)) {
28
+ socket.write('HTTP/1.1 401 Unauthorized\r\n\r\n');
29
+ socket.destroy();
30
+ return;
31
+ }
32
+ const url = new URL(req.url || '/', `http://${req.headers.host}`);
33
+ const match = url.pathname.match(/^\/vnc\/([a-f0-9]+)$/);
34
+ if (!match) {
35
+ socket.write('HTTP/1.1 404 Not Found\r\n\r\n');
36
+ socket.destroy();
37
+ return;
38
+ }
39
+ const sessionId = match[1];
40
+ const session = this.sessionManager.getSession(sessionId);
41
+ if (!session) {
42
+ socket.write('HTTP/1.1 404 Not Found\r\n\r\n');
43
+ socket.destroy();
44
+ return;
45
+ }
46
+ this.wss.handleUpgrade(req, socket, head, (ws) => {
47
+ proxyVncWebSocket(ws, session.vncPort, sessionId);
48
+ });
49
+ });
50
+ return new Promise((resolve) => {
51
+ this.server.listen(this.config.webPort, this.config.host, () => {
52
+ log.info(`Web 管理服务已启动: http://${this.config.host}:${this.config.webPort}`);
53
+ resolve();
54
+ });
55
+ });
56
+ }
57
+ async stop() {
58
+ if (this.wss) {
59
+ this.wss.close();
60
+ this.wss = null;
61
+ }
62
+ if (this.server) {
63
+ return new Promise((resolve) => {
64
+ this.server.close(() => {
65
+ log.info('Web 管理服务已停止');
66
+ resolve();
67
+ });
68
+ });
69
+ }
70
+ }
71
+ async handleRequest(req, res) {
72
+ if (!this.checkAuth(req)) {
73
+ res.writeHead(401, { 'WWW-Authenticate': 'Basic realm="browser-server"' });
74
+ res.end('Unauthorized');
75
+ return;
76
+ }
77
+ const url = new URL(req.url || '/', `http://${req.headers.host}`);
78
+ if (req.method === 'GET' && url.pathname === '/') {
79
+ return this.servePage(res, renderHomePage(this.sessionManager.listSessions(), this.config));
80
+ }
81
+ if (req.method === 'GET' && url.pathname.match(/^\/viewer\/[a-f0-9]+$/)) {
82
+ const sessionId = url.pathname.split('/')[2];
83
+ const session = this.sessionManager.getSession(sessionId);
84
+ if (!session)
85
+ return this.serve404(res);
86
+ return this.servePage(res, renderViewerPage(session, this.config));
87
+ }
88
+ if (req.method === 'POST' && url.pathname === '/create') {
89
+ this.handleCreate(req, res);
90
+ return;
91
+ }
92
+ if (req.method === 'POST' && url.pathname.match(/^\/delete\/[a-f0-9]+$/)) {
93
+ const sessionId = url.pathname.split('/')[2];
94
+ this.handleDelete(sessionId, res);
95
+ return;
96
+ }
97
+ this.serve404(res);
98
+ }
99
+ async handleCreate(req, res) {
100
+ const body = await readBody(req);
101
+ const params = new URLSearchParams(body);
102
+ const name = params.get('name')?.trim();
103
+ if (!name) {
104
+ res.writeHead(400);
105
+ res.end('会话名称不能为空');
106
+ return;
107
+ }
108
+ try {
109
+ const session = this.sessionManager.createSession(name);
110
+ await this.sessionManager.startSession(session);
111
+ res.writeHead(302, { Location: '/' });
112
+ res.end();
113
+ }
114
+ catch (err) {
115
+ log.error('创建会话失败:', err instanceof Error ? err.message : err);
116
+ res.writeHead(500);
117
+ res.end(`创建失败: ${err instanceof Error ? err.message : err}`);
118
+ }
119
+ }
120
+ async handleDelete(sessionId, res) {
121
+ try {
122
+ const ok = await this.sessionManager.deleteSession(sessionId);
123
+ if (!ok) {
124
+ res.writeHead(404);
125
+ res.end('会话不存在');
126
+ return;
127
+ }
128
+ res.writeHead(302, { Location: '/' });
129
+ res.end();
130
+ }
131
+ catch (err) {
132
+ log.error('删除会话失败:', err instanceof Error ? err.message : err);
133
+ res.writeHead(500);
134
+ res.end(`删除失败: ${err instanceof Error ? err.message : err}`);
135
+ }
136
+ }
137
+ checkAuth(req) {
138
+ const auth = req.headers.authorization;
139
+ if (!auth || !auth.startsWith('Basic '))
140
+ return false;
141
+ const decoded = Buffer.from(auth.slice(6), 'base64').toString('utf-8');
142
+ const [user, pass] = decoded.split(':');
143
+ return user === this.config.user && pass === this.config.pass;
144
+ }
145
+ servePage(res, html) {
146
+ res.writeHead(200, { 'Content-Type': 'text/html; charset=utf-8' });
147
+ res.end(html);
148
+ }
149
+ serve404(res) {
150
+ res.writeHead(404);
151
+ res.end('Not Found');
152
+ }
153
+ }
154
+ function readBody(req) {
155
+ return new Promise((resolve) => {
156
+ let body = '';
157
+ req.on('data', (chunk) => { body += chunk; });
158
+ req.on('end', () => resolve(body));
159
+ });
160
+ }
161
+ //# sourceMappingURL=web-server.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"web-server.js","sourceRoot":"","sources":["../../src/server/web-server.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,33 @@
1
+ export interface ProcessGroupConfig {
2
+ sessionId: string;
3
+ displayNum: number;
4
+ cdpPort: number;
5
+ vncPort: number;
6
+ chromeDataDir: string;
7
+ chromeLogPath: string;
8
+ }
9
+ export declare class ProcessGroup {
10
+ private config;
11
+ private xvfb;
12
+ private chromium;
13
+ private vnc;
14
+ private startedAt;
15
+ private tabCount;
16
+ private guardInterval;
17
+ constructor(config: ProcessGroupConfig);
18
+ start(): Promise<void>;
19
+ stop(): Promise<void>;
20
+ getTabCount(): number;
21
+ setTabCount(count: number): void;
22
+ getUptime(): number;
23
+ getCdpPort(): number;
24
+ getVncPort(): number;
25
+ private spawnXvfb;
26
+ private spawnChromium;
27
+ private spawnVnc;
28
+ private attachErrorHandler;
29
+ private guard;
30
+ private killProcess;
31
+ private waitForProcess;
32
+ }
33
+ //# sourceMappingURL=process-group.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"process-group.d.ts","sourceRoot":"","sources":["../../src/session/process-group.ts"],"names":[],"mappings":"AAIA,MAAM,WAAW,kBAAkB;IACjC,SAAS,EAAE,MAAM,CAAC;IAClB,UAAU,EAAE,MAAM,CAAC;IACnB,OAAO,EAAE,MAAM,CAAC;IAChB,OAAO,EAAE,MAAM,CAAC;IAChB,aAAa,EAAE,MAAM,CAAC;IACtB,aAAa,EAAE,MAAM,CAAC;CACvB;AAED,qBAAa,YAAY;IACvB,OAAO,CAAC,MAAM,CAAqB;IACnC,OAAO,CAAC,IAAI,CAA6B;IACzC,OAAO,CAAC,QAAQ,CAA6B;IAC7C,OAAO,CAAC,GAAG,CAA6B;IACxC,OAAO,CAAC,SAAS,CAAa;IAC9B,OAAO,CAAC,QAAQ,CAAa;IAC7B,OAAO,CAAC,aAAa,CAA+C;gBAExD,MAAM,EAAE,kBAAkB;IAIhC,KAAK,IAAI,OAAO,CAAC,IAAI,CAAC;IAmBtB,IAAI,IAAI,OAAO,CAAC,IAAI,CAAC;IAmB3B,WAAW,IAAI,MAAM;IAIrB,WAAW,CAAC,KAAK,EAAE,MAAM,GAAG,IAAI;IAIhC,SAAS,IAAI,MAAM;IAInB,UAAU,IAAI,MAAM;IAIpB,UAAU,IAAI,MAAM;IAIpB,OAAO,CAAC,SAAS;IAcjB,OAAO,CAAC,aAAa;IAkCrB,OAAO,CAAC,QAAQ;IAmBhB,OAAO,CAAC,kBAAkB;IAS1B,OAAO,CAAC,KAAK;IAoBb,OAAO,CAAC,WAAW;IAcnB,OAAO,CAAC,cAAc;CAGvB"}
@@ -0,0 +1,185 @@
1
+ import { spawn, execSync } from 'node:child_process';
2
+ import * as fs from 'node:fs';
3
+ import { log } from '../utils/logger.js';
4
+ export class ProcessGroup {
5
+ config;
6
+ xvfb = null;
7
+ chromium = null;
8
+ vnc = null;
9
+ startedAt = 0;
10
+ tabCount = 1;
11
+ guardInterval = null;
12
+ constructor(config) {
13
+ this.config = config;
14
+ }
15
+ async start() {
16
+ this.startedAt = Date.now();
17
+ const { sessionId, displayNum, cdpPort, vncPort, chromeDataDir, chromeLogPath } = this.config;
18
+ const display = `:${displayNum}`;
19
+ log.info(`启动进程组 [${sessionId}]: display=${display}, cdp=${cdpPort}, vnc=${vncPort}`);
20
+ this.xvfb = this.spawnXvfb(display);
21
+ await this.waitForProcess('Xvfb', 1000);
22
+ this.chromium = this.spawnChromium(display, cdpPort, chromeDataDir, chromeLogPath);
23
+ await this.waitForProcess('Chromium', 2000);
24
+ this.vnc = this.spawnVnc(display, vncPort);
25
+ this.guardInterval = setInterval(() => this.guard(), 5000);
26
+ log.info(`进程组 [${sessionId}] 启动完成`);
27
+ }
28
+ async stop() {
29
+ if (this.guardInterval) {
30
+ clearInterval(this.guardInterval);
31
+ this.guardInterval = null;
32
+ }
33
+ this.killProcess(this.vnc, 'VNC');
34
+ this.vnc = null;
35
+ this.killProcess(this.chromium, 'Chromium');
36
+ this.chromium = null;
37
+ this.killProcess(this.xvfb, 'Xvfb');
38
+ this.xvfb = null;
39
+ await sleep(500);
40
+ log.info(`进程组 [${this.config.sessionId}] 已停止`);
41
+ }
42
+ getTabCount() {
43
+ return this.tabCount;
44
+ }
45
+ setTabCount(count) {
46
+ this.tabCount = count;
47
+ }
48
+ getUptime() {
49
+ return this.startedAt > 0 ? Date.now() - this.startedAt : 0;
50
+ }
51
+ getCdpPort() {
52
+ return this.config.cdpPort;
53
+ }
54
+ getVncPort() {
55
+ return this.config.vncPort;
56
+ }
57
+ spawnXvfb(display) {
58
+ const child = spawn('Xvfb', [
59
+ display,
60
+ '-screen', '0', '1280x720x24',
61
+ '-nolisten', 'tcp',
62
+ '-ac',
63
+ ], {
64
+ stdio: 'ignore',
65
+ detached: false,
66
+ });
67
+ this.attachErrorHandler(child, 'Xvfb');
68
+ return child;
69
+ }
70
+ spawnChromium(display, cdpPort, dataDir, logPath) {
71
+ const chromeBin = findChromium();
72
+ const logStream = fs.openSync(logPath, 'a');
73
+ const child = spawn(chromeBin, [
74
+ '--no-first-run',
75
+ '--no-default-browser-check',
76
+ `--user-data-dir=${dataDir}`,
77
+ `--remote-debugging-port=${cdpPort}`,
78
+ '--remote-debugging-address=127.0.0.1',
79
+ '--disable-background-networking',
80
+ '--disable-default-apps',
81
+ '--disable-extensions',
82
+ '--disable-sync',
83
+ '--disable-translate',
84
+ '--disable-dev-shm-usage',
85
+ '--no-sandbox',
86
+ '--disable-setuid-sandbox',
87
+ '--metrics-recording-only',
88
+ '--safebrowsing-disable-auto-update',
89
+ 'about:blank',
90
+ ], {
91
+ stdio: ['ignore', logStream, logStream],
92
+ detached: false,
93
+ env: { ...process.env, DISPLAY: display },
94
+ });
95
+ this.attachErrorHandler(child, 'Chromium');
96
+ child.on('exit', () => {
97
+ try {
98
+ fs.closeSync(logStream);
99
+ }
100
+ catch { /* fd may already be closed */ }
101
+ });
102
+ return child;
103
+ }
104
+ spawnVnc(display, vncPort) {
105
+ const child = spawn('x11vnc', [
106
+ '-display', display,
107
+ '-rfbport', String(vncPort),
108
+ '-listen', '127.0.0.1',
109
+ '-nopw',
110
+ '-forever',
111
+ '-shared',
112
+ '-xkb',
113
+ '-ncache', '10',
114
+ '-noxdamage',
115
+ ], {
116
+ stdio: 'ignore',
117
+ detached: false,
118
+ });
119
+ this.attachErrorHandler(child, 'x11vnc');
120
+ return child;
121
+ }
122
+ attachErrorHandler(child, name) {
123
+ child.on('error', (err) => {
124
+ log.error(`[${this.config.sessionId}] ${name} 进程错误:`, err.message);
125
+ });
126
+ child.on('exit', (code, signal) => {
127
+ log.warn(`[${this.config.sessionId}] ${name} 进程退出: code=${code}, signal=${signal}`);
128
+ });
129
+ }
130
+ guard() {
131
+ if (this.xvfb && this.xvfb.exitCode !== null) {
132
+ log.warn(`[${this.config.sessionId}] Xvfb 崩溃,重启中...`);
133
+ this.xvfb = this.spawnXvfb(`:${this.config.displayNum}`);
134
+ }
135
+ if (this.chromium && this.chromium.exitCode !== null) {
136
+ log.warn(`[${this.config.sessionId}] Chromium 崩溃,重启中...`);
137
+ this.chromium = this.spawnChromium(`:${this.config.displayNum}`, this.config.cdpPort, this.config.chromeDataDir, this.config.chromeLogPath);
138
+ }
139
+ if (this.vnc && this.vnc.exitCode !== null) {
140
+ log.warn(`[${this.config.sessionId}] VNC 崩溃,重启中...`);
141
+ this.vnc = this.spawnVnc(`:${this.config.displayNum}`, this.config.vncPort);
142
+ }
143
+ }
144
+ killProcess(child, name) {
145
+ if (!child || child.exitCode !== null)
146
+ return;
147
+ try {
148
+ child.kill('SIGTERM');
149
+ setTimeout(() => {
150
+ if (child.exitCode === null) {
151
+ child.kill('SIGKILL');
152
+ }
153
+ }, 3000);
154
+ }
155
+ catch (err) {
156
+ log.error(`终止 ${name} 失败:`, err instanceof Error ? err.message : err);
157
+ }
158
+ }
159
+ waitForProcess(_name, ms) {
160
+ return sleep(ms);
161
+ }
162
+ }
163
+ function findChromium() {
164
+ const candidates = [
165
+ 'chromium-browser',
166
+ 'chromium',
167
+ 'google-chrome-stable',
168
+ 'google-chrome',
169
+ ];
170
+ for (const bin of candidates) {
171
+ try {
172
+ const result = execSync(`which ${bin}`, { encoding: 'utf-8', stdio: 'pipe' }).trim();
173
+ if (result)
174
+ return result;
175
+ }
176
+ catch {
177
+ // continue
178
+ }
179
+ }
180
+ throw new Error('未找到 Chromium/Chrome 浏览器。请安装 chromium-browser 或 google-chrome。');
181
+ }
182
+ function sleep(ms) {
183
+ return new Promise(resolve => setTimeout(resolve, ms));
184
+ }
185
+ //# sourceMappingURL=process-group.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"process-group.js","sourceRoot":"","sources":["../../src/session/process-group.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,26 @@
1
+ import { ProcessGroup } from './process-group.js';
2
+ import type { SessionInfo } from '../utils/types.js';
3
+ export declare class SessionManager {
4
+ private dataDir;
5
+ private sessionsPath;
6
+ private data;
7
+ private processGroups;
8
+ constructor(dataDir: string);
9
+ private loadSessionsData;
10
+ private saveSessionsData;
11
+ private sessionDir;
12
+ private chromeDataDir;
13
+ private chromeLogPath;
14
+ createSession(name: string): SessionInfo;
15
+ deleteSession(id: string): Promise<boolean>;
16
+ getSession(id: string): SessionInfo | undefined;
17
+ listSessions(): SessionInfo[];
18
+ startSession(session: SessionInfo): Promise<void>;
19
+ stopSession(id: string): Promise<void>;
20
+ restoreSessions(): Promise<void>;
21
+ stopAllSessions(): Promise<void>;
22
+ getProcessGroup(id: string): ProcessGroup | undefined;
23
+ getSessionTabCount(id: string): number;
24
+ getSessionUptime(id: string): number;
25
+ }
26
+ //# sourceMappingURL=session-manager.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"session-manager.d.ts","sourceRoot":"","sources":["../../src/session/session-manager.ts"],"names":[],"mappings":"AAKA,OAAO,EAAE,YAAY,EAAE,MAAM,oBAAoB,CAAC;AAClD,OAAO,KAAK,EAAE,WAAW,EAAgB,MAAM,mBAAmB,CAAC;AAMnE,qBAAa,cAAc;IACzB,OAAO,CAAC,OAAO,CAAS;IACxB,OAAO,CAAC,YAAY,CAAS;IAC7B,OAAO,CAAC,IAAI,CAAe;IAC3B,OAAO,CAAC,aAAa,CAAwC;gBAEjD,OAAO,EAAE,MAAM;IAM3B,OAAO,CAAC,gBAAgB;IAWxB,OAAO,CAAC,gBAAgB;IAIxB,OAAO,CAAC,UAAU;IAIlB,OAAO,CAAC,aAAa;IAIrB,OAAO,CAAC,aAAa;IAIrB,aAAa,CAAC,IAAI,EAAE,MAAM,GAAG,WAAW;IAoBlC,aAAa,CAAC,EAAE,EAAE,MAAM,GAAG,OAAO,CAAC,OAAO,CAAC;IAkBjD,UAAU,CAAC,EAAE,EAAE,MAAM,GAAG,WAAW,GAAG,SAAS;IAI/C,YAAY,IAAI,WAAW,EAAE;IAIvB,YAAY,CAAC,OAAO,EAAE,WAAW,GAAG,OAAO,CAAC,IAAI,CAAC;IAmBjD,WAAW,CAAC,EAAE,EAAE,MAAM,GAAG,OAAO,CAAC,IAAI,CAAC;IAQtC,eAAe,IAAI,OAAO,CAAC,IAAI,CAAC;IAWhC,eAAe,IAAI,OAAO,CAAC,IAAI,CAAC;IAQtC,eAAe,CAAC,EAAE,EAAE,MAAM,GAAG,YAAY,GAAG,SAAS;IAIrD,kBAAkB,CAAC,EAAE,EAAE,MAAM,GAAG,MAAM;IAKtC,gBAAgB,CAAC,EAAE,EAAE,MAAM,GAAG,MAAM;CAIrC"}