@checkly/playwright-core 1.48.22 → 1.51.11-beta.1

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 (307) hide show
  1. package/ThirdPartyNotices.txt +60 -48
  2. package/bin/reinstall_chrome_beta_linux.sh +10 -8
  3. package/bin/reinstall_chrome_stable_linux.sh +10 -8
  4. package/bin/reinstall_msedge_beta_linux.sh +17 -9
  5. package/bin/reinstall_msedge_dev_linux.sh +17 -9
  6. package/bin/reinstall_msedge_stable_linux.sh +17 -9
  7. package/browsers.json +33 -12
  8. package/index.js +0 -1
  9. package/lib/androidServerImpl.js +5 -5
  10. package/lib/browserServerImpl.js +18 -9
  11. package/lib/checkly/secretsFilter.js +1 -1
  12. package/lib/cli/driver.js +6 -6
  13. package/lib/cli/program.js +38 -43
  14. package/lib/cli/programWithTestStub.js +2 -1
  15. package/lib/client/android.js +37 -58
  16. package/lib/client/artifact.js +4 -7
  17. package/lib/client/browser.js +10 -15
  18. package/lib/client/browserContext.js +69 -62
  19. package/lib/client/browserType.js +30 -64
  20. package/lib/client/channelOwner.js +53 -68
  21. package/lib/client/clientHelper.js +4 -6
  22. package/lib/client/clientInstrumentation.js +2 -0
  23. package/lib/client/clientStackTrace.js +65 -0
  24. package/lib/client/connection.js +48 -41
  25. package/lib/client/consoleMessage.js +4 -7
  26. package/lib/client/electron.js +10 -10
  27. package/lib/client/elementHandle.js +32 -33
  28. package/lib/client/errors.js +2 -2
  29. package/lib/client/eventEmitter.js +5 -8
  30. package/lib/client/fetch.js +61 -62
  31. package/lib/client/fileUtils.js +31 -0
  32. package/lib/client/frame.js +22 -23
  33. package/lib/client/harRouter.js +7 -9
  34. package/lib/client/jsHandle.js +3 -4
  35. package/lib/client/localUtils.js +24 -0
  36. package/lib/client/locator.js +27 -13
  37. package/lib/client/network.js +44 -37
  38. package/lib/client/page.js +50 -41
  39. package/lib/client/platform.js +71 -0
  40. package/lib/client/playwright.js +21 -1
  41. package/lib/client/selectors.js +8 -2
  42. package/lib/client/stream.js +2 -21
  43. package/lib/client/timeoutSettings.js +65 -0
  44. package/lib/client/tracing.js +29 -7
  45. package/lib/client/video.js +2 -2
  46. package/lib/client/waiter.js +19 -16
  47. package/lib/client/webSocket.js +106 -0
  48. package/lib/client/worker.js +4 -4
  49. package/lib/client/writableStream.js +2 -21
  50. package/lib/generated/consoleApiSource.js +1 -1
  51. package/lib/generated/injectedScriptSource.js +1 -1
  52. package/lib/generated/pollingRecorderSource.js +1 -1
  53. package/lib/generated/recorderSource.js +7 -0
  54. package/lib/generated/webSocketMockSource.js +1 -1
  55. package/lib/inProcessFactory.js +8 -3
  56. package/lib/outofprocess.js +12 -8
  57. package/lib/protocol/debug.js +1 -1
  58. package/lib/protocol/validator.js +77 -11
  59. package/lib/protocol/validatorPrimitives.js +1 -2
  60. package/lib/remote/playwrightConnection.js +21 -12
  61. package/lib/remote/playwrightServer.js +22 -8
  62. package/lib/server/android/android.js +17 -14
  63. package/lib/server/android/backendAdb.js +14 -14
  64. package/lib/server/artifact.js +3 -3
  65. package/lib/server/bidi/bidiBrowser.js +11 -8
  66. package/lib/server/bidi/bidiChromium.js +5 -4
  67. package/lib/server/bidi/bidiConnection.js +1 -3
  68. package/lib/server/bidi/bidiExecutionContext.js +71 -25
  69. package/lib/server/bidi/bidiFirefox.js +4 -3
  70. package/lib/server/bidi/bidiInput.js +11 -12
  71. package/lib/server/bidi/bidiNetworkManager.js +11 -12
  72. package/lib/server/bidi/bidiOverCdp.js +2 -2
  73. package/lib/server/bidi/bidiPage.js +39 -66
  74. package/lib/server/bidi/third_party/bidiKeyboard.js +9 -7
  75. package/lib/server/bidi/third_party/firefoxPrefs.js +19 -3
  76. package/lib/server/browser.js +2 -2
  77. package/lib/server/browserContext.js +60 -55
  78. package/lib/server/browserType.js +22 -18
  79. package/lib/server/callLog.js +79 -0
  80. package/lib/server/chromium/chromium.js +20 -16
  81. package/lib/server/chromium/chromiumSwitches.js +32 -9
  82. package/lib/server/chromium/crBrowser.js +25 -24
  83. package/lib/server/chromium/crConnection.js +2 -2
  84. package/lib/server/chromium/crCoverage.js +1 -1
  85. package/lib/server/chromium/crDevTools.js +1 -1
  86. package/lib/server/chromium/crDragDrop.js +1 -1
  87. package/lib/server/chromium/crExecutionContext.js +26 -35
  88. package/lib/server/chromium/crInput.js +17 -6
  89. package/lib/server/chromium/crNetworkManager.js +3 -3
  90. package/lib/server/chromium/crPage.js +43 -55
  91. package/lib/server/chromium/crPdf.js +1 -1
  92. package/lib/server/chromium/crProtocolHelper.js +3 -3
  93. package/lib/server/chromium/crServiceWorker.js +2 -2
  94. package/lib/server/chromium/videoRecorder.js +2 -2
  95. package/lib/server/clock.js +1 -1
  96. package/lib/server/codegen/csharp.js +21 -16
  97. package/lib/server/codegen/java.js +17 -9
  98. package/lib/server/codegen/javascript.js +37 -7
  99. package/lib/server/codegen/jsonl.js +1 -1
  100. package/lib/server/codegen/languages.js +2 -2
  101. package/lib/server/codegen/python.js +14 -17
  102. package/lib/server/debugController.js +23 -45
  103. package/lib/server/debugger.js +1 -1
  104. package/lib/server/deviceDescriptors.js +1 -1
  105. package/lib/server/deviceDescriptorsSource.json +131 -131
  106. package/lib/server/dispatchers/androidDispatcher.js +14 -3
  107. package/lib/server/dispatchers/artifactDispatcher.js +3 -3
  108. package/lib/server/dispatchers/browserContextDispatcher.js +21 -32
  109. package/lib/server/dispatchers/browserTypeDispatcher.js +1 -1
  110. package/lib/server/dispatchers/cdpSessionDispatcher.js +1 -1
  111. package/lib/server/dispatchers/debugControllerDispatcher.js +5 -3
  112. package/lib/server/dispatchers/dispatcher.js +27 -25
  113. package/lib/server/dispatchers/electronDispatcher.js +3 -3
  114. package/lib/server/dispatchers/elementHandlerDispatcher.js +7 -2
  115. package/lib/server/dispatchers/frameDispatcher.js +9 -3
  116. package/lib/server/dispatchers/jsonPipeDispatcher.js +2 -2
  117. package/lib/server/dispatchers/localUtilsDispatcher.js +25 -291
  118. package/lib/server/dispatchers/networkDispatchers.js +3 -3
  119. package/lib/server/dispatchers/pageDispatcher.js +10 -8
  120. package/lib/server/dispatchers/playwrightDispatcher.js +5 -5
  121. package/lib/server/dispatchers/streamDispatcher.js +4 -3
  122. package/lib/server/dispatchers/tracingDispatcher.js +10 -0
  123. package/lib/server/dispatchers/webSocketRouteDispatcher.js +18 -18
  124. package/lib/server/dispatchers/writableStreamDispatcher.js +5 -6
  125. package/lib/server/dom.js +102 -48
  126. package/lib/server/download.js +1 -1
  127. package/lib/server/electron/electron.js +17 -16
  128. package/lib/server/errors.js +1 -1
  129. package/lib/server/fetch.js +49 -33
  130. package/lib/server/fileUploadUtils.js +7 -4
  131. package/lib/server/firefox/ffBrowser.js +17 -9
  132. package/lib/server/firefox/ffConnection.js +1 -1
  133. package/lib/server/firefox/ffExecutionContext.js +25 -29
  134. package/lib/server/firefox/ffInput.js +15 -4
  135. package/lib/server/firefox/ffNetworkManager.js +3 -3
  136. package/lib/server/firefox/ffPage.js +29 -50
  137. package/lib/server/firefox/firefox.js +7 -9
  138. package/lib/server/frameSelectors.js +1 -1
  139. package/lib/server/frames.js +65 -55
  140. package/lib/server/har/harRecorder.js +4 -4
  141. package/lib/server/har/harTracer.js +8 -9
  142. package/lib/server/harBackend.js +157 -0
  143. package/lib/server/helper.js +2 -2
  144. package/lib/server/index.js +1 -14
  145. package/lib/server/input.js +3 -4
  146. package/lib/server/instrumentation.js +2 -2
  147. package/lib/server/isomorphic/utilityScriptSerializers.js +3 -0
  148. package/lib/server/javascript.js +26 -33
  149. package/lib/server/launchApp.js +7 -7
  150. package/lib/server/localUtils.js +203 -0
  151. package/lib/server/network.js +5 -5
  152. package/lib/server/page.js +76 -37
  153. package/lib/server/pipeTransport.js +1 -1
  154. package/lib/server/playwright.js +5 -5
  155. package/lib/server/progress.js +1 -4
  156. package/lib/server/protocolError.js +1 -1
  157. package/lib/server/recorder/chat.js +177 -0
  158. package/lib/server/recorder/codeGenerator.js +154 -0
  159. package/lib/server/recorder/contextRecorder.js +9 -18
  160. package/lib/server/recorder/csharp.js +311 -0
  161. package/lib/server/recorder/java.js +249 -0
  162. package/lib/server/recorder/javascript.js +230 -0
  163. package/lib/server/recorder/jsonl.js +48 -0
  164. package/lib/server/recorder/language.js +45 -0
  165. package/lib/server/recorder/python.js +276 -0
  166. package/lib/server/recorder/recorderActions.js +6 -0
  167. package/lib/server/recorder/recorderApp.js +25 -28
  168. package/lib/server/recorder/recorderCollection.js +9 -21
  169. package/lib/server/recorder/recorderInTraceViewer.js +1 -1
  170. package/lib/server/recorder/recorderRunner.js +7 -3
  171. package/lib/server/recorder/recorderUtils.js +5 -28
  172. package/lib/server/recorder/throttledFile.js +3 -4
  173. package/lib/server/recorder/utils.js +46 -0
  174. package/lib/server/recorder.js +83 -43
  175. package/lib/server/registry/browserFetcher.js +10 -8
  176. package/lib/server/registry/dependencies.js +20 -20
  177. package/lib/server/registry/index.js +215 -61
  178. package/lib/server/registry/nativeDeps.js +10 -4
  179. package/lib/server/registry/oopDownloadBrowserMain.js +3 -3
  180. package/lib/server/screenshotter.js +1 -1
  181. package/lib/server/selectors.js +3 -3
  182. package/lib/server/socksClientCertificatesInterceptor.js +8 -8
  183. package/lib/server/socksInterceptor.js +8 -5
  184. package/lib/server/storageScript.js +160 -0
  185. package/lib/server/timeoutSettings.js +74 -0
  186. package/lib/server/trace/recorder/snapshotter.js +10 -20
  187. package/lib/server/trace/recorder/snapshotterInjected.js +26 -1
  188. package/lib/server/trace/recorder/tracing.js +108 -45
  189. package/lib/server/trace/test/inMemorySnapshotter.js +6 -6
  190. package/lib/server/trace/viewer/traceViewer.js +21 -7
  191. package/lib/server/transport.js +3 -2
  192. package/lib/server/utils/ascii.js +31 -0
  193. package/lib/server/utils/comparators.js +159 -0
  194. package/lib/server/utils/crypto.js +171 -0
  195. package/lib/server/utils/debug.js +38 -0
  196. package/lib/server/utils/debugLogger.js +93 -0
  197. package/lib/server/utils/env.js +53 -0
  198. package/lib/server/utils/eventsHelper.js +38 -0
  199. package/lib/server/utils/expectUtils.js +33 -0
  200. package/lib/server/utils/fileUtils.js +204 -0
  201. package/lib/server/utils/happyEyeballs.js +207 -0
  202. package/lib/server/utils/hostPlatform.js +145 -0
  203. package/lib/server/utils/httpServer.js +233 -0
  204. package/lib/server/utils/image_tools/colorUtils.js +98 -0
  205. package/lib/server/utils/image_tools/compare.js +108 -0
  206. package/lib/server/utils/image_tools/imageChannel.js +70 -0
  207. package/lib/server/utils/image_tools/stats.js +102 -0
  208. package/lib/server/utils/linuxUtils.js +58 -0
  209. package/lib/server/utils/network.js +160 -0
  210. package/lib/server/utils/nodePlatform.js +140 -0
  211. package/lib/server/utils/pipeTransport.js +82 -0
  212. package/lib/server/utils/processLauncher.js +248 -0
  213. package/lib/server/utils/profiler.js +52 -0
  214. package/lib/server/utils/socksProxy.js +570 -0
  215. package/lib/server/utils/spawnAsync.js +45 -0
  216. package/lib/server/utils/task.js +58 -0
  217. package/lib/server/utils/userAgent.js +91 -0
  218. package/lib/server/utils/wsServer.js +128 -0
  219. package/lib/server/utils/zipFile.js +75 -0
  220. package/lib/server/utils/zones.js +54 -0
  221. package/lib/server/webkit/webkit.js +4 -4
  222. package/lib/server/webkit/wkBrowser.js +10 -10
  223. package/lib/server/webkit/wkConnection.js +1 -1
  224. package/lib/server/webkit/wkExecutionContext.js +26 -30
  225. package/lib/server/webkit/wkInput.js +17 -7
  226. package/lib/server/webkit/wkInterceptableRequest.js +2 -2
  227. package/lib/server/webkit/wkPage.js +80 -66
  228. package/lib/server/webkit/wkProvisionalPage.js +1 -1
  229. package/lib/server/webkit/wkWorkers.js +2 -2
  230. package/lib/utils/fileUtils.js +1 -1
  231. package/lib/utils/glob.js +84 -0
  232. package/lib/utils/isomorphic/ariaSnapshot.js +392 -0
  233. package/lib/utils/isomorphic/assert.js +25 -0
  234. package/lib/utils/isomorphic/colors.js +65 -0
  235. package/lib/utils/isomorphic/cssParser.js +5 -5
  236. package/lib/utils/isomorphic/headers.js +52 -0
  237. package/lib/utils/isomorphic/locatorGenerators.js +38 -8
  238. package/lib/utils/isomorphic/locatorParser.js +19 -13
  239. package/lib/utils/isomorphic/manualPromise.js +107 -0
  240. package/lib/utils/isomorphic/mimeType.js +11 -3
  241. package/lib/utils/isomorphic/multimap.js +73 -0
  242. package/lib/utils/isomorphic/rtti.js +41 -0
  243. package/lib/utils/isomorphic/semaphore.js +51 -0
  244. package/lib/utils/isomorphic/stackTrace.js +169 -0
  245. package/lib/utils/isomorphic/stringUtils.js +31 -1
  246. package/lib/utils/isomorphic/time.js +25 -0
  247. package/lib/utils/isomorphic/timeoutRunner.js +66 -0
  248. package/lib/utils/isomorphic/traceUtils.js +23 -0
  249. package/lib/utils/isomorphic/types.js +5 -0
  250. package/lib/utils/isomorphic/urlMatch.js +7 -5
  251. package/lib/utils/mimeType.js +30 -0
  252. package/lib/utils/stackTrace.js +1 -1
  253. package/lib/utils.js +447 -0
  254. package/lib/utilsBundle.js +4 -29
  255. package/lib/utilsBundleImpl/index.js +188 -33
  256. package/lib/vite/htmlReport/index.html +20 -17
  257. package/lib/vite/recorder/assets/codeMirrorModule-B9YMkrwa.js +24 -0
  258. package/lib/vite/recorder/assets/index-ELPgmkwA.js +184 -0
  259. package/lib/vite/recorder/assets/{index-BW-aOBcL.css → index-eHBmevrY.css} +1 -1
  260. package/lib/vite/recorder/index.html +2 -2
  261. package/lib/vite/traceViewer/assets/codeMirrorModule-gU1OOCQO.js +24 -0
  262. package/lib/vite/traceViewer/assets/defaultSettingsView-B5n_FjMx.js +1 -0
  263. package/lib/vite/traceViewer/assets/inspectorTab-6Tru8Mn_.js +235 -0
  264. package/lib/vite/traceViewer/assets/workbench-B_Nj4NA2.js +25 -0
  265. package/lib/vite/traceViewer/assets/{xtermModule-BeNbaIVa.js → xtermModule-BoAIEibi.js} +7 -7
  266. package/lib/vite/traceViewer/defaultSettingsView.CO3FR0CX.css +1 -0
  267. package/lib/vite/traceViewer/{embedded.BkvOrz5Z.js → embedded.DpNPH6mk.js} +2 -2
  268. package/lib/vite/traceViewer/embedded.html +6 -6
  269. package/lib/vite/traceViewer/index.CuE3SYGw.js +2 -0
  270. package/lib/vite/traceViewer/index.html +27 -9
  271. package/lib/vite/traceViewer/inspectorTab.CXDulcFG.css +1 -0
  272. package/lib/vite/traceViewer/{recorder.DNMfnSiu.js → recorder.BD-uZJs7.js} +2 -2
  273. package/lib/vite/traceViewer/recorder.html +4 -4
  274. package/lib/vite/traceViewer/sw.bundle.js +3 -3
  275. package/lib/vite/traceViewer/{uiMode.CAYqod-m.css → uiMode.BatfzHMG.css} +1 -1
  276. package/lib/vite/traceViewer/uiMode.DHrNgddz.js +5 -0
  277. package/lib/vite/traceViewer/uiMode.html +8 -7
  278. package/lib/vite/traceViewer/workbench.B9vIAzH9.css +1 -0
  279. package/lib/zipBundleImpl.js +4 -4
  280. package/package.json +7 -7
  281. package/types/protocol.d.ts +820 -153
  282. package/types/types.d.ts +285 -82
  283. package/bin/PrintDeps.exe +0 -0
  284. package/bin/README.md +0 -2
  285. package/lib/vite/recorder/assets/codeMirrorModule-d0KhC1qL.js +0 -24
  286. package/lib/vite/recorder/assets/index-Bxxcmxlu.js +0 -42
  287. package/lib/vite/traceViewer/assets/codeMirrorModule-pBPtArIT.js +0 -24
  288. package/lib/vite/traceViewer/assets/codeMirrorModule-tzBrK1V4.js +0 -24
  289. package/lib/vite/traceViewer/assets/inspectorTab-BuJ3wAX_.js +0 -64
  290. package/lib/vite/traceViewer/assets/inspectorTab-Soeeuvzv.js +0 -64
  291. package/lib/vite/traceViewer/assets/testServerConnection-DeE2kSzz.js +0 -1
  292. package/lib/vite/traceViewer/assets/workbench-DdmJ9AJV.js +0 -9
  293. package/lib/vite/traceViewer/assets/workbench-lypYlf00.js +0 -9
  294. package/lib/vite/traceViewer/embedded.DInvAijy.js +0 -2
  295. package/lib/vite/traceViewer/index.Dha3cgqs.js +0 -2
  296. package/lib/vite/traceViewer/index._Iolt-uE.js +0 -2
  297. package/lib/vite/traceViewer/inspectorTab.DLjBDrQR.css +0 -1
  298. package/lib/vite/traceViewer/recorder.DTSaNaly.js +0 -2
  299. package/lib/vite/traceViewer/uiMode.BM7yhjzl.js +0 -5
  300. package/lib/vite/traceViewer/uiMode.Cr1tvTWS.js +0 -5
  301. package/lib/vite/traceViewer/workbench.DlsCx8k5.css +0 -1
  302. /package/lib/vite/recorder/assets/{codeMirrorModule-ez37Vkbh.css → codeMirrorModule-C3UTv-Ge.css} +0 -0
  303. /package/lib/vite/traceViewer/{codeMirrorModule.ez37Vkbh.css → codeMirrorModule.C3UTv-Ge.css} +0 -0
  304. /package/lib/vite/traceViewer/{embedded.w7WN2u1R.css → embedded.mLhjB5IF.css} +0 -0
  305. /package/lib/vite/traceViewer/{index.CrbWWHbf.css → index.CFOW-Ezb.css} +0 -0
  306. /package/lib/vite/traceViewer/{recorder.B_SY1GJM.css → recorder.tn0RQdqM.css} +0 -0
  307. /package/lib/vite/traceViewer/{xtermModule.DSXBckUd.css → xtermModule.Beg8tuEN.css} +0 -0
@@ -0,0 +1,159 @@
1
+ "use strict";
2
+
3
+ Object.defineProperty(exports, "__esModule", {
4
+ value: true
5
+ });
6
+ exports.compareBuffersOrStrings = compareBuffersOrStrings;
7
+ exports.getComparator = getComparator;
8
+ var _compare = require("./image_tools/compare");
9
+ var _pixelmatch = _interopRequireDefault(require("../../third_party/pixelmatch"));
10
+ var _utilsBundle = require("../../utilsBundle");
11
+ function _interopRequireDefault(e) { return e && e.__esModule ? e : { default: e }; }
12
+ /**
13
+ * Copyright 2017 Google Inc. All rights reserved.
14
+ * Modifications copyright (c) Microsoft Corporation.
15
+ *
16
+ * Licensed under the Apache License, Version 2.0 (the "License");
17
+ * you may not use this file except in compliance with the License.
18
+ * You may obtain a copy of the License at
19
+ *
20
+ * http://www.apache.org/licenses/LICENSE-2.0
21
+ *
22
+ * Unless required by applicable law or agreed to in writing, software
23
+ * distributed under the License is distributed on an "AS IS" BASIS,
24
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
25
+ * See the License for the specific language governing permissions and
26
+ * limitations under the License.
27
+ */
28
+
29
+ // @ts-ignore
30
+
31
+ function getComparator(mimeType) {
32
+ if (mimeType === 'image/png') return compareImages.bind(null, 'image/png');
33
+ if (mimeType === 'image/jpeg') return compareImages.bind(null, 'image/jpeg');
34
+ if (mimeType === 'text/plain') return compareText;
35
+ return compareBuffersOrStrings;
36
+ }
37
+ const JPEG_JS_MAX_BUFFER_SIZE_IN_MB = 5 * 1024; // ~5 GB
38
+
39
+ function compareBuffersOrStrings(actualBuffer, expectedBuffer) {
40
+ if (typeof actualBuffer === 'string') return compareText(actualBuffer, expectedBuffer);
41
+ if (!actualBuffer || !(actualBuffer instanceof Buffer)) return {
42
+ errorMessage: 'Actual result should be a Buffer or a string.'
43
+ };
44
+ if (Buffer.compare(actualBuffer, expectedBuffer)) return {
45
+ errorMessage: 'Buffers differ'
46
+ };
47
+ return null;
48
+ }
49
+ function compareImages(mimeType, actualBuffer, expectedBuffer, options = {}) {
50
+ var _options$comparator, _ref;
51
+ if (!actualBuffer || !(actualBuffer instanceof Buffer)) return {
52
+ errorMessage: 'Actual result should be a Buffer.'
53
+ };
54
+ validateBuffer(expectedBuffer, mimeType);
55
+ let actual = mimeType === 'image/png' ? _utilsBundle.PNG.sync.read(actualBuffer) : _utilsBundle.jpegjs.decode(actualBuffer, {
56
+ maxMemoryUsageInMB: JPEG_JS_MAX_BUFFER_SIZE_IN_MB
57
+ });
58
+ let expected = mimeType === 'image/png' ? _utilsBundle.PNG.sync.read(expectedBuffer) : _utilsBundle.jpegjs.decode(expectedBuffer, {
59
+ maxMemoryUsageInMB: JPEG_JS_MAX_BUFFER_SIZE_IN_MB
60
+ });
61
+ const size = {
62
+ width: Math.max(expected.width, actual.width),
63
+ height: Math.max(expected.height, actual.height)
64
+ };
65
+ let sizesMismatchError = '';
66
+ if (expected.width !== actual.width || expected.height !== actual.height) {
67
+ sizesMismatchError = `Expected an image ${expected.width}px by ${expected.height}px, received ${actual.width}px by ${actual.height}px. `;
68
+ actual = resizeImage(actual, size);
69
+ expected = resizeImage(expected, size);
70
+ }
71
+ const diff = new _utilsBundle.PNG({
72
+ width: size.width,
73
+ height: size.height
74
+ });
75
+ let count;
76
+ if (options.comparator === 'ssim-cie94') {
77
+ count = (0, _compare.compare)(expected.data, actual.data, diff.data, size.width, size.height, {
78
+ // All ΔE* formulae are originally designed to have the difference of 1.0 stand for a "just noticeable difference" (JND).
79
+ // See https://en.wikipedia.org/wiki/Color_difference#CIELAB_%CE%94E*
80
+ maxColorDeltaE94: 1.0
81
+ });
82
+ } else if (((_options$comparator = options.comparator) !== null && _options$comparator !== void 0 ? _options$comparator : 'pixelmatch') === 'pixelmatch') {
83
+ var _options$threshold;
84
+ count = (0, _pixelmatch.default)(expected.data, actual.data, diff.data, size.width, size.height, {
85
+ threshold: (_options$threshold = options.threshold) !== null && _options$threshold !== void 0 ? _options$threshold : 0.2
86
+ });
87
+ } else {
88
+ throw new Error(`Configuration specifies unknown comparator "${options.comparator}"`);
89
+ }
90
+ const maxDiffPixels1 = options.maxDiffPixels;
91
+ const maxDiffPixels2 = options.maxDiffPixelRatio !== undefined ? expected.width * expected.height * options.maxDiffPixelRatio : undefined;
92
+ let maxDiffPixels;
93
+ if (maxDiffPixels1 !== undefined && maxDiffPixels2 !== undefined) maxDiffPixels = Math.min(maxDiffPixels1, maxDiffPixels2);else maxDiffPixels = (_ref = maxDiffPixels1 !== null && maxDiffPixels1 !== void 0 ? maxDiffPixels1 : maxDiffPixels2) !== null && _ref !== void 0 ? _ref : 0;
94
+ const ratio = Math.ceil(count / (expected.width * expected.height) * 100) / 100;
95
+ const pixelsMismatchError = count > maxDiffPixels ? `${count} pixels (ratio ${ratio.toFixed(2)} of all image pixels) are different.` : '';
96
+ if (pixelsMismatchError || sizesMismatchError) return {
97
+ errorMessage: sizesMismatchError + pixelsMismatchError,
98
+ diff: _utilsBundle.PNG.sync.write(diff)
99
+ };
100
+ return null;
101
+ }
102
+ function validateBuffer(buffer, mimeType) {
103
+ if (mimeType === 'image/png') {
104
+ const pngMagicNumber = [137, 80, 78, 71, 13, 10, 26, 10];
105
+ if (buffer.length < pngMagicNumber.length || !pngMagicNumber.every((byte, index) => buffer[index] === byte)) throw new Error('could not decode image as PNG.');
106
+ } else if (mimeType === 'image/jpeg') {
107
+ const jpegMagicNumber = [255, 216];
108
+ if (buffer.length < jpegMagicNumber.length || !jpegMagicNumber.every((byte, index) => buffer[index] === byte)) throw new Error('could not decode image as JPEG.');
109
+ }
110
+ }
111
+ function compareText(actual, expectedBuffer) {
112
+ if (typeof actual !== 'string') return {
113
+ errorMessage: 'Actual result should be a string'
114
+ };
115
+ let expected = expectedBuffer.toString('utf-8');
116
+ if (expected === actual) return null;
117
+ // Eliminate '\'
118
+ if (!actual.endsWith('\n')) actual += '\n';
119
+ if (!expected.endsWith('\n')) expected += '\n';
120
+ const lines = _utilsBundle.diff.createPatch('file', expected, actual, undefined, undefined, {
121
+ context: 5
122
+ }).split('\n');
123
+ const coloredLines = lines.slice(4).map(line => {
124
+ if (line.startsWith('-')) return _utilsBundle.colors.red(line);
125
+ if (line.startsWith('+')) return _utilsBundle.colors.green(line);
126
+ if (line.startsWith('@@')) return _utilsBundle.colors.dim(line);
127
+ return line;
128
+ });
129
+ const errorMessage = coloredLines.join('\n');
130
+ return {
131
+ errorMessage
132
+ };
133
+ }
134
+ function resizeImage(image, size) {
135
+ if (image.width === size.width && image.height === size.height) return image;
136
+ const buffer = new Uint8Array(size.width * size.height * 4);
137
+ for (let y = 0; y < size.height; y++) {
138
+ for (let x = 0; x < size.width; x++) {
139
+ const to = (y * size.width + x) * 4;
140
+ if (y < image.height && x < image.width) {
141
+ const from = (y * image.width + x) * 4;
142
+ buffer[to] = image.data[from];
143
+ buffer[to + 1] = image.data[from + 1];
144
+ buffer[to + 2] = image.data[from + 2];
145
+ buffer[to + 3] = image.data[from + 3];
146
+ } else {
147
+ buffer[to] = 0;
148
+ buffer[to + 1] = 0;
149
+ buffer[to + 2] = 0;
150
+ buffer[to + 3] = 0;
151
+ }
152
+ }
153
+ }
154
+ return {
155
+ data: Buffer.from(buffer),
156
+ width: size.width,
157
+ height: size.height
158
+ };
159
+ }
@@ -0,0 +1,171 @@
1
+ "use strict";
2
+
3
+ Object.defineProperty(exports, "__esModule", {
4
+ value: true
5
+ });
6
+ exports.calculateSha1 = calculateSha1;
7
+ exports.createGuid = createGuid;
8
+ exports.generateSelfSignedCertificate = generateSelfSignedCertificate;
9
+ var _crypto = _interopRequireDefault(require("crypto"));
10
+ var _assert = require("../../utils/isomorphic/assert");
11
+ function _interopRequireDefault(e) { return e && e.__esModule ? e : { default: e }; }
12
+ /**
13
+ * Copyright (c) Microsoft Corporation.
14
+ *
15
+ * Licensed under the Apache License, Version 2.0 (the "License");
16
+ * you may not use this file except in compliance with the License.
17
+ * You may obtain a copy of the License at
18
+ *
19
+ * http://www.apache.org/licenses/LICENSE-2.0
20
+ *
21
+ * Unless required by applicable law or agreed to in writing, software
22
+ * distributed under the License is distributed on an "AS IS" BASIS,
23
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
24
+ * See the License for the specific language governing permissions and
25
+ * limitations under the License.
26
+ */
27
+
28
+ function createGuid() {
29
+ return _crypto.default.randomBytes(16).toString('hex');
30
+ }
31
+ function calculateSha1(buffer) {
32
+ const hash = _crypto.default.createHash('sha1');
33
+ hash.update(buffer);
34
+ return hash.digest('hex');
35
+ }
36
+
37
+ // Variable-length quantity encoding aka. base-128 encoding
38
+ function encodeBase128(value) {
39
+ const bytes = [];
40
+ do {
41
+ let byte = value & 0x7f;
42
+ value >>>= 7;
43
+ if (bytes.length > 0) byte |= 0x80;
44
+ bytes.push(byte);
45
+ } while (value > 0);
46
+ return Buffer.from(bytes.reverse());
47
+ }
48
+
49
+ // ASN1/DER Speficiation: https://www.itu.int/rec/T-REC-X.680-X.693-202102-I/en
50
+ class DER {
51
+ static encodeSequence(data) {
52
+ return this._encode(0x30, Buffer.concat(data));
53
+ }
54
+ static encodeInteger(data) {
55
+ (0, _assert.assert)(data >= -128 && data <= 127);
56
+ return this._encode(0x02, Buffer.from([data]));
57
+ }
58
+ static encodeObjectIdentifier(oid) {
59
+ const parts = oid.split('.').map(v => Number(v));
60
+ // Encode the second part, which could be large, using base-128 encoding if necessary
61
+ const output = [encodeBase128(40 * parts[0] + parts[1])];
62
+ for (let i = 2; i < parts.length; i++) output.push(encodeBase128(parts[i]));
63
+ return this._encode(0x06, Buffer.concat(output));
64
+ }
65
+ static encodeNull() {
66
+ return Buffer.from([0x05, 0x00]);
67
+ }
68
+ static encodeSet(data) {
69
+ (0, _assert.assert)(data.length === 1, 'Only one item in the set is supported. We\'d need to sort the data to support more.');
70
+ // We expect the data to be already sorted.
71
+ return this._encode(0x31, Buffer.concat(data));
72
+ }
73
+ static encodeExplicitContextDependent(tag, data) {
74
+ return this._encode(0xa0 + tag, data);
75
+ }
76
+ static encodePrintableString(data) {
77
+ return this._encode(0x13, Buffer.from(data));
78
+ }
79
+ static encodeBitString(data) {
80
+ // The first byte of the content is the number of unused bits at the end
81
+ const unusedBits = 0; // Assuming all bits are used
82
+ const content = Buffer.concat([Buffer.from([unusedBits]), data]);
83
+ return this._encode(0x03, content);
84
+ }
85
+ static encodeDate(date) {
86
+ const year = date.getUTCFullYear();
87
+ const isGeneralizedTime = year >= 2050;
88
+ const parts = [isGeneralizedTime ? year.toString() : year.toString().slice(-2), (date.getUTCMonth() + 1).toString().padStart(2, '0'), date.getUTCDate().toString().padStart(2, '0'), date.getUTCHours().toString().padStart(2, '0'), date.getUTCMinutes().toString().padStart(2, '0'), date.getUTCSeconds().toString().padStart(2, '0')];
89
+ const encodedDate = parts.join('') + 'Z';
90
+ const tag = isGeneralizedTime ? 0x18 : 0x17; // 0x18 for GeneralizedTime, 0x17 for UTCTime
91
+ return this._encode(tag, Buffer.from(encodedDate));
92
+ }
93
+ static _encode(tag, data) {
94
+ const lengthBytes = this._encodeLength(data.length);
95
+ return Buffer.concat([Buffer.from([tag]), lengthBytes, data]);
96
+ }
97
+ static _encodeLength(length) {
98
+ if (length < 128) {
99
+ return Buffer.from([length]);
100
+ } else {
101
+ const lengthBytes = [];
102
+ while (length > 0) {
103
+ lengthBytes.unshift(length & 0xFF);
104
+ length >>= 8;
105
+ }
106
+ return Buffer.from([0x80 | lengthBytes.length, ...lengthBytes]);
107
+ }
108
+ }
109
+ }
110
+
111
+ // X.509 Specification: https://datatracker.ietf.org/doc/html/rfc2459#section-4.1
112
+ function generateSelfSignedCertificate() {
113
+ const {
114
+ privateKey,
115
+ publicKey
116
+ } = _crypto.default.generateKeyPairSync('rsa', {
117
+ modulusLength: 2048
118
+ });
119
+ const publicKeyDer = publicKey.export({
120
+ type: 'pkcs1',
121
+ format: 'der'
122
+ });
123
+ const oneYearInMilliseconds = 365 * 24 * 60 * 60 * 1_000;
124
+ const notBefore = new Date(new Date().getTime() - oneYearInMilliseconds);
125
+ const notAfter = new Date(new Date().getTime() + oneYearInMilliseconds);
126
+
127
+ // List of fields / structure: https://datatracker.ietf.org/doc/html/rfc2459#section-4.1
128
+ const tbsCertificate = DER.encodeSequence([DER.encodeExplicitContextDependent(0, DER.encodeInteger(1)),
129
+ // version
130
+ DER.encodeInteger(1),
131
+ // serialNumber
132
+ DER.encodeSequence([DER.encodeObjectIdentifier('1.2.840.113549.1.1.11'),
133
+ // sha256WithRSAEncryption PKCS #1
134
+ DER.encodeNull()]),
135
+ // signature
136
+ DER.encodeSequence([DER.encodeSet([DER.encodeSequence([DER.encodeObjectIdentifier('2.5.4.3'),
137
+ // commonName X.520 DN component
138
+ DER.encodePrintableString('localhost')])]), DER.encodeSet([DER.encodeSequence([DER.encodeObjectIdentifier('2.5.4.10'),
139
+ // organizationName X.520 DN component
140
+ DER.encodePrintableString('Playwright Client Certificate Support')])])]),
141
+ // issuer
142
+ DER.encodeSequence([DER.encodeDate(notBefore),
143
+ // notBefore
144
+ DER.encodeDate(notAfter) // notAfter
145
+ ]),
146
+ // validity
147
+ DER.encodeSequence([DER.encodeSet([DER.encodeSequence([DER.encodeObjectIdentifier('2.5.4.3'),
148
+ // commonName X.520 DN component
149
+ DER.encodePrintableString('localhost')])]), DER.encodeSet([DER.encodeSequence([DER.encodeObjectIdentifier('2.5.4.10'),
150
+ // organizationName X.520 DN component
151
+ DER.encodePrintableString('Playwright Client Certificate Support')])])]),
152
+ // subject
153
+ DER.encodeSequence([DER.encodeSequence([DER.encodeObjectIdentifier('1.2.840.113549.1.1.1'),
154
+ // rsaEncryption PKCS #1
155
+ DER.encodeNull()]), DER.encodeBitString(publicKeyDer)]) // SubjectPublicKeyInfo
156
+ ]);
157
+ const signature = _crypto.default.sign('sha256', tbsCertificate, privateKey);
158
+ const certificate = DER.encodeSequence([tbsCertificate, DER.encodeSequence([DER.encodeObjectIdentifier('1.2.840.113549.1.1.11'),
159
+ // sha256WithRSAEncryption PKCS #1
160
+ DER.encodeNull()]), DER.encodeBitString(signature)]);
161
+ const certPem = ['-----BEGIN CERTIFICATE-----',
162
+ // Split the base64 string into lines of 64 characters
163
+ certificate.toString('base64').match(/.{1,64}/g).join('\n'), '-----END CERTIFICATE-----'].join('\n');
164
+ return {
165
+ cert: certPem,
166
+ key: privateKey.export({
167
+ type: 'pkcs1',
168
+ format: 'pem'
169
+ })
170
+ };
171
+ }
@@ -0,0 +1,38 @@
1
+ "use strict";
2
+
3
+ Object.defineProperty(exports, "__esModule", {
4
+ value: true
5
+ });
6
+ exports.debugMode = debugMode;
7
+ exports.isUnderTest = isUnderTest;
8
+ exports.setUnderTest = setUnderTest;
9
+ var _env = require("./env");
10
+ /**
11
+ * Copyright (c) Microsoft Corporation.
12
+ *
13
+ * Licensed under the Apache License, Version 2.0 (the "License");
14
+ * you may not use this file except in compliance with the License.
15
+ * You may obtain a copy of the License at
16
+ *
17
+ * http://www.apache.org/licenses/LICENSE-2.0
18
+ *
19
+ * Unless required by applicable law or agreed to in writing, software
20
+ * distributed under the License is distributed on an "AS IS" BASIS,
21
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
22
+ * See the License for the specific language governing permissions and
23
+ * limitations under the License.
24
+ */
25
+
26
+ const _debugMode = (0, _env.getFromENV)('PWDEBUG') || '';
27
+ function debugMode() {
28
+ if (_debugMode === 'console') return 'console';
29
+ if (_debugMode === '0' || _debugMode === 'false') return '';
30
+ return _debugMode ? 'inspector' : '';
31
+ }
32
+ let _isUnderTest = !!process.env.PWTEST_UNDER_TEST;
33
+ function setUnderTest() {
34
+ _isUnderTest = true;
35
+ }
36
+ function isUnderTest() {
37
+ return _isUnderTest;
38
+ }
@@ -0,0 +1,93 @@
1
+ "use strict";
2
+
3
+ Object.defineProperty(exports, "__esModule", {
4
+ value: true
5
+ });
6
+ exports.debugLogger = exports.RecentLogsCollector = void 0;
7
+ var _fs = _interopRequireDefault(require("fs"));
8
+ var _utilsBundle = require("../../utilsBundle");
9
+ function _interopRequireDefault(e) { return e && e.__esModule ? e : { default: e }; }
10
+ /**
11
+ * Copyright (c) Microsoft Corporation.
12
+ *
13
+ * Licensed under the Apache License, Version 2.0 (the "License");
14
+ * you may not use this file except in compliance with the License.
15
+ * You may obtain a copy of the License at
16
+ *
17
+ * http://www.apache.org/licenses/LICENSE-2.0
18
+ *
19
+ * Unless required by applicable law or agreed to in writing, software
20
+ * distributed under the License is distributed on an "AS IS" BASIS,
21
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
22
+ * See the License for the specific language governing permissions and
23
+ * limitations under the License.
24
+ */
25
+
26
+ const debugLoggerColorMap = {
27
+ 'api': 45,
28
+ // cyan
29
+ 'protocol': 34,
30
+ // green
31
+ 'install': 34,
32
+ // green
33
+ 'download': 34,
34
+ // green
35
+ 'browser': 0,
36
+ // reset
37
+ 'socks': 92,
38
+ // purple
39
+ 'client-certificates': 92,
40
+ // purple
41
+ 'error': 160,
42
+ // red,
43
+ 'channel': 33,
44
+ // blue
45
+ 'server': 45,
46
+ // cyan
47
+ 'server:channel': 34,
48
+ // green
49
+ 'server:metadata': 33,
50
+ // blue,
51
+ 'recorder': 45 // cyan
52
+ };
53
+ class DebugLogger {
54
+ constructor() {
55
+ this._debuggers = new Map();
56
+ if (process.env.DEBUG_FILE) {
57
+ const ansiRegex = new RegExp(['[\\u001B\\u009B][[\\]()#;?]*(?:(?:(?:[a-zA-Z\\d]*(?:;[-a-zA-Z\\d\\/#&.:=?%@~_]*)*)?\\u0007)', '(?:(?:\\d{1,4}(?:;\\d{0,4})*)?[\\dA-PR-TZcf-ntqry=><~]))'].join('|'), 'g');
58
+ const stream = _fs.default.createWriteStream(process.env.DEBUG_FILE);
59
+ _utilsBundle.debug.log = data => {
60
+ stream.write(data.replace(ansiRegex, ''));
61
+ stream.write('\n');
62
+ };
63
+ }
64
+ }
65
+ log(name, message) {
66
+ let cachedDebugger = this._debuggers.get(name);
67
+ if (!cachedDebugger) {
68
+ cachedDebugger = (0, _utilsBundle.debug)(`pw:${name}`);
69
+ this._debuggers.set(name, cachedDebugger);
70
+ cachedDebugger.color = debugLoggerColorMap[name] || 0;
71
+ }
72
+ cachedDebugger(message);
73
+ }
74
+ isEnabled(name) {
75
+ return _utilsBundle.debug.enabled(`pw:${name}`);
76
+ }
77
+ }
78
+ const debugLogger = exports.debugLogger = new DebugLogger();
79
+ const kLogCount = 150;
80
+ class RecentLogsCollector {
81
+ constructor() {
82
+ this._logs = [];
83
+ }
84
+ log(message) {
85
+ this._logs.push(message);
86
+ if (this._logs.length === kLogCount * 2) this._logs.splice(0, kLogCount);
87
+ }
88
+ recentLogs() {
89
+ if (this._logs.length > kLogCount) return this._logs.slice(-kLogCount);
90
+ return this._logs;
91
+ }
92
+ }
93
+ exports.RecentLogsCollector = RecentLogsCollector;
@@ -0,0 +1,53 @@
1
+ "use strict";
2
+
3
+ Object.defineProperty(exports, "__esModule", {
4
+ value: true
5
+ });
6
+ exports.getAsBooleanFromENV = getAsBooleanFromENV;
7
+ exports.getFromENV = getFromENV;
8
+ exports.getPackageManager = getPackageManager;
9
+ exports.getPackageManagerExecCommand = getPackageManagerExecCommand;
10
+ exports.isLikelyNpxGlobal = isLikelyNpxGlobal;
11
+ /**
12
+ * Copyright (c) Microsoft Corporation.
13
+ *
14
+ * Licensed under the Apache License, Version 2.0 (the "License");
15
+ * you may not use this file except in compliance with the License.
16
+ * You may obtain a copy of the License at
17
+ *
18
+ * http://www.apache.org/licenses/LICENSE-2.0
19
+ *
20
+ * Unless required by applicable law or agreed to in writing, software
21
+ * distributed under the License is distributed on an "AS IS" BASIS,
22
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
23
+ * See the License for the specific language governing permissions and
24
+ * limitations under the License.
25
+ */
26
+
27
+ function getFromENV(name) {
28
+ let value = process.env[name];
29
+ value = value === undefined ? process.env[`npm_config_${name.toLowerCase()}`] : value;
30
+ value = value === undefined ? process.env[`npm_package_config_${name.toLowerCase()}`] : value;
31
+ return value;
32
+ }
33
+ function getAsBooleanFromENV(name, defaultValue) {
34
+ const value = getFromENV(name);
35
+ if (value === 'false' || value === '0') return false;
36
+ if (value) return true;
37
+ return !!defaultValue;
38
+ }
39
+ function getPackageManager() {
40
+ const env = process.env.npm_config_user_agent || '';
41
+ if (env.includes('yarn')) return 'yarn';
42
+ if (env.includes('pnpm')) return 'pnpm';
43
+ return 'npm';
44
+ }
45
+ function getPackageManagerExecCommand() {
46
+ const packageManager = getPackageManager();
47
+ if (packageManager === 'yarn') return 'yarn';
48
+ if (packageManager === 'pnpm') return 'pnpm exec';
49
+ return 'npx';
50
+ }
51
+ function isLikelyNpxGlobal() {
52
+ return process.argv.length >= 2 && process.argv[1].includes('_npx');
53
+ }
@@ -0,0 +1,38 @@
1
+ "use strict";
2
+
3
+ Object.defineProperty(exports, "__esModule", {
4
+ value: true
5
+ });
6
+ exports.eventsHelper = void 0;
7
+ /**
8
+ * Copyright 2017 Google Inc. All rights reserved.
9
+ * Modifications copyright (c) Microsoft Corporation.
10
+ *
11
+ * Licensed under the Apache License, Version 2.0 (the "License");
12
+ * you may not use this file except in compliance with the License.
13
+ * You may obtain a copy of the License at
14
+ *
15
+ * http://www.apache.org/licenses/LICENSE-2.0
16
+ *
17
+ * Unless required by applicable law or agreed to in writing, software
18
+ * distributed under the License is distributed on an "AS IS" BASIS,
19
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
20
+ * See the License for the specific language governing permissions and
21
+ * limitations under the License.
22
+ */
23
+
24
+ class EventsHelper {
25
+ static addEventListener(emitter, eventName, handler) {
26
+ emitter.on(eventName, handler);
27
+ return {
28
+ emitter,
29
+ eventName,
30
+ handler
31
+ };
32
+ }
33
+ static removeEventListeners(listeners) {
34
+ for (const listener of listeners) listener.emitter.removeListener(listener.eventName, listener.handler);
35
+ listeners.splice(0, listeners.length);
36
+ }
37
+ }
38
+ const eventsHelper = exports.eventsHelper = EventsHelper;
@@ -0,0 +1,33 @@
1
+ "use strict";
2
+
3
+ Object.defineProperty(exports, "__esModule", {
4
+ value: true
5
+ });
6
+ exports.serializeExpectedTextValues = serializeExpectedTextValues;
7
+ var _rtti = require("../../utils/isomorphic/rtti");
8
+ /**
9
+ * Copyright (c) Microsoft Corporation.
10
+ *
11
+ * Licensed under the Apache License, Version 2.0 (the "License");
12
+ * you may not use this file except in compliance with the License.
13
+ * You may obtain a copy of the License at
14
+ *
15
+ * http://www.apache.org/licenses/LICENSE-2.0
16
+ *
17
+ * Unless required by applicable law or agreed to in writing, software
18
+ * distributed under the License is distributed on an "AS IS" BASIS,
19
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
20
+ * See the License for the specific language governing permissions and
21
+ * limitations under the License.
22
+ */
23
+
24
+ function serializeExpectedTextValues(items, options = {}) {
25
+ return items.map(i => ({
26
+ string: (0, _rtti.isString)(i) ? i : undefined,
27
+ regexSource: (0, _rtti.isRegExp)(i) ? i.source : undefined,
28
+ regexFlags: (0, _rtti.isRegExp)(i) ? i.flags : undefined,
29
+ matchSubstring: options.matchSubstring,
30
+ ignoreCase: options.ignoreCase,
31
+ normalizeWhiteSpace: options.normalizeWhiteSpace
32
+ }));
33
+ }