@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,139 @@
1
+ "use strict";
2
+
3
+ Object.defineProperty(exports, "__esModule", {
4
+ value: true
5
+ });
6
+ exports.HarRecorder = void 0;
7
+ var _fs = _interopRequireDefault(require("fs"));
8
+ var _path = _interopRequireDefault(require("path"));
9
+ var _artifact = require("../artifact");
10
+ var _harTracer = require("./harTracer");
11
+ var _zipBundle = require("../../zipBundle");
12
+ var _manualPromise = require("../../utils/manualPromise");
13
+ var _utils = require("../../utils");
14
+ function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; }
15
+ /**
16
+ * 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 HarRecorder {
32
+ constructor(context, page, options) {
33
+ this._artifact = void 0;
34
+ this._isFlushed = false;
35
+ this._tracer = void 0;
36
+ this._entries = [];
37
+ this._zipFile = null;
38
+ this._writtenZipEntries = new Set();
39
+ this._artifact = new _artifact.Artifact(context, _path.default.join(context._browser.options.artifactsDir, `${(0, _utils.createGuid)()}.har`));
40
+ const urlFilterRe = options.urlRegexSource !== undefined && options.urlRegexFlags !== undefined ? new RegExp(options.urlRegexSource, options.urlRegexFlags) : undefined;
41
+ const expectsZip = options.path.endsWith('.zip');
42
+ const content = options.content || (expectsZip ? 'attach' : 'embed');
43
+ this._tracer = new _harTracer.HarTracer(context, page, this, {
44
+ content,
45
+ slimMode: options.mode === 'minimal',
46
+ includeTraceInfo: false,
47
+ recordRequestOverrides: true,
48
+ waitForContentOnStop: true,
49
+ urlFilter: urlFilterRe !== null && urlFilterRe !== void 0 ? urlFilterRe : options.urlGlob
50
+ });
51
+ this._zipFile = content === 'attach' || expectsZip ? new _zipBundle.yazl.ZipFile() : null;
52
+ this._tracer.start({
53
+ omitScripts: false
54
+ });
55
+ }
56
+ onEntryStarted(entry) {
57
+ this._entries.push(entry);
58
+ }
59
+ onEntryFinished(entry) {}
60
+ onContentBlob(sha1, buffer) {
61
+ if (!this._zipFile || this._writtenZipEntries.has(sha1)) return;
62
+ this._writtenZipEntries.add(sha1);
63
+ this._zipFile.addBuffer(buffer, sha1);
64
+ }
65
+ async flush() {
66
+ if (this._isFlushed) return;
67
+ this._isFlushed = true;
68
+ await this._tracer.flush();
69
+ const log = this._tracer.stop();
70
+ log.entries = this._entries;
71
+ const harFileContent = jsonStringify({
72
+ log
73
+ });
74
+ if (this._zipFile) {
75
+ const result = new _manualPromise.ManualPromise();
76
+ this._zipFile.on('error', error => result.reject(error));
77
+ this._zipFile.addBuffer(Buffer.from(harFileContent, 'utf-8'), 'har.har');
78
+ this._zipFile.end();
79
+ this._zipFile.outputStream.pipe(_fs.default.createWriteStream(this._artifact.localPath())).on('close', () => {
80
+ result.resolve();
81
+ });
82
+ await result;
83
+ } else {
84
+ await _fs.default.promises.writeFile(this._artifact.localPath(), harFileContent);
85
+ }
86
+ }
87
+ async export() {
88
+ await this.flush();
89
+ this._artifact.reportFinished();
90
+ return this._artifact;
91
+ }
92
+ }
93
+ exports.HarRecorder = HarRecorder;
94
+ function jsonStringify(object) {
95
+ const tokens = [];
96
+ innerJsonStringify(object, tokens, '', false, undefined);
97
+ return tokens.join('');
98
+ }
99
+ function innerJsonStringify(object, tokens, indent, flat, parentKey) {
100
+ if (typeof object !== 'object' || object === null) {
101
+ tokens.push(JSON.stringify(object));
102
+ return;
103
+ }
104
+ const isArray = Array.isArray(object);
105
+ if (!isArray && object.constructor.name !== 'Object') {
106
+ tokens.push(JSON.stringify(object));
107
+ return;
108
+ }
109
+ const entries = isArray ? object : Object.entries(object).filter(e => e[1] !== undefined);
110
+ if (!entries.length) {
111
+ tokens.push(isArray ? `[]` : `{}`);
112
+ return;
113
+ }
114
+ const childIndent = `${indent} `;
115
+ let brackets;
116
+ if (isArray) brackets = flat ? {
117
+ open: '[',
118
+ close: ']'
119
+ } : {
120
+ open: `[\n${childIndent}`,
121
+ close: `\n${indent}]`
122
+ };else brackets = flat ? {
123
+ open: '{ ',
124
+ close: ' }'
125
+ } : {
126
+ open: `{\n${childIndent}`,
127
+ close: `\n${indent}}`
128
+ };
129
+ tokens.push(brackets.open);
130
+ for (let i = 0; i < entries.length; ++i) {
131
+ const entry = entries[i];
132
+ if (i) tokens.push(flat ? `, ` : `,\n${childIndent}`);
133
+ if (!isArray) tokens.push(`${JSON.stringify(entry[0])}: `);
134
+ const key = isArray ? undefined : entry[0];
135
+ const flatten = flat || key === 'timings' || parentKey === 'headers';
136
+ innerJsonStringify(isArray ? entry : entry[1], tokens, childIndent, flatten, key);
137
+ }
138
+ tokens.push(brackets.close);
139
+ }
@@ -0,0 +1,539 @@
1
+ "use strict";
2
+
3
+ Object.defineProperty(exports, "__esModule", {
4
+ value: true
5
+ });
6
+ exports.HarTracer = void 0;
7
+ var _browserContext = require("../browserContext");
8
+ var _fetch = require("../fetch");
9
+ var _helper = require("../helper");
10
+ var network = _interopRequireWildcard(require("../network"));
11
+ var _utils = require("../../utils");
12
+ var _eventsHelper = require("../../utils/eventsHelper");
13
+ var _utilsBundle = require("../../utilsBundle");
14
+ var _manualPromise = require("../../utils/manualPromise");
15
+ var _userAgent = require("../../utils/userAgent");
16
+ var _network2 = require("../../utils/network");
17
+ var _frames = require("../frames");
18
+ var _mimeType = require("../../utils/mimeType");
19
+ 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); }
20
+ 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; }
21
+ /**
22
+ * Copyright (c) Microsoft Corporation.
23
+ *
24
+ * Licensed under the Apache License, Version 2.0 (the "License");
25
+ * you may not use this file except in compliance with the License.
26
+ * You may obtain a copy of the License at
27
+ *
28
+ * http://www.apache.org/licenses/LICENSE-2.0
29
+ *
30
+ * Unless required by applicable law or agreed to in writing, software
31
+ * distributed under the License is distributed on an "AS IS" BASIS,
32
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
33
+ * See the License for the specific language governing permissions and
34
+ * limitations under the License.
35
+ */
36
+
37
+ const FALLBACK_HTTP_VERSION = 'HTTP/1.1';
38
+ class HarTracer {
39
+ constructor(context, page, delegate, options) {
40
+ this._context = void 0;
41
+ this._barrierPromises = new Set();
42
+ this._delegate = void 0;
43
+ this._options = void 0;
44
+ this._pageEntries = new Map();
45
+ this._eventListeners = [];
46
+ this._started = false;
47
+ this._entrySymbol = void 0;
48
+ this._baseURL = void 0;
49
+ this._page = void 0;
50
+ this._context = context;
51
+ this._page = page;
52
+ this._delegate = delegate;
53
+ this._options = options;
54
+ if (options.slimMode) {
55
+ options.omitSecurityDetails = true;
56
+ options.omitCookies = true;
57
+ options.omitTiming = true;
58
+ options.omitServerIP = true;
59
+ options.omitSizes = true;
60
+ options.omitPages = true;
61
+ }
62
+ this._entrySymbol = Symbol('requestHarEntry');
63
+ this._baseURL = context instanceof _fetch.APIRequestContext ? context._defaultOptions().baseURL : context._options.baseURL;
64
+ }
65
+ start(options) {
66
+ if (this._started) return;
67
+ this._options.omitScripts = options.omitScripts;
68
+ this._started = true;
69
+ const apiRequest = this._context instanceof _fetch.APIRequestContext ? this._context : this._context.fetchRequest;
70
+ this._eventListeners = [_eventsHelper.eventsHelper.addEventListener(apiRequest, _fetch.APIRequestContext.Events.Request, event => this._onAPIRequest(event)), _eventsHelper.eventsHelper.addEventListener(apiRequest, _fetch.APIRequestContext.Events.RequestFinished, event => this._onAPIRequestFinished(event))];
71
+ if (this._context instanceof _browserContext.BrowserContext) {
72
+ this._eventListeners.push(_eventsHelper.eventsHelper.addEventListener(this._context, _browserContext.BrowserContext.Events.Page, page => this._createPageEntryIfNeeded(page)), _eventsHelper.eventsHelper.addEventListener(this._context, _browserContext.BrowserContext.Events.Request, request => this._onRequest(request)), _eventsHelper.eventsHelper.addEventListener(this._context, _browserContext.BrowserContext.Events.RequestFinished, ({
73
+ request,
74
+ response
75
+ }) => this._onRequestFinished(request, response).catch(() => {})), _eventsHelper.eventsHelper.addEventListener(this._context, _browserContext.BrowserContext.Events.RequestFailed, request => this._onRequestFailed(request)), _eventsHelper.eventsHelper.addEventListener(this._context, _browserContext.BrowserContext.Events.Response, response => this._onResponse(response)), _eventsHelper.eventsHelper.addEventListener(this._context, _browserContext.BrowserContext.Events.RequestAborted, request => this._onRequestAborted(request)), _eventsHelper.eventsHelper.addEventListener(this._context, _browserContext.BrowserContext.Events.RequestFulfilled, request => this._onRequestFulfilled(request)), _eventsHelper.eventsHelper.addEventListener(this._context, _browserContext.BrowserContext.Events.RequestContinued, request => this._onRequestContinued(request)));
76
+ }
77
+ }
78
+ _shouldIncludeEntryWithUrl(urlString) {
79
+ return !this._options.urlFilter || (0, _network2.urlMatches)(this._baseURL, urlString, this._options.urlFilter);
80
+ }
81
+ _entryForRequest(request) {
82
+ return request[this._entrySymbol];
83
+ }
84
+ _createPageEntryIfNeeded(page) {
85
+ if (!page) return;
86
+ if (this._options.omitPages) return;
87
+ if (this._page && page !== this._page) return;
88
+ let pageEntry = this._pageEntries.get(page);
89
+ if (!pageEntry) {
90
+ const date = new Date();
91
+ pageEntry = {
92
+ startedDateTime: date.toISOString(),
93
+ id: page.guid,
94
+ title: '',
95
+ pageTimings: this._options.omitTiming ? {} : {
96
+ onContentLoad: -1,
97
+ onLoad: -1
98
+ }
99
+ };
100
+ pageEntry[startedDateSymbol] = date;
101
+ page.mainFrame().on(_frames.Frame.Events.AddLifecycle, event => {
102
+ if (event === 'load') this._onLoad(page, pageEntry);
103
+ if (event === 'domcontentloaded') this._onDOMContentLoaded(page, pageEntry);
104
+ });
105
+ this._pageEntries.set(page, pageEntry);
106
+ }
107
+ return pageEntry;
108
+ }
109
+ _onDOMContentLoaded(page, pageEntry) {
110
+ const promise = page.mainFrame().evaluateExpression(String(() => {
111
+ return {
112
+ title: document.title,
113
+ domContentLoaded: performance.timing.domContentLoadedEventStart
114
+ };
115
+ }), {
116
+ isFunction: true,
117
+ world: 'utility'
118
+ }).then(result => {
119
+ pageEntry.title = result.title;
120
+ if (!this._options.omitTiming) pageEntry.pageTimings.onContentLoad = result.domContentLoaded;
121
+ }).catch(() => {});
122
+ this._addBarrier(page, promise);
123
+ }
124
+ _onLoad(page, pageEntry) {
125
+ const promise = page.mainFrame().evaluateExpression(String(() => {
126
+ return {
127
+ title: document.title,
128
+ loaded: performance.timing.loadEventStart
129
+ };
130
+ }), {
131
+ isFunction: true,
132
+ world: 'utility'
133
+ }).then(result => {
134
+ pageEntry.title = result.title;
135
+ if (!this._options.omitTiming) pageEntry.pageTimings.onLoad = result.loaded;
136
+ }).catch(() => {});
137
+ this._addBarrier(page, promise);
138
+ }
139
+ _addBarrier(target, promise) {
140
+ if (!target) return null;
141
+ if (!this._options.waitForContentOnStop) return;
142
+ const race = target.openScope.safeRace(promise);
143
+ this._barrierPromises.add(race);
144
+ race.then(() => this._barrierPromises.delete(race));
145
+ }
146
+ _onAPIRequest(event) {
147
+ var _event$postData;
148
+ if (!this._shouldIncludeEntryWithUrl(event.url.toString())) return;
149
+ const harEntry = createHarEntry(event.method, event.url, undefined, this._options);
150
+ harEntry._apiRequest = true;
151
+ if (!this._options.omitCookies) harEntry.request.cookies = event.cookies;
152
+ harEntry.request.headers = Object.entries(event.headers).map(([name, value]) => ({
153
+ name,
154
+ value
155
+ }));
156
+ harEntry.request.postData = this._postDataForBuffer(event.postData || null, event.headers['content-type'], this._options.content);
157
+ if (!this._options.omitSizes) harEntry.request.bodySize = ((_event$postData = event.postData) === null || _event$postData === void 0 ? void 0 : _event$postData.length) || 0;
158
+ event[this._entrySymbol] = harEntry;
159
+ if (this._started) this._delegate.onEntryStarted(harEntry);
160
+ }
161
+ _onAPIRequestFinished(event) {
162
+ const harEntry = this._entryForRequest(event.requestEvent);
163
+ if (!harEntry) return;
164
+ harEntry.response.status = event.statusCode;
165
+ harEntry.response.statusText = event.statusMessage;
166
+ harEntry.response.httpVersion = event.httpVersion;
167
+ harEntry.response.redirectURL = event.headers.location || '';
168
+ for (let i = 0; i < event.rawHeaders.length; i += 2) {
169
+ harEntry.response.headers.push({
170
+ name: event.rawHeaders[i],
171
+ value: event.rawHeaders[i + 1]
172
+ });
173
+ }
174
+ harEntry.response.cookies = this._options.omitCookies ? [] : event.cookies.map(c => {
175
+ return {
176
+ ...c,
177
+ expires: c.expires === -1 ? undefined : safeDateToISOString(c.expires)
178
+ };
179
+ });
180
+ const content = harEntry.response.content;
181
+ const contentType = event.headers['content-type'];
182
+ if (contentType) content.mimeType = contentType;
183
+ this._storeResponseContent(event.body, content, 'other');
184
+ if (this._started) this._delegate.onEntryFinished(harEntry);
185
+ }
186
+ _onRequest(request) {
187
+ var _request$frame, _request$frame2;
188
+ if (!this._shouldIncludeEntryWithUrl(request.url())) return;
189
+ const page = (_request$frame = request.frame()) === null || _request$frame === void 0 ? void 0 : _request$frame._page;
190
+ if (this._page && page !== this._page) return;
191
+ const url = network.parsedURL(request.url());
192
+ if (!url) return;
193
+ const pageEntry = this._createPageEntryIfNeeded(page);
194
+ const harEntry = createHarEntry(request.method(), url, (_request$frame2 = request.frame()) === null || _request$frame2 === void 0 ? void 0 : _request$frame2.guid, this._options);
195
+ if (pageEntry) harEntry.pageref = pageEntry.id;
196
+ this._recordRequestHeadersAndCookies(harEntry, request.headers());
197
+ harEntry.request.postData = this._postDataForRequest(request, this._options.content);
198
+ if (!this._options.omitSizes) harEntry.request.bodySize = request.bodySize();
199
+ if (request.redirectedFrom()) {
200
+ const fromEntry = this._entryForRequest(request.redirectedFrom());
201
+ if (fromEntry) fromEntry.response.redirectURL = request.url();
202
+ }
203
+ request[this._entrySymbol] = harEntry;
204
+ (0, _utils.assert)(this._started);
205
+ this._delegate.onEntryStarted(harEntry);
206
+ }
207
+ _recordRequestHeadersAndCookies(harEntry, headers) {
208
+ if (!this._options.omitCookies) {
209
+ harEntry.request.cookies = [];
210
+ for (const header of headers.filter(header => header.name.toLowerCase() === 'cookie')) harEntry.request.cookies.push(...header.value.split(';').map(parseCookie));
211
+ }
212
+ harEntry.request.headers = headers;
213
+ }
214
+ _recordRequestOverrides(harEntry, request) {
215
+ if (!request._hasOverrides() || !this._options.recordRequestOverrides) return;
216
+ harEntry.request.method = request.method();
217
+ harEntry.request.url = request.url();
218
+ harEntry.request.postData = this._postDataForRequest(request, this._options.content);
219
+ this._recordRequestHeadersAndCookies(harEntry, request.headers());
220
+ }
221
+ async _onRequestFinished(request, response) {
222
+ var _request$frame3;
223
+ if (!response) return;
224
+ const harEntry = this._entryForRequest(request);
225
+ if (!harEntry) return;
226
+ const page = (_request$frame3 = request.frame()) === null || _request$frame3 === void 0 ? void 0 : _request$frame3._page;
227
+
228
+ // In WebKit security details and server ip are reported in Network.loadingFinished, so we populate
229
+ // it here to not hang in case of long chunked responses, see https://github.com/microsoft/playwright/issues/21182.
230
+ if (!this._options.omitServerIP) {
231
+ this._addBarrier(page || request.serviceWorker(), response.serverAddr().then(server => {
232
+ if (server !== null && server !== void 0 && server.ipAddress) harEntry.serverIPAddress = server.ipAddress;
233
+ if (server !== null && server !== void 0 && server.port) harEntry._serverPort = server.port;
234
+ }));
235
+ }
236
+ if (!this._options.omitSecurityDetails) {
237
+ this._addBarrier(page || request.serviceWorker(), response.securityDetails().then(details => {
238
+ if (details) harEntry._securityDetails = details;
239
+ }));
240
+ }
241
+ const httpVersion = response.httpVersion();
242
+ harEntry.request.httpVersion = httpVersion;
243
+ harEntry.response.httpVersion = httpVersion;
244
+ const compressionCalculationBarrier = this._options.omitSizes ? undefined : {
245
+ _encodedBodySize: -1,
246
+ _decodedBodySize: -1,
247
+ barrier: new _manualPromise.ManualPromise(),
248
+ _check: function () {
249
+ if (this._encodedBodySize !== -1 && this._decodedBodySize !== -1) {
250
+ harEntry.response.content.compression = Math.max(0, this._decodedBodySize - this._encodedBodySize);
251
+ this.barrier.resolve();
252
+ }
253
+ },
254
+ setEncodedBodySize: function (encodedBodySize) {
255
+ this._encodedBodySize = encodedBodySize;
256
+ this._check();
257
+ },
258
+ setDecodedBodySize: function (decodedBodySize) {
259
+ this._decodedBodySize = decodedBodySize;
260
+ this._check();
261
+ }
262
+ };
263
+ if (compressionCalculationBarrier) this._addBarrier(page || request.serviceWorker(), compressionCalculationBarrier.barrier);
264
+ const promise = response.body().then(buffer => {
265
+ if (this._options.omitScripts && request.resourceType() === 'script') {
266
+ compressionCalculationBarrier === null || compressionCalculationBarrier === void 0 || compressionCalculationBarrier.setDecodedBodySize(0);
267
+ return;
268
+ }
269
+ const content = harEntry.response.content;
270
+ compressionCalculationBarrier === null || compressionCalculationBarrier === void 0 || compressionCalculationBarrier.setDecodedBodySize(buffer.length);
271
+ this._storeResponseContent(buffer, content, request.resourceType());
272
+ }).catch(() => {
273
+ compressionCalculationBarrier === null || compressionCalculationBarrier === void 0 || compressionCalculationBarrier.setDecodedBodySize(0);
274
+ }).then(() => {
275
+ if (this._started) this._delegate.onEntryFinished(harEntry);
276
+ });
277
+ this._addBarrier(page || request.serviceWorker(), promise);
278
+
279
+ // Respose end timing is only available after the response event was received.
280
+ const timing = response.timing();
281
+ harEntry.timings.receive = response.request()._responseEndTiming !== -1 ? _helper.helper.millisToRoundishMillis(response.request()._responseEndTiming - timing.responseStart) : -1;
282
+ this._computeHarEntryTotalTime(harEntry);
283
+ if (!this._options.omitSizes) {
284
+ this._addBarrier(page || request.serviceWorker(), response.sizes().then(sizes => {
285
+ harEntry.response.bodySize = sizes.responseBodySize;
286
+ harEntry.response.headersSize = sizes.responseHeadersSize;
287
+ harEntry.response._transferSize = sizes.transferSize;
288
+ harEntry.request.headersSize = sizes.requestHeadersSize;
289
+ compressionCalculationBarrier === null || compressionCalculationBarrier === void 0 || compressionCalculationBarrier.setEncodedBodySize(sizes.responseBodySize);
290
+ }));
291
+ }
292
+ }
293
+ async _onRequestFailed(request) {
294
+ const harEntry = this._entryForRequest(request);
295
+ if (!harEntry) return;
296
+ if (request._failureText !== null) harEntry.response._failureText = request._failureText;
297
+ this._recordRequestOverrides(harEntry, request);
298
+ if (this._started) this._delegate.onEntryFinished(harEntry);
299
+ }
300
+ _onRequestAborted(request) {
301
+ const harEntry = this._entryForRequest(request);
302
+ if (harEntry) harEntry._wasAborted = true;
303
+ }
304
+ _onRequestFulfilled(request) {
305
+ const harEntry = this._entryForRequest(request);
306
+ if (harEntry) harEntry._wasFulfilled = true;
307
+ }
308
+ _onRequestContinued(request) {
309
+ const harEntry = this._entryForRequest(request);
310
+ if (harEntry) harEntry._wasContinued = true;
311
+ }
312
+ _storeResponseContent(buffer, content, resourceType) {
313
+ if (!buffer) {
314
+ content.size = 0;
315
+ return;
316
+ }
317
+ if (!this._options.omitSizes) content.size = buffer.length;
318
+ if (this._options.content === 'embed') {
319
+ // Sometimes, we can receive a font/media file with textual mime type. Browser
320
+ // still interprets them correctly, but the 'content-type' header is obviously wrong.
321
+ if ((0, _mimeType.isTextualMimeType)(content.mimeType) && resourceType !== 'font') {
322
+ content.text = buffer.toString();
323
+ } else {
324
+ content.text = buffer.toString('base64');
325
+ content.encoding = 'base64';
326
+ }
327
+ } else if (this._options.content === 'attach') {
328
+ const sha1 = (0, _utils.calculateSha1)(buffer) + '.' + (_utilsBundle.mime.getExtension(content.mimeType) || 'dat');
329
+ if (this._options.includeTraceInfo) content._sha1 = sha1;else content._file = sha1;
330
+ if (this._started) this._delegate.onContentBlob(sha1, buffer);
331
+ }
332
+ }
333
+ _onResponse(response) {
334
+ var _response$frame;
335
+ const harEntry = this._entryForRequest(response.request());
336
+ if (!harEntry) return;
337
+ const page = (_response$frame = response.frame()) === null || _response$frame === void 0 ? void 0 : _response$frame._page;
338
+ const pageEntry = this._createPageEntryIfNeeded(page);
339
+ const request = response.request();
340
+
341
+ // Prefer "response received" time over "request sent" time
342
+ // for the purpose of matching requests that were used in a particular snapshot.
343
+ // Note that both snapshot time and request time are taken here in the Node process.
344
+ if (this._options.includeTraceInfo) harEntry._monotonicTime = (0, _utils.monotonicTime)();
345
+ harEntry.response = {
346
+ status: response.status(),
347
+ statusText: response.statusText(),
348
+ httpVersion: response.httpVersion(),
349
+ // These are bad values that will be overwritten bellow.
350
+ cookies: [],
351
+ headers: [],
352
+ content: {
353
+ size: -1,
354
+ mimeType: 'x-unknown'
355
+ },
356
+ headersSize: -1,
357
+ bodySize: -1,
358
+ redirectURL: '',
359
+ _transferSize: this._options.omitSizes ? undefined : -1
360
+ };
361
+ if (!this._options.omitTiming) {
362
+ const startDateTime = pageEntry ? pageEntry[startedDateSymbol].valueOf() : 0;
363
+ const timing = response.timing();
364
+ if (pageEntry && startDateTime > timing.startTime) pageEntry.startedDateTime = new Date(timing.startTime).toISOString();
365
+ const dns = timing.domainLookupEnd !== -1 ? _helper.helper.millisToRoundishMillis(timing.domainLookupEnd - timing.domainLookupStart) : -1;
366
+ const connect = timing.connectEnd !== -1 ? _helper.helper.millisToRoundishMillis(timing.connectEnd - timing.connectStart) : -1;
367
+ const ssl = timing.connectEnd !== -1 ? _helper.helper.millisToRoundishMillis(timing.connectEnd - timing.secureConnectionStart) : -1;
368
+ const wait = timing.responseStart !== -1 ? _helper.helper.millisToRoundishMillis(timing.responseStart - timing.requestStart) : -1;
369
+ const receive = -1;
370
+ harEntry.timings = {
371
+ dns,
372
+ connect,
373
+ ssl,
374
+ send: 0,
375
+ wait,
376
+ receive
377
+ };
378
+ this._computeHarEntryTotalTime(harEntry);
379
+ }
380
+ this._recordRequestOverrides(harEntry, request);
381
+ this._addBarrier(page || request.serviceWorker(), request.rawRequestHeaders().then(headers => {
382
+ this._recordRequestHeadersAndCookies(harEntry, headers);
383
+ }));
384
+ // Record available headers including redirect location in case the tracing is stopped before
385
+ // response extra info is received (in Chromium).
386
+ this._recordResponseHeaders(harEntry, response.headers());
387
+ this._addBarrier(page || request.serviceWorker(), response.rawResponseHeaders().then(headers => {
388
+ this._recordResponseHeaders(harEntry, headers);
389
+ }));
390
+ }
391
+ _recordResponseHeaders(harEntry, headers) {
392
+ if (!this._options.omitCookies) {
393
+ harEntry.response.cookies = headers.filter(header => header.name.toLowerCase() === 'set-cookie').map(header => parseCookie(header.value));
394
+ }
395
+ harEntry.response.headers = headers;
396
+ const contentType = headers.find(header => header.name.toLowerCase() === 'content-type');
397
+ if (contentType) harEntry.response.content.mimeType = contentType.value;
398
+ }
399
+ _computeHarEntryTotalTime(harEntry) {
400
+ harEntry.time = [harEntry.timings.dns, harEntry.timings.connect, harEntry.timings.ssl, harEntry.timings.wait, harEntry.timings.receive].reduce((pre, cur) => (cur || -1) > 0 ? cur + pre : pre, 0);
401
+ }
402
+ async flush() {
403
+ await Promise.all(this._barrierPromises);
404
+ }
405
+ stop() {
406
+ this._started = false;
407
+ _eventsHelper.eventsHelper.removeEventListeners(this._eventListeners);
408
+ this._barrierPromises.clear();
409
+ const context = this._context instanceof _browserContext.BrowserContext ? this._context : undefined;
410
+ const log = {
411
+ version: '1.2',
412
+ creator: {
413
+ name: 'Playwright',
414
+ version: (0, _userAgent.getPlaywrightVersion)()
415
+ },
416
+ browser: {
417
+ name: (context === null || context === void 0 ? void 0 : context._browser.options.name) || '',
418
+ version: (context === null || context === void 0 ? void 0 : context._browser.version()) || ''
419
+ },
420
+ pages: this._pageEntries.size ? Array.from(this._pageEntries.values()) : undefined,
421
+ entries: []
422
+ };
423
+ if (!this._options.omitTiming) {
424
+ for (const pageEntry of log.pages || []) {
425
+ const startDateTime = pageEntry[startedDateSymbol].valueOf();
426
+ if (typeof pageEntry.pageTimings.onContentLoad === 'number' && pageEntry.pageTimings.onContentLoad >= 0) pageEntry.pageTimings.onContentLoad -= startDateTime;else pageEntry.pageTimings.onContentLoad = -1;
427
+ if (typeof pageEntry.pageTimings.onLoad === 'number' && pageEntry.pageTimings.onLoad >= 0) pageEntry.pageTimings.onLoad -= startDateTime;else pageEntry.pageTimings.onLoad = -1;
428
+ }
429
+ }
430
+ this._pageEntries.clear();
431
+ return log;
432
+ }
433
+ _postDataForRequest(request, content) {
434
+ const postData = request.postDataBuffer();
435
+ if (!postData) return;
436
+ const contentType = request.headerValue('content-type');
437
+ return this._postDataForBuffer(postData, contentType, content);
438
+ }
439
+ _postDataForBuffer(postData, contentType, content) {
440
+ var _contentType;
441
+ if (!postData) return;
442
+ (_contentType = contentType) !== null && _contentType !== void 0 ? _contentType : contentType = 'application/octet-stream';
443
+ const result = {
444
+ mimeType: contentType,
445
+ text: '',
446
+ params: []
447
+ };
448
+ if (content === 'embed' && contentType !== 'application/octet-stream') result.text = postData.toString();
449
+ if (content === 'attach') {
450
+ const sha1 = (0, _utils.calculateSha1)(postData) + '.' + (_utilsBundle.mime.getExtension(contentType) || 'dat');
451
+ if (this._options.includeTraceInfo) result._sha1 = sha1;else result._file = sha1;
452
+ this._delegate.onContentBlob(sha1, postData);
453
+ }
454
+ if (contentType === 'application/x-www-form-urlencoded') {
455
+ const parsed = new URLSearchParams(postData.toString());
456
+ for (const [name, value] of parsed.entries()) result.params.push({
457
+ name,
458
+ value
459
+ });
460
+ }
461
+ return result;
462
+ }
463
+ }
464
+ exports.HarTracer = HarTracer;
465
+ function createHarEntry(method, url, frameref, options) {
466
+ const harEntry = {
467
+ _frameref: options.includeTraceInfo ? frameref : undefined,
468
+ _monotonicTime: options.includeTraceInfo ? (0, _utils.monotonicTime)() : undefined,
469
+ startedDateTime: new Date().toISOString(),
470
+ time: -1,
471
+ request: {
472
+ method: method,
473
+ url: url.toString(),
474
+ httpVersion: FALLBACK_HTTP_VERSION,
475
+ cookies: [],
476
+ headers: [],
477
+ queryString: [...url.searchParams].map(e => ({
478
+ name: e[0],
479
+ value: e[1]
480
+ })),
481
+ headersSize: -1,
482
+ bodySize: -1
483
+ },
484
+ response: {
485
+ status: -1,
486
+ statusText: '',
487
+ httpVersion: FALLBACK_HTTP_VERSION,
488
+ cookies: [],
489
+ headers: [],
490
+ content: {
491
+ size: -1,
492
+ mimeType: 'x-unknown'
493
+ },
494
+ headersSize: -1,
495
+ bodySize: -1,
496
+ redirectURL: '',
497
+ _transferSize: options.omitSizes ? undefined : -1
498
+ },
499
+ cache: {},
500
+ timings: {
501
+ send: -1,
502
+ wait: -1,
503
+ receive: -1
504
+ }
505
+ };
506
+ return harEntry;
507
+ }
508
+ function parseCookie(c) {
509
+ const cookie = {
510
+ name: '',
511
+ value: ''
512
+ };
513
+ let first = true;
514
+ for (const pair of c.split(/; */)) {
515
+ const indexOfEquals = pair.indexOf('=');
516
+ const name = indexOfEquals !== -1 ? pair.substr(0, indexOfEquals).trim() : pair.trim();
517
+ const value = indexOfEquals !== -1 ? pair.substr(indexOfEquals + 1, pair.length).trim() : '';
518
+ if (first) {
519
+ first = false;
520
+ cookie.name = name;
521
+ cookie.value = value;
522
+ continue;
523
+ }
524
+ if (name === 'Domain') cookie.domain = value;
525
+ if (name === 'Expires') cookie.expires = safeDateToISOString(value);
526
+ if (name === 'HttpOnly') cookie.httpOnly = true;
527
+ if (name === 'Max-Age') cookie.expires = safeDateToISOString(Date.now() + +value * 1000);
528
+ if (name === 'Path') cookie.path = value;
529
+ if (name === 'SameSite') cookie.sameSite = value;
530
+ if (name === 'Secure') cookie.secure = true;
531
+ }
532
+ return cookie;
533
+ }
534
+ function safeDateToISOString(value) {
535
+ try {
536
+ return new Date(value).toISOString();
537
+ } catch (e) {}
538
+ }
539
+ const startedDateSymbol = Symbol('startedDate');