| 
									
										
										
										
											2021-04-02 09:00:49 -07: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. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-10-28 07:31:30 -08:00
										 |  |  | import { baseTest } from '../config/baseTest'; | 
					
						
							| 
									
										
										
										
											2025-03-17 19:41:44 +01:00
										 |  |  | import path from 'path'; | 
					
						
							|  |  |  | import fs from 'fs'; | 
					
						
							|  |  |  | import os from 'os'; | 
					
						
							| 
									
										
										
										
											2023-03-13 12:52:52 -07:00
										 |  |  | import type { ElectronApplication, Page, Electron } from '@playwright/test'; | 
					
						
							| 
									
										
										
										
											2022-04-06 13:57:14 -08:00
										 |  |  | import type { PageTestFixtures, PageWorkerFixtures } from '../page/pageTestApi'; | 
					
						
							|  |  |  | import type { TraceViewerFixtures } from '../config/traceViewerFixtures'; | 
					
						
							|  |  |  | import { traceViewerFixtures } from '../config/traceViewerFixtures'; | 
					
						
							| 
									
										
										
										
											2025-03-17 19:41:44 +01:00
										 |  |  | import { removeFolders } from '../../packages/playwright-core/lib/server/utils/fileUtils'; | 
					
						
							| 
									
										
										
										
											2021-10-11 10:52:17 -04:00
										 |  |  | export { expect } from '@playwright/test'; | 
					
						
							| 
									
										
										
										
											2021-04-02 09:00:49 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-16 19:58:26 -07:00
										 |  |  | type ElectronTestFixtures = PageTestFixtures & { | 
					
						
							| 
									
										
										
										
											2021-04-02 09:00:49 -07:00
										 |  |  |   electronApp: ElectronApplication; | 
					
						
							| 
									
										
										
										
											2023-03-13 12:52:52 -07:00
										 |  |  |   launchElectronApp: (appFile: string, args?: string[], options?: Parameters<Electron['launch']>[0]) => Promise<ElectronApplication>; | 
					
						
							| 
									
										
										
										
											2021-04-02 09:00:49 -07:00
										 |  |  |   newWindow: () => Promise<Page>; | 
					
						
							| 
									
										
										
										
											2025-03-17 19:41:44 +01:00
										 |  |  |   createUserDataDir: () => Promise<string>; | 
					
						
							| 
									
										
										
										
											2021-04-02 09:00:49 -07:00
										 |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-04-05 15:10:12 -08:00
										 |  |  | export const electronTest = baseTest.extend<TraceViewerFixtures>(traceViewerFixtures).extend<ElectronTestFixtures, PageWorkerFixtures>({ | 
					
						
							| 
									
										
										
										
											2024-01-03 19:49:37 +01:00
										 |  |  |   browserVersion: [({}, use) => use(process.env.ELECTRON_CHROMIUM_VERSION), { scope: 'worker' }], | 
					
						
							|  |  |  |   browserMajorVersion: [({}, use) =>  use(Number(process.env.ELECTRON_CHROMIUM_VERSION.split('.')[0])), { scope: 'worker' }], | 
					
						
							| 
									
										
										
										
											2024-06-21 00:43:26 +02:00
										 |  |  |   electronMajorVersion: [({}, use) => use(parseInt(require('electron/package.json').version.split('.')[0], 10)), { scope: 'worker' }], | 
					
						
							| 
									
										
										
										
											2021-10-28 07:31:30 -08:00
										 |  |  |   isAndroid: [false, { scope: 'worker' }], | 
					
						
							|  |  |  |   isElectron: [true, { scope: 'worker' }], | 
					
						
							| 
									
										
										
										
											2022-09-08 09:11:47 +02:00
										 |  |  |   isWebView2: [false, { scope: 'worker' }], | 
					
						
							| 
									
										
										
										
											2024-11-20 17:13:07 +01:00
										 |  |  |   isHeadlessShell: [false, { scope: 'worker' }], | 
					
						
							| 
									
										
										
										
											2021-04-29 11:11:32 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-03-17 19:41:44 +01:00
										 |  |  |   createUserDataDir: async ({ mode }, run) => { | 
					
						
							|  |  |  |     const dirs: string[] = []; | 
					
						
							|  |  |  |     // We do not put user data dir in testOutputPath,
 | 
					
						
							|  |  |  |     // because we do not want to upload them as test result artifacts.
 | 
					
						
							|  |  |  |     await run(async () => { | 
					
						
							|  |  |  |       const dir = await fs.promises.mkdtemp(path.join(os.tmpdir(), 'playwright-test-')); | 
					
						
							|  |  |  |       dirs.push(dir); | 
					
						
							|  |  |  |       return dir; | 
					
						
							|  |  |  |     }); | 
					
						
							|  |  |  |     await removeFolders(dirs); | 
					
						
							|  |  |  |   }, | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   launchElectronApp: async ({ playwright, createUserDataDir }, use) => { | 
					
						
							| 
									
										
										
										
											2021-04-29 11:11:32 -07:00
										 |  |  |     // This env prevents 'Electron Security Policy' console message.
 | 
					
						
							|  |  |  |     process.env['ELECTRON_DISABLE_SECURITY_WARNINGS'] = 'true'; | 
					
						
							| 
									
										
										
										
											2022-12-20 12:50:53 -08:00
										 |  |  |     const apps: ElectronApplication[] = []; | 
					
						
							| 
									
										
										
										
											2023-03-13 12:52:52 -07:00
										 |  |  |     await use(async (appFile: string, args: string[] = [], options?: Parameters<Electron['launch']>[0]) => { | 
					
						
							| 
									
										
										
										
											2025-03-17 19:41:44 +01:00
										 |  |  |       const userDataDir = await createUserDataDir(); | 
					
						
							|  |  |  |       const app = await playwright._electron.launch({ | 
					
						
							|  |  |  |         ...options, | 
					
						
							|  |  |  |         args: [path.join(__dirname, appFile), ...args], | 
					
						
							|  |  |  |         env: { | 
					
						
							|  |  |  |           ...process.env, | 
					
						
							|  |  |  |           PWTEST_ELECTRON_USER_DATA_DIR: userDataDir, | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |       }); | 
					
						
							| 
									
										
										
										
											2022-12-20 12:50:53 -08:00
										 |  |  |       apps.push(app); | 
					
						
							|  |  |  |       return app; | 
					
						
							| 
									
										
										
										
											2021-04-29 11:11:32 -07:00
										 |  |  |     }); | 
					
						
							| 
									
										
										
										
											2022-12-20 12:50:53 -08:00
										 |  |  |     for (const app of apps) | 
					
						
							|  |  |  |       await app.close(); | 
					
						
							|  |  |  |   }, | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   electronApp: async ({ launchElectronApp }, use) => { | 
					
						
							|  |  |  |     await use(await launchElectronApp('electron-app.js')); | 
					
						
							| 
									
										
										
										
											2021-05-16 19:58:26 -07:00
										 |  |  |   }, | 
					
						
							| 
									
										
										
										
											2021-04-29 11:11:32 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-16 19:58:26 -07:00
										 |  |  |   newWindow: async ({ electronApp }, run) => { | 
					
						
							|  |  |  |     const windows: Page[] = []; | 
					
						
							|  |  |  |     await run(async () => { | 
					
						
							| 
									
										
										
										
											2022-08-18 20:12:33 +02:00
										 |  |  |       const [window] = await Promise.all([ | 
					
						
							| 
									
										
										
										
											2021-05-16 19:58:26 -07:00
										 |  |  |         electronApp.waitForEvent('window'), | 
					
						
							| 
									
										
										
										
											2021-12-08 17:34:50 -08:00
										 |  |  |         electronApp.evaluate(async electron => { | 
					
						
							|  |  |  |           // Avoid "Error: Cannot create BrowserWindow before app is ready".
 | 
					
						
							|  |  |  |           await electron.app.whenReady(); | 
					
						
							| 
									
										
										
										
											2021-05-16 19:58:26 -07:00
										 |  |  |           const window = new electron.BrowserWindow({ | 
					
						
							|  |  |  |             width: 800, | 
					
						
							|  |  |  |             height: 600, | 
					
						
							|  |  |  |             // Sandboxed windows share process with their window.open() children
 | 
					
						
							|  |  |  |             // and can script them. We use that heavily in our tests.
 | 
					
						
							|  |  |  |             webPreferences: { sandbox: true } | 
					
						
							|  |  |  |           }); | 
					
						
							| 
									
										
										
										
											2023-06-02 21:59:12 +02:00
										 |  |  |           await window.loadURL('about:blank'); | 
					
						
							| 
									
										
										
										
											2021-05-16 19:58:26 -07:00
										 |  |  |         }) | 
					
						
							|  |  |  |       ]); | 
					
						
							|  |  |  |       windows.push(window); | 
					
						
							|  |  |  |       return window; | 
					
						
							|  |  |  |     }); | 
					
						
							|  |  |  |     for (const window of windows) | 
					
						
							| 
									
										
										
										
											2021-04-29 11:11:32 -07:00
										 |  |  |       await window.close(); | 
					
						
							| 
									
										
										
										
											2021-05-16 19:58:26 -07:00
										 |  |  |   }, | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   page: async ({ newWindow }, run) => { | 
					
						
							|  |  |  |     await run(await newWindow()); | 
					
						
							|  |  |  |   }, | 
					
						
							| 
									
										
										
										
											2022-04-05 15:10:12 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  |   context: async ({ electronApp }, run) => { | 
					
						
							|  |  |  |     await run(electronApp.context()); | 
					
						
							|  |  |  |   }, | 
					
						
							| 
									
										
										
										
											2021-10-28 07:31:30 -08:00
										 |  |  | }); |