@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.
Files changed (305) hide show
  1. package/LICENSE +202 -0
  2. package/NOTICE +5 -0
  3. package/README.md +3 -0
  4. package/ThirdPartyNotices.txt +1513 -0
  5. package/bin/PrintDeps.exe +0 -0
  6. package/bin/README.md +2 -0
  7. package/bin/install_media_pack.ps1 +5 -0
  8. package/bin/reinstall_chrome_beta_linux.sh +40 -0
  9. package/bin/reinstall_chrome_beta_mac.sh +13 -0
  10. package/bin/reinstall_chrome_beta_win.ps1 +23 -0
  11. package/bin/reinstall_chrome_stable_linux.sh +40 -0
  12. package/bin/reinstall_chrome_stable_mac.sh +12 -0
  13. package/bin/reinstall_chrome_stable_win.ps1 +23 -0
  14. package/bin/reinstall_msedge_beta_linux.sh +40 -0
  15. package/bin/reinstall_msedge_beta_mac.sh +11 -0
  16. package/bin/reinstall_msedge_beta_win.ps1 +22 -0
  17. package/bin/reinstall_msedge_dev_linux.sh +40 -0
  18. package/bin/reinstall_msedge_dev_mac.sh +11 -0
  19. package/bin/reinstall_msedge_dev_win.ps1 +22 -0
  20. package/bin/reinstall_msedge_stable_linux.sh +40 -0
  21. package/bin/reinstall_msedge_stable_mac.sh +11 -0
  22. package/bin/reinstall_msedge_stable_win.ps1 +23 -0
  23. package/browsers.json +64 -0
  24. package/cli.js +17 -0
  25. package/index.d.ts +17 -0
  26. package/index.js +33 -0
  27. package/index.mjs +28 -0
  28. package/lib/androidServerImpl.js +69 -0
  29. package/lib/browserServerImpl.js +92 -0
  30. package/lib/cli/driver.js +97 -0
  31. package/lib/cli/program.js +582 -0
  32. package/lib/cli/programWithTestStub.js +67 -0
  33. package/lib/client/accessibility.js +50 -0
  34. package/lib/client/android.js +473 -0
  35. package/lib/client/api.js +272 -0
  36. package/lib/client/artifact.js +79 -0
  37. package/lib/client/browser.js +145 -0
  38. package/lib/client/browserContext.js +509 -0
  39. package/lib/client/browserType.js +233 -0
  40. package/lib/client/cdpSession.js +53 -0
  41. package/lib/client/channelOwner.js +229 -0
  42. package/lib/client/clientHelper.js +57 -0
  43. package/lib/client/clientInstrumentation.js +40 -0
  44. package/lib/client/connection.js +327 -0
  45. package/lib/client/consoleMessage.js +55 -0
  46. package/lib/client/coverage.js +41 -0
  47. package/lib/client/dialog.js +57 -0
  48. package/lib/client/download.js +62 -0
  49. package/lib/client/electron.js +130 -0
  50. package/lib/client/elementHandle.js +291 -0
  51. package/lib/client/errors.js +77 -0
  52. package/lib/client/events.js +93 -0
  53. package/lib/client/fetch.js +343 -0
  54. package/lib/client/fileChooser.js +45 -0
  55. package/lib/client/frame.js +506 -0
  56. package/lib/client/harRouter.js +93 -0
  57. package/lib/client/input.js +111 -0
  58. package/lib/client/jsHandle.js +123 -0
  59. package/lib/client/jsonPipe.js +35 -0
  60. package/lib/client/localUtils.js +35 -0
  61. package/lib/client/locator.js +432 -0
  62. package/lib/client/network.js +601 -0
  63. package/lib/client/page.js +707 -0
  64. package/lib/client/playwright.js +74 -0
  65. package/lib/client/selectors.js +67 -0
  66. package/lib/client/stream.js +54 -0
  67. package/lib/client/tracing.js +135 -0
  68. package/lib/client/types.js +24 -0
  69. package/lib/client/video.js +51 -0
  70. package/lib/client/waiter.js +158 -0
  71. package/lib/client/webError.js +37 -0
  72. package/lib/client/worker.js +71 -0
  73. package/lib/client/writableStream.js +54 -0
  74. package/lib/common/socksProxy.js +569 -0
  75. package/lib/common/timeoutSettings.js +73 -0
  76. package/lib/common/types.js +5 -0
  77. package/lib/generated/consoleApiSource.js +7 -0
  78. package/lib/generated/injectedScriptSource.js +7 -0
  79. package/lib/generated/recorderSource.js +7 -0
  80. package/lib/generated/utilityScriptSource.js +7 -0
  81. package/lib/image_tools/colorUtils.js +98 -0
  82. package/lib/image_tools/compare.js +108 -0
  83. package/lib/image_tools/imageChannel.js +70 -0
  84. package/lib/image_tools/stats.js +102 -0
  85. package/lib/inProcessFactory.js +54 -0
  86. package/lib/inprocess.js +20 -0
  87. package/lib/outofprocess.js +67 -0
  88. package/lib/protocol/debug.js +27 -0
  89. package/lib/protocol/serializers.js +172 -0
  90. package/lib/protocol/transport.js +82 -0
  91. package/lib/protocol/validator.js +2599 -0
  92. package/lib/protocol/validatorPrimitives.js +139 -0
  93. package/lib/remote/playwrightConnection.js +274 -0
  94. package/lib/remote/playwrightServer.js +110 -0
  95. package/lib/server/accessibility.js +62 -0
  96. package/lib/server/android/android.js +441 -0
  97. package/lib/server/android/backendAdb.js +172 -0
  98. package/lib/server/artifact.js +104 -0
  99. package/lib/server/browser.js +129 -0
  100. package/lib/server/browserContext.js +609 -0
  101. package/lib/server/browserType.js +300 -0
  102. package/lib/server/chromium/appIcon.png +0 -0
  103. package/lib/server/chromium/chromium.js +346 -0
  104. package/lib/server/chromium/chromiumSwitches.js +41 -0
  105. package/lib/server/chromium/crAccessibility.js +237 -0
  106. package/lib/server/chromium/crBrowser.js +521 -0
  107. package/lib/server/chromium/crConnection.js +228 -0
  108. package/lib/server/chromium/crCoverage.js +246 -0
  109. package/lib/server/chromium/crDevTools.js +104 -0
  110. package/lib/server/chromium/crDragDrop.js +144 -0
  111. package/lib/server/chromium/crExecutionContext.js +156 -0
  112. package/lib/server/chromium/crInput.js +171 -0
  113. package/lib/server/chromium/crNetworkManager.js +723 -0
  114. package/lib/server/chromium/crPage.js +1173 -0
  115. package/lib/server/chromium/crPdf.js +147 -0
  116. package/lib/server/chromium/crProtocolHelper.js +131 -0
  117. package/lib/server/chromium/crServiceWorker.js +115 -0
  118. package/lib/server/chromium/defaultFontFamilies.js +145 -0
  119. package/lib/server/chromium/videoRecorder.js +155 -0
  120. package/lib/server/console.js +59 -0
  121. package/lib/server/cookieStore.js +112 -0
  122. package/lib/server/debugController.js +236 -0
  123. package/lib/server/debugger.js +132 -0
  124. package/lib/server/deviceDescriptors.js +21 -0
  125. package/lib/server/deviceDescriptorsSource.json +1549 -0
  126. package/lib/server/dialog.js +70 -0
  127. package/lib/server/dispatchers/androidDispatcher.js +193 -0
  128. package/lib/server/dispatchers/artifactDispatcher.js +118 -0
  129. package/lib/server/dispatchers/browserContextDispatcher.js +306 -0
  130. package/lib/server/dispatchers/browserDispatcher.js +170 -0
  131. package/lib/server/dispatchers/browserTypeDispatcher.js +55 -0
  132. package/lib/server/dispatchers/cdpSessionDispatcher.js +48 -0
  133. package/lib/server/dispatchers/debugControllerDispatcher.js +103 -0
  134. package/lib/server/dispatchers/dialogDispatcher.js +44 -0
  135. package/lib/server/dispatchers/dispatcher.js +400 -0
  136. package/lib/server/dispatchers/electronDispatcher.js +80 -0
  137. package/lib/server/dispatchers/elementHandlerDispatcher.js +228 -0
  138. package/lib/server/dispatchers/frameDispatcher.js +287 -0
  139. package/lib/server/dispatchers/jsHandleDispatcher.js +102 -0
  140. package/lib/server/dispatchers/jsonPipeDispatcher.js +61 -0
  141. package/lib/server/dispatchers/localUtilsDispatcher.js +399 -0
  142. package/lib/server/dispatchers/networkDispatchers.js +221 -0
  143. package/lib/server/dispatchers/pageDispatcher.js +363 -0
  144. package/lib/server/dispatchers/playwrightDispatcher.js +105 -0
  145. package/lib/server/dispatchers/selectorsDispatcher.js +36 -0
  146. package/lib/server/dispatchers/streamDispatcher.js +62 -0
  147. package/lib/server/dispatchers/tracingDispatcher.js +54 -0
  148. package/lib/server/dispatchers/writableStreamDispatcher.js +55 -0
  149. package/lib/server/dom.js +808 -0
  150. package/lib/server/download.js +53 -0
  151. package/lib/server/electron/electron.js +254 -0
  152. package/lib/server/electron/loader.js +57 -0
  153. package/lib/server/errors.js +68 -0
  154. package/lib/server/fetch.js +611 -0
  155. package/lib/server/fileChooser.js +42 -0
  156. package/lib/server/fileUploadUtils.js +71 -0
  157. package/lib/server/firefox/ffAccessibility.js +215 -0
  158. package/lib/server/firefox/ffBrowser.js +447 -0
  159. package/lib/server/firefox/ffConnection.js +168 -0
  160. package/lib/server/firefox/ffExecutionContext.js +138 -0
  161. package/lib/server/firefox/ffInput.js +150 -0
  162. package/lib/server/firefox/ffNetworkManager.js +231 -0
  163. package/lib/server/firefox/ffPage.js +558 -0
  164. package/lib/server/firefox/firefox.js +91 -0
  165. package/lib/server/formData.js +75 -0
  166. package/lib/server/frameSelectors.js +171 -0
  167. package/lib/server/frames.js +1597 -0
  168. package/lib/server/har/harRecorder.js +139 -0
  169. package/lib/server/har/harTracer.js +539 -0
  170. package/lib/server/helper.js +103 -0
  171. package/lib/server/index.js +96 -0
  172. package/lib/server/input.js +301 -0
  173. package/lib/server/instrumentation.js +74 -0
  174. package/lib/server/isomorphic/utilityScriptSerializers.js +212 -0
  175. package/lib/server/javascript.js +305 -0
  176. package/lib/server/launchApp.js +90 -0
  177. package/lib/server/macEditingCommands.js +139 -0
  178. package/lib/server/network.js +607 -0
  179. package/lib/server/page.js +793 -0
  180. package/lib/server/pipeTransport.js +85 -0
  181. package/lib/server/playwright.js +82 -0
  182. package/lib/server/progress.js +102 -0
  183. package/lib/server/protocolError.js +49 -0
  184. package/lib/server/recorder/codeGenerator.js +153 -0
  185. package/lib/server/recorder/csharp.js +310 -0
  186. package/lib/server/recorder/java.js +216 -0
  187. package/lib/server/recorder/javascript.js +229 -0
  188. package/lib/server/recorder/jsonl.js +47 -0
  189. package/lib/server/recorder/language.js +44 -0
  190. package/lib/server/recorder/python.js +275 -0
  191. package/lib/server/recorder/recorderActions.js +5 -0
  192. package/lib/server/recorder/recorderApp.js +181 -0
  193. package/lib/server/recorder/recorderUtils.js +48 -0
  194. package/lib/server/recorder/utils.js +45 -0
  195. package/lib/server/recorder.js +700 -0
  196. package/lib/server/registry/browserFetcher.js +168 -0
  197. package/lib/server/registry/dependencies.js +322 -0
  198. package/lib/server/registry/index.js +925 -0
  199. package/lib/server/registry/nativeDeps.js +383 -0
  200. package/lib/server/registry/oopDownloadBrowserMain.js +138 -0
  201. package/lib/server/screenshotter.js +354 -0
  202. package/lib/server/selectors.js +73 -0
  203. package/lib/server/socksInterceptor.js +100 -0
  204. package/lib/server/trace/recorder/snapshotter.js +168 -0
  205. package/lib/server/trace/recorder/snapshotterInjected.js +493 -0
  206. package/lib/server/trace/recorder/tracing.js +552 -0
  207. package/lib/server/trace/test/inMemorySnapshotter.js +93 -0
  208. package/lib/server/trace/viewer/traceViewer.js +229 -0
  209. package/lib/server/transport.js +191 -0
  210. package/lib/server/types.js +24 -0
  211. package/lib/server/usKeyboardLayout.js +555 -0
  212. package/lib/server/webkit/webkit.js +87 -0
  213. package/lib/server/webkit/wkAccessibility.js +194 -0
  214. package/lib/server/webkit/wkBrowser.js +328 -0
  215. package/lib/server/webkit/wkConnection.js +173 -0
  216. package/lib/server/webkit/wkExecutionContext.js +146 -0
  217. package/lib/server/webkit/wkInput.js +169 -0
  218. package/lib/server/webkit/wkInterceptableRequest.js +158 -0
  219. package/lib/server/webkit/wkPage.js +1198 -0
  220. package/lib/server/webkit/wkProvisionalPage.js +59 -0
  221. package/lib/server/webkit/wkWorkers.js +104 -0
  222. package/lib/third_party/diff_match_patch.js +2222 -0
  223. package/lib/third_party/pixelmatch.js +255 -0
  224. package/lib/utils/ascii.js +31 -0
  225. package/lib/utils/comparators.js +171 -0
  226. package/lib/utils/crypto.js +33 -0
  227. package/lib/utils/debug.js +46 -0
  228. package/lib/utils/debugLogger.js +89 -0
  229. package/lib/utils/env.js +47 -0
  230. package/lib/utils/eventsHelper.js +38 -0
  231. package/lib/utils/fileUtils.js +66 -0
  232. package/lib/utils/glob.js +83 -0
  233. package/lib/utils/happy-eyeballs.js +154 -0
  234. package/lib/utils/headers.js +52 -0
  235. package/lib/utils/hostPlatform.js +124 -0
  236. package/lib/utils/httpServer.js +195 -0
  237. package/lib/utils/index.js +324 -0
  238. package/lib/utils/isomorphic/cssParser.js +250 -0
  239. package/lib/utils/isomorphic/cssTokenizer.js +979 -0
  240. package/lib/utils/isomorphic/locatorGenerators.js +651 -0
  241. package/lib/utils/isomorphic/locatorParser.js +179 -0
  242. package/lib/utils/isomorphic/locatorUtils.js +62 -0
  243. package/lib/utils/isomorphic/selectorParser.js +397 -0
  244. package/lib/utils/isomorphic/stringUtils.js +107 -0
  245. package/lib/utils/isomorphic/traceUtils.js +39 -0
  246. package/lib/utils/linuxUtils.js +78 -0
  247. package/lib/utils/manualPromise.js +109 -0
  248. package/lib/utils/mimeType.js +29 -0
  249. package/lib/utils/multimap.js +75 -0
  250. package/lib/utils/network.js +189 -0
  251. package/lib/utils/processLauncher.js +248 -0
  252. package/lib/utils/profiler.js +53 -0
  253. package/lib/utils/rtti.js +41 -0
  254. package/lib/utils/semaphore.js +51 -0
  255. package/lib/utils/spawnAsync.js +45 -0
  256. package/lib/utils/stackTrace.js +123 -0
  257. package/lib/utils/task.js +58 -0
  258. package/lib/utils/time.js +37 -0
  259. package/lib/utils/timeoutRunner.js +131 -0
  260. package/lib/utils/traceUtils.js +44 -0
  261. package/lib/utils/userAgent.js +105 -0
  262. package/lib/utils/wsServer.js +125 -0
  263. package/lib/utils/zipFile.js +75 -0
  264. package/lib/utils/zones.js +99 -0
  265. package/lib/utilsBundle.js +81 -0
  266. package/lib/utilsBundleImpl/index.js +51 -0
  267. package/lib/utilsBundleImpl/xdg-open +1066 -0
  268. package/lib/vite/htmlReport/index.html +66 -0
  269. package/lib/vite/recorder/assets/codeMirrorModule-Hs9-1ZG4.css +1 -0
  270. package/lib/vite/recorder/assets/codeMirrorModule-I9ks4y7D.js +24 -0
  271. package/lib/vite/recorder/assets/codicon-zGuYmc9o.ttf +0 -0
  272. package/lib/vite/recorder/assets/index-ljsTwXtJ.css +1 -0
  273. package/lib/vite/recorder/assets/index-yg8ypzl6.js +47 -0
  274. package/lib/vite/recorder/index.html +29 -0
  275. package/lib/vite/recorder/playwright-logo.svg +9 -0
  276. package/lib/vite/traceViewer/assets/codeMirrorModule-GluP1cQ1.js +24 -0
  277. package/lib/vite/traceViewer/assets/codeMirrorModule-fqJB1XDu.js +24 -0
  278. package/lib/vite/traceViewer/assets/codeMirrorModule-y3M3aAqy.js +24 -0
  279. package/lib/vite/traceViewer/assets/wsPort-Rvwd4WC-.js +69 -0
  280. package/lib/vite/traceViewer/assets/wsPort-dlD7vDkY.js +69 -0
  281. package/lib/vite/traceViewer/assets/wsPort-qOE2NWrO.js +69 -0
  282. package/lib/vite/traceViewer/assets/xtermModule-Yt6xwiJ_.js +9 -0
  283. package/lib/vite/traceViewer/codeMirrorModule.Hs9-1ZG4.css +1 -0
  284. package/lib/vite/traceViewer/codicon.zGuYmc9o.ttf +0 -0
  285. package/lib/vite/traceViewer/index.-g_5lMbJ.css +1 -0
  286. package/lib/vite/traceViewer/index.4X7zDysg.js +2 -0
  287. package/lib/vite/traceViewer/index.HkJgzlGy.js +2 -0
  288. package/lib/vite/traceViewer/index.html +26 -0
  289. package/lib/vite/traceViewer/index.kRjx5sAJ.js +2 -0
  290. package/lib/vite/traceViewer/playwright-logo.svg +9 -0
  291. package/lib/vite/traceViewer/snapshot.html +21 -0
  292. package/lib/vite/traceViewer/sw.bundle.js +4 -0
  293. package/lib/vite/traceViewer/uiMode.1Wcp_Kto.js +10 -0
  294. package/lib/vite/traceViewer/uiMode.GTNzARcV.js +10 -0
  295. package/lib/vite/traceViewer/uiMode.html +17 -0
  296. package/lib/vite/traceViewer/uiMode.pWy0Re7G.css +1 -0
  297. package/lib/vite/traceViewer/uiMode.zV-7Lf9v.js +10 -0
  298. package/lib/vite/traceViewer/wsPort.kSgQKQ0y.css +1 -0
  299. package/lib/vite/traceViewer/xtermModule.0lwXJFHT.css +32 -0
  300. package/lib/zipBundle.js +25 -0
  301. package/lib/zipBundleImpl.js +5 -0
  302. package/package.json +43 -0
  303. package/types/protocol.d.ts +20304 -0
  304. package/types/structs.d.ts +45 -0
  305. 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');