| 
									
										
										
										
											2022-04-05 15:10:12 -08:00
										 |  |  | /** | 
					
						
							|  |  |  |  * Copyright (c) Microsoft Corporation. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * Licensed under the Apache License, Version 2.0 (the "License"); | 
					
						
							|  |  |  |  * you may not use this file except in compliance with the License. | 
					
						
							|  |  |  |  * You may obtain a copy of the License at | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * http://www.apache.org/licenses/LICENSE-2.0
 | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * Unless required by applicable law or agreed to in writing, software | 
					
						
							|  |  |  |  * distributed under the License is distributed on an "AS IS" BASIS, | 
					
						
							|  |  |  |  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. | 
					
						
							|  |  |  |  * See the License for the specific language governing permissions and | 
					
						
							|  |  |  |  * limitations under the License. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-03-21 07:40:54 -07:00
										 |  |  | import type { Fixtures, FrameLocator, Locator, Page, Browser, BrowserContext } from '@playwright/test'; | 
					
						
							| 
									
										
										
										
											2023-03-29 20:43:08 -07:00
										 |  |  | import { step } from './baseTest'; | 
					
						
							| 
									
										
										
										
											2024-03-18 09:50:11 -07:00
										 |  |  | import { runTraceViewerApp } from '../../packages/playwright-core/lib/server'; | 
					
						
							| 
									
										
										
										
											2022-04-05 15:10:12 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  | type BaseTestFixtures = { | 
					
						
							|  |  |  |   context: BrowserContext; | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | type BaseWorkerFixtures = { | 
					
						
							|  |  |  |   headless: boolean; | 
					
						
							|  |  |  |   browser: Browser; | 
					
						
							|  |  |  |   browserName: 'chromium' | 'firefox' | 'webkit'; | 
					
						
							|  |  |  |   playwright: typeof import('@playwright/test'); | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | export type TraceViewerFixtures = { | 
					
						
							| 
									
										
										
										
											2023-01-27 23:20:25 +01:00
										 |  |  |   showTraceViewer: (trace: string[], options?: {host?: string, port?: number}) => Promise<TraceViewerPage>; | 
					
						
							| 
									
										
										
										
											2024-09-06 16:24:33 +02:00
										 |  |  |   runAndTrace: (body: () => Promise<void>, optsOverrides?: Parameters<BrowserContext['tracing']['start']>[0]) => Promise<TraceViewerPage>; | 
					
						
							| 
									
										
										
										
											2022-04-05 15:10:12 -08:00
										 |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | class TraceViewerPage { | 
					
						
							|  |  |  |   actionTitles: Locator; | 
					
						
							| 
									
										
										
										
											2024-11-05 12:45:37 +01:00
										 |  |  |   actionsTree: Locator; | 
					
						
							| 
									
										
										
										
											2022-04-05 15:10:12 -08:00
										 |  |  |   callLines: Locator; | 
					
						
							|  |  |  |   consoleLines: Locator; | 
					
						
							| 
									
										
										
										
											2023-09-01 20:12:05 -07:00
										 |  |  |   logLines: Locator; | 
					
						
							| 
									
										
										
										
											2024-02-02 16:41:08 -08:00
										 |  |  |   errorMessages: Locator; | 
					
						
							| 
									
										
										
										
											2022-04-05 15:10:12 -08:00
										 |  |  |   consoleLineMessages: Locator; | 
					
						
							|  |  |  |   consoleStacks: Locator; | 
					
						
							|  |  |  |   stackFrames: Locator; | 
					
						
							|  |  |  |   networkRequests: Locator; | 
					
						
							| 
									
										
										
										
											2024-07-25 17:14:46 +02:00
										 |  |  |   metadataTab: Locator; | 
					
						
							| 
									
										
										
										
											2022-04-05 15:10:12 -08:00
										 |  |  |   snapshotContainer: Locator; | 
					
						
							| 
									
										
										
										
											2024-11-05 12:45:37 +01:00
										 |  |  |   sourceCodeTab: Locator; | 
					
						
							| 
									
										
										
										
											2022-04-05 15:10:12 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-01-08 05:08:00 -08:00
										 |  |  |   settingsDialog: Locator; | 
					
						
							|  |  |  |   darkModeSetting: Locator; | 
					
						
							|  |  |  |   displayCanvasContentSetting: Locator; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-04-05 15:10:12 -08:00
										 |  |  |   constructor(public page: Page) { | 
					
						
							|  |  |  |     this.actionTitles = page.locator('.action-title'); | 
					
						
							| 
									
										
										
										
											2024-11-05 12:45:37 +01:00
										 |  |  |     this.actionsTree = page.getByTestId('actions-tree'); | 
					
						
							| 
									
										
										
										
											2023-03-07 12:43:16 -08:00
										 |  |  |     this.callLines = page.locator('.call-tab .call-line'); | 
					
						
							| 
									
										
										
										
											2023-09-01 20:12:05 -07:00
										 |  |  |     this.logLines = page.getByTestId('log-list').locator('.list-view-entry'); | 
					
						
							| 
									
										
										
										
											2022-04-05 15:10:12 -08:00
										 |  |  |     this.consoleLines = page.locator('.console-line'); | 
					
						
							|  |  |  |     this.consoleLineMessages = page.locator('.console-line-message'); | 
					
						
							| 
									
										
										
										
											2024-02-02 16:41:08 -08:00
										 |  |  |     this.errorMessages = page.locator('.error-message'); | 
					
						
							| 
									
										
										
										
											2022-04-05 15:10:12 -08:00
										 |  |  |     this.consoleStacks = page.locator('.console-stack'); | 
					
						
							| 
									
										
										
										
											2023-09-07 17:14:39 -07:00
										 |  |  |     this.stackFrames = page.getByTestId('stack-trace-list').locator('.list-view-entry'); | 
					
						
							|  |  |  |     this.networkRequests = page.getByTestId('network-list').locator('.list-view-entry'); | 
					
						
							| 
									
										
										
										
											2023-03-21 07:40:54 -07:00
										 |  |  |     this.snapshotContainer = page.locator('.snapshot-container iframe.snapshot-visible[name=snapshot]'); | 
					
						
							| 
									
										
										
										
											2024-07-31 16:37:16 -07:00
										 |  |  |     this.metadataTab = page.getByTestId('metadata-view'); | 
					
						
							| 
									
										
										
										
											2024-11-05 12:45:37 +01:00
										 |  |  |     this.sourceCodeTab = page.getByTestId('source-code'); | 
					
						
							| 
									
										
										
										
											2025-01-08 05:08:00 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  |     this.settingsDialog = page.getByTestId('settings-toolbar-dialog'); | 
					
						
							|  |  |  |     this.darkModeSetting = page.locator('.setting').getByText('Dark mode'); | 
					
						
							|  |  |  |     this.displayCanvasContentSetting = page.locator('.setting').getByText('Display canvas content'); | 
					
						
							| 
									
										
										
										
											2022-04-05 15:10:12 -08:00
										 |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   async actionIconsText(action: string) { | 
					
						
							| 
									
										
										
										
											2024-10-17 16:57:45 -07:00
										 |  |  |     const entry = await this.page.waitForSelector(`.tree-view-entry:has-text("${action}")`); | 
					
						
							| 
									
										
										
										
											2022-04-05 15:10:12 -08:00
										 |  |  |     await entry.waitForSelector('.action-icon-value:visible'); | 
					
						
							|  |  |  |     return await entry.$$eval('.action-icon-value:visible', ee => ee.map(e => e.textContent)); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   async actionIcons(action: string) { | 
					
						
							| 
									
										
										
										
											2024-10-17 16:57:45 -07:00
										 |  |  |     return await this.page.waitForSelector(`.tree-view-entry:has-text("${action}") .action-icons`); | 
					
						
							| 
									
										
										
										
											2022-04-05 15:10:12 -08:00
										 |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-11-05 04:45:54 -08:00
										 |  |  |   @step | 
					
						
							|  |  |  |   async expandAction(title: string, ordinal: number = 0) { | 
					
						
							|  |  |  |     await this.actionsTree.locator('.tree-view-entry', { hasText: title }).nth(ordinal).locator('.codicon-chevron-right').click(); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-09-21 10:17:59 -07:00
										 |  |  |   @step | 
					
						
							| 
									
										
										
										
											2022-04-05 15:10:12 -08:00
										 |  |  |   async selectAction(title: string, ordinal: number = 0) { | 
					
						
							|  |  |  |     await this.page.locator(`.action-title:has-text("${title}")`).nth(ordinal).click(); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-10-02 13:30:44 +02:00
										 |  |  |   @step | 
					
						
							|  |  |  |   async hoverAction(title: string, ordinal: number = 0) { | 
					
						
							|  |  |  |     await this.page.locator(`.action-title:has-text("${title}")`).nth(ordinal).hover(); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-09-21 10:17:59 -07:00
										 |  |  |   @step | 
					
						
							| 
									
										
										
										
											2022-04-05 15:10:12 -08:00
										 |  |  |   async selectSnapshot(name: string) { | 
					
						
							| 
									
										
										
										
											2023-02-17 11:19:53 -08:00
										 |  |  |     await this.page.click(`.snapshot-tab .tabbed-pane-tab-label:has-text("${name}")`); | 
					
						
							| 
									
										
										
										
											2022-04-05 15:10:12 -08:00
										 |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-02-02 16:41:08 -08:00
										 |  |  |   async showErrorsTab() { | 
					
						
							|  |  |  |     await this.page.click('text="Errors"'); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-04-05 15:10:12 -08:00
										 |  |  |   async showConsoleTab() { | 
					
						
							|  |  |  |     await this.page.click('text="Console"'); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   async showSourceTab() { | 
					
						
							|  |  |  |     await this.page.click('text="Source"'); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   async showNetworkTab() { | 
					
						
							|  |  |  |     await this.page.click('text="Network"'); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-07-25 17:14:46 +02:00
										 |  |  |   async showMetadataTab() { | 
					
						
							|  |  |  |     await this.page.click('text="Metadata"'); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-01-08 05:08:00 -08:00
										 |  |  |   async showSettings() { | 
					
						
							|  |  |  |     await this.page.locator('.settings-gear').click(); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-03-29 20:43:08 -07:00
										 |  |  |   @step | 
					
						
							| 
									
										
										
										
											2023-03-21 07:40:54 -07:00
										 |  |  |   async snapshotFrame(actionName: string, ordinal: number = 0, hasSubframe: boolean = false): Promise<FrameLocator> { | 
					
						
							|  |  |  |     await this.selectAction(actionName, ordinal); | 
					
						
							|  |  |  |     while (this.page.frames().length < (hasSubframe ? 4 : 3)) | 
					
						
							| 
									
										
										
										
											2022-04-05 15:10:12 -08:00
										 |  |  |       await this.page.waitForEvent('frameattached'); | 
					
						
							| 
									
										
										
										
											2023-03-21 07:40:54 -07:00
										 |  |  |     return this.page.frameLocator('iframe.snapshot-visible[name=snapshot]'); | 
					
						
							| 
									
										
										
										
											2022-04-05 15:10:12 -08:00
										 |  |  |   } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | export const traceViewerFixtures: Fixtures<TraceViewerFixtures, {}, BaseTestFixtures, BaseWorkerFixtures> = { | 
					
						
							| 
									
										
										
										
											2023-08-04 14:59:48 -07:00
										 |  |  |   showTraceViewer: async ({ playwright, browserName, headless }, use, testInfo) => { | 
					
						
							| 
									
										
										
										
											2022-09-27 12:45:42 -07:00
										 |  |  |     const browsers: Browser[] = []; | 
					
						
							|  |  |  |     const contextImpls: any[] = []; | 
					
						
							| 
									
										
										
										
											2023-01-27 23:20:25 +01:00
										 |  |  |     await use(async (traces: string[], { host, port } = {}) => { | 
					
						
							| 
									
										
										
										
											2024-03-18 09:50:11 -07:00
										 |  |  |       const pageImpl = await runTraceViewerApp(traces, browserName, { headless, host, port }); | 
					
						
							| 
									
										
										
										
											2025-04-30 18:57:59 -07:00
										 |  |  |       const contextImpl = pageImpl.browserContext; | 
					
						
							| 
									
										
										
										
											2022-09-27 12:45:42 -07:00
										 |  |  |       const browser = await playwright.chromium.connectOverCDP(contextImpl._browser.options.wsEndpoint); | 
					
						
							|  |  |  |       browsers.push(browser); | 
					
						
							|  |  |  |       contextImpls.push(contextImpl); | 
					
						
							| 
									
										
										
										
											2022-04-05 15:10:12 -08:00
										 |  |  |       return new TraceViewerPage(browser.contexts()[0].pages()[0]); | 
					
						
							|  |  |  |     }); | 
					
						
							| 
									
										
										
										
											2022-09-27 12:45:42 -07:00
										 |  |  |     for (const browser of browsers) | 
					
						
							|  |  |  |       await browser.close(); | 
					
						
							|  |  |  |     for (const contextImpl of contextImpls) | 
					
						
							| 
									
										
										
										
											2023-10-16 20:32:13 -07:00
										 |  |  |       await contextImpl._browser.close({ reason: 'Trace viewer closed' }); | 
					
						
							| 
									
										
										
										
											2022-04-05 15:10:12 -08:00
										 |  |  |   }, | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   runAndTrace: async ({ context, showTraceViewer }, use, testInfo) => { | 
					
						
							| 
									
										
										
										
											2024-09-06 16:24:33 +02:00
										 |  |  |     await use(async (body: () => Promise<void>, optsOverrides = {}) => { | 
					
						
							| 
									
										
										
										
											2024-09-17 15:32:30 +02:00
										 |  |  |       const traceFile = testInfo.outputPath('trace.zip'); | 
					
						
							| 
									
										
										
										
											2024-09-06 16:24:33 +02:00
										 |  |  |       await context.tracing.start({ snapshots: true, screenshots: true, sources: true, ...optsOverrides }); | 
					
						
							| 
									
										
										
										
											2022-04-05 15:10:12 -08:00
										 |  |  |       await body(); | 
					
						
							|  |  |  |       await context.tracing.stop({ path: traceFile }); | 
					
						
							|  |  |  |       return showTraceViewer([traceFile]); | 
					
						
							|  |  |  |     }); | 
					
						
							|  |  |  |   }, | 
					
						
							| 
									
										
										
										
											2023-01-27 23:20:25 +01:00
										 |  |  | }; |