| 
									
										
										
										
											2020-07-27 13:02:28 -07:00
										 |  |  | #!/usr/bin/env node
 | 
					
						
							|  |  |  | /** | 
					
						
							|  |  |  |  * Copyright 2019 Google Inc. All rights reserved. | 
					
						
							|  |  |  |  * Modifications 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-01-01 15:17:27 -08:00
										 |  |  | const fs = require('fs'); | 
					
						
							| 
									
										
										
										
											2020-07-27 13:02:28 -07:00
										 |  |  | const ts = require('typescript'); | 
					
						
							|  |  |  | const path = require('path'); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | async function checkDeps() { | 
					
						
							| 
									
										
										
										
											2021-12-01 18:14:13 -08:00
										 |  |  |   const packages = path.normalize(path.join(__dirname, '..', 'packages')); | 
					
						
							|  |  |  |   const corePackageJson = await innerCheckDeps(path.join(packages, 'playwright-core'), true); | 
					
						
							|  |  |  |   const testPackageJson = await innerCheckDeps(path.join(packages, 'playwright-test'), false); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   let hasVersionMismatch = false; | 
					
						
							|  |  |  |   for (const [key, value] of Object.entries(corePackageJson.dependencies)) { | 
					
						
							|  |  |  |     const value2 = testPackageJson.dependencies[key]; | 
					
						
							|  |  |  |     if (value2 && value2 !== value) { | 
					
						
							|  |  |  |       hasVersionMismatch = true; | 
					
						
							|  |  |  |       console.log(`Dependency version mismatch ${key}: ${value} != ${value2}`); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   process.exit(hasVersionMismatch ? 1 : 0); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | async function innerCheckDeps(root, checkDepsFile) { | 
					
						
							|  |  |  |   const deps = new Set(); | 
					
						
							|  |  |  |   const src = path.join(root, 'src'); | 
					
						
							| 
									
										
										
										
											2021-09-02 19:56:30 +02:00
										 |  |  |   const packageJSON = require(path.join(root, 'package.json')); | 
					
						
							| 
									
										
										
										
											2020-07-27 13:02:28 -07:00
										 |  |  |   const program = ts.createProgram({ | 
					
						
							|  |  |  |     options: { | 
					
						
							|  |  |  |       allowJs: true, | 
					
						
							|  |  |  |       target: ts.ScriptTarget.ESNext, | 
					
						
							|  |  |  |       strict: true, | 
					
						
							|  |  |  |     }, | 
					
						
							| 
									
										
										
										
											2021-01-01 15:17:27 -08:00
										 |  |  |     rootNames: listAllFiles(src), | 
					
						
							| 
									
										
										
										
											2020-07-27 13:02:28 -07:00
										 |  |  |   }); | 
					
						
							|  |  |  |   const sourceFiles = program.getSourceFiles(); | 
					
						
							|  |  |  |   const errors = []; | 
					
						
							| 
									
										
										
										
											2020-12-22 11:01:25 -08:00
										 |  |  |   const usedDeps = new Set(); | 
					
						
							| 
									
										
										
										
											2020-07-27 13:02:28 -07:00
										 |  |  |   sourceFiles.filter(x => !x.fileName.includes('node_modules')).map(x => visit(x, x.fileName)); | 
					
						
							| 
									
										
										
										
											2020-12-22 11:01:25 -08:00
										 |  |  |   for (const key of Object.keys(DEPS)) { | 
					
						
							|  |  |  |     if (!usedDeps.has(key) && DEPS[key].length) | 
					
						
							|  |  |  |       errors.push(`Stale DEPS entry "${key}"`); | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2021-12-01 18:14:13 -08:00
										 |  |  |   if (checkDepsFile && errors.length) { | 
					
						
							|  |  |  |     for (const error of errors) | 
					
						
							|  |  |  |       console.log(error); | 
					
						
							| 
									
										
										
										
											2020-08-24 14:48:03 -07:00
										 |  |  |     console.log(`--------------------------------------------------------`); | 
					
						
							|  |  |  |     console.log(`Changing the project structure or adding new components?`); | 
					
						
							| 
									
										
										
										
											2021-09-02 19:56:30 +02:00
										 |  |  |     console.log(`Update DEPS in ./${path.relative(root, __filename)}`); | 
					
						
							| 
									
										
										
										
											2020-08-24 14:48:03 -07:00
										 |  |  |     console.log(`--------------------------------------------------------`); | 
					
						
							| 
									
										
										
										
											2021-12-01 18:14:13 -08:00
										 |  |  |     process.exit(1); | 
					
						
							| 
									
										
										
										
											2020-08-24 14:48:03 -07:00
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2021-12-01 18:14:13 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  |   for (const dep of deps) { | 
					
						
							| 
									
										
										
										
											2022-02-01 21:27:34 +01:00
										 |  |  |     const resolved = require.resolve(dep, { paths: [root] }); | 
					
						
							| 
									
										
										
										
											2021-12-01 18:14:13 -08:00
										 |  |  |     if (dep === resolved || !resolved.includes('node_modules')) | 
					
						
							|  |  |  |       deps.delete(dep); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   for (const dep of Object.keys(packageJSON.dependencies)) | 
					
						
							|  |  |  |     deps.delete(dep); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (deps.size) { | 
					
						
							|  |  |  |     console.log('Dependencies are not declared in package.json:'); | 
					
						
							|  |  |  |     for (const dep of deps) | 
					
						
							|  |  |  |       console.log(`  ${dep}`); | 
					
						
							|  |  |  |     process.exit(1); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   return packageJSON; | 
					
						
							| 
									
										
										
										
											2020-07-27 13:02:28 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  |   function visit(node, fileName) { | 
					
						
							|  |  |  |     if (ts.isImportDeclaration(node) && ts.isStringLiteral(node.moduleSpecifier)) { | 
					
						
							|  |  |  |       const importName = node.moduleSpecifier.text; | 
					
						
							| 
									
										
										
										
											2021-12-01 18:14:13 -08:00
										 |  |  |       if (!importName.startsWith('.') && !node.importClause.isTypeOnly && !fileName.includes(path.sep + 'web' + path.sep)) { | 
					
						
							|  |  |  |         if (importName.startsWith('@')) | 
					
						
							|  |  |  |           deps.add(importName.split('/').slice(0, 2).join('/')); | 
					
						
							|  |  |  |         else | 
					
						
							|  |  |  |           deps.add(importName.split('/')[0]); | 
					
						
							|  |  |  |       } | 
					
						
							| 
									
										
										
										
											2020-08-23 21:24:16 -07:00
										 |  |  |       const importPath = path.resolve(path.dirname(fileName), importName) + '.ts'; | 
					
						
							| 
									
										
										
										
											2021-12-01 18:14:13 -08:00
										 |  |  |       if (checkDepsFile && !allowImport(fileName, importPath)) | 
					
						
							| 
									
										
										
										
											2021-12-09 17:21:17 -08:00
										 |  |  |         errors.push(`Disallowed import ${path.relative(root, importPath)} in ${path.relative(root, fileName)}`); | 
					
						
							| 
									
										
										
										
											2021-12-01 18:14:13 -08:00
										 |  |  |       if (checkDepsFile && !allowExternalImport(fileName, importPath, importName)) | 
					
						
							| 
									
										
										
										
											2021-09-02 19:56:30 +02:00
										 |  |  |         errors.push(`Disallowed external dependency ${importName} from ${path.relative(root, fileName)}`); | 
					
						
							| 
									
										
										
										
											2020-07-27 13:02:28 -07:00
										 |  |  |     } | 
					
						
							|  |  |  |     ts.forEachChild(node, x => visit(x, fileName)); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   function allowImport(from, to) { | 
					
						
							| 
									
										
										
										
											2020-12-22 11:01:25 -08:00
										 |  |  |     if (!to.startsWith(src + path.sep)) | 
					
						
							| 
									
										
										
										
											2020-12-09 15:06:57 -08:00
										 |  |  |       return true; | 
					
						
							| 
									
										
										
										
											2021-01-06 09:31:42 -08:00
										 |  |  |     if (!fs.existsSync(to)) | 
					
						
							|  |  |  |       return true; | 
					
						
							| 
									
										
										
										
											2020-12-22 11:01:25 -08:00
										 |  |  |     from = path.relative(root, from).replace(/\\/g, '/'); | 
					
						
							|  |  |  |     to = path.relative(root, to).replace(/\\/g, '/'); | 
					
						
							| 
									
										
										
										
											2020-08-23 21:24:16 -07:00
										 |  |  |     const fromDirectory = from.substring(0, from.lastIndexOf('/') + 1); | 
					
						
							| 
									
										
										
										
											2020-08-22 15:46:42 -07:00
										 |  |  |     const toDirectory = to.substring(0, to.lastIndexOf('/') + 1); | 
					
						
							| 
									
										
										
										
											2020-08-23 21:24:16 -07:00
										 |  |  |     if (fromDirectory === toDirectory) | 
					
						
							|  |  |  |       return true; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     while (!DEPS[from]) { | 
					
						
							|  |  |  |       if (from.endsWith('/')) | 
					
						
							|  |  |  |         from = from.substring(0, from.length - 1); | 
					
						
							|  |  |  |       if (from.lastIndexOf('/') === -1) | 
					
						
							| 
									
										
										
										
											2020-08-24 14:48:03 -07:00
										 |  |  |         throw new Error(`Cannot find DEPS for ${fromDirectory}`); | 
					
						
							| 
									
										
										
										
											2020-08-23 21:24:16 -07:00
										 |  |  |       from = from.substring(0, from.lastIndexOf('/') + 1); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-12-22 11:01:25 -08:00
										 |  |  |     usedDeps.add(from); | 
					
						
							| 
									
										
										
										
											2020-08-24 14:48:03 -07:00
										 |  |  |     for (const dep of DEPS[from]) { | 
					
						
							| 
									
										
										
										
											2020-08-23 21:24:16 -07:00
										 |  |  |       if (to === dep || toDirectory === dep) | 
					
						
							|  |  |  |         return true; | 
					
						
							|  |  |  |       if (dep.endsWith('**')) { | 
					
						
							|  |  |  |         const parent = dep.substring(0, dep.length - 2); | 
					
						
							|  |  |  |         if (to.startsWith(parent)) | 
					
						
							| 
									
										
										
										
											2020-08-22 07:07:13 -07:00
										 |  |  |           return true; | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-08-23 21:24:16 -07:00
										 |  |  |     return false; | 
					
						
							| 
									
										
										
										
											2020-07-27 13:02:28 -07:00
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2021-09-02 19:56:30 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-10-08 08:01:31 -07:00
										 |  |  |   function allowExternalImport(from, importPath, importName) { | 
					
						
							| 
									
										
										
										
											2021-09-02 19:56:30 +02:00
										 |  |  |     const EXTERNAL_IMPORT_ALLOWLIST = ['electron']; | 
					
						
							|  |  |  |     // Only external imports are relevant. Files in src/web are bundled via webpack.
 | 
					
						
							|  |  |  |     if (importName.startsWith('.') || importPath.startsWith(path.join(src, 'web'))) | 
					
						
							|  |  |  |       return true; | 
					
						
							|  |  |  |     if (EXTERNAL_IMPORT_ALLOWLIST.includes(importName)) | 
					
						
							|  |  |  |       return true; | 
					
						
							|  |  |  |     try { | 
					
						
							| 
									
										
										
										
											2021-10-08 08:01:31 -07:00
										 |  |  |       const resolvedImport = require.resolve(importName); | 
					
						
							| 
									
										
										
										
											2021-09-02 19:56:30 +02:00
										 |  |  |       const resolvedImportRelativeToNodeModules = path.relative(path.join(root, 'node_modules'), resolvedImport); | 
					
						
							|  |  |  |       // Filter out internal Node.js modules
 | 
					
						
							|  |  |  |       if (!resolvedImportRelativeToNodeModules.startsWith(importName)) | 
					
						
							|  |  |  |         return true; | 
					
						
							|  |  |  |       const resolvedImportRelativeToNodeModulesParts = resolvedImportRelativeToNodeModules.split(path.sep); | 
					
						
							|  |  |  |       if (packageJSON.dependencies[resolvedImportRelativeToNodeModulesParts[0]]) | 
					
						
							|  |  |  |         return true; | 
					
						
							|  |  |  |       // handle e.g. @babel/code-frame
 | 
					
						
							|  |  |  |       if (resolvedImportRelativeToNodeModulesParts.length >= 2 && packageJSON.dependencies[resolvedImportRelativeToNodeModulesParts.splice(0, 2).join(path.sep)]) | 
					
						
							|  |  |  |         return true; | 
					
						
							|  |  |  |       return false; | 
					
						
							|  |  |  |     } catch (error) { | 
					
						
							|  |  |  |       if (error.code !== 'MODULE_NOT_FOUND') | 
					
						
							| 
									
										
										
										
											2021-10-08 08:01:31 -07:00
										 |  |  |         throw error; | 
					
						
							| 
									
										
										
										
											2021-09-02 19:56:30 +02:00
										 |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2020-07-27 13:02:28 -07:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-01-01 15:17:27 -08:00
										 |  |  | function listAllFiles(dir) { | 
					
						
							|  |  |  |   const dirs = fs.readdirSync(dir, { withFileTypes: true }); | 
					
						
							| 
									
										
										
										
											2021-09-02 19:56:30 +02:00
										 |  |  |   const result = []; | 
					
						
							| 
									
										
										
										
											2021-01-01 15:17:27 -08:00
										 |  |  |   dirs.map(d => { | 
					
						
							|  |  |  |     const res = path.resolve(dir, d.name); | 
					
						
							|  |  |  |     if (d.isDirectory()) | 
					
						
							|  |  |  |       result.push(...listAllFiles(res)); | 
					
						
							|  |  |  |     else | 
					
						
							|  |  |  |       result.push(res); | 
					
						
							|  |  |  |   }); | 
					
						
							|  |  |  |   return result; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-08-23 21:24:16 -07:00
										 |  |  | const DEPS = {}; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | DEPS['src/protocol/'] = ['src/utils/']; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-08-24 06:51:51 -07:00
										 |  |  | // Client depends on chromium protocol for types.
 | 
					
						
							| 
									
										
										
										
											2021-06-23 18:01:48 -07:00
										 |  |  | DEPS['src/client/'] = ['src/common/', 'src/utils/', 'src/protocol/', 'src/server/chromium/protocol.d.ts']; | 
					
						
							| 
									
										
										
										
											2022-02-02 21:26:45 -08:00
										 |  |  | DEPS['src/outofprocess.ts'] = ['src/client/', 'src/protocol/', 'src/utils/']; | 
					
						
							| 
									
										
										
										
											2020-08-23 21:24:16 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-02-10 21:50:29 -08:00
										 |  |  | DEPS['src/dispatchers/'] = ['src/common/', 'src/utils/', 'src/protocol/', 'src/server/**']; | 
					
						
							| 
									
										
										
										
											2020-08-24 14:48:03 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  | // Generic dependencies for server-side code.
 | 
					
						
							| 
									
										
										
										
											2020-08-24 06:51:51 -07:00
										 |  |  | DEPS['src/server/'] = [ | 
					
						
							| 
									
										
										
										
											2021-02-10 18:52:28 -08:00
										 |  |  |   'src/common/', | 
					
						
							| 
									
										
										
										
											2020-08-24 06:51:51 -07:00
										 |  |  |   'src/utils/', | 
					
						
							|  |  |  |   'src/generated/', | 
					
						
							| 
									
										
										
										
											2020-08-24 14:48:03 -07:00
										 |  |  |   // Can depend on files directly in the server directory.
 | 
					
						
							|  |  |  |   'src/server/', | 
					
						
							|  |  |  |   // Can depend on any files in these subdirectories.
 | 
					
						
							|  |  |  |   'src/server/common/**', | 
					
						
							|  |  |  |   'src/server/injected/**', | 
					
						
							| 
									
										
										
										
											2021-01-24 19:21:19 -08:00
										 |  |  |   'src/server/supplements/**', | 
					
						
							| 
									
										
										
										
											2021-01-25 14:49:26 -08:00
										 |  |  |   'src/protocol/**', | 
					
						
							| 
									
										
										
										
											2020-08-24 06:51:51 -07:00
										 |  |  | ]; | 
					
						
							| 
									
										
										
										
											2020-08-23 21:24:16 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-08-24 14:48:03 -07:00
										 |  |  | // No dependencies for code shared between node and page.
 | 
					
						
							|  |  |  | DEPS['src/server/common/'] = []; | 
					
						
							|  |  |  | // Strict dependencies for injected code.
 | 
					
						
							| 
									
										
										
										
											2021-09-23 16:46:46 -07:00
										 |  |  | DEPS['src/server/injected/'] = ['src/server/common/', 'src/protocol/channels.ts']; | 
					
						
							| 
									
										
										
										
											2021-01-24 19:21:19 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-11-06 16:31:11 -08:00
										 |  |  | // Electron and Clank use chromium internally.
 | 
					
						
							| 
									
										
										
										
											2020-12-22 11:01:25 -08:00
										 |  |  | DEPS['src/server/android/'] = [...DEPS['src/server/'], 'src/server/chromium/', 'src/protocol/']; | 
					
						
							| 
									
										
										
										
											2020-08-24 14:48:03 -07:00
										 |  |  | DEPS['src/server/electron/'] = [...DEPS['src/server/'], 'src/server/chromium/']; | 
					
						
							| 
									
										
										
										
											2020-08-23 21:24:16 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-23 18:34:52 -07:00
										 |  |  | DEPS['src/server/playwright.ts'] = [...DEPS['src/server/'], 'src/server/chromium/', 'src/server/webkit/', 'src/server/firefox/', 'src/server/android/', 'src/server/electron/']; | 
					
						
							| 
									
										
										
										
											2021-04-27 11:07:07 -07:00
										 |  |  | DEPS['src/server/browserContext.ts'] = [...DEPS['src/server/'], 'src/server/trace/recorder/tracing.ts']; | 
					
						
							| 
									
										
										
										
											2022-01-22 11:25:13 -08:00
										 |  |  | DEPS['src/server/fetch.ts'] = [...DEPS['src/server/'], 'src/server/trace/recorder/tracing.ts']; | 
					
						
							| 
									
										
										
										
											2021-10-01 17:06:13 -07:00
										 |  |  | DEPS['src/cli/driver.ts'] = DEPS['src/inProcessFactory.ts'] = DEPS['src/browserServerImpl.ts'] = ['src/**']; | 
					
						
							| 
									
										
										
										
											2020-08-22 07:07:13 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-09-04 16:31:52 -07:00
										 |  |  | // Tracing is a client/server plugin, nothing should depend on it.
 | 
					
						
							| 
									
										
										
										
											2021-02-17 14:05:41 -08:00
										 |  |  | DEPS['src/web/recorder/'] = ['src/common/', 'src/web/', 'src/web/components/', 'src/server/supplements/recorder/recorderTypes.ts']; | 
					
						
							| 
									
										
										
										
											2021-10-15 14:22:49 -08:00
										 |  |  | DEPS['src/web/traceViewer/'] = ['src/common/', 'src/web/', 'src/server/trace/common/', 'src/protocol/callMetadata.ts']; | 
					
						
							| 
									
										
										
										
											2021-07-02 16:45:09 -07:00
										 |  |  | DEPS['src/web/traceViewer/ui/'] = ['src/common/', 'src/protocol/', 'src/web/traceViewer/', 'src/web/', 'src/server/trace/viewer/', 'src/server/trace/', 'src/server/trace/common/', 'src/server/snapshot/snapshotTypes.ts', 'src/protocol/channels.ts']; | 
					
						
							| 
									
										
										
										
											2021-10-12 13:42:50 -08:00
										 |  |  | DEPS['src/web/traceViewer/inMemorySnapshotter.ts'] = ['src/**']; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-11-30 14:57:17 -08:00
										 |  |  | // The service is a cross-cutting feature, and so it depends on a bunch of things.
 | 
					
						
							| 
									
										
										
										
											2021-07-07 21:14:16 +02:00
										 |  |  | DEPS['src/remote/'] = ['src/client/', 'src/debug/', 'src/dispatchers/', 'src/server/', 'src/server/supplements/', 'src/server/electron/', 'src/server/trace/', 'src/utils/**']; | 
					
						
							| 
									
										
										
										
											2020-11-30 14:57:17 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-12-22 14:54:13 -08:00
										 |  |  | // CLI should only use client-side features.
 | 
					
						
							| 
									
										
											  
											
												feat: introduce experimental general-purpose grid (#8941)
This patch adds a general-purpose grid framework to parallelize
Playwright across multiple agents.
This patch adds two CLI commands to manage grid:
- `npx playwright experimental-grid-server` - to launch grid
- `npx playwrigth experimental-grid-agent` - to launch agent in a host
  environment.
Grid server accepts an `--agent-factory` argument. A simple
`factory.js` might look like this:
```js
const child_process = require('child_process');
module.exports = {
  name: 'My Simple Factory',
  capacity: Infinity, // How many workers launch per agent
  timeout: 10_000, // 10 seconds timeout to create agent
  launch: ({agentId, gridURL, playwrightVersion}) => child_process.spawn(`npx`, [
    'playwright'
    'experimental-grid-agent',
    '--grid-url', gridURL,
    '--agent-id', agentId,
  ], {
    cwd: __dirname,
    shell: true,
    stdio: 'inherit',
  }),
};
```
With this `factory.js`, grid server could be launched like this:
```bash
npx playwright experimental-grid-server --factory=./factory.js
```
Once launched, it could be used with Playwright Test using env variable:
```bash
PW_GRID=http://localhost:3000 npx playwright test
```
											
										 
											2021-09-16 01:20:36 -07:00
										 |  |  | DEPS['src/cli/'] = ['src/cli/**', 'src/client/**', 'src/generated/', 'src/server/injected/', 'src/debug/injected/', 'src/server/trace/**', 'src/utils/**', 'src/grid/**']; | 
					
						
							| 
									
										
										
										
											2020-12-22 14:54:13 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-02-12 10:11:30 -08:00
										 |  |  | DEPS['src/server/supplements/recorder/recorderApp.ts'] = ['src/common/', 'src/utils/', 'src/server/', 'src/server/chromium/']; | 
					
						
							| 
									
										
										
										
											2021-02-25 13:09:26 -08:00
										 |  |  | DEPS['src/server/supplements/recorderSupplement.ts'] = ['src/server/snapshot/', ...DEPS['src/server/']]; | 
					
						
							| 
									
										
										
										
											2021-07-02 16:45:09 -07:00
										 |  |  | DEPS['src/utils/'] = ['src/common/', 'src/protocol/']; | 
					
						
							| 
									
										
										
										
											2021-01-31 16:37:13 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-02-24 13:39:51 -08:00
										 |  |  | // Trace viewer
 | 
					
						
							| 
									
										
										
										
											2021-02-25 09:33:32 -08:00
										 |  |  | DEPS['src/server/trace/common/'] = ['src/server/snapshot/', ...DEPS['src/server/']]; | 
					
						
							|  |  |  | DEPS['src/server/trace/recorder/'] = ['src/server/trace/common/', ...DEPS['src/server/trace/common/']]; | 
					
						
							| 
									
										
										
										
											2021-07-02 14:33:38 -07:00
										 |  |  | DEPS['src/server/trace/viewer/'] = ['src/server/trace/common/', 'src/server/trace/recorder/', 'src/server/chromium/', ...DEPS['src/server/trace/common/']]; | 
					
						
							| 
									
										
											  
											
												feat: introduce experimental general-purpose grid (#8941)
This patch adds a general-purpose grid framework to parallelize
Playwright across multiple agents.
This patch adds two CLI commands to manage grid:
- `npx playwright experimental-grid-server` - to launch grid
- `npx playwrigth experimental-grid-agent` - to launch agent in a host
  environment.
Grid server accepts an `--agent-factory` argument. A simple
`factory.js` might look like this:
```js
const child_process = require('child_process');
module.exports = {
  name: 'My Simple Factory',
  capacity: Infinity, // How many workers launch per agent
  timeout: 10_000, // 10 seconds timeout to create agent
  launch: ({agentId, gridURL, playwrightVersion}) => child_process.spawn(`npx`, [
    'playwright'
    'experimental-grid-agent',
    '--grid-url', gridURL,
    '--agent-id', agentId,
  ], {
    cwd: __dirname,
    shell: true,
    stdio: 'inherit',
  }),
};
```
With this `factory.js`, grid server could be launched like this:
```bash
npx playwright experimental-grid-server --factory=./factory.js
```
Once launched, it could be used with Playwright Test using env variable:
```bash
PW_GRID=http://localhost:3000 npx playwright test
```
											
										 
											2021-09-16 01:20:36 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-10-11 10:52:17 -04:00
										 |  |  | // TODO(einbinder) re-enable these checks
 | 
					
						
							|  |  |  | // // Playwright Test
 | 
					
						
							|  |  |  | // DEPS['src/test/'] = ['src/test/**', 'src/utils/utils.ts', 'src/utils/**', 'src/protocol/channels.ts'];
 | 
					
						
							|  |  |  | // DEPS['src/test/index.ts'] = [... DEPS['src/test/'], 'src/grid/gridClient.ts' ];
 | 
					
						
							| 
									
										
										
										
											2021-06-06 17:09:53 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
											  
											
												feat: introduce experimental general-purpose grid (#8941)
This patch adds a general-purpose grid framework to parallelize
Playwright across multiple agents.
This patch adds two CLI commands to manage grid:
- `npx playwright experimental-grid-server` - to launch grid
- `npx playwrigth experimental-grid-agent` - to launch agent in a host
  environment.
Grid server accepts an `--agent-factory` argument. A simple
`factory.js` might look like this:
```js
const child_process = require('child_process');
module.exports = {
  name: 'My Simple Factory',
  capacity: Infinity, // How many workers launch per agent
  timeout: 10_000, // 10 seconds timeout to create agent
  launch: ({agentId, gridURL, playwrightVersion}) => child_process.spawn(`npx`, [
    'playwright'
    'experimental-grid-agent',
    '--grid-url', gridURL,
    '--agent-id', agentId,
  ], {
    cwd: __dirname,
    shell: true,
    stdio: 'inherit',
  }),
};
```
With this `factory.js`, grid server could be launched like this:
```bash
npx playwright experimental-grid-server --factory=./factory.js
```
Once launched, it could be used with Playwright Test using env variable:
```bash
PW_GRID=http://localhost:3000 npx playwright test
```
											
										 
											2021-09-16 01:20:36 -07:00
										 |  |  | // Grid
 | 
					
						
							|  |  |  | DEPS['src/grid/'] = ['src/utils/**', 'src/dispatchers/**', 'src/server/', 'src/client/']; | 
					
						
							| 
									
										
										
										
											2021-08-05 13:36:47 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-11-30 14:57:17 -08:00
										 |  |  | checkDeps().catch(e => { | 
					
						
							|  |  |  |   console.error(e && e.stack ? e.stack : e); | 
					
						
							|  |  |  |   process.exit(1); | 
					
						
							|  |  |  | }); |