@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,230 @@
1
+ import { z } from 'zod';
2
+ import type { Browser, Page, BrowserContext } from 'playwright-core';
3
+ export interface BrowserConnection {
4
+ browser: Browser;
5
+ context: BrowserContext;
6
+ activePage: Page;
7
+ pages: Page[];
8
+ }
9
+ export declare function defineTools(): {
10
+ navigate_page: {
11
+ description: string;
12
+ schema: {
13
+ url: z.ZodString;
14
+ };
15
+ handler: (conn: BrowserConnection, args: {
16
+ url: string;
17
+ }) => Promise<{
18
+ content: {
19
+ type: "text";
20
+ text: string;
21
+ }[];
22
+ }>;
23
+ };
24
+ click: {
25
+ description: string;
26
+ schema: {
27
+ selector: z.ZodString;
28
+ };
29
+ handler: (conn: BrowserConnection, args: {
30
+ selector: string;
31
+ }) => Promise<{
32
+ content: {
33
+ type: "text";
34
+ text: string;
35
+ }[];
36
+ }>;
37
+ };
38
+ fill: {
39
+ description: string;
40
+ schema: {
41
+ selector: z.ZodString;
42
+ value: z.ZodString;
43
+ };
44
+ handler: (conn: BrowserConnection, args: {
45
+ selector: string;
46
+ value: string;
47
+ }) => Promise<{
48
+ content: {
49
+ type: "text";
50
+ text: string;
51
+ }[];
52
+ }>;
53
+ };
54
+ hover: {
55
+ description: string;
56
+ schema: {
57
+ selector: z.ZodString;
58
+ };
59
+ handler: (conn: BrowserConnection, args: {
60
+ selector: string;
61
+ }) => Promise<{
62
+ content: {
63
+ type: "text";
64
+ text: string;
65
+ }[];
66
+ }>;
67
+ };
68
+ press_key: {
69
+ description: string;
70
+ schema: {
71
+ key: z.ZodString;
72
+ };
73
+ handler: (conn: BrowserConnection, args: {
74
+ key: string;
75
+ }) => Promise<{
76
+ content: {
77
+ type: "text";
78
+ text: string;
79
+ }[];
80
+ }>;
81
+ };
82
+ take_screenshot: {
83
+ description: string;
84
+ schema: {
85
+ fullPage: z.ZodOptional<z.ZodBoolean>;
86
+ };
87
+ handler: (conn: BrowserConnection, args: {
88
+ fullPage?: boolean;
89
+ }) => Promise<{
90
+ content: {
91
+ type: "image";
92
+ data: string;
93
+ mimeType: string;
94
+ }[];
95
+ }>;
96
+ };
97
+ take_snapshot: {
98
+ description: string;
99
+ schema: {};
100
+ handler: (conn: BrowserConnection, _args: Record<string, never>) => Promise<{
101
+ content: {
102
+ type: "text";
103
+ text: string;
104
+ }[];
105
+ }>;
106
+ };
107
+ evaluate_script: {
108
+ description: string;
109
+ schema: {
110
+ expression: z.ZodString;
111
+ };
112
+ handler: (conn: BrowserConnection, args: {
113
+ expression: string;
114
+ }) => Promise<{
115
+ content: {
116
+ type: "text";
117
+ text: string;
118
+ }[];
119
+ }>;
120
+ };
121
+ list_pages: {
122
+ description: string;
123
+ schema: {};
124
+ handler: (conn: BrowserConnection, _args: Record<string, never>) => Promise<{
125
+ content: {
126
+ type: "text";
127
+ text: string;
128
+ }[];
129
+ }>;
130
+ };
131
+ new_page: {
132
+ description: string;
133
+ schema: {
134
+ url: z.ZodOptional<z.ZodString>;
135
+ };
136
+ handler: (conn: BrowserConnection, args: {
137
+ url?: string;
138
+ }) => Promise<{
139
+ content: {
140
+ type: "text";
141
+ text: string;
142
+ }[];
143
+ }>;
144
+ };
145
+ close_page: {
146
+ description: string;
147
+ schema: {};
148
+ handler: (conn: BrowserConnection, _args: Record<string, never>) => Promise<{
149
+ content: {
150
+ type: "text";
151
+ text: string;
152
+ }[];
153
+ }>;
154
+ };
155
+ select_page: {
156
+ description: string;
157
+ schema: {
158
+ index: z.ZodNumber;
159
+ };
160
+ handler: (conn: BrowserConnection, args: {
161
+ index: number;
162
+ }) => Promise<{
163
+ content: {
164
+ type: "text";
165
+ text: string;
166
+ }[];
167
+ isError: boolean;
168
+ } | {
169
+ content: {
170
+ type: "text";
171
+ text: string;
172
+ }[];
173
+ isError?: undefined;
174
+ }>;
175
+ };
176
+ wait_for: {
177
+ description: string;
178
+ schema: {
179
+ selector: z.ZodOptional<z.ZodString>;
180
+ timeout: z.ZodOptional<z.ZodNumber>;
181
+ };
182
+ handler: (conn: BrowserConnection, args: {
183
+ selector?: string;
184
+ timeout?: number;
185
+ }) => Promise<{
186
+ content: {
187
+ type: "text";
188
+ text: string;
189
+ }[];
190
+ }>;
191
+ };
192
+ emulate: {
193
+ description: string;
194
+ schema: {
195
+ device: z.ZodString;
196
+ };
197
+ handler: (conn: BrowserConnection, args: {
198
+ device: string;
199
+ }) => Promise<{
200
+ content: {
201
+ type: "text";
202
+ text: string;
203
+ }[];
204
+ isError: boolean;
205
+ } | {
206
+ content: {
207
+ type: "text";
208
+ text: string;
209
+ }[];
210
+ isError?: undefined;
211
+ }>;
212
+ };
213
+ resize_page: {
214
+ description: string;
215
+ schema: {
216
+ width: z.ZodNumber;
217
+ height: z.ZodNumber;
218
+ };
219
+ handler: (conn: BrowserConnection, args: {
220
+ width: number;
221
+ height: number;
222
+ }) => Promise<{
223
+ content: {
224
+ type: "text";
225
+ text: string;
226
+ }[];
227
+ }>;
228
+ };
229
+ };
230
+ //# sourceMappingURL=tools.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"tools.d.ts","sourceRoot":"","sources":["../../src/mcp/tools.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,CAAC,EAAE,MAAM,KAAK,CAAC;AACxB,OAAO,KAAK,EAAE,OAAO,EAAE,IAAI,EAAE,cAAc,EAAE,MAAM,iBAAiB,CAAC;AAErE,MAAM,WAAW,iBAAiB;IAChC,OAAO,EAAE,OAAO,CAAC;IACjB,OAAO,EAAE,cAAc,CAAC;IACxB,UAAU,EAAE,IAAI,CAAC;IACjB,KAAK,EAAE,IAAI,EAAE,CAAC;CACf;AAUD,wBAAgB,WAAW;;;;;;wBAKC,iBAAiB,QAAQ;YAAE,GAAG,EAAE,MAAM,CAAA;SAAE;;;;;;;;;;;;wBAUxC,iBAAiB,QAAQ;YAAE,QAAQ,EAAE,MAAM,CAAA;SAAE;;;;;;;;;;;;;wBAa7C,iBAAiB,QAAQ;YAAE,QAAQ,EAAE,MAAM,CAAC;YAAC,KAAK,EAAE,MAAM,CAAA;SAAE;;;;;;;;;;;;wBAU5D,iBAAiB,QAAQ;YAAE,QAAQ,EAAE,MAAM,CAAA;SAAE;;;;;;;;;;;;wBAU7C,iBAAiB,QAAQ;YAAE,GAAG,EAAE,MAAM,CAAA;SAAE;;;;;;;;;;;;wBAYxC,iBAAiB,QAAQ;YAAE,QAAQ,CAAC,EAAE,OAAO,CAAA;SAAE;;;;;;;;;;;wBAmB/C,iBAAiB,SAAS,MAAM,CAAC,MAAM,EAAE,KAAK,CAAC;;;;;;;;;;;;wBA+B/C,iBAAiB,QAAQ;YAAE,UAAU,EAAE,MAAM,CAAA;SAAE;;;;;;;;;;wBAY/C,iBAAiB,SAAS,MAAM,CAAC,MAAM,EAAE,KAAK,CAAC;;;;;;;;;;;;wBAa/C,iBAAiB,QAAQ;YAAE,GAAG,CAAC,EAAE,MAAM,CAAA;SAAE;;;;;;;;;;wBAazC,iBAAiB,SAAS,MAAM,CAAC,MAAM,EAAE,KAAK,CAAC;;;;;;;;;;;;wBAY/C,iBAAiB,QAAQ;YAAE,KAAK,EAAE,MAAM,CAAA;SAAE;;;;;;;;;;;;;;;;;;;;wBAiB1C,iBAAiB,QAAQ;YAAE,QAAQ,CAAC,EAAE,MAAM,CAAC;YAAC,OAAO,CAAC,EAAE,MAAM,CAAA;SAAE;;;;;;;;;;;;wBAkBhE,iBAAiB,QAAQ;YAAE,MAAM,EAAE,MAAM,CAAA;SAAE;;;;;;;;;;;;;;;;;;;;wBAmB3C,iBAAiB,QAAQ;YAAE,KAAK,EAAE,MAAM,CAAC;YAAC,MAAM,EAAE,MAAM,CAAA;SAAE;;;;;;;EAOrF"}
@@ -0,0 +1,219 @@
1
+ import { z } from 'zod';
2
+ async function getActivePage(conn) {
3
+ if (conn.activePage.isClosed()) {
4
+ const pages = conn.context.pages();
5
+ conn.activePage = pages[pages.length - 1] || await conn.context.newPage();
6
+ }
7
+ return conn.activePage;
8
+ }
9
+ export function defineTools() {
10
+ return {
11
+ navigate_page: {
12
+ description: '导航到指定 URL',
13
+ schema: { url: z.string().describe('目标 URL') },
14
+ handler: async (conn, args) => {
15
+ const page = await getActivePage(conn);
16
+ await page.goto(args.url, { waitUntil: 'domcontentloaded', timeout: 30_000 });
17
+ return { content: [{ type: 'text', text: `已导航到 ${page.url()}` }] };
18
+ },
19
+ },
20
+ click: {
21
+ description: '点击页面元素',
22
+ schema: { selector: z.string().describe('CSS 选择器') },
23
+ handler: async (conn, args) => {
24
+ const page = await getActivePage(conn);
25
+ await page.click(args.selector, { timeout: 10_000 });
26
+ return { content: [{ type: 'text', text: `已点击 ${args.selector}` }] };
27
+ },
28
+ },
29
+ fill: {
30
+ description: '填写输入框',
31
+ schema: {
32
+ selector: z.string().describe('CSS 选择器'),
33
+ value: z.string().describe('填入的值'),
34
+ },
35
+ handler: async (conn, args) => {
36
+ const page = await getActivePage(conn);
37
+ await page.fill(args.selector, args.value, { timeout: 10_000 });
38
+ return { content: [{ type: 'text', text: `已填写 ${args.selector}` }] };
39
+ },
40
+ },
41
+ hover: {
42
+ description: '悬停在元素上',
43
+ schema: { selector: z.string().describe('CSS 选择器') },
44
+ handler: async (conn, args) => {
45
+ const page = await getActivePage(conn);
46
+ await page.hover(args.selector, { timeout: 10_000 });
47
+ return { content: [{ type: 'text', text: `已悬停 ${args.selector}` }] };
48
+ },
49
+ },
50
+ press_key: {
51
+ description: '按下键盘按键',
52
+ schema: { key: z.string().describe('按键名称,如 Enter, Tab, Escape, ArrowDown') },
53
+ handler: async (conn, args) => {
54
+ const page = await getActivePage(conn);
55
+ await page.keyboard.press(args.key);
56
+ return { content: [{ type: 'text', text: `已按下 ${args.key}` }] };
57
+ },
58
+ },
59
+ take_screenshot: {
60
+ description: '截取当前页面截图',
61
+ schema: {
62
+ fullPage: z.boolean().optional().describe('是否截取整个页面'),
63
+ },
64
+ handler: async (conn, args) => {
65
+ const page = await getActivePage(conn);
66
+ const buffer = await page.screenshot({
67
+ fullPage: args.fullPage ?? false,
68
+ type: 'png',
69
+ });
70
+ return {
71
+ content: [{
72
+ type: 'image',
73
+ data: buffer.toString('base64'),
74
+ mimeType: 'image/png',
75
+ }],
76
+ };
77
+ },
78
+ },
79
+ take_snapshot: {
80
+ description: '获取页面 DOM 快照(精简的 HTML 结构),用于理解页面结构',
81
+ schema: {},
82
+ handler: async (conn, _args) => {
83
+ const page = await getActivePage(conn);
84
+ const snapshot = await page.evaluate(() => {
85
+ function walk(el, depth) {
86
+ if (depth > 5)
87
+ return '';
88
+ const tag = el.tagName.toLowerCase();
89
+ const id = el.id ? `#${el.id}` : '';
90
+ const cls = el.className && typeof el.className === 'string'
91
+ ? '.' + el.className.trim().split(/\s+/).slice(0, 2).join('.')
92
+ : '';
93
+ const text = el.childNodes.length === 1 && el.childNodes[0].nodeType === 3
94
+ ? el.textContent?.trim().slice(0, 50) || ''
95
+ : '';
96
+ const indent = ' '.repeat(depth);
97
+ let result = `${indent}<${tag}${id}${cls}>${text ? ' ' + text : ''}\n`;
98
+ for (let i = 0; i < el.children.length; i++) {
99
+ result += walk(el.children[i], depth + 1);
100
+ }
101
+ return result;
102
+ }
103
+ return walk(document.body, 0);
104
+ });
105
+ return {
106
+ content: [{ type: 'text', text: snapshot }],
107
+ };
108
+ },
109
+ },
110
+ evaluate_script: {
111
+ description: '在页面中执行 JavaScript 表达式',
112
+ schema: { expression: z.string().describe('JavaScript 表达式') },
113
+ handler: async (conn, args) => {
114
+ const page = await getActivePage(conn);
115
+ const result = await page.evaluate(args.expression);
116
+ return {
117
+ content: [{ type: 'text', text: typeof result === 'string' ? result : JSON.stringify(result, null, 2) }],
118
+ };
119
+ },
120
+ },
121
+ list_pages: {
122
+ description: '列出所有打开的标签页',
123
+ schema: {},
124
+ handler: async (conn, _args) => {
125
+ const pages = conn.context.pages();
126
+ const list = await Promise.all(pages.map(async (p, i) => {
127
+ const isActive = p === conn.activePage;
128
+ return `${isActive ? '* ' : ' '}[${i}] ${p.url()} - ${await p.title()}`;
129
+ }));
130
+ return { content: [{ type: 'text', text: list.join('\n') || '无打开的页面' }] };
131
+ },
132
+ },
133
+ new_page: {
134
+ description: '打开新标签页',
135
+ schema: { url: z.string().optional().describe('可选的初始 URL') },
136
+ handler: async (conn, args) => {
137
+ const page = await conn.context.newPage();
138
+ conn.activePage = page;
139
+ if (args.url) {
140
+ await page.goto(args.url, { waitUntil: 'domcontentloaded', timeout: 30_000 });
141
+ }
142
+ return { content: [{ type: 'text', text: `已打开新标签页${args.url ? ': ' + page.url() : ''}` }] };
143
+ },
144
+ },
145
+ close_page: {
146
+ description: '关闭当前标签页',
147
+ schema: {},
148
+ handler: async (conn, _args) => {
149
+ const page = await getActivePage(conn);
150
+ await page.close();
151
+ const pages = conn.context.pages();
152
+ conn.activePage = pages[pages.length - 1] || await conn.context.newPage();
153
+ return { content: [{ type: 'text', text: '已关闭标签页' }] };
154
+ },
155
+ },
156
+ select_page: {
157
+ description: '切换到指定标签页',
158
+ schema: { index: z.number().describe('标签页索引(从 0 开始)') },
159
+ handler: async (conn, args) => {
160
+ const pages = conn.context.pages();
161
+ if (args.index < 0 || args.index >= pages.length) {
162
+ return { content: [{ type: 'text', text: `无效索引 ${args.index},共 ${pages.length} 个标签页` }], isError: true };
163
+ }
164
+ conn.activePage = pages[args.index];
165
+ await conn.activePage.bringToFront();
166
+ return { content: [{ type: 'text', text: `已切换到标签页 [${args.index}]: ${conn.activePage.url()}` }] };
167
+ },
168
+ },
169
+ wait_for: {
170
+ description: '等待元素出现或指定时间',
171
+ schema: {
172
+ selector: z.string().optional().describe('CSS 选择器(不传则等待指定毫秒)'),
173
+ timeout: z.number().optional().describe('超时毫秒数,默认 5000'),
174
+ },
175
+ handler: async (conn, args) => {
176
+ const page = await getActivePage(conn);
177
+ const timeout = args.timeout ?? 5000;
178
+ if (args.selector) {
179
+ await page.waitForSelector(args.selector, { timeout });
180
+ return { content: [{ type: 'text', text: `元素 ${args.selector} 已出现` }] };
181
+ }
182
+ else {
183
+ await page.waitForTimeout(timeout);
184
+ return { content: [{ type: 'text', text: `已等待 ${timeout}ms` }] };
185
+ }
186
+ },
187
+ },
188
+ emulate: {
189
+ description: '模拟移动设备',
190
+ schema: {
191
+ device: z.string().describe('设备名称,如 "iPhone 14 Pro Max", "Pixel 7"'),
192
+ },
193
+ handler: async (conn, args) => {
194
+ const pw = await import('playwright-core');
195
+ const devices = pw.devices;
196
+ const device = devices[args.device];
197
+ if (!device) {
198
+ return { content: [{ type: 'text', text: `未知设备: ${args.device}` }], isError: true };
199
+ }
200
+ const page = await getActivePage(conn);
201
+ await page.setViewportSize(device.viewport);
202
+ return { content: [{ type: 'text', text: `已模拟 ${args.device} (${device.viewport.width}x${device.viewport.height})` }] };
203
+ },
204
+ },
205
+ resize_page: {
206
+ description: '调整浏览器视口大小',
207
+ schema: {
208
+ width: z.number().describe('宽度(像素)'),
209
+ height: z.number().describe('高度(像素)'),
210
+ },
211
+ handler: async (conn, args) => {
212
+ const page = await getActivePage(conn);
213
+ await page.setViewportSize({ width: args.width, height: args.height });
214
+ return { content: [{ type: 'text', text: `视口已调整为 ${args.width}x${args.height}` }] };
215
+ },
216
+ },
217
+ };
218
+ }
219
+ //# sourceMappingURL=tools.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"tools.js","sourceRoot":"","sources":["../../src/mcp/tools.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,15 @@
1
+ import type { SessionManager } from '../session/session-manager.js';
2
+ import type { ServerConfig } from '../utils/types.js';
3
+ export declare class McpServer {
4
+ private config;
5
+ private sessionManager;
6
+ private server;
7
+ private connectionPool;
8
+ private transports;
9
+ constructor(config: ServerConfig, sessionManager: SessionManager);
10
+ start(): Promise<void>;
11
+ stop(): Promise<void>;
12
+ private handleRequest;
13
+ private createTransportForSession;
14
+ }
15
+ //# sourceMappingURL=mcp-server.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"mcp-server.d.ts","sourceRoot":"","sources":["../../src/server/mcp-server.ts"],"names":[],"mappings":"AAMA,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,cAAc,CAAiB;IACvC,OAAO,CAAC,UAAU,CAAyD;gBAE/D,MAAM,EAAE,YAAY,EAAE,cAAc,EAAE,cAAc;IAM1D,KAAK,IAAI,OAAO,CAAC,IAAI,CAAC;IAmBtB,IAAI,IAAI,OAAO,CAAC,IAAI,CAAC;YAiBb,aAAa;YA2Bb,yBAAyB;CA4BxC"}
@@ -0,0 +1,96 @@
1
+ import * as http from 'node:http';
2
+ import { McpServer as McpSdkServer } from '@modelcontextprotocol/sdk/server/mcp.js';
3
+ import { StreamableHTTPServerTransport } from '@modelcontextprotocol/sdk/server/streamableHttp.js';
4
+ import { log } from '../utils/logger.js';
5
+ import { ConnectionPool } from '../mcp/connection-pool.js';
6
+ import { defineTools } from '../mcp/tools.js';
7
+ export class McpServer {
8
+ config;
9
+ sessionManager;
10
+ server = null;
11
+ connectionPool;
12
+ transports = new Map();
13
+ constructor(config, sessionManager) {
14
+ this.config = config;
15
+ this.sessionManager = sessionManager;
16
+ this.connectionPool = new ConnectionPool();
17
+ }
18
+ async start() {
19
+ this.server = http.createServer((req, res) => {
20
+ this.handleRequest(req, res).catch((err) => {
21
+ log.error('MCP 请求处理异常:', err instanceof Error ? err.message : err);
22
+ if (!res.headersSent) {
23
+ res.writeHead(500);
24
+ res.end(JSON.stringify({ error: 'Internal Server Error' }));
25
+ }
26
+ });
27
+ });
28
+ return new Promise((resolve) => {
29
+ this.server.listen(this.config.mcpPort, this.config.host, () => {
30
+ log.info(`MCP 服务已启动: http://${this.config.host}:${this.config.mcpPort}`);
31
+ resolve();
32
+ });
33
+ });
34
+ }
35
+ async stop() {
36
+ for (const [, transport] of this.transports) {
37
+ await transport.close();
38
+ }
39
+ this.transports.clear();
40
+ await this.connectionPool.disconnectAll();
41
+ if (this.server) {
42
+ return new Promise((resolve) => {
43
+ this.server.close(() => {
44
+ log.info('MCP 服务已停止');
45
+ resolve();
46
+ });
47
+ });
48
+ }
49
+ }
50
+ async handleRequest(req, res) {
51
+ const url = new URL(req.url || '/', `http://${req.headers.host}`);
52
+ const match = url.pathname.match(/^\/mcp\/([a-f0-9]+)$/);
53
+ if (!match) {
54
+ res.writeHead(404);
55
+ res.end(JSON.stringify({ error: 'Not Found. Use /mcp/:sessionId' }));
56
+ return;
57
+ }
58
+ const sessionId = match[1];
59
+ const session = this.sessionManager.getSession(sessionId);
60
+ if (!session) {
61
+ res.writeHead(404);
62
+ res.end(JSON.stringify({ error: `Session ${sessionId} not found` }));
63
+ return;
64
+ }
65
+ let transport = this.transports.get(sessionId);
66
+ if (!transport) {
67
+ transport = await this.createTransportForSession(sessionId, session.cdpPort);
68
+ this.transports.set(sessionId, transport);
69
+ }
70
+ await transport.handleRequest(req, res);
71
+ }
72
+ async createTransportForSession(sessionId, cdpPort) {
73
+ const transport = new StreamableHTTPServerTransport({
74
+ sessionIdGenerator: undefined,
75
+ });
76
+ const mcpServer = new McpSdkServer({ name: `browser-server/${sessionId}`, version: '0.1.0' }, { capabilities: { tools: {} } });
77
+ const tools = defineTools();
78
+ for (const [name, tool] of Object.entries(tools)) {
79
+ mcpServer.tool(name, tool.description, tool.schema, async (args) => {
80
+ try {
81
+ const conn = await this.connectionPool.getConnection(sessionId, cdpPort);
82
+ return await tool.handler(conn, args);
83
+ }
84
+ catch (err) {
85
+ const msg = err instanceof Error ? err.message : String(err);
86
+ log.error(`[${sessionId}] 工具 ${name} 执行失败:`, msg);
87
+ return { content: [{ type: 'text', text: `错误: ${msg}` }], isError: true };
88
+ }
89
+ });
90
+ }
91
+ await mcpServer.connect(transport);
92
+ log.info(`[${sessionId}] MCP transport 已创建`);
93
+ return transport;
94
+ }
95
+ }
96
+ //# sourceMappingURL=mcp-server.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"mcp-server.js","sourceRoot":"","sources":["../../src/server/mcp-server.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,7 @@
1
+ import { WebSocket } from 'ws';
2
+ /**
3
+ * 将 incoming WebSocket 连接代理到本地 VNC (x11vnc) 的 TCP 端口。
4
+ * noVNC 客户端通过 WebSocket 连接此代理,代理将数据透传到 VNC server。
5
+ */
6
+ export declare function proxyVncWebSocket(ws: WebSocket, vncPort: number, sessionId: string): void;
7
+ //# sourceMappingURL=vnc-proxy.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"vnc-proxy.d.ts","sourceRoot":"","sources":["../../src/server/vnc-proxy.ts"],"names":[],"mappings":"AAEA,OAAO,EAAmB,SAAS,EAAE,MAAM,IAAI,CAAC;AAGhD;;;GAGG;AACH,wBAAgB,iBAAiB,CAC/B,EAAE,EAAE,SAAS,EACb,OAAO,EAAE,MAAM,EACf,SAAS,EAAE,MAAM,GAChB,IAAI,CAoCN"}
@@ -0,0 +1,38 @@
1
+ import { Socket } from 'node:net';
2
+ import { WebSocket } from 'ws';
3
+ import { log } from '../utils/logger.js';
4
+ /**
5
+ * 将 incoming WebSocket 连接代理到本地 VNC (x11vnc) 的 TCP 端口。
6
+ * noVNC 客户端通过 WebSocket 连接此代理,代理将数据透传到 VNC server。
7
+ */
8
+ export function proxyVncWebSocket(ws, vncPort, sessionId) {
9
+ const tcp = new Socket();
10
+ tcp.connect(vncPort, '127.0.0.1', () => {
11
+ log.debug(`[${sessionId}] VNC 代理已连接到 localhost:${vncPort}`);
12
+ });
13
+ tcp.on('data', (data) => {
14
+ if (ws.readyState === WebSocket.OPEN) {
15
+ ws.send(data);
16
+ }
17
+ });
18
+ ws.on('message', (data) => {
19
+ if (tcp.writable) {
20
+ tcp.write(data);
21
+ }
22
+ });
23
+ tcp.on('error', (err) => {
24
+ log.error(`[${sessionId}] VNC TCP 错误:`, err.message);
25
+ ws.close();
26
+ });
27
+ tcp.on('close', () => {
28
+ ws.close();
29
+ });
30
+ ws.on('close', () => {
31
+ tcp.destroy();
32
+ });
33
+ ws.on('error', (err) => {
34
+ log.error(`[${sessionId}] VNC WebSocket 错误:`, err.message);
35
+ tcp.destroy();
36
+ });
37
+ }
38
+ //# sourceMappingURL=vnc-proxy.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"vnc-proxy.js","sourceRoot":"","sources":["../../src/server/vnc-proxy.ts"],"names":[],"mappings":"AACA,OAAO,EAAE,MAAM,EAAE,MAAM,UAAU,CAAC;AAClC,OAAO,EAAmB,SAAS,EAAE,MAAM,IAAI,CAAC;AAChD,OAAO,EAAE,GAAG,EAAE,MAAM,oBAAoB,CAAC;AAEzC;;;GAGG;AACH,MAAM,UAAU,iBAAiB,CAC/B,EAAa,EACb,OAAe,EACf,SAAiB;IAEjB,MAAM,GAAG,GAAG,IAAI,MAAM,EAAE,CAAC;IAEzB,GAAG,CAAC,OAAO,CAAC,OAAO,EAAE,WAAW,EAAE,GAAG,EAAE;QACrC,GAAG,CAAC,KAAK,CAAC,IAAI,SAAS,0BAA0B,OAAO,EAAE,CAAC,CAAC;IAC9D,CAAC,CAAC,CAAC;IAEH,GAAG,CAAC,EAAE,CAAC,MAAM,EAAE,CAAC,IAAI,EAAE,EAAE;QACtB,IAAI,EAAE,CAAC,UAAU,KAAK,SAAS,CAAC,IAAI,EAAE,CAAC;YACrC,EAAE,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;QAChB,CAAC;IACH,CAAC,CAAC,CAAC;IAEH,EAAE,CAAC,EAAE,CAAC,SAAS,EAAE,CAAC,IAAI,EAAE,EAAE;QACxB,IAAI,GAAG,CAAC,QAAQ,EAAE,CAAC;YACjB,GAAG,CAAC,KAAK,CAAC,IAAc,CAAC,CAAC;QAC5B,CAAC;IACH,CAAC,CAAC,CAAC;IAEH,GAAG,CAAC,EAAE,CAAC,OAAO,EAAE,CAAC,GAAG,EAAE,EAAE;QACtB,GAAG,CAAC,KAAK,CAAC,IAAI,SAAS,eAAe,EAAE,GAAG,CAAC,OAAO,CAAC,CAAC;QACrD,EAAE,CAAC,KAAK,EAAE,CAAC;IACb,CAAC,CAAC,CAAC;IAEH,GAAG,CAAC,EAAE,CAAC,OAAO,EAAE,GAAG,EAAE;QACnB,EAAE,CAAC,KAAK,EAAE,CAAC;IACb,CAAC,CAAC,CAAC;IAEH,EAAE,CAAC,EAAE,CAAC,OAAO,EAAE,GAAG,EAAE;QAClB,GAAG,CAAC,OAAO,EAAE,CAAC;IAChB,CAAC,CAAC,CAAC;IAEH,EAAE,CAAC,EAAE,CAAC,OAAO,EAAE,CAAC,GAAG,EAAE,EAAE;QACrB,GAAG,CAAC,KAAK,CAAC,IAAI,SAAS,qBAAqB,EAAE,GAAG,CAAC,OAAO,CAAC,CAAC;QAC3D,GAAG,CAAC,OAAO,EAAE,CAAC;IAChB,CAAC,CAAC,CAAC;AACL,CAAC"}
@@ -0,0 +1,4 @@
1
+ import type { SessionInfo, ServerConfig } from '../utils/types.js';
2
+ export declare function renderHomePage(sessions: SessionInfo[], config: ServerConfig): string;
3
+ export declare function renderViewerPage(session: SessionInfo, config: ServerConfig): string;
4
+ //# sourceMappingURL=web-pages.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"web-pages.d.ts","sourceRoot":"","sources":["../../src/server/web-pages.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,WAAW,EAAE,YAAY,EAAE,MAAM,mBAAmB,CAAC;AA2DnE,wBAAgB,cAAc,CAAC,QAAQ,EAAE,WAAW,EAAE,EAAE,MAAM,EAAE,YAAY,GAAG,MAAM,CAqDpF;AAED,wBAAgB,gBAAgB,CAAC,OAAO,EAAE,WAAW,EAAE,MAAM,EAAE,YAAY,GAAG,MAAM,CAiDnF"}