mcp-accessibility-scanner 1.1.2 → 2.0.1
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 → README.md} +49 -0
- package/lib/browserContextFactory.js +83 -99
- package/lib/browserContextFactory.js.map +1 -1
- package/lib/browserServerBackend.js +78 -0
- package/lib/browserServerBackend.js.map +1 -0
- package/lib/config.js +89 -19
- package/lib/config.js.map +1 -1
- package/lib/context.js +109 -174
- package/lib/context.js.map +1 -1
- package/lib/extension/cdpRelay.js +359 -0
- package/lib/extension/cdpRelay.js.map +1 -0
- package/lib/extension/extensionContextFactory.js +57 -0
- package/lib/extension/extensionContextFactory.js.map +1 -0
- package/lib/extension/protocol.js +19 -0
- package/lib/extension/protocol.js.map +1 -0
- package/lib/index.js +7 -3
- package/lib/index.js.map +1 -1
- package/lib/{transport.js → mcp/http.js} +65 -63
- package/lib/mcp/http.js.map +1 -0
- package/lib/mcp/inProcessTransport.js +73 -0
- package/lib/mcp/inProcessTransport.js.map +1 -0
- package/lib/mcp/manualPromise.js.map +1 -0
- package/lib/mcp/mdb.js +199 -0
- package/lib/mcp/mdb.js.map +1 -0
- package/lib/mcp/proxyBackend.js +105 -0
- package/lib/mcp/proxyBackend.js.map +1 -0
- package/lib/mcp/server.js +124 -0
- package/lib/mcp/server.js.map +1 -0
- package/lib/mcp/tool.js +33 -0
- package/lib/mcp/tool.js.map +1 -0
- package/lib/program.js +83 -26
- package/lib/program.js.map +1 -1
- package/lib/response.js +166 -0
- package/lib/response.js.map +1 -0
- package/lib/sessionLog.js +122 -0
- package/lib/sessionLog.js.map +1 -0
- package/lib/tab.js +173 -25
- package/lib/tab.js.map +1 -1
- package/lib/tools/common.js +12 -25
- package/lib/tools/common.js.map +1 -1
- package/lib/tools/console.js +4 -15
- package/lib/tools/console.js.map +1 -1
- package/lib/tools/dialogs.js +14 -19
- package/lib/tools/dialogs.js.map +1 -1
- package/lib/tools/evaluate.js +54 -0
- package/lib/tools/evaluate.js.map +1 -0
- package/lib/tools/files.js +12 -19
- package/lib/tools/files.js.map +1 -1
- package/lib/tools/form.js +58 -0
- package/lib/tools/form.js.map +1 -0
- package/lib/tools/install.js +7 -10
- package/lib/tools/install.js.map +1 -1
- package/lib/tools/keyboard.js +49 -17
- package/lib/tools/keyboard.js.map +1 -1
- package/lib/tools/mouse.js +100 -0
- package/lib/tools/mouse.js.map +1 -0
- package/lib/tools/navigate.js +13 -52
- package/lib/tools/navigate.js.map +1 -1
- package/lib/tools/network.js +5 -15
- package/lib/tools/network.js.map +1 -1
- package/lib/tools/pdf.js +8 -17
- package/lib/tools/pdf.js.map +1 -1
- package/lib/tools/screenshot.js +36 -34
- package/lib/tools/screenshot.js.map +1 -1
- package/lib/tools/snapshot.js +103 -154
- package/lib/tools/snapshot.js.map +1 -1
- package/lib/tools/tabs.js +66 -81
- package/lib/tools/tabs.js.map +1 -1
- package/lib/tools/tool.js +15 -0
- package/lib/tools/tool.js.map +1 -1
- package/lib/tools/utils.js +7 -13
- package/lib/tools/utils.js.map +1 -1
- package/lib/tools/verify.js +138 -0
- package/lib/tools/verify.js.map +1 -0
- package/lib/tools/wait.js +11 -15
- package/lib/tools/wait.js.map +1 -1
- package/lib/tools.js +19 -23
- package/lib/tools.js.map +1 -1
- package/lib/{javascript.js → utils/codegen.js} +2 -2
- package/lib/utils/codegen.js.map +1 -0
- package/lib/{fileUtils.js → utils/fileUtils.js} +6 -2
- package/lib/utils/fileUtils.js.map +1 -0
- package/lib/utils/guid.js +23 -0
- package/lib/utils/guid.js.map +1 -0
- package/lib/utils/log.js +22 -0
- package/lib/utils/log.js.map +1 -0
- package/lib/{package.js → utils/package.js} +4 -4
- package/lib/utils/package.js.map +1 -0
- package/lib/vscode/host.js +129 -0
- package/lib/vscode/host.js.map +1 -0
- package/lib/vscode/main.js +63 -0
- package/lib/vscode/main.js.map +1 -0
- package/package.json +20 -20
- package/lib/browserServer.js +0 -152
- package/lib/browserServer.js.map +0 -1
- package/lib/connection.js +0 -83
- package/lib/connection.js.map +0 -1
- package/lib/fileUtils.js.map +0 -1
- package/lib/httpServer.js +0 -202
- package/lib/httpServer.js.map +0 -1
- package/lib/javascript.js.map +0 -1
- package/lib/manualPromise.js.map +0 -1
- package/lib/package.js.map +0 -1
- package/lib/pageSnapshot.js +0 -44
- package/lib/pageSnapshot.js.map +0 -1
- package/lib/server.js +0 -49
- package/lib/server.js.map +0 -1
- package/lib/tools/vision.js +0 -190
- package/lib/tools/vision.js.map +0 -1
- package/lib/transport.js.map +0 -1
- /package/lib/{manualPromise.js → mcp/manualPromise.js} +0 -0
package/{Readme.md → README.md}
RENAMED
|
@@ -66,6 +66,55 @@ Here's the Claude Desktop configuration:
|
|
|
66
66
|
}
|
|
67
67
|
```
|
|
68
68
|
|
|
69
|
+
### Advanced Configuration
|
|
70
|
+
|
|
71
|
+
You can pass a configuration file to customize Playwright behavior:
|
|
72
|
+
|
|
73
|
+
```json
|
|
74
|
+
{
|
|
75
|
+
"mcpServers": {
|
|
76
|
+
"accessibility-scanner": {
|
|
77
|
+
"command": "npx",
|
|
78
|
+
"args": ["-y", "mcp-accessibility-scanner", "--config", "/path/to/config.json"]
|
|
79
|
+
}
|
|
80
|
+
}
|
|
81
|
+
}
|
|
82
|
+
```
|
|
83
|
+
|
|
84
|
+
#### Configuration Options
|
|
85
|
+
|
|
86
|
+
Create a `config.json` file with the following options:
|
|
87
|
+
|
|
88
|
+
```json
|
|
89
|
+
{
|
|
90
|
+
"browser": {
|
|
91
|
+
"browserName": "chromium",
|
|
92
|
+
"launchOptions": {
|
|
93
|
+
"headless": true,
|
|
94
|
+
"channel": "chrome"
|
|
95
|
+
}
|
|
96
|
+
},
|
|
97
|
+
"timeouts": {
|
|
98
|
+
"navigationTimeout": 60000,
|
|
99
|
+
"defaultTimeout": 5000
|
|
100
|
+
},
|
|
101
|
+
"network": {
|
|
102
|
+
"allowedOrigins": ["example.com", "trusted-site.com"],
|
|
103
|
+
"blockedOrigins": ["ads.example.com"]
|
|
104
|
+
}
|
|
105
|
+
}
|
|
106
|
+
```
|
|
107
|
+
|
|
108
|
+
**Available Options:**
|
|
109
|
+
|
|
110
|
+
- `browser.browserName`: Browser to use (`chromium`, `firefox`, `webkit`)
|
|
111
|
+
- `browser.launchOptions.headless`: Run browser in headless mode (default: `true` on Linux without display, `false` otherwise)
|
|
112
|
+
- `browser.launchOptions.channel`: Browser channel (`chrome`, `chrome-beta`, `msedge`, etc.)
|
|
113
|
+
- `timeouts.navigationTimeout`: Maximum time for page navigation in milliseconds (default: `60000`)
|
|
114
|
+
- `timeouts.defaultTimeout`: Default timeout for Playwright operations in milliseconds (default: `5000`)
|
|
115
|
+
- `network.allowedOrigins`: List of origins to allow (blocks all others if specified)
|
|
116
|
+
- `network.blockedOrigins`: List of origins to block
|
|
117
|
+
|
|
69
118
|
## Available Tools
|
|
70
119
|
|
|
71
120
|
The MCP server provides comprehensive browser automation and accessibility scanning tools:
|
|
@@ -13,38 +13,39 @@
|
|
|
13
13
|
* See the License for the specific language governing permissions and
|
|
14
14
|
* limitations under the License.
|
|
15
15
|
*/
|
|
16
|
-
import fs from '
|
|
17
|
-
import net from '
|
|
18
|
-
import path from '
|
|
19
|
-
import os from 'node:os';
|
|
20
|
-
import debug from 'debug';
|
|
16
|
+
import fs from 'fs';
|
|
17
|
+
import net from 'net';
|
|
18
|
+
import path from 'path';
|
|
21
19
|
import * as playwright from 'playwright';
|
|
22
|
-
|
|
23
|
-
|
|
24
|
-
|
|
25
|
-
|
|
26
|
-
|
|
27
|
-
|
|
28
|
-
|
|
29
|
-
|
|
30
|
-
|
|
31
|
-
|
|
32
|
-
|
|
33
|
-
|
|
20
|
+
// @ts-ignore
|
|
21
|
+
import { registryDirectory } from 'playwright-core/lib/server/registry/index';
|
|
22
|
+
// @ts-ignore
|
|
23
|
+
import { startTraceViewerServer } from 'playwright-core/lib/server';
|
|
24
|
+
import { logUnhandledError, testDebug } from './utils/log.js';
|
|
25
|
+
import { createHash } from './utils/guid.js';
|
|
26
|
+
import { outputFile } from './config.js';
|
|
27
|
+
export function contextFactory(config) {
|
|
28
|
+
if (config.browser.remoteEndpoint)
|
|
29
|
+
return new RemoteContextFactory(config);
|
|
30
|
+
if (config.browser.cdpEndpoint)
|
|
31
|
+
return new CdpContextFactory(config);
|
|
32
|
+
if (config.browser.isolated)
|
|
33
|
+
return new IsolatedContextFactory(config);
|
|
34
|
+
return new PersistentContextFactory(config);
|
|
34
35
|
}
|
|
35
36
|
class BaseContextFactory {
|
|
36
|
-
|
|
37
|
+
config;
|
|
38
|
+
_logName;
|
|
37
39
|
_browserPromise;
|
|
38
|
-
name
|
|
39
|
-
|
|
40
|
-
this.
|
|
41
|
-
this.browserConfig = browserConfig;
|
|
40
|
+
constructor(name, config) {
|
|
41
|
+
this._logName = name;
|
|
42
|
+
this.config = config;
|
|
42
43
|
}
|
|
43
|
-
async _obtainBrowser() {
|
|
44
|
+
async _obtainBrowser(clientInfo) {
|
|
44
45
|
if (this._browserPromise)
|
|
45
46
|
return this._browserPromise;
|
|
46
|
-
testDebug(`obtain browser (${this.
|
|
47
|
-
this._browserPromise = this._doObtainBrowser();
|
|
47
|
+
testDebug(`obtain browser (${this._logName})`);
|
|
48
|
+
this._browserPromise = this._doObtainBrowser(clientInfo);
|
|
48
49
|
void this._browserPromise.then(browser => {
|
|
49
50
|
browser.on('disconnected', () => {
|
|
50
51
|
this._browserPromise = undefined;
|
|
@@ -54,12 +55,12 @@ class BaseContextFactory {
|
|
|
54
55
|
});
|
|
55
56
|
return this._browserPromise;
|
|
56
57
|
}
|
|
57
|
-
async _doObtainBrowser() {
|
|
58
|
+
async _doObtainBrowser(clientInfo) {
|
|
58
59
|
throw new Error('Not implemented');
|
|
59
60
|
}
|
|
60
|
-
async createContext() {
|
|
61
|
-
testDebug(`create browser context (${this.
|
|
62
|
-
const browser = await this._obtainBrowser();
|
|
61
|
+
async createContext(clientInfo) {
|
|
62
|
+
testDebug(`create browser context (${this._logName})`);
|
|
63
|
+
const browser = await this._obtainBrowser(clientInfo);
|
|
63
64
|
const browserContext = await this._doCreateContext(browser);
|
|
64
65
|
return { browserContext, close: () => this._closeBrowserContext(browserContext, browser) };
|
|
65
66
|
}
|
|
@@ -67,25 +68,26 @@ class BaseContextFactory {
|
|
|
67
68
|
throw new Error('Not implemented');
|
|
68
69
|
}
|
|
69
70
|
async _closeBrowserContext(browserContext, browser) {
|
|
70
|
-
testDebug(`close browser context (${this.
|
|
71
|
+
testDebug(`close browser context (${this._logName})`);
|
|
71
72
|
if (browser.contexts().length === 1)
|
|
72
73
|
this._browserPromise = undefined;
|
|
73
|
-
await browserContext.close().catch(
|
|
74
|
+
await browserContext.close().catch(logUnhandledError);
|
|
74
75
|
if (browser.contexts().length === 0) {
|
|
75
|
-
testDebug(`close browser (${this.
|
|
76
|
-
await browser.close().catch(
|
|
76
|
+
testDebug(`close browser (${this._logName})`);
|
|
77
|
+
await browser.close().catch(logUnhandledError);
|
|
77
78
|
}
|
|
78
79
|
}
|
|
79
80
|
}
|
|
80
81
|
class IsolatedContextFactory extends BaseContextFactory {
|
|
81
|
-
constructor(
|
|
82
|
-
super('isolated',
|
|
82
|
+
constructor(config) {
|
|
83
|
+
super('isolated', config);
|
|
83
84
|
}
|
|
84
|
-
async _doObtainBrowser() {
|
|
85
|
-
await injectCdpPort(this.
|
|
86
|
-
const browserType = playwright[this.
|
|
85
|
+
async _doObtainBrowser(clientInfo) {
|
|
86
|
+
await injectCdpPort(this.config.browser);
|
|
87
|
+
const browserType = playwright[this.config.browser.browserName];
|
|
87
88
|
return browserType.launch({
|
|
88
|
-
|
|
89
|
+
tracesDir: await startTraceServer(this.config, clientInfo.rootPath),
|
|
90
|
+
...this.config.browser.launchOptions,
|
|
89
91
|
handleSIGINT: false,
|
|
90
92
|
handleSIGTERM: false,
|
|
91
93
|
}).catch(error => {
|
|
@@ -95,53 +97,57 @@ class IsolatedContextFactory extends BaseContextFactory {
|
|
|
95
97
|
});
|
|
96
98
|
}
|
|
97
99
|
async _doCreateContext(browser) {
|
|
98
|
-
return browser.newContext(this.
|
|
100
|
+
return browser.newContext(this.config.browser.contextOptions);
|
|
99
101
|
}
|
|
100
102
|
}
|
|
101
103
|
class CdpContextFactory extends BaseContextFactory {
|
|
102
|
-
constructor(
|
|
103
|
-
super('cdp',
|
|
104
|
+
constructor(config) {
|
|
105
|
+
super('cdp', config);
|
|
104
106
|
}
|
|
105
107
|
async _doObtainBrowser() {
|
|
106
|
-
return playwright.chromium.connectOverCDP(this.
|
|
108
|
+
return playwright.chromium.connectOverCDP(this.config.browser.cdpEndpoint);
|
|
107
109
|
}
|
|
108
110
|
async _doCreateContext(browser) {
|
|
109
|
-
return this.
|
|
111
|
+
return this.config.browser.isolated ? await browser.newContext() : browser.contexts()[0];
|
|
110
112
|
}
|
|
111
113
|
}
|
|
112
114
|
class RemoteContextFactory extends BaseContextFactory {
|
|
113
|
-
constructor(
|
|
114
|
-
super('remote',
|
|
115
|
+
constructor(config) {
|
|
116
|
+
super('remote', config);
|
|
115
117
|
}
|
|
116
118
|
async _doObtainBrowser() {
|
|
117
|
-
const url = new URL(this.
|
|
118
|
-
url.searchParams.set('browser', this.
|
|
119
|
-
if (this.
|
|
120
|
-
url.searchParams.set('launch-options', JSON.stringify(this.
|
|
121
|
-
return playwright[this.
|
|
119
|
+
const url = new URL(this.config.browser.remoteEndpoint);
|
|
120
|
+
url.searchParams.set('browser', this.config.browser.browserName);
|
|
121
|
+
if (this.config.browser.launchOptions)
|
|
122
|
+
url.searchParams.set('launch-options', JSON.stringify(this.config.browser.launchOptions));
|
|
123
|
+
return playwright[this.config.browser.browserName].connect(String(url));
|
|
122
124
|
}
|
|
123
125
|
async _doCreateContext(browser) {
|
|
124
126
|
return browser.newContext();
|
|
125
127
|
}
|
|
126
128
|
}
|
|
127
129
|
class PersistentContextFactory {
|
|
128
|
-
|
|
130
|
+
config;
|
|
131
|
+
name = 'persistent';
|
|
132
|
+
description = 'Create a new persistent browser context';
|
|
129
133
|
_userDataDirs = new Set();
|
|
130
|
-
constructor(
|
|
131
|
-
this.
|
|
134
|
+
constructor(config) {
|
|
135
|
+
this.config = config;
|
|
132
136
|
}
|
|
133
|
-
async createContext() {
|
|
134
|
-
await injectCdpPort(this.
|
|
137
|
+
async createContext(clientInfo) {
|
|
138
|
+
await injectCdpPort(this.config.browser);
|
|
135
139
|
testDebug('create browser context (persistent)');
|
|
136
|
-
const userDataDir = this.
|
|
140
|
+
const userDataDir = this.config.browser.userDataDir ?? await this._createUserDataDir(clientInfo.rootPath);
|
|
141
|
+
const tracesDir = await startTraceServer(this.config, clientInfo.rootPath);
|
|
137
142
|
this._userDataDirs.add(userDataDir);
|
|
138
143
|
testDebug('lock user data dir', userDataDir);
|
|
139
|
-
const browserType = playwright[this.
|
|
144
|
+
const browserType = playwright[this.config.browser.browserName];
|
|
140
145
|
for (let i = 0; i < 5; i++) {
|
|
141
146
|
try {
|
|
142
147
|
const browserContext = await browserType.launchPersistentContext(userDataDir, {
|
|
143
|
-
|
|
144
|
-
...this.
|
|
148
|
+
tracesDir,
|
|
149
|
+
...this.config.browser.launchOptions,
|
|
150
|
+
...this.config.browser.contextOptions,
|
|
145
151
|
handleSIGINT: false,
|
|
146
152
|
handleSIGTERM: false,
|
|
147
153
|
});
|
|
@@ -168,49 +174,16 @@ class PersistentContextFactory {
|
|
|
168
174
|
this._userDataDirs.delete(userDataDir);
|
|
169
175
|
testDebug('close browser context complete (persistent)');
|
|
170
176
|
}
|
|
171
|
-
async _createUserDataDir() {
|
|
172
|
-
|
|
173
|
-
|
|
174
|
-
|
|
175
|
-
|
|
176
|
-
|
|
177
|
-
else if (process.platform === 'win32')
|
|
178
|
-
cacheDirectory = process.env.LOCALAPPDATA || path.join(os.homedir(), 'AppData', 'Local');
|
|
179
|
-
else
|
|
180
|
-
throw new Error('Unsupported platform: ' + process.platform);
|
|
181
|
-
const result = path.join(cacheDirectory, 'ms-playwright', `mcp-${this.browserConfig.launchOptions?.channel ?? this.browserConfig?.browserName}-profile`);
|
|
177
|
+
async _createUserDataDir(rootPath) {
|
|
178
|
+
const dir = process.env.PWMCP_PROFILES_DIR_FOR_TEST ?? registryDirectory;
|
|
179
|
+
const browserToken = this.config.browser.launchOptions?.channel ?? this.config.browser?.browserName;
|
|
180
|
+
// Hesitant putting hundreds of files into the user's workspace, so using it for hashing instead.
|
|
181
|
+
const rootPathToken = rootPath ? `-${createHash(rootPath)}` : '';
|
|
182
|
+
const result = path.join(dir, `mcp-${browserToken}${rootPathToken}`);
|
|
182
183
|
await fs.promises.mkdir(result, { recursive: true });
|
|
183
184
|
return result;
|
|
184
185
|
}
|
|
185
186
|
}
|
|
186
|
-
export class BrowserServerContextFactory extends BaseContextFactory {
|
|
187
|
-
constructor(browserConfig) {
|
|
188
|
-
super('persistent', browserConfig);
|
|
189
|
-
}
|
|
190
|
-
async _doObtainBrowser() {
|
|
191
|
-
const response = await fetch(new URL(`/json/launch`, this.browserConfig.browserAgent), {
|
|
192
|
-
method: 'POST',
|
|
193
|
-
body: JSON.stringify({
|
|
194
|
-
browserType: this.browserConfig.browserName,
|
|
195
|
-
userDataDir: this.browserConfig.userDataDir ?? await this._createUserDataDir(),
|
|
196
|
-
launchOptions: this.browserConfig.launchOptions,
|
|
197
|
-
contextOptions: this.browserConfig.contextOptions,
|
|
198
|
-
}),
|
|
199
|
-
});
|
|
200
|
-
const info = await response.json();
|
|
201
|
-
if (info.error)
|
|
202
|
-
throw new Error(info.error);
|
|
203
|
-
return await playwright.chromium.connectOverCDP(`http://localhost:${info.cdpPort}/`);
|
|
204
|
-
}
|
|
205
|
-
async _doCreateContext(browser) {
|
|
206
|
-
return this.browserConfig.isolated ? await browser.newContext() : browser.contexts()[0];
|
|
207
|
-
}
|
|
208
|
-
async _createUserDataDir() {
|
|
209
|
-
const dir = await userDataDir(this.browserConfig);
|
|
210
|
-
await fs.promises.mkdir(dir, { recursive: true });
|
|
211
|
-
return dir;
|
|
212
|
-
}
|
|
213
|
-
}
|
|
214
187
|
async function injectCdpPort(browserConfig) {
|
|
215
188
|
if (browserConfig.browserName === 'chromium')
|
|
216
189
|
browserConfig.launchOptions.cdpPort = await findFreePort();
|
|
@@ -225,4 +198,15 @@ async function findFreePort() {
|
|
|
225
198
|
server.on('error', reject);
|
|
226
199
|
});
|
|
227
200
|
}
|
|
201
|
+
async function startTraceServer(config, rootPath) {
|
|
202
|
+
if (!config.saveTrace)
|
|
203
|
+
return undefined;
|
|
204
|
+
const tracesDir = await outputFile(config, rootPath, `traces-${Date.now()}`);
|
|
205
|
+
const server = await startTraceViewerServer();
|
|
206
|
+
const urlPrefix = server.urlPrefix('human-readable');
|
|
207
|
+
const url = urlPrefix + '/trace/index.html?trace=' + tracesDir + '/trace.json';
|
|
208
|
+
// eslint-disable-next-line no-console
|
|
209
|
+
console.error('\nTrace viewer listening on ' + url);
|
|
210
|
+
return tracesDir;
|
|
211
|
+
}
|
|
228
212
|
//# sourceMappingURL=browserContextFactory.js.map
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"browserContextFactory.js","sourceRoot":"","sources":["../src/browserContextFactory.ts"],"names":[],"mappings":"AAAA;;;;;;;;;;;;;;GAcG;AAEH,OAAO,EAAE,MAAM,
|
|
1
|
+
{"version":3,"file":"browserContextFactory.js","sourceRoot":"","sources":["../src/browserContextFactory.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,78 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Copyright (c) Microsoft Corporation.
|
|
3
|
+
*
|
|
4
|
+
* Licensed under the Apache License, Version 2.0 (the "License");
|
|
5
|
+
* you may not use this file except in compliance with the License.
|
|
6
|
+
* You may obtain a copy of the License at
|
|
7
|
+
*
|
|
8
|
+
* http://www.apache.org/licenses/LICENSE-2.0
|
|
9
|
+
*
|
|
10
|
+
* Unless required by applicable law or agreed to in writing, software
|
|
11
|
+
* distributed under the License is distributed on an "AS IS" BASIS,
|
|
12
|
+
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
13
|
+
* See the License for the specific language governing permissions and
|
|
14
|
+
* limitations under the License.
|
|
15
|
+
*/
|
|
16
|
+
import { fileURLToPath } from 'url';
|
|
17
|
+
import { Context } from './context.js';
|
|
18
|
+
import { logUnhandledError } from './utils/log.js';
|
|
19
|
+
import { Response } from './response.js';
|
|
20
|
+
import { SessionLog } from './sessionLog.js';
|
|
21
|
+
import { filteredTools } from './tools.js';
|
|
22
|
+
import { toMcpTool } from './mcp/tool.js';
|
|
23
|
+
export class BrowserServerBackend {
|
|
24
|
+
_tools;
|
|
25
|
+
_context;
|
|
26
|
+
_sessionLog;
|
|
27
|
+
_config;
|
|
28
|
+
_browserContextFactory;
|
|
29
|
+
constructor(config, factory) {
|
|
30
|
+
this._config = config;
|
|
31
|
+
this._browserContextFactory = factory;
|
|
32
|
+
this._tools = filteredTools(config);
|
|
33
|
+
}
|
|
34
|
+
async initialize(server, clientVersion, roots) {
|
|
35
|
+
let rootPath;
|
|
36
|
+
if (roots.length > 0) {
|
|
37
|
+
const firstRootUri = roots[0]?.uri;
|
|
38
|
+
const url = firstRootUri ? new URL(firstRootUri) : undefined;
|
|
39
|
+
rootPath = url ? fileURLToPath(url) : undefined;
|
|
40
|
+
}
|
|
41
|
+
this._sessionLog = this._config.saveSession ? await SessionLog.create(this._config, rootPath) : undefined;
|
|
42
|
+
this._context = new Context({
|
|
43
|
+
tools: this._tools,
|
|
44
|
+
config: this._config,
|
|
45
|
+
browserContextFactory: this._browserContextFactory,
|
|
46
|
+
sessionLog: this._sessionLog,
|
|
47
|
+
clientInfo: { ...clientVersion, rootPath },
|
|
48
|
+
});
|
|
49
|
+
}
|
|
50
|
+
async listTools() {
|
|
51
|
+
return this._tools.map(tool => toMcpTool(tool.schema));
|
|
52
|
+
}
|
|
53
|
+
async callTool(name, rawArguments) {
|
|
54
|
+
const tool = this._tools.find(tool => tool.schema.name === name);
|
|
55
|
+
if (!tool)
|
|
56
|
+
throw new Error(`Tool "${name}" not found`);
|
|
57
|
+
const parsedArguments = tool.schema.inputSchema.parse(rawArguments || {});
|
|
58
|
+
const context = this._context;
|
|
59
|
+
const response = new Response(context, name, parsedArguments);
|
|
60
|
+
context.setRunningTool(name);
|
|
61
|
+
try {
|
|
62
|
+
await tool.handle(context, parsedArguments, response);
|
|
63
|
+
await response.finish();
|
|
64
|
+
this._sessionLog?.logResponse(response);
|
|
65
|
+
}
|
|
66
|
+
catch (error) {
|
|
67
|
+
response.addError(String(error));
|
|
68
|
+
}
|
|
69
|
+
finally {
|
|
70
|
+
context.setRunningTool(undefined);
|
|
71
|
+
}
|
|
72
|
+
return response.serialize();
|
|
73
|
+
}
|
|
74
|
+
serverClosed() {
|
|
75
|
+
void this._context?.dispose().catch(logUnhandledError);
|
|
76
|
+
}
|
|
77
|
+
}
|
|
78
|
+
//# sourceMappingURL=browserServerBackend.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"browserServerBackend.js","sourceRoot":"","sources":["../src/browserServerBackend.ts"],"names":[],"mappings":"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"}
|
package/lib/config.js
CHANGED
|
@@ -17,14 +17,13 @@ import fs from 'fs';
|
|
|
17
17
|
import os from 'os';
|
|
18
18
|
import path from 'path';
|
|
19
19
|
import { devices } from 'playwright';
|
|
20
|
-
import { sanitizeForFilePath } from './
|
|
21
|
-
const headlessMode = true;
|
|
20
|
+
import { sanitizeForFilePath } from './utils/fileUtils.js';
|
|
22
21
|
const defaultConfig = {
|
|
23
22
|
browser: {
|
|
24
23
|
browserName: 'chromium',
|
|
25
24
|
launchOptions: {
|
|
26
25
|
channel: 'chrome',
|
|
27
|
-
headless:
|
|
26
|
+
headless: os.platform() === 'linux' && !process.env.DISPLAY,
|
|
28
27
|
chromiumSandbox: true,
|
|
29
28
|
},
|
|
30
29
|
contextOptions: {
|
|
@@ -36,21 +35,26 @@ const defaultConfig = {
|
|
|
36
35
|
blockedOrigins: undefined,
|
|
37
36
|
},
|
|
38
37
|
server: {},
|
|
39
|
-
|
|
38
|
+
saveTrace: false,
|
|
39
|
+
timeouts: {
|
|
40
|
+
navigationTimeout: 60000,
|
|
41
|
+
defaultTimeout: 5000,
|
|
42
|
+
},
|
|
40
43
|
};
|
|
41
44
|
export async function resolveConfig(config) {
|
|
42
45
|
return mergeConfig(defaultConfig, config);
|
|
43
46
|
}
|
|
44
47
|
export async function resolveCLIConfig(cliOptions) {
|
|
45
48
|
const configInFile = await loadConfig(cliOptions.config);
|
|
46
|
-
const
|
|
47
|
-
const
|
|
48
|
-
|
|
49
|
-
|
|
50
|
-
|
|
49
|
+
const envOverrides = configFromEnv();
|
|
50
|
+
const cliOverrides = configFromCLIOptions(cliOptions);
|
|
51
|
+
let result = defaultConfig;
|
|
52
|
+
result = mergeConfig(result, configInFile);
|
|
53
|
+
result = mergeConfig(result, envOverrides);
|
|
54
|
+
result = mergeConfig(result, cliOverrides);
|
|
51
55
|
return result;
|
|
52
56
|
}
|
|
53
|
-
export
|
|
57
|
+
export function configFromCLIOptions(cliOptions) {
|
|
54
58
|
let browserName;
|
|
55
59
|
let channel;
|
|
56
60
|
switch (cliOptions.browser) {
|
|
@@ -77,10 +81,10 @@ export async function configFromCLIOptions(cliOptions) {
|
|
|
77
81
|
const launchOptions = {
|
|
78
82
|
channel,
|
|
79
83
|
executablePath: cliOptions.executablePath,
|
|
80
|
-
headless:
|
|
84
|
+
headless: cliOptions.headless,
|
|
81
85
|
};
|
|
82
86
|
// --no-sandbox was passed, disable the sandbox
|
|
83
|
-
if (
|
|
87
|
+
if (cliOptions.sandbox === false)
|
|
84
88
|
launchOptions.chromiumSandbox = false;
|
|
85
89
|
if (cliOptions.proxyServer) {
|
|
86
90
|
launchOptions.proxy = {
|
|
@@ -101,7 +105,7 @@ export async function configFromCLIOptions(cliOptions) {
|
|
|
101
105
|
try {
|
|
102
106
|
const [width, height] = cliOptions.viewportSize.split(',').map(n => +n);
|
|
103
107
|
if (isNaN(width) || isNaN(height))
|
|
104
|
-
throw new Error('
|
|
108
|
+
throw new Error('bad values');
|
|
105
109
|
contextOptions.viewport = { width, height };
|
|
106
110
|
}
|
|
107
111
|
catch (e) {
|
|
@@ -114,7 +118,6 @@ export async function configFromCLIOptions(cliOptions) {
|
|
|
114
118
|
contextOptions.serviceWorkers = 'block';
|
|
115
119
|
return {
|
|
116
120
|
browser: {
|
|
117
|
-
browserAgent: cliOptions.browserAgent ?? process.env.PW_BROWSER_AGENT,
|
|
118
121
|
browserName,
|
|
119
122
|
isolated: cliOptions.isolated,
|
|
120
123
|
userDataDir: cliOptions.userDataDir,
|
|
@@ -126,17 +129,52 @@ export async function configFromCLIOptions(cliOptions) {
|
|
|
126
129
|
port: cliOptions.port,
|
|
127
130
|
host: cliOptions.host,
|
|
128
131
|
},
|
|
129
|
-
capabilities: cliOptions.caps
|
|
130
|
-
vision: !!cliOptions.vision,
|
|
132
|
+
capabilities: cliOptions.caps,
|
|
131
133
|
network: {
|
|
132
134
|
allowedOrigins: cliOptions.allowedOrigins,
|
|
133
135
|
blockedOrigins: cliOptions.blockedOrigins,
|
|
134
136
|
},
|
|
137
|
+
saveSession: cliOptions.saveSession,
|
|
135
138
|
saveTrace: cliOptions.saveTrace,
|
|
136
139
|
outputDir: cliOptions.outputDir,
|
|
137
140
|
imageResponses: cliOptions.imageResponses,
|
|
141
|
+
timeouts: {
|
|
142
|
+
navigationTimeout: cliOptions.navigationTimeout,
|
|
143
|
+
defaultTimeout: cliOptions.defaultTimeout,
|
|
144
|
+
}
|
|
138
145
|
};
|
|
139
146
|
}
|
|
147
|
+
function configFromEnv() {
|
|
148
|
+
const options = {};
|
|
149
|
+
options.allowedOrigins = semicolonSeparatedList(process.env.PLAYWRIGHT_MCP_ALLOWED_ORIGINS);
|
|
150
|
+
options.blockedOrigins = semicolonSeparatedList(process.env.PLAYWRIGHT_MCP_BLOCKED_ORIGINS);
|
|
151
|
+
options.blockServiceWorkers = envToBoolean(process.env.PLAYWRIGHT_MCP_BLOCK_SERVICE_WORKERS);
|
|
152
|
+
options.browser = envToString(process.env.PLAYWRIGHT_MCP_BROWSER);
|
|
153
|
+
options.caps = commaSeparatedList(process.env.PLAYWRIGHT_MCP_CAPS);
|
|
154
|
+
options.cdpEndpoint = envToString(process.env.PLAYWRIGHT_MCP_CDP_ENDPOINT);
|
|
155
|
+
options.config = envToString(process.env.PLAYWRIGHT_MCP_CONFIG);
|
|
156
|
+
options.device = envToString(process.env.PLAYWRIGHT_MCP_DEVICE);
|
|
157
|
+
options.executablePath = envToString(process.env.PLAYWRIGHT_MCP_EXECUTABLE_PATH);
|
|
158
|
+
options.headless = envToBoolean(process.env.PLAYWRIGHT_MCP_HEADLESS);
|
|
159
|
+
options.host = envToString(process.env.PLAYWRIGHT_MCP_HOST);
|
|
160
|
+
options.ignoreHttpsErrors = envToBoolean(process.env.PLAYWRIGHT_MCP_IGNORE_HTTPS_ERRORS);
|
|
161
|
+
options.isolated = envToBoolean(process.env.PLAYWRIGHT_MCP_ISOLATED);
|
|
162
|
+
if (process.env.PLAYWRIGHT_MCP_IMAGE_RESPONSES === 'omit')
|
|
163
|
+
options.imageResponses = 'omit';
|
|
164
|
+
options.sandbox = envToBoolean(process.env.PLAYWRIGHT_MCP_SANDBOX);
|
|
165
|
+
options.outputDir = envToString(process.env.PLAYWRIGHT_MCP_OUTPUT_DIR);
|
|
166
|
+
options.port = envToNumber(process.env.PLAYWRIGHT_MCP_PORT);
|
|
167
|
+
options.proxyBypass = envToString(process.env.PLAYWRIGHT_MCP_PROXY_BYPASS);
|
|
168
|
+
options.proxyServer = envToString(process.env.PLAYWRIGHT_MCP_PROXY_SERVER);
|
|
169
|
+
options.saveTrace = envToBoolean(process.env.PLAYWRIGHT_MCP_SAVE_TRACE);
|
|
170
|
+
options.storageState = envToString(process.env.PLAYWRIGHT_MCP_STORAGE_STATE);
|
|
171
|
+
options.userAgent = envToString(process.env.PLAYWRIGHT_MCP_USER_AGENT);
|
|
172
|
+
options.userDataDir = envToString(process.env.PLAYWRIGHT_MCP_USER_DATA_DIR);
|
|
173
|
+
options.viewportSize = envToString(process.env.PLAYWRIGHT_MCP_VIEWPORT_SIZE);
|
|
174
|
+
options.navigationTimeout = envToNumber(process.env.PLAYWRIGHT_MCP_NAVIGATION_TIMEOUT);
|
|
175
|
+
options.defaultTimeout = envToNumber(process.env.PLAYWRIGHT_MCP_DEFAULT_TIMEOUT);
|
|
176
|
+
return configFromCLIOptions(options);
|
|
177
|
+
}
|
|
140
178
|
async function loadConfig(configFile) {
|
|
141
179
|
if (!configFile)
|
|
142
180
|
return {};
|
|
@@ -147,10 +185,13 @@ async function loadConfig(configFile) {
|
|
|
147
185
|
throw new Error(`Failed to load config file: ${configFile}, ${error}`);
|
|
148
186
|
}
|
|
149
187
|
}
|
|
150
|
-
export async function outputFile(config, name) {
|
|
151
|
-
|
|
188
|
+
export async function outputFile(config, rootPath, name) {
|
|
189
|
+
const outputDir = config.outputDir
|
|
190
|
+
?? (rootPath ? path.join(rootPath, '.playwright-mcp') : undefined)
|
|
191
|
+
?? path.join(os.tmpdir(), 'playwright-mcp-output', sanitizeForFilePath(new Date().toISOString()));
|
|
192
|
+
await fs.promises.mkdir(outputDir, { recursive: true });
|
|
152
193
|
const fileName = sanitizeForFilePath(name);
|
|
153
|
-
return path.join(
|
|
194
|
+
return path.join(outputDir, fileName);
|
|
154
195
|
}
|
|
155
196
|
function pickDefined(obj) {
|
|
156
197
|
return Object.fromEntries(Object.entries(obj ?? {}).filter(([_, v]) => v !== undefined));
|
|
@@ -185,6 +226,35 @@ function mergeConfig(base, overrides) {
|
|
|
185
226
|
...pickDefined(base.server),
|
|
186
227
|
...pickDefined(overrides.server),
|
|
187
228
|
},
|
|
229
|
+
timeouts: {
|
|
230
|
+
navigationTimeout: overrides.timeouts?.navigationTimeout ?? base.timeouts.navigationTimeout,
|
|
231
|
+
defaultTimeout: overrides.timeouts?.defaultTimeout ?? base.timeouts.defaultTimeout,
|
|
232
|
+
},
|
|
188
233
|
};
|
|
189
234
|
}
|
|
235
|
+
export function semicolonSeparatedList(value) {
|
|
236
|
+
if (!value)
|
|
237
|
+
return undefined;
|
|
238
|
+
return value.split(';').map(v => v.trim());
|
|
239
|
+
}
|
|
240
|
+
export function commaSeparatedList(value) {
|
|
241
|
+
if (!value)
|
|
242
|
+
return undefined;
|
|
243
|
+
return value.split(',').map(v => v.trim());
|
|
244
|
+
}
|
|
245
|
+
function envToNumber(value) {
|
|
246
|
+
if (!value)
|
|
247
|
+
return undefined;
|
|
248
|
+
return +value;
|
|
249
|
+
}
|
|
250
|
+
function envToBoolean(value) {
|
|
251
|
+
if (value === 'true' || value === '1')
|
|
252
|
+
return true;
|
|
253
|
+
if (value === 'false' || value === '0')
|
|
254
|
+
return false;
|
|
255
|
+
return undefined;
|
|
256
|
+
}
|
|
257
|
+
function envToString(value) {
|
|
258
|
+
return value ? value.trim() : undefined;
|
|
259
|
+
}
|
|
190
260
|
//# sourceMappingURL=config.js.map
|