@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,1198 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
|
|
3
|
+
Object.defineProperty(exports, "__esModule", {
|
|
4
|
+
value: true
|
|
5
|
+
});
|
|
6
|
+
exports.WKPage = void 0;
|
|
7
|
+
var _path = _interopRequireDefault(require("path"));
|
|
8
|
+
var _utilsBundle = require("../../utilsBundle");
|
|
9
|
+
var _stackTrace = require("../../utils/stackTrace");
|
|
10
|
+
var _utils = require("../../utils");
|
|
11
|
+
var _hostPlatform = require("../../utils/hostPlatform");
|
|
12
|
+
var dialog = _interopRequireWildcard(require("../dialog"));
|
|
13
|
+
var dom = _interopRequireWildcard(require("../dom"));
|
|
14
|
+
var _eventsHelper = require("../../utils/eventsHelper");
|
|
15
|
+
var _helper = require("../helper");
|
|
16
|
+
var network = _interopRequireWildcard(require("../network"));
|
|
17
|
+
var _page = require("../page");
|
|
18
|
+
var _wkAccessibility = require("./wkAccessibility");
|
|
19
|
+
var _wkConnection = require("./wkConnection");
|
|
20
|
+
var _wkExecutionContext = require("./wkExecutionContext");
|
|
21
|
+
var _wkInput = require("./wkInput");
|
|
22
|
+
var _wkInterceptableRequest = require("./wkInterceptableRequest");
|
|
23
|
+
var _wkProvisionalPage = require("./wkProvisionalPage");
|
|
24
|
+
var _wkWorkers = require("./wkWorkers");
|
|
25
|
+
var _debugLogger = require("../../utils/debugLogger");
|
|
26
|
+
var _manualPromise = require("../../utils/manualPromise");
|
|
27
|
+
var _browserContext = require("../browserContext");
|
|
28
|
+
var _errors = require("../errors");
|
|
29
|
+
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); }
|
|
30
|
+
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; }
|
|
31
|
+
function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; }
|
|
32
|
+
/**
|
|
33
|
+
* Copyright 2017 Google Inc. All rights reserved.
|
|
34
|
+
* Modifications copyright (c) Microsoft Corporation.
|
|
35
|
+
*
|
|
36
|
+
* Licensed under the Apache License, Version 2.0 (the "License");
|
|
37
|
+
* you may not use this file except in compliance with the License.
|
|
38
|
+
* You may obtain a copy of the License at
|
|
39
|
+
*
|
|
40
|
+
* http://www.apache.org/licenses/LICENSE-2.0
|
|
41
|
+
*
|
|
42
|
+
* Unless required by applicable law or agreed to in writing, software
|
|
43
|
+
* distributed under the License is distributed on an "AS IS" BASIS,
|
|
44
|
+
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
45
|
+
* See the License for the specific language governing permissions and
|
|
46
|
+
* limitations under the License.
|
|
47
|
+
*/
|
|
48
|
+
|
|
49
|
+
const UTILITY_WORLD_NAME = '__playwright_utility_world__';
|
|
50
|
+
class WKPage {
|
|
51
|
+
constructor(browserContext, pageProxySession, opener) {
|
|
52
|
+
this.rawMouse = void 0;
|
|
53
|
+
this.rawKeyboard = void 0;
|
|
54
|
+
this.rawTouchscreen = void 0;
|
|
55
|
+
this._session = void 0;
|
|
56
|
+
this._provisionalPage = null;
|
|
57
|
+
this._page = void 0;
|
|
58
|
+
this._pagePromise = new _manualPromise.ManualPromise();
|
|
59
|
+
this._pageProxySession = void 0;
|
|
60
|
+
this._opener = void 0;
|
|
61
|
+
this._requestIdToRequest = new Map();
|
|
62
|
+
this._requestIdToRequestWillBeSentEvent = new Map();
|
|
63
|
+
this._workers = void 0;
|
|
64
|
+
this._contextIdToContext = void 0;
|
|
65
|
+
this._sessionListeners = [];
|
|
66
|
+
this._eventListeners = void 0;
|
|
67
|
+
this._browserContext = void 0;
|
|
68
|
+
this._initializedPage = null;
|
|
69
|
+
this._firstNonInitialNavigationCommittedPromise = void 0;
|
|
70
|
+
this._firstNonInitialNavigationCommittedFulfill = () => {};
|
|
71
|
+
this._firstNonInitialNavigationCommittedReject = e => {};
|
|
72
|
+
this._lastConsoleMessage = null;
|
|
73
|
+
this._requestIdToResponseReceivedPayloadEvent = new Map();
|
|
74
|
+
// Holds window features for the next popup being opened via window.open,
|
|
75
|
+
// until the popup page proxy arrives.
|
|
76
|
+
this._nextWindowOpenPopupFeatures = void 0;
|
|
77
|
+
this._recordingVideoFile = null;
|
|
78
|
+
this._screencastGeneration = 0;
|
|
79
|
+
this._pageProxySession = pageProxySession;
|
|
80
|
+
this._opener = opener;
|
|
81
|
+
this.rawKeyboard = new _wkInput.RawKeyboardImpl(pageProxySession);
|
|
82
|
+
this.rawMouse = new _wkInput.RawMouseImpl(pageProxySession);
|
|
83
|
+
this.rawTouchscreen = new _wkInput.RawTouchscreenImpl(pageProxySession);
|
|
84
|
+
this._contextIdToContext = new Map();
|
|
85
|
+
this._page = new _page.Page(this, browserContext);
|
|
86
|
+
this.rawMouse.setPage(this._page);
|
|
87
|
+
this._workers = new _wkWorkers.WKWorkers(this._page);
|
|
88
|
+
this._session = undefined;
|
|
89
|
+
this._browserContext = browserContext;
|
|
90
|
+
this._page.on(_page.Page.Events.FrameDetached, frame => this._removeContextsForFrame(frame, false));
|
|
91
|
+
this._eventListeners = [_eventsHelper.eventsHelper.addEventListener(this._pageProxySession, 'Target.targetCreated', this._onTargetCreated.bind(this)), _eventsHelper.eventsHelper.addEventListener(this._pageProxySession, 'Target.targetDestroyed', this._onTargetDestroyed.bind(this)), _eventsHelper.eventsHelper.addEventListener(this._pageProxySession, 'Target.dispatchMessageFromTarget', this._onDispatchMessageFromTarget.bind(this)), _eventsHelper.eventsHelper.addEventListener(this._pageProxySession, 'Target.didCommitProvisionalTarget', this._onDidCommitProvisionalTarget.bind(this)), _eventsHelper.eventsHelper.addEventListener(this._pageProxySession, 'Screencast.screencastFrame', this._onScreencastFrame.bind(this))];
|
|
92
|
+
this._firstNonInitialNavigationCommittedPromise = new Promise((f, r) => {
|
|
93
|
+
this._firstNonInitialNavigationCommittedFulfill = f;
|
|
94
|
+
this._firstNonInitialNavigationCommittedReject = r;
|
|
95
|
+
});
|
|
96
|
+
if (opener && !browserContext._options.noDefaultViewport && opener._nextWindowOpenPopupFeatures) {
|
|
97
|
+
const viewportSize = _helper.helper.getViewportSizeFromWindowFeatures(opener._nextWindowOpenPopupFeatures);
|
|
98
|
+
opener._nextWindowOpenPopupFeatures = undefined;
|
|
99
|
+
if (viewportSize) this._page._emulatedSize = {
|
|
100
|
+
viewport: viewportSize,
|
|
101
|
+
screen: viewportSize
|
|
102
|
+
};
|
|
103
|
+
}
|
|
104
|
+
}
|
|
105
|
+
potentiallyUninitializedPage() {
|
|
106
|
+
return this._page;
|
|
107
|
+
}
|
|
108
|
+
async _initializePageProxySession() {
|
|
109
|
+
if (this._page._browserContext.isSettingStorageState()) return;
|
|
110
|
+
const promises = [this._pageProxySession.send('Dialog.enable'), this._pageProxySession.send('Emulation.setActiveAndFocused', {
|
|
111
|
+
active: true
|
|
112
|
+
})];
|
|
113
|
+
const contextOptions = this._browserContext._options;
|
|
114
|
+
if (contextOptions.javaScriptEnabled === false) promises.push(this._pageProxySession.send('Emulation.setJavaScriptEnabled', {
|
|
115
|
+
enabled: false
|
|
116
|
+
}));
|
|
117
|
+
promises.push(this._updateViewport());
|
|
118
|
+
promises.push(this.updateHttpCredentials());
|
|
119
|
+
if (this._browserContext._permissions.size) {
|
|
120
|
+
for (const [key, value] of this._browserContext._permissions) promises.push(this._grantPermissions(key, value));
|
|
121
|
+
}
|
|
122
|
+
if (this._browserContext._options.recordVideo) {
|
|
123
|
+
const outputFile = _path.default.join(this._browserContext._options.recordVideo.dir, (0, _utils.createGuid)() + '.webm');
|
|
124
|
+
promises.push(this._browserContext._ensureVideosPath().then(() => {
|
|
125
|
+
return this._startVideo({
|
|
126
|
+
// validateBrowserContextOptions ensures correct video size.
|
|
127
|
+
...this._browserContext._options.recordVideo.size,
|
|
128
|
+
outputFile
|
|
129
|
+
});
|
|
130
|
+
}));
|
|
131
|
+
}
|
|
132
|
+
await Promise.all(promises);
|
|
133
|
+
}
|
|
134
|
+
_setSession(session) {
|
|
135
|
+
_eventsHelper.eventsHelper.removeEventListeners(this._sessionListeners);
|
|
136
|
+
this._session = session;
|
|
137
|
+
this.rawKeyboard.setSession(session);
|
|
138
|
+
this.rawMouse.setSession(session);
|
|
139
|
+
this._addSessionListeners();
|
|
140
|
+
this._workers.setSession(session);
|
|
141
|
+
}
|
|
142
|
+
|
|
143
|
+
// This method is called for provisional targets as well. The session passed as the parameter
|
|
144
|
+
// may be different from the current session and may be destroyed without becoming current.
|
|
145
|
+
async _initializeSession(session, provisional, resourceTreeHandler) {
|
|
146
|
+
await this._initializeSessionMayThrow(session, resourceTreeHandler).catch(e => {
|
|
147
|
+
// Provisional session can be disposed at any time, for example due to new navigation initiating
|
|
148
|
+
// a new provisional page.
|
|
149
|
+
if (provisional && session.isDisposed()) return;
|
|
150
|
+
// Swallow initialization errors due to newer target swap in,
|
|
151
|
+
// since we will reinitialize again.
|
|
152
|
+
if (this._session === session) throw e;
|
|
153
|
+
});
|
|
154
|
+
}
|
|
155
|
+
async _initializeSessionMayThrow(session, resourceTreeHandler) {
|
|
156
|
+
const [, frameTree] = await Promise.all([
|
|
157
|
+
// Page agent must be enabled before Runtime.
|
|
158
|
+
session.send('Page.enable'), session.send('Page.getResourceTree')]);
|
|
159
|
+
resourceTreeHandler(frameTree);
|
|
160
|
+
const promises = [
|
|
161
|
+
// Resource tree should be received before first execution context.
|
|
162
|
+
session.send('Runtime.enable'), session.send('Page.createUserWorld', {
|
|
163
|
+
name: UTILITY_WORLD_NAME
|
|
164
|
+
}).catch(_ => {}),
|
|
165
|
+
// Worlds are per-process
|
|
166
|
+
session.send('Console.enable'), session.send('Network.enable'), this._workers.initializeSession(session)];
|
|
167
|
+
if (this._page.needsRequestInterception()) {
|
|
168
|
+
promises.push(session.send('Network.setInterceptionEnabled', {
|
|
169
|
+
enabled: true
|
|
170
|
+
}));
|
|
171
|
+
promises.push(session.send('Network.setResourceCachingDisabled', {
|
|
172
|
+
disabled: true
|
|
173
|
+
}));
|
|
174
|
+
promises.push(session.send('Network.addInterception', {
|
|
175
|
+
url: '.*',
|
|
176
|
+
stage: 'request',
|
|
177
|
+
isRegex: true
|
|
178
|
+
}));
|
|
179
|
+
}
|
|
180
|
+
if (this._page._browserContext.isSettingStorageState()) {
|
|
181
|
+
await Promise.all(promises);
|
|
182
|
+
return;
|
|
183
|
+
}
|
|
184
|
+
const contextOptions = this._browserContext._options;
|
|
185
|
+
if (contextOptions.userAgent) promises.push(this.updateUserAgent());
|
|
186
|
+
const emulatedMedia = this._page.emulatedMedia();
|
|
187
|
+
if (emulatedMedia.media || emulatedMedia.colorScheme || emulatedMedia.reducedMotion || emulatedMedia.forcedColors) promises.push(WKPage._setEmulateMedia(session, emulatedMedia.media, emulatedMedia.colorScheme, emulatedMedia.reducedMotion, emulatedMedia.forcedColors));
|
|
188
|
+
for (const binding of this._page.allBindings()) promises.push(session.send('Runtime.addBinding', {
|
|
189
|
+
name: binding.name
|
|
190
|
+
}));
|
|
191
|
+
const bootstrapScript = this._calculateBootstrapScript();
|
|
192
|
+
if (bootstrapScript.length) promises.push(session.send('Page.setBootstrapScript', {
|
|
193
|
+
source: bootstrapScript
|
|
194
|
+
}));
|
|
195
|
+
this._page.frames().map(frame => frame.evaluateExpression(bootstrapScript).catch(e => {}));
|
|
196
|
+
if (contextOptions.bypassCSP) promises.push(session.send('Page.setBypassCSP', {
|
|
197
|
+
enabled: true
|
|
198
|
+
}));
|
|
199
|
+
const emulatedSize = this._page.emulatedSize();
|
|
200
|
+
if (emulatedSize) {
|
|
201
|
+
promises.push(session.send('Page.setScreenSizeOverride', {
|
|
202
|
+
width: emulatedSize.screen.width,
|
|
203
|
+
height: emulatedSize.screen.height
|
|
204
|
+
}));
|
|
205
|
+
}
|
|
206
|
+
promises.push(this.updateEmulateMedia());
|
|
207
|
+
promises.push(session.send('Network.setExtraHTTPHeaders', {
|
|
208
|
+
headers: (0, _utils.headersArrayToObject)(this._calculateExtraHTTPHeaders(), false /* lowerCase */)
|
|
209
|
+
}));
|
|
210
|
+
if (contextOptions.offline) promises.push(session.send('Network.setEmulateOfflineState', {
|
|
211
|
+
offline: true
|
|
212
|
+
}));
|
|
213
|
+
promises.push(session.send('Page.setTouchEmulationEnabled', {
|
|
214
|
+
enabled: !!contextOptions.hasTouch
|
|
215
|
+
}));
|
|
216
|
+
if (contextOptions.timezoneId) {
|
|
217
|
+
promises.push(session.send('Page.setTimeZone', {
|
|
218
|
+
timeZone: contextOptions.timezoneId
|
|
219
|
+
}).catch(e => {
|
|
220
|
+
throw new Error(`Invalid timezone ID: ${contextOptions.timezoneId}`);
|
|
221
|
+
}));
|
|
222
|
+
}
|
|
223
|
+
if (this._page.fileChooserIntercepted()) promises.push(session.send('Page.setInterceptFileChooserDialog', {
|
|
224
|
+
enabled: true
|
|
225
|
+
}));
|
|
226
|
+
promises.push(session.send('Page.overrideSetting', {
|
|
227
|
+
setting: 'DeviceOrientationEventEnabled',
|
|
228
|
+
value: contextOptions.isMobile
|
|
229
|
+
}));
|
|
230
|
+
promises.push(session.send('Page.overrideSetting', {
|
|
231
|
+
setting: 'FullScreenEnabled',
|
|
232
|
+
value: !contextOptions.isMobile
|
|
233
|
+
}));
|
|
234
|
+
promises.push(session.send('Page.overrideSetting', {
|
|
235
|
+
setting: 'NotificationsEnabled',
|
|
236
|
+
value: !contextOptions.isMobile
|
|
237
|
+
}));
|
|
238
|
+
promises.push(session.send('Page.overrideSetting', {
|
|
239
|
+
setting: 'PointerLockEnabled',
|
|
240
|
+
value: !contextOptions.isMobile
|
|
241
|
+
}));
|
|
242
|
+
promises.push(session.send('Page.overrideSetting', {
|
|
243
|
+
setting: 'InputTypeMonthEnabled',
|
|
244
|
+
value: contextOptions.isMobile
|
|
245
|
+
}));
|
|
246
|
+
promises.push(session.send('Page.overrideSetting', {
|
|
247
|
+
setting: 'InputTypeWeekEnabled',
|
|
248
|
+
value: contextOptions.isMobile
|
|
249
|
+
}));
|
|
250
|
+
await Promise.all(promises);
|
|
251
|
+
}
|
|
252
|
+
_onDidCommitProvisionalTarget(event) {
|
|
253
|
+
const {
|
|
254
|
+
oldTargetId,
|
|
255
|
+
newTargetId
|
|
256
|
+
} = event;
|
|
257
|
+
(0, _utils.assert)(this._provisionalPage);
|
|
258
|
+
(0, _utils.assert)(this._provisionalPage._session.sessionId === newTargetId, 'Unknown new target: ' + newTargetId);
|
|
259
|
+
(0, _utils.assert)(this._session.sessionId === oldTargetId, 'Unknown old target: ' + oldTargetId);
|
|
260
|
+
const newSession = this._provisionalPage._session;
|
|
261
|
+
this._provisionalPage.commit();
|
|
262
|
+
this._provisionalPage.dispose();
|
|
263
|
+
this._provisionalPage = null;
|
|
264
|
+
this._setSession(newSession);
|
|
265
|
+
}
|
|
266
|
+
_onTargetDestroyed(event) {
|
|
267
|
+
const {
|
|
268
|
+
targetId,
|
|
269
|
+
crashed
|
|
270
|
+
} = event;
|
|
271
|
+
if (this._provisionalPage && this._provisionalPage._session.sessionId === targetId) {
|
|
272
|
+
this._provisionalPage._session.dispose();
|
|
273
|
+
this._provisionalPage.dispose();
|
|
274
|
+
this._provisionalPage = null;
|
|
275
|
+
} else if (this._session.sessionId === targetId) {
|
|
276
|
+
this._session.dispose();
|
|
277
|
+
_eventsHelper.eventsHelper.removeEventListeners(this._sessionListeners);
|
|
278
|
+
if (crashed) {
|
|
279
|
+
this._session.markAsCrashed();
|
|
280
|
+
this._page._didCrash();
|
|
281
|
+
}
|
|
282
|
+
}
|
|
283
|
+
}
|
|
284
|
+
didClose() {
|
|
285
|
+
this._pageProxySession.dispose();
|
|
286
|
+
_eventsHelper.eventsHelper.removeEventListeners(this._sessionListeners);
|
|
287
|
+
_eventsHelper.eventsHelper.removeEventListeners(this._eventListeners);
|
|
288
|
+
if (this._session) this._session.dispose();
|
|
289
|
+
if (this._provisionalPage) {
|
|
290
|
+
this._provisionalPage._session.dispose();
|
|
291
|
+
this._provisionalPage.dispose();
|
|
292
|
+
this._provisionalPage = null;
|
|
293
|
+
}
|
|
294
|
+
this._firstNonInitialNavigationCommittedReject(new _errors.TargetClosedError());
|
|
295
|
+
this._page._didClose();
|
|
296
|
+
}
|
|
297
|
+
dispatchMessageToSession(message) {
|
|
298
|
+
this._pageProxySession.dispatchMessage(message);
|
|
299
|
+
}
|
|
300
|
+
handleProvisionalLoadFailed(event) {
|
|
301
|
+
if (!this._initializedPage) {
|
|
302
|
+
this._firstNonInitialNavigationCommittedReject(new Error('Initial load failed'));
|
|
303
|
+
return;
|
|
304
|
+
}
|
|
305
|
+
if (!this._provisionalPage) return;
|
|
306
|
+
let errorText = event.error;
|
|
307
|
+
if (errorText.includes('cancelled')) errorText += '; maybe frame was detached?';
|
|
308
|
+
this._page._frameManager.frameAbortedNavigation(this._page.mainFrame()._id, errorText, event.loaderId);
|
|
309
|
+
}
|
|
310
|
+
handleWindowOpen(event) {
|
|
311
|
+
(0, _utils.debugAssert)(!this._nextWindowOpenPopupFeatures);
|
|
312
|
+
this._nextWindowOpenPopupFeatures = event.windowFeatures;
|
|
313
|
+
}
|
|
314
|
+
async pageOrError() {
|
|
315
|
+
return this._pagePromise;
|
|
316
|
+
}
|
|
317
|
+
async _onTargetCreated(event) {
|
|
318
|
+
const {
|
|
319
|
+
targetInfo
|
|
320
|
+
} = event;
|
|
321
|
+
const session = new _wkConnection.WKSession(this._pageProxySession.connection, targetInfo.targetId, message => {
|
|
322
|
+
this._pageProxySession.send('Target.sendMessageToTarget', {
|
|
323
|
+
message: JSON.stringify(message),
|
|
324
|
+
targetId: targetInfo.targetId
|
|
325
|
+
}).catch(e => {
|
|
326
|
+
session.dispatchMessage({
|
|
327
|
+
id: message.id,
|
|
328
|
+
error: {
|
|
329
|
+
message: e.message
|
|
330
|
+
}
|
|
331
|
+
});
|
|
332
|
+
});
|
|
333
|
+
});
|
|
334
|
+
(0, _utils.assert)(targetInfo.type === 'page', 'Only page targets are expected in WebKit, received: ' + targetInfo.type);
|
|
335
|
+
if (!targetInfo.isProvisional) {
|
|
336
|
+
(0, _utils.assert)(!this._initializedPage);
|
|
337
|
+
let pageOrError;
|
|
338
|
+
try {
|
|
339
|
+
this._setSession(session);
|
|
340
|
+
await Promise.all([this._initializePageProxySession(), this._initializeSession(session, false, ({
|
|
341
|
+
frameTree
|
|
342
|
+
}) => this._handleFrameTree(frameTree))]);
|
|
343
|
+
pageOrError = this._page;
|
|
344
|
+
} catch (e) {
|
|
345
|
+
pageOrError = e;
|
|
346
|
+
}
|
|
347
|
+
if (targetInfo.isPaused) this._pageProxySession.sendMayFail('Target.resume', {
|
|
348
|
+
targetId: targetInfo.targetId
|
|
349
|
+
});
|
|
350
|
+
if (pageOrError instanceof _page.Page && this._page.mainFrame().url() === '') {
|
|
351
|
+
try {
|
|
352
|
+
// Initial empty page has an empty url. We should wait until the first real url has been loaded,
|
|
353
|
+
// even if that url is about:blank. This is especially important for popups, where we need the
|
|
354
|
+
// actual url before interacting with it.
|
|
355
|
+
await this._firstNonInitialNavigationCommittedPromise;
|
|
356
|
+
} catch (e) {
|
|
357
|
+
pageOrError = e;
|
|
358
|
+
}
|
|
359
|
+
} else {
|
|
360
|
+
// Avoid rejection on disconnect.
|
|
361
|
+
this._firstNonInitialNavigationCommittedPromise.catch(() => {});
|
|
362
|
+
}
|
|
363
|
+
await this._page.initOpener(this._opener);
|
|
364
|
+
// Note: it is important to call |reportAsNew| before resolving pageOrError promise,
|
|
365
|
+
// so that anyone who awaits pageOrError got a ready and reported page.
|
|
366
|
+
this._initializedPage = pageOrError instanceof _page.Page ? pageOrError : null;
|
|
367
|
+
this._page.reportAsNew(pageOrError instanceof _page.Page ? undefined : pageOrError);
|
|
368
|
+
this._pagePromise.resolve(pageOrError);
|
|
369
|
+
} else {
|
|
370
|
+
(0, _utils.assert)(targetInfo.isProvisional);
|
|
371
|
+
(0, _utils.assert)(!this._provisionalPage);
|
|
372
|
+
this._provisionalPage = new _wkProvisionalPage.WKProvisionalPage(session, this);
|
|
373
|
+
if (targetInfo.isPaused) {
|
|
374
|
+
this._provisionalPage.initializationPromise.then(() => {
|
|
375
|
+
this._pageProxySession.sendMayFail('Target.resume', {
|
|
376
|
+
targetId: targetInfo.targetId
|
|
377
|
+
});
|
|
378
|
+
});
|
|
379
|
+
}
|
|
380
|
+
}
|
|
381
|
+
}
|
|
382
|
+
_onDispatchMessageFromTarget(event) {
|
|
383
|
+
const {
|
|
384
|
+
targetId,
|
|
385
|
+
message
|
|
386
|
+
} = event;
|
|
387
|
+
if (this._provisionalPage && this._provisionalPage._session.sessionId === targetId) this._provisionalPage._session.dispatchMessage(JSON.parse(message));else if (this._session.sessionId === targetId) this._session.dispatchMessage(JSON.parse(message));else throw new Error('Unknown target: ' + targetId);
|
|
388
|
+
}
|
|
389
|
+
_addSessionListeners() {
|
|
390
|
+
this._sessionListeners = [_eventsHelper.eventsHelper.addEventListener(this._session, 'Page.frameNavigated', event => this._onFrameNavigated(event.frame, false)), _eventsHelper.eventsHelper.addEventListener(this._session, 'Page.navigatedWithinDocument', event => this._onFrameNavigatedWithinDocument(event.frameId, event.url)), _eventsHelper.eventsHelper.addEventListener(this._session, 'Page.frameAttached', event => this._onFrameAttached(event.frameId, event.parentFrameId)), _eventsHelper.eventsHelper.addEventListener(this._session, 'Page.frameDetached', event => this._onFrameDetached(event.frameId)), _eventsHelper.eventsHelper.addEventListener(this._session, 'Page.willCheckNavigationPolicy', event => this._onWillCheckNavigationPolicy(event.frameId)), _eventsHelper.eventsHelper.addEventListener(this._session, 'Page.didCheckNavigationPolicy', event => this._onDidCheckNavigationPolicy(event.frameId, event.cancel)), _eventsHelper.eventsHelper.addEventListener(this._session, 'Page.frameScheduledNavigation', event => this._onFrameScheduledNavigation(event.frameId, event.delay, event.targetIsCurrentFrame)), _eventsHelper.eventsHelper.addEventListener(this._session, 'Page.loadEventFired', event => this._page._frameManager.frameLifecycleEvent(event.frameId, 'load')), _eventsHelper.eventsHelper.addEventListener(this._session, 'Page.domContentEventFired', event => this._page._frameManager.frameLifecycleEvent(event.frameId, 'domcontentloaded')), _eventsHelper.eventsHelper.addEventListener(this._session, 'Runtime.executionContextCreated', event => this._onExecutionContextCreated(event.context)), _eventsHelper.eventsHelper.addEventListener(this._session, 'Runtime.bindingCalled', event => this._onBindingCalled(event.contextId, event.argument)), _eventsHelper.eventsHelper.addEventListener(this._session, 'Console.messageAdded', event => this._onConsoleMessage(event)), _eventsHelper.eventsHelper.addEventListener(this._session, 'Console.messageRepeatCountUpdated', event => this._onConsoleRepeatCountUpdated(event)), _eventsHelper.eventsHelper.addEventListener(this._pageProxySession, 'Dialog.javascriptDialogOpening', event => this._onDialog(event)), _eventsHelper.eventsHelper.addEventListener(this._session, 'Page.fileChooserOpened', event => this._onFileChooserOpened(event)), _eventsHelper.eventsHelper.addEventListener(this._session, 'Network.requestWillBeSent', e => this._onRequestWillBeSent(this._session, e)), _eventsHelper.eventsHelper.addEventListener(this._session, 'Network.requestIntercepted', e => this._onRequestIntercepted(this._session, e)), _eventsHelper.eventsHelper.addEventListener(this._session, 'Network.responseReceived', e => this._onResponseReceived(this._session, e)), _eventsHelper.eventsHelper.addEventListener(this._session, 'Network.loadingFinished', e => this._onLoadingFinished(e)), _eventsHelper.eventsHelper.addEventListener(this._session, 'Network.loadingFailed', e => this._onLoadingFailed(this._session, e)), _eventsHelper.eventsHelper.addEventListener(this._session, 'Network.webSocketCreated', e => this._page._frameManager.onWebSocketCreated(e.requestId, e.url)), _eventsHelper.eventsHelper.addEventListener(this._session, 'Network.webSocketWillSendHandshakeRequest', e => this._page._frameManager.onWebSocketRequest(e.requestId)), _eventsHelper.eventsHelper.addEventListener(this._session, 'Network.webSocketHandshakeResponseReceived', e => this._page._frameManager.onWebSocketResponse(e.requestId, e.response.status, e.response.statusText)), _eventsHelper.eventsHelper.addEventListener(this._session, 'Network.webSocketFrameSent', e => e.response.payloadData && this._page._frameManager.onWebSocketFrameSent(e.requestId, e.response.opcode, e.response.payloadData)), _eventsHelper.eventsHelper.addEventListener(this._session, 'Network.webSocketFrameReceived', e => e.response.payloadData && this._page._frameManager.webSocketFrameReceived(e.requestId, e.response.opcode, e.response.payloadData)), _eventsHelper.eventsHelper.addEventListener(this._session, 'Network.webSocketClosed', e => this._page._frameManager.webSocketClosed(e.requestId)), _eventsHelper.eventsHelper.addEventListener(this._session, 'Network.webSocketFrameError', e => this._page._frameManager.webSocketError(e.requestId, e.errorMessage))];
|
|
391
|
+
}
|
|
392
|
+
async _updateState(method, params) {
|
|
393
|
+
await this._forAllSessions(session => session.send(method, params).then());
|
|
394
|
+
}
|
|
395
|
+
async _forAllSessions(callback) {
|
|
396
|
+
const sessions = [this._session];
|
|
397
|
+
// If the state changes during provisional load, push it to the provisional page
|
|
398
|
+
// as well to always be in sync with the backend.
|
|
399
|
+
if (this._provisionalPage) sessions.push(this._provisionalPage._session);
|
|
400
|
+
await Promise.all(sessions.map(session => callback(session).catch(e => {})));
|
|
401
|
+
}
|
|
402
|
+
_onWillCheckNavigationPolicy(frameId) {
|
|
403
|
+
// It may happen that new policy check occurs while there is an ongoing
|
|
404
|
+
// provisional load, in this case it should be safe to ignore it as it will
|
|
405
|
+
// either:
|
|
406
|
+
// - end up canceled, e.g. ctrl+click opening link in new tab, having no effect
|
|
407
|
+
// on this page
|
|
408
|
+
// - start new provisional load which we will miss in our signal trackers but
|
|
409
|
+
// we certainly won't hang waiting for it to finish and there is high chance
|
|
410
|
+
// that the current provisional page will commit navigation canceling the new
|
|
411
|
+
// one.
|
|
412
|
+
if (this._provisionalPage) return;
|
|
413
|
+
this._page._frameManager.frameRequestedNavigation(frameId);
|
|
414
|
+
}
|
|
415
|
+
_onDidCheckNavigationPolicy(frameId, cancel) {
|
|
416
|
+
if (!cancel) return;
|
|
417
|
+
// This is a cross-process navigation that is canceled in the original page and continues in
|
|
418
|
+
// the provisional page. Bail out as we are tracking it.
|
|
419
|
+
if (this._provisionalPage) return;
|
|
420
|
+
this._page._frameManager.frameAbortedNavigation(frameId, 'Navigation canceled by policy check');
|
|
421
|
+
}
|
|
422
|
+
_onFrameScheduledNavigation(frameId, delay, targetIsCurrentFrame) {
|
|
423
|
+
if (targetIsCurrentFrame) this._page._frameManager.frameRequestedNavigation(frameId);
|
|
424
|
+
}
|
|
425
|
+
_handleFrameTree(frameTree) {
|
|
426
|
+
this._onFrameAttached(frameTree.frame.id, frameTree.frame.parentId || null);
|
|
427
|
+
this._onFrameNavigated(frameTree.frame, true);
|
|
428
|
+
this._page._frameManager.frameLifecycleEvent(frameTree.frame.id, 'domcontentloaded');
|
|
429
|
+
this._page._frameManager.frameLifecycleEvent(frameTree.frame.id, 'load');
|
|
430
|
+
if (!frameTree.childFrames) return;
|
|
431
|
+
for (const child of frameTree.childFrames) this._handleFrameTree(child);
|
|
432
|
+
}
|
|
433
|
+
_onFrameAttached(frameId, parentFrameId) {
|
|
434
|
+
return this._page._frameManager.frameAttached(frameId, parentFrameId);
|
|
435
|
+
}
|
|
436
|
+
_onFrameNavigated(framePayload, initial) {
|
|
437
|
+
const frame = this._page._frameManager.frame(framePayload.id);
|
|
438
|
+
(0, _utils.assert)(frame);
|
|
439
|
+
this._removeContextsForFrame(frame, true);
|
|
440
|
+
if (!framePayload.parentId) this._workers.clear();
|
|
441
|
+
this._page._frameManager.frameCommittedNewDocumentNavigation(framePayload.id, framePayload.url, framePayload.name || '', framePayload.loaderId, initial);
|
|
442
|
+
if (!initial) this._firstNonInitialNavigationCommittedFulfill();
|
|
443
|
+
}
|
|
444
|
+
_onFrameNavigatedWithinDocument(frameId, url) {
|
|
445
|
+
this._page._frameManager.frameCommittedSameDocumentNavigation(frameId, url);
|
|
446
|
+
}
|
|
447
|
+
_onFrameDetached(frameId) {
|
|
448
|
+
this._page._frameManager.frameDetached(frameId);
|
|
449
|
+
}
|
|
450
|
+
_removeContextsForFrame(frame, notifyFrame) {
|
|
451
|
+
for (const [contextId, context] of this._contextIdToContext) {
|
|
452
|
+
if (context.frame === frame) {
|
|
453
|
+
this._contextIdToContext.delete(contextId);
|
|
454
|
+
if (notifyFrame) frame._contextDestroyed(context);
|
|
455
|
+
}
|
|
456
|
+
}
|
|
457
|
+
}
|
|
458
|
+
_onExecutionContextCreated(contextPayload) {
|
|
459
|
+
if (this._contextIdToContext.has(contextPayload.id)) return;
|
|
460
|
+
const frame = this._page._frameManager.frame(contextPayload.frameId);
|
|
461
|
+
if (!frame) return;
|
|
462
|
+
const delegate = new _wkExecutionContext.WKExecutionContext(this._session, contextPayload.id);
|
|
463
|
+
let worldName = null;
|
|
464
|
+
if (contextPayload.type === 'normal') worldName = 'main';else if (contextPayload.type === 'user' && contextPayload.name === UTILITY_WORLD_NAME) worldName = 'utility';
|
|
465
|
+
const context = new dom.FrameExecutionContext(delegate, frame, worldName);
|
|
466
|
+
context[contextDelegateSymbol] = delegate;
|
|
467
|
+
if (worldName) frame._contextCreated(worldName, context);
|
|
468
|
+
this._contextIdToContext.set(contextPayload.id, context);
|
|
469
|
+
}
|
|
470
|
+
async _onBindingCalled(contextId, argument) {
|
|
471
|
+
const pageOrError = await this.pageOrError();
|
|
472
|
+
if (!(pageOrError instanceof Error)) {
|
|
473
|
+
const context = this._contextIdToContext.get(contextId);
|
|
474
|
+
if (context) await this._page._onBindingCalled(argument, context);
|
|
475
|
+
}
|
|
476
|
+
}
|
|
477
|
+
async navigateFrame(frame, url, referrer) {
|
|
478
|
+
if (this._pageProxySession.isDisposed()) throw new _errors.TargetClosedError();
|
|
479
|
+
const pageProxyId = this._pageProxySession.sessionId;
|
|
480
|
+
const result = await this._pageProxySession.connection.browserSession.send('Playwright.navigate', {
|
|
481
|
+
url,
|
|
482
|
+
pageProxyId,
|
|
483
|
+
frameId: frame._id,
|
|
484
|
+
referrer
|
|
485
|
+
});
|
|
486
|
+
return {
|
|
487
|
+
newDocumentId: result.loaderId
|
|
488
|
+
};
|
|
489
|
+
}
|
|
490
|
+
_onConsoleMessage(event) {
|
|
491
|
+
// Note: do no introduce await in this function, otherwise we lose the ordering.
|
|
492
|
+
// For example, frame.setContent relies on this.
|
|
493
|
+
const {
|
|
494
|
+
type,
|
|
495
|
+
level,
|
|
496
|
+
text,
|
|
497
|
+
parameters,
|
|
498
|
+
url,
|
|
499
|
+
line: lineNumber,
|
|
500
|
+
column: columnNumber,
|
|
501
|
+
source
|
|
502
|
+
} = event.message;
|
|
503
|
+
if (level === 'error' && source === 'javascript') {
|
|
504
|
+
const {
|
|
505
|
+
name,
|
|
506
|
+
message
|
|
507
|
+
} = (0, _stackTrace.splitErrorMessage)(text);
|
|
508
|
+
let stack;
|
|
509
|
+
if (event.message.stackTrace) {
|
|
510
|
+
stack = text + '\n' + event.message.stackTrace.callFrames.map(callFrame => {
|
|
511
|
+
return ` at ${callFrame.functionName || 'unknown'} (${callFrame.url}:${callFrame.lineNumber}:${callFrame.columnNumber})`;
|
|
512
|
+
}).join('\n');
|
|
513
|
+
} else {
|
|
514
|
+
stack = '';
|
|
515
|
+
}
|
|
516
|
+
this._lastConsoleMessage = null;
|
|
517
|
+
const error = new Error(message);
|
|
518
|
+
error.stack = stack;
|
|
519
|
+
error.name = name;
|
|
520
|
+
this._page.emitOnContextOnceInitialized(_browserContext.BrowserContext.Events.PageError, error, this._page);
|
|
521
|
+
return;
|
|
522
|
+
}
|
|
523
|
+
let derivedType = type || '';
|
|
524
|
+
if (type === 'log') derivedType = level;else if (type === 'timing') derivedType = 'timeEnd';
|
|
525
|
+
const handles = [];
|
|
526
|
+
for (const p of parameters || []) {
|
|
527
|
+
let context;
|
|
528
|
+
if (p.objectId) {
|
|
529
|
+
const objectId = JSON.parse(p.objectId);
|
|
530
|
+
context = this._contextIdToContext.get(objectId.injectedScriptId);
|
|
531
|
+
} else {
|
|
532
|
+
// Pick any context if the parameter is a value.
|
|
533
|
+
context = [...this._contextIdToContext.values()].find(c => c.frame === this._page.mainFrame());
|
|
534
|
+
}
|
|
535
|
+
if (!context) return;
|
|
536
|
+
handles.push(context.createHandle(p));
|
|
537
|
+
}
|
|
538
|
+
this._lastConsoleMessage = {
|
|
539
|
+
derivedType,
|
|
540
|
+
text,
|
|
541
|
+
handles,
|
|
542
|
+
count: 0,
|
|
543
|
+
location: {
|
|
544
|
+
url: url || '',
|
|
545
|
+
lineNumber: (lineNumber || 1) - 1,
|
|
546
|
+
columnNumber: (columnNumber || 1) - 1
|
|
547
|
+
}
|
|
548
|
+
};
|
|
549
|
+
this._onConsoleRepeatCountUpdated({
|
|
550
|
+
count: 1
|
|
551
|
+
});
|
|
552
|
+
}
|
|
553
|
+
_onConsoleRepeatCountUpdated(event) {
|
|
554
|
+
if (this._lastConsoleMessage) {
|
|
555
|
+
const {
|
|
556
|
+
derivedType,
|
|
557
|
+
text,
|
|
558
|
+
handles,
|
|
559
|
+
count,
|
|
560
|
+
location
|
|
561
|
+
} = this._lastConsoleMessage;
|
|
562
|
+
for (let i = count; i < event.count; ++i) this._page._addConsoleMessage(derivedType, handles, location, handles.length ? undefined : text);
|
|
563
|
+
this._lastConsoleMessage.count = event.count;
|
|
564
|
+
}
|
|
565
|
+
}
|
|
566
|
+
_onDialog(event) {
|
|
567
|
+
this._page.emitOnContext(_browserContext.BrowserContext.Events.Dialog, new dialog.Dialog(this._page, event.type, event.message, async (accept, promptText) => {
|
|
568
|
+
await this._pageProxySession.send('Dialog.handleJavaScriptDialog', {
|
|
569
|
+
accept,
|
|
570
|
+
promptText
|
|
571
|
+
});
|
|
572
|
+
}, event.defaultPrompt));
|
|
573
|
+
}
|
|
574
|
+
async _onFileChooserOpened(event) {
|
|
575
|
+
let handle;
|
|
576
|
+
try {
|
|
577
|
+
const context = await this._page._frameManager.frame(event.frameId)._mainContext();
|
|
578
|
+
handle = context.createHandle(event.element).asElement();
|
|
579
|
+
} catch (e) {
|
|
580
|
+
// During async processing, frame/context may go away. We should not throw.
|
|
581
|
+
return;
|
|
582
|
+
}
|
|
583
|
+
await this._page._onFileChooserOpened(handle);
|
|
584
|
+
}
|
|
585
|
+
static async _setEmulateMedia(session, mediaType, colorScheme, reducedMotion, forcedColors) {
|
|
586
|
+
const promises = [];
|
|
587
|
+
promises.push(session.send('Page.setEmulatedMedia', {
|
|
588
|
+
media: mediaType === 'no-override' ? '' : mediaType
|
|
589
|
+
}));
|
|
590
|
+
let appearance = undefined;
|
|
591
|
+
switch (colorScheme) {
|
|
592
|
+
case 'light':
|
|
593
|
+
appearance = 'Light';
|
|
594
|
+
break;
|
|
595
|
+
case 'dark':
|
|
596
|
+
appearance = 'Dark';
|
|
597
|
+
break;
|
|
598
|
+
case 'no-override':
|
|
599
|
+
appearance = undefined;
|
|
600
|
+
break;
|
|
601
|
+
}
|
|
602
|
+
promises.push(session.send('Page.overrideUserPreference', {
|
|
603
|
+
name: 'PrefersColorScheme',
|
|
604
|
+
value: appearance
|
|
605
|
+
}));
|
|
606
|
+
let reducedMotionWk = undefined;
|
|
607
|
+
switch (reducedMotion) {
|
|
608
|
+
case 'reduce':
|
|
609
|
+
reducedMotionWk = 'Reduce';
|
|
610
|
+
break;
|
|
611
|
+
case 'no-preference':
|
|
612
|
+
reducedMotionWk = 'NoPreference';
|
|
613
|
+
break;
|
|
614
|
+
case 'no-override':
|
|
615
|
+
reducedMotionWk = undefined;
|
|
616
|
+
break;
|
|
617
|
+
}
|
|
618
|
+
promises.push(session.send('Page.overrideUserPreference', {
|
|
619
|
+
name: 'PrefersReducedMotion',
|
|
620
|
+
value: reducedMotionWk
|
|
621
|
+
}));
|
|
622
|
+
let forcedColorsWk = undefined;
|
|
623
|
+
switch (forcedColors) {
|
|
624
|
+
case 'active':
|
|
625
|
+
forcedColorsWk = 'Active';
|
|
626
|
+
break;
|
|
627
|
+
case 'none':
|
|
628
|
+
forcedColorsWk = 'None';
|
|
629
|
+
break;
|
|
630
|
+
case 'no-override':
|
|
631
|
+
forcedColorsWk = undefined;
|
|
632
|
+
break;
|
|
633
|
+
}
|
|
634
|
+
promises.push(session.send('Page.setForcedColors', {
|
|
635
|
+
forcedColors: forcedColorsWk
|
|
636
|
+
}));
|
|
637
|
+
await Promise.all(promises);
|
|
638
|
+
}
|
|
639
|
+
async updateExtraHTTPHeaders() {
|
|
640
|
+
await this._updateState('Network.setExtraHTTPHeaders', {
|
|
641
|
+
headers: (0, _utils.headersArrayToObject)(this._calculateExtraHTTPHeaders(), false /* lowerCase */)
|
|
642
|
+
});
|
|
643
|
+
}
|
|
644
|
+
_calculateExtraHTTPHeaders() {
|
|
645
|
+
const locale = this._browserContext._options.locale;
|
|
646
|
+
const headers = network.mergeHeaders([this._browserContext._options.extraHTTPHeaders, this._page.extraHTTPHeaders(), locale ? network.singleHeader('Accept-Language', locale) : undefined]);
|
|
647
|
+
return headers;
|
|
648
|
+
}
|
|
649
|
+
async updateEmulateMedia() {
|
|
650
|
+
const emulatedMedia = this._page.emulatedMedia();
|
|
651
|
+
const colorScheme = emulatedMedia.colorScheme;
|
|
652
|
+
const reducedMotion = emulatedMedia.reducedMotion;
|
|
653
|
+
const forcedColors = emulatedMedia.forcedColors;
|
|
654
|
+
await this._forAllSessions(session => WKPage._setEmulateMedia(session, emulatedMedia.media, colorScheme, reducedMotion, forcedColors));
|
|
655
|
+
}
|
|
656
|
+
async updateEmulatedViewportSize() {
|
|
657
|
+
this._browserContext._validateEmulatedViewport(this._page.viewportSize());
|
|
658
|
+
await this._updateViewport();
|
|
659
|
+
}
|
|
660
|
+
async updateUserAgent() {
|
|
661
|
+
const contextOptions = this._browserContext._options;
|
|
662
|
+
this._updateState('Page.overrideUserAgent', {
|
|
663
|
+
value: contextOptions.userAgent
|
|
664
|
+
});
|
|
665
|
+
}
|
|
666
|
+
async bringToFront() {
|
|
667
|
+
this._pageProxySession.send('Target.activate', {
|
|
668
|
+
targetId: this._session.sessionId
|
|
669
|
+
});
|
|
670
|
+
}
|
|
671
|
+
async _updateViewport() {
|
|
672
|
+
const options = this._browserContext._options;
|
|
673
|
+
const deviceSize = this._page.emulatedSize();
|
|
674
|
+
if (deviceSize === null) return;
|
|
675
|
+
const viewportSize = deviceSize.viewport;
|
|
676
|
+
const screenSize = deviceSize.screen;
|
|
677
|
+
const promises = [this._pageProxySession.send('Emulation.setDeviceMetricsOverride', {
|
|
678
|
+
width: viewportSize.width,
|
|
679
|
+
height: viewportSize.height,
|
|
680
|
+
fixedLayout: !!options.isMobile,
|
|
681
|
+
deviceScaleFactor: options.deviceScaleFactor || 1
|
|
682
|
+
}), this._session.send('Page.setScreenSizeOverride', {
|
|
683
|
+
width: screenSize.width,
|
|
684
|
+
height: screenSize.height
|
|
685
|
+
})];
|
|
686
|
+
if (options.isMobile) {
|
|
687
|
+
const angle = viewportSize.width > viewportSize.height ? 90 : 0;
|
|
688
|
+
promises.push(this._session.send('Page.setOrientationOverride', {
|
|
689
|
+
angle
|
|
690
|
+
}));
|
|
691
|
+
}
|
|
692
|
+
await Promise.all(promises);
|
|
693
|
+
}
|
|
694
|
+
async updateRequestInterception() {
|
|
695
|
+
const enabled = this._page.needsRequestInterception();
|
|
696
|
+
await Promise.all([this._updateState('Network.setInterceptionEnabled', {
|
|
697
|
+
enabled
|
|
698
|
+
}), this._updateState('Network.setResourceCachingDisabled', {
|
|
699
|
+
disabled: enabled
|
|
700
|
+
}), this._updateState('Network.addInterception', {
|
|
701
|
+
url: '.*',
|
|
702
|
+
stage: 'request',
|
|
703
|
+
isRegex: true
|
|
704
|
+
})]);
|
|
705
|
+
}
|
|
706
|
+
async updateOffline() {
|
|
707
|
+
await this._updateState('Network.setEmulateOfflineState', {
|
|
708
|
+
offline: !!this._browserContext._options.offline
|
|
709
|
+
});
|
|
710
|
+
}
|
|
711
|
+
async updateHttpCredentials() {
|
|
712
|
+
const credentials = this._browserContext._options.httpCredentials || {
|
|
713
|
+
username: '',
|
|
714
|
+
password: '',
|
|
715
|
+
origin: ''
|
|
716
|
+
};
|
|
717
|
+
await this._pageProxySession.send('Emulation.setAuthCredentials', {
|
|
718
|
+
username: credentials.username,
|
|
719
|
+
password: credentials.password,
|
|
720
|
+
origin: credentials.origin
|
|
721
|
+
});
|
|
722
|
+
}
|
|
723
|
+
async updateFileChooserInterception() {
|
|
724
|
+
const enabled = this._page.fileChooserIntercepted();
|
|
725
|
+
await this._session.send('Page.setInterceptFileChooserDialog', {
|
|
726
|
+
enabled
|
|
727
|
+
}).catch(() => {}); // target can be closed.
|
|
728
|
+
}
|
|
729
|
+
async reload() {
|
|
730
|
+
await this._session.send('Page.reload');
|
|
731
|
+
}
|
|
732
|
+
goBack() {
|
|
733
|
+
return this._session.send('Page.goBack').then(() => true).catch(error => {
|
|
734
|
+
if (error instanceof Error && error.message.includes(`Protocol error (Page.goBack): Failed to go`)) return false;
|
|
735
|
+
throw error;
|
|
736
|
+
});
|
|
737
|
+
}
|
|
738
|
+
goForward() {
|
|
739
|
+
return this._session.send('Page.goForward').then(() => true).catch(error => {
|
|
740
|
+
if (error instanceof Error && error.message.includes(`Protocol error (Page.goForward): Failed to go`)) return false;
|
|
741
|
+
throw error;
|
|
742
|
+
});
|
|
743
|
+
}
|
|
744
|
+
async exposeBinding(binding) {
|
|
745
|
+
this._session.send('Runtime.addBinding', {
|
|
746
|
+
name: binding.name
|
|
747
|
+
});
|
|
748
|
+
await this._updateBootstrapScript();
|
|
749
|
+
await Promise.all(this._page.frames().map(frame => frame.evaluateExpression(binding.source).catch(e => {})));
|
|
750
|
+
}
|
|
751
|
+
async removeExposedBindings() {
|
|
752
|
+
await this._updateBootstrapScript();
|
|
753
|
+
}
|
|
754
|
+
async addInitScript(script) {
|
|
755
|
+
await this._updateBootstrapScript();
|
|
756
|
+
}
|
|
757
|
+
async removeInitScripts() {
|
|
758
|
+
await this._updateBootstrapScript();
|
|
759
|
+
}
|
|
760
|
+
_calculateBootstrapScript() {
|
|
761
|
+
const scripts = [];
|
|
762
|
+
if (!this._page.context()._options.isMobile) {
|
|
763
|
+
scripts.push('delete window.orientation');
|
|
764
|
+
scripts.push('delete window.ondevicemotion');
|
|
765
|
+
scripts.push('delete window.ondeviceorientation');
|
|
766
|
+
}
|
|
767
|
+
scripts.push('if (!window.safari) window.safari = { pushNotification: { toString() { return "[object SafariRemoteNotification]"; } } };');
|
|
768
|
+
scripts.push('if (!window.GestureEvent) window.GestureEvent = function GestureEvent() {};');
|
|
769
|
+
for (const binding of this._page.allBindings()) scripts.push(binding.source);
|
|
770
|
+
scripts.push(...this._browserContext.initScripts);
|
|
771
|
+
scripts.push(...this._page.initScripts);
|
|
772
|
+
return scripts.join(';\n');
|
|
773
|
+
}
|
|
774
|
+
async _updateBootstrapScript() {
|
|
775
|
+
await this._updateState('Page.setBootstrapScript', {
|
|
776
|
+
source: this._calculateBootstrapScript()
|
|
777
|
+
});
|
|
778
|
+
}
|
|
779
|
+
async closePage(runBeforeUnload) {
|
|
780
|
+
await this._stopVideo();
|
|
781
|
+
await this._pageProxySession.sendMayFail('Target.close', {
|
|
782
|
+
targetId: this._session.sessionId,
|
|
783
|
+
runBeforeUnload
|
|
784
|
+
});
|
|
785
|
+
}
|
|
786
|
+
async setBackgroundColor(color) {
|
|
787
|
+
await this._session.send('Page.setDefaultBackgroundColorOverride', {
|
|
788
|
+
color
|
|
789
|
+
});
|
|
790
|
+
}
|
|
791
|
+
_toolbarHeight() {
|
|
792
|
+
var _this$_page$_browserC;
|
|
793
|
+
if ((_this$_page$_browserC = this._page._browserContext._browser) !== null && _this$_page$_browserC !== void 0 && _this$_page$_browserC.options.headful) return _hostPlatform.hostPlatform === 'mac10.15' ? 55 : 59;
|
|
794
|
+
return 0;
|
|
795
|
+
}
|
|
796
|
+
async _startVideo(options) {
|
|
797
|
+
(0, _utils.assert)(!this._recordingVideoFile);
|
|
798
|
+
const {
|
|
799
|
+
screencastId
|
|
800
|
+
} = await this._pageProxySession.send('Screencast.startVideo', {
|
|
801
|
+
file: options.outputFile,
|
|
802
|
+
width: options.width,
|
|
803
|
+
height: options.height,
|
|
804
|
+
toolbarHeight: this._toolbarHeight()
|
|
805
|
+
});
|
|
806
|
+
this._recordingVideoFile = options.outputFile;
|
|
807
|
+
this._browserContext._browser._videoStarted(this._browserContext, screencastId, options.outputFile, this.pageOrError());
|
|
808
|
+
}
|
|
809
|
+
async _stopVideo() {
|
|
810
|
+
if (!this._recordingVideoFile) return;
|
|
811
|
+
await this._pageProxySession.sendMayFail('Screencast.stopVideo');
|
|
812
|
+
this._recordingVideoFile = null;
|
|
813
|
+
}
|
|
814
|
+
validateScreenshotDimension(side, omitDeviceScaleFactor) {
|
|
815
|
+
// Cairo based implementations (Linux and Windows) have hard limit of 32767
|
|
816
|
+
// (see https://github.com/microsoft/playwright/issues/16727).
|
|
817
|
+
if (process.platform === 'darwin') return;
|
|
818
|
+
if (!omitDeviceScaleFactor && this._page._browserContext._options.deviceScaleFactor) side = Math.ceil(side * this._page._browserContext._options.deviceScaleFactor);
|
|
819
|
+
if (side > 32767) throw new Error('Cannot take screenshot larger than 32767 pixels on any dimension');
|
|
820
|
+
}
|
|
821
|
+
async takeScreenshot(progress, format, documentRect, viewportRect, quality, fitsViewport, scale) {
|
|
822
|
+
const rect = documentRect || viewportRect;
|
|
823
|
+
const omitDeviceScaleFactor = scale === 'css';
|
|
824
|
+
this.validateScreenshotDimension(rect.width, omitDeviceScaleFactor);
|
|
825
|
+
this.validateScreenshotDimension(rect.height, omitDeviceScaleFactor);
|
|
826
|
+
const result = await this._session.send('Page.snapshotRect', {
|
|
827
|
+
...rect,
|
|
828
|
+
coordinateSystem: documentRect ? 'Page' : 'Viewport',
|
|
829
|
+
omitDeviceScaleFactor
|
|
830
|
+
});
|
|
831
|
+
const prefix = 'data:image/png;base64,';
|
|
832
|
+
let buffer = Buffer.from(result.dataURL.substr(prefix.length), 'base64');
|
|
833
|
+
if (format === 'jpeg') buffer = _utilsBundle.jpegjs.encode(_utilsBundle.PNG.sync.read(buffer), quality).data;
|
|
834
|
+
return buffer;
|
|
835
|
+
}
|
|
836
|
+
async getContentFrame(handle) {
|
|
837
|
+
const nodeInfo = await this._session.send('DOM.describeNode', {
|
|
838
|
+
objectId: handle._objectId
|
|
839
|
+
});
|
|
840
|
+
if (!nodeInfo.contentFrameId) return null;
|
|
841
|
+
return this._page._frameManager.frame(nodeInfo.contentFrameId);
|
|
842
|
+
}
|
|
843
|
+
async getOwnerFrame(handle) {
|
|
844
|
+
if (!handle._objectId) return null;
|
|
845
|
+
const nodeInfo = await this._session.send('DOM.describeNode', {
|
|
846
|
+
objectId: handle._objectId
|
|
847
|
+
});
|
|
848
|
+
return nodeInfo.ownerFrameId || null;
|
|
849
|
+
}
|
|
850
|
+
isElementHandle(remoteObject) {
|
|
851
|
+
return remoteObject.subtype === 'node';
|
|
852
|
+
}
|
|
853
|
+
async getBoundingBox(handle) {
|
|
854
|
+
const quads = await this.getContentQuads(handle);
|
|
855
|
+
if (!quads || !quads.length) return null;
|
|
856
|
+
let minX = Infinity;
|
|
857
|
+
let maxX = -Infinity;
|
|
858
|
+
let minY = Infinity;
|
|
859
|
+
let maxY = -Infinity;
|
|
860
|
+
for (const quad of quads) {
|
|
861
|
+
for (const point of quad) {
|
|
862
|
+
minX = Math.min(minX, point.x);
|
|
863
|
+
maxX = Math.max(maxX, point.x);
|
|
864
|
+
minY = Math.min(minY, point.y);
|
|
865
|
+
maxY = Math.max(maxY, point.y);
|
|
866
|
+
}
|
|
867
|
+
}
|
|
868
|
+
return {
|
|
869
|
+
x: minX,
|
|
870
|
+
y: minY,
|
|
871
|
+
width: maxX - minX,
|
|
872
|
+
height: maxY - minY
|
|
873
|
+
};
|
|
874
|
+
}
|
|
875
|
+
async scrollRectIntoViewIfNeeded(handle, rect) {
|
|
876
|
+
return await this._session.send('DOM.scrollIntoViewIfNeeded', {
|
|
877
|
+
objectId: handle._objectId,
|
|
878
|
+
rect
|
|
879
|
+
}).then(() => 'done').catch(e => {
|
|
880
|
+
if (e instanceof Error && e.message.includes('Node does not have a layout object')) return 'error:notvisible';
|
|
881
|
+
if (e instanceof Error && e.message.includes('Node is detached from document')) return 'error:notconnected';
|
|
882
|
+
throw e;
|
|
883
|
+
});
|
|
884
|
+
}
|
|
885
|
+
async setScreencastOptions(options) {
|
|
886
|
+
if (options) {
|
|
887
|
+
const so = {
|
|
888
|
+
...options,
|
|
889
|
+
toolbarHeight: this._toolbarHeight()
|
|
890
|
+
};
|
|
891
|
+
const {
|
|
892
|
+
generation
|
|
893
|
+
} = await this._pageProxySession.send('Screencast.startScreencast', so);
|
|
894
|
+
this._screencastGeneration = generation;
|
|
895
|
+
} else {
|
|
896
|
+
await this._pageProxySession.send('Screencast.stopScreencast');
|
|
897
|
+
}
|
|
898
|
+
}
|
|
899
|
+
_onScreencastFrame(event) {
|
|
900
|
+
const generation = this._screencastGeneration;
|
|
901
|
+
this._page.throttleScreencastFrameAck(() => {
|
|
902
|
+
this._pageProxySession.send('Screencast.screencastFrameAck', {
|
|
903
|
+
generation
|
|
904
|
+
}).catch(e => _debugLogger.debugLogger.log('error', e));
|
|
905
|
+
});
|
|
906
|
+
const buffer = Buffer.from(event.data, 'base64');
|
|
907
|
+
this._page.emit(_page.Page.Events.ScreencastFrame, {
|
|
908
|
+
buffer,
|
|
909
|
+
width: event.deviceWidth,
|
|
910
|
+
height: event.deviceHeight
|
|
911
|
+
});
|
|
912
|
+
}
|
|
913
|
+
rafCountForStablePosition() {
|
|
914
|
+
return process.platform === 'win32' ? 5 : 1;
|
|
915
|
+
}
|
|
916
|
+
async getContentQuads(handle) {
|
|
917
|
+
const result = await this._session.sendMayFail('DOM.getContentQuads', {
|
|
918
|
+
objectId: handle._objectId
|
|
919
|
+
});
|
|
920
|
+
if (!result) return null;
|
|
921
|
+
return result.quads.map(quad => [{
|
|
922
|
+
x: quad[0],
|
|
923
|
+
y: quad[1]
|
|
924
|
+
}, {
|
|
925
|
+
x: quad[2],
|
|
926
|
+
y: quad[3]
|
|
927
|
+
}, {
|
|
928
|
+
x: quad[4],
|
|
929
|
+
y: quad[5]
|
|
930
|
+
}, {
|
|
931
|
+
x: quad[6],
|
|
932
|
+
y: quad[7]
|
|
933
|
+
}]);
|
|
934
|
+
}
|
|
935
|
+
async setInputFiles(handle, files) {
|
|
936
|
+
const objectId = handle._objectId;
|
|
937
|
+
const protocolFiles = files.map(file => ({
|
|
938
|
+
name: file.name,
|
|
939
|
+
type: file.mimeType,
|
|
940
|
+
data: file.buffer
|
|
941
|
+
}));
|
|
942
|
+
await this._session.send('DOM.setInputFiles', {
|
|
943
|
+
objectId,
|
|
944
|
+
files: protocolFiles
|
|
945
|
+
});
|
|
946
|
+
}
|
|
947
|
+
async setInputFilePaths(progress, handle, paths) {
|
|
948
|
+
const pageProxyId = this._pageProxySession.sessionId;
|
|
949
|
+
const objectId = handle._objectId;
|
|
950
|
+
await Promise.all([this._pageProxySession.connection.browserSession.send('Playwright.grantFileReadAccess', {
|
|
951
|
+
pageProxyId,
|
|
952
|
+
paths
|
|
953
|
+
}), this._session.send('DOM.setInputFiles', {
|
|
954
|
+
objectId,
|
|
955
|
+
paths
|
|
956
|
+
})]);
|
|
957
|
+
}
|
|
958
|
+
async adoptElementHandle(handle, to) {
|
|
959
|
+
const result = await this._session.sendMayFail('DOM.resolveNode', {
|
|
960
|
+
objectId: handle._objectId,
|
|
961
|
+
executionContextId: to[contextDelegateSymbol]._contextId
|
|
962
|
+
});
|
|
963
|
+
if (!result || result.object.subtype === 'null') throw new Error(dom.kUnableToAdoptErrorMessage);
|
|
964
|
+
return to.createHandle(result.object);
|
|
965
|
+
}
|
|
966
|
+
async getAccessibilityTree(needle) {
|
|
967
|
+
return (0, _wkAccessibility.getAccessibilityTree)(this._session, needle);
|
|
968
|
+
}
|
|
969
|
+
async inputActionEpilogue() {}
|
|
970
|
+
async resetForReuse() {}
|
|
971
|
+
async getFrameElement(frame) {
|
|
972
|
+
const parent = frame.parentFrame();
|
|
973
|
+
if (!parent) throw new Error('Frame has been detached.');
|
|
974
|
+
const context = await parent._mainContext();
|
|
975
|
+
const result = await this._session.send('DOM.resolveNode', {
|
|
976
|
+
frameId: frame._id,
|
|
977
|
+
executionContextId: context[contextDelegateSymbol]._contextId
|
|
978
|
+
});
|
|
979
|
+
if (!result || result.object.subtype === 'null') throw new Error('Frame has been detached.');
|
|
980
|
+
return context.createHandle(result.object);
|
|
981
|
+
}
|
|
982
|
+
_onRequestWillBeSent(session, event) {
|
|
983
|
+
if (event.request.url.startsWith('data:')) return;
|
|
984
|
+
|
|
985
|
+
// We do not support intercepting redirects.
|
|
986
|
+
if (this._page.needsRequestInterception() && !event.redirectResponse) this._requestIdToRequestWillBeSentEvent.set(event.requestId, event);else this._onRequest(session, event, false);
|
|
987
|
+
}
|
|
988
|
+
_onRequest(session, event, intercepted) {
|
|
989
|
+
let redirectedFrom = null;
|
|
990
|
+
if (event.redirectResponse) {
|
|
991
|
+
const request = this._requestIdToRequest.get(event.requestId);
|
|
992
|
+
// If we connect late to the target, we could have missed the requestWillBeSent event.
|
|
993
|
+
if (request) {
|
|
994
|
+
this._handleRequestRedirect(request, event.redirectResponse, event.timestamp);
|
|
995
|
+
redirectedFrom = request;
|
|
996
|
+
}
|
|
997
|
+
}
|
|
998
|
+
const frame = redirectedFrom ? redirectedFrom.request.frame() : this._page._frameManager.frame(event.frameId);
|
|
999
|
+
// sometimes we get stray network events for detached frames
|
|
1000
|
+
// TODO(einbinder) why?
|
|
1001
|
+
if (!frame) return;
|
|
1002
|
+
|
|
1003
|
+
// TODO(einbinder) this will fail if we are an XHR document request
|
|
1004
|
+
const isNavigationRequest = event.type === 'Document';
|
|
1005
|
+
const documentId = isNavigationRequest ? event.loaderId : undefined;
|
|
1006
|
+
const request = new _wkInterceptableRequest.WKInterceptableRequest(session, frame, event, redirectedFrom, documentId);
|
|
1007
|
+
let route;
|
|
1008
|
+
if (intercepted) {
|
|
1009
|
+
route = new _wkInterceptableRequest.WKRouteImpl(session, request._requestId);
|
|
1010
|
+
// There is no point in waiting for the raw headers in Network.responseReceived when intercepting.
|
|
1011
|
+
// Use provisional headers as raw headers, so that client can call allHeaders() from the route handler.
|
|
1012
|
+
request.request.setRawRequestHeaders(null);
|
|
1013
|
+
}
|
|
1014
|
+
this._requestIdToRequest.set(event.requestId, request);
|
|
1015
|
+
this._page._frameManager.requestStarted(request.request, route);
|
|
1016
|
+
}
|
|
1017
|
+
_handleRequestRedirect(request, responsePayload, timestamp) {
|
|
1018
|
+
const response = request.createResponse(responsePayload);
|
|
1019
|
+
response._securityDetailsFinished();
|
|
1020
|
+
response._serverAddrFinished();
|
|
1021
|
+
response.setResponseHeadersSize(null);
|
|
1022
|
+
response.setEncodedBodySize(null);
|
|
1023
|
+
response._requestFinished(responsePayload.timing ? _helper.helper.secondsToRoundishMillis(timestamp - request._timestamp) : -1);
|
|
1024
|
+
this._requestIdToRequest.delete(request._requestId);
|
|
1025
|
+
this._page._frameManager.requestReceivedResponse(response);
|
|
1026
|
+
this._page._frameManager.reportRequestFinished(request.request, response);
|
|
1027
|
+
}
|
|
1028
|
+
_onRequestIntercepted(session, event) {
|
|
1029
|
+
const requestWillBeSentEvent = this._requestIdToRequestWillBeSentEvent.get(event.requestId);
|
|
1030
|
+
if (!requestWillBeSentEvent) {
|
|
1031
|
+
// Intercepted, although we do not intend to allow interception.
|
|
1032
|
+
// Just continue.
|
|
1033
|
+
session.sendMayFail('Network.interceptWithRequest', {
|
|
1034
|
+
requestId: event.requestId
|
|
1035
|
+
});
|
|
1036
|
+
return;
|
|
1037
|
+
}
|
|
1038
|
+
this._requestIdToRequestWillBeSentEvent.delete(event.requestId);
|
|
1039
|
+
this._onRequest(session, requestWillBeSentEvent, true);
|
|
1040
|
+
}
|
|
1041
|
+
_onResponseReceived(session, event) {
|
|
1042
|
+
const requestWillBeSentEvent = this._requestIdToRequestWillBeSentEvent.get(event.requestId);
|
|
1043
|
+
if (requestWillBeSentEvent) {
|
|
1044
|
+
this._requestIdToRequestWillBeSentEvent.delete(event.requestId);
|
|
1045
|
+
// We received a response, so the request won't be intercepted (e.g. it was handled by a
|
|
1046
|
+
// service worker and we don't intercept service workers).
|
|
1047
|
+
this._onRequest(session, requestWillBeSentEvent, false);
|
|
1048
|
+
}
|
|
1049
|
+
const request = this._requestIdToRequest.get(event.requestId);
|
|
1050
|
+
// FileUpload sends a response without a matching request.
|
|
1051
|
+
if (!request) return;
|
|
1052
|
+
this._requestIdToResponseReceivedPayloadEvent.set(request._requestId, event);
|
|
1053
|
+
const response = request.createResponse(event.response);
|
|
1054
|
+
this._page._frameManager.requestReceivedResponse(response);
|
|
1055
|
+
if (response.status() === 204) {
|
|
1056
|
+
this._onLoadingFailed(session, {
|
|
1057
|
+
requestId: event.requestId,
|
|
1058
|
+
errorText: 'Aborted: 204 No Content',
|
|
1059
|
+
timestamp: event.timestamp
|
|
1060
|
+
});
|
|
1061
|
+
}
|
|
1062
|
+
}
|
|
1063
|
+
_onLoadingFinished(event) {
|
|
1064
|
+
const request = this._requestIdToRequest.get(event.requestId);
|
|
1065
|
+
// For certain requestIds we never receive requestWillBeSent event.
|
|
1066
|
+
// @see https://crbug.com/750469
|
|
1067
|
+
if (!request) return;
|
|
1068
|
+
|
|
1069
|
+
// Under certain conditions we never get the Network.responseReceived
|
|
1070
|
+
// event from protocol. @see https://crbug.com/883475
|
|
1071
|
+
const response = request.request._existingResponse();
|
|
1072
|
+
if (response) {
|
|
1073
|
+
var _event$metrics, _event$metrics2, _responseReceivedPayl, _responseReceivedPayl2, _responseReceivedPayl3, _event$metrics3, _event$metrics$respon, _event$metrics4, _event$metrics$respon2, _event$metrics5;
|
|
1074
|
+
const responseReceivedPayload = this._requestIdToResponseReceivedPayloadEvent.get(request._requestId);
|
|
1075
|
+
response._serverAddrFinished(parseRemoteAddress(event === null || event === void 0 || (_event$metrics = event.metrics) === null || _event$metrics === void 0 ? void 0 : _event$metrics.remoteAddress));
|
|
1076
|
+
response._securityDetailsFinished({
|
|
1077
|
+
protocol: isLoadedSecurely(response.url(), response.timing()) ? (_event$metrics2 = event.metrics) === null || _event$metrics2 === void 0 || (_event$metrics2 = _event$metrics2.securityConnection) === null || _event$metrics2 === void 0 ? void 0 : _event$metrics2.protocol : undefined,
|
|
1078
|
+
subjectName: responseReceivedPayload === null || responseReceivedPayload === void 0 || (_responseReceivedPayl = responseReceivedPayload.response.security) === null || _responseReceivedPayl === void 0 || (_responseReceivedPayl = _responseReceivedPayl.certificate) === null || _responseReceivedPayl === void 0 ? void 0 : _responseReceivedPayl.subject,
|
|
1079
|
+
validFrom: responseReceivedPayload === null || responseReceivedPayload === void 0 || (_responseReceivedPayl2 = responseReceivedPayload.response.security) === null || _responseReceivedPayl2 === void 0 || (_responseReceivedPayl2 = _responseReceivedPayl2.certificate) === null || _responseReceivedPayl2 === void 0 ? void 0 : _responseReceivedPayl2.validFrom,
|
|
1080
|
+
validTo: responseReceivedPayload === null || responseReceivedPayload === void 0 || (_responseReceivedPayl3 = responseReceivedPayload.response.security) === null || _responseReceivedPayl3 === void 0 || (_responseReceivedPayl3 = _responseReceivedPayl3.certificate) === null || _responseReceivedPayl3 === void 0 ? void 0 : _responseReceivedPayl3.validUntil
|
|
1081
|
+
});
|
|
1082
|
+
if ((_event$metrics3 = event.metrics) !== null && _event$metrics3 !== void 0 && _event$metrics3.protocol) response._setHttpVersion(event.metrics.protocol);
|
|
1083
|
+
response.setEncodedBodySize((_event$metrics$respon = (_event$metrics4 = event.metrics) === null || _event$metrics4 === void 0 ? void 0 : _event$metrics4.responseBodyBytesReceived) !== null && _event$metrics$respon !== void 0 ? _event$metrics$respon : null);
|
|
1084
|
+
response.setResponseHeadersSize((_event$metrics$respon2 = (_event$metrics5 = event.metrics) === null || _event$metrics5 === void 0 ? void 0 : _event$metrics5.responseHeaderBytesReceived) !== null && _event$metrics$respon2 !== void 0 ? _event$metrics$respon2 : null);
|
|
1085
|
+
response._requestFinished(_helper.helper.secondsToRoundishMillis(event.timestamp - request._timestamp));
|
|
1086
|
+
} else {
|
|
1087
|
+
// Use provisional headers if we didn't have the response with raw headers.
|
|
1088
|
+
request.request.setRawRequestHeaders(null);
|
|
1089
|
+
}
|
|
1090
|
+
this._requestIdToResponseReceivedPayloadEvent.delete(request._requestId);
|
|
1091
|
+
this._requestIdToRequest.delete(request._requestId);
|
|
1092
|
+
this._page._frameManager.reportRequestFinished(request.request, response);
|
|
1093
|
+
}
|
|
1094
|
+
_onLoadingFailed(session, event) {
|
|
1095
|
+
const requestWillBeSentEvent = this._requestIdToRequestWillBeSentEvent.get(event.requestId);
|
|
1096
|
+
if (requestWillBeSentEvent) {
|
|
1097
|
+
this._requestIdToRequestWillBeSentEvent.delete(event.requestId);
|
|
1098
|
+
// If loading failed, the request won't be intercepted (e.g. it was handled by a
|
|
1099
|
+
// service worker and we don't intercept service workers).
|
|
1100
|
+
this._onRequest(session, requestWillBeSentEvent, false);
|
|
1101
|
+
}
|
|
1102
|
+
const request = this._requestIdToRequest.get(event.requestId);
|
|
1103
|
+
// For certain requestIds we never receive requestWillBeSent event.
|
|
1104
|
+
// @see https://crbug.com/750469
|
|
1105
|
+
if (!request) return;
|
|
1106
|
+
const response = request.request._existingResponse();
|
|
1107
|
+
if (response) {
|
|
1108
|
+
response._serverAddrFinished();
|
|
1109
|
+
response._securityDetailsFinished();
|
|
1110
|
+
response.setResponseHeadersSize(null);
|
|
1111
|
+
response.setEncodedBodySize(null);
|
|
1112
|
+
response._requestFinished(_helper.helper.secondsToRoundishMillis(event.timestamp - request._timestamp));
|
|
1113
|
+
} else {
|
|
1114
|
+
// Use provisional headers if we didn't have the response with raw headers.
|
|
1115
|
+
request.request.setRawRequestHeaders(null);
|
|
1116
|
+
}
|
|
1117
|
+
this._requestIdToRequest.delete(request._requestId);
|
|
1118
|
+
request.request._setFailureText(event.errorText);
|
|
1119
|
+
this._page._frameManager.requestFailed(request.request, event.errorText.includes('cancelled'));
|
|
1120
|
+
}
|
|
1121
|
+
async _grantPermissions(origin, permissions) {
|
|
1122
|
+
const webPermissionToProtocol = new Map([['geolocation', 'geolocation'], ['clipboard-read', 'clipboard-read']]);
|
|
1123
|
+
const filtered = permissions.map(permission => {
|
|
1124
|
+
const protocolPermission = webPermissionToProtocol.get(permission);
|
|
1125
|
+
if (!protocolPermission) throw new Error('Unknown permission: ' + permission);
|
|
1126
|
+
return protocolPermission;
|
|
1127
|
+
});
|
|
1128
|
+
await this._pageProxySession.send('Emulation.grantPermissions', {
|
|
1129
|
+
origin,
|
|
1130
|
+
permissions: filtered
|
|
1131
|
+
});
|
|
1132
|
+
}
|
|
1133
|
+
async _clearPermissions() {
|
|
1134
|
+
await this._pageProxySession.send('Emulation.resetPermissions', {});
|
|
1135
|
+
}
|
|
1136
|
+
shouldToggleStyleSheetToSyncAnimations() {
|
|
1137
|
+
return true;
|
|
1138
|
+
}
|
|
1139
|
+
}
|
|
1140
|
+
|
|
1141
|
+
/**
|
|
1142
|
+
* WebKit Remote Addresses look like:
|
|
1143
|
+
*
|
|
1144
|
+
* macOS:
|
|
1145
|
+
* ::1.8911
|
|
1146
|
+
* 2606:2800:220:1:248:1893:25c8:1946.443
|
|
1147
|
+
* 127.0.0.1:8000
|
|
1148
|
+
*
|
|
1149
|
+
* ubuntu:
|
|
1150
|
+
* ::1:8907
|
|
1151
|
+
* 127.0.0.1:8000
|
|
1152
|
+
*
|
|
1153
|
+
* NB: They look IPv4 and IPv6's with ports but use an alternative notation.
|
|
1154
|
+
*/
|
|
1155
|
+
exports.WKPage = WKPage;
|
|
1156
|
+
function parseRemoteAddress(value) {
|
|
1157
|
+
if (!value) return;
|
|
1158
|
+
try {
|
|
1159
|
+
const colon = value.lastIndexOf(':');
|
|
1160
|
+
const dot = value.lastIndexOf('.');
|
|
1161
|
+
if (dot < 0) {
|
|
1162
|
+
// IPv6ish:port
|
|
1163
|
+
return {
|
|
1164
|
+
ipAddress: `[${value.slice(0, colon)}]`,
|
|
1165
|
+
port: +value.slice(colon + 1)
|
|
1166
|
+
};
|
|
1167
|
+
}
|
|
1168
|
+
if (colon > dot) {
|
|
1169
|
+
// IPv4:port
|
|
1170
|
+
const [address, port] = value.split(':');
|
|
1171
|
+
return {
|
|
1172
|
+
ipAddress: address,
|
|
1173
|
+
port: +port
|
|
1174
|
+
};
|
|
1175
|
+
} else {
|
|
1176
|
+
// IPv6ish.port
|
|
1177
|
+
const [address, port] = value.split('.');
|
|
1178
|
+
return {
|
|
1179
|
+
ipAddress: `[${address}]`,
|
|
1180
|
+
port: +port
|
|
1181
|
+
};
|
|
1182
|
+
}
|
|
1183
|
+
} catch (_) {}
|
|
1184
|
+
}
|
|
1185
|
+
|
|
1186
|
+
/**
|
|
1187
|
+
* Adapted from Source/WebInspectorUI/UserInterface/Models/Resource.js in
|
|
1188
|
+
* WebKit codebase.
|
|
1189
|
+
*/
|
|
1190
|
+
function isLoadedSecurely(url, timing) {
|
|
1191
|
+
try {
|
|
1192
|
+
const u = new URL(url);
|
|
1193
|
+
if (u.protocol !== 'https:' && u.protocol !== 'wss:' && u.protocol !== 'sftp:') return false;
|
|
1194
|
+
if (timing.secureConnectionStart === -1 && timing.connectStart !== -1) return false;
|
|
1195
|
+
return true;
|
|
1196
|
+
} catch (_) {}
|
|
1197
|
+
}
|
|
1198
|
+
const contextDelegateSymbol = Symbol('delegate');
|