@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.
- package/README.md +148 -0
- package/dist/cli.d.ts +3 -0
- package/dist/cli.d.ts.map +1 -0
- package/dist/cli.js +161 -0
- package/dist/cli.js.map +1 -0
- package/dist/daemon.d.ts +2 -0
- package/dist/daemon.d.ts.map +1 -0
- package/dist/daemon.js +57 -0
- package/dist/daemon.js.map +1 -0
- package/dist/index.d.ts +8 -0
- package/dist/index.d.ts.map +1 -0
- package/dist/index.js +7 -0
- package/dist/index.js.map +1 -0
- package/dist/mcp/connection-pool.d.ts +14 -0
- package/dist/mcp/connection-pool.d.ts.map +1 -0
- package/dist/mcp/connection-pool.js +65 -0
- package/dist/mcp/connection-pool.js.map +1 -0
- package/dist/mcp/tools.d.ts +230 -0
- package/dist/mcp/tools.d.ts.map +1 -0
- package/dist/mcp/tools.js +219 -0
- package/dist/mcp/tools.js.map +1 -0
- package/dist/server/mcp-server.d.ts +15 -0
- package/dist/server/mcp-server.d.ts.map +1 -0
- package/dist/server/mcp-server.js +96 -0
- package/dist/server/mcp-server.js.map +1 -0
- package/dist/server/vnc-proxy.d.ts +7 -0
- package/dist/server/vnc-proxy.d.ts.map +1 -0
- package/dist/server/vnc-proxy.js +38 -0
- package/dist/server/vnc-proxy.js.map +1 -0
- package/dist/server/web-pages.d.ts +4 -0
- package/dist/server/web-pages.d.ts.map +1 -0
- package/dist/server/web-pages.js +157 -0
- package/dist/server/web-pages.js.map +1 -0
- package/dist/server/web-server.d.ts +18 -0
- package/dist/server/web-server.d.ts.map +1 -0
- package/dist/server/web-server.js +161 -0
- package/dist/server/web-server.js.map +1 -0
- package/dist/session/process-group.d.ts +33 -0
- package/dist/session/process-group.d.ts.map +1 -0
- package/dist/session/process-group.js +185 -0
- package/dist/session/process-group.js.map +1 -0
- package/dist/session/session-manager.d.ts +26 -0
- package/dist/session/session-manager.d.ts.map +1 -0
- package/dist/session/session-manager.js +133 -0
- package/dist/session/session-manager.js.map +1 -0
- package/dist/utils/config.d.ts +14 -0
- package/dist/utils/config.d.ts.map +1 -0
- package/dist/utils/config.js +72 -0
- package/dist/utils/config.js.map +1 -0
- package/dist/utils/logger.d.ts +9 -0
- package/dist/utils/logger.d.ts.map +1 -0
- package/dist/utils/logger.js +33 -0
- package/dist/utils/logger.js.map +1 -0
- package/dist/utils/types.d.ts +26 -0
- package/dist/utils/types.d.ts.map +1 -0
- package/dist/utils/types.js +2 -0
- package/dist/utils/types.js.map +1 -0
- 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, '&').replace(/</g, '<').replace(/>/g, '>').replace(/"/g, '"');
|
|
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} · 创建于 ${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">← 返回会话列表</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"}
|