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