playwright-codegen-pro-core 1.0.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 (468) hide show
  1. package/LICENSE +202 -0
  2. package/NOTICE +5 -0
  3. package/README.md +3 -0
  4. package/ThirdPartyNotices.txt +3552 -0
  5. package/bin/install_media_pack.ps1 +5 -0
  6. package/bin/install_webkit_wsl.ps1 +33 -0
  7. package/bin/reinstall_chrome_beta_linux.sh +42 -0
  8. package/bin/reinstall_chrome_beta_mac.sh +13 -0
  9. package/bin/reinstall_chrome_beta_win.ps1 +24 -0
  10. package/bin/reinstall_chrome_stable_linux.sh +42 -0
  11. package/bin/reinstall_chrome_stable_mac.sh +12 -0
  12. package/bin/reinstall_chrome_stable_win.ps1 +24 -0
  13. package/bin/reinstall_msedge_beta_linux.sh +48 -0
  14. package/bin/reinstall_msedge_beta_mac.sh +11 -0
  15. package/bin/reinstall_msedge_beta_win.ps1 +23 -0
  16. package/bin/reinstall_msedge_dev_linux.sh +48 -0
  17. package/bin/reinstall_msedge_dev_mac.sh +11 -0
  18. package/bin/reinstall_msedge_dev_win.ps1 +23 -0
  19. package/bin/reinstall_msedge_stable_linux.sh +48 -0
  20. package/bin/reinstall_msedge_stable_mac.sh +11 -0
  21. package/bin/reinstall_msedge_stable_win.ps1 +24 -0
  22. package/browsers.json +81 -0
  23. package/bundles/mcp/package-lock.json +1135 -0
  24. package/bundles/mcp/package.json +10 -0
  25. package/bundles/mcp/raw-body.ts +43 -0
  26. package/bundles/utils/package-lock.json +466 -0
  27. package/bundles/utils/package.json +40 -0
  28. package/bundles/zip/package-lock.json +257 -0
  29. package/bundles/zip/package.json +16 -0
  30. package/cli.js +18 -0
  31. package/index.d.ts +17 -0
  32. package/index.js +32 -0
  33. package/index.mjs +28 -0
  34. package/lib/androidServerImpl.js +65 -0
  35. package/lib/browserServerImpl.js +120 -0
  36. package/lib/cli/driver.js +98 -0
  37. package/lib/cli/program.js +599 -0
  38. package/lib/cli/programWithTestStub.js +74 -0
  39. package/lib/client/android.js +361 -0
  40. package/lib/client/api.js +137 -0
  41. package/lib/client/artifact.js +79 -0
  42. package/lib/client/browser.js +169 -0
  43. package/lib/client/browserContext.js +563 -0
  44. package/lib/client/browserType.js +153 -0
  45. package/lib/client/cdpSession.js +55 -0
  46. package/lib/client/channelOwner.js +194 -0
  47. package/lib/client/clientHelper.js +64 -0
  48. package/lib/client/clientInstrumentation.js +55 -0
  49. package/lib/client/clientStackTrace.js +69 -0
  50. package/lib/client/clock.js +68 -0
  51. package/lib/client/connect.js +143 -0
  52. package/lib/client/connection.js +322 -0
  53. package/lib/client/consoleMessage.js +61 -0
  54. package/lib/client/coverage.js +44 -0
  55. package/lib/client/debugger.js +57 -0
  56. package/lib/client/dialog.js +56 -0
  57. package/lib/client/disposable.js +76 -0
  58. package/lib/client/download.js +62 -0
  59. package/lib/client/electron.js +138 -0
  60. package/lib/client/elementHandle.js +281 -0
  61. package/lib/client/errors.js +77 -0
  62. package/lib/client/eventEmitter.js +314 -0
  63. package/lib/client/events.js +103 -0
  64. package/lib/client/fetch.js +367 -0
  65. package/lib/client/fileChooser.js +46 -0
  66. package/lib/client/fileUtils.js +34 -0
  67. package/lib/client/frame.js +404 -0
  68. package/lib/client/harRouter.js +99 -0
  69. package/lib/client/input.js +84 -0
  70. package/lib/client/jsHandle.js +105 -0
  71. package/lib/client/jsonPipe.js +39 -0
  72. package/lib/client/localUtils.js +60 -0
  73. package/lib/client/locator.js +367 -0
  74. package/lib/client/network.js +750 -0
  75. package/lib/client/page.js +729 -0
  76. package/lib/client/platform.js +77 -0
  77. package/lib/client/playwright.js +71 -0
  78. package/lib/client/screencast.js +48 -0
  79. package/lib/client/selectors.js +57 -0
  80. package/lib/client/stream.js +39 -0
  81. package/lib/client/timeoutSettings.js +79 -0
  82. package/lib/client/tracing.js +126 -0
  83. package/lib/client/types.js +28 -0
  84. package/lib/client/video.js +68 -0
  85. package/lib/client/waiter.js +142 -0
  86. package/lib/client/webError.js +39 -0
  87. package/lib/client/worker.js +85 -0
  88. package/lib/client/writableStream.js +39 -0
  89. package/lib/generated/bindingsControllerSource.js +28 -0
  90. package/lib/generated/clockSource.js +28 -0
  91. package/lib/generated/injectedScriptSource.js +28 -0
  92. package/lib/generated/pollingRecorderSource.js +28 -0
  93. package/lib/generated/storageScriptSource.js +28 -0
  94. package/lib/generated/utilityScriptSource.js +28 -0
  95. package/lib/generated/webSocketMockSource.js +336 -0
  96. package/lib/inProcessFactory.js +60 -0
  97. package/lib/inprocess.js +3 -0
  98. package/lib/mcpBundle.js +81 -0
  99. package/lib/mcpBundleImpl/index.js +91 -0
  100. package/lib/outofprocess.js +76 -0
  101. package/lib/protocol/serializers.js +197 -0
  102. package/lib/protocol/validator.js +3035 -0
  103. package/lib/protocol/validatorPrimitives.js +193 -0
  104. package/lib/remote/playwrightConnection.js +131 -0
  105. package/lib/remote/playwrightPipeServer.js +100 -0
  106. package/lib/remote/playwrightServer.js +339 -0
  107. package/lib/remote/playwrightWebSocketServer.js +73 -0
  108. package/lib/remote/serverTransport.js +96 -0
  109. package/lib/server/android/android.js +465 -0
  110. package/lib/server/android/backendAdb.js +177 -0
  111. package/lib/server/artifact.js +127 -0
  112. package/lib/server/bidi/bidiBrowser.js +571 -0
  113. package/lib/server/bidi/bidiChromium.js +162 -0
  114. package/lib/server/bidi/bidiConnection.js +213 -0
  115. package/lib/server/bidi/bidiDeserializer.js +116 -0
  116. package/lib/server/bidi/bidiExecutionContext.js +267 -0
  117. package/lib/server/bidi/bidiFirefox.js +128 -0
  118. package/lib/server/bidi/bidiInput.js +146 -0
  119. package/lib/server/bidi/bidiNetworkManager.js +411 -0
  120. package/lib/server/bidi/bidiOverCdp.js +102 -0
  121. package/lib/server/bidi/bidiPage.js +599 -0
  122. package/lib/server/bidi/bidiPdf.js +106 -0
  123. package/lib/server/bidi/third_party/bidiCommands.d.js +22 -0
  124. package/lib/server/bidi/third_party/bidiKeyboard.js +256 -0
  125. package/lib/server/bidi/third_party/bidiProtocol.js +24 -0
  126. package/lib/server/bidi/third_party/bidiProtocolCore.js +180 -0
  127. package/lib/server/bidi/third_party/bidiProtocolPermissions.js +42 -0
  128. package/lib/server/bidi/third_party/bidiSerializer.js +148 -0
  129. package/lib/server/bidi/third_party/firefoxPrefs.js +261 -0
  130. package/lib/server/browser.js +223 -0
  131. package/lib/server/browserContext.js +703 -0
  132. package/lib/server/browserType.js +338 -0
  133. package/lib/server/callLog.js +82 -0
  134. package/lib/server/chromium/appIcon.png +0 -0
  135. package/lib/server/chromium/chromium.js +399 -0
  136. package/lib/server/chromium/chromiumSwitches.js +104 -0
  137. package/lib/server/chromium/crBrowser.js +532 -0
  138. package/lib/server/chromium/crConnection.js +197 -0
  139. package/lib/server/chromium/crCoverage.js +235 -0
  140. package/lib/server/chromium/crDevTools.js +111 -0
  141. package/lib/server/chromium/crDragDrop.js +131 -0
  142. package/lib/server/chromium/crExecutionContext.js +146 -0
  143. package/lib/server/chromium/crInput.js +187 -0
  144. package/lib/server/chromium/crNetworkManager.js +711 -0
  145. package/lib/server/chromium/crPage.js +1004 -0
  146. package/lib/server/chromium/crPdf.js +121 -0
  147. package/lib/server/chromium/crProtocolHelper.js +145 -0
  148. package/lib/server/chromium/crServiceWorker.js +137 -0
  149. package/lib/server/chromium/defaultFontFamilies.js +162 -0
  150. package/lib/server/chromium/protocol.d.js +16 -0
  151. package/lib/server/clock.js +149 -0
  152. package/lib/server/codegen/csharp.js +327 -0
  153. package/lib/server/codegen/java.js +274 -0
  154. package/lib/server/codegen/javascript.js +247 -0
  155. package/lib/server/codegen/jsonl.js +52 -0
  156. package/lib/server/codegen/language.js +132 -0
  157. package/lib/server/codegen/languages.js +68 -0
  158. package/lib/server/codegen/python.js +279 -0
  159. package/lib/server/codegen/types.js +16 -0
  160. package/lib/server/console.js +61 -0
  161. package/lib/server/cookieStore.js +206 -0
  162. package/lib/server/debugController.js +197 -0
  163. package/lib/server/debugger.js +117 -0
  164. package/lib/server/deviceDescriptors.js +39 -0
  165. package/lib/server/deviceDescriptorsSource.json +1779 -0
  166. package/lib/server/dialog.js +116 -0
  167. package/lib/server/dispatchers/androidDispatcher.js +325 -0
  168. package/lib/server/dispatchers/artifactDispatcher.js +118 -0
  169. package/lib/server/dispatchers/browserContextDispatcher.js +381 -0
  170. package/lib/server/dispatchers/browserDispatcher.js +124 -0
  171. package/lib/server/dispatchers/browserTypeDispatcher.js +71 -0
  172. package/lib/server/dispatchers/cdpSessionDispatcher.js +47 -0
  173. package/lib/server/dispatchers/debugControllerDispatcher.js +78 -0
  174. package/lib/server/dispatchers/debuggerDispatcher.js +80 -0
  175. package/lib/server/dispatchers/dialogDispatcher.js +47 -0
  176. package/lib/server/dispatchers/dispatcher.js +364 -0
  177. package/lib/server/dispatchers/disposableDispatcher.js +39 -0
  178. package/lib/server/dispatchers/electronDispatcher.js +90 -0
  179. package/lib/server/dispatchers/elementHandlerDispatcher.js +181 -0
  180. package/lib/server/dispatchers/frameDispatcher.js +227 -0
  181. package/lib/server/dispatchers/jsHandleDispatcher.js +85 -0
  182. package/lib/server/dispatchers/jsonPipeDispatcher.js +58 -0
  183. package/lib/server/dispatchers/localUtilsDispatcher.js +185 -0
  184. package/lib/server/dispatchers/networkDispatchers.js +214 -0
  185. package/lib/server/dispatchers/pageDispatcher.js +430 -0
  186. package/lib/server/dispatchers/playwrightDispatcher.js +108 -0
  187. package/lib/server/dispatchers/streamDispatcher.js +67 -0
  188. package/lib/server/dispatchers/tracingDispatcher.js +68 -0
  189. package/lib/server/dispatchers/webSocketRouteDispatcher.js +164 -0
  190. package/lib/server/dispatchers/writableStreamDispatcher.js +79 -0
  191. package/lib/server/disposable.js +41 -0
  192. package/lib/server/dom.js +833 -0
  193. package/lib/server/download.js +71 -0
  194. package/lib/server/electron/electron.js +272 -0
  195. package/lib/server/electron/loader.js +29 -0
  196. package/lib/server/errors.js +69 -0
  197. package/lib/server/fetch.js +621 -0
  198. package/lib/server/fileChooser.js +43 -0
  199. package/lib/server/fileUploadUtils.js +84 -0
  200. package/lib/server/firefox/ffBrowser.js +415 -0
  201. package/lib/server/firefox/ffConnection.js +142 -0
  202. package/lib/server/firefox/ffExecutionContext.js +150 -0
  203. package/lib/server/firefox/ffInput.js +175 -0
  204. package/lib/server/firefox/ffNetworkManager.js +256 -0
  205. package/lib/server/firefox/ffPage.js +495 -0
  206. package/lib/server/firefox/firefox.js +114 -0
  207. package/lib/server/firefox/protocol.d.js +16 -0
  208. package/lib/server/formData.js +147 -0
  209. package/lib/server/frameSelectors.js +160 -0
  210. package/lib/server/frames.js +1495 -0
  211. package/lib/server/har/harRecorder.js +147 -0
  212. package/lib/server/har/harTracer.js +608 -0
  213. package/lib/server/harBackend.js +157 -0
  214. package/lib/server/helper.js +96 -0
  215. package/lib/server/index.js +58 -0
  216. package/lib/server/input.js +322 -0
  217. package/lib/server/instrumentation.js +72 -0
  218. package/lib/server/javascript.js +291 -0
  219. package/lib/server/launchApp.js +127 -0
  220. package/lib/server/localUtils.js +214 -0
  221. package/lib/server/macEditingCommands.js +143 -0
  222. package/lib/server/network.js +668 -0
  223. package/lib/server/page.js +884 -0
  224. package/lib/server/pipeTransport.js +89 -0
  225. package/lib/server/playwright.js +69 -0
  226. package/lib/server/progress.js +136 -0
  227. package/lib/server/protocolError.js +52 -0
  228. package/lib/server/recorder/chat.js +161 -0
  229. package/lib/server/recorder/networkCapture.js +282 -0
  230. package/lib/server/recorder/recorderApp.js +581 -0
  231. package/lib/server/recorder/recorderRunner.js +138 -0
  232. package/lib/server/recorder/recorderSignalProcessor.js +83 -0
  233. package/lib/server/recorder/recorderUtils.js +157 -0
  234. package/lib/server/recorder/sessionExporter.js +40 -0
  235. package/lib/server/recorder/sessionPromptBuilder.js +128 -0
  236. package/lib/server/recorder/sessionRedactor.js +74 -0
  237. package/lib/server/recorder/throttledFile.js +57 -0
  238. package/lib/server/recorder.js +538 -0
  239. package/lib/server/registry/browserFetcher.js +177 -0
  240. package/lib/server/registry/dependencies.js +371 -0
  241. package/lib/server/registry/index.js +1395 -0
  242. package/lib/server/registry/nativeDeps.js +1281 -0
  243. package/lib/server/registry/oopDownloadBrowserMain.js +127 -0
  244. package/lib/server/screencast.js +238 -0
  245. package/lib/server/screenshotter.js +333 -0
  246. package/lib/server/selectors.js +112 -0
  247. package/lib/server/socksClientCertificatesInterceptor.js +383 -0
  248. package/lib/server/socksInterceptor.js +95 -0
  249. package/lib/server/trace/recorder/snapshotter.js +147 -0
  250. package/lib/server/trace/recorder/snapshotterInjected.js +561 -0
  251. package/lib/server/trace/recorder/tracing.js +615 -0
  252. package/lib/server/trace/viewer/traceViewer.js +244 -0
  253. package/lib/server/transport.js +181 -0
  254. package/lib/server/types.js +28 -0
  255. package/lib/server/usKeyboardLayout.js +152 -0
  256. package/lib/server/utils/ascii.js +44 -0
  257. package/lib/server/utils/comparators.js +139 -0
  258. package/lib/server/utils/crypto.js +216 -0
  259. package/lib/server/utils/debug.js +42 -0
  260. package/lib/server/utils/debugLogger.js +122 -0
  261. package/lib/server/utils/disposable.js +32 -0
  262. package/lib/server/utils/env.js +73 -0
  263. package/lib/server/utils/eventsHelper.js +41 -0
  264. package/lib/server/utils/expectUtils.js +123 -0
  265. package/lib/server/utils/fileUtils.js +191 -0
  266. package/lib/server/utils/happyEyeballs.js +207 -0
  267. package/lib/server/utils/hostPlatform.js +123 -0
  268. package/lib/server/utils/httpServer.js +205 -0
  269. package/lib/server/utils/image_tools/colorUtils.js +89 -0
  270. package/lib/server/utils/image_tools/compare.js +109 -0
  271. package/lib/server/utils/image_tools/imageChannel.js +78 -0
  272. package/lib/server/utils/image_tools/stats.js +102 -0
  273. package/lib/server/utils/linuxUtils.js +71 -0
  274. package/lib/server/utils/network.js +243 -0
  275. package/lib/server/utils/nodePlatform.js +154 -0
  276. package/lib/server/utils/pipeTransport.js +84 -0
  277. package/lib/server/utils/processLauncher.js +243 -0
  278. package/lib/server/utils/profiler.js +65 -0
  279. package/lib/server/utils/socksProxy.js +511 -0
  280. package/lib/server/utils/spawnAsync.js +41 -0
  281. package/lib/server/utils/task.js +51 -0
  282. package/lib/server/utils/userAgent.js +98 -0
  283. package/lib/server/utils/wsServer.js +121 -0
  284. package/lib/server/utils/zipFile.js +74 -0
  285. package/lib/server/utils/zones.js +57 -0
  286. package/lib/server/videoRecorder.js +133 -0
  287. package/lib/server/webkit/protocol.d.js +16 -0
  288. package/lib/server/webkit/webkit.js +108 -0
  289. package/lib/server/webkit/wkBrowser.js +331 -0
  290. package/lib/server/webkit/wkConnection.js +144 -0
  291. package/lib/server/webkit/wkExecutionContext.js +154 -0
  292. package/lib/server/webkit/wkInput.js +181 -0
  293. package/lib/server/webkit/wkInterceptableRequest.js +197 -0
  294. package/lib/server/webkit/wkPage.js +1164 -0
  295. package/lib/server/webkit/wkProvisionalPage.js +83 -0
  296. package/lib/server/webkit/wkWorkers.js +106 -0
  297. package/lib/serverRegistry.js +147 -0
  298. package/lib/third_party/pixelmatch.js +255 -0
  299. package/lib/tools/backend/browserBackend.js +79 -0
  300. package/lib/tools/backend/common.js +63 -0
  301. package/lib/tools/backend/config.js +41 -0
  302. package/lib/tools/backend/console.js +66 -0
  303. package/lib/tools/backend/context.js +290 -0
  304. package/lib/tools/backend/cookies.js +152 -0
  305. package/lib/tools/backend/devtools.js +69 -0
  306. package/lib/tools/backend/dialogs.js +59 -0
  307. package/lib/tools/backend/evaluate.js +64 -0
  308. package/lib/tools/backend/files.js +60 -0
  309. package/lib/tools/backend/form.js +64 -0
  310. package/lib/tools/backend/keyboard.js +155 -0
  311. package/lib/tools/backend/logFile.js +95 -0
  312. package/lib/tools/backend/mouse.js +168 -0
  313. package/lib/tools/backend/navigate.js +106 -0
  314. package/lib/tools/backend/network.js +135 -0
  315. package/lib/tools/backend/pdf.js +48 -0
  316. package/lib/tools/backend/recorder.js +74 -0
  317. package/lib/tools/backend/response.js +302 -0
  318. package/lib/tools/backend/route.js +140 -0
  319. package/lib/tools/backend/runCode.js +76 -0
  320. package/lib/tools/backend/screenshot.js +88 -0
  321. package/lib/tools/backend/sessionLog.js +74 -0
  322. package/lib/tools/backend/snapshot.js +208 -0
  323. package/lib/tools/backend/storage.js +67 -0
  324. package/lib/tools/backend/tab.js +445 -0
  325. package/lib/tools/backend/tabs.js +67 -0
  326. package/lib/tools/backend/tool.js +47 -0
  327. package/lib/tools/backend/tools.js +104 -0
  328. package/lib/tools/backend/tracing.js +75 -0
  329. package/lib/tools/backend/utils.js +83 -0
  330. package/lib/tools/backend/verify.js +151 -0
  331. package/lib/tools/backend/video.js +89 -0
  332. package/lib/tools/backend/wait.js +63 -0
  333. package/lib/tools/backend/webstorage.js +223 -0
  334. package/lib/tools/cli-client/cli.js +6 -0
  335. package/lib/tools/cli-client/help.json +101 -0
  336. package/lib/tools/cli-client/minimist.js +128 -0
  337. package/lib/tools/cli-client/program.js +382 -0
  338. package/lib/tools/cli-client/registry.js +176 -0
  339. package/lib/tools/cli-client/session.js +289 -0
  340. package/lib/tools/cli-client/skill/SKILL.md +326 -0
  341. package/lib/tools/cli-client/skill/references/element-attributes.md +23 -0
  342. package/lib/tools/cli-client/skill/references/playwright-tests.md +39 -0
  343. package/lib/tools/cli-client/skill/references/request-mocking.md +87 -0
  344. package/lib/tools/cli-client/skill/references/running-code.md +231 -0
  345. package/lib/tools/cli-client/skill/references/session-management.md +169 -0
  346. package/lib/tools/cli-client/skill/references/storage-state.md +275 -0
  347. package/lib/tools/cli-client/skill/references/test-generation.md +88 -0
  348. package/lib/tools/cli-client/skill/references/tracing.md +139 -0
  349. package/lib/tools/cli-client/skill/references/video-recording.md +46 -0
  350. package/lib/tools/cli-daemon/command.js +73 -0
  351. package/lib/tools/cli-daemon/commands.js +933 -0
  352. package/lib/tools/cli-daemon/daemon.js +178 -0
  353. package/lib/tools/cli-daemon/helpGenerator.js +173 -0
  354. package/lib/tools/cli-daemon/program.js +118 -0
  355. package/lib/tools/dashboard/appIcon.png +0 -0
  356. package/lib/tools/dashboard/dashboardApp.js +286 -0
  357. package/lib/tools/dashboard/dashboardController.js +296 -0
  358. package/lib/tools/exports.js +60 -0
  359. package/lib/tools/mcp/browserFactory.js +233 -0
  360. package/lib/tools/mcp/cdpRelay.js +353 -0
  361. package/lib/tools/mcp/cli-stub.js +7 -0
  362. package/lib/tools/mcp/config.d.js +16 -0
  363. package/lib/tools/mcp/config.js +401 -0
  364. package/lib/tools/mcp/configIni.js +189 -0
  365. package/lib/tools/mcp/extensionContextFactory.js +59 -0
  366. package/lib/tools/mcp/index.js +62 -0
  367. package/lib/tools/mcp/log.js +35 -0
  368. package/lib/tools/mcp/program.js +107 -0
  369. package/lib/tools/mcp/protocol.js +28 -0
  370. package/lib/tools/mcp/watchdog.js +44 -0
  371. package/lib/tools/trace/SKILL.md +163 -0
  372. package/lib/tools/trace/installSkill.js +48 -0
  373. package/lib/tools/trace/traceActions.js +142 -0
  374. package/lib/tools/trace/traceAttachments.js +69 -0
  375. package/lib/tools/trace/traceCli.js +80 -0
  376. package/lib/tools/trace/traceConsole.js +97 -0
  377. package/lib/tools/trace/traceErrors.js +55 -0
  378. package/lib/tools/trace/traceOpen.js +69 -0
  379. package/lib/tools/trace/traceParser.js +96 -0
  380. package/lib/tools/trace/traceRequests.js +158 -0
  381. package/lib/tools/trace/traceScreenshot.js +68 -0
  382. package/lib/tools/trace/traceSnapshot.js +149 -0
  383. package/lib/tools/trace/traceUtils.js +135 -0
  384. package/lib/tools/utils/connect.js +32 -0
  385. package/lib/tools/utils/mcp/http.js +152 -0
  386. package/lib/tools/utils/mcp/server.js +230 -0
  387. package/lib/tools/utils/mcp/tool.js +47 -0
  388. package/lib/tools/utils/socketConnection.js +108 -0
  389. package/lib/utils/isomorphic/ariaSnapshot.js +455 -0
  390. package/lib/utils/isomorphic/assert.js +31 -0
  391. package/lib/utils/isomorphic/colors.js +72 -0
  392. package/lib/utils/isomorphic/cssParser.js +245 -0
  393. package/lib/utils/isomorphic/cssTokenizer.js +1051 -0
  394. package/lib/utils/isomorphic/formatUtils.js +64 -0
  395. package/lib/utils/isomorphic/headers.js +53 -0
  396. package/lib/utils/isomorphic/imageUtils.js +141 -0
  397. package/lib/utils/isomorphic/locatorGenerators.js +689 -0
  398. package/lib/utils/isomorphic/locatorParser.js +176 -0
  399. package/lib/utils/isomorphic/locatorUtils.js +81 -0
  400. package/lib/utils/isomorphic/lruCache.js +51 -0
  401. package/lib/utils/isomorphic/manualPromise.js +114 -0
  402. package/lib/utils/isomorphic/mimeType.js +464 -0
  403. package/lib/utils/isomorphic/multimap.js +80 -0
  404. package/lib/utils/isomorphic/protocolFormatter.js +81 -0
  405. package/lib/utils/isomorphic/protocolMetainfo.js +347 -0
  406. package/lib/utils/isomorphic/rtti.js +43 -0
  407. package/lib/utils/isomorphic/selectorParser.js +386 -0
  408. package/lib/utils/isomorphic/semaphore.js +54 -0
  409. package/lib/utils/isomorphic/stackTrace.js +158 -0
  410. package/lib/utils/isomorphic/stringUtils.js +204 -0
  411. package/lib/utils/isomorphic/time.js +49 -0
  412. package/lib/utils/isomorphic/timeoutRunner.js +66 -0
  413. package/lib/utils/isomorphic/trace/entries.js +16 -0
  414. package/lib/utils/isomorphic/trace/snapshotRenderer.js +492 -0
  415. package/lib/utils/isomorphic/trace/snapshotServer.js +120 -0
  416. package/lib/utils/isomorphic/trace/snapshotStorage.js +89 -0
  417. package/lib/utils/isomorphic/trace/traceLoader.js +131 -0
  418. package/lib/utils/isomorphic/trace/traceModel.js +366 -0
  419. package/lib/utils/isomorphic/trace/traceModernizer.js +401 -0
  420. package/lib/utils/isomorphic/trace/traceUtils.js +58 -0
  421. package/lib/utils/isomorphic/trace/versions/traceV3.js +16 -0
  422. package/lib/utils/isomorphic/trace/versions/traceV4.js +16 -0
  423. package/lib/utils/isomorphic/trace/versions/traceV5.js +16 -0
  424. package/lib/utils/isomorphic/trace/versions/traceV6.js +16 -0
  425. package/lib/utils/isomorphic/trace/versions/traceV7.js +16 -0
  426. package/lib/utils/isomorphic/trace/versions/traceV8.js +16 -0
  427. package/lib/utils/isomorphic/types.js +16 -0
  428. package/lib/utils/isomorphic/urlMatch.js +243 -0
  429. package/lib/utils/isomorphic/utilityScriptSerializers.js +262 -0
  430. package/lib/utils/isomorphic/yaml.js +84 -0
  431. package/lib/utils.js +113 -0
  432. package/lib/utilsBundle.js +91 -0
  433. package/lib/utilsBundleImpl/index.js +218 -0
  434. package/lib/utilsBundleImpl/xdg-open +1066 -0
  435. package/lib/vite/dashboard/assets/index-Bn2lDGZX.js +50 -0
  436. package/lib/vite/dashboard/assets/index-DDWrEWMc.css +1 -0
  437. package/lib/vite/dashboard/index.html +28 -0
  438. package/lib/vite/htmlReport/index.html +89 -0
  439. package/lib/vite/recorder/assets/codeMirrorModule-DYBRYzYX.css +1 -0
  440. package/lib/vite/recorder/assets/codeMirrorModule-RoSmqW0t.js +32 -0
  441. package/lib/vite/recorder/assets/codicon-DCmgc-ay.ttf +0 -0
  442. package/lib/vite/recorder/assets/index-CLrDthbi.js +193 -0
  443. package/lib/vite/recorder/assets/index-OFPIkgDs.css +1 -0
  444. package/lib/vite/recorder/index.html +29 -0
  445. package/lib/vite/recorder/playwright-logo.svg +9 -0
  446. package/lib/vite/traceViewer/assets/codeMirrorModule-Cigrr2OM.js +32 -0
  447. package/lib/vite/traceViewer/assets/defaultSettingsView-ConrJv9G.js +262 -0
  448. package/lib/vite/traceViewer/assets/xtermModule-CsJ4vdCR.js +9 -0
  449. package/lib/vite/traceViewer/codeMirrorModule.DYBRYzYX.css +1 -0
  450. package/lib/vite/traceViewer/codicon.DCmgc-ay.ttf +0 -0
  451. package/lib/vite/traceViewer/defaultSettingsView.B4dS75f0.css +1 -0
  452. package/lib/vite/traceViewer/index.CzXZzn5A.css +1 -0
  453. package/lib/vite/traceViewer/index.EVGp-u_4.js +2 -0
  454. package/lib/vite/traceViewer/index.html +43 -0
  455. package/lib/vite/traceViewer/manifest.webmanifest +16 -0
  456. package/lib/vite/traceViewer/playwright-logo.svg +9 -0
  457. package/lib/vite/traceViewer/snapshot.html +21 -0
  458. package/lib/vite/traceViewer/sw.bundle.js +5 -0
  459. package/lib/vite/traceViewer/uiMode.Bewj7-uD.js +6 -0
  460. package/lib/vite/traceViewer/uiMode.Btcz36p_.css +1 -0
  461. package/lib/vite/traceViewer/uiMode.html +17 -0
  462. package/lib/vite/traceViewer/xtermModule.DYP7pi_n.css +32 -0
  463. package/lib/zipBundle.js +34 -0
  464. package/lib/zipBundleImpl.js +5 -0
  465. package/package.json +46 -0
  466. package/types/protocol.d.ts +24365 -0
  467. package/types/structs.d.ts +45 -0
  468. package/types/types.d.ts +23498 -0
@@ -0,0 +1,581 @@
1
+ "use strict";
2
+ var __create = Object.create;
3
+ var __defProp = Object.defineProperty;
4
+ var __getOwnPropDesc = Object.getOwnPropertyDescriptor;
5
+ var __getOwnPropNames = Object.getOwnPropertyNames;
6
+ var __getProtoOf = Object.getPrototypeOf;
7
+ var __hasOwnProp = Object.prototype.hasOwnProperty;
8
+ var __export = (target, all) => {
9
+ for (var name in all)
10
+ __defProp(target, name, { get: all[name], enumerable: true });
11
+ };
12
+ var __copyProps = (to, from, except, desc) => {
13
+ if (from && typeof from === "object" || typeof from === "function") {
14
+ for (let key of __getOwnPropNames(from))
15
+ if (!__hasOwnProp.call(to, key) && key !== except)
16
+ __defProp(to, key, { get: () => from[key], enumerable: !(desc = __getOwnPropDesc(from, key)) || desc.enumerable });
17
+ }
18
+ return to;
19
+ };
20
+ var __toESM = (mod, isNodeMode, target) => (target = mod != null ? __create(__getProtoOf(mod)) : {}, __copyProps(
21
+ // If the importer is in node compatibility mode or this is not an ESM
22
+ // file that has been converted to a CommonJS file using a Babel-
23
+ // compatible transform (i.e. "__esModule" has not been set), then set
24
+ // "default" to the CommonJS "module.exports" for node compatibility.
25
+ isNodeMode || !mod || !mod.__esModule ? __defProp(target, "default", { value: mod, enumerable: true }) : target,
26
+ mod
27
+ ));
28
+ var __toCommonJS = (mod) => __copyProps(__defProp({}, "__esModule", { value: true }), mod);
29
+ var recorderApp_exports = {};
30
+ __export(recorderApp_exports, {
31
+ ProgrammaticRecorderApp: () => ProgrammaticRecorderApp,
32
+ RecorderApp: () => RecorderApp
33
+ });
34
+ module.exports = __toCommonJS(recorderApp_exports);
35
+ var import_fs = __toESM(require("fs"));
36
+ var import_path = __toESM(require("path"));
37
+ var import_debug = require("../utils/debug");
38
+ var import_utilsBundle = require("../../utilsBundle");
39
+ var import_launchApp = require("../launchApp");
40
+ var import_launchApp2 = require("../launchApp");
41
+ var import_progress = require("../progress");
42
+ var import_throttledFile = require("./throttledFile");
43
+ var import_languages = require("../codegen/languages");
44
+ var import_recorderUtils = require("./recorderUtils");
45
+ var import_language = require("../codegen/language");
46
+ var import_recorder = require("../recorder");
47
+ var import_browserContext = require("../browserContext");
48
+ var import_networkCapture = require("./networkCapture");
49
+ var import_sessionExporter = require("./sessionExporter");
50
+ var import_chat = require("./chat");
51
+ var import_sessionRedactor = require("./sessionRedactor");
52
+ var import_sessionPromptBuilder = require("./sessionPromptBuilder");
53
+ class RecorderApp {
54
+ constructor(recorder, params, page, wsEndpointForTest) {
55
+ this._throttledOutputFile = null;
56
+ this._actions = [];
57
+ this._userSources = [];
58
+ this._recorderSources = [];
59
+ this._networkCapture = null;
60
+ this._inspectedContext = null;
61
+ this._scenarioName = "my scenario";
62
+ this._throttledSessionFile = null;
63
+ this._page = page;
64
+ this._recorder = recorder;
65
+ this._frontend = createRecorderFrontend(page);
66
+ this.wsEndpointForTest = process.env.PW_AI_ENDPOINT || wsEndpointForTest;
67
+ this._languageGeneratorOptions = {
68
+ browserName: params.browserName,
69
+ launchOptions: { headless: false, ...params.launchOptions, tracesDir: void 0 },
70
+ contextOptions: { ...params.contextOptions },
71
+ deviceName: params.device,
72
+ saveStorage: params.saveStorage
73
+ };
74
+ this._aiCodegen = !!params.aiCodegen;
75
+ this._throttledOutputFile = params.outputFile ? new import_throttledFile.ThrottledFile(params.outputFile) : null;
76
+ if (this._aiCodegen)
77
+ this._throttledSessionFile = new import_throttledFile.ThrottledFile(import_path.default.join(process.cwd(), ".playwright-session.md"));
78
+ this._primaryGeneratorId = process.env.TEST_INSPECTOR_LANGUAGE || params.language || determinePrimaryGeneratorId(params.sdkLanguage);
79
+ this._selectedGeneratorId = this._primaryGeneratorId;
80
+ for (const languageGenerator of (0, import_languages.languageSet)()) {
81
+ if (languageGenerator.id === this._primaryGeneratorId)
82
+ this._recorder.setLanguage(languageGenerator.highlighter);
83
+ }
84
+ }
85
+ async _init(inspectedContext) {
86
+ this._inspectedContext = inspectedContext;
87
+ await (0, import_launchApp.syncLocalStorageWithSettings)(this._page, "recorder");
88
+ const controller = new import_progress.ProgressController();
89
+ await controller.run(async (progress) => {
90
+ await this._page.addRequestInterceptor(progress, (route) => {
91
+ if (!route.request().url().startsWith("https://playwright/")) {
92
+ route.continue({ isFallback: true }).catch(() => {
93
+ });
94
+ return;
95
+ }
96
+ const uri = route.request().url().substring("https://playwright/".length);
97
+ const file = require.resolve("../../vite/recorder/" + uri);
98
+ import_fs.default.promises.readFile(file).then((buffer) => {
99
+ route.fulfill({
100
+ status: 200,
101
+ headers: [
102
+ { name: "Content-Type", value: import_utilsBundle.mime.getType(import_path.default.extname(file)) || "application/octet-stream" }
103
+ ],
104
+ body: buffer.toString("base64"),
105
+ isBase64: true
106
+ }).catch(() => {
107
+ });
108
+ });
109
+ });
110
+ await this._createDispatcher(progress);
111
+ this._page.once("close", () => {
112
+ this._throttledSessionFile?.flush();
113
+ this._networkCapture?.dispose();
114
+ this._networkCapture = null;
115
+ this._recorder.close();
116
+ inspectedContext.close({ reason: "Recorder window closed" }).catch(() => {
117
+ });
118
+ this._page.browserContext.close({ reason: "Recorder window closed" }).catch(() => {
119
+ });
120
+ delete inspectedContext[recorderAppSymbol];
121
+ });
122
+ await this._page.mainFrame().goto(progress, "https://playwright/index.html");
123
+ });
124
+ const url = this._recorder.url();
125
+ if (url)
126
+ this._frontend.pageNavigated({ url });
127
+ this._frontend.modeChanged({ mode: this._recorder.mode() });
128
+ this._frontend.configUpdated({ aiCodegen: this._aiCodegen });
129
+ this._frontend.pauseStateChanged({ paused: this._recorder.paused() });
130
+ this._updateActions("reveal");
131
+ this._onUserSourcesChanged(this._recorder.userSources(), this._recorder.pausedSourceId());
132
+ this._frontend.callLogsUpdated({ callLogs: this._recorder.callLog() });
133
+ this._wireListeners(this._recorder);
134
+ if (this._aiCodegen && this._recorder.mode() === "recording" && !this._networkCapture && this._inspectedContext) {
135
+ this._networkCapture = new import_networkCapture.NetworkCapture(this._inspectedContext, {
136
+ onNetworkEventsUpdated: () => this._updateNetworkPanel()
137
+ });
138
+ this._networkCapture.start();
139
+ }
140
+ }
141
+ async _createDispatcher(progress) {
142
+ const dispatcher = {
143
+ clear: async () => {
144
+ this._actions = [];
145
+ this._updateActions("reveal");
146
+ this._recorder.clear();
147
+ },
148
+ fileChanged: async (params) => {
149
+ const source = [...this._recorderSources, ...this._userSources].find((s) => s.id === params.fileId);
150
+ if (source) {
151
+ if (source.isRecorded)
152
+ this._selectedGeneratorId = source.id;
153
+ await this._recorder.setLanguage(source.language);
154
+ }
155
+ },
156
+ setAutoExpect: async (params) => {
157
+ this._languageGeneratorOptions.generateAutoExpect = params.autoExpect;
158
+ this._updateActions();
159
+ },
160
+ setMode: async (params) => {
161
+ await this._recorder.setMode(params.mode);
162
+ },
163
+ resume: async () => {
164
+ this._recorder.resume();
165
+ },
166
+ pause: async () => {
167
+ this._recorder.pause();
168
+ },
169
+ step: async () => {
170
+ this._recorder.step();
171
+ },
172
+ highlightRequested: async (params) => {
173
+ if (params.selector)
174
+ await this._recorder.setHighlightedSelector(params.selector);
175
+ if (params.ariaTemplate)
176
+ await this._recorder.setHighlightedAriaTemplate(params.ariaTemplate);
177
+ },
178
+ ...this._aiCodegen ? {
179
+ generateTest: async (params) => {
180
+ void this._runGeneration(params.scenarioName, params.outputFile);
181
+ },
182
+ setScenarioName: async (params) => {
183
+ this._scenarioName = params.name;
184
+ }
185
+ } : {}
186
+ };
187
+ await this._page.exposeBinding(progress, "sendCommand", false, async (_, data) => {
188
+ const { method, params } = data;
189
+ return await dispatcher[method].call(dispatcher, params);
190
+ });
191
+ }
192
+ static async show(context, params) {
193
+ if (process.env.PW_CODEGEN_NO_INSPECTOR)
194
+ return;
195
+ const recorder = await import_recorder.Recorder.forContext(context, params);
196
+ if (params.recorderMode === "api") {
197
+ const browserName = context._browser.options.name;
198
+ await ProgrammaticRecorderApp.run(context, recorder, browserName, params);
199
+ return;
200
+ }
201
+ await RecorderApp._show(recorder, context, params);
202
+ }
203
+ async close() {
204
+ this._networkCapture?.dispose();
205
+ this._networkCapture = null;
206
+ await this._page.close();
207
+ }
208
+ static showInspectorNoReply(context) {
209
+ if (process.env.PW_CODEGEN_NO_INSPECTOR)
210
+ return;
211
+ void import_recorder.Recorder.forContext(context, {}).then((recorder) => RecorderApp._show(recorder, context, {})).catch(() => {
212
+ });
213
+ }
214
+ static async _show(recorder, inspectedContext, params) {
215
+ if (inspectedContext[recorderAppSymbol])
216
+ return;
217
+ inspectedContext[recorderAppSymbol] = true;
218
+ const sdkLanguage = inspectedContext._browser.sdkLanguage();
219
+ const isChromium = inspectedContext._browser.options.browserType === "chromium";
220
+ const headed = !!inspectedContext._browser.options.headful;
221
+ const recorderPlaywright = require("../playwright").createPlaywright({ sdkLanguage: "javascript", isInternalPlaywright: true });
222
+ const { context: appContext, page } = await (0, import_launchApp2.launchApp)(recorderPlaywright.chromium, {
223
+ sdkLanguage,
224
+ windowSize: { width: 600, height: 600 },
225
+ windowPosition: { x: 1020, y: 10 },
226
+ persistentContextOptions: {
227
+ noDefaultViewport: true,
228
+ headless: !!process.env.PWTEST_CLI_HEADLESS || (0, import_debug.isUnderTest)() && !headed,
229
+ cdpPort: (0, import_debug.isUnderTest)() ? 0 : void 0,
230
+ handleSIGINT: params.handleSIGINT,
231
+ executablePath: isChromium ? inspectedContext._browser.options.customExecutablePath : void 0,
232
+ // Use the same channel as the inspected context to guarantee that the browser is installed.
233
+ channel: isChromium ? inspectedContext._browser.options.channel : void 0
234
+ }
235
+ });
236
+ const controller = new import_progress.ProgressController();
237
+ await controller.run(async (progress) => {
238
+ await appContext._browser._defaultContext._loadDefaultContextAsIs(progress);
239
+ });
240
+ const appParams = {
241
+ browserName: inspectedContext._browser.options.name,
242
+ sdkLanguage: inspectedContext._browser.sdkLanguage(),
243
+ wsEndpointForTest: inspectedContext._browser.options.wsEndpoint,
244
+ headed: !!inspectedContext._browser.options.headful,
245
+ executablePath: isChromium ? inspectedContext._browser.options.customExecutablePath : void 0,
246
+ channel: isChromium ? inspectedContext._browser.options.channel : void 0,
247
+ ...params
248
+ };
249
+ const recorderApp = new RecorderApp(recorder, appParams, page, appContext._browser.options.wsEndpoint);
250
+ await recorderApp._init(inspectedContext);
251
+ inspectedContext.recorderAppForTest = recorderApp;
252
+ }
253
+ _wireListeners(recorder) {
254
+ recorder.on(import_recorder.RecorderEvent.ActionAdded, (action) => {
255
+ this._onActionAdded(action);
256
+ });
257
+ recorder.on(import_recorder.RecorderEvent.SignalAdded, (signal) => {
258
+ this._onSignalAdded(signal);
259
+ });
260
+ recorder.on(import_recorder.RecorderEvent.PageNavigated, (url) => {
261
+ this._frontend.pageNavigated({ url });
262
+ });
263
+ recorder.on(import_recorder.RecorderEvent.ContextClosed, () => {
264
+ this._throttledOutputFile?.flush();
265
+ this._throttledSessionFile?.flush();
266
+ this._networkCapture?.dispose();
267
+ this._networkCapture = null;
268
+ this._page.browserContext.close({ reason: "Recorder window closed" }).catch(() => {
269
+ });
270
+ });
271
+ recorder.on(import_recorder.RecorderEvent.ModeChanged, (mode) => {
272
+ this._frontend.modeChanged({ mode });
273
+ if (this._aiCodegen && mode === "recording" && !this._networkCapture && this._inspectedContext) {
274
+ this._networkCapture = new import_networkCapture.NetworkCapture(this._inspectedContext, {
275
+ onNetworkEventsUpdated: () => this._updateNetworkPanel()
276
+ });
277
+ this._networkCapture.start();
278
+ }
279
+ if (mode !== "recording" && this._networkCapture) {
280
+ this._networkCapture.stop();
281
+ this._networkCapture = null;
282
+ }
283
+ });
284
+ recorder.on(import_recorder.RecorderEvent.PausedStateChanged, (paused) => {
285
+ this._frontend.pauseStateChanged({ paused });
286
+ });
287
+ recorder.on(import_recorder.RecorderEvent.UserSourcesChanged, (sources, pausedSourceId) => {
288
+ this._onUserSourcesChanged(sources, pausedSourceId);
289
+ });
290
+ recorder.on(import_recorder.RecorderEvent.ElementPicked, (elementInfo, userGesture) => {
291
+ if (userGesture)
292
+ this._page.bringToFront();
293
+ this._frontend.elementPicked({ elementInfo, userGesture });
294
+ });
295
+ recorder.on(import_recorder.RecorderEvent.CallLogsUpdated, (callLogs) => {
296
+ this._frontend.callLogsUpdated({ callLogs });
297
+ });
298
+ }
299
+ _onActionAdded(action) {
300
+ this._actions.push(action);
301
+ this._networkCapture?.onActionAdded(action);
302
+ this._updateActions("reveal");
303
+ }
304
+ _onSignalAdded(signal) {
305
+ const lastAction = this._actions.findLast((a) => a.frame.pageGuid === signal.frame.pageGuid);
306
+ if (lastAction)
307
+ lastAction.action.signals.push(signal.signal);
308
+ this._updateActions();
309
+ if (signal.signal.name === "navigation") {
310
+ this._networkCapture?.onNavigationSignal(
311
+ signal.frame.pageGuid,
312
+ signal.signal.url
313
+ );
314
+ }
315
+ }
316
+ _onUserSourcesChanged(sources, pausedSourceId) {
317
+ if (!sources.length && !this._userSources.length)
318
+ return;
319
+ this._userSources = sources;
320
+ this._pushAllSources();
321
+ this._revealSource(pausedSourceId);
322
+ }
323
+ _pushAllSources() {
324
+ const sources = [...this._userSources, ...this._recorderSources];
325
+ this._frontend.sourcesUpdated({ sources });
326
+ }
327
+ _revealSource(sourceId) {
328
+ if (!sourceId)
329
+ return;
330
+ this._frontend.sourceRevealRequested({ sourceId });
331
+ }
332
+ _updateActions(reveal) {
333
+ const recorderSources = [];
334
+ const actions = (0, import_recorderUtils.collapseActions)(this._actions);
335
+ let revealSourceId;
336
+ for (const languageGenerator of (0, import_languages.languageSet)()) {
337
+ const { header, footer, actionTexts, text } = (0, import_language.generateCode)(actions, languageGenerator, this._languageGeneratorOptions);
338
+ const source = {
339
+ isRecorded: true,
340
+ label: languageGenerator.name,
341
+ group: languageGenerator.groupName,
342
+ id: languageGenerator.id,
343
+ text,
344
+ header,
345
+ footer,
346
+ actions: actionTexts,
347
+ language: languageGenerator.highlighter,
348
+ highlight: []
349
+ };
350
+ source.revealLine = text.split("\n").length - 1;
351
+ recorderSources.push(source);
352
+ if (languageGenerator.id === this._primaryGeneratorId)
353
+ this._throttledOutputFile?.setContent(source.text);
354
+ if (reveal === "reveal" && source.id === this._selectedGeneratorId)
355
+ revealSourceId = source.id;
356
+ }
357
+ this._recorderSources = recorderSources;
358
+ this._pushAllSources();
359
+ this._revealSource(revealSourceId);
360
+ if (this._throttledSessionFile) {
361
+ const enriched = this._networkCapture ? this._networkCapture.getEnrichedActions(this._actions) : actions;
362
+ const redacted = (0, import_sessionRedactor.redactSession)(enriched, false);
363
+ const prompt = (0, import_sessionPromptBuilder.buildPrompt)(redacted, {
364
+ scenarioName: this._scenarioName,
365
+ outputFile: "tests/" + this._scenarioName.replace(/\s+/g, "-") + ".spec.ts",
366
+ mode: "clipboard",
367
+ pageHasWebSockets: false
368
+ });
369
+ this._throttledSessionFile.setContent(prompt);
370
+ }
371
+ }
372
+ _updateNetworkPanel() {
373
+ const entries = [];
374
+ const collapsed = (0, import_recorderUtils.collapseActions)(this._actions);
375
+ for (let i = 0; i < collapsed.length; i++) {
376
+ for (const event of collapsed[i].networkEvents ?? []) {
377
+ entries.push({
378
+ url: event.url,
379
+ method: event.method,
380
+ status: event.status,
381
+ bucket: event.bucket,
382
+ linkedStepIndex: i,
383
+ operationName: event.operationName
384
+ });
385
+ }
386
+ }
387
+ this._frontend.networkEntriesUpdated({ entries });
388
+ }
389
+ _emitGenerationStatus(params) {
390
+ this._frontend.generationStatusChanged(params);
391
+ }
392
+ async _exportPrompt(scenarioName, outputFile) {
393
+ try {
394
+ this._emitGenerationStatus({ status: "analyzing", message: "Building prompt...", progress: 30 });
395
+ await this._networkCapture?.waitForPendingResponses(15e3);
396
+ const enrichedActions = this._networkCapture ? this._networkCapture.getEnrichedActions(this._actions) : (0, import_recorderUtils.collapseActions)(this._actions);
397
+ const redactedSession = (0, import_sessionRedactor.redactSession)(enrichedActions, false);
398
+ const prompt = (0, import_sessionPromptBuilder.buildPrompt)(redactedSession, {
399
+ scenarioName,
400
+ outputFile,
401
+ mode: "clipboard",
402
+ pageHasWebSockets: false
403
+ });
404
+ const promptFilePath = import_path.default.join(process.cwd(), ".playwright-prompt.md");
405
+ await import_fs.default.promises.writeFile(promptFilePath, prompt, "utf-8");
406
+ this._frontend.promptReady({ prompt, filePath: promptFilePath });
407
+ this._emitGenerationStatus({
408
+ status: "exported",
409
+ message: "Prompt copied to clipboard and saved to .playwright-prompt.md",
410
+ progress: 100,
411
+ promptFilePath
412
+ });
413
+ } catch (error) {
414
+ this._emitGenerationStatus({ status: "error", message: String(error?.message ?? error), progress: 0 });
415
+ }
416
+ }
417
+ async _runGeneration(scenarioName, outputFile) {
418
+ if (!process.env.PW_AI_ENDPOINT) {
419
+ await this._exportPrompt(scenarioName, outputFile);
420
+ return;
421
+ }
422
+ try {
423
+ this._emitGenerationStatus({ status: "finalizing", message: "Waiting for pending API responses...", progress: 10 });
424
+ await this._networkCapture?.waitForPendingResponses(15e3);
425
+ this._emitGenerationStatus({ status: "analyzing", message: "Analyzing session...", progress: 30 });
426
+ const enrichedActions = this._networkCapture ? this._networkCapture.getEnrichedActions(this._actions) : (0, import_recorderUtils.collapseActions)(this._actions);
427
+ this._emitGenerationStatus({ status: "writing", message: "AI writing test...", progress: 50 });
428
+ const result = await (0, import_sessionExporter.exportSession)(enrichedActions, {
429
+ scenarioName,
430
+ outputFile,
431
+ wsEndpoint: this.wsEndpointForTest,
432
+ pageHasWebSockets: false
433
+ });
434
+ await import_fs.default.promises.writeFile(outputFile, result.code, "utf-8");
435
+ this._emitGenerationStatus({ status: "running", message: "Running generated test...", progress: 75 });
436
+ const runResult = await this._runGeneratedTest(outputFile);
437
+ if (runResult.passed) {
438
+ this._emitGenerationStatus({ status: "done", message: `Test passed in ${runResult.durationMs}ms`, progress: 100, outputFile });
439
+ } else {
440
+ const chat = new import_chat.Chat(this.wsEndpointForTest);
441
+ await this._repairAndRerun(chat, result.code, outputFile, runResult.error ?? "Test failed", 1);
442
+ }
443
+ } catch (error) {
444
+ this._emitGenerationStatus({ status: "error", message: String(error?.message ?? error), progress: 0 });
445
+ }
446
+ }
447
+ async _runGeneratedTest(outputFile) {
448
+ const start = Date.now();
449
+ return new Promise((resolve) => {
450
+ const { spawn } = require("child_process");
451
+ const proc = spawn("npx", ["playwright", "test", outputFile, "--reporter=json"], {
452
+ cwd: process.cwd(),
453
+ stdio: ["ignore", "pipe", "pipe"]
454
+ });
455
+ let stdout = "";
456
+ let stderr = "";
457
+ proc.stdout?.on("data", (d) => stdout += d.toString());
458
+ proc.stderr?.on("data", (d) => stderr += d.toString());
459
+ proc.on("close", (code) => {
460
+ const durationMs = Date.now() - start;
461
+ if (code === 0) {
462
+ resolve({ passed: true, durationMs });
463
+ return;
464
+ }
465
+ try {
466
+ const report = JSON.parse(stdout);
467
+ const failed = report?.suites?.[0]?.specs?.[0]?.tests?.[0]?.results?.[0];
468
+ const errorMsg = failed?.error?.message ?? stderr.slice(0, 1e3);
469
+ resolve({ passed: false, durationMs, error: errorMsg });
470
+ } catch {
471
+ resolve({ passed: false, durationMs, error: stderr.slice(0, 1e3) || "Test failed" });
472
+ }
473
+ });
474
+ proc.on("error", (err) => {
475
+ resolve({ passed: false, durationMs: Date.now() - start, error: err.message });
476
+ });
477
+ });
478
+ }
479
+ async _repairAndRerun(chat, code, outputFile, error, attempt) {
480
+ if (attempt > 3) {
481
+ this._emitGenerationStatus({
482
+ status: "error",
483
+ message: `Test failed after ${attempt - 1} repair attempt(s). Last error: ${error.slice(0, 200)}`,
484
+ progress: 0
485
+ });
486
+ return;
487
+ }
488
+ this._emitGenerationStatus({
489
+ status: "repairing",
490
+ message: `Repair attempt ${attempt}/3...`,
491
+ progress: 75 + attempt * 5
492
+ });
493
+ const repairPrompt = `The following Playwright test failed. Please fix it.
494
+
495
+ ## Failed Test
496
+ \`\`\`typescript
497
+ ${code}
498
+ \`\`\`
499
+
500
+ ## Error
501
+ ${error}
502
+
503
+ ## Output Format
504
+ Return the fixed TypeScript code as a raw JSON string. Your entire response must be valid JSON parseable by JSON.parse().
505
+ Example: "import { test, expect } from '@playwright/test';\\n\\ntest.describe(..."
506
+ Do not include any explanation, markdown fences, or wrapper objects.
507
+ The code must start with: import { test, expect } from '@playwright/test';`;
508
+ const fixedCode = await chat.post(repairPrompt);
509
+ if (!fixedCode) {
510
+ this._emitGenerationStatus({ status: "error", message: "AI repair failed", progress: 0 });
511
+ return;
512
+ }
513
+ await import_fs.default.promises.writeFile(outputFile, fixedCode, "utf-8");
514
+ const runResult = await this._runGeneratedTest(outputFile);
515
+ if (runResult.passed)
516
+ this._emitGenerationStatus({ status: "done", message: `Test passed after ${attempt} repair(s)`, progress: 100, outputFile });
517
+ else
518
+ await this._repairAndRerun(chat, fixedCode, outputFile, runResult.error ?? "Test failed", attempt + 1);
519
+ }
520
+ }
521
+ function determinePrimaryGeneratorId(sdkLanguage) {
522
+ for (const language of (0, import_languages.languageSet)()) {
523
+ if (language.highlighter === sdkLanguage)
524
+ return language.id;
525
+ }
526
+ return sdkLanguage;
527
+ }
528
+ class ProgrammaticRecorderApp {
529
+ static async run(inspectedContext, recorder, browserName, params) {
530
+ let lastAction = null;
531
+ const languages = [...(0, import_languages.languageSet)()];
532
+ const languageGeneratorOptions = {
533
+ browserName,
534
+ launchOptions: { headless: false, ...params.launchOptions, tracesDir: void 0 },
535
+ contextOptions: { ...params.contextOptions },
536
+ deviceName: params.device,
537
+ saveStorage: params.saveStorage
538
+ };
539
+ const languageGenerator = languages.find((l) => l.id === params.language) ?? languages.find((l) => l.id === "playwright-test");
540
+ recorder.on(import_recorder.RecorderEvent.ActionAdded, (action) => {
541
+ const page = findPageByGuid(inspectedContext, action.frame.pageGuid);
542
+ if (!page)
543
+ return;
544
+ const { actionTexts } = (0, import_language.generateCode)([action], languageGenerator, languageGeneratorOptions);
545
+ if (!lastAction || !(0, import_recorderUtils.shouldMergeAction)(action, lastAction))
546
+ inspectedContext.emit(import_browserContext.BrowserContext.Events.RecorderEvent, { event: "actionAdded", data: action, page, code: actionTexts.join("\n") });
547
+ else
548
+ inspectedContext.emit(import_browserContext.BrowserContext.Events.RecorderEvent, { event: "actionUpdated", data: action, page, code: actionTexts.join("\n") });
549
+ lastAction = action;
550
+ });
551
+ recorder.on(import_recorder.RecorderEvent.SignalAdded, (signal) => {
552
+ const page = findPageByGuid(inspectedContext, signal.frame.pageGuid);
553
+ if (!page)
554
+ return;
555
+ inspectedContext.emit(import_browserContext.BrowserContext.Events.RecorderEvent, { event: "signalAdded", data: signal, page, code: "" });
556
+ });
557
+ }
558
+ }
559
+ function findPageByGuid(context, guid) {
560
+ return context.pages().find((p) => p.guid === guid);
561
+ }
562
+ function createRecorderFrontend(page) {
563
+ return new Proxy({}, {
564
+ get: (_target, prop) => {
565
+ if (typeof prop !== "string")
566
+ return void 0;
567
+ return (params) => {
568
+ page.mainFrame().evaluateExpression(((event) => {
569
+ window.dispatch(event);
570
+ }).toString(), { isFunction: true }, { method: prop, params }).catch(() => {
571
+ });
572
+ };
573
+ }
574
+ });
575
+ }
576
+ const recorderAppSymbol = Symbol("recorderApp");
577
+ // Annotate the CommonJS export names for ESM import in node:
578
+ 0 && (module.exports = {
579
+ ProgrammaticRecorderApp,
580
+ RecorderApp
581
+ });