| 
									
										
										
										
											2021-09-21 16:24:48 -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-11 10:52:17 -04:00
										 |  |  | import type { Fixtures } from '@playwright/test'; | 
					
						
							| 
									
										
										
										
											2022-04-06 13:57:14 -08:00
										 |  |  | import type { ChildProcess } from 'child_process'; | 
					
						
							|  |  |  | import { execSync, spawn } from 'child_process'; | 
					
						
							| 
									
										
										
										
											2021-09-21 16:24:48 -07:00
										 |  |  | import net from 'net'; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | type TestChildParams = { | 
					
						
							|  |  |  |   command: string[], | 
					
						
							|  |  |  |   cwd?: string, | 
					
						
							| 
									
										
										
										
											2021-10-28 23:08:04 +02:00
										 |  |  |   env?: NodeJS.ProcessEnv, | 
					
						
							| 
									
										
										
										
											2021-09-21 16:24:48 -07:00
										 |  |  |   shell?: boolean, | 
					
						
							|  |  |  |   onOutput?: () => void; | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | export class TestChildProcess { | 
					
						
							|  |  |  |   params: TestChildParams; | 
					
						
							|  |  |  |   process: ChildProcess; | 
					
						
							|  |  |  |   output = ''; | 
					
						
							|  |  |  |   onOutput?: () => void; | 
					
						
							| 
									
										
										
										
											2022-06-18 15:47:26 -07:00
										 |  |  |   exited: Promise<{ exitCode: number, signal: string | null }>; | 
					
						
							| 
									
										
										
										
											2021-09-21 16:24:48 -07:00
										 |  |  |   exitCode: Promise<number>; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-09-22 21:13:32 -07:00
										 |  |  |   private _outputCallbacks = new Set<() => void>(); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-09-21 16:24:48 -07:00
										 |  |  |   constructor(params: TestChildParams) { | 
					
						
							|  |  |  |     this.params = params; | 
					
						
							|  |  |  |     this.process = spawn(params.command[0], params.command.slice(1), { | 
					
						
							|  |  |  |       env: { | 
					
						
							|  |  |  |         ...process.env, | 
					
						
							|  |  |  |         ...params.env, | 
					
						
							| 
									
										
										
										
											2021-10-28 23:08:04 +02:00
										 |  |  |       }, | 
					
						
							| 
									
										
										
										
											2021-09-21 16:24:48 -07:00
										 |  |  |       cwd: params.cwd, | 
					
						
							|  |  |  |       shell: params.shell, | 
					
						
							| 
									
										
										
										
											2021-10-28 23:08:04 +02:00
										 |  |  |       // On non-windows platforms, `detached: true` makes child process a leader of a new
 | 
					
						
							|  |  |  |       // process group, making it possible to kill child process tree with `.kill(-pid)` command.
 | 
					
						
							|  |  |  |       // @see https://nodejs.org/api/child_process.html#child_process_options_detached
 | 
					
						
							|  |  |  |       detached: process.platform !== 'win32', | 
					
						
							| 
									
										
										
										
											2021-09-21 16:24:48 -07:00
										 |  |  |     }); | 
					
						
							|  |  |  |     if (process.env.PWTEST_DEBUG) | 
					
						
							|  |  |  |       process.stdout.write(`\n\nLaunching ${params.command.join(' ')}\n`); | 
					
						
							|  |  |  |     this.onOutput = params.onOutput; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     const appendChunk = (chunk: string | Buffer) => { | 
					
						
							|  |  |  |       this.output += String(chunk); | 
					
						
							|  |  |  |       if (process.env.PWTEST_DEBUG) | 
					
						
							|  |  |  |         process.stdout.write(String(chunk)); | 
					
						
							|  |  |  |       this.onOutput?.(); | 
					
						
							| 
									
										
										
										
											2021-09-22 21:13:32 -07:00
										 |  |  |       for (const cb of this._outputCallbacks) | 
					
						
							|  |  |  |         cb(); | 
					
						
							|  |  |  |       this._outputCallbacks.clear(); | 
					
						
							| 
									
										
										
										
											2021-09-21 16:24:48 -07:00
										 |  |  |     }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     this.process.stderr.on('data', appendChunk); | 
					
						
							|  |  |  |     this.process.stdout.on('data', appendChunk); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-10-28 23:08:04 +02:00
										 |  |  |     const killProcessGroup = this._killProcessGroup.bind(this); | 
					
						
							|  |  |  |     process.on('exit', killProcessGroup); | 
					
						
							| 
									
										
										
										
											2021-09-21 16:24:48 -07:00
										 |  |  |     this.exited = new Promise(f => { | 
					
						
							|  |  |  |       this.process.on('exit', (exitCode, signal) => f({ exitCode, signal })); | 
					
						
							| 
									
										
										
										
											2021-10-28 23:08:04 +02:00
										 |  |  |       process.off('exit', killProcessGroup); | 
					
						
							| 
									
										
										
										
											2021-09-21 16:24:48 -07:00
										 |  |  |     }); | 
					
						
							|  |  |  |     this.exitCode = this.exited.then(r => r.exitCode); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   async close() { | 
					
						
							|  |  |  |     if (!this.process.killed) | 
					
						
							| 
									
										
										
										
											2021-10-28 23:08:04 +02:00
										 |  |  |       this._killProcessGroup(); | 
					
						
							| 
									
										
										
										
											2021-09-21 16:24:48 -07:00
										 |  |  |     return this.exited; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-10-28 23:08:04 +02:00
										 |  |  |   private _killProcessGroup() { | 
					
						
							|  |  |  |     if (!this.process.pid || this.process.killed) | 
					
						
							|  |  |  |       return; | 
					
						
							|  |  |  |     try { | 
					
						
							|  |  |  |       if (process.platform === 'win32') | 
					
						
							| 
									
										
										
										
											2021-11-11 18:32:22 +01:00
										 |  |  |         execSync(`taskkill /pid ${this.process.pid} /T /F /FI "MEMUSAGE gt 0"`, { stdio: 'ignore' }); | 
					
						
							| 
									
										
										
										
											2021-10-28 23:08:04 +02:00
										 |  |  |       else | 
					
						
							|  |  |  |         process.kill(-this.process.pid, 'SIGKILL'); | 
					
						
							|  |  |  |     } catch (e) { | 
					
						
							|  |  |  |       // the process might have already stopped
 | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-09-21 16:24:48 -07:00
										 |  |  |   async cleanExit() { | 
					
						
							|  |  |  |     const r = await this.exited; | 
					
						
							|  |  |  |     if (r.exitCode) | 
					
						
							|  |  |  |       throw new Error(`Process failed with exit code ${r.exitCode}`); | 
					
						
							|  |  |  |     if (r.signal) | 
					
						
							| 
									
										
										
										
											2022-02-01 16:09:41 -03:00
										 |  |  |       throw new Error(`Process received signal: ${r.signal}`); | 
					
						
							| 
									
										
										
										
											2021-09-21 16:24:48 -07:00
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2021-09-22 21:13:32 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  |   async waitForOutput(substring: string) { | 
					
						
							|  |  |  |     while (!this.output.includes(substring)) | 
					
						
							|  |  |  |       await new Promise<void>(f => this._outputCallbacks.add(f)); | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2021-09-21 16:24:48 -07:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | export type CommonFixtures = { | 
					
						
							|  |  |  |   childProcess: (params: TestChildParams) => TestChildProcess; | 
					
						
							|  |  |  |   waitForPort: (port: number) => Promise<void>; | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-11-03 13:47:51 -07:00
										 |  |  | export type CommonWorkerFixtures = { | 
					
						
							|  |  |  |   daemonProcess: (params: TestChildParams) => TestChildProcess; | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | export const commonFixtures: Fixtures<CommonFixtures, CommonWorkerFixtures> = { | 
					
						
							| 
									
										
										
										
											2021-09-21 16:24:48 -07:00
										 |  |  |   childProcess: async ({}, use, testInfo) => { | 
					
						
							|  |  |  |     const processes: TestChildProcess[] = []; | 
					
						
							|  |  |  |     await use(params => { | 
					
						
							|  |  |  |       const process = new TestChildProcess(params); | 
					
						
							|  |  |  |       processes.push(process); | 
					
						
							|  |  |  |       return process; | 
					
						
							|  |  |  |     }); | 
					
						
							|  |  |  |     await Promise.all(processes.map(child => child.close())); | 
					
						
							| 
									
										
										
										
											2021-11-18 14:35:51 -08:00
										 |  |  |     if (testInfo.status !== 'passed' && testInfo.status !== 'skipped' && !process.env.PWTEST_DEBUG) { | 
					
						
							| 
									
										
										
										
											2021-09-21 16:24:48 -07:00
										 |  |  |       for (const process of processes) { | 
					
						
							|  |  |  |         console.log('====== ' + process.params.command.join(' ')); | 
					
						
							|  |  |  |         console.log(process.output); | 
					
						
							|  |  |  |         console.log('========================================='); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   }, | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-11-03 13:47:51 -07:00
										 |  |  |   daemonProcess: [async ({}, use) => { | 
					
						
							|  |  |  |     const processes: TestChildProcess[] = []; | 
					
						
							|  |  |  |     await use(params => { | 
					
						
							|  |  |  |       const process = new TestChildProcess(params); | 
					
						
							|  |  |  |       processes.push(process); | 
					
						
							|  |  |  |       return process; | 
					
						
							|  |  |  |     }); | 
					
						
							|  |  |  |     await Promise.all(processes.map(child => child.close())); | 
					
						
							|  |  |  |   }, { scope: 'worker' }], | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-09-21 16:24:48 -07:00
										 |  |  |   waitForPort: async ({}, use) => { | 
					
						
							|  |  |  |     const token = { canceled: false }; | 
					
						
							|  |  |  |     await use(async port => { | 
					
						
							|  |  |  |       while (!token.canceled) { | 
					
						
							|  |  |  |         const promise = new Promise<boolean>(resolve => { | 
					
						
							| 
									
										
										
										
											2022-04-22 13:42:52 +02:00
										 |  |  |           const conn = net.connect(port, '127.0.0.1') | 
					
						
							| 
									
										
										
										
											2021-09-21 16:24:48 -07:00
										 |  |  |               .on('error', () => resolve(false)) | 
					
						
							|  |  |  |               .on('connect', () => { | 
					
						
							|  |  |  |                 conn.end(); | 
					
						
							|  |  |  |                 resolve(true); | 
					
						
							|  |  |  |               }); | 
					
						
							|  |  |  |         }); | 
					
						
							|  |  |  |         if (await promise) | 
					
						
							|  |  |  |           return; | 
					
						
							|  |  |  |         await new Promise(x => setTimeout(x, 100)); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     }); | 
					
						
							|  |  |  |     token.canceled = true; | 
					
						
							|  |  |  |   }, | 
					
						
							|  |  |  | }; |