@gabrielbryk/arc-devtools-mcp 1.3.0
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/LICENSE +202 -0
- package/README.md +977 -0
- package/build/src/HeapSnapshotManager.js +148 -0
- package/build/src/McpContext.js +737 -0
- package/build/src/McpPage.js +315 -0
- package/build/src/McpResponse.js +990 -0
- package/build/src/Mutex.js +38 -0
- package/build/src/PageCollector.js +297 -0
- package/build/src/ServiceWorkerCollector.js +171 -0
- package/build/src/SlimMcpResponse.js +19 -0
- package/build/src/TextSnapshot.js +236 -0
- package/build/src/ToolHandler.js +223 -0
- package/build/src/WaitForHelper.js +190 -0
- package/build/src/bin/check-latest-version.js +50 -0
- package/build/src/bin/chrome-devtools-cli-options.js +978 -0
- package/build/src/bin/chrome-devtools-mcp-cli-options.js +412 -0
- package/build/src/bin/chrome-devtools-mcp-main.js +72 -0
- package/build/src/bin/chrome-devtools-mcp.js +23 -0
- package/build/src/bin/chrome-devtools.js +189 -0
- package/build/src/browser.js +253 -0
- package/build/src/daemon/client.js +160 -0
- package/build/src/daemon/daemon.js +261 -0
- package/build/src/daemon/types.js +7 -0
- package/build/src/daemon/utils.js +115 -0
- package/build/src/devtools/DevToolsConnectionAdapter.js +70 -0
- package/build/src/devtools/DevtoolsUtils.js +369 -0
- package/build/src/devtools/McpHostBindingAdapter.js +165 -0
- package/build/src/formatters/ConsoleFormatter.js +288 -0
- package/build/src/formatters/HeapSnapshotFormatter.js +97 -0
- package/build/src/formatters/IssueFormatter.js +193 -0
- package/build/src/formatters/NetworkFormatter.js +238 -0
- package/build/src/formatters/SnapshotFormatter.js +135 -0
- package/build/src/index.js +153 -0
- package/build/src/issue-descriptions.js +40 -0
- package/build/src/logger.js +37 -0
- package/build/src/polyfill.js +8 -0
- package/build/src/telemetry/ClearcutLogger.js +169 -0
- package/build/src/telemetry/WatchdogClient.js +61 -0
- package/build/src/telemetry/errors.js +18 -0
- package/build/src/telemetry/flagUtils.js +89 -0
- package/build/src/telemetry/metricsRegistry.js +89 -0
- package/build/src/telemetry/persistence.js +72 -0
- package/build/src/telemetry/transformation.js +134 -0
- package/build/src/telemetry/types.js +31 -0
- package/build/src/telemetry/watchdog/ClearcutSender.js +205 -0
- package/build/src/telemetry/watchdog/main.js +128 -0
- package/build/src/third_party/THIRD_PARTY_NOTICES +3637 -0
- package/build/src/third_party/bundled-packages.json +12 -0
- package/build/src/third_party/devtools-formatter-worker.js +15301 -0
- package/build/src/third_party/devtools-heap-snapshot-worker.js +9870 -0
- package/build/src/third_party/index.js +159597 -0
- package/build/src/third_party/issue-descriptions/CoepCoopSandboxedIframeCannotNavigateToCoopPage.md +4 -0
- package/build/src/third_party/issue-descriptions/CoepCorpNotSameOrigin.md +8 -0
- package/build/src/third_party/issue-descriptions/CoepCorpNotSameOriginAfterDefaultedToSameOriginByCoep.md +18 -0
- package/build/src/third_party/issue-descriptions/CoepCorpNotSameSite.md +7 -0
- package/build/src/third_party/issue-descriptions/CoepFrameResourceNeedsCoepHeader.md +10 -0
- package/build/src/third_party/issue-descriptions/CompatibilityModeQuirks.md +5 -0
- package/build/src/third_party/issue-descriptions/CookieAttributeValueExceedsMaxSize.md +5 -0
- package/build/src/third_party/issue-descriptions/LowTextContrast.md +5 -0
- package/build/src/third_party/issue-descriptions/SameSiteExcludeContextDowngradeRead.md +8 -0
- package/build/src/third_party/issue-descriptions/SameSiteExcludeContextDowngradeSet.md +8 -0
- package/build/src/third_party/issue-descriptions/SameSiteExcludeNavigationContextDowngrade.md +8 -0
- package/build/src/third_party/issue-descriptions/SameSiteNoneInsecureErrorRead.md +8 -0
- package/build/src/third_party/issue-descriptions/SameSiteNoneInsecureErrorSet.md +8 -0
- package/build/src/third_party/issue-descriptions/SameSiteNoneInsecureWarnRead.md +8 -0
- package/build/src/third_party/issue-descriptions/SameSiteNoneInsecureWarnSet.md +8 -0
- package/build/src/third_party/issue-descriptions/SameSiteUnspecifiedLaxAllowUnsafeRead.md +9 -0
- package/build/src/third_party/issue-descriptions/SameSiteUnspecifiedLaxAllowUnsafeSet.md +9 -0
- package/build/src/third_party/issue-descriptions/SameSiteWarnCrossDowngradeRead.md +8 -0
- package/build/src/third_party/issue-descriptions/SameSiteWarnCrossDowngradeSet.md +8 -0
- package/build/src/third_party/issue-descriptions/SameSiteWarnStrictLaxDowngradeStrict.md +8 -0
- package/build/src/third_party/issue-descriptions/arInsecureContext.md +7 -0
- package/build/src/third_party/issue-descriptions/arInvalidInfoHeader.md +5 -0
- package/build/src/third_party/issue-descriptions/arInvalidRegisterOsSourceHeader.md +5 -0
- package/build/src/third_party/issue-descriptions/arInvalidRegisterOsTriggerHeader.md +5 -0
- package/build/src/third_party/issue-descriptions/arInvalidRegisterSourceHeader.md +5 -0
- package/build/src/third_party/issue-descriptions/arInvalidRegisterTriggerHeader.md +5 -0
- package/build/src/third_party/issue-descriptions/arNavigationRegistrationUniqueScopeAlreadySet.md +5 -0
- package/build/src/third_party/issue-descriptions/arNavigationRegistrationWithoutTransientUserActivation.md +6 -0
- package/build/src/third_party/issue-descriptions/arNoRegisterOsSourceHeader.md +5 -0
- package/build/src/third_party/issue-descriptions/arNoRegisterOsTriggerHeader.md +5 -0
- package/build/src/third_party/issue-descriptions/arNoRegisterSourceHeader.md +5 -0
- package/build/src/third_party/issue-descriptions/arNoRegisterTriggerHeader.md +5 -0
- package/build/src/third_party/issue-descriptions/arNoWebOrOsSupport.md +4 -0
- package/build/src/third_party/issue-descriptions/arOsSourceIgnored.md +18 -0
- package/build/src/third_party/issue-descriptions/arOsTriggerIgnored.md +19 -0
- package/build/src/third_party/issue-descriptions/arPermissionPolicyDisabled.md +8 -0
- package/build/src/third_party/issue-descriptions/arSourceAndTriggerHeaders.md +9 -0
- package/build/src/third_party/issue-descriptions/arSourceIgnored.md +13 -0
- package/build/src/third_party/issue-descriptions/arTriggerIgnored.md +12 -0
- package/build/src/third_party/issue-descriptions/arUntrustworthyReportingOrigin.md +10 -0
- package/build/src/third_party/issue-descriptions/arWebAndOsHeaders.md +11 -0
- package/build/src/third_party/issue-descriptions/bounceTrackingMitigations.md +3 -0
- package/build/src/third_party/issue-descriptions/clientHintMetaTagAllowListInvalidOrigin.md +4 -0
- package/build/src/third_party/issue-descriptions/clientHintMetaTagModifiedHTML.md +4 -0
- package/build/src/third_party/issue-descriptions/connectionAllowlistInvalidAllowlistItemType.md +12 -0
- package/build/src/third_party/issue-descriptions/connectionAllowlistInvalidHeader.md +12 -0
- package/build/src/third_party/issue-descriptions/connectionAllowlistInvalidUrlPattern.md +8 -0
- package/build/src/third_party/issue-descriptions/connectionAllowlistItemNotInnerList.md +12 -0
- package/build/src/third_party/issue-descriptions/connectionAllowlistMoreThanOneList.md +7 -0
- package/build/src/third_party/issue-descriptions/connectionAllowlistReportingEndpointNotToken.md +10 -0
- package/build/src/third_party/issue-descriptions/cookieCrossSiteRedirectDowngrade.md +12 -0
- package/build/src/third_party/issue-descriptions/cookieExcludeBlockedWithinRelatedWebsiteSet.md +4 -0
- package/build/src/third_party/issue-descriptions/cookieExcludeDomainNonAscii.md +11 -0
- package/build/src/third_party/issue-descriptions/cookieExcludePortMismatch.md +8 -0
- package/build/src/third_party/issue-descriptions/cookieExcludeSchemeMismatch.md +7 -0
- package/build/src/third_party/issue-descriptions/cookieExcludeThirdPartyPhaseoutRead.md +6 -0
- package/build/src/third_party/issue-descriptions/cookieExcludeThirdPartyPhaseoutSet.md +6 -0
- package/build/src/third_party/issue-descriptions/cookieWarnDomainNonAscii.md +11 -0
- package/build/src/third_party/issue-descriptions/cookieWarnMetadataGrantRead.md +4 -0
- package/build/src/third_party/issue-descriptions/cookieWarnMetadataGrantSet.md +4 -0
- package/build/src/third_party/issue-descriptions/cookieWarnThirdPartyPhaseoutRead.md +6 -0
- package/build/src/third_party/issue-descriptions/cookieWarnThirdPartyPhaseoutSet.md +6 -0
- package/build/src/third_party/issue-descriptions/corsAllowCredentialsRequired.md +6 -0
- package/build/src/third_party/issue-descriptions/corsDisabledScheme.md +7 -0
- package/build/src/third_party/issue-descriptions/corsDisallowedByMode.md +7 -0
- package/build/src/third_party/issue-descriptions/corsHeaderDisallowedByPreflightResponse.md +5 -0
- package/build/src/third_party/issue-descriptions/corsInvalidHeaderValues.md +7 -0
- package/build/src/third_party/issue-descriptions/corsLocalNetworkAccessPermissionDenied.md +19 -0
- package/build/src/third_party/issue-descriptions/corsMethodDisallowedByPreflightResponse.md +5 -0
- package/build/src/third_party/issue-descriptions/corsNoCorsRedirectModeNotFollow.md +5 -0
- package/build/src/third_party/issue-descriptions/corsOriginMismatch.md +6 -0
- package/build/src/third_party/issue-descriptions/corsPreflightResponseInvalid.md +5 -0
- package/build/src/third_party/issue-descriptions/corsRedirectContainsCredentials.md +5 -0
- package/build/src/third_party/issue-descriptions/corsWildcardOriginNotAllowed.md +8 -0
- package/build/src/third_party/issue-descriptions/cspEvalViolation.md +9 -0
- package/build/src/third_party/issue-descriptions/cspInlineViolation.md +10 -0
- package/build/src/third_party/issue-descriptions/cspTrustedTypesPolicyViolation.md +5 -0
- package/build/src/third_party/issue-descriptions/cspTrustedTypesSinkViolation.md +8 -0
- package/build/src/third_party/issue-descriptions/cspURLViolation.md +10 -0
- package/build/src/third_party/issue-descriptions/deprecation.md +3 -0
- package/build/src/third_party/issue-descriptions/emailVerificationRequestAccountsEmptyList.md +1 -0
- package/build/src/third_party/issue-descriptions/emailVerificationRequestAccountsHttpNotFound.md +1 -0
- package/build/src/third_party/issue-descriptions/emailVerificationRequestAccountsInvalidContentType.md +1 -0
- package/build/src/third_party/issue-descriptions/emailVerificationRequestAccountsInvalidResponse.md +1 -0
- package/build/src/third_party/issue-descriptions/emailVerificationRequestAccountsNoResponse.md +1 -0
- package/build/src/third_party/issue-descriptions/emailVerificationRequestDnsFetchFailed.md +1 -0
- package/build/src/third_party/issue-descriptions/emailVerificationRequestDnsInvalidRecord.md +1 -0
- package/build/src/third_party/issue-descriptions/emailVerificationRequestEmailVerificationWellKnownHttpNotFound.md +1 -0
- package/build/src/third_party/issue-descriptions/emailVerificationRequestEmailVerificationWellKnownInvalidContentType.md +1 -0
- package/build/src/third_party/issue-descriptions/emailVerificationRequestEmailVerificationWellKnownInvalidResponse.md +1 -0
- package/build/src/third_party/issue-descriptions/emailVerificationRequestEmailVerificationWellKnownNoResponse.md +1 -0
- package/build/src/third_party/issue-descriptions/emailVerificationRequestInvalidEmail.md +1 -0
- package/build/src/third_party/issue-descriptions/emailVerificationRequestJwksHttpNotFound.md +1 -0
- package/build/src/third_party/issue-descriptions/emailVerificationRequestJwksInvalidResponse.md +1 -0
- package/build/src/third_party/issue-descriptions/emailVerificationRequestKeyBindingSigningFailed.md +1 -0
- package/build/src/third_party/issue-descriptions/emailVerificationRequestRpOriginIsOpaque.md +1 -0
- package/build/src/third_party/issue-descriptions/emailVerificationRequestTokenHttpNotFound.md +1 -0
- package/build/src/third_party/issue-descriptions/emailVerificationRequestTokenInvalidContentType.md +1 -0
- package/build/src/third_party/issue-descriptions/emailVerificationRequestTokenInvalidResponse.md +1 -0
- package/build/src/third_party/issue-descriptions/emailVerificationRequestTokenInvalidSdJwt.md +1 -0
- package/build/src/third_party/issue-descriptions/emailVerificationRequestTokenMalformedSdJwt.md +1 -0
- package/build/src/third_party/issue-descriptions/emailVerificationRequestTokenNoResponse.md +1 -0
- package/build/src/third_party/issue-descriptions/emailVerificationRequestTokenVerificationKbInvalidAudience.md +1 -0
- package/build/src/third_party/issue-descriptions/emailVerificationRequestTokenVerificationKbInvalidIssuedAt.md +1 -0
- package/build/src/third_party/issue-descriptions/emailVerificationRequestTokenVerificationKbInvalidNonce.md +1 -0
- package/build/src/third_party/issue-descriptions/emailVerificationRequestTokenVerificationKbInvalidSdHash.md +1 -0
- package/build/src/third_party/issue-descriptions/emailVerificationRequestTokenVerificationKbInvalidTyp.md +1 -0
- package/build/src/third_party/issue-descriptions/emailVerificationRequestTokenVerificationKbMissingAud.md +1 -0
- package/build/src/third_party/issue-descriptions/emailVerificationRequestTokenVerificationKbMissingCnf.md +1 -0
- package/build/src/third_party/issue-descriptions/emailVerificationRequestTokenVerificationKbMissingIat.md +1 -0
- package/build/src/third_party/issue-descriptions/emailVerificationRequestTokenVerificationKbMissingNonce.md +1 -0
- package/build/src/third_party/issue-descriptions/emailVerificationRequestTokenVerificationKbMissingSdHash.md +1 -0
- package/build/src/third_party/issue-descriptions/emailVerificationRequestTokenVerificationKbSignatureFailed.md +1 -0
- package/build/src/third_party/issue-descriptions/emailVerificationRequestTokenVerificationSdJwtInvalidEmail.md +1 -0
- package/build/src/third_party/issue-descriptions/emailVerificationRequestTokenVerificationSdJwtInvalidEmailVerified.md +1 -0
- package/build/src/third_party/issue-descriptions/emailVerificationRequestTokenVerificationSdJwtInvalidHolderKey.md +1 -0
- package/build/src/third_party/issue-descriptions/emailVerificationRequestTokenVerificationSdJwtInvalidIssuedAt.md +1 -0
- package/build/src/third_party/issue-descriptions/emailVerificationRequestTokenVerificationSdJwtInvalidIssuer.md +1 -0
- package/build/src/third_party/issue-descriptions/emailVerificationRequestTokenVerificationSdJwtJwksMissingKeys.md +1 -0
- package/build/src/third_party/issue-descriptions/emailVerificationRequestTokenVerificationSdJwtMissingCnf.md +1 -0
- package/build/src/third_party/issue-descriptions/emailVerificationRequestTokenVerificationSdJwtMissingEmail.md +1 -0
- package/build/src/third_party/issue-descriptions/emailVerificationRequestTokenVerificationSdJwtMissingIat.md +1 -0
- package/build/src/third_party/issue-descriptions/emailVerificationRequestTokenVerificationSdJwtMissingIss.md +1 -0
- package/build/src/third_party/issue-descriptions/emailVerificationRequestTokenVerificationSdJwtSignatureFailed.md +1 -0
- package/build/src/third_party/issue-descriptions/emailVerificationRequestTokenVerificationSdJwtUnsupportedHeaderAlg.md +1 -0
- package/build/src/third_party/issue-descriptions/emailVerificationRequestUserLoggedOut.md +1 -0
- package/build/src/third_party/issue-descriptions/emailVerificationRequestWellKnownAccountsEndpointCrossOrigin.md +1 -0
- package/build/src/third_party/issue-descriptions/emailVerificationRequestWellKnownHttpNotFound.md +1 -0
- package/build/src/third_party/issue-descriptions/emailVerificationRequestWellKnownInvalidContentType.md +1 -0
- package/build/src/third_party/issue-descriptions/emailVerificationRequestWellKnownInvalidResponse.md +1 -0
- package/build/src/third_party/issue-descriptions/emailVerificationRequestWellKnownIssuanceEndpointCrossOrigin.md +1 -0
- package/build/src/third_party/issue-descriptions/emailVerificationRequestWellKnownListEmpty.md +1 -0
- package/build/src/third_party/issue-descriptions/emailVerificationRequestWellKnownMissingAccountsEndpoint.md +1 -0
- package/build/src/third_party/issue-descriptions/emailVerificationRequestWellKnownMissingIssuanceEndpoint.md +1 -0
- package/build/src/third_party/issue-descriptions/emailVerificationRequestWellKnownNoResponse.md +1 -0
- package/build/src/third_party/issue-descriptions/emailVerificationRequestWellKnownUnsupportedSigningAlgorithm.md +1 -0
- package/build/src/third_party/issue-descriptions/federatedAuthRequestAccountsHttpNotFound.md +1 -0
- package/build/src/third_party/issue-descriptions/federatedAuthRequestAccountsInvalidResponse.md +1 -0
- package/build/src/third_party/issue-descriptions/federatedAuthRequestAccountsNoResponse.md +1 -0
- package/build/src/third_party/issue-descriptions/federatedAuthRequestApprovalDeclined.md +1 -0
- package/build/src/third_party/issue-descriptions/federatedAuthRequestCanceled.md +1 -0
- package/build/src/third_party/issue-descriptions/federatedAuthRequestErrorFetchingSignin.md +1 -0
- package/build/src/third_party/issue-descriptions/federatedAuthRequestErrorIdToken.md +1 -0
- package/build/src/third_party/issue-descriptions/federatedAuthRequestIdTokenHttpNotFound.md +1 -0
- package/build/src/third_party/issue-descriptions/federatedAuthRequestIdTokenInvalidRequest.md +1 -0
- package/build/src/third_party/issue-descriptions/federatedAuthRequestIdTokenInvalidResponse.md +1 -0
- package/build/src/third_party/issue-descriptions/federatedAuthRequestIdTokenNoResponse.md +1 -0
- package/build/src/third_party/issue-descriptions/federatedAuthRequestInvalidSigninResponse.md +1 -0
- package/build/src/third_party/issue-descriptions/federatedAuthRequestManifestHttpNotFound.md +1 -0
- package/build/src/third_party/issue-descriptions/federatedAuthRequestManifestInvalidResponse.md +1 -0
- package/build/src/third_party/issue-descriptions/federatedAuthRequestManifestNoResponse.md +1 -0
- package/build/src/third_party/issue-descriptions/federatedAuthRequestTooManyRequests.md +1 -0
- package/build/src/third_party/issue-descriptions/federatedAuthUserInfoRequestInvalidAccountsResponse.md +1 -0
- package/build/src/third_party/issue-descriptions/federatedAuthUserInfoRequestInvalidConfigOrWellKnown.md +1 -0
- package/build/src/third_party/issue-descriptions/federatedAuthUserInfoRequestNoAccountSharingPermission.md +1 -0
- package/build/src/third_party/issue-descriptions/federatedAuthUserInfoRequestNoApiPermission.md +1 -0
- package/build/src/third_party/issue-descriptions/federatedAuthUserInfoRequestNoReturningUserFromFetchedAccounts.md +1 -0
- package/build/src/third_party/issue-descriptions/federatedAuthUserInfoRequestNotIframe.md +1 -0
- package/build/src/third_party/issue-descriptions/federatedAuthUserInfoRequestNotPotentiallyTrustworthy.md +1 -0
- package/build/src/third_party/issue-descriptions/federatedAuthUserInfoRequestNotSameOrigin.md +1 -0
- package/build/src/third_party/issue-descriptions/federatedAuthUserInfoRequestNotSignedInWithIdp.md +1 -0
- package/build/src/third_party/issue-descriptions/fetchingPartitionedBlobURL.md +7 -0
- package/build/src/third_party/issue-descriptions/genericBackUINavigationWouldSkipAd.md +4 -0
- package/build/src/third_party/issue-descriptions/genericFormAriaLabelledByToNonExistingIdError.md +8 -0
- package/build/src/third_party/issue-descriptions/genericFormAutocompleteAttributeEmptyError.md +5 -0
- package/build/src/third_party/issue-descriptions/genericFormDuplicateIdForInputError.md +5 -0
- package/build/src/third_party/issue-descriptions/genericFormEmptyIdAndNameAttributesForInputError.md +5 -0
- package/build/src/third_party/issue-descriptions/genericFormInputAssignedAutocompleteValueToIdOrNameAttributeError.md +5 -0
- package/build/src/third_party/issue-descriptions/genericFormInputHasWrongButWellIntendedAutocompleteValueError.md +5 -0
- package/build/src/third_party/issue-descriptions/genericFormInputWithNoLabelError.md +5 -0
- package/build/src/third_party/issue-descriptions/genericFormLabelForMatchesNonExistingIdError.md +5 -0
- package/build/src/third_party/issue-descriptions/genericFormLabelForNameError.md +5 -0
- package/build/src/third_party/issue-descriptions/genericFormLabelHasNeitherForNorNestedInputError.md +5 -0
- package/build/src/third_party/issue-descriptions/genericFormModelContextMissingToolDescription.md +5 -0
- package/build/src/third_party/issue-descriptions/genericFormModelContextMissingToolName.md +5 -0
- package/build/src/third_party/issue-descriptions/genericFormModelContextParameterMissingName.md +5 -0
- package/build/src/third_party/issue-descriptions/genericFormModelContextParameterMissingTitleAndDescription.md +5 -0
- package/build/src/third_party/issue-descriptions/genericFormModelContextRequiredParameterMissingName.md +5 -0
- package/build/src/third_party/issue-descriptions/genericNavigationEntryMarkedSkippable.md +7 -0
- package/build/src/third_party/issue-descriptions/genericResponseWasBlockedByORB.md +4 -0
- package/build/src/third_party/issue-descriptions/heavyAd.md +10 -0
- package/build/src/third_party/issue-descriptions/mixedContent.md +5 -0
- package/build/src/third_party/issue-descriptions/navigatingPartitionedBlobURL.md +5 -0
- package/build/src/third_party/issue-descriptions/permissionElementActivationDisabled.md +7 -0
- package/build/src/third_party/issue-descriptions/permissionElementActivationDisabledWithOccluder.md +9 -0
- package/build/src/third_party/issue-descriptions/permissionElementActivationDisabledWithOccluderParent.md +9 -0
- package/build/src/third_party/issue-descriptions/permissionElementCspFrameAncestorsMissing.md +5 -0
- package/build/src/third_party/issue-descriptions/permissionElementFencedFrameDisallowed.md +5 -0
- package/build/src/third_party/issue-descriptions/permissionElementFontSizeTooLarge.md +5 -0
- package/build/src/third_party/issue-descriptions/permissionElementFontSizeTooSmall.md +5 -0
- package/build/src/third_party/issue-descriptions/permissionElementGeolocationDeprecated.md +5 -0
- package/build/src/third_party/issue-descriptions/permissionElementInsetBoxShadowUnsupported.md +5 -0
- package/build/src/third_party/issue-descriptions/permissionElementInvalidDisplayStyle.md +5 -0
- package/build/src/third_party/issue-descriptions/permissionElementInvalidSizeValue.md +5 -0
- package/build/src/third_party/issue-descriptions/permissionElementInvalidType.md +5 -0
- package/build/src/third_party/issue-descriptions/permissionElementInvalidTypeActivation.md +5 -0
- package/build/src/third_party/issue-descriptions/permissionElementLowContrast.md +5 -0
- package/build/src/third_party/issue-descriptions/permissionElementNonOpaqueColor.md +5 -0
- package/build/src/third_party/issue-descriptions/permissionElementPaddingBottomUnsupported.md +6 -0
- package/build/src/third_party/issue-descriptions/permissionElementPaddingRightUnsupported.md +6 -0
- package/build/src/third_party/issue-descriptions/permissionElementPermissionsPolicyBlocked.md +5 -0
- package/build/src/third_party/issue-descriptions/permissionElementRegistrationFailed.md +5 -0
- package/build/src/third_party/issue-descriptions/permissionElementRequestInProgress.md +5 -0
- package/build/src/third_party/issue-descriptions/permissionElementSecurityChecksFailed.md +5 -0
- package/build/src/third_party/issue-descriptions/permissionElementTypeNotSupported.md +5 -0
- package/build/src/third_party/issue-descriptions/permissionElementUntrustedEvent.md +7 -0
- package/build/src/third_party/issue-descriptions/placeholderDescriptionForInvisibleIssues.md +3 -0
- package/build/src/third_party/issue-descriptions/propertyRuleInvalidNameIssue.md +3 -0
- package/build/src/third_party/issue-descriptions/propertyRuleIssue.md +7 -0
- package/build/src/third_party/issue-descriptions/selectElementAccessibilityDisallowedOptGroupChild.md +7 -0
- package/build/src/third_party/issue-descriptions/selectElementAccessibilityDisallowedSelectChild.md +7 -0
- package/build/src/third_party/issue-descriptions/selectElementAccessibilityInteractiveContentAttributesSelectDescendant.md +3 -0
- package/build/src/third_party/issue-descriptions/selectElementAccessibilityInteractiveContentLegendChild.md +3 -0
- package/build/src/third_party/issue-descriptions/selectElementAccessibilityInteractiveContentOptionChild.md +3 -0
- package/build/src/third_party/issue-descriptions/selectElementAccessibilityNonPhrasingContentOptionChild.md +3 -0
- package/build/src/third_party/issue-descriptions/selectivePermissionsIntervention.md +7 -0
- package/build/src/third_party/issue-descriptions/sharedArrayBuffer.md +7 -0
- package/build/src/third_party/issue-descriptions/sharedDictionaryUseErrorCrossOriginNoCorsRequest.md +1 -0
- package/build/src/third_party/issue-descriptions/sharedDictionaryUseErrorDictionaryLoadFailure.md +3 -0
- package/build/src/third_party/issue-descriptions/sharedDictionaryUseErrorMatchingDictionaryNotUsed.md +3 -0
- package/build/src/third_party/issue-descriptions/sharedDictionaryUseErrorUnexpectedContentDictionaryHeader.md +1 -0
- package/build/src/third_party/issue-descriptions/sharedDictionaryWriteErrorCossOriginNoCorsRequest.md +1 -0
- package/build/src/third_party/issue-descriptions/sharedDictionaryWriteErrorDisallowedBySettings.md +1 -0
- package/build/src/third_party/issue-descriptions/sharedDictionaryWriteErrorExpiredResponse.md +3 -0
- package/build/src/third_party/issue-descriptions/sharedDictionaryWriteErrorFeatureDisabled.md +3 -0
- package/build/src/third_party/issue-descriptions/sharedDictionaryWriteErrorInsufficientResources.md +1 -0
- package/build/src/third_party/issue-descriptions/sharedDictionaryWriteErrorInvalidMatchField.md +1 -0
- package/build/src/third_party/issue-descriptions/sharedDictionaryWriteErrorInvalidStructuredHeader.md +1 -0
- package/build/src/third_party/issue-descriptions/sharedDictionaryWriteErrorInvalidTTLField.md +1 -0
- package/build/src/third_party/issue-descriptions/sharedDictionaryWriteErrorNavigationRequest.md +3 -0
- package/build/src/third_party/issue-descriptions/sharedDictionaryWriteErrorNoMatchField.md +1 -0
- package/build/src/third_party/issue-descriptions/sharedDictionaryWriteErrorNonIntegerTTLField.md +1 -0
- package/build/src/third_party/issue-descriptions/sharedDictionaryWriteErrorNonListMatchDestField.md +1 -0
- package/build/src/third_party/issue-descriptions/sharedDictionaryWriteErrorNonSecureContext.md +3 -0
- package/build/src/third_party/issue-descriptions/sharedDictionaryWriteErrorNonStringIdField.md +1 -0
- package/build/src/third_party/issue-descriptions/sharedDictionaryWriteErrorNonStringInMatchDestList.md +1 -0
- package/build/src/third_party/issue-descriptions/sharedDictionaryWriteErrorNonStringMatchField.md +1 -0
- package/build/src/third_party/issue-descriptions/sharedDictionaryWriteErrorNonTokenTypeField.md +1 -0
- package/build/src/third_party/issue-descriptions/sharedDictionaryWriteErrorRequestAborted.md +1 -0
- package/build/src/third_party/issue-descriptions/sharedDictionaryWriteErrorShuttingDown.md +1 -0
- package/build/src/third_party/issue-descriptions/sharedDictionaryWriteErrorTooLongIdField.md +3 -0
- package/build/src/third_party/issue-descriptions/sharedDictionaryWriteErrorUnsupportedType.md +3 -0
- package/build/src/third_party/issue-descriptions/sriInvalidSignatureHeader.md +14 -0
- package/build/src/third_party/issue-descriptions/sriInvalidSignatureInputHeader.md +15 -0
- package/build/src/third_party/issue-descriptions/sriMissingSignatureHeader.md +8 -0
- package/build/src/third_party/issue-descriptions/sriMissingSignatureInputHeader.md +7 -0
- package/build/src/third_party/issue-descriptions/sriSignatureHeaderValueIsIncorrectLength.md +11 -0
- package/build/src/third_party/issue-descriptions/sriSignatureHeaderValueIsNotByteSequence.md +14 -0
- package/build/src/third_party/issue-descriptions/sriSignatureHeaderValueIsParameterized.md +15 -0
- package/build/src/third_party/issue-descriptions/sriSignatureInputHeaderInvalidComponentName.md +8 -0
- package/build/src/third_party/issue-descriptions/sriSignatureInputHeaderInvalidComponentType.md +13 -0
- package/build/src/third_party/issue-descriptions/sriSignatureInputHeaderInvalidDerivedComponentParameter.md +4 -0
- package/build/src/third_party/issue-descriptions/sriSignatureInputHeaderInvalidHeaderComponentParameter.md +5 -0
- package/build/src/third_party/issue-descriptions/sriSignatureInputHeaderInvalidParameter.md +11 -0
- package/build/src/third_party/issue-descriptions/sriSignatureInputHeaderKeyIdLength.md +12 -0
- package/build/src/third_party/issue-descriptions/sriSignatureInputHeaderMissingLabel.md +6 -0
- package/build/src/third_party/issue-descriptions/sriSignatureInputHeaderMissingRequiredParameters.md +8 -0
- package/build/src/third_party/issue-descriptions/sriSignatureInputHeaderValueMissingComponents.md +11 -0
- package/build/src/third_party/issue-descriptions/sriSignatureInputHeaderValueNotInnerList.md +11 -0
- package/build/src/third_party/issue-descriptions/sriValidationFailedIntegrityMismatch.md +10 -0
- package/build/src/third_party/issue-descriptions/sriValidationFailedInvalidLength.md +5 -0
- package/build/src/third_party/issue-descriptions/sriValidationFailedSignatureExpired.md +6 -0
- package/build/src/third_party/issue-descriptions/sriValidationFailedSignatureMismatch.md +11 -0
- package/build/src/third_party/issue-descriptions/stylesheetLateImport.md +4 -0
- package/build/src/third_party/issue-descriptions/stylesheetRequestFailed.md +3 -0
- package/build/src/third_party/issue-descriptions/summaryElementAccessibilityInteractiveContentSummaryDescendant.md +3 -0
- package/build/src/third_party/issue-descriptions/unencodedDigestIncorrectDigestLength.md +12 -0
- package/build/src/third_party/issue-descriptions/unencodedDigestIncorrectDigestType.md +17 -0
- package/build/src/third_party/issue-descriptions/unencodedDigestMalformedDictionary.md +14 -0
- package/build/src/third_party/issue-descriptions/unencodedDigestUnknownAlgorithm.md +15 -0
- package/build/src/third_party/lighthouse-devtools-mcp-bundle.js +61598 -0
- package/build/src/tools/ToolDefinition.js +73 -0
- package/build/src/tools/categories.js +36 -0
- package/build/src/tools/console.js +98 -0
- package/build/src/tools/emulation.js +84 -0
- package/build/src/tools/extensions.js +101 -0
- package/build/src/tools/input.js +469 -0
- package/build/src/tools/lighthouse.js +136 -0
- package/build/src/tools/memory.js +227 -0
- package/build/src/tools/network.js +125 -0
- package/build/src/tools/pages.js +419 -0
- package/build/src/tools/performance.js +200 -0
- package/build/src/tools/screencast.js +117 -0
- package/build/src/tools/screenshot.js +169 -0
- package/build/src/tools/script.js +151 -0
- package/build/src/tools/slim/tools.js +88 -0
- package/build/src/tools/snapshot.js +61 -0
- package/build/src/tools/thirdPartyDeveloper.js +85 -0
- package/build/src/tools/tools.js +56 -0
- package/build/src/tools/webmcp.js +66 -0
- package/build/src/trace-processing/parse.js +85 -0
- package/build/src/types.js +7 -0
- package/build/src/utils/check-for-updates.js +74 -0
- package/build/src/utils/files.js +61 -0
- package/build/src/utils/id.js +16 -0
- package/build/src/utils/keyboard.js +297 -0
- package/build/src/utils/pagination.js +50 -0
- package/build/src/utils/string.js +37 -0
- package/build/src/utils/types.js +7 -0
- package/build/src/version.js +10 -0
- package/package.json +100 -0
- package/skills/a11y-debugging/SKILL.md +89 -0
- package/skills/a11y-debugging/references/a11y-snippets.md +92 -0
- package/skills/chrome-devtools/SKILL.md +72 -0
- package/skills/chrome-devtools-cli/SKILL.md +153 -0
- package/skills/chrome-devtools-cli/references/installation.md +14 -0
- package/skills/debug-optimize-lcp/SKILL.md +121 -0
- package/skills/debug-optimize-lcp/references/elements-and-size.md +27 -0
- package/skills/debug-optimize-lcp/references/lcp-breakdown.md +23 -0
- package/skills/debug-optimize-lcp/references/lcp-snippets.md +79 -0
- package/skills/debug-optimize-lcp/references/optimization-strategies.md +38 -0
- package/skills/memory-leak-debugging/SKILL.md +50 -0
- package/skills/memory-leak-debugging/references/common-leaks.md +33 -0
- package/skills/memory-leak-debugging/references/compare_snapshots.js +109 -0
- package/skills/memory-leak-debugging/references/memlab.md +29 -0
- package/skills/troubleshooting/SKILL.md +98 -0
|
@@ -0,0 +1,238 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* @license
|
|
3
|
+
* Copyright 2025 Google LLC
|
|
4
|
+
* SPDX-License-Identifier: Apache-2.0
|
|
5
|
+
* */
|
|
6
|
+
import { isUtf8 } from 'node:buffer';
|
|
7
|
+
import { DevTools, } from '../third_party/index.js';
|
|
8
|
+
const BODY_CONTEXT_SIZE_LIMIT = 10000;
|
|
9
|
+
export class NetworkFormatter {
|
|
10
|
+
#request;
|
|
11
|
+
#options;
|
|
12
|
+
#requestBody;
|
|
13
|
+
#responseBody;
|
|
14
|
+
#requestBodyFilePath;
|
|
15
|
+
#responseBodyFilePath;
|
|
16
|
+
constructor(request, options) {
|
|
17
|
+
this.#request = request;
|
|
18
|
+
this.#options = options;
|
|
19
|
+
}
|
|
20
|
+
static async from(request, options) {
|
|
21
|
+
const instance = new NetworkFormatter(request, options);
|
|
22
|
+
if (options.fetchData) {
|
|
23
|
+
await instance.#loadDetailedData();
|
|
24
|
+
}
|
|
25
|
+
return instance;
|
|
26
|
+
}
|
|
27
|
+
async #loadDetailedData() {
|
|
28
|
+
// Load Request Body
|
|
29
|
+
if (this.#request.hasPostData()) {
|
|
30
|
+
let data;
|
|
31
|
+
try {
|
|
32
|
+
data =
|
|
33
|
+
this.#request.postData() ?? (await this.#request.fetchPostData());
|
|
34
|
+
}
|
|
35
|
+
catch {
|
|
36
|
+
// Ignore parsing errors
|
|
37
|
+
}
|
|
38
|
+
const requestBodyNotAvailableMessage = '<Request body not available anymore>';
|
|
39
|
+
if (this.#options.requestFilePath) {
|
|
40
|
+
if (!this.#options.saveFile) {
|
|
41
|
+
throw new Error('saveFile is not provided');
|
|
42
|
+
}
|
|
43
|
+
if (data) {
|
|
44
|
+
const result = await this.#options.saveFile(Buffer.from(data), this.#options.requestFilePath, '.network-request');
|
|
45
|
+
this.#requestBodyFilePath = result.filename;
|
|
46
|
+
}
|
|
47
|
+
else {
|
|
48
|
+
this.#requestBody = requestBodyNotAvailableMessage;
|
|
49
|
+
}
|
|
50
|
+
}
|
|
51
|
+
else {
|
|
52
|
+
if (data) {
|
|
53
|
+
this.#requestBody = getSizeLimitedString(data, BODY_CONTEXT_SIZE_LIMIT);
|
|
54
|
+
}
|
|
55
|
+
else {
|
|
56
|
+
this.#requestBody = requestBodyNotAvailableMessage;
|
|
57
|
+
}
|
|
58
|
+
}
|
|
59
|
+
}
|
|
60
|
+
// Load Response Body
|
|
61
|
+
const response = this.#request.response();
|
|
62
|
+
if (response) {
|
|
63
|
+
const responseBodyNotAvailableMessage = '<Response body not available anymore>';
|
|
64
|
+
if (this.#options.responseFilePath) {
|
|
65
|
+
try {
|
|
66
|
+
const buffer = await response.buffer();
|
|
67
|
+
if (!this.#options.saveFile) {
|
|
68
|
+
throw new Error('saveFile is not provided');
|
|
69
|
+
}
|
|
70
|
+
const result = await this.#options.saveFile(buffer, this.#options.responseFilePath, '.network-response');
|
|
71
|
+
this.#responseBodyFilePath = result.filename;
|
|
72
|
+
}
|
|
73
|
+
catch {
|
|
74
|
+
// Flatten error handling for buffer() failure and save failure
|
|
75
|
+
}
|
|
76
|
+
if (!this.#responseBodyFilePath) {
|
|
77
|
+
this.#responseBody = responseBodyNotAvailableMessage;
|
|
78
|
+
}
|
|
79
|
+
}
|
|
80
|
+
else {
|
|
81
|
+
this.#responseBody = await this.#getFormattedResponseBody(response, BODY_CONTEXT_SIZE_LIMIT);
|
|
82
|
+
}
|
|
83
|
+
}
|
|
84
|
+
}
|
|
85
|
+
toString() {
|
|
86
|
+
return convertNetworkRequestConciseToString(this.toJSON());
|
|
87
|
+
}
|
|
88
|
+
toStringDetailed() {
|
|
89
|
+
return converNetworkRequestDetailedToStringDetailed(this.toJSONDetailed());
|
|
90
|
+
}
|
|
91
|
+
toJSON() {
|
|
92
|
+
return {
|
|
93
|
+
requestId: this.#options.requestId,
|
|
94
|
+
method: this.#request.method(),
|
|
95
|
+
url: this.#request.url(),
|
|
96
|
+
status: this.#getStatusFromRequest(this.#request),
|
|
97
|
+
selectedInDevToolsUI: this.#options.selectedInDevToolsUI,
|
|
98
|
+
};
|
|
99
|
+
}
|
|
100
|
+
#redactNetworkHeaders(headers) {
|
|
101
|
+
const headersList = Object.entries(headers).map(item => {
|
|
102
|
+
return { name: item[0], value: item[1] };
|
|
103
|
+
});
|
|
104
|
+
const redacted = DevTools.NetworkRequestFormatter.sanitizeHeaders(headersList);
|
|
105
|
+
return redacted.reduce((acc, item) => {
|
|
106
|
+
acc[item.name] = item.value;
|
|
107
|
+
return acc;
|
|
108
|
+
}, {});
|
|
109
|
+
}
|
|
110
|
+
toJSONDetailed() {
|
|
111
|
+
const redirectChain = this.#request.redirectChain();
|
|
112
|
+
const formattedRedirectChain = redirectChain.reverse().map(request => {
|
|
113
|
+
const id = this.#options.requestIdResolver
|
|
114
|
+
? this.#options.requestIdResolver(request)
|
|
115
|
+
: undefined;
|
|
116
|
+
const formatter = new NetworkFormatter(request, {
|
|
117
|
+
requestId: id,
|
|
118
|
+
saveFile: this.#options.saveFile,
|
|
119
|
+
redactNetworkHeaders: this.#options.redactNetworkHeaders,
|
|
120
|
+
});
|
|
121
|
+
return formatter.toJSON();
|
|
122
|
+
});
|
|
123
|
+
const responseHeaders = this.#request.response()?.headers();
|
|
124
|
+
return {
|
|
125
|
+
...this.toJSON(),
|
|
126
|
+
requestHeaders: this.#options.redactNetworkHeaders
|
|
127
|
+
? this.#redactNetworkHeaders(this.#request.headers())
|
|
128
|
+
: this.#request.headers(),
|
|
129
|
+
requestBody: this.#requestBody,
|
|
130
|
+
requestBodyFilePath: this.#requestBodyFilePath,
|
|
131
|
+
responseHeaders: this.#options.redactNetworkHeaders && responseHeaders
|
|
132
|
+
? this.#redactNetworkHeaders(responseHeaders)
|
|
133
|
+
: this.#request.response()?.headers(),
|
|
134
|
+
responseBody: this.#responseBody,
|
|
135
|
+
responseBodyFilePath: this.#responseBodyFilePath,
|
|
136
|
+
failure: this.#request.failure()?.errorText,
|
|
137
|
+
redirectChain: formattedRedirectChain.length
|
|
138
|
+
? formattedRedirectChain
|
|
139
|
+
: undefined,
|
|
140
|
+
};
|
|
141
|
+
}
|
|
142
|
+
#getStatusFromRequest(request) {
|
|
143
|
+
const httpResponse = request.response();
|
|
144
|
+
const failure = request.failure();
|
|
145
|
+
let status;
|
|
146
|
+
if (httpResponse) {
|
|
147
|
+
status = httpResponse.status().toString();
|
|
148
|
+
}
|
|
149
|
+
else if (failure) {
|
|
150
|
+
status = failure.errorText;
|
|
151
|
+
}
|
|
152
|
+
else {
|
|
153
|
+
status = 'pending';
|
|
154
|
+
}
|
|
155
|
+
return status;
|
|
156
|
+
}
|
|
157
|
+
async #getFormattedResponseBody(httpResponse, sizeLimit = BODY_CONTEXT_SIZE_LIMIT) {
|
|
158
|
+
try {
|
|
159
|
+
const responseBuffer = await httpResponse.buffer();
|
|
160
|
+
if (isUtf8(responseBuffer)) {
|
|
161
|
+
const responseAsTest = responseBuffer.toString('utf-8');
|
|
162
|
+
if (responseAsTest.length === 0) {
|
|
163
|
+
return '<empty response>';
|
|
164
|
+
}
|
|
165
|
+
return getSizeLimitedString(responseAsTest, sizeLimit);
|
|
166
|
+
}
|
|
167
|
+
return '<binary data>';
|
|
168
|
+
}
|
|
169
|
+
catch {
|
|
170
|
+
return '<not available anymore>';
|
|
171
|
+
}
|
|
172
|
+
}
|
|
173
|
+
}
|
|
174
|
+
function getSizeLimitedString(text, sizeLimit) {
|
|
175
|
+
if (text.length > sizeLimit) {
|
|
176
|
+
return text.substring(0, sizeLimit) + '... <truncated>';
|
|
177
|
+
}
|
|
178
|
+
return text;
|
|
179
|
+
}
|
|
180
|
+
function convertNetworkRequestConciseToString(data) {
|
|
181
|
+
// TODO truncate the URL
|
|
182
|
+
return `reqid=${data.requestId} ${data.method} ${data.url} [${data.status}]${data.selectedInDevToolsUI ? ` [selected in the DevTools Network panel]` : ''}`;
|
|
183
|
+
}
|
|
184
|
+
function formatHeadlers(headers) {
|
|
185
|
+
const response = [];
|
|
186
|
+
for (const [name, value] of Object.entries(headers)) {
|
|
187
|
+
response.push(`- ${name}:${value}`);
|
|
188
|
+
}
|
|
189
|
+
return response;
|
|
190
|
+
}
|
|
191
|
+
function converNetworkRequestDetailedToStringDetailed(data) {
|
|
192
|
+
const response = [];
|
|
193
|
+
response.push(`## Request ${data.url}`);
|
|
194
|
+
response.push(`Status: ${data.status}`);
|
|
195
|
+
response.push(`### Request Headers`);
|
|
196
|
+
for (const line of formatHeadlers(data.requestHeaders)) {
|
|
197
|
+
response.push(line);
|
|
198
|
+
}
|
|
199
|
+
if (data.requestBody) {
|
|
200
|
+
response.push(`### Request Body`);
|
|
201
|
+
response.push(data.requestBody);
|
|
202
|
+
}
|
|
203
|
+
else if (data.requestBodyFilePath) {
|
|
204
|
+
response.push(`### Request Body`);
|
|
205
|
+
response.push(`Saved to ${data.requestBodyFilePath}.`);
|
|
206
|
+
}
|
|
207
|
+
if (data.responseHeaders) {
|
|
208
|
+
response.push(`### Response Headers`);
|
|
209
|
+
for (const line of formatHeadlers(data.responseHeaders)) {
|
|
210
|
+
response.push(line);
|
|
211
|
+
}
|
|
212
|
+
}
|
|
213
|
+
if (data.responseBody) {
|
|
214
|
+
response.push(`### Response Body`);
|
|
215
|
+
response.push(data.responseBody);
|
|
216
|
+
}
|
|
217
|
+
else if (data.responseBodyFilePath) {
|
|
218
|
+
response.push(`### Response Body`);
|
|
219
|
+
response.push(`Saved to ${data.responseBodyFilePath}.`);
|
|
220
|
+
}
|
|
221
|
+
if (data.failure) {
|
|
222
|
+
response.push(`### Request failed with`);
|
|
223
|
+
response.push(data.failure);
|
|
224
|
+
}
|
|
225
|
+
const redirectChain = data.redirectChain;
|
|
226
|
+
if (redirectChain?.length) {
|
|
227
|
+
response.push(`### Redirect chain`);
|
|
228
|
+
let indent = 0;
|
|
229
|
+
// `redirectChain` is already ordered by toJSONDetailed(); don't reverse it
|
|
230
|
+
// again here or the text output contradicts structuredContent (the JSON).
|
|
231
|
+
for (const request of redirectChain) {
|
|
232
|
+
response.push(`${' '.repeat(indent)}${convertNetworkRequestConciseToString(request)}`);
|
|
233
|
+
indent++;
|
|
234
|
+
}
|
|
235
|
+
}
|
|
236
|
+
return response.join('\n');
|
|
237
|
+
}
|
|
238
|
+
//# sourceMappingURL=NetworkFormatter.js.map
|
|
@@ -0,0 +1,135 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* @license
|
|
3
|
+
* Copyright 2025 Google LLC
|
|
4
|
+
* SPDX-License-Identifier: Apache-2.0
|
|
5
|
+
*/
|
|
6
|
+
export class SnapshotFormatter {
|
|
7
|
+
#snapshot;
|
|
8
|
+
constructor(snapshot) {
|
|
9
|
+
this.#snapshot = snapshot;
|
|
10
|
+
}
|
|
11
|
+
toString() {
|
|
12
|
+
const chunks = [];
|
|
13
|
+
const root = this.#snapshot.root;
|
|
14
|
+
// Top-level content of the snapshot.
|
|
15
|
+
if (!this.#snapshot.verbose &&
|
|
16
|
+
this.#snapshot.hasSelectedElement &&
|
|
17
|
+
!this.#snapshot.selectedElementUid) {
|
|
18
|
+
chunks.push(`Note: there is a selected element in the DevTools Elements panel but it is not included into the current a11y tree snapshot.
|
|
19
|
+
Get a verbose snapshot to include all elements if you are interested in the selected element.\n\n`);
|
|
20
|
+
}
|
|
21
|
+
chunks.push(this.#formatNode(root, 0));
|
|
22
|
+
return chunks.join('');
|
|
23
|
+
}
|
|
24
|
+
toJSON() {
|
|
25
|
+
return this.#nodeToJSON(this.#snapshot.root);
|
|
26
|
+
}
|
|
27
|
+
#formatNode(node, depth = 0) {
|
|
28
|
+
const chunks = [];
|
|
29
|
+
const attributes = this.#getAttributes(node);
|
|
30
|
+
const line = ' '.repeat(depth * 2) +
|
|
31
|
+
attributes.join(' ') +
|
|
32
|
+
(node.id === this.#snapshot.selectedElementUid
|
|
33
|
+
? ' [selected in the DevTools Elements panel]'
|
|
34
|
+
: '') +
|
|
35
|
+
'\n';
|
|
36
|
+
chunks.push(line);
|
|
37
|
+
for (const child of node.children) {
|
|
38
|
+
chunks.push(this.#formatNode(child, depth + 1));
|
|
39
|
+
}
|
|
40
|
+
return chunks.join('');
|
|
41
|
+
}
|
|
42
|
+
#nodeToJSON(node) {
|
|
43
|
+
const rawAttrs = this.#getAttributesMap(node);
|
|
44
|
+
const children = node.children.map(child => this.#nodeToJSON(child));
|
|
45
|
+
const result = structuredClone(rawAttrs);
|
|
46
|
+
if (children.length > 0) {
|
|
47
|
+
result.children = children;
|
|
48
|
+
}
|
|
49
|
+
return result;
|
|
50
|
+
}
|
|
51
|
+
#getAttributes(serializedAXNodeRoot) {
|
|
52
|
+
const attributes = [`uid=${serializedAXNodeRoot.id}`];
|
|
53
|
+
if (serializedAXNodeRoot.role) {
|
|
54
|
+
attributes.push(serializedAXNodeRoot.role === 'none'
|
|
55
|
+
? 'ignored'
|
|
56
|
+
: serializedAXNodeRoot.role);
|
|
57
|
+
}
|
|
58
|
+
if (serializedAXNodeRoot.name) {
|
|
59
|
+
attributes.push(`"${serializedAXNodeRoot.name}"`);
|
|
60
|
+
}
|
|
61
|
+
const simpleAttrs = this.#getAttributesMap(serializedAXNodeRoot,
|
|
62
|
+
/* excludeSpecial */ true);
|
|
63
|
+
for (const attr of Object.keys(serializedAXNodeRoot).sort()) {
|
|
64
|
+
if (excludedAttributes.has(attr)) {
|
|
65
|
+
continue;
|
|
66
|
+
}
|
|
67
|
+
const mapped = booleanPropertyMap[attr];
|
|
68
|
+
if (mapped && simpleAttrs[mapped]) {
|
|
69
|
+
attributes.push(mapped);
|
|
70
|
+
}
|
|
71
|
+
const val = simpleAttrs[attr];
|
|
72
|
+
if (val === true) {
|
|
73
|
+
attributes.push(attr);
|
|
74
|
+
}
|
|
75
|
+
else if (typeof val === 'string' || typeof val === 'number') {
|
|
76
|
+
attributes.push(`${attr}="${val}"`);
|
|
77
|
+
}
|
|
78
|
+
}
|
|
79
|
+
return attributes;
|
|
80
|
+
}
|
|
81
|
+
#getAttributesMap(node, excludeSpecial = false) {
|
|
82
|
+
const result = {};
|
|
83
|
+
if (!excludeSpecial) {
|
|
84
|
+
result.id = node.id;
|
|
85
|
+
if (node.role) {
|
|
86
|
+
result.role = node.role;
|
|
87
|
+
}
|
|
88
|
+
if (node.name) {
|
|
89
|
+
result.name = node.name;
|
|
90
|
+
}
|
|
91
|
+
}
|
|
92
|
+
// Re-implementing the exact logic from original function for #getAttributes to be safe:
|
|
93
|
+
return {
|
|
94
|
+
...result,
|
|
95
|
+
...this.#extractedAttributes(node),
|
|
96
|
+
};
|
|
97
|
+
}
|
|
98
|
+
#extractedAttributes(node) {
|
|
99
|
+
const result = {};
|
|
100
|
+
for (const attr of Object.keys(node).sort()) {
|
|
101
|
+
if (excludedAttributes.has(attr)) {
|
|
102
|
+
continue;
|
|
103
|
+
}
|
|
104
|
+
const value = node[attr];
|
|
105
|
+
if (typeof value === 'boolean') {
|
|
106
|
+
if (booleanPropertyMap[attr]) {
|
|
107
|
+
result[booleanPropertyMap[attr]] = true;
|
|
108
|
+
}
|
|
109
|
+
if (value) {
|
|
110
|
+
result[attr] = true;
|
|
111
|
+
}
|
|
112
|
+
}
|
|
113
|
+
else if (typeof value === 'string' || typeof value === 'number') {
|
|
114
|
+
result[attr] = value;
|
|
115
|
+
}
|
|
116
|
+
}
|
|
117
|
+
return result;
|
|
118
|
+
}
|
|
119
|
+
}
|
|
120
|
+
const booleanPropertyMap = {
|
|
121
|
+
disabled: 'disableable',
|
|
122
|
+
expanded: 'expandable',
|
|
123
|
+
focused: 'focusable',
|
|
124
|
+
selected: 'selectable',
|
|
125
|
+
};
|
|
126
|
+
const excludedAttributes = new Set([
|
|
127
|
+
'id',
|
|
128
|
+
'role',
|
|
129
|
+
'name',
|
|
130
|
+
'elementHandle',
|
|
131
|
+
'children',
|
|
132
|
+
'backendNodeId',
|
|
133
|
+
'loaderId',
|
|
134
|
+
]);
|
|
135
|
+
//# sourceMappingURL=SnapshotFormatter.js.map
|
|
@@ -0,0 +1,153 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* @license
|
|
3
|
+
* Copyright 2026 Google LLC
|
|
4
|
+
* SPDX-License-Identifier: Apache-2.0
|
|
5
|
+
*/
|
|
6
|
+
import { ensureBrowserConnected, ensureBrowserLaunched } from './browser.js';
|
|
7
|
+
import { loadIssueDescriptions } from './issue-descriptions.js';
|
|
8
|
+
import { logger } from './logger.js';
|
|
9
|
+
import { McpContext } from './McpContext.js';
|
|
10
|
+
import { Mutex } from './Mutex.js';
|
|
11
|
+
import { ClearcutLogger } from './telemetry/ClearcutLogger.js';
|
|
12
|
+
import { FilePersistence } from './telemetry/persistence.js';
|
|
13
|
+
import { McpServer, SetLevelRequestSchema, ListRootsResultSchema, RootsListChangedNotificationSchema, } from './third_party/index.js';
|
|
14
|
+
import { ToolHandler } from './ToolHandler.js';
|
|
15
|
+
import { createTools } from './tools/tools.js';
|
|
16
|
+
import { VERSION } from './version.js';
|
|
17
|
+
export { buildFlag } from './ToolHandler.js';
|
|
18
|
+
export async function createMcpServer(serverArgs, options) {
|
|
19
|
+
if (serverArgs.usageStatistics) {
|
|
20
|
+
ClearcutLogger.initialize({
|
|
21
|
+
persistence: new FilePersistence(),
|
|
22
|
+
logFile: serverArgs.logFile,
|
|
23
|
+
appVersion: VERSION,
|
|
24
|
+
clearcutEndpoint: serverArgs.clearcutEndpoint,
|
|
25
|
+
clearcutForceFlushIntervalMs: serverArgs.clearcutForceFlushIntervalMs,
|
|
26
|
+
clearcutIncludePidHeader: serverArgs.clearcutIncludePidHeader,
|
|
27
|
+
});
|
|
28
|
+
}
|
|
29
|
+
const server = new McpServer({
|
|
30
|
+
name: 'chrome_devtools',
|
|
31
|
+
title: 'Chrome DevTools MCP server',
|
|
32
|
+
version: VERSION,
|
|
33
|
+
}, { capabilities: { logging: {} } });
|
|
34
|
+
server.server.setRequestHandler(SetLevelRequestSchema, () => {
|
|
35
|
+
return {};
|
|
36
|
+
});
|
|
37
|
+
const updateRoots = async () => {
|
|
38
|
+
if (!server.server.getClientCapabilities()?.roots) {
|
|
39
|
+
return;
|
|
40
|
+
}
|
|
41
|
+
try {
|
|
42
|
+
const roots = await server.server.request({ method: 'roots/list' }, ListRootsResultSchema);
|
|
43
|
+
context?.setRoots(roots.roots);
|
|
44
|
+
}
|
|
45
|
+
catch (e) {
|
|
46
|
+
logger?.('Failed to list roots', e);
|
|
47
|
+
}
|
|
48
|
+
};
|
|
49
|
+
server.server.oninitialized = () => {
|
|
50
|
+
const clientName = server.server.getClientVersion()?.name;
|
|
51
|
+
if (clientName) {
|
|
52
|
+
ClearcutLogger.get()?.setClientName(clientName);
|
|
53
|
+
}
|
|
54
|
+
if (server.server.getClientCapabilities()?.roots) {
|
|
55
|
+
void updateRoots();
|
|
56
|
+
server.server.setNotificationHandler(RootsListChangedNotificationSchema, () => {
|
|
57
|
+
void updateRoots();
|
|
58
|
+
});
|
|
59
|
+
}
|
|
60
|
+
};
|
|
61
|
+
let context;
|
|
62
|
+
async function getContext() {
|
|
63
|
+
const chromeArgs = (serverArgs.chromeArg ?? []).map(String);
|
|
64
|
+
const ignoreDefaultChromeArgs = (serverArgs.ignoreDefaultChromeArg ?? []).map(String);
|
|
65
|
+
if (serverArgs.proxyServer) {
|
|
66
|
+
chromeArgs.push(`--proxy-server=${serverArgs.proxyServer}`);
|
|
67
|
+
}
|
|
68
|
+
const devtools = serverArgs.experimentalDevtools ?? false;
|
|
69
|
+
const blocklist = serverArgs.blockedUrlPattern
|
|
70
|
+
? serverArgs.blockedUrlPattern.map(String)
|
|
71
|
+
: undefined;
|
|
72
|
+
const allowlist = serverArgs.allowedUrlPattern
|
|
73
|
+
? serverArgs.allowedUrlPattern.map(String)
|
|
74
|
+
: undefined;
|
|
75
|
+
const browser = serverArgs.browserUrl || serverArgs.wsEndpoint || serverArgs.autoConnect
|
|
76
|
+
? await ensureBrowserConnected({
|
|
77
|
+
browserURL: serverArgs.browserUrl,
|
|
78
|
+
wsEndpoint: serverArgs.wsEndpoint,
|
|
79
|
+
wsHeaders: serverArgs.wsHeaders,
|
|
80
|
+
// Important: only pass channel, if autoConnect is true.
|
|
81
|
+
channel: serverArgs.autoConnect
|
|
82
|
+
? serverArgs.channel
|
|
83
|
+
: undefined,
|
|
84
|
+
userDataDir: serverArgs.userDataDir,
|
|
85
|
+
devtools,
|
|
86
|
+
blocklist,
|
|
87
|
+
allowlist,
|
|
88
|
+
})
|
|
89
|
+
: await ensureBrowserLaunched({
|
|
90
|
+
headless: serverArgs.headless,
|
|
91
|
+
executablePath: serverArgs.executablePath,
|
|
92
|
+
channel: serverArgs.channel,
|
|
93
|
+
isolated: serverArgs.isolated ?? false,
|
|
94
|
+
userDataDir: serverArgs.userDataDir,
|
|
95
|
+
logFile: options.logFile,
|
|
96
|
+
viewport: serverArgs.viewport,
|
|
97
|
+
chromeArgs,
|
|
98
|
+
ignoreDefaultChromeArgs,
|
|
99
|
+
acceptInsecureCerts: serverArgs.acceptInsecureCerts,
|
|
100
|
+
devtools,
|
|
101
|
+
enableExtensions: serverArgs.categoryExtensions,
|
|
102
|
+
viaCli: serverArgs.viaCli,
|
|
103
|
+
blocklist,
|
|
104
|
+
allowlist,
|
|
105
|
+
});
|
|
106
|
+
if (context?.browser !== browser) {
|
|
107
|
+
context = await McpContext.from(browser, logger, {
|
|
108
|
+
experimentalDevToolsDebugging: devtools,
|
|
109
|
+
experimentalIncludeAllPages: serverArgs.experimentalIncludeAllPages,
|
|
110
|
+
performanceCrux: serverArgs.performanceCrux,
|
|
111
|
+
arc: serverArgs.arc,
|
|
112
|
+
hasNetworkBlockOrAllowlist: Boolean((blocklist && blocklist.length > 0) ||
|
|
113
|
+
(allowlist && allowlist.length > 0)),
|
|
114
|
+
});
|
|
115
|
+
await updateRoots();
|
|
116
|
+
}
|
|
117
|
+
return context;
|
|
118
|
+
}
|
|
119
|
+
const toolMutex = new Mutex();
|
|
120
|
+
function registerTool(tool) {
|
|
121
|
+
const toolHandler = new ToolHandler(tool, serverArgs, getContext, toolMutex);
|
|
122
|
+
if (!toolHandler.shouldRegister) {
|
|
123
|
+
return;
|
|
124
|
+
}
|
|
125
|
+
server.registerTool(tool.name, {
|
|
126
|
+
description: tool.description,
|
|
127
|
+
inputSchema: toolHandler.registeredInputSchema,
|
|
128
|
+
annotations: tool.annotations,
|
|
129
|
+
}, async (params) => {
|
|
130
|
+
return await toolHandler.handle(params);
|
|
131
|
+
});
|
|
132
|
+
}
|
|
133
|
+
const tools = createTools(serverArgs);
|
|
134
|
+
for (const tool of tools) {
|
|
135
|
+
registerTool(tool);
|
|
136
|
+
}
|
|
137
|
+
await loadIssueDescriptions();
|
|
138
|
+
return { server };
|
|
139
|
+
}
|
|
140
|
+
export const logDisclaimers = (args) => {
|
|
141
|
+
console.error(`chrome-devtools-mcp exposes content of the browser instance to the MCP clients allowing them to inspect,
|
|
142
|
+
debug, and modify any data in the browser or DevTools.
|
|
143
|
+
Avoid sharing sensitive or personal information that you do not want to share with MCP clients.`);
|
|
144
|
+
if (!args.slim && args.performanceCrux) {
|
|
145
|
+
console.error(`Performance tools may send trace URLs to the Google CrUX API to fetch real-user experience data. To disable, run with --no-performance-crux.`);
|
|
146
|
+
}
|
|
147
|
+
if (!args.slim && args.usageStatistics) {
|
|
148
|
+
console.error(`
|
|
149
|
+
Google collects usage statistics to improve Chrome DevTools MCP. To opt-out, run with --no-usage-statistics.
|
|
150
|
+
For more details, visit: https://github.com/ChromeDevTools/chrome-devtools-mcp#usage-statistics`);
|
|
151
|
+
}
|
|
152
|
+
};
|
|
153
|
+
//# sourceMappingURL=index.js.map
|
|
@@ -0,0 +1,40 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* @license
|
|
3
|
+
* Copyright 2025 Google LLC
|
|
4
|
+
* SPDX-License-Identifier: Apache-2.0
|
|
5
|
+
*/
|
|
6
|
+
import * as fs from 'node:fs';
|
|
7
|
+
import * as path from 'node:path';
|
|
8
|
+
const DESCRIPTIONS_PATH = path.join(import.meta.dirname, 'third_party/issue-descriptions');
|
|
9
|
+
let issueDescriptions = {};
|
|
10
|
+
/**
|
|
11
|
+
* Reads all issue descriptions from the filesystem into memory.
|
|
12
|
+
*/
|
|
13
|
+
export async function loadIssueDescriptions() {
|
|
14
|
+
if (Object.keys(issueDescriptions).length > 0) {
|
|
15
|
+
return;
|
|
16
|
+
}
|
|
17
|
+
const files = await fs.promises.readdir(DESCRIPTIONS_PATH);
|
|
18
|
+
const descriptions = {};
|
|
19
|
+
for (const file of files) {
|
|
20
|
+
if (!file.endsWith('.md')) {
|
|
21
|
+
continue;
|
|
22
|
+
}
|
|
23
|
+
const content = await fs.promises.readFile(path.join(DESCRIPTIONS_PATH, file), 'utf-8');
|
|
24
|
+
descriptions[file] = content;
|
|
25
|
+
}
|
|
26
|
+
issueDescriptions = descriptions;
|
|
27
|
+
}
|
|
28
|
+
/**
|
|
29
|
+
* Gets an issue description from the in-memory cache.
|
|
30
|
+
* @param fileName The file name of the issue description.
|
|
31
|
+
* @returns The description of the issue, or null if it doesn't exist.
|
|
32
|
+
*/
|
|
33
|
+
export function getIssueDescription(fileName) {
|
|
34
|
+
return issueDescriptions[fileName] ?? null;
|
|
35
|
+
}
|
|
36
|
+
export const ISSUE_UTILS = {
|
|
37
|
+
loadIssueDescriptions,
|
|
38
|
+
getIssueDescription,
|
|
39
|
+
};
|
|
40
|
+
//# sourceMappingURL=issue-descriptions.js.map
|
|
@@ -0,0 +1,37 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* @license
|
|
3
|
+
* Copyright 2025 Google LLC
|
|
4
|
+
* SPDX-License-Identifier: Apache-2.0
|
|
5
|
+
*/
|
|
6
|
+
import fs from 'node:fs';
|
|
7
|
+
import { debug } from './third_party/index.js';
|
|
8
|
+
const mcpDebugNamespace = 'mcp:log';
|
|
9
|
+
const namespacesToEnable = [
|
|
10
|
+
mcpDebugNamespace,
|
|
11
|
+
...(process.env['DEBUG'] ? [process.env['DEBUG']] : []),
|
|
12
|
+
];
|
|
13
|
+
export function saveLogsToFile(fileName) {
|
|
14
|
+
// Enable overrides everything so we need to add them
|
|
15
|
+
debug.enable(namespacesToEnable.join(','));
|
|
16
|
+
const logFile = fs.createWriteStream(fileName, { flags: 'a+' });
|
|
17
|
+
debug.log = function (...chunks) {
|
|
18
|
+
logFile.write(`${chunks.join(' ')}\n`);
|
|
19
|
+
};
|
|
20
|
+
logFile.on('error', function (error) {
|
|
21
|
+
console.error(`Error when opening/writing to log file: ${error.message}`);
|
|
22
|
+
logFile.end();
|
|
23
|
+
process.exit(1);
|
|
24
|
+
});
|
|
25
|
+
return logFile;
|
|
26
|
+
}
|
|
27
|
+
export function flushLogs(logFile, timeoutMs = 2000) {
|
|
28
|
+
return new Promise((resolve, reject) => {
|
|
29
|
+
const timeout = setTimeout(reject, timeoutMs);
|
|
30
|
+
logFile.end(() => {
|
|
31
|
+
clearTimeout(timeout);
|
|
32
|
+
resolve();
|
|
33
|
+
});
|
|
34
|
+
});
|
|
35
|
+
}
|
|
36
|
+
export const logger = debug(mcpDebugNamespace);
|
|
37
|
+
//# sourceMappingURL=logger.js.map
|