@checkly/playwright-core 1.41.2-beta.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/NOTICE +5 -0
- package/README.md +3 -0
- package/ThirdPartyNotices.txt +1513 -0
- package/bin/PrintDeps.exe +0 -0
- package/bin/README.md +2 -0
- package/bin/install_media_pack.ps1 +5 -0
- package/bin/reinstall_chrome_beta_linux.sh +40 -0
- package/bin/reinstall_chrome_beta_mac.sh +13 -0
- package/bin/reinstall_chrome_beta_win.ps1 +23 -0
- package/bin/reinstall_chrome_stable_linux.sh +40 -0
- package/bin/reinstall_chrome_stable_mac.sh +12 -0
- package/bin/reinstall_chrome_stable_win.ps1 +23 -0
- package/bin/reinstall_msedge_beta_linux.sh +40 -0
- package/bin/reinstall_msedge_beta_mac.sh +11 -0
- package/bin/reinstall_msedge_beta_win.ps1 +22 -0
- package/bin/reinstall_msedge_dev_linux.sh +40 -0
- package/bin/reinstall_msedge_dev_mac.sh +11 -0
- package/bin/reinstall_msedge_dev_win.ps1 +22 -0
- package/bin/reinstall_msedge_stable_linux.sh +40 -0
- package/bin/reinstall_msedge_stable_mac.sh +11 -0
- package/bin/reinstall_msedge_stable_win.ps1 +23 -0
- package/browsers.json +64 -0
- package/cli.js +17 -0
- package/index.d.ts +17 -0
- package/index.js +33 -0
- package/index.mjs +28 -0
- package/lib/androidServerImpl.js +69 -0
- package/lib/browserServerImpl.js +92 -0
- package/lib/cli/driver.js +97 -0
- package/lib/cli/program.js +582 -0
- package/lib/cli/programWithTestStub.js +67 -0
- package/lib/client/accessibility.js +50 -0
- package/lib/client/android.js +473 -0
- package/lib/client/api.js +272 -0
- package/lib/client/artifact.js +79 -0
- package/lib/client/browser.js +145 -0
- package/lib/client/browserContext.js +509 -0
- package/lib/client/browserType.js +233 -0
- package/lib/client/cdpSession.js +53 -0
- package/lib/client/channelOwner.js +229 -0
- package/lib/client/clientHelper.js +57 -0
- package/lib/client/clientInstrumentation.js +40 -0
- package/lib/client/connection.js +327 -0
- package/lib/client/consoleMessage.js +55 -0
- package/lib/client/coverage.js +41 -0
- package/lib/client/dialog.js +57 -0
- package/lib/client/download.js +62 -0
- package/lib/client/electron.js +130 -0
- package/lib/client/elementHandle.js +291 -0
- package/lib/client/errors.js +77 -0
- package/lib/client/events.js +93 -0
- package/lib/client/fetch.js +343 -0
- package/lib/client/fileChooser.js +45 -0
- package/lib/client/frame.js +506 -0
- package/lib/client/harRouter.js +93 -0
- package/lib/client/input.js +111 -0
- package/lib/client/jsHandle.js +123 -0
- package/lib/client/jsonPipe.js +35 -0
- package/lib/client/localUtils.js +35 -0
- package/lib/client/locator.js +432 -0
- package/lib/client/network.js +601 -0
- package/lib/client/page.js +707 -0
- package/lib/client/playwright.js +74 -0
- package/lib/client/selectors.js +67 -0
- package/lib/client/stream.js +54 -0
- package/lib/client/tracing.js +135 -0
- package/lib/client/types.js +24 -0
- package/lib/client/video.js +51 -0
- package/lib/client/waiter.js +158 -0
- package/lib/client/webError.js +37 -0
- package/lib/client/worker.js +71 -0
- package/lib/client/writableStream.js +54 -0
- package/lib/common/socksProxy.js +569 -0
- package/lib/common/timeoutSettings.js +73 -0
- package/lib/common/types.js +5 -0
- package/lib/generated/consoleApiSource.js +7 -0
- package/lib/generated/injectedScriptSource.js +7 -0
- package/lib/generated/recorderSource.js +7 -0
- package/lib/generated/utilityScriptSource.js +7 -0
- package/lib/image_tools/colorUtils.js +98 -0
- package/lib/image_tools/compare.js +108 -0
- package/lib/image_tools/imageChannel.js +70 -0
- package/lib/image_tools/stats.js +102 -0
- package/lib/inProcessFactory.js +54 -0
- package/lib/inprocess.js +20 -0
- package/lib/outofprocess.js +67 -0
- package/lib/protocol/debug.js +27 -0
- package/lib/protocol/serializers.js +172 -0
- package/lib/protocol/transport.js +82 -0
- package/lib/protocol/validator.js +2599 -0
- package/lib/protocol/validatorPrimitives.js +139 -0
- package/lib/remote/playwrightConnection.js +274 -0
- package/lib/remote/playwrightServer.js +110 -0
- package/lib/server/accessibility.js +62 -0
- package/lib/server/android/android.js +441 -0
- package/lib/server/android/backendAdb.js +172 -0
- package/lib/server/artifact.js +104 -0
- package/lib/server/browser.js +129 -0
- package/lib/server/browserContext.js +609 -0
- package/lib/server/browserType.js +300 -0
- package/lib/server/chromium/appIcon.png +0 -0
- package/lib/server/chromium/chromium.js +346 -0
- package/lib/server/chromium/chromiumSwitches.js +41 -0
- package/lib/server/chromium/crAccessibility.js +237 -0
- package/lib/server/chromium/crBrowser.js +521 -0
- package/lib/server/chromium/crConnection.js +228 -0
- package/lib/server/chromium/crCoverage.js +246 -0
- package/lib/server/chromium/crDevTools.js +104 -0
- package/lib/server/chromium/crDragDrop.js +144 -0
- package/lib/server/chromium/crExecutionContext.js +156 -0
- package/lib/server/chromium/crInput.js +171 -0
- package/lib/server/chromium/crNetworkManager.js +723 -0
- package/lib/server/chromium/crPage.js +1173 -0
- package/lib/server/chromium/crPdf.js +147 -0
- package/lib/server/chromium/crProtocolHelper.js +131 -0
- package/lib/server/chromium/crServiceWorker.js +115 -0
- package/lib/server/chromium/defaultFontFamilies.js +145 -0
- package/lib/server/chromium/videoRecorder.js +155 -0
- package/lib/server/console.js +59 -0
- package/lib/server/cookieStore.js +112 -0
- package/lib/server/debugController.js +236 -0
- package/lib/server/debugger.js +132 -0
- package/lib/server/deviceDescriptors.js +21 -0
- package/lib/server/deviceDescriptorsSource.json +1549 -0
- package/lib/server/dialog.js +70 -0
- package/lib/server/dispatchers/androidDispatcher.js +193 -0
- package/lib/server/dispatchers/artifactDispatcher.js +118 -0
- package/lib/server/dispatchers/browserContextDispatcher.js +306 -0
- package/lib/server/dispatchers/browserDispatcher.js +170 -0
- package/lib/server/dispatchers/browserTypeDispatcher.js +55 -0
- package/lib/server/dispatchers/cdpSessionDispatcher.js +48 -0
- package/lib/server/dispatchers/debugControllerDispatcher.js +103 -0
- package/lib/server/dispatchers/dialogDispatcher.js +44 -0
- package/lib/server/dispatchers/dispatcher.js +400 -0
- package/lib/server/dispatchers/electronDispatcher.js +80 -0
- package/lib/server/dispatchers/elementHandlerDispatcher.js +228 -0
- package/lib/server/dispatchers/frameDispatcher.js +287 -0
- package/lib/server/dispatchers/jsHandleDispatcher.js +102 -0
- package/lib/server/dispatchers/jsonPipeDispatcher.js +61 -0
- package/lib/server/dispatchers/localUtilsDispatcher.js +399 -0
- package/lib/server/dispatchers/networkDispatchers.js +221 -0
- package/lib/server/dispatchers/pageDispatcher.js +363 -0
- package/lib/server/dispatchers/playwrightDispatcher.js +105 -0
- package/lib/server/dispatchers/selectorsDispatcher.js +36 -0
- package/lib/server/dispatchers/streamDispatcher.js +62 -0
- package/lib/server/dispatchers/tracingDispatcher.js +54 -0
- package/lib/server/dispatchers/writableStreamDispatcher.js +55 -0
- package/lib/server/dom.js +808 -0
- package/lib/server/download.js +53 -0
- package/lib/server/electron/electron.js +254 -0
- package/lib/server/electron/loader.js +57 -0
- package/lib/server/errors.js +68 -0
- package/lib/server/fetch.js +611 -0
- package/lib/server/fileChooser.js +42 -0
- package/lib/server/fileUploadUtils.js +71 -0
- package/lib/server/firefox/ffAccessibility.js +215 -0
- package/lib/server/firefox/ffBrowser.js +447 -0
- package/lib/server/firefox/ffConnection.js +168 -0
- package/lib/server/firefox/ffExecutionContext.js +138 -0
- package/lib/server/firefox/ffInput.js +150 -0
- package/lib/server/firefox/ffNetworkManager.js +231 -0
- package/lib/server/firefox/ffPage.js +558 -0
- package/lib/server/firefox/firefox.js +91 -0
- package/lib/server/formData.js +75 -0
- package/lib/server/frameSelectors.js +171 -0
- package/lib/server/frames.js +1597 -0
- package/lib/server/har/harRecorder.js +139 -0
- package/lib/server/har/harTracer.js +539 -0
- package/lib/server/helper.js +103 -0
- package/lib/server/index.js +96 -0
- package/lib/server/input.js +301 -0
- package/lib/server/instrumentation.js +74 -0
- package/lib/server/isomorphic/utilityScriptSerializers.js +212 -0
- package/lib/server/javascript.js +305 -0
- package/lib/server/launchApp.js +90 -0
- package/lib/server/macEditingCommands.js +139 -0
- package/lib/server/network.js +607 -0
- package/lib/server/page.js +793 -0
- package/lib/server/pipeTransport.js +85 -0
- package/lib/server/playwright.js +82 -0
- package/lib/server/progress.js +102 -0
- package/lib/server/protocolError.js +49 -0
- package/lib/server/recorder/codeGenerator.js +153 -0
- package/lib/server/recorder/csharp.js +310 -0
- package/lib/server/recorder/java.js +216 -0
- package/lib/server/recorder/javascript.js +229 -0
- package/lib/server/recorder/jsonl.js +47 -0
- package/lib/server/recorder/language.js +44 -0
- package/lib/server/recorder/python.js +275 -0
- package/lib/server/recorder/recorderActions.js +5 -0
- package/lib/server/recorder/recorderApp.js +181 -0
- package/lib/server/recorder/recorderUtils.js +48 -0
- package/lib/server/recorder/utils.js +45 -0
- package/lib/server/recorder.js +700 -0
- package/lib/server/registry/browserFetcher.js +168 -0
- package/lib/server/registry/dependencies.js +322 -0
- package/lib/server/registry/index.js +925 -0
- package/lib/server/registry/nativeDeps.js +383 -0
- package/lib/server/registry/oopDownloadBrowserMain.js +138 -0
- package/lib/server/screenshotter.js +354 -0
- package/lib/server/selectors.js +73 -0
- package/lib/server/socksInterceptor.js +100 -0
- package/lib/server/trace/recorder/snapshotter.js +168 -0
- package/lib/server/trace/recorder/snapshotterInjected.js +493 -0
- package/lib/server/trace/recorder/tracing.js +552 -0
- package/lib/server/trace/test/inMemorySnapshotter.js +93 -0
- package/lib/server/trace/viewer/traceViewer.js +229 -0
- package/lib/server/transport.js +191 -0
- package/lib/server/types.js +24 -0
- package/lib/server/usKeyboardLayout.js +555 -0
- package/lib/server/webkit/webkit.js +87 -0
- package/lib/server/webkit/wkAccessibility.js +194 -0
- package/lib/server/webkit/wkBrowser.js +328 -0
- package/lib/server/webkit/wkConnection.js +173 -0
- package/lib/server/webkit/wkExecutionContext.js +146 -0
- package/lib/server/webkit/wkInput.js +169 -0
- package/lib/server/webkit/wkInterceptableRequest.js +158 -0
- package/lib/server/webkit/wkPage.js +1198 -0
- package/lib/server/webkit/wkProvisionalPage.js +59 -0
- package/lib/server/webkit/wkWorkers.js +104 -0
- package/lib/third_party/diff_match_patch.js +2222 -0
- package/lib/third_party/pixelmatch.js +255 -0
- package/lib/utils/ascii.js +31 -0
- package/lib/utils/comparators.js +171 -0
- package/lib/utils/crypto.js +33 -0
- package/lib/utils/debug.js +46 -0
- package/lib/utils/debugLogger.js +89 -0
- package/lib/utils/env.js +47 -0
- package/lib/utils/eventsHelper.js +38 -0
- package/lib/utils/fileUtils.js +66 -0
- package/lib/utils/glob.js +83 -0
- package/lib/utils/happy-eyeballs.js +154 -0
- package/lib/utils/headers.js +52 -0
- package/lib/utils/hostPlatform.js +124 -0
- package/lib/utils/httpServer.js +195 -0
- package/lib/utils/index.js +324 -0
- package/lib/utils/isomorphic/cssParser.js +250 -0
- package/lib/utils/isomorphic/cssTokenizer.js +979 -0
- package/lib/utils/isomorphic/locatorGenerators.js +651 -0
- package/lib/utils/isomorphic/locatorParser.js +179 -0
- package/lib/utils/isomorphic/locatorUtils.js +62 -0
- package/lib/utils/isomorphic/selectorParser.js +397 -0
- package/lib/utils/isomorphic/stringUtils.js +107 -0
- package/lib/utils/isomorphic/traceUtils.js +39 -0
- package/lib/utils/linuxUtils.js +78 -0
- package/lib/utils/manualPromise.js +109 -0
- package/lib/utils/mimeType.js +29 -0
- package/lib/utils/multimap.js +75 -0
- package/lib/utils/network.js +189 -0
- package/lib/utils/processLauncher.js +248 -0
- package/lib/utils/profiler.js +53 -0
- package/lib/utils/rtti.js +41 -0
- package/lib/utils/semaphore.js +51 -0
- package/lib/utils/spawnAsync.js +45 -0
- package/lib/utils/stackTrace.js +123 -0
- package/lib/utils/task.js +58 -0
- package/lib/utils/time.js +37 -0
- package/lib/utils/timeoutRunner.js +131 -0
- package/lib/utils/traceUtils.js +44 -0
- package/lib/utils/userAgent.js +105 -0
- package/lib/utils/wsServer.js +125 -0
- package/lib/utils/zipFile.js +75 -0
- package/lib/utils/zones.js +99 -0
- package/lib/utilsBundle.js +81 -0
- package/lib/utilsBundleImpl/index.js +51 -0
- package/lib/utilsBundleImpl/xdg-open +1066 -0
- package/lib/vite/htmlReport/index.html +66 -0
- package/lib/vite/recorder/assets/codeMirrorModule-Hs9-1ZG4.css +1 -0
- package/lib/vite/recorder/assets/codeMirrorModule-I9ks4y7D.js +24 -0
- package/lib/vite/recorder/assets/codicon-zGuYmc9o.ttf +0 -0
- package/lib/vite/recorder/assets/index-ljsTwXtJ.css +1 -0
- package/lib/vite/recorder/assets/index-yg8ypzl6.js +47 -0
- package/lib/vite/recorder/index.html +29 -0
- package/lib/vite/recorder/playwright-logo.svg +9 -0
- package/lib/vite/traceViewer/assets/codeMirrorModule-GluP1cQ1.js +24 -0
- package/lib/vite/traceViewer/assets/codeMirrorModule-fqJB1XDu.js +24 -0
- package/lib/vite/traceViewer/assets/codeMirrorModule-y3M3aAqy.js +24 -0
- package/lib/vite/traceViewer/assets/wsPort-Rvwd4WC-.js +69 -0
- package/lib/vite/traceViewer/assets/wsPort-dlD7vDkY.js +69 -0
- package/lib/vite/traceViewer/assets/wsPort-qOE2NWrO.js +69 -0
- package/lib/vite/traceViewer/assets/xtermModule-Yt6xwiJ_.js +9 -0
- package/lib/vite/traceViewer/codeMirrorModule.Hs9-1ZG4.css +1 -0
- package/lib/vite/traceViewer/codicon.zGuYmc9o.ttf +0 -0
- package/lib/vite/traceViewer/index.-g_5lMbJ.css +1 -0
- package/lib/vite/traceViewer/index.4X7zDysg.js +2 -0
- package/lib/vite/traceViewer/index.HkJgzlGy.js +2 -0
- package/lib/vite/traceViewer/index.html +26 -0
- package/lib/vite/traceViewer/index.kRjx5sAJ.js +2 -0
- package/lib/vite/traceViewer/playwright-logo.svg +9 -0
- package/lib/vite/traceViewer/snapshot.html +21 -0
- package/lib/vite/traceViewer/sw.bundle.js +4 -0
- package/lib/vite/traceViewer/uiMode.1Wcp_Kto.js +10 -0
- package/lib/vite/traceViewer/uiMode.GTNzARcV.js +10 -0
- package/lib/vite/traceViewer/uiMode.html +17 -0
- package/lib/vite/traceViewer/uiMode.pWy0Re7G.css +1 -0
- package/lib/vite/traceViewer/uiMode.zV-7Lf9v.js +10 -0
- package/lib/vite/traceViewer/wsPort.kSgQKQ0y.css +1 -0
- package/lib/vite/traceViewer/xtermModule.0lwXJFHT.css +32 -0
- package/lib/zipBundle.js +25 -0
- package/lib/zipBundleImpl.js +5 -0
- package/package.json +43 -0
- package/types/protocol.d.ts +20304 -0
- package/types/structs.d.ts +45 -0
- package/types/types.d.ts +20626 -0
|
@@ -0,0 +1,723 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
|
|
3
|
+
Object.defineProperty(exports, "__esModule", {
|
|
4
|
+
value: true
|
|
5
|
+
});
|
|
6
|
+
exports.CRNetworkManager = void 0;
|
|
7
|
+
var _helper = require("../helper");
|
|
8
|
+
var _eventsHelper = require("../../utils/eventsHelper");
|
|
9
|
+
var network = _interopRequireWildcard(require("../network"));
|
|
10
|
+
var _utils = require("../../utils");
|
|
11
|
+
var _protocolError = require("../protocolError");
|
|
12
|
+
function _getRequireWildcardCache(e) { if ("function" != typeof WeakMap) return null; var r = new WeakMap(), t = new WeakMap(); return (_getRequireWildcardCache = function (e) { return e ? t : r; })(e); }
|
|
13
|
+
function _interopRequireWildcard(e, r) { if (!r && e && e.__esModule) return e; if (null === e || "object" != typeof e && "function" != typeof e) return { default: e }; var t = _getRequireWildcardCache(r); if (t && t.has(e)) return t.get(e); var n = { __proto__: null }, a = Object.defineProperty && Object.getOwnPropertyDescriptor; for (var u in e) if ("default" !== u && Object.prototype.hasOwnProperty.call(e, u)) { var i = a ? Object.getOwnPropertyDescriptor(e, u) : null; i && (i.get || i.set) ? Object.defineProperty(n, u, i) : n[u] = e[u]; } return n.default = e, t && t.set(e, n), n; }
|
|
14
|
+
/**
|
|
15
|
+
* Copyright 2017 Google Inc. All rights reserved.
|
|
16
|
+
* Modifications copyright (c) Microsoft Corporation.
|
|
17
|
+
*
|
|
18
|
+
* Licensed under the Apache License, Version 2.0 (the "License");
|
|
19
|
+
* you may not use this file except in compliance with the License.
|
|
20
|
+
* You may obtain a copy of the License at
|
|
21
|
+
*
|
|
22
|
+
* http://www.apache.org/licenses/LICENSE-2.0
|
|
23
|
+
*
|
|
24
|
+
* Unless required by applicable law or agreed to in writing, software
|
|
25
|
+
* distributed under the License is distributed on an "AS IS" BASIS,
|
|
26
|
+
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
27
|
+
* See the License for the specific language governing permissions and
|
|
28
|
+
* limitations under the License.
|
|
29
|
+
*/
|
|
30
|
+
|
|
31
|
+
class CRNetworkManager {
|
|
32
|
+
constructor(session, page, serviceWorker, parentManager) {
|
|
33
|
+
this._session = void 0;
|
|
34
|
+
this._page = void 0;
|
|
35
|
+
this._serviceWorker = void 0;
|
|
36
|
+
this._parentManager = void 0;
|
|
37
|
+
this._requestIdToRequest = new Map();
|
|
38
|
+
this._requestIdToRequestWillBeSentEvent = new Map();
|
|
39
|
+
this._credentials = null;
|
|
40
|
+
this._attemptedAuthentications = new Set();
|
|
41
|
+
this._userRequestInterceptionEnabled = false;
|
|
42
|
+
this._protocolRequestInterceptionEnabled = false;
|
|
43
|
+
this._requestIdToRequestPausedEvent = new Map();
|
|
44
|
+
this._eventListeners = void 0;
|
|
45
|
+
this._responseExtraInfoTracker = new ResponseExtraInfoTracker();
|
|
46
|
+
this._session = session;
|
|
47
|
+
this._page = page;
|
|
48
|
+
this._serviceWorker = serviceWorker;
|
|
49
|
+
this._parentManager = parentManager;
|
|
50
|
+
this._eventListeners = this.instrumentNetworkEvents({
|
|
51
|
+
session
|
|
52
|
+
});
|
|
53
|
+
}
|
|
54
|
+
instrumentNetworkEvents(sessionInfo) {
|
|
55
|
+
const listeners = [_eventsHelper.eventsHelper.addEventListener(sessionInfo.session, 'Fetch.requestPaused', this._onRequestPaused.bind(this, sessionInfo)), _eventsHelper.eventsHelper.addEventListener(sessionInfo.session, 'Fetch.authRequired', this._onAuthRequired.bind(this)), _eventsHelper.eventsHelper.addEventListener(sessionInfo.session, 'Network.requestWillBeSent', this._onRequestWillBeSent.bind(this, sessionInfo)), _eventsHelper.eventsHelper.addEventListener(sessionInfo.session, 'Network.requestWillBeSentExtraInfo', this._onRequestWillBeSentExtraInfo.bind(this)), _eventsHelper.eventsHelper.addEventListener(sessionInfo.session, 'Network.requestServedFromCache', this._onRequestServedFromCache.bind(this)), _eventsHelper.eventsHelper.addEventListener(sessionInfo.session, 'Network.responseReceived', this._onResponseReceived.bind(this, sessionInfo)), _eventsHelper.eventsHelper.addEventListener(sessionInfo.session, 'Network.responseReceivedExtraInfo', this._onResponseReceivedExtraInfo.bind(this)), _eventsHelper.eventsHelper.addEventListener(sessionInfo.session, 'Network.loadingFinished', this._onLoadingFinished.bind(this)), _eventsHelper.eventsHelper.addEventListener(sessionInfo.session, 'Network.loadingFailed', this._onLoadingFailed.bind(this, sessionInfo))];
|
|
56
|
+
if (this._page) {
|
|
57
|
+
listeners.push(...[_eventsHelper.eventsHelper.addEventListener(sessionInfo.session, 'Network.webSocketCreated', e => this._page._frameManager.onWebSocketCreated(e.requestId, e.url)), _eventsHelper.eventsHelper.addEventListener(sessionInfo.session, 'Network.webSocketWillSendHandshakeRequest', e => this._page._frameManager.onWebSocketRequest(e.requestId)), _eventsHelper.eventsHelper.addEventListener(sessionInfo.session, 'Network.webSocketHandshakeResponseReceived', e => this._page._frameManager.onWebSocketResponse(e.requestId, e.response.status, e.response.statusText)), _eventsHelper.eventsHelper.addEventListener(sessionInfo.session, 'Network.webSocketFrameSent', e => e.response.payloadData && this._page._frameManager.onWebSocketFrameSent(e.requestId, e.response.opcode, e.response.payloadData)), _eventsHelper.eventsHelper.addEventListener(sessionInfo.session, 'Network.webSocketFrameReceived', e => e.response.payloadData && this._page._frameManager.webSocketFrameReceived(e.requestId, e.response.opcode, e.response.payloadData)), _eventsHelper.eventsHelper.addEventListener(sessionInfo.session, 'Network.webSocketClosed', e => this._page._frameManager.webSocketClosed(e.requestId)), _eventsHelper.eventsHelper.addEventListener(sessionInfo.session, 'Network.webSocketFrameError', e => this._page._frameManager.webSocketError(e.requestId, e.errorMessage))]);
|
|
58
|
+
}
|
|
59
|
+
return listeners;
|
|
60
|
+
}
|
|
61
|
+
async initialize() {
|
|
62
|
+
await this._session.send('Network.enable');
|
|
63
|
+
}
|
|
64
|
+
dispose() {
|
|
65
|
+
_eventsHelper.eventsHelper.removeEventListeners(this._eventListeners);
|
|
66
|
+
}
|
|
67
|
+
async authenticate(credentials) {
|
|
68
|
+
this._credentials = credentials;
|
|
69
|
+
await this._updateProtocolRequestInterception();
|
|
70
|
+
}
|
|
71
|
+
async setOffline(offline) {
|
|
72
|
+
await this._session.send('Network.emulateNetworkConditions', {
|
|
73
|
+
offline,
|
|
74
|
+
// values of 0 remove any active throttling. crbug.com/456324#c9
|
|
75
|
+
latency: 0,
|
|
76
|
+
downloadThroughput: -1,
|
|
77
|
+
uploadThroughput: -1
|
|
78
|
+
});
|
|
79
|
+
}
|
|
80
|
+
async setRequestInterception(value) {
|
|
81
|
+
this._userRequestInterceptionEnabled = value;
|
|
82
|
+
await this._updateProtocolRequestInterception();
|
|
83
|
+
}
|
|
84
|
+
async _updateProtocolRequestInterception() {
|
|
85
|
+
const enabled = this._userRequestInterceptionEnabled || !!this._credentials;
|
|
86
|
+
if (enabled === this._protocolRequestInterceptionEnabled) return;
|
|
87
|
+
this._protocolRequestInterceptionEnabled = enabled;
|
|
88
|
+
if (enabled) {
|
|
89
|
+
await Promise.all([this._session.send('Network.setCacheDisabled', {
|
|
90
|
+
cacheDisabled: true
|
|
91
|
+
}), this._session.send('Fetch.enable', {
|
|
92
|
+
handleAuthRequests: true,
|
|
93
|
+
patterns: [{
|
|
94
|
+
urlPattern: '*',
|
|
95
|
+
requestStage: 'Request'
|
|
96
|
+
}]
|
|
97
|
+
})]);
|
|
98
|
+
} else {
|
|
99
|
+
await Promise.all([this._session.send('Network.setCacheDisabled', {
|
|
100
|
+
cacheDisabled: false
|
|
101
|
+
}), this._session.send('Fetch.disable')]);
|
|
102
|
+
}
|
|
103
|
+
}
|
|
104
|
+
async clearCache() {
|
|
105
|
+
// Sending 'Network.setCacheDisabled' with 'cacheDisabled = true' will clear the MemoryCache.
|
|
106
|
+
await this._session.send('Network.setCacheDisabled', {
|
|
107
|
+
cacheDisabled: true
|
|
108
|
+
});
|
|
109
|
+
if (!this._protocolRequestInterceptionEnabled) await this._session.send('Network.setCacheDisabled', {
|
|
110
|
+
cacheDisabled: false
|
|
111
|
+
});
|
|
112
|
+
await this._session.send('Network.clearBrowserCache');
|
|
113
|
+
}
|
|
114
|
+
_onRequestWillBeSent(sessionInfo, event) {
|
|
115
|
+
// Request interception doesn't happen for data URLs with Network Service.
|
|
116
|
+
if (this._protocolRequestInterceptionEnabled && !event.request.url.startsWith('data:')) {
|
|
117
|
+
const requestId = event.requestId;
|
|
118
|
+
const requestPausedEvent = this._requestIdToRequestPausedEvent.get(requestId);
|
|
119
|
+
if (requestPausedEvent) {
|
|
120
|
+
this._onRequest(sessionInfo, event, requestPausedEvent);
|
|
121
|
+
this._requestIdToRequestPausedEvent.delete(requestId);
|
|
122
|
+
} else {
|
|
123
|
+
this._requestIdToRequestWillBeSentEvent.set(event.requestId, event);
|
|
124
|
+
}
|
|
125
|
+
} else {
|
|
126
|
+
this._onRequest(sessionInfo, event, null);
|
|
127
|
+
}
|
|
128
|
+
}
|
|
129
|
+
_onRequestServedFromCache(event) {
|
|
130
|
+
this._responseExtraInfoTracker.requestServedFromCache(event);
|
|
131
|
+
}
|
|
132
|
+
_onRequestWillBeSentExtraInfo(event) {
|
|
133
|
+
this._responseExtraInfoTracker.requestWillBeSentExtraInfo(event);
|
|
134
|
+
}
|
|
135
|
+
_onAuthRequired(event) {
|
|
136
|
+
let response = 'Default';
|
|
137
|
+
const shouldProvideCredentials = this._shouldProvideCredentials(event.request.url);
|
|
138
|
+
if (this._attemptedAuthentications.has(event.requestId)) {
|
|
139
|
+
response = 'CancelAuth';
|
|
140
|
+
} else if (shouldProvideCredentials) {
|
|
141
|
+
response = 'ProvideCredentials';
|
|
142
|
+
this._attemptedAuthentications.add(event.requestId);
|
|
143
|
+
}
|
|
144
|
+
const {
|
|
145
|
+
username,
|
|
146
|
+
password
|
|
147
|
+
} = shouldProvideCredentials && this._credentials ? this._credentials : {
|
|
148
|
+
username: undefined,
|
|
149
|
+
password: undefined
|
|
150
|
+
};
|
|
151
|
+
this._session._sendMayFail('Fetch.continueWithAuth', {
|
|
152
|
+
requestId: event.requestId,
|
|
153
|
+
authChallengeResponse: {
|
|
154
|
+
response,
|
|
155
|
+
username,
|
|
156
|
+
password
|
|
157
|
+
}
|
|
158
|
+
});
|
|
159
|
+
}
|
|
160
|
+
_shouldProvideCredentials(url) {
|
|
161
|
+
if (!this._credentials) return false;
|
|
162
|
+
return !this._credentials.origin || new URL(url).origin.toLowerCase() === this._credentials.origin.toLowerCase();
|
|
163
|
+
}
|
|
164
|
+
_onRequestPaused(sessionInfo, event) {
|
|
165
|
+
if (!event.networkId) {
|
|
166
|
+
// Fetch without networkId means that request was not recognized by inspector, and
|
|
167
|
+
// it will never receive Network.requestWillBeSent. Continue the request to not affect it.
|
|
168
|
+
this._session._sendMayFail('Fetch.continueRequest', {
|
|
169
|
+
requestId: event.requestId
|
|
170
|
+
});
|
|
171
|
+
return;
|
|
172
|
+
}
|
|
173
|
+
if (event.request.url.startsWith('data:')) return;
|
|
174
|
+
const requestId = event.networkId;
|
|
175
|
+
const requestWillBeSentEvent = this._requestIdToRequestWillBeSentEvent.get(requestId);
|
|
176
|
+
if (requestWillBeSentEvent) {
|
|
177
|
+
this._onRequest(sessionInfo, requestWillBeSentEvent, event);
|
|
178
|
+
this._requestIdToRequestWillBeSentEvent.delete(requestId);
|
|
179
|
+
} else {
|
|
180
|
+
var _existingRequest$_rou;
|
|
181
|
+
const existingRequest = this._requestIdToRequest.get(requestId);
|
|
182
|
+
const alreadyContinuedParams = existingRequest === null || existingRequest === void 0 || (_existingRequest$_rou = existingRequest._route) === null || _existingRequest$_rou === void 0 ? void 0 : _existingRequest$_rou._alreadyContinuedParams;
|
|
183
|
+
if (alreadyContinuedParams && !event.redirectedRequestId) {
|
|
184
|
+
// Sometimes Chromium network stack restarts the request internally.
|
|
185
|
+
// For example, when no-cors request hits a "less public address space", it should be resent with cors.
|
|
186
|
+
// There are some more examples here: https://source.chromium.org/chromium/chromium/src/+/main:services/network/url_loader.cc;l=1205-1234;drc=d5dd931e0ad3d9ffe74888ec62a3cc106efd7ea6
|
|
187
|
+
// There are probably even more cases deep inside the network stack.
|
|
188
|
+
//
|
|
189
|
+
// Anyway, in this case, continue the request in the same way as before, and it should go through.
|
|
190
|
+
//
|
|
191
|
+
// Note: make sure not to prematurely continue the redirect, which shares the
|
|
192
|
+
// `networkId` between the original request and the redirect.
|
|
193
|
+
this._session._sendMayFail('Fetch.continueRequest', {
|
|
194
|
+
...alreadyContinuedParams,
|
|
195
|
+
requestId: event.requestId
|
|
196
|
+
});
|
|
197
|
+
return;
|
|
198
|
+
}
|
|
199
|
+
this._requestIdToRequestPausedEvent.set(requestId, event);
|
|
200
|
+
}
|
|
201
|
+
}
|
|
202
|
+
_onRequest(sessionInfo, requestWillBeSentEvent, requestPausedEvent) {
|
|
203
|
+
var _this$_page, _this$_page2, _this$_page3;
|
|
204
|
+
if (requestWillBeSentEvent.request.url.startsWith('data:')) return;
|
|
205
|
+
let redirectedFrom = null;
|
|
206
|
+
if (requestWillBeSentEvent.redirectResponse) {
|
|
207
|
+
const request = this._requestIdToRequest.get(requestWillBeSentEvent.requestId);
|
|
208
|
+
// If we connect late to the target, we could have missed the requestWillBeSent event.
|
|
209
|
+
if (request) {
|
|
210
|
+
this._handleRequestRedirect(request, requestWillBeSentEvent.redirectResponse, requestWillBeSentEvent.timestamp, requestWillBeSentEvent.redirectHasExtraInfo);
|
|
211
|
+
redirectedFrom = request;
|
|
212
|
+
}
|
|
213
|
+
}
|
|
214
|
+
let frame = requestWillBeSentEvent.frameId ? (_this$_page = this._page) === null || _this$_page === void 0 ? void 0 : _this$_page._frameManager.frame(requestWillBeSentEvent.frameId) : sessionInfo.workerFrame;
|
|
215
|
+
// Requests from workers lack frameId, because we receive Network.requestWillBeSent
|
|
216
|
+
// on the worker target. However, we receive Fetch.requestPaused on the page target,
|
|
217
|
+
// and lack workerFrame there. Luckily, Fetch.requestPaused provides a frameId.
|
|
218
|
+
if (!frame && this._page && requestPausedEvent && requestPausedEvent.frameId) frame = this._page._frameManager.frame(requestPausedEvent.frameId);
|
|
219
|
+
|
|
220
|
+
// Check if it's main resource request interception (targetId === main frame id).
|
|
221
|
+
if (!frame && this._page && requestWillBeSentEvent.frameId === ((_this$_page2 = this._page) === null || _this$_page2 === void 0 ? void 0 : _this$_page2._delegate)._targetId) {
|
|
222
|
+
// Main resource request for the page is being intercepted so the Frame is not created
|
|
223
|
+
// yet. Precreate it here for the purposes of request interception. It will be updated
|
|
224
|
+
// later as soon as the request continues and we receive frame tree from the page.
|
|
225
|
+
frame = this._page._frameManager.frameAttached(requestWillBeSentEvent.frameId, null);
|
|
226
|
+
}
|
|
227
|
+
|
|
228
|
+
// CORS options preflight request is generated by the network stack. If interception is enabled,
|
|
229
|
+
// we accept all CORS options, assuming that this was intended when setting route.
|
|
230
|
+
//
|
|
231
|
+
// Note: it would be better to match the URL against interception patterns.
|
|
232
|
+
const isInterceptedOptionsPreflight = !!requestPausedEvent && requestPausedEvent.request.method === 'OPTIONS' && requestWillBeSentEvent.initiator.type === 'preflight';
|
|
233
|
+
if (isInterceptedOptionsPreflight && (this._page || this._serviceWorker).needsRequestInterception()) {
|
|
234
|
+
const requestHeaders = requestPausedEvent.request.headers;
|
|
235
|
+
const responseHeaders = [{
|
|
236
|
+
name: 'Access-Control-Allow-Origin',
|
|
237
|
+
value: requestHeaders['Origin'] || '*'
|
|
238
|
+
}, {
|
|
239
|
+
name: 'Access-Control-Allow-Methods',
|
|
240
|
+
value: requestHeaders['Access-Control-Request-Method'] || 'GET, POST, OPTIONS, DELETE'
|
|
241
|
+
}, {
|
|
242
|
+
name: 'Access-Control-Allow-Credentials',
|
|
243
|
+
value: 'true'
|
|
244
|
+
}];
|
|
245
|
+
if (requestHeaders['Access-Control-Request-Headers']) responseHeaders.push({
|
|
246
|
+
name: 'Access-Control-Allow-Headers',
|
|
247
|
+
value: requestHeaders['Access-Control-Request-Headers']
|
|
248
|
+
});
|
|
249
|
+
this._session._sendMayFail('Fetch.fulfillRequest', {
|
|
250
|
+
requestId: requestPausedEvent.requestId,
|
|
251
|
+
responseCode: 204,
|
|
252
|
+
responsePhrase: network.STATUS_TEXTS['204'],
|
|
253
|
+
responseHeaders,
|
|
254
|
+
body: ''
|
|
255
|
+
});
|
|
256
|
+
return;
|
|
257
|
+
}
|
|
258
|
+
|
|
259
|
+
// Non-service-worker requests MUST have a frame—if they don't, we pretend there was no request
|
|
260
|
+
if (!frame && !this._serviceWorker) {
|
|
261
|
+
if (requestPausedEvent) this._session._sendMayFail('Fetch.continueRequest', {
|
|
262
|
+
requestId: requestPausedEvent.requestId
|
|
263
|
+
});
|
|
264
|
+
return;
|
|
265
|
+
}
|
|
266
|
+
let route = null;
|
|
267
|
+
if (requestPausedEvent) {
|
|
268
|
+
// We do not support intercepting redirects.
|
|
269
|
+
if (redirectedFrom || !this._userRequestInterceptionEnabled && this._protocolRequestInterceptionEnabled) {
|
|
270
|
+
var _redirectedFrom;
|
|
271
|
+
let headers = undefined;
|
|
272
|
+
const previousHeaderOverrides = (_redirectedFrom = redirectedFrom) === null || _redirectedFrom === void 0 || (_redirectedFrom = _redirectedFrom._originalRequestRoute) === null || _redirectedFrom === void 0 || (_redirectedFrom = _redirectedFrom._alreadyContinuedParams) === null || _redirectedFrom === void 0 ? void 0 : _redirectedFrom.headers;
|
|
273
|
+
// Chromium does not preserve header overrides between redirects, so we have to do it ourselves.
|
|
274
|
+
if (previousHeaderOverrides) headers = network.mergeHeaders([(0, _utils.headersObjectToArray)(requestPausedEvent.request.headers, '\n'), previousHeaderOverrides]);
|
|
275
|
+
this._session._sendMayFail('Fetch.continueRequest', {
|
|
276
|
+
requestId: requestPausedEvent.requestId,
|
|
277
|
+
headers
|
|
278
|
+
});
|
|
279
|
+
} else {
|
|
280
|
+
route = new RouteImpl(this._session, requestPausedEvent.requestId);
|
|
281
|
+
}
|
|
282
|
+
}
|
|
283
|
+
const isNavigationRequest = requestWillBeSentEvent.requestId === requestWillBeSentEvent.loaderId && requestWillBeSentEvent.type === 'Document';
|
|
284
|
+
const documentId = isNavigationRequest ? requestWillBeSentEvent.loaderId : undefined;
|
|
285
|
+
const request = new InterceptableRequest({
|
|
286
|
+
session: sessionInfo.session,
|
|
287
|
+
context: (this._page || this._serviceWorker)._browserContext,
|
|
288
|
+
frame: frame || null,
|
|
289
|
+
serviceWorker: this._serviceWorker || null,
|
|
290
|
+
documentId,
|
|
291
|
+
route,
|
|
292
|
+
requestWillBeSentEvent,
|
|
293
|
+
requestPausedEvent,
|
|
294
|
+
redirectedFrom
|
|
295
|
+
});
|
|
296
|
+
this._requestIdToRequest.set(requestWillBeSentEvent.requestId, request);
|
|
297
|
+
if (requestPausedEvent) {
|
|
298
|
+
// We will not receive extra info when intercepting the request.
|
|
299
|
+
// Use the headers from the Fetch.requestPausedPayload and release the allHeaders()
|
|
300
|
+
// right away, so that client can call it from the route handler.
|
|
301
|
+
request.request.setRawRequestHeaders((0, _utils.headersObjectToArray)(requestPausedEvent.request.headers, '\n'));
|
|
302
|
+
}
|
|
303
|
+
(((_this$_page3 = this._page) === null || _this$_page3 === void 0 ? void 0 : _this$_page3._frameManager) || this._serviceWorker).requestStarted(request.request, route || undefined);
|
|
304
|
+
}
|
|
305
|
+
_createResponse(request, responsePayload, hasExtraInfo) {
|
|
306
|
+
var _responsePayload$secu, _responsePayload$secu2, _responsePayload$secu3, _responsePayload$secu4, _responsePayload$secu5;
|
|
307
|
+
const getResponseBody = async () => {
|
|
308
|
+
const contentLengthHeader = Object.entries(responsePayload.headers).find(header => header[0].toLowerCase() === 'content-length');
|
|
309
|
+
const expectedLength = contentLengthHeader ? +contentLengthHeader[1] : undefined;
|
|
310
|
+
const session = request.session;
|
|
311
|
+
const response = await session.send('Network.getResponseBody', {
|
|
312
|
+
requestId: request._requestId
|
|
313
|
+
});
|
|
314
|
+
if (response.body || !expectedLength) return Buffer.from(response.body, response.base64Encoded ? 'base64' : 'utf8');
|
|
315
|
+
|
|
316
|
+
// For <link prefetch we are going to receive empty body with non-empty content-length expectation. Reach out for the actual content.
|
|
317
|
+
const resource = await session.send('Network.loadNetworkResource', {
|
|
318
|
+
url: request.request.url(),
|
|
319
|
+
frameId: this._serviceWorker ? undefined : request.request.frame()._id,
|
|
320
|
+
options: {
|
|
321
|
+
disableCache: false,
|
|
322
|
+
includeCredentials: true
|
|
323
|
+
}
|
|
324
|
+
});
|
|
325
|
+
const chunks = [];
|
|
326
|
+
while (resource.resource.stream) {
|
|
327
|
+
const chunk = await session.send('IO.read', {
|
|
328
|
+
handle: resource.resource.stream
|
|
329
|
+
});
|
|
330
|
+
chunks.push(Buffer.from(chunk.data, chunk.base64Encoded ? 'base64' : 'utf-8'));
|
|
331
|
+
if (chunk.eof) {
|
|
332
|
+
await session.send('IO.close', {
|
|
333
|
+
handle: resource.resource.stream
|
|
334
|
+
});
|
|
335
|
+
break;
|
|
336
|
+
}
|
|
337
|
+
}
|
|
338
|
+
return Buffer.concat(chunks);
|
|
339
|
+
};
|
|
340
|
+
const timingPayload = responsePayload.timing;
|
|
341
|
+
let timing;
|
|
342
|
+
if (timingPayload && !this._responseExtraInfoTracker.servedFromCache(request._requestId)) {
|
|
343
|
+
timing = {
|
|
344
|
+
startTime: (timingPayload.requestTime - request._timestamp + request._wallTime) * 1000,
|
|
345
|
+
domainLookupStart: timingPayload.dnsStart,
|
|
346
|
+
domainLookupEnd: timingPayload.dnsEnd,
|
|
347
|
+
connectStart: timingPayload.connectStart,
|
|
348
|
+
secureConnectionStart: timingPayload.sslStart,
|
|
349
|
+
connectEnd: timingPayload.connectEnd,
|
|
350
|
+
requestStart: timingPayload.sendStart,
|
|
351
|
+
responseStart: timingPayload.receiveHeadersEnd
|
|
352
|
+
};
|
|
353
|
+
} else {
|
|
354
|
+
timing = {
|
|
355
|
+
startTime: request._wallTime * 1000,
|
|
356
|
+
domainLookupStart: -1,
|
|
357
|
+
domainLookupEnd: -1,
|
|
358
|
+
connectStart: -1,
|
|
359
|
+
secureConnectionStart: -1,
|
|
360
|
+
connectEnd: -1,
|
|
361
|
+
requestStart: -1,
|
|
362
|
+
responseStart: -1
|
|
363
|
+
};
|
|
364
|
+
}
|
|
365
|
+
const response = new network.Response(request.request, responsePayload.status, responsePayload.statusText, (0, _utils.headersObjectToArray)(responsePayload.headers), timing, getResponseBody, !!responsePayload.fromServiceWorker, responsePayload.protocol);
|
|
366
|
+
if (responsePayload !== null && responsePayload !== void 0 && responsePayload.remoteIPAddress && typeof (responsePayload === null || responsePayload === void 0 ? void 0 : responsePayload.remotePort) === 'number') {
|
|
367
|
+
response._serverAddrFinished({
|
|
368
|
+
ipAddress: responsePayload.remoteIPAddress,
|
|
369
|
+
port: responsePayload.remotePort
|
|
370
|
+
});
|
|
371
|
+
} else {
|
|
372
|
+
response._serverAddrFinished();
|
|
373
|
+
}
|
|
374
|
+
response._securityDetailsFinished({
|
|
375
|
+
protocol: responsePayload === null || responsePayload === void 0 || (_responsePayload$secu = responsePayload.securityDetails) === null || _responsePayload$secu === void 0 ? void 0 : _responsePayload$secu.protocol,
|
|
376
|
+
subjectName: responsePayload === null || responsePayload === void 0 || (_responsePayload$secu2 = responsePayload.securityDetails) === null || _responsePayload$secu2 === void 0 ? void 0 : _responsePayload$secu2.subjectName,
|
|
377
|
+
issuer: responsePayload === null || responsePayload === void 0 || (_responsePayload$secu3 = responsePayload.securityDetails) === null || _responsePayload$secu3 === void 0 ? void 0 : _responsePayload$secu3.issuer,
|
|
378
|
+
validFrom: responsePayload === null || responsePayload === void 0 || (_responsePayload$secu4 = responsePayload.securityDetails) === null || _responsePayload$secu4 === void 0 ? void 0 : _responsePayload$secu4.validFrom,
|
|
379
|
+
validTo: responsePayload === null || responsePayload === void 0 || (_responsePayload$secu5 = responsePayload.securityDetails) === null || _responsePayload$secu5 === void 0 ? void 0 : _responsePayload$secu5.validTo
|
|
380
|
+
});
|
|
381
|
+
this._responseExtraInfoTracker.processResponse(request._requestId, response, hasExtraInfo);
|
|
382
|
+
return response;
|
|
383
|
+
}
|
|
384
|
+
_deleteRequest(request) {
|
|
385
|
+
this._requestIdToRequest.delete(request._requestId);
|
|
386
|
+
if (request._interceptionId) this._attemptedAuthentications.delete(request._interceptionId);
|
|
387
|
+
}
|
|
388
|
+
_handleRequestRedirect(request, responsePayload, timestamp, hasExtraInfo) {
|
|
389
|
+
var _this$_page4, _this$_page5;
|
|
390
|
+
const response = this._createResponse(request, responsePayload, hasExtraInfo);
|
|
391
|
+
response.setTransferSize(null);
|
|
392
|
+
response.setEncodedBodySize(null);
|
|
393
|
+
response._requestFinished((timestamp - request._timestamp) * 1000);
|
|
394
|
+
this._deleteRequest(request);
|
|
395
|
+
(((_this$_page4 = this._page) === null || _this$_page4 === void 0 ? void 0 : _this$_page4._frameManager) || this._serviceWorker).requestReceivedResponse(response);
|
|
396
|
+
(((_this$_page5 = this._page) === null || _this$_page5 === void 0 ? void 0 : _this$_page5._frameManager) || this._serviceWorker).reportRequestFinished(request.request, response);
|
|
397
|
+
}
|
|
398
|
+
_onResponseReceivedExtraInfo(event) {
|
|
399
|
+
this._responseExtraInfoTracker.responseReceivedExtraInfo(event);
|
|
400
|
+
}
|
|
401
|
+
_onResponseReceived(sessionInfo, event) {
|
|
402
|
+
var _this$_page6;
|
|
403
|
+
let request = this._requestIdToRequest.get(event.requestId);
|
|
404
|
+
// For frame-level Requests that are handled by a Service Worker's fetch handler, we'll never get a requestPaused event, so we need to
|
|
405
|
+
// manually create the request. In an ideal world, crNetworkManager would be able to know this on Network.requestWillBeSent, but there
|
|
406
|
+
// is not enough metadata there.
|
|
407
|
+
if (!request && event.response.fromServiceWorker) {
|
|
408
|
+
const requestWillBeSentEvent = this._requestIdToRequestWillBeSentEvent.get(event.requestId);
|
|
409
|
+
if (requestWillBeSentEvent) {
|
|
410
|
+
this._requestIdToRequestWillBeSentEvent.delete(event.requestId);
|
|
411
|
+
this._onRequest(sessionInfo, requestWillBeSentEvent, null /* requestPausedPayload */);
|
|
412
|
+
request = this._requestIdToRequest.get(event.requestId);
|
|
413
|
+
}
|
|
414
|
+
}
|
|
415
|
+
// FileUpload sends a response without a matching request.
|
|
416
|
+
if (!request) return;
|
|
417
|
+
const response = this._createResponse(request, event.response, event.hasExtraInfo);
|
|
418
|
+
(((_this$_page6 = this._page) === null || _this$_page6 === void 0 ? void 0 : _this$_page6._frameManager) || this._serviceWorker).requestReceivedResponse(response);
|
|
419
|
+
}
|
|
420
|
+
_onLoadingFinished(event) {
|
|
421
|
+
var _this$_page7;
|
|
422
|
+
this._responseExtraInfoTracker.loadingFinished(event);
|
|
423
|
+
let request = this._requestIdToRequest.get(event.requestId);
|
|
424
|
+
if (!request) request = this._maybeAdoptMainRequest(event.requestId);
|
|
425
|
+
// For certain requestIds we never receive requestWillBeSent event.
|
|
426
|
+
// @see https://crbug.com/750469
|
|
427
|
+
if (!request) return;
|
|
428
|
+
|
|
429
|
+
// Under certain conditions we never get the Network.responseReceived
|
|
430
|
+
// event from protocol. @see https://crbug.com/883475
|
|
431
|
+
const response = request.request._existingResponse();
|
|
432
|
+
if (response) {
|
|
433
|
+
response.setTransferSize(event.encodedDataLength);
|
|
434
|
+
response.responseHeadersSize().then(size => response.setEncodedBodySize(event.encodedDataLength - size));
|
|
435
|
+
response._requestFinished(_helper.helper.secondsToRoundishMillis(event.timestamp - request._timestamp));
|
|
436
|
+
}
|
|
437
|
+
this._deleteRequest(request);
|
|
438
|
+
(((_this$_page7 = this._page) === null || _this$_page7 === void 0 ? void 0 : _this$_page7._frameManager) || this._serviceWorker).reportRequestFinished(request.request, response);
|
|
439
|
+
}
|
|
440
|
+
_onLoadingFailed(sessionInfo, event) {
|
|
441
|
+
var _this$_page8;
|
|
442
|
+
this._responseExtraInfoTracker.loadingFailed(event);
|
|
443
|
+
let request = this._requestIdToRequest.get(event.requestId);
|
|
444
|
+
if (!request) request = this._maybeAdoptMainRequest(event.requestId);
|
|
445
|
+
if (!request) {
|
|
446
|
+
const requestWillBeSentEvent = this._requestIdToRequestWillBeSentEvent.get(event.requestId);
|
|
447
|
+
if (requestWillBeSentEvent) {
|
|
448
|
+
// This is a case where request has failed before we had a chance to intercept it.
|
|
449
|
+
// We stop waiting for Fetch.requestPaused (it might never come), and dispatch request event
|
|
450
|
+
// right away, followed by requestfailed event.
|
|
451
|
+
this._requestIdToRequestWillBeSentEvent.delete(event.requestId);
|
|
452
|
+
this._onRequest(sessionInfo, requestWillBeSentEvent, null);
|
|
453
|
+
request = this._requestIdToRequest.get(event.requestId);
|
|
454
|
+
}
|
|
455
|
+
}
|
|
456
|
+
|
|
457
|
+
// For certain requestIds we never receive requestWillBeSent event.
|
|
458
|
+
// @see https://crbug.com/750469
|
|
459
|
+
if (!request) return;
|
|
460
|
+
const response = request.request._existingResponse();
|
|
461
|
+
if (response) {
|
|
462
|
+
response.setTransferSize(null);
|
|
463
|
+
response.setEncodedBodySize(null);
|
|
464
|
+
response._requestFinished(_helper.helper.secondsToRoundishMillis(event.timestamp - request._timestamp));
|
|
465
|
+
}
|
|
466
|
+
this._deleteRequest(request);
|
|
467
|
+
request.request._setFailureText(event.errorText);
|
|
468
|
+
(((_this$_page8 = this._page) === null || _this$_page8 === void 0 ? void 0 : _this$_page8._frameManager) || this._serviceWorker).requestFailed(request.request, !!event.canceled);
|
|
469
|
+
}
|
|
470
|
+
_maybeAdoptMainRequest(requestId) {
|
|
471
|
+
// OOPIF has a main request that starts in the parent session but finishes in the child session.
|
|
472
|
+
if (!this._parentManager) return;
|
|
473
|
+
const request = this._parentManager._requestIdToRequest.get(requestId);
|
|
474
|
+
// Main requests have matching loaderId and requestId.
|
|
475
|
+
if (!request || request._documentId !== requestId) return;
|
|
476
|
+
this._requestIdToRequest.set(requestId, request);
|
|
477
|
+
request.session = this._session;
|
|
478
|
+
this._parentManager._requestIdToRequest.delete(requestId);
|
|
479
|
+
if (request._interceptionId && this._parentManager._attemptedAuthentications.has(request._interceptionId)) {
|
|
480
|
+
this._parentManager._attemptedAuthentications.delete(request._interceptionId);
|
|
481
|
+
this._attemptedAuthentications.add(request._interceptionId);
|
|
482
|
+
}
|
|
483
|
+
return request;
|
|
484
|
+
}
|
|
485
|
+
}
|
|
486
|
+
exports.CRNetworkManager = CRNetworkManager;
|
|
487
|
+
class InterceptableRequest {
|
|
488
|
+
constructor(options) {
|
|
489
|
+
this.request = void 0;
|
|
490
|
+
this._requestId = void 0;
|
|
491
|
+
this._interceptionId = void 0;
|
|
492
|
+
this._documentId = void 0;
|
|
493
|
+
this._timestamp = void 0;
|
|
494
|
+
this._wallTime = void 0;
|
|
495
|
+
this._route = void 0;
|
|
496
|
+
// Only first request in the chain can be intercepted, so this will
|
|
497
|
+
// store the first and only Route in the chain (if any).
|
|
498
|
+
this._originalRequestRoute = void 0;
|
|
499
|
+
this.session = void 0;
|
|
500
|
+
const {
|
|
501
|
+
session,
|
|
502
|
+
context,
|
|
503
|
+
frame,
|
|
504
|
+
documentId,
|
|
505
|
+
route,
|
|
506
|
+
requestWillBeSentEvent,
|
|
507
|
+
requestPausedEvent,
|
|
508
|
+
redirectedFrom,
|
|
509
|
+
serviceWorker
|
|
510
|
+
} = options;
|
|
511
|
+
this.session = session;
|
|
512
|
+
this._timestamp = requestWillBeSentEvent.timestamp;
|
|
513
|
+
this._wallTime = requestWillBeSentEvent.wallTime;
|
|
514
|
+
this._requestId = requestWillBeSentEvent.requestId;
|
|
515
|
+
this._interceptionId = requestPausedEvent && requestPausedEvent.requestId;
|
|
516
|
+
this._documentId = documentId;
|
|
517
|
+
this._route = route;
|
|
518
|
+
this._originalRequestRoute = route !== null && route !== void 0 ? route : redirectedFrom === null || redirectedFrom === void 0 ? void 0 : redirectedFrom._originalRequestRoute;
|
|
519
|
+
const {
|
|
520
|
+
headers,
|
|
521
|
+
method,
|
|
522
|
+
url,
|
|
523
|
+
postDataEntries = null
|
|
524
|
+
} = requestPausedEvent ? requestPausedEvent.request : requestWillBeSentEvent.request;
|
|
525
|
+
const type = (requestWillBeSentEvent.type || '').toLowerCase();
|
|
526
|
+
let postDataBuffer = null;
|
|
527
|
+
if (postDataEntries && postDataEntries.length && postDataEntries[0].bytes) postDataBuffer = Buffer.from(postDataEntries[0].bytes, 'base64');
|
|
528
|
+
this.request = new network.Request(context, frame, serviceWorker, (redirectedFrom === null || redirectedFrom === void 0 ? void 0 : redirectedFrom.request) || null, documentId, url, type, method, postDataBuffer, (0, _utils.headersObjectToArray)(headers));
|
|
529
|
+
}
|
|
530
|
+
}
|
|
531
|
+
class RouteImpl {
|
|
532
|
+
constructor(session, interceptionId) {
|
|
533
|
+
this._session = void 0;
|
|
534
|
+
this._interceptionId = void 0;
|
|
535
|
+
this._alreadyContinuedParams = void 0;
|
|
536
|
+
this._session = session;
|
|
537
|
+
this._interceptionId = interceptionId;
|
|
538
|
+
}
|
|
539
|
+
async continue(request, overrides) {
|
|
540
|
+
this._alreadyContinuedParams = {
|
|
541
|
+
requestId: this._interceptionId,
|
|
542
|
+
url: overrides.url,
|
|
543
|
+
headers: overrides.headers,
|
|
544
|
+
method: overrides.method,
|
|
545
|
+
postData: overrides.postData ? overrides.postData.toString('base64') : undefined
|
|
546
|
+
};
|
|
547
|
+
await catchDisallowedErrors(async () => {
|
|
548
|
+
await this._session.send('Fetch.continueRequest', this._alreadyContinuedParams);
|
|
549
|
+
});
|
|
550
|
+
}
|
|
551
|
+
async fulfill(response) {
|
|
552
|
+
const body = response.isBase64 ? response.body : Buffer.from(response.body).toString('base64');
|
|
553
|
+
const responseHeaders = splitSetCookieHeader(response.headers);
|
|
554
|
+
await catchDisallowedErrors(async () => {
|
|
555
|
+
await this._session.send('Fetch.fulfillRequest', {
|
|
556
|
+
requestId: this._interceptionId,
|
|
557
|
+
responseCode: response.status,
|
|
558
|
+
responsePhrase: network.STATUS_TEXTS[String(response.status)],
|
|
559
|
+
responseHeaders,
|
|
560
|
+
body
|
|
561
|
+
});
|
|
562
|
+
});
|
|
563
|
+
}
|
|
564
|
+
async abort(errorCode = 'failed') {
|
|
565
|
+
const errorReason = errorReasons[errorCode];
|
|
566
|
+
(0, _utils.assert)(errorReason, 'Unknown error code: ' + errorCode);
|
|
567
|
+
await catchDisallowedErrors(async () => {
|
|
568
|
+
await this._session.send('Fetch.failRequest', {
|
|
569
|
+
requestId: this._interceptionId,
|
|
570
|
+
errorReason
|
|
571
|
+
});
|
|
572
|
+
});
|
|
573
|
+
}
|
|
574
|
+
}
|
|
575
|
+
|
|
576
|
+
// In certain cases, protocol will return error if the request was already canceled
|
|
577
|
+
// or the page was closed. We should tolerate these errors but propagate other.
|
|
578
|
+
async function catchDisallowedErrors(callback) {
|
|
579
|
+
try {
|
|
580
|
+
return await callback();
|
|
581
|
+
} catch (e) {
|
|
582
|
+
if ((0, _protocolError.isProtocolError)(e) && e.message.includes('Invalid http status code or phrase')) throw e;
|
|
583
|
+
}
|
|
584
|
+
}
|
|
585
|
+
function splitSetCookieHeader(headers) {
|
|
586
|
+
const index = headers.findIndex(({
|
|
587
|
+
name
|
|
588
|
+
}) => name.toLowerCase() === 'set-cookie');
|
|
589
|
+
if (index === -1) return headers;
|
|
590
|
+
const header = headers[index];
|
|
591
|
+
const values = header.value.split('\n');
|
|
592
|
+
if (values.length === 1) return headers;
|
|
593
|
+
const result = headers.slice();
|
|
594
|
+
result.splice(index, 1, ...values.map(value => ({
|
|
595
|
+
name: header.name,
|
|
596
|
+
value
|
|
597
|
+
})));
|
|
598
|
+
return result;
|
|
599
|
+
}
|
|
600
|
+
const errorReasons = {
|
|
601
|
+
'aborted': 'Aborted',
|
|
602
|
+
'accessdenied': 'AccessDenied',
|
|
603
|
+
'addressunreachable': 'AddressUnreachable',
|
|
604
|
+
'blockedbyclient': 'BlockedByClient',
|
|
605
|
+
'blockedbyresponse': 'BlockedByResponse',
|
|
606
|
+
'connectionaborted': 'ConnectionAborted',
|
|
607
|
+
'connectionclosed': 'ConnectionClosed',
|
|
608
|
+
'connectionfailed': 'ConnectionFailed',
|
|
609
|
+
'connectionrefused': 'ConnectionRefused',
|
|
610
|
+
'connectionreset': 'ConnectionReset',
|
|
611
|
+
'internetdisconnected': 'InternetDisconnected',
|
|
612
|
+
'namenotresolved': 'NameNotResolved',
|
|
613
|
+
'timedout': 'TimedOut',
|
|
614
|
+
'failed': 'Failed'
|
|
615
|
+
};
|
|
616
|
+
// This class aligns responses with response headers from extra info:
|
|
617
|
+
// - Network.requestWillBeSent, Network.responseReceived, Network.loadingFinished/loadingFailed are
|
|
618
|
+
// dispatched using one channel.
|
|
619
|
+
// - Network.requestWillBeSentExtraInfo and Network.responseReceivedExtraInfo are dispatched on
|
|
620
|
+
// another channel. Those channels are not associated, so events come in random order.
|
|
621
|
+
//
|
|
622
|
+
// This class will associate responses with the new headers. These extra info headers will become
|
|
623
|
+
// available to client reliably upon requestfinished event only. It consumes CDP
|
|
624
|
+
// signals on one end and processResponse(network.Response) signals on the other hands. It then makes
|
|
625
|
+
// sure that responses have all the extra headers in place by the time request finishes.
|
|
626
|
+
//
|
|
627
|
+
// The shape of the instrumentation API is deliberately following the CDP, so that it
|
|
628
|
+
// is clear what is called when and what this means to the tracker without extra
|
|
629
|
+
// documentation.
|
|
630
|
+
class ResponseExtraInfoTracker {
|
|
631
|
+
constructor() {
|
|
632
|
+
this._requests = new Map();
|
|
633
|
+
}
|
|
634
|
+
requestWillBeSentExtraInfo(event) {
|
|
635
|
+
const info = this._getOrCreateEntry(event.requestId);
|
|
636
|
+
info.requestWillBeSentExtraInfo.push(event);
|
|
637
|
+
this._patchHeaders(info, info.requestWillBeSentExtraInfo.length - 1);
|
|
638
|
+
this._checkFinished(info);
|
|
639
|
+
}
|
|
640
|
+
requestServedFromCache(event) {
|
|
641
|
+
const info = this._getOrCreateEntry(event.requestId);
|
|
642
|
+
info.servedFromCache = true;
|
|
643
|
+
}
|
|
644
|
+
servedFromCache(requestId) {
|
|
645
|
+
const info = this._requests.get(requestId);
|
|
646
|
+
return !!(info !== null && info !== void 0 && info.servedFromCache);
|
|
647
|
+
}
|
|
648
|
+
responseReceivedExtraInfo(event) {
|
|
649
|
+
const info = this._getOrCreateEntry(event.requestId);
|
|
650
|
+
info.responseReceivedExtraInfo.push(event);
|
|
651
|
+
this._patchHeaders(info, info.responseReceivedExtraInfo.length - 1);
|
|
652
|
+
this._checkFinished(info);
|
|
653
|
+
}
|
|
654
|
+
processResponse(requestId, response, hasExtraInfo) {
|
|
655
|
+
var _info;
|
|
656
|
+
let info = this._requests.get(requestId);
|
|
657
|
+
// Cached responses have erroneous "hasExtraInfo" flag.
|
|
658
|
+
// https://bugs.chromium.org/p/chromium/issues/detail?id=1340398
|
|
659
|
+
if (!hasExtraInfo || (_info = info) !== null && _info !== void 0 && _info.servedFromCache) {
|
|
660
|
+
// Use "provisional" headers as "raw" ones.
|
|
661
|
+
response.request().setRawRequestHeaders(null);
|
|
662
|
+
response.setResponseHeadersSize(null);
|
|
663
|
+
response.setRawResponseHeaders(null);
|
|
664
|
+
return;
|
|
665
|
+
}
|
|
666
|
+
info = this._getOrCreateEntry(requestId);
|
|
667
|
+
info.responses.push(response);
|
|
668
|
+
this._patchHeaders(info, info.responses.length - 1);
|
|
669
|
+
}
|
|
670
|
+
loadingFinished(event) {
|
|
671
|
+
const info = this._requests.get(event.requestId);
|
|
672
|
+
if (!info) return;
|
|
673
|
+
info.loadingFinished = event;
|
|
674
|
+
this._checkFinished(info);
|
|
675
|
+
}
|
|
676
|
+
loadingFailed(event) {
|
|
677
|
+
const info = this._requests.get(event.requestId);
|
|
678
|
+
if (!info) return;
|
|
679
|
+
info.loadingFailed = event;
|
|
680
|
+
this._checkFinished(info);
|
|
681
|
+
}
|
|
682
|
+
_getOrCreateEntry(requestId) {
|
|
683
|
+
let info = this._requests.get(requestId);
|
|
684
|
+
if (!info) {
|
|
685
|
+
info = {
|
|
686
|
+
requestId: requestId,
|
|
687
|
+
requestWillBeSentExtraInfo: [],
|
|
688
|
+
responseReceivedExtraInfo: [],
|
|
689
|
+
responses: []
|
|
690
|
+
};
|
|
691
|
+
this._requests.set(requestId, info);
|
|
692
|
+
}
|
|
693
|
+
return info;
|
|
694
|
+
}
|
|
695
|
+
_patchHeaders(info, index) {
|
|
696
|
+
const response = info.responses[index];
|
|
697
|
+
const requestExtraInfo = info.requestWillBeSentExtraInfo[index];
|
|
698
|
+
if (response && requestExtraInfo) {
|
|
699
|
+
response.request().setRawRequestHeaders((0, _utils.headersObjectToArray)(requestExtraInfo.headers, '\n'));
|
|
700
|
+
info.requestWillBeSentExtraInfo[index] = undefined;
|
|
701
|
+
}
|
|
702
|
+
const responseExtraInfo = info.responseReceivedExtraInfo[index];
|
|
703
|
+
if (response && responseExtraInfo) {
|
|
704
|
+
var _responseExtraInfo$he;
|
|
705
|
+
response.setResponseHeadersSize(((_responseExtraInfo$he = responseExtraInfo.headersText) === null || _responseExtraInfo$he === void 0 ? void 0 : _responseExtraInfo$he.length) || 0);
|
|
706
|
+
response.setRawResponseHeaders((0, _utils.headersObjectToArray)(responseExtraInfo.headers, '\n'));
|
|
707
|
+
info.responseReceivedExtraInfo[index] = undefined;
|
|
708
|
+
}
|
|
709
|
+
}
|
|
710
|
+
_checkFinished(info) {
|
|
711
|
+
if (!info.loadingFinished && !info.loadingFailed) return;
|
|
712
|
+
if (info.responses.length <= info.responseReceivedExtraInfo.length) {
|
|
713
|
+
// We have extra info for each response.
|
|
714
|
+
this._stopTracking(info.requestId);
|
|
715
|
+
return;
|
|
716
|
+
}
|
|
717
|
+
|
|
718
|
+
// We are not done yet.
|
|
719
|
+
}
|
|
720
|
+
_stopTracking(requestId) {
|
|
721
|
+
this._requests.delete(requestId);
|
|
722
|
+
}
|
|
723
|
+
}
|