@mcp-b/chrome-devtools-mcp 2.3.0 → 2.3.1-beta.20260528050333

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 (67) hide show
  1. package/package.json +1 -1
  2. package/build/src/DevToolsConnectionAdapter.js +0 -70
  3. package/build/src/DevtoolsUtils.js +0 -290
  4. package/build/src/McpContext.js +0 -687
  5. package/build/src/McpPage.js +0 -95
  6. package/build/src/McpResponse.js +0 -588
  7. package/build/src/Mutex.js +0 -37
  8. package/build/src/PageCollector.js +0 -308
  9. package/build/src/SlimMcpResponse.js +0 -18
  10. package/build/src/WaitForHelper.js +0 -135
  11. package/build/src/bin/chrome-devtools-cli-options.js +0 -651
  12. package/build/src/bin/chrome-devtools-mcp-cli-options.js +0 -317
  13. package/build/src/bin/chrome-devtools-mcp-main.js +0 -35
  14. package/build/src/bin/chrome-devtools-mcp.js +0 -21
  15. package/build/src/bin/chrome-devtools.js +0 -185
  16. package/build/src/bin/cliDefinitions.js +0 -615
  17. package/build/src/browser.js +0 -198
  18. package/build/src/daemon/client.js +0 -152
  19. package/build/src/daemon/daemon.js +0 -206
  20. package/build/src/daemon/types.js +0 -6
  21. package/build/src/daemon/utils.js +0 -108
  22. package/build/src/formatters/ConsoleFormatter.js +0 -234
  23. package/build/src/formatters/IssueFormatter.js +0 -192
  24. package/build/src/formatters/NetworkFormatter.js +0 -215
  25. package/build/src/formatters/SnapshotFormatter.js +0 -131
  26. package/build/src/index.js +0 -202
  27. package/build/src/issue-descriptions.js +0 -39
  28. package/build/src/logger.js +0 -36
  29. package/build/src/polyfill.js +0 -7
  30. package/build/src/telemetry/ClearcutLogger.js +0 -102
  31. package/build/src/telemetry/WatchdogClient.js +0 -60
  32. package/build/src/telemetry/flagUtils.js +0 -45
  33. package/build/src/telemetry/metricUtils.js +0 -14
  34. package/build/src/telemetry/persistence.js +0 -53
  35. package/build/src/telemetry/types.js +0 -33
  36. package/build/src/telemetry/watchdog/ClearcutSender.js +0 -203
  37. package/build/src/telemetry/watchdog/main.js +0 -127
  38. package/build/src/third_party/devtools-formatter-worker.js +0 -7
  39. package/build/src/third_party/index.js +0 -26
  40. package/build/src/third_party/lighthouse-devtools-mcp-bundle.js +0 -54183
  41. package/build/src/tools/ToolDefinition.js +0 -72
  42. package/build/src/tools/categories.js +0 -24
  43. package/build/src/tools/console.js +0 -85
  44. package/build/src/tools/emulation.js +0 -55
  45. package/build/src/tools/extensions.js +0 -96
  46. package/build/src/tools/input.js +0 -368
  47. package/build/src/tools/lighthouse.js +0 -123
  48. package/build/src/tools/memory.js +0 -28
  49. package/build/src/tools/network.js +0 -120
  50. package/build/src/tools/pages.js +0 -319
  51. package/build/src/tools/performance.js +0 -190
  52. package/build/src/tools/screencast.js +0 -79
  53. package/build/src/tools/screenshot.js +0 -84
  54. package/build/src/tools/script.js +0 -119
  55. package/build/src/tools/slim/tools.js +0 -81
  56. package/build/src/tools/snapshot.js +0 -56
  57. package/build/src/tools/tools.js +0 -52
  58. package/build/src/tools/webmcp.js +0 -416
  59. package/build/src/trace-processing/parse.js +0 -84
  60. package/build/src/types.js +0 -6
  61. package/build/src/utils/ExtensionRegistry.js +0 -35
  62. package/build/src/utils/files.js +0 -19
  63. package/build/src/utils/keyboard.js +0 -296
  64. package/build/src/utils/pagination.js +0 -49
  65. package/build/src/utils/string.js +0 -36
  66. package/build/src/utils/types.js +0 -6
  67. package/build/src/version.js +0 -9
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@mcp-b/chrome-devtools-mcp",
3
- "version": "2.3.0",
3
+ "version": "2.3.1-beta.20260528050333",
4
4
  "description": "MCP server for Chrome DevTools",
5
5
  "homepage": "https://github.com/WebMCP-org/npm-packages/tree/main/packages/chrome-devtools-mcp#readme",
6
6
  "bugs": {
@@ -1,70 +0,0 @@
1
- /**
2
- * @license
3
- * Copyright 2025 Google LLC
4
- * SPDX-License-Identifier: Apache-2.0
5
- */
6
- import { CDPSessionEvent } from './third_party/index.js';
7
- /**
8
- * This class makes a puppeteer connection look like DevTools CDPConnection.
9
- *
10
- * Since we connect "root" DevTools targets to specific pages, we scope everything to a puppeteer CDP session.
11
- *
12
- * We don't have to recursively listen for 'sessionattached' as the "root" CDP session sees all child session attached
13
- * events, regardless how deeply nested they are.
14
- */
15
- export class PuppeteerDevToolsConnection {
16
- #connection;
17
- #observers = new Set();
18
- #sessionEventHandlers = new Map();
19
- constructor(session) {
20
- this.#connection = session.connection();
21
- session.on(CDPSessionEvent.SessionAttached, this.#startForwardingCdpEvents.bind(this));
22
- session.on(CDPSessionEvent.SessionDetached, this.#stopForwardingCdpEvents.bind(this));
23
- this.#startForwardingCdpEvents(session);
24
- }
25
- send(method, params, sessionId) {
26
- if (sessionId === undefined) {
27
- throw new Error('Attempting to send on the root session. This must not happen');
28
- }
29
- const session = this.#connection.session(sessionId);
30
- if (!session) {
31
- throw new Error('Unknown session ' + sessionId);
32
- }
33
- // Rolled protocol version between puppeteer and DevTools doesn't necessarily match
34
- /* eslint-disable @typescript-eslint/no-explicit-any */
35
- return session
36
- .send(method, params)
37
- .then((result) => ({ result }))
38
- .catch((error) => ({ error }));
39
- /* eslint-enable @typescript-eslint/no-explicit-any */
40
- }
41
- observe(observer) {
42
- this.#observers.add(observer);
43
- }
44
- unobserve(observer) {
45
- this.#observers.delete(observer);
46
- }
47
- #startForwardingCdpEvents(session) {
48
- const handler = this.#handleEvent.bind(this, session.id());
49
- this.#sessionEventHandlers.set(session.id(), handler);
50
- session.on('*', handler);
51
- }
52
- #stopForwardingCdpEvents(session) {
53
- const handler = this.#sessionEventHandlers.get(session.id());
54
- if (handler) {
55
- session.off('*', handler);
56
- }
57
- }
58
- #handleEvent(sessionId, type, event // eslint-disable-line @typescript-eslint/no-explicit-any
59
- ) {
60
- if (typeof type === 'string' &&
61
- type !== CDPSessionEvent.SessionAttached &&
62
- type !== CDPSessionEvent.SessionDetached) {
63
- this.#observers.forEach((observer) => observer.onEvent({
64
- method: type,
65
- sessionId,
66
- params: event,
67
- }));
68
- }
69
- }
70
- }
@@ -1,290 +0,0 @@
1
- /**
2
- * @license
3
- * Copyright 2025 Google LLC
4
- * SPDX-License-Identifier: Apache-2.0
5
- */
6
- import { PuppeteerDevToolsConnection } from './DevToolsConnectionAdapter.js';
7
- import { Mutex } from './Mutex.js';
8
- import { DevTools } from './third_party/index.js';
9
- /**
10
- * A mock implementation of an issues manager that only implements the methods
11
- * that are actually used by the IssuesAggregator
12
- */
13
- export class FakeIssuesManager extends DevTools.Common.ObjectWrapper
14
- .ObjectWrapper {
15
- issues() {
16
- return [];
17
- }
18
- }
19
- // DevTools CDP errors can get noisy.
20
- DevTools.ProtocolClient.InspectorBackend.test.suppressRequestErrors = true;
21
- DevTools.I18n.DevToolsLocale.DevToolsLocale.instance({
22
- create: true,
23
- data: {
24
- navigatorLanguage: 'en-US',
25
- settingLanguage: 'en-US',
26
- lookupClosestDevToolsLocale: (l) => l,
27
- },
28
- });
29
- DevTools.I18n.i18n.registerLocaleDataForTest('en-US', {});
30
- DevTools.Formatter.FormatterWorkerPool.FormatterWorkerPool.instance({
31
- forceNew: true,
32
- entrypointURL: import.meta.resolve('./third_party/devtools-formatter-worker.js'),
33
- });
34
- export class UniverseManager {
35
- #browser;
36
- #createUniverseFor;
37
- #universes = new WeakMap();
38
- /** Guard access to #universes so we don't create unnecessary universes */
39
- #mutex = new Mutex();
40
- constructor(browser, factory = DEFAULT_FACTORY) {
41
- this.#browser = browser;
42
- this.#createUniverseFor = factory;
43
- }
44
- async init(pages) {
45
- try {
46
- await this.#mutex.acquire();
47
- const promises = [];
48
- for (const page of pages) {
49
- promises.push(this.#createUniverseFor(page).then((targetUniverse) => this.#universes.set(page, targetUniverse)));
50
- }
51
- this.#browser.on('targetcreated', this.#onTargetCreated);
52
- this.#browser.on('targetdestroyed', this.#onTargetDestroyed);
53
- await Promise.all(promises);
54
- }
55
- finally {
56
- this.#mutex.release();
57
- }
58
- }
59
- get(page) {
60
- return this.#universes.get(page) ?? null;
61
- }
62
- dispose() {
63
- this.#browser.off('targetcreated', this.#onTargetCreated);
64
- this.#browser.off('targetdestroyed', this.#onTargetDestroyed);
65
- }
66
- #onTargetCreated = async (target) => {
67
- const page = await target.page();
68
- try {
69
- await this.#mutex.acquire();
70
- if (!page || this.#universes.has(page)) {
71
- return;
72
- }
73
- this.#universes.set(page, await this.#createUniverseFor(page));
74
- }
75
- finally {
76
- this.#mutex.release();
77
- }
78
- };
79
- #onTargetDestroyed = async (target) => {
80
- const page = await target.page();
81
- try {
82
- await this.#mutex.acquire();
83
- if (!page || !this.#universes.has(page)) {
84
- return;
85
- }
86
- this.#universes.delete(page);
87
- }
88
- finally {
89
- this.#mutex.release();
90
- }
91
- };
92
- }
93
- const DEFAULT_FACTORY = async (page) => {
94
- const settingStorage = new DevTools.Common.Settings.SettingsStorage({});
95
- const universe = new DevTools.Foundation.Universe.Universe({
96
- settingsCreationOptions: {
97
- syncedStorage: settingStorage,
98
- globalStorage: settingStorage,
99
- localStorage: settingStorage,
100
- settingRegistrations: DevTools.Common.SettingRegistration.getRegisteredSettings(),
101
- },
102
- overrideAutoStartModels: new Set([DevTools.DebuggerModel]),
103
- });
104
- const session = await page.createCDPSession();
105
- const connection = new PuppeteerDevToolsConnection(session);
106
- const targetManager = universe.context.get(DevTools.TargetManager);
107
- targetManager.observeModels(DevTools.DebuggerModel, SKIP_ALL_PAUSES);
108
- const target = targetManager.createTarget('main', '', 'frame', // eslint-disable-line @typescript-eslint/no-explicit-any
109
- /* parentTarget */ null, session.id(), undefined, connection);
110
- return { target, universe };
111
- };
112
- // We don't want to pause any DevTools universe session ever on the MCP side.
113
- //
114
- // Note that calling `setSkipAllPauses` only affects the session on which it was
115
- // sent. This means DevTools can still pause, step and do whatever. We just won't
116
- // see the `Debugger.paused`/`Debugger.resumed` events on the MCP side.
117
- const SKIP_ALL_PAUSES = {
118
- modelAdded(model) {
119
- void model.agent.invoke_setSkipAllPauses({ skip: true });
120
- },
121
- modelRemoved() {
122
- // Do nothing.
123
- },
124
- };
125
- /**
126
- * Constructed from Runtime.ExceptionDetails of an uncaught error.
127
- *
128
- * TODO: Also construct from a RemoteObject of subtype 'error'.
129
- *
130
- * Consists of the message, a fully resolved stack trace and a fully resolved 'cause' chain.
131
- */
132
- export class SymbolizedError {
133
- message;
134
- stackTrace;
135
- cause;
136
- constructor(message, stackTrace, cause) {
137
- this.message = message;
138
- this.stackTrace = stackTrace;
139
- this.cause = cause;
140
- }
141
- static async fromDetails(opts) {
142
- const message = SymbolizedError.#getMessage(opts.details);
143
- if (!opts.includeStackAndCause || !opts.devTools) {
144
- return new SymbolizedError(message, opts.resolvedStackTraceForTesting, opts.resolvedCauseForTesting);
145
- }
146
- let stackTrace;
147
- if (opts.resolvedStackTraceForTesting) {
148
- stackTrace = opts.resolvedStackTraceForTesting;
149
- }
150
- else if (opts.details.stackTrace) {
151
- try {
152
- stackTrace = await createStackTrace(opts.devTools, opts.details.stackTrace, opts.targetId);
153
- }
154
- catch {
155
- // ignore
156
- }
157
- }
158
- // TODO: Turn opts.details.exception into a JSHandle and retrieve the 'cause' property.
159
- // If its an Error, recursively create a SymbolizedError.
160
- let cause;
161
- if (opts.resolvedCauseForTesting) {
162
- cause = opts.resolvedCauseForTesting;
163
- }
164
- else if (opts.details.exception) {
165
- try {
166
- const causeRemoteObj = await SymbolizedError.#lookupCause(opts.devTools, opts.details.exception, opts.targetId);
167
- if (causeRemoteObj) {
168
- cause = await SymbolizedError.fromError({
169
- devTools: opts.devTools,
170
- error: causeRemoteObj,
171
- targetId: opts.targetId,
172
- });
173
- }
174
- }
175
- catch {
176
- // Ignore
177
- }
178
- }
179
- return new SymbolizedError(message, stackTrace, cause);
180
- }
181
- static async fromError(opts) {
182
- const details = await SymbolizedError.#getExceptionDetails(opts.devTools, opts.error, opts.targetId);
183
- if (details) {
184
- return SymbolizedError.fromDetails({
185
- details,
186
- devTools: opts.devTools,
187
- targetId: opts.targetId,
188
- includeStackAndCause: true,
189
- });
190
- }
191
- return new SymbolizedError(SymbolizedError.#getMessageFromException(opts.error));
192
- }
193
- static #getMessage(details) {
194
- // For Runtime.exceptionThrown with a present exception object, `details.text` will be "Uncaught" and
195
- // we have to manually parse out the error text from the exception description.
196
- // In the case of Runtime.getExceptionDetails, `details.text` has the Error.message.
197
- if (details.text === 'Uncaught' && details.exception) {
198
- return 'Uncaught ' + SymbolizedError.#getMessageFromException(details.exception);
199
- }
200
- return details.text;
201
- }
202
- static #getMessageFromException(error) {
203
- const messageWithRest = error.description?.split('\n at ', 2) ?? [];
204
- return messageWithRest[0] ?? '';
205
- }
206
- static async #getExceptionDetails(devTools, error, targetId) {
207
- if (!devTools || (error.type !== 'object' && error.subtype !== 'error')) {
208
- return null;
209
- }
210
- const targetManager = devTools.universe.context.get(DevTools.TargetManager);
211
- const target = targetId
212
- ? targetManager.targetById(targetId) || devTools.target
213
- : devTools.target;
214
- const model = target.model(DevTools.RuntimeModel);
215
- return ((await model.getExceptionDetails(error.objectId)) ?? null);
216
- }
217
- static async #lookupCause(devTools, error, targetId) {
218
- if (!devTools || (error.type !== 'object' && error.subtype !== 'error')) {
219
- return null;
220
- }
221
- const targetManager = devTools.universe.context.get(DevTools.TargetManager);
222
- const target = targetId
223
- ? targetManager.targetById(targetId) || devTools.target
224
- : devTools.target;
225
- const properties = await target.runtimeAgent().invoke_getProperties({
226
- objectId: error.objectId,
227
- });
228
- if (properties.getError()) {
229
- return null;
230
- }
231
- return properties.result.find((prop) => prop.name === 'cause')?.value ?? null;
232
- }
233
- static createForTesting(message, stackTrace, cause) {
234
- return new SymbolizedError(message, stackTrace, cause);
235
- }
236
- }
237
- export async function createStackTraceForConsoleMessage(devTools, consoleMessage) {
238
- const message = consoleMessage;
239
- const rawStackTrace = message._rawStackTrace();
240
- if (rawStackTrace) {
241
- return createStackTrace(devTools, rawStackTrace, message._targetId());
242
- }
243
- return undefined;
244
- }
245
- export async function createStackTrace(devTools, rawStackTrace, targetId) {
246
- const targetManager = devTools.universe.context.get(DevTools.TargetManager);
247
- const target = targetId ? targetManager.targetById(targetId) || devTools.target : devTools.target;
248
- const model = target.model(DevTools.DebuggerModel);
249
- // DevTools doesn't wait for source maps to attach before building a stack trace, rather it'll send
250
- // an update event once a source map was attached and the stack trace retranslated. This doesn't
251
- // work in the MCP case, so we'll collect all script IDs upfront and wait for any pending source map
252
- // loads before creating the stack trace. We might also have to wait for Debugger.ScriptParsed events if
253
- // the stack trace is created particularly early.
254
- const scriptIds = new Set();
255
- for (const frame of rawStackTrace.callFrames) {
256
- scriptIds.add(frame.scriptId);
257
- }
258
- for (let asyncStack = rawStackTrace.parent; asyncStack; asyncStack = asyncStack.parent) {
259
- for (const frame of asyncStack.callFrames) {
260
- scriptIds.add(frame.scriptId);
261
- }
262
- }
263
- const signal = AbortSignal.timeout(1_000);
264
- await Promise.all([...scriptIds].map((id) => waitForScript(model, id, signal)
265
- .then((script) => model.sourceMapManager().sourceMapForClientPromise(script))
266
- .catch()));
267
- const binding = devTools.universe.context.get(DevTools.DebuggerWorkspaceBinding);
268
- // DevTools uses branded types for ScriptId and others. Casting the puppeteer protocol type to the DevTools protocol type is safe.
269
- return binding.createStackTraceFromProtocolRuntime(rawStackTrace, target);
270
- }
271
- // Waits indefinitely for the script so pair it with Promise.race.
272
- async function waitForScript(model, scriptId, signal) {
273
- while (true) {
274
- if (signal.aborted) {
275
- throw signal.reason;
276
- }
277
- const script = model.scriptForId(scriptId);
278
- if (script) {
279
- return script;
280
- }
281
- await new Promise((resolve, reject) => {
282
- signal.addEventListener('abort', () => reject(signal.reason), {
283
- once: true,
284
- });
285
- void model
286
- .once('ParsedScriptSource')
287
- .then(resolve);
288
- });
289
- }
290
- }