forked from mirrors/gecko-dev
		
	Bug 1791532 - [puppeteer] Sync puppeteer v17.1.2 r=webdriver-reviewers,jdescottes
Differential Revision: https://phabricator.services.mozilla.com/D160331
This commit is contained in:
		
							parent
							
								
									51aa9fcd1f
								
							
						
					
					
						commit
						428f90d6b8
					
				
					 99 changed files with 4253 additions and 2404 deletions
				
			
		
							
								
								
									
										17
									
								
								.hgignore
									
									
									
									
									
								
							
							
						
						
									
										17
									
								
								.hgignore
									
									
									
									
									
								
							|  | @ -108,20 +108,23 @@ _OPT\.OBJ/ | |||
| ^devtools/.*/node_modules/ | ||||
| 
 | ||||
| # Ignore node_module directories and npm artifacts | ||||
| ^remote/test/puppeteer/.github | ||||
| ^remote/test/puppeteer/.husky | ||||
| ^remote/test/puppeteer/.local-chromium/ | ||||
| ^remote/test/puppeteer/.local-firefox/ | ||||
| ^remote/test/puppeteer/.*\.tsbuildinfo | ||||
| ^remote/test/puppeteer/\.github | ||||
| ^remote/test/puppeteer/\.husky | ||||
| ^remote/test/puppeteer/\.local-chromium/ | ||||
| ^remote/test/puppeteer/\.local-firefox/ | ||||
| ^remote/test/puppeteer/coverage/ | ||||
| ^remote/test/puppeteer/docker/ | ||||
| ^remote/test/puppeteer/docs/puppeteer\.api\.json | ||||
| ^remote/test/puppeteer/experimental/ | ||||
| ^remote/test/puppeteer/lib/ | ||||
| ^remote/test/puppeteer/node_modules/ | ||||
| ^remote/test/puppeteer/package-lock.json | ||||
| ^remote/test/puppeteer/package-lock\.json | ||||
| ^remote/test/puppeteer/puppeteer.*\.tgz | ||||
| ^remote/test/puppeteer/src/generated | ||||
| ^remote/test/puppeteer/test/build | ||||
| ^remote/test/puppeteer/test/output-firefox | ||||
| ^remote/test/puppeteer/test/output-chromium | ||||
| ^remote/test/puppeteer/utils/testserver/tsconfig.tsbuildinfo | ||||
| ^remote/test/puppeteer/website | ||||
| 
 | ||||
| # git checkout of libstagefright | ||||
|  | @ -261,4 +264,4 @@ toolkit/components/certviewer/content/package-lock.json | |||
| ^tools/esmify/jscodeshift | ||||
| ^tools/esmify/jscodeshift.cmd | ||||
| ^tools/esmify/jscodeshift.ps1 | ||||
| ^tools/esmify/package-lock.json | ||||
| ^tools/esmify/package-lock.json | ||||
|  |  | |||
							
								
								
									
										5
									
								
								remote/.gitignore
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										5
									
								
								remote/.gitignore
									
									
									
									
										vendored
									
									
								
							|  | @ -1,15 +1,18 @@ | |||
| test/puppeteer/**/*.tsbuildinfo | ||||
| test/puppeteer/.github | ||||
| test/puppeteer/.husky | ||||
| test/puppeteer/.local-chromium/ | ||||
| test/puppeteer/.local-firefox/ | ||||
| test/puppeteer/coverage/ | ||||
| test/puppeteer/docker/ | ||||
| test/puppeteer/docs/puppeteer.api.json | ||||
| test/puppeteer/experimental/ | ||||
| test/puppeteer/lib/ | ||||
| test/puppeteer/node_modules/ | ||||
| test/puppeteer/package-lock.json | ||||
| test/puppeteer/puppeteer*.tgz | ||||
| test/puppeteer/src/generated | ||||
| test/puppeteer/test/build | ||||
| test/puppeteer/test/output-firefox | ||||
| test/puppeteer/test/output-chromium | ||||
| test/puppeteer/utils/testserver/tsconfig.tsbuildinfo | ||||
| test/puppeteer/website | ||||
|  |  | |||
|  | @ -710,6 +710,12 @@ def install_puppeteer(command_context, product, ci): | |||
| 
 | ||||
|     command = "ci" if ci else "install" | ||||
|     npm(command, cwd=os.path.join(command_context.topsrcdir, puppeteer_dir), env=env) | ||||
|     npm( | ||||
|         "run", | ||||
|         "build:dev", | ||||
|         cwd=os.path.join(command_context.topsrcdir, puppeteer_dir), | ||||
|         env=env, | ||||
|     ) | ||||
| 
 | ||||
| 
 | ||||
| def exit(code, error=None): | ||||
|  |  | |||
|  | @ -698,6 +698,9 @@ | |||
|   "ignoreHTTPSErrors Response.securityDetails Network redirects should report SecurityDetails (ignorehttpserrors.spec.js)": [ | ||||
|     "PASS", "FAIL" | ||||
|   ], | ||||
|   "InjectedUtil tests should work (injected.spec.js)": [ | ||||
|     "PASS" | ||||
|   ], | ||||
|   "input tests input should upload the file (input.spec.js)": [ | ||||
|     "SKIP" | ||||
|   ], | ||||
|  | @ -924,7 +927,7 @@ | |||
|     "PASS" | ||||
|   ], | ||||
|   "Launcher specs Puppeteer Puppeteer.launch userDataDir option should restore cookies (launcher.spec.js)": [ | ||||
|     "SKIP" | ||||
|     "PASS" | ||||
|   ], | ||||
|   "Launcher specs Puppeteer Puppeteer.launch should return the default arguments (launcher.spec.js)": [ | ||||
|     "PASS" | ||||
|  | @ -1097,9 +1100,6 @@ | |||
|   "navigation Page.goto should fail when navigating to bad SSL after redirects (navigation.spec.js)": [ | ||||
|     "PASS" | ||||
|   ], | ||||
|   "navigation Page.goto should throw if networkidle is passed as an option (navigation.spec.js)": [ | ||||
|     "PASS" | ||||
|   ], | ||||
|   "navigation Page.goto should fail when main resources failed to load (navigation.spec.js)": [ | ||||
|     "PASS" | ||||
|   ], | ||||
|  | @ -2176,6 +2176,9 @@ | |||
|   "Screenshots Page.screenshot should clip rect (screenshot.spec.js)": [ | ||||
|     "FAIL" | ||||
|   ], | ||||
|   "Screenshots Page.screenshot should use scale for clip (screenshot.spec.js)": [ | ||||
|     "FAIL" | ||||
|   ], | ||||
|   "Screenshots Page.screenshot should get screenshot bigger than the viewport (screenshot.spec.js)": [ | ||||
|     "PASS" | ||||
|   ], | ||||
|  |  | |||
|  | @ -7,7 +7,7 @@ build/ | |||
| lib/ | ||||
| 
 | ||||
| # Generated files | ||||
| tsconfig.tsbuildinfo | ||||
| **/*.tsbuildinfo | ||||
| puppeteer.api.json | ||||
| puppeteer*.tgz | ||||
| yarn.lock | ||||
|  | @ -18,9 +18,11 @@ yarn.lock | |||
| test/output-*/ | ||||
| .dev_profile* | ||||
| coverage/ | ||||
| src/generated | ||||
| 
 | ||||
| # IDE Artifacts | ||||
| .vscode | ||||
| .devcontainer | ||||
| 
 | ||||
| # Misc | ||||
| .DS_Store | ||||
|  | @ -32,6 +34,7 @@ coverage/ | |||
| npm-debug.log* | ||||
| yarn-debug.log* | ||||
| yarn-error.log* | ||||
| 
 | ||||
| ## [END] Keep in sync with .gitignore | ||||
| 
 | ||||
| # ESLint ignores. | ||||
|  |  | |||
|  | @ -7,7 +7,7 @@ build/ | |||
| lib/ | ||||
| 
 | ||||
| # Generated files | ||||
| tsconfig.tsbuildinfo | ||||
| **/*.tsbuildinfo | ||||
| puppeteer.api.json | ||||
| puppeteer*.tgz | ||||
| yarn.lock | ||||
|  | @ -18,9 +18,11 @@ yarn.lock | |||
| test/output-*/ | ||||
| .dev_profile* | ||||
| coverage/ | ||||
| src/generated | ||||
| 
 | ||||
| # IDE Artifacts | ||||
| .vscode | ||||
| .devcontainer | ||||
| 
 | ||||
| # Misc | ||||
| .DS_Store | ||||
|  |  | |||
|  | @ -1,3 +1,3 @@ | |||
| { | ||||
|   ".": "16.1.1" | ||||
|   ".": "17.1.2" | ||||
| } | ||||
|  |  | |||
|  | @ -2,6 +2,65 @@ | |||
| 
 | ||||
| All notable changes to this project will be documented in this file. See [standard-version](https://github.com/conventional-changelog/standard-version) for commit guidelines. | ||||
| 
 | ||||
| ## [17.1.2](https://github.com/puppeteer/puppeteer/compare/v17.1.1...v17.1.2) (2022-09-07) | ||||
| 
 | ||||
| 
 | ||||
| ### Bug Fixes | ||||
| 
 | ||||
| * add missing code coverage ranges that span only a single character ([#8911](https://github.com/puppeteer/puppeteer/issues/8911)) ([0c577b9](https://github.com/puppeteer/puppeteer/commit/0c577b9bf8855dc0ccb6098cd43a25c528f6d7f5)) | ||||
| * add Page.getDefaultTimeout getter ([#8903](https://github.com/puppeteer/puppeteer/issues/8903)) ([3240095](https://github.com/puppeteer/puppeteer/commit/32400954c50cbddc48468ad118c3f8a47653b9d3)), closes [#8901](https://github.com/puppeteer/puppeteer/issues/8901) | ||||
| * don't detect project root for puppeteer-core ([#8907](https://github.com/puppeteer/puppeteer/issues/8907)) ([b4f5ea1](https://github.com/puppeteer/puppeteer/commit/b4f5ea1167a60c870194c70d22f5372ada5b7c4c)), closes [#8896](https://github.com/puppeteer/puppeteer/issues/8896) | ||||
| * support scale for screenshot clips ([#8908](https://github.com/puppeteer/puppeteer/issues/8908)) ([260e428](https://github.com/puppeteer/puppeteer/commit/260e4282275ab1d05c86e5643e2a02c01f269a9c)), closes [#5329](https://github.com/puppeteer/puppeteer/issues/5329) | ||||
| * work around a race in waitForFileChooser ([#8905](https://github.com/puppeteer/puppeteer/issues/8905)) ([053d960](https://github.com/puppeteer/puppeteer/commit/053d960fb593e514e7914d7da9af436afc39a12f)), closes [#6040](https://github.com/puppeteer/puppeteer/issues/6040) | ||||
| 
 | ||||
| ## [17.1.1](https://github.com/puppeteer/puppeteer/compare/v17.1.0...v17.1.1) (2022-09-05) | ||||
| 
 | ||||
| 
 | ||||
| ### Bug Fixes | ||||
| 
 | ||||
| * restore deferred promise debugging ([#8895](https://github.com/puppeteer/puppeteer/issues/8895)) ([7b42250](https://github.com/puppeteer/puppeteer/commit/7b42250c7bb91ac873307acda493726ffc4c54a8)) | ||||
| 
 | ||||
| ## [17.1.0](https://github.com/puppeteer/puppeteer/compare/v17.0.0...v17.1.0) (2022-09-02) | ||||
| 
 | ||||
| 
 | ||||
| ### Features | ||||
| 
 | ||||
| * **chromium:** roll to Chromium 106.0.5249.0 (r1036745) ([#8869](https://github.com/puppeteer/puppeteer/issues/8869)) ([6e9a47a](https://github.com/puppeteer/puppeteer/commit/6e9a47a6faa06d241dec0bcf7bcdf49370517008)) | ||||
| 
 | ||||
| 
 | ||||
| ### Bug Fixes | ||||
| 
 | ||||
| * allow getting a frame from an elementhandle ([#8875](https://github.com/puppeteer/puppeteer/issues/8875)) ([3732757](https://github.com/puppeteer/puppeteer/commit/3732757450b4363041ccbacc3b236289a156abb0)) | ||||
| * typos in documentation ([#8858](https://github.com/puppeteer/puppeteer/issues/8858)) ([8d95a9b](https://github.com/puppeteer/puppeteer/commit/8d95a9bc920b98820aa655ad4eb2d8fd9b2b893a)) | ||||
| * use the timeout setting in waitForFileChooser ([#8856](https://github.com/puppeteer/puppeteer/issues/8856)) ([f477b46](https://github.com/puppeteer/puppeteer/commit/f477b46f212da9206102da695697760eea539f05)) | ||||
| 
 | ||||
| ## [17.0.0](https://github.com/puppeteer/puppeteer/compare/v16.2.0...v17.0.0) (2022-08-26) | ||||
| 
 | ||||
| 
 | ||||
| ### ⚠ BREAKING CHANGES | ||||
| 
 | ||||
| * remove `root` from `WaitForSelectorOptions` (#8848) | ||||
| * internalize execution context (#8844) | ||||
| 
 | ||||
| ### Bug Fixes | ||||
| 
 | ||||
| * allow multiple navigations to happen in LifecycleWatcher ([#8826](https://github.com/puppeteer/puppeteer/issues/8826)) ([341b669](https://github.com/puppeteer/puppeteer/commit/341b669a5e45ecbb9ffb0f28c45b520660f27ad2)), closes [#8811](https://github.com/puppeteer/puppeteer/issues/8811) | ||||
| * internalize execution context ([#8844](https://github.com/puppeteer/puppeteer/issues/8844)) ([2f33237](https://github.com/puppeteer/puppeteer/commit/2f33237d0443de77d58dca4454b0c9a1d2b57d03)) | ||||
| * remove `root` from `WaitForSelectorOptions` ([#8848](https://github.com/puppeteer/puppeteer/issues/8848)) ([1155c8e](https://github.com/puppeteer/puppeteer/commit/1155c8eac85b176c3334cc3d98adfe7d943dfbe6)) | ||||
| * remove deferred promise timeouts ([#8835](https://github.com/puppeteer/puppeteer/issues/8835)) ([202ffce](https://github.com/puppeteer/puppeteer/commit/202ffce0aa4f34dba35fbb8e7d740af16efee35f)), closes [#8832](https://github.com/puppeteer/puppeteer/issues/8832) | ||||
| 
 | ||||
| ## [16.2.0](https://github.com/puppeteer/puppeteer/compare/v16.1.1...v16.2.0) (2022-08-18) | ||||
| 
 | ||||
| 
 | ||||
| ### Features | ||||
| 
 | ||||
| * add Khmer (Cambodian) language support ([#8809](https://github.com/puppeteer/puppeteer/issues/8809)) ([34f8737](https://github.com/puppeteer/puppeteer/commit/34f873721804d57a5faf3eab8ef50340c69ed180)) | ||||
| 
 | ||||
| 
 | ||||
| ### Bug Fixes | ||||
| 
 | ||||
| * handle service workers in extensions ([#8807](https://github.com/puppeteer/puppeteer/issues/8807)) ([2a0eefb](https://github.com/puppeteer/puppeteer/commit/2a0eefb99f0ae00dacc9e768a253308c0d18a4c3)), closes [#8800](https://github.com/puppeteer/puppeteer/issues/8800) | ||||
| 
 | ||||
| ## [16.1.1](https://github.com/puppeteer/puppeteer/compare/v16.1.0...v16.1.1) (2022-08-16) | ||||
| 
 | ||||
| 
 | ||||
|  |  | |||
|  | @ -5,6 +5,6 @@ origin: | |||
|   description: Headless Chrome Node API | ||||
|   license: Apache-2.0 | ||||
|   name: puppeteer | ||||
|   release: a4938d7edc53fbb1d217914981155ce3bbcc149f | ||||
|   release: 0d2d99efeca73fba255fb10b28b5d3f50c2e20e4 | ||||
|   url: /Users/alexandraborovova/Projects/puppeteer | ||||
| schema: 1 | ||||
|  |  | |||
							
								
								
									
										1576
									
								
								remote/test/puppeteer/package-lock.json
									
									
									
										generated
									
									
									
								
							
							
						
						
									
										1576
									
								
								remote/test/puppeteer/package-lock.json
									
									
									
										generated
									
									
									
								
							
										
											
												File diff suppressed because it is too large
												Load diff
											
										
									
								
							|  | @ -1,6 +1,6 @@ | |||
| { | ||||
|   "name": "puppeteer", | ||||
|   "version": "16.1.1", | ||||
|   "version": "17.1.2", | ||||
|   "description": "A high-level API to control headless Chrome over the DevTools Protocol", | ||||
|   "keywords": [ | ||||
|     "puppeteer", | ||||
|  | @ -30,35 +30,35 @@ | |||
|     "test": "c8 --check-coverage --lines 93 run-s test:chrome:* test:firefox", | ||||
|     "test:types": "tsd", | ||||
|     "test:install": "scripts/test-install.sh", | ||||
|     "test:firefox": "cross-env PUPPETEER_PRODUCT=firefox mocha", | ||||
|     "test:firefox": "cross-env PUPPETEER_PRODUCT=firefox MOZ_WEBRENDER=0 PUPPETEER_DEFERRED_PROMISE_DEBUG_TIMEOUT=20000 mocha", | ||||
|     "test:chrome": "run-s test:chrome:*", | ||||
|     "test:chrome:headless": "cross-env HEADLESS=true mocha", | ||||
|     "test:chrome:headless-chrome": "cross-env HEADLESS=chrome mocha", | ||||
|     "test:chrome:headful": "cross-env HEADLESS=false mocha", | ||||
|     "test:chrome:headless": "cross-env HEADLESS=true PUPPETEER_DEFERRED_PROMISE_DEBUG_TIMEOUT=20000 mocha", | ||||
|     "test:chrome:headless-chrome": "cross-env HEADLESS=chrome PUPPETEER_DEFERRED_PROMISE_DEBUG_TIMEOUT=20000 mocha", | ||||
|     "test:chrome:headful": "cross-env HEADLESS=false PUPPETEER_DEFERRED_PROMISE_DEBUG_TIMEOUT=20000 mocha", | ||||
|     "prepublishOnly": "npm run build", | ||||
|     "prepare": "node typescript-if-required.js && husky install", | ||||
|     "lint": "run-s lint:prettier lint:eslint", | ||||
|     "lint:prettier": "prettier --check .", | ||||
|     "lint:eslint": "([ \"$CI\" = true ] && eslint --ext js --ext ts --quiet -f codeframe . || eslint --ext js --ext ts .)", | ||||
|     "install": "node install.js", | ||||
|     "generate:types": "node utils/export_all.js && api-extractor run --local --verbose && eslint --ext ts --no-ignore --no-eslintrc -c .eslintrc.types.cjs --fix lib/types.d.ts", | ||||
|     "generate:markdown": "ts-node -O '{\"module\":\"commonjs\"}' utils/generate_docs.ts && prettier --ignore-path none --write docs", | ||||
|     "generate:esm-package-json": "echo '{\"type\": \"module\"}' > lib/esm/package.json", | ||||
|     "generate:sources": "tsx utils/generate_sources.ts", | ||||
|     "generate:artifacts": "tsx utils/generate_artifacts.ts", | ||||
|     "generate:markdown": "tsx utils/generate_docs.ts", | ||||
|     "format": "run-s format:*", | ||||
|     "format:prettier": "prettier --write .", | ||||
|     "format:eslint": "eslint --ext js --ext ts --fix .", | ||||
|     "docs": "run-s build generate:markdown", | ||||
|     "debug": "npm run build && mocha --inspect-brk", | ||||
|     "debug": "npm run build:dev && mocha --inspect-brk", | ||||
|     "commitlint": "commitlint --from=HEAD~1", | ||||
|     "clean": "rimraf lib && rimraf test/build", | ||||
|     "check": "run-p check:*", | ||||
|     "check:protocol-revision": "ts-node -s scripts/ensure-correct-devtools-protocol-package", | ||||
|     "check:pinned-deps": "ts-node -s scripts/ensure-pinned-deps", | ||||
|     "build": "run-s build:tsc generate:types generate:esm-package-json", | ||||
|     "build:tsc": "tsc --version && run-p build:tsc:*", | ||||
|     "build:tsc:esm": "tsc -b src/tsconfig.esm.json", | ||||
|     "build:tsc:cjs": "tsc -b src/tsconfig.cjs.json", | ||||
|     "build:tsc:test": "tsc -b test" | ||||
|     "check:protocol-revision": "tsx scripts/ensure-correct-devtools-protocol-package", | ||||
|     "check:pinned-deps": "tsx scripts/ensure-pinned-deps", | ||||
|     "build": "npm run build:prod", | ||||
|     "build:dev": "run-s generate:sources build:tsc:dev generate:artifacts", | ||||
|     "build:prod": "run-s generate:sources build:tsc:prod generate:artifacts", | ||||
|     "build:tsc:dev": "tsc -b test", | ||||
|     "build:tsc:prod": "tsc -b tsconfig.lib.json" | ||||
|   }, | ||||
|   "files": [ | ||||
|     "lib", | ||||
|  | @ -71,10 +71,9 @@ | |||
|   "dependencies": { | ||||
|     "cross-fetch": "3.1.5", | ||||
|     "debug": "4.3.4", | ||||
|     "devtools-protocol": "0.0.1019158", | ||||
|     "devtools-protocol": "0.0.1036444", | ||||
|     "extract-zip": "2.0.1", | ||||
|     "https-proxy-agent": "5.0.1", | ||||
|     "pkg-dir": "4.2.0", | ||||
|     "progress": "2.0.3", | ||||
|     "proxy-from-env": "1.1.0", | ||||
|     "rimraf": "3.0.2", | ||||
|  | @ -90,6 +89,7 @@ | |||
|     "@microsoft/api-extractor-model": "7.23.0", | ||||
|     "@types/debug": "4.1.7", | ||||
|     "@types/diff": "5.0.2", | ||||
|     "@types/glob": "7.2.0", | ||||
|     "@types/mime": "3.0.1", | ||||
|     "@types/mocha": "9.1.1", | ||||
|     "@types/node": "18.7.1", | ||||
|  | @ -109,6 +109,7 @@ | |||
|     "commonmark": "0.30.0", | ||||
|     "cross-env": "7.0.3", | ||||
|     "diff": "5.1.0", | ||||
|     "esbuild": "0.15.5", | ||||
|     "eslint": "8.21.0", | ||||
|     "eslint-config-prettier": "8.5.0", | ||||
|     "eslint-formatter-codeframe": "7.32.1", | ||||
|  | @ -120,6 +121,7 @@ | |||
|     "eslint-plugin-unused-imports": "2.0.0", | ||||
|     "esprima": "4.0.1", | ||||
|     "expect": "25.2.7", | ||||
|     "glob": "8.0.3", | ||||
|     "gts": "4.0.0", | ||||
|     "husky": "8.0.1", | ||||
|     "jpeg-js": "0.4.4", | ||||
|  | @ -136,6 +138,7 @@ | |||
|     "source-map-support": "0.5.21", | ||||
|     "text-diff": "1.0.1", | ||||
|     "tsd": "0.22.0", | ||||
|     "tsx": "3.8.2", | ||||
|     "typescript": "4.7.4" | ||||
|   } | ||||
| } | ||||
|  |  | |||
|  | @ -15,15 +15,11 @@ | |||
|  */ | ||||
| 
 | ||||
| import {Protocol} from 'devtools-protocol'; | ||||
| import {assert} from './assert.js'; | ||||
| import {assert} from '../util/assert.js'; | ||||
| import {CDPSession} from './Connection.js'; | ||||
| import { | ||||
|   IsolatedWorld, | ||||
|   PageBinding, | ||||
|   WaitForSelectorOptions, | ||||
| } from './IsolatedWorld.js'; | ||||
| import {ElementHandle} from './ElementHandle.js'; | ||||
| import {JSHandle} from './JSHandle.js'; | ||||
| import {Frame} from './Frame.js'; | ||||
| import {MAIN_WORLD, PageBinding, PUPPETEER_WORLD} from './IsolatedWorld.js'; | ||||
| import {InternalQueryHandler} from './QueryHandler.js'; | ||||
| 
 | ||||
| async function queryAXTree( | ||||
|  | @ -90,52 +86,86 @@ function parseAriaSelector(selector: string): ARIAQueryOption { | |||
|   return queryOptions; | ||||
| } | ||||
| 
 | ||||
| const queryOne = async ( | ||||
|   element: ElementHandle<Node>, | ||||
|   selector: string | ||||
| ): Promise<ElementHandle<Node> | null> => { | ||||
|   const exeCtx = element.executionContext(); | ||||
| const queryOneId = async (element: ElementHandle<Node>, selector: string) => { | ||||
|   const {name, role} = parseAriaSelector(selector); | ||||
|   const res = await queryAXTree(exeCtx._client, element, name, role); | ||||
|   const res = await queryAXTree(element.client, element, name, role); | ||||
|   if (!res[0] || !res[0].backendDOMNodeId) { | ||||
|     return null; | ||||
|   } | ||||
|   return (await exeCtx._world!.adoptBackendNode( | ||||
|     res[0].backendDOMNodeId | ||||
|   return res[0].backendDOMNodeId; | ||||
| }; | ||||
| 
 | ||||
| const queryOne: InternalQueryHandler['queryOne'] = async ( | ||||
|   element, | ||||
|   selector | ||||
| ) => { | ||||
|   const id = await queryOneId(element, selector); | ||||
|   if (!id) { | ||||
|     return null; | ||||
|   } | ||||
|   return (await element.frame.worlds[MAIN_WORLD].adoptBackendNode( | ||||
|     id | ||||
|   )) as ElementHandle<Node>; | ||||
| }; | ||||
| 
 | ||||
| const waitFor = async ( | ||||
|   isolatedWorld: IsolatedWorld, | ||||
|   selector: string, | ||||
|   options: WaitForSelectorOptions | ||||
| ): Promise<ElementHandle<Element> | null> => { | ||||
| const waitFor: InternalQueryHandler['waitFor'] = async ( | ||||
|   elementOrFrame, | ||||
|   selector, | ||||
|   options | ||||
| ) => { | ||||
|   let frame: Frame; | ||||
|   let element: ElementHandle<Node> | undefined; | ||||
|   if (elementOrFrame instanceof Frame) { | ||||
|     frame = elementOrFrame; | ||||
|   } else { | ||||
|     frame = elementOrFrame.frame; | ||||
|     element = await frame.worlds[PUPPETEER_WORLD].adoptHandle(elementOrFrame); | ||||
|   } | ||||
|   const binding: PageBinding = { | ||||
|     name: 'ariaQuerySelector', | ||||
|     pptrFunction: async (selector: string) => { | ||||
|       const root = options.root || (await isolatedWorld.document()); | ||||
|       const element = await queryOne(root, selector); | ||||
|       return element; | ||||
|       const id = await queryOneId( | ||||
|         element || (await frame.worlds[PUPPETEER_WORLD].document()), | ||||
|         selector | ||||
|       ); | ||||
|       if (!id) { | ||||
|         return null; | ||||
|       } | ||||
|       return (await frame.worlds[PUPPETEER_WORLD].adoptBackendNode( | ||||
|         id | ||||
|       )) as ElementHandle<Node>; | ||||
|     }, | ||||
|   }; | ||||
|   return (await isolatedWorld._waitForSelectorInPage( | ||||
|   const result = await frame.worlds[PUPPETEER_WORLD]._waitForSelectorInPage( | ||||
|     (_: Element, selector: string) => { | ||||
|       return ( | ||||
|         globalThis as unknown as { | ||||
|           ariaQuerySelector(selector: string): void; | ||||
|           ariaQuerySelector(selector: string): Node | null; | ||||
|         } | ||||
|       ).ariaQuerySelector(selector); | ||||
|     }, | ||||
|     element, | ||||
|     selector, | ||||
|     options, | ||||
|     binding | ||||
|   )) as ElementHandle<Element> | null; | ||||
|   ); | ||||
|   if (element) { | ||||
|     await element.dispose(); | ||||
|   } | ||||
|   if (!result) { | ||||
|     return null; | ||||
|   } | ||||
|   if (!(result instanceof ElementHandle)) { | ||||
|     await result.dispose(); | ||||
|     return null; | ||||
|   } | ||||
|   return result.frame.worlds[MAIN_WORLD].transferHandle(result); | ||||
| }; | ||||
| 
 | ||||
| const queryAll = async ( | ||||
|   element: ElementHandle<Node>, | ||||
|   selector: string | ||||
| ): Promise<Array<ElementHandle<Node>>> => { | ||||
| const queryAll: InternalQueryHandler['queryAll'] = async ( | ||||
|   element, | ||||
|   selector | ||||
| ) => { | ||||
|   const exeCtx = element.executionContext(); | ||||
|   const {name, role} = parseAriaSelector(selector); | ||||
|   const res = await queryAXTree(exeCtx._client, element, name, role); | ||||
|  | @ -149,18 +179,6 @@ const queryAll = async ( | |||
|   ); | ||||
| }; | ||||
| 
 | ||||
| const queryAllArray = async ( | ||||
|   element: ElementHandle<Node>, | ||||
|   selector: string | ||||
| ): Promise<JSHandle<Node[]>> => { | ||||
|   const elementHandles = await queryAll(element, selector); | ||||
|   const exeCtx = element.executionContext(); | ||||
|   const jsHandle = exeCtx.evaluateHandle((...elements) => { | ||||
|     return elements; | ||||
|   }, ...elementHandles); | ||||
|   return jsHandle; | ||||
| }; | ||||
| 
 | ||||
| /** | ||||
|  * @internal | ||||
|  */ | ||||
|  | @ -168,5 +186,4 @@ export const ariaHandler: InternalQueryHandler = { | |||
|   queryOne, | ||||
|   waitFor, | ||||
|   queryAll, | ||||
|   queryAllArray, | ||||
| }; | ||||
|  |  | |||
|  | @ -16,7 +16,7 @@ | |||
| 
 | ||||
| import {ChildProcess} from 'child_process'; | ||||
| import {Protocol} from 'devtools-protocol'; | ||||
| import {assert} from './assert.js'; | ||||
| import {assert} from '../util/assert.js'; | ||||
| import {CDPSession, Connection, ConnectionEmittedEvents} from './Connection.js'; | ||||
| import {EventEmitter} from './EventEmitter.js'; | ||||
| import {waitWithTimeout} from './util.js'; | ||||
|  |  | |||
|  | @ -14,9 +14,10 @@ | |||
|  * limitations under the License. | ||||
|  */ | ||||
| 
 | ||||
| import {debugError, isErrorLike} from './util.js'; | ||||
| import {debugError} from './util.js'; | ||||
| import {isErrorLike} from '../util/ErrorLike.js'; | ||||
| import {isNode} from '../environment.js'; | ||||
| import {assert} from './assert.js'; | ||||
| import {assert} from '../util/assert.js'; | ||||
| import { | ||||
|   Browser, | ||||
|   IsPageTargetCallback, | ||||
|  |  | |||
|  | @ -15,7 +15,7 @@ | |||
|  */ | ||||
| 
 | ||||
| import Protocol from 'devtools-protocol'; | ||||
| import {assert} from './assert.js'; | ||||
| import {assert} from '../util/assert.js'; | ||||
| import {CDPSession, Connection} from './Connection.js'; | ||||
| import {EventEmitter} from './EventEmitter.js'; | ||||
| import {Target} from './Target.js'; | ||||
|  | @ -317,11 +317,12 @@ export class ChromeTargetManager extends EventEmitter implements TargetManager { | |||
|     ) { | ||||
|       this.#finishInitializationIfReady(targetInfo.targetId); | ||||
|       await silentDetach(); | ||||
|       if (parentSession instanceof CDPSession) { | ||||
|         const target = this.#targetFactory(targetInfo); | ||||
|         this.#attachedTargetsByTargetId.set(targetInfo.targetId, target); | ||||
|         this.emit(TargetManagerEmittedEvents.TargetAvailable, target); | ||||
|       if (this.#attachedTargetsByTargetId.has(targetInfo.targetId)) { | ||||
|         return; | ||||
|       } | ||||
|       const target = this.#targetFactory(targetInfo); | ||||
|       this.#attachedTargetsByTargetId.set(targetInfo.targetId, target); | ||||
|       this.emit(TargetManagerEmittedEvents.TargetAvailable, target); | ||||
|       return; | ||||
|     } | ||||
| 
 | ||||
|  |  | |||
|  | @ -13,7 +13,7 @@ | |||
|  * See the License for the specific language governing permissions and | ||||
|  * limitations under the License. | ||||
|  */ | ||||
| import {assert} from './assert.js'; | ||||
| import {assert} from '../util/assert.js'; | ||||
| import {debug} from './Debug.js'; | ||||
| const debugProtocolSend = debug('puppeteer:protocol:SEND ►'); | ||||
| const debugProtocolReceive = debug('puppeteer:protocol:RECV ◀'); | ||||
|  |  | |||
|  | @ -14,7 +14,7 @@ | |||
|  * limitations under the License. | ||||
|  */ | ||||
| 
 | ||||
| import {assert} from './assert.js'; | ||||
| import {assert} from '../util/assert.js'; | ||||
| import {addEventListener, debugError, PuppeteerEventListener} from './util.js'; | ||||
| import {Protocol} from 'devtools-protocol'; | ||||
| import {CDPSession} from './Connection.js'; | ||||
|  | @ -484,6 +484,6 @@ function convertToDisjointRanges( | |||
|   } | ||||
|   // Filter out empty ranges.
 | ||||
|   return results.filter(range => { | ||||
|     return range.end - range.start > 1; | ||||
|     return range.end - range.start > 0; | ||||
|   }); | ||||
| } | ||||
|  |  | |||
|  | @ -14,7 +14,7 @@ | |||
|  * limitations under the License. | ||||
|  */ | ||||
| 
 | ||||
| import {assert} from './assert.js'; | ||||
| import {assert} from '../util/assert.js'; | ||||
| import {CDPSession} from './Connection.js'; | ||||
| import {Protocol} from 'devtools-protocol'; | ||||
| 
 | ||||
|  |  | |||
|  | @ -1,13 +1,9 @@ | |||
| import {Protocol} from 'devtools-protocol'; | ||||
| import {assert} from './assert.js'; | ||||
| import {CDPSession} from './Connection.js'; | ||||
| import {assert} from '../util/assert.js'; | ||||
| import {ExecutionContext} from './ExecutionContext.js'; | ||||
| import {Frame, FrameManager} from './FrameManager.js'; | ||||
| import { | ||||
|   MAIN_WORLD, | ||||
|   PUPPETEER_WORLD, | ||||
|   WaitForSelectorOptions, | ||||
| } from './IsolatedWorld.js'; | ||||
| import {Frame} from './Frame.js'; | ||||
| import {FrameManager} from './FrameManager.js'; | ||||
| import {WaitForSelectorOptions} from './IsolatedWorld.js'; | ||||
| import { | ||||
|   BoundingBox, | ||||
|   BoxModel, | ||||
|  | @ -19,7 +15,7 @@ import { | |||
| } from './JSHandle.js'; | ||||
| import {Page, ScreenshotOptions} from './Page.js'; | ||||
| import {getQueryHandlerAndSelector} from './QueryHandler.js'; | ||||
| import {EvaluateFunc, NodeFor} from './types.js'; | ||||
| import {EvaluateFunc, HandleFor, NodeFor} from './types.js'; | ||||
| import {KeyInput} from './USKeyboardLayout.js'; | ||||
| import {debugError, isString} from './util.js'; | ||||
| 
 | ||||
|  | @ -71,24 +67,29 @@ export class ElementHandle< | |||
|   ElementType extends Node = Element | ||||
| > extends JSHandle<ElementType> { | ||||
|   #frame: Frame; | ||||
|   #page: Page; | ||||
|   #frameManager: FrameManager; | ||||
| 
 | ||||
|   /** | ||||
|    * @internal | ||||
|    */ | ||||
|   constructor( | ||||
|     context: ExecutionContext, | ||||
|     client: CDPSession, | ||||
|     remoteObject: Protocol.Runtime.RemoteObject, | ||||
|     frame: Frame, | ||||
|     page: Page, | ||||
|     frameManager: FrameManager | ||||
|     frame: Frame | ||||
|   ) { | ||||
|     super(context, client, remoteObject); | ||||
|     super(context, remoteObject); | ||||
|     this.#frame = frame; | ||||
|     this.#page = page; | ||||
|     this.#frameManager = frameManager; | ||||
|   } | ||||
| 
 | ||||
|   get #frameManager(): FrameManager { | ||||
|     return this.#frame._frameManager; | ||||
|   } | ||||
| 
 | ||||
|   get #page(): Page { | ||||
|     return this.#frame.page(); | ||||
|   } | ||||
| 
 | ||||
|   get frame(): Frame { | ||||
|     return this.#frame; | ||||
|   } | ||||
| 
 | ||||
|   /** | ||||
|  | @ -228,13 +229,24 @@ export class ElementHandle< | |||
|   ): Promise<Awaited<ReturnType<Func>>> { | ||||
|     const {updatedSelector, queryHandler} = | ||||
|       getQueryHandlerAndSelector(selector); | ||||
|     assert(queryHandler.queryAllArray); | ||||
|     const arrayHandle = (await queryHandler.queryAllArray( | ||||
|     assert( | ||||
|       queryHandler.queryAll, | ||||
|       'Cannot handle queries for a multiple element with the given selector' | ||||
|     ); | ||||
|     const handles = (await queryHandler.queryAll( | ||||
|       this, | ||||
|       updatedSelector | ||||
|     )) as JSHandle<Array<NodeFor<Selector>>>; | ||||
|     const result = await arrayHandle.evaluate(pageFunction, ...args); | ||||
|     await arrayHandle.dispose(); | ||||
|     )) as Array<HandleFor<NodeFor<Selector>>>; | ||||
|     const elements = await this.evaluateHandle((_, ...elements) => { | ||||
|       return elements; | ||||
|     }, ...handles); | ||||
|     const [result] = await Promise.all([ | ||||
|       elements.evaluate(pageFunction, ...args), | ||||
|       ...handles.map(handle => { | ||||
|         return handle.dispose(); | ||||
|       }), | ||||
|     ]); | ||||
|     await elements.dispose(); | ||||
|     return result; | ||||
|   } | ||||
| 
 | ||||
|  | @ -291,27 +303,16 @@ export class ElementHandle< | |||
|    */ | ||||
|   async waitForSelector<Selector extends string>( | ||||
|     selector: Selector, | ||||
|     options: Exclude<WaitForSelectorOptions, 'root'> = {} | ||||
|     options: WaitForSelectorOptions = {} | ||||
|   ): Promise<ElementHandle<NodeFor<Selector>> | null> { | ||||
|     const frame = this.executionContext().frame(); | ||||
|     assert(frame); | ||||
|     const adoptedRoot = await frame.worlds[PUPPETEER_WORLD].adoptHandle(this); | ||||
|     const handle = await frame.worlds[PUPPETEER_WORLD].waitForSelector( | ||||
|       selector, | ||||
|       { | ||||
|         ...options, | ||||
|         root: adoptedRoot, | ||||
|       } | ||||
|     ); | ||||
|     await adoptedRoot.dispose(); | ||||
|     if (!handle) { | ||||
|       return null; | ||||
|     } | ||||
|     const result = (await frame.worlds[MAIN_WORLD].adoptHandle( | ||||
|       handle | ||||
|     )) as ElementHandle<NodeFor<Selector>>; | ||||
|     await handle.dispose(); | ||||
|     return result; | ||||
|     const {updatedSelector, queryHandler} = | ||||
|       getQueryHandlerAndSelector(selector); | ||||
|     assert(queryHandler.waitFor, 'Query handler does not support waiting'); | ||||
|     return (await queryHandler.waitFor( | ||||
|       this, | ||||
|       updatedSelector, | ||||
|       options | ||||
|     )) as ElementHandle<NodeFor<Selector>> | null; | ||||
|   } | ||||
| 
 | ||||
|   /** | ||||
|  |  | |||
|  | @ -15,9 +15,7 @@ | |||
|  */ | ||||
| 
 | ||||
| import {Protocol} from 'devtools-protocol'; | ||||
| import {assert} from './assert.js'; | ||||
| import {CDPSession} from './Connection.js'; | ||||
| import {Frame} from './FrameManager.js'; | ||||
| import {IsolatedWorld} from './IsolatedWorld.js'; | ||||
| import {JSHandle} from './JSHandle.js'; | ||||
| import {EvaluateFunc, HandleFor} from './types.js'; | ||||
|  | @ -35,8 +33,6 @@ export const EVALUATION_SCRIPT_URL = 'pptr://__puppeteer_evaluation_script__'; | |||
| const SOURCE_URL_REGEX = /^[\040\t]*\/\/[@#] sourceURL=\s*(\S*?)\s*$/m; | ||||
| 
 | ||||
| /** | ||||
|  * @deprecated Do not use directly. | ||||
|  * | ||||
|  * Represents a context for JavaScript execution. | ||||
|  * | ||||
|  * @example | ||||
|  | @ -55,6 +51,8 @@ const SOURCE_URL_REGEX = /^[\040\t]*\/\/[@#] sourceURL=\s*(\S*?)\s*$/m; | |||
|  * @remarks | ||||
|  * Besides pages, execution contexts can be found in | ||||
|  * {@link WebWorker | workers}. | ||||
|  * | ||||
|  * @internal | ||||
|  */ | ||||
| export class ExecutionContext { | ||||
|   /** | ||||
|  | @ -88,18 +86,6 @@ export class ExecutionContext { | |||
|     this._contextName = contextPayload.name; | ||||
|   } | ||||
| 
 | ||||
|   /** | ||||
|    * @returns The frame associated with this execution context. | ||||
|    * | ||||
|    * @remarks | ||||
|    * Not every execution context is associated with a frame. For example, | ||||
|    * {@link WebWorker | workers} have execution contexts that are not associated | ||||
|    * with frames. | ||||
|    */ | ||||
|   frame(): Frame | null { | ||||
|     return this._world ? this._world.frame() : null; | ||||
|   } | ||||
| 
 | ||||
|   /** | ||||
|    * Evaluates the given function. | ||||
|    * | ||||
|  | @ -355,61 +341,24 @@ export class ExecutionContext { | |||
|       } | ||||
|       return {value: arg}; | ||||
|     } | ||||
| 
 | ||||
|     function rewriteError(error: Error): Protocol.Runtime.EvaluateResponse { | ||||
|       if (error.message.includes('Object reference chain is too long')) { | ||||
|         return {result: {type: 'undefined'}}; | ||||
|       } | ||||
|       if (error.message.includes("Object couldn't be returned by value")) { | ||||
|         return {result: {type: 'undefined'}}; | ||||
|       } | ||||
| 
 | ||||
|       if ( | ||||
|         error.message.endsWith('Cannot find context with specified id') || | ||||
|         error.message.endsWith('Inspected target navigated or closed') | ||||
|       ) { | ||||
|         throw new Error( | ||||
|           'Execution context was destroyed, most likely because of a navigation.' | ||||
|         ); | ||||
|       } | ||||
|       throw error; | ||||
|     } | ||||
|   } | ||||
| 
 | ||||
|   /** | ||||
|    * Iterates through the JavaScript heap and finds all the objects with the | ||||
|    * given prototype. | ||||
|    * | ||||
|    * @example | ||||
|    * | ||||
|    * ```ts
 | ||||
|    * // Create a Map object
 | ||||
|    * await page.evaluate(() => (window.map = new Map())); | ||||
|    * // Get a handle to the Map object prototype
 | ||||
|    * const mapPrototype = await page.evaluateHandle(() => Map.prototype); | ||||
|    * // Query all map instances into an array
 | ||||
|    * const mapInstances = await page.queryObjects(mapPrototype); | ||||
|    * // Count amount of map objects in heap
 | ||||
|    * const count = await page.evaluate(maps => maps.length, mapInstances); | ||||
|    * await mapInstances.dispose(); | ||||
|    * await mapPrototype.dispose(); | ||||
|    * ``` | ||||
|    * | ||||
|    * @param prototypeHandle - a handle to the object prototype | ||||
|    * @returns A handle to an array of objects with the given prototype. | ||||
|    */ | ||||
|   async queryObjects<Prototype>( | ||||
|     prototypeHandle: JSHandle<Prototype> | ||||
|   ): Promise<HandleFor<Prototype[]>> { | ||||
|     assert(!prototypeHandle.disposed, 'Prototype JSHandle is disposed!'); | ||||
|     const remoteObject = prototypeHandle.remoteObject(); | ||||
|     assert( | ||||
|       remoteObject.objectId, | ||||
|       'Prototype JSHandle must not be referencing primitive value' | ||||
|     ); | ||||
|     const response = await this._client.send('Runtime.queryObjects', { | ||||
|       prototypeObjectId: remoteObject.objectId, | ||||
|     }); | ||||
|     return createJSHandle(this, response.objects) as HandleFor<Prototype[]>; | ||||
|   } | ||||
| } | ||||
| 
 | ||||
| const rewriteError = (error: Error): Protocol.Runtime.EvaluateResponse => { | ||||
|   if (error.message.includes('Object reference chain is too long')) { | ||||
|     return {result: {type: 'undefined'}}; | ||||
|   } | ||||
|   if (error.message.includes("Object couldn't be returned by value")) { | ||||
|     return {result: {type: 'undefined'}}; | ||||
|   } | ||||
| 
 | ||||
|   if ( | ||||
|     error.message.endsWith('Cannot find context with specified id') || | ||||
|     error.message.endsWith('Inspected target navigated or closed') | ||||
|   ) { | ||||
|     throw new Error( | ||||
|       'Execution context was destroyed, most likely because of a navigation.' | ||||
|     ); | ||||
|   } | ||||
|   throw error; | ||||
| }; | ||||
|  |  | |||
|  | @ -15,7 +15,7 @@ | |||
|  */ | ||||
| 
 | ||||
| import {Protocol} from 'devtools-protocol'; | ||||
| import {assert} from './assert.js'; | ||||
| import {assert} from '../util/assert.js'; | ||||
| import {ElementHandle} from './ElementHandle.js'; | ||||
| 
 | ||||
| /** | ||||
|  |  | |||
|  | @ -15,7 +15,7 @@ | |||
|  */ | ||||
| 
 | ||||
| import Protocol from 'devtools-protocol'; | ||||
| import {assert} from './assert.js'; | ||||
| import {assert} from '../util/assert.js'; | ||||
| import {CDPSession, Connection} from './Connection.js'; | ||||
| import {Target} from './Target.js'; | ||||
| import {TargetFilterCallback} from './Browser.js'; | ||||
|  |  | |||
							
								
								
									
										1097
									
								
								remote/test/puppeteer/src/common/Frame.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										1097
									
								
								remote/test/puppeteer/src/common/Frame.ts
									
									
									
									
									
										Normal file
									
								
							
										
											
												File diff suppressed because it is too large
												Load diff
											
										
									
								
							
										
											
												File diff suppressed because it is too large
												Load diff
											
										
									
								
							|  | @ -15,10 +15,10 @@ | |||
|  */ | ||||
| import {Protocol} from 'devtools-protocol'; | ||||
| import {ProtocolMapping} from 'devtools-protocol/types/protocol-mapping.js'; | ||||
| import {assert} from './assert.js'; | ||||
| import {assert} from '../util/assert.js'; | ||||
| import {ProtocolError} from './Errors.js'; | ||||
| import {EventEmitter} from './EventEmitter.js'; | ||||
| import {Frame} from './FrameManager.js'; | ||||
| import {Frame} from './Frame.js'; | ||||
| import {debugError, isString} from './util.js'; | ||||
| import {HTTPResponse} from './HTTPResponse.js'; | ||||
| 
 | ||||
|  |  | |||
|  | @ -16,7 +16,7 @@ | |||
| import {ProtocolMapping} from 'devtools-protocol/types/protocol-mapping.js'; | ||||
| 
 | ||||
| import {EventEmitter} from './EventEmitter.js'; | ||||
| import {Frame} from './FrameManager.js'; | ||||
| import {Frame} from './Frame.js'; | ||||
| import {HTTPRequest} from './HTTPRequest.js'; | ||||
| import {SecurityDetails} from './SecurityDetails.js'; | ||||
| import {Protocol} from 'devtools-protocol'; | ||||
|  |  | |||
|  | @ -14,7 +14,7 @@ | |||
|  * limitations under the License. | ||||
|  */ | ||||
| 
 | ||||
| import {assert} from './assert.js'; | ||||
| import {assert} from '../util/assert.js'; | ||||
| import {CDPSession} from './Connection.js'; | ||||
| import {_keyDefinitions, KeyDefinition, KeyInput} from './USKeyboardLayout.js'; | ||||
| import {Protocol} from 'devtools-protocol'; | ||||
|  |  | |||
|  | @ -15,24 +15,23 @@ | |||
|  */ | ||||
| 
 | ||||
| import {Protocol} from 'devtools-protocol'; | ||||
| import {assert} from './assert.js'; | ||||
| import {source as injectedSource} from '../generated/injected.js'; | ||||
| import {assert} from '../util/assert.js'; | ||||
| import {createDeferredPromise} from '../util/DeferredPromise.js'; | ||||
| import {CDPSession} from './Connection.js'; | ||||
| import {ElementHandle} from './ElementHandle.js'; | ||||
| import {TimeoutError} from './Errors.js'; | ||||
| import {ExecutionContext} from './ExecutionContext.js'; | ||||
| import {Frame, FrameManager} from './FrameManager.js'; | ||||
| import {Frame} from './Frame.js'; | ||||
| import {FrameManager} from './FrameManager.js'; | ||||
| import {MouseButton} from './Input.js'; | ||||
| import {JSHandle} from './JSHandle.js'; | ||||
| import {LifecycleWatcher, PuppeteerLifeCycleEvent} from './LifecycleWatcher.js'; | ||||
| import {getQueryHandlerAndSelector} from './QueryHandler.js'; | ||||
| import {TimeoutSettings} from './TimeoutSettings.js'; | ||||
| import {EvaluateFunc, HandleFor, NodeFor} from './types.js'; | ||||
| import { | ||||
|   createDeferredPromise, | ||||
|   createJSHandle, | ||||
|   debugError, | ||||
|   DeferredPromise, | ||||
|   importFS, | ||||
|   isNumber, | ||||
|   isString, | ||||
|   makePredicateString, | ||||
|  | @ -77,10 +76,6 @@ export interface WaitForSelectorOptions { | |||
|    * @defaultValue `30000` (30 seconds) | ||||
|    */ | ||||
|   timeout?: number; | ||||
|   /** | ||||
|    * @deprecated Do not use. Use the {@link ElementHandle.waitForSelector} | ||||
|    */ | ||||
|   root?: ElementHandle<Node>; | ||||
| } | ||||
| 
 | ||||
| /** | ||||
|  | @ -118,12 +113,10 @@ export interface IsolatedWorldChart { | |||
|  * @internal | ||||
|  */ | ||||
| export class IsolatedWorld { | ||||
|   #frameManager: FrameManager; | ||||
|   #client: CDPSession; | ||||
|   #frame: Frame; | ||||
|   #timeoutSettings: TimeoutSettings; | ||||
|   #documentPromise: Promise<ElementHandle<Document>> | null = null; | ||||
|   #contextPromise: DeferredPromise<ExecutionContext> = createDeferredPromise(); | ||||
|   #injected: boolean; | ||||
|   #document?: ElementHandle<Document>; | ||||
|   #context = createDeferredPromise<ExecutionContext>(); | ||||
|   #detached = false; | ||||
| 
 | ||||
|   // Set of bindings that have been registered in the current context.
 | ||||
|  | @ -145,44 +138,48 @@ export class IsolatedWorld { | |||
|     return `${name}_${contextId}`; | ||||
|   }; | ||||
| 
 | ||||
|   constructor( | ||||
|     client: CDPSession, | ||||
|     frameManager: FrameManager, | ||||
|     frame: Frame, | ||||
|     timeoutSettings: TimeoutSettings | ||||
|   ) { | ||||
|   constructor(frame: Frame, injected = false) { | ||||
|     // Keep own reference to client because it might differ from the FrameManager's
 | ||||
|     // client for OOP iframes.
 | ||||
|     this.#client = client; | ||||
|     this.#frameManager = frameManager; | ||||
|     this.#frame = frame; | ||||
|     this.#timeoutSettings = timeoutSettings; | ||||
|     this.#injected = injected; | ||||
|     this.#client.on('Runtime.bindingCalled', this.#onBindingCalled); | ||||
|   } | ||||
| 
 | ||||
|   get #client(): CDPSession { | ||||
|     return this.#frame._client(); | ||||
|   } | ||||
| 
 | ||||
|   get #frameManager(): FrameManager { | ||||
|     return this.#frame._frameManager; | ||||
|   } | ||||
| 
 | ||||
|   get #timeoutSettings(): TimeoutSettings { | ||||
|     return this.#frameManager.timeoutSettings; | ||||
|   } | ||||
| 
 | ||||
|   frame(): Frame { | ||||
|     return this.#frame; | ||||
|   } | ||||
| 
 | ||||
|   clearContext(): void { | ||||
|     this.#documentPromise = null; | ||||
|     this.#contextPromise = createDeferredPromise(); | ||||
|     this.#document = undefined; | ||||
|     this.#context = createDeferredPromise(); | ||||
|   } | ||||
| 
 | ||||
|   setContext(context: ExecutionContext): void { | ||||
|     assert( | ||||
|       this.#contextPromise, | ||||
|       `ExecutionContext ${context._contextId} has already been set.` | ||||
|     ); | ||||
|     if (this.#injected) { | ||||
|       context.evaluate(injectedSource).catch(debugError); | ||||
|     } | ||||
|     this.#ctxBindings.clear(); | ||||
|     this.#contextPromise.resolve(context); | ||||
|     this.#context.resolve(context); | ||||
|     for (const waitTask of this._waitTasks) { | ||||
|       waitTask.rerun(); | ||||
|     } | ||||
|   } | ||||
| 
 | ||||
|   hasContext(): boolean { | ||||
|     return this.#contextPromise.resolved(); | ||||
|     return this.#context.resolved(); | ||||
|   } | ||||
| 
 | ||||
|   _detach(): void { | ||||
|  | @ -201,10 +198,10 @@ export class IsolatedWorld { | |||
|         `Execution context is not available in detached frame "${this.#frame.url()}" (are you trying to evaluate?)` | ||||
|       ); | ||||
|     } | ||||
|     if (this.#contextPromise === null) { | ||||
|     if (this.#context === null) { | ||||
|       throw new Error(`Execution content promise is missing`); | ||||
|     } | ||||
|     return this.#contextPromise; | ||||
|     return this.#context; | ||||
|   } | ||||
| 
 | ||||
|   async evaluateHandle< | ||||
|  | @ -244,15 +241,14 @@ export class IsolatedWorld { | |||
|   } | ||||
| 
 | ||||
|   async document(): Promise<ElementHandle<Document>> { | ||||
|     if (this.#documentPromise) { | ||||
|       return this.#documentPromise; | ||||
|     if (this.#document) { | ||||
|       return this.#document; | ||||
|     } | ||||
|     this.#documentPromise = this.executionContext().then(async context => { | ||||
|       return await context.evaluateHandle(() => { | ||||
|         return document; | ||||
|       }); | ||||
|     const context = await this.executionContext(); | ||||
|     this.#document = await context.evaluateHandle(() => { | ||||
|       return document; | ||||
|     }); | ||||
|     return this.#documentPromise; | ||||
|     return this.#document; | ||||
|   } | ||||
| 
 | ||||
|   async $x(expression: string): Promise<Array<ElementHandle<Node>>> { | ||||
|  | @ -290,20 +286,6 @@ export class IsolatedWorld { | |||
|     return document.$$eval(selector, pageFunction, ...args); | ||||
|   } | ||||
| 
 | ||||
|   async waitForSelector<Selector extends string>( | ||||
|     selector: Selector, | ||||
|     options: WaitForSelectorOptions | ||||
|   ): Promise<ElementHandle<NodeFor<Selector>> | null> { | ||||
|     const {updatedSelector, queryHandler} = | ||||
|       getQueryHandlerAndSelector(selector); | ||||
|     assert(queryHandler.waitFor, 'Query handler does not support waiting'); | ||||
|     return (await queryHandler.waitFor( | ||||
|       this, | ||||
|       updatedSelector, | ||||
|       options | ||||
|     )) as ElementHandle<NodeFor<Selector>> | null; | ||||
|   } | ||||
| 
 | ||||
|   async content(): Promise<string> { | ||||
|     return await this.evaluate(() => { | ||||
|       let retVal = ''; | ||||
|  | @ -351,191 +333,6 @@ export class IsolatedWorld { | |||
|     } | ||||
|   } | ||||
| 
 | ||||
|   /** | ||||
|    * Adds a script tag into the current context. | ||||
|    * | ||||
|    * @remarks | ||||
|    * You can pass a URL, filepath or string of contents. Note that when running Puppeteer | ||||
|    * in a browser environment you cannot pass a filepath and should use either | ||||
|    * `url` or `content`. | ||||
|    */ | ||||
|   async addScriptTag(options: { | ||||
|     url?: string; | ||||
|     path?: string; | ||||
|     content?: string; | ||||
|     id?: string; | ||||
|     type?: string; | ||||
|   }): Promise<ElementHandle<HTMLScriptElement>> { | ||||
|     const { | ||||
|       url = null, | ||||
|       path = null, | ||||
|       content = null, | ||||
|       id = '', | ||||
|       type = '', | ||||
|     } = options; | ||||
|     if (url !== null) { | ||||
|       try { | ||||
|         const context = await this.executionContext(); | ||||
|         return await context.evaluateHandle(addScriptUrl, url, id, type); | ||||
|       } catch (error) { | ||||
|         throw new Error(`Loading script from ${url} failed`); | ||||
|       } | ||||
|     } | ||||
| 
 | ||||
|     if (path !== null) { | ||||
|       let fs; | ||||
|       try { | ||||
|         fs = (await import('fs')).promises; | ||||
|       } catch (error) { | ||||
|         if (error instanceof TypeError) { | ||||
|           throw new Error( | ||||
|             'Can only pass a filepath to addScriptTag in a Node-like environment.' | ||||
|           ); | ||||
|         } | ||||
|         throw error; | ||||
|       } | ||||
|       let contents = await fs.readFile(path, 'utf8'); | ||||
|       contents += '//# sourceURL=' + path.replace(/\n/g, ''); | ||||
|       const context = await this.executionContext(); | ||||
|       return await context.evaluateHandle(addScriptContent, contents, id, type); | ||||
|     } | ||||
| 
 | ||||
|     if (content !== null) { | ||||
|       const context = await this.executionContext(); | ||||
|       return await context.evaluateHandle(addScriptContent, content, id, type); | ||||
|     } | ||||
| 
 | ||||
|     throw new Error( | ||||
|       'Provide an object with a `url`, `path` or `content` property' | ||||
|     ); | ||||
| 
 | ||||
|     async function addScriptUrl(url: string, id: string, type: string) { | ||||
|       const script = document.createElement('script'); | ||||
|       script.src = url; | ||||
|       if (id) { | ||||
|         script.id = id; | ||||
|       } | ||||
|       if (type) { | ||||
|         script.type = type; | ||||
|       } | ||||
|       const promise = new Promise((res, rej) => { | ||||
|         script.onload = res; | ||||
|         script.onerror = rej; | ||||
|       }); | ||||
|       document.head.appendChild(script); | ||||
|       await promise; | ||||
|       return script; | ||||
|     } | ||||
| 
 | ||||
|     function addScriptContent( | ||||
|       content: string, | ||||
|       id: string, | ||||
|       type = 'text/javascript' | ||||
|     ) { | ||||
|       const script = document.createElement('script'); | ||||
|       script.type = type; | ||||
|       script.text = content; | ||||
|       if (id) { | ||||
|         script.id = id; | ||||
|       } | ||||
|       let error = null; | ||||
|       script.onerror = e => { | ||||
|         return (error = e); | ||||
|       }; | ||||
|       document.head.appendChild(script); | ||||
|       if (error) { | ||||
|         throw error; | ||||
|       } | ||||
|       return script; | ||||
|     } | ||||
|   } | ||||
| 
 | ||||
|   /** | ||||
|    * Adds a style tag into the current context. | ||||
|    * | ||||
|    * @remarks | ||||
|    * You can pass a URL, filepath or string of contents. Note that when running Puppeteer | ||||
|    * in a browser environment you cannot pass a filepath and should use either | ||||
|    * `url` or `content`. | ||||
|    */ | ||||
|   async addStyleTag(options: { | ||||
|     url?: string; | ||||
|     path?: string; | ||||
|     content?: string; | ||||
|   }): Promise<ElementHandle<HTMLStyleElement | HTMLLinkElement>> { | ||||
|     const {url = null, path = null, content = null} = options; | ||||
|     if (url !== null) { | ||||
|       try { | ||||
|         const context = await this.executionContext(); | ||||
|         return (await context.evaluateHandle( | ||||
|           addStyleUrl, | ||||
|           url | ||||
|         )) as ElementHandle<HTMLLinkElement>; | ||||
|       } catch (error) { | ||||
|         throw new Error(`Loading style from ${url} failed`); | ||||
|       } | ||||
|     } | ||||
| 
 | ||||
|     if (path !== null) { | ||||
|       let fs: typeof import('fs').promises; | ||||
|       try { | ||||
|         fs = (await importFS()).promises; | ||||
|       } catch (error) { | ||||
|         if (error instanceof TypeError) { | ||||
|           throw new Error( | ||||
|             'Cannot pass a filepath to addStyleTag in the browser environment.' | ||||
|           ); | ||||
|         } | ||||
|         throw error; | ||||
|       } | ||||
| 
 | ||||
|       let contents = await fs.readFile(path, 'utf8'); | ||||
|       contents += '/*# sourceURL=' + path.replace(/\n/g, '') + '*/'; | ||||
|       const context = await this.executionContext(); | ||||
|       return (await context.evaluateHandle( | ||||
|         addStyleContent, | ||||
|         contents | ||||
|       )) as ElementHandle<HTMLStyleElement>; | ||||
|     } | ||||
| 
 | ||||
|     if (content !== null) { | ||||
|       const context = await this.executionContext(); | ||||
|       return (await context.evaluateHandle( | ||||
|         addStyleContent, | ||||
|         content | ||||
|       )) as ElementHandle<HTMLStyleElement>; | ||||
|     } | ||||
| 
 | ||||
|     throw new Error( | ||||
|       'Provide an object with a `url`, `path` or `content` property' | ||||
|     ); | ||||
| 
 | ||||
|     async function addStyleUrl(url: string): Promise<HTMLElement> { | ||||
|       const link = document.createElement('link'); | ||||
|       link.rel = 'stylesheet'; | ||||
|       link.href = url; | ||||
|       const promise = new Promise((res, rej) => { | ||||
|         link.onload = res; | ||||
|         link.onerror = rej; | ||||
|       }); | ||||
|       document.head.appendChild(link); | ||||
|       await promise; | ||||
|       return link; | ||||
|     } | ||||
| 
 | ||||
|     async function addStyleContent(content: string): Promise<HTMLElement> { | ||||
|       const style = document.createElement('style'); | ||||
|       style.appendChild(document.createTextNode(content)); | ||||
|       const promise = new Promise((res, rej) => { | ||||
|         style.onload = res; | ||||
|         style.onerror = rej; | ||||
|       }); | ||||
|       document.head.appendChild(style); | ||||
|       await promise; | ||||
|       return style; | ||||
|     } | ||||
|   } | ||||
| 
 | ||||
|   async click( | ||||
|     selector: string, | ||||
|     options: {delay?: number; button?: MouseButton; clickCount?: number} | ||||
|  | @ -703,10 +500,11 @@ export class IsolatedWorld { | |||
| 
 | ||||
|   async _waitForSelectorInPage( | ||||
|     queryOne: Function, | ||||
|     root: ElementHandle<Node> | undefined, | ||||
|     selector: string, | ||||
|     options: WaitForSelectorOptions, | ||||
|     binding?: PageBinding | ||||
|   ): Promise<ElementHandle<Node> | null> { | ||||
|   ): Promise<JSHandle<unknown> | null> { | ||||
|     const { | ||||
|       visible: waitForVisible = false, | ||||
|       hidden: waitForHidden = false, | ||||
|  | @ -722,9 +520,7 @@ export class IsolatedWorld { | |||
|       waitForVisible: boolean, | ||||
|       waitForHidden: boolean | ||||
|     ): Promise<Node | null | boolean> { | ||||
|       const node = predicateQueryHandler | ||||
|         ? ((await predicateQueryHandler(root, selector)) as Element) | ||||
|         : root.querySelector(selector); | ||||
|       const node = (await predicateQueryHandler(root, selector)) as Element; | ||||
|       return checkWaitForOptions(node, waitForVisible, waitForHidden); | ||||
|     } | ||||
|     const waitTaskOptions: WaitTaskOptions = { | ||||
|  | @ -736,16 +532,10 @@ export class IsolatedWorld { | |||
|       timeout, | ||||
|       args: [selector, waitForVisible, waitForHidden], | ||||
|       binding, | ||||
|       root: options.root, | ||||
|       root, | ||||
|     }; | ||||
|     const waitTask = new WaitTask(waitTaskOptions); | ||||
|     const jsHandle = await waitTask.promise; | ||||
|     const elementHandle = jsHandle.asElement(); | ||||
|     if (!elementHandle) { | ||||
|       await jsHandle.dispose(); | ||||
|       return null; | ||||
|     } | ||||
|     return elementHandle; | ||||
|     return waitTask.promise; | ||||
|   } | ||||
| 
 | ||||
|   waitForFunction( | ||||
|  | @ -796,6 +586,12 @@ export class IsolatedWorld { | |||
|     }); | ||||
|     return (await this.adoptBackendNode(nodeInfo.node.backendNodeId)) as T; | ||||
|   } | ||||
| 
 | ||||
|   async transferHandle<T extends JSHandle<Node>>(handle: T): Promise<T> { | ||||
|     const result = await this.adoptHandle(handle); | ||||
|     await handle.dispose(); | ||||
|     return result; | ||||
|   } | ||||
| } | ||||
| 
 | ||||
| /** | ||||
|  |  | |||
|  | @ -15,7 +15,7 @@ | |||
|  */ | ||||
| 
 | ||||
| import {Protocol} from 'devtools-protocol'; | ||||
| import {assert} from './assert.js'; | ||||
| import {assert} from '../util/assert.js'; | ||||
| import {CDPSession} from './Connection.js'; | ||||
| import type {ElementHandle} from './ElementHandle.js'; | ||||
| import {ExecutionContext} from './ExecutionContext.js'; | ||||
|  | @ -78,7 +78,6 @@ export class JSHandle<T = unknown> { | |||
|    */ | ||||
|   [__JSHandleSymbol]?: T; | ||||
| 
 | ||||
|   #client: CDPSession; | ||||
|   #disposed = false; | ||||
|   #context: ExecutionContext; | ||||
|   #remoteObject: Protocol.Runtime.RemoteObject; | ||||
|  | @ -87,7 +86,7 @@ export class JSHandle<T = unknown> { | |||
|    * @internal | ||||
|    */ | ||||
|   get client(): CDPSession { | ||||
|     return this.#client; | ||||
|     return this.#context._client; | ||||
|   } | ||||
| 
 | ||||
|   /** | ||||
|  | @ -102,16 +101,14 @@ export class JSHandle<T = unknown> { | |||
|    */ | ||||
|   constructor( | ||||
|     context: ExecutionContext, | ||||
|     client: CDPSession, | ||||
|     remoteObject: Protocol.Runtime.RemoteObject | ||||
|   ) { | ||||
|     this.#context = context; | ||||
|     this.#client = client; | ||||
|     this.#remoteObject = remoteObject; | ||||
|   } | ||||
| 
 | ||||
|   /** | ||||
|    * @returns The execution context the handle belongs to. | ||||
|    * @internal | ||||
|    */ | ||||
|   executionContext(): ExecutionContext { | ||||
|     return this.#context; | ||||
|  | @ -196,7 +193,7 @@ export class JSHandle<T = unknown> { | |||
|     assert(this.#remoteObject.objectId); | ||||
|     // We use Runtime.getProperties rather than iterative building because the
 | ||||
|     // iterative approach might create a distorted snapshot.
 | ||||
|     const response = await this.#client.send('Runtime.getProperties', { | ||||
|     const response = await this.client.send('Runtime.getProperties', { | ||||
|       objectId: this.#remoteObject.objectId, | ||||
|       ownProperties: true, | ||||
|     }); | ||||
|  | @ -247,7 +244,7 @@ export class JSHandle<T = unknown> { | |||
|       return; | ||||
|     } | ||||
|     this.#disposed = true; | ||||
|     await releaseObject(this.#client, this.#remoteObject); | ||||
|     await releaseObject(this.client, this.#remoteObject); | ||||
|   } | ||||
| 
 | ||||
|   /** | ||||
|  | @ -279,11 +276,11 @@ export class JSHandle<T = unknown> { | |||
|  */ | ||||
| export interface Offset { | ||||
|   /** | ||||
|    * x-offset for the clickable point relative to the top-left corder of the border box. | ||||
|    * x-offset for the clickable point relative to the top-left corner of the border box. | ||||
|    */ | ||||
|   x: number; | ||||
|   /** | ||||
|    * y-offset for the clickable point relative to the top-left corder of the border box. | ||||
|    * y-offset for the clickable point relative to the top-left corner of the border box. | ||||
|    */ | ||||
|   y: number; | ||||
| } | ||||
|  | @ -307,7 +304,7 @@ export interface ClickOptions { | |||
|    */ | ||||
|   clickCount?: number; | ||||
|   /** | ||||
|    * Offset for the clickable point relative to the top-left corder of the border box. | ||||
|    * Offset for the clickable point relative to the top-left corner of the border box. | ||||
|    */ | ||||
|   offset?: Offset; | ||||
| } | ||||
|  |  | |||
|  | @ -14,20 +14,19 @@ | |||
|  * limitations under the License. | ||||
|  */ | ||||
| 
 | ||||
| import {assert} from './assert.js'; | ||||
| import {assert} from '../util/assert.js'; | ||||
| import { | ||||
|   addEventListener, | ||||
|   PuppeteerEventListener, | ||||
|   removeEventListeners, | ||||
| } from './util.js'; | ||||
| import { | ||||
|   DeferredPromise, | ||||
|   createDeferredPromise, | ||||
| } from './util.js'; | ||||
| } from '../util/DeferredPromise.js'; | ||||
| import {TimeoutError} from './Errors.js'; | ||||
| import { | ||||
|   FrameManager, | ||||
|   Frame, | ||||
|   FrameManagerEmittedEvents, | ||||
| } from './FrameManager.js'; | ||||
| import {FrameManager, FrameManagerEmittedEvents} from './FrameManager.js'; | ||||
| import {Frame} from './Frame.js'; | ||||
| import {HTTPRequest} from './HTTPRequest.js'; | ||||
| import {HTTPResponse} from './HTTPResponse.js'; | ||||
| import {NetworkManagerEmittedEvents} from './NetworkManager.js'; | ||||
|  | @ -180,9 +179,10 @@ export class LifecycleWatcher { | |||
|       return; | ||||
|     } | ||||
|     this.#navigationRequest = request; | ||||
|     this.#navigationResponseReceived?.reject( | ||||
|       new Error('New navigation request was received') | ||||
|     ); | ||||
|     // Resolve previous navigation response in case there are multiple
 | ||||
|     // navigation requests reported by the backend. This generally should not
 | ||||
|     // happen by it looks like it's possible.
 | ||||
|     this.#navigationResponseReceived?.resolve(); | ||||
|     this.#navigationResponseReceived = createDeferredPromise(); | ||||
|     if (request.response() !== null) { | ||||
|       this.#navigationResponseReceived?.resolve(); | ||||
|  |  | |||
|  | @ -16,18 +16,15 @@ | |||
| 
 | ||||
| import {Protocol} from 'devtools-protocol'; | ||||
| import {ProtocolMapping} from 'devtools-protocol/types/protocol-mapping.js'; | ||||
| import {assert} from './assert.js'; | ||||
| import {assert} from '../util/assert.js'; | ||||
| import {EventEmitter} from './EventEmitter.js'; | ||||
| import {Frame} from './FrameManager.js'; | ||||
| import {Frame} from './Frame.js'; | ||||
| import {HTTPRequest} from './HTTPRequest.js'; | ||||
| import {HTTPResponse} from './HTTPResponse.js'; | ||||
| import {FetchRequestId, NetworkEventManager} from './NetworkEventManager.js'; | ||||
| import { | ||||
|   debugError, | ||||
|   isString, | ||||
|   createDeferredPromiseWithTimer, | ||||
|   DeferredPromise, | ||||
| } from './util.js'; | ||||
| import {debugError, isString} from './util.js'; | ||||
| import {DeferredPromise} from '../util/DeferredPromise.js'; | ||||
| import {createDebuggableDeferredPromise} from '../util/DebuggableDeferredPromise.js'; | ||||
| 
 | ||||
| /** | ||||
|  * @public | ||||
|  | @ -145,9 +142,8 @@ export class NetworkManager extends EventEmitter { | |||
|     if (this.#deferredInitPromise) { | ||||
|       return this.#deferredInitPromise; | ||||
|     } | ||||
|     this.#deferredInitPromise = createDeferredPromiseWithTimer<void>( | ||||
|       'NetworkManager initialization timed out', | ||||
|       30000 | ||||
|     this.#deferredInitPromise = createDebuggableDeferredPromise( | ||||
|       'NetworkManager initialization timed out' | ||||
|     ); | ||||
|     const init = Promise.all([ | ||||
|       this.#ignoreHTTPSErrors | ||||
|  |  | |||
|  | @ -16,26 +16,32 @@ | |||
| 
 | ||||
| import {Protocol} from 'devtools-protocol'; | ||||
| import type {Readable} from 'stream'; | ||||
| import {assert} from '../util/assert.js'; | ||||
| import { | ||||
|   createDeferredPromise, | ||||
|   DeferredPromise, | ||||
| } from '../util/DeferredPromise.js'; | ||||
| import {isErrorLike} from '../util/ErrorLike.js'; | ||||
| import {Accessibility} from './Accessibility.js'; | ||||
| import {assert} from './assert.js'; | ||||
| import {Browser, BrowserContext} from './Browser.js'; | ||||
| import {CDPSession, CDPSessionEmittedEvents} from './Connection.js'; | ||||
| import {ConsoleMessage, ConsoleMessageType} from './ConsoleMessage.js'; | ||||
| import {Coverage} from './Coverage.js'; | ||||
| import {Dialog} from './Dialog.js'; | ||||
| import {MAIN_WORLD, WaitForSelectorOptions} from './IsolatedWorld.js'; | ||||
| import {ElementHandle} from './ElementHandle.js'; | ||||
| import {EmulationManager} from './EmulationManager.js'; | ||||
| import {EventEmitter, Handler} from './EventEmitter.js'; | ||||
| import {FileChooser} from './FileChooser.js'; | ||||
| import { | ||||
|   Frame, | ||||
|   FrameManager, | ||||
|   FrameManagerEmittedEvents, | ||||
| } from './FrameManager.js'; | ||||
|   FrameAddScriptTagOptions, | ||||
|   FrameAddStyleTagOptions, | ||||
| } from './Frame.js'; | ||||
| import {FrameManager, FrameManagerEmittedEvents} from './FrameManager.js'; | ||||
| import {HTTPRequest} from './HTTPRequest.js'; | ||||
| import {HTTPResponse} from './HTTPResponse.js'; | ||||
| import {Keyboard, Mouse, MouseButton, Touchscreen} from './Input.js'; | ||||
| import {MAIN_WORLD, WaitForSelectorOptions} from './IsolatedWorld.js'; | ||||
| import {JSHandle} from './JSHandle.js'; | ||||
| import {PuppeteerLifeCycleEvent} from './LifecycleWatcher.js'; | ||||
| import { | ||||
|  | @ -56,10 +62,9 @@ import { | |||
|   debugError, | ||||
|   evaluationString, | ||||
|   getExceptionMessage, | ||||
|   importFS, | ||||
|   getReadableAsBuffer, | ||||
|   getReadableFromProtocolStream, | ||||
|   isErrorLike, | ||||
|   importFS, | ||||
|   isNumber, | ||||
|   isString, | ||||
|   pageBindingDeliverErrorString, | ||||
|  | @ -70,8 +75,6 @@ import { | |||
|   valueFromRemoteObject, | ||||
|   waitForEvent, | ||||
|   waitWithTimeout, | ||||
|   createDeferredPromiseWithTimer, | ||||
|   DeferredPromise, | ||||
| } from './util.js'; | ||||
| import {WebWorker} from './WebWorker.js'; | ||||
| 
 | ||||
|  | @ -160,6 +163,10 @@ export interface ScreenshotClip { | |||
|   y: number; | ||||
|   width: number; | ||||
|   height: number; | ||||
|   /** | ||||
|    * @defaultValue 1 | ||||
|    */ | ||||
|   scale?: number; | ||||
| } | ||||
| 
 | ||||
| /** | ||||
|  | @ -760,24 +767,28 @@ export class Page extends EventEmitter { | |||
|    * await fileChooser.accept(['/tmp/myfile.pdf']); | ||||
|    * ``` | ||||
|    */ | ||||
|   async waitForFileChooser( | ||||
|     options: WaitTimeoutOptions = {} | ||||
|   ): Promise<FileChooser> { | ||||
|     if (!this.#fileChooserPromises.size) { | ||||
|       await this.#client.send('Page.setInterceptFileChooserDialog', { | ||||
|   waitForFileChooser(options: WaitTimeoutOptions = {}): Promise<FileChooser> { | ||||
|     const needsEnable = this.#fileChooserPromises.size === 0; | ||||
|     const {timeout = this.#timeoutSettings.timeout()} = options; | ||||
|     const promise = createDeferredPromise<FileChooser>({ | ||||
|       message: `Waiting for \`FileChooser\` failed: ${timeout}ms exceeded`, | ||||
|       timeout, | ||||
|     }); | ||||
|     this.#fileChooserPromises.add(promise); | ||||
|     let enablePromise: Promise<void> | undefined; | ||||
|     if (needsEnable) { | ||||
|       enablePromise = this.#client.send('Page.setInterceptFileChooserDialog', { | ||||
|         enabled: true, | ||||
|       }); | ||||
|     } | ||||
| 
 | ||||
|     const {timeout = this.#timeoutSettings.timeout()} = options; | ||||
|     const promise = createDeferredPromiseWithTimer<FileChooser>( | ||||
|       `Waiting for \`FileChooser\` failed: ${timeout}ms exceeded` | ||||
|     ); | ||||
|     this.#fileChooserPromises.add(promise); | ||||
|     return promise.catch(error => { | ||||
|       this.#fileChooserPromises.delete(promise); | ||||
|       throw error; | ||||
|     }); | ||||
|     return Promise.all([promise, enablePromise]) | ||||
|       .then(([result]) => { | ||||
|         return result; | ||||
|       }) | ||||
|       .catch(error => { | ||||
|         this.#fileChooserPromises.delete(promise); | ||||
|         throw error; | ||||
|       }); | ||||
|   } | ||||
| 
 | ||||
|   /** | ||||
|  | @ -1038,6 +1049,13 @@ export class Page extends EventEmitter { | |||
|     this.#timeoutSettings.setDefaultTimeout(timeout); | ||||
|   } | ||||
| 
 | ||||
|   /** | ||||
|    * @returns Maximum time in milliseconds. | ||||
|    */ | ||||
|   getDefaultTimeout(): number { | ||||
|     return this.#timeoutSettings.timeout(); | ||||
|   } | ||||
| 
 | ||||
|   /** | ||||
|    * Runs `document.querySelector` within the page. If no element matches the | ||||
|    * selector, the return value resolves to `null`. | ||||
|  | @ -1137,11 +1155,6 @@ export class Page extends EventEmitter { | |||
|    * This method iterates the JavaScript heap and finds all objects with the | ||||
|    * given prototype. | ||||
|    * | ||||
|    * @remarks | ||||
|    * Shortcut for | ||||
|    * {@link ExecutionContext.queryObjects | | ||||
|    * page.mainFrame().executionContext().queryObjects(prototypeHandle)}. | ||||
|    * | ||||
|    * @example | ||||
|    * | ||||
|    * ```ts
 | ||||
|  | @ -1165,7 +1178,16 @@ export class Page extends EventEmitter { | |||
|     prototypeHandle: JSHandle<Prototype> | ||||
|   ): Promise<JSHandle<Prototype[]>> { | ||||
|     const context = await this.mainFrame().executionContext(); | ||||
|     return context.queryObjects(prototypeHandle); | ||||
|     assert(!prototypeHandle.disposed, 'Prototype JSHandle is disposed!'); | ||||
|     const remoteObject = prototypeHandle.remoteObject(); | ||||
|     assert( | ||||
|       remoteObject.objectId, | ||||
|       'Prototype JSHandle must not be referencing primitive value' | ||||
|     ); | ||||
|     const response = await context._client.send('Runtime.queryObjects', { | ||||
|       prototypeObjectId: remoteObject.objectId, | ||||
|     }); | ||||
|     return createJSHandle(context, response.objects) as HandleFor<Prototype[]>; | ||||
|   } | ||||
| 
 | ||||
|   /** | ||||
|  | @ -1408,30 +1430,35 @@ export class Page extends EventEmitter { | |||
|    * Shortcut for | ||||
|    * {@link Frame.addScriptTag | page.mainFrame().addScriptTag(options)}. | ||||
|    * | ||||
|    * @returns Promise which resolves to the added tag when the script's onload | ||||
|    * fires or when the script content was injected into frame. | ||||
|    * @param options - Options for the script. | ||||
|    * @returns An {@link ElementHandle | element handle} to the injected | ||||
|    * `<script>` element. | ||||
|    */ | ||||
|   async addScriptTag(options: { | ||||
|     url?: string; | ||||
|     path?: string; | ||||
|     content?: string; | ||||
|     type?: string; | ||||
|     id?: string; | ||||
|   }): Promise<ElementHandle<HTMLScriptElement>> { | ||||
|   async addScriptTag( | ||||
|     options: FrameAddScriptTagOptions | ||||
|   ): Promise<ElementHandle<HTMLScriptElement>> { | ||||
|     return this.mainFrame().addScriptTag(options); | ||||
|   } | ||||
| 
 | ||||
|   /** | ||||
|    * Adds a `<link rel="stylesheet">` tag into the page with the desired URL or a | ||||
|    * `<style type="text/css">` tag with the content. | ||||
|    * @returns Promise which resolves to the added tag when the stylesheet's | ||||
|    * onload fires or when the CSS content was injected into frame. | ||||
|    * Adds a `<link rel="stylesheet">` tag into the page with the desired URL or | ||||
|    * a `<style type="text/css">` tag with the content. | ||||
|    * | ||||
|    * Shortcut for | ||||
|    * {@link Frame.addStyleTag | page.mainFrame().addStyleTag(options)}. | ||||
|    * | ||||
|    * @returns An {@link ElementHandle | element handle} to the injected `<link>` | ||||
|    * or `<style>` element. | ||||
|    */ | ||||
|   async addStyleTag(options: { | ||||
|     url?: string; | ||||
|     path?: string; | ||||
|     content?: string; | ||||
|   }): Promise<ElementHandle<Node>> { | ||||
|   async addStyleTag( | ||||
|     options: Omit<FrameAddStyleTagOptions, 'url'> | ||||
|   ): Promise<ElementHandle<HTMLStyleElement>>; | ||||
|   async addStyleTag( | ||||
|     options: FrameAddStyleTagOptions | ||||
|   ): Promise<ElementHandle<HTMLLinkElement>>; | ||||
|   async addStyleTag( | ||||
|     options: FrameAddStyleTagOptions | ||||
|   ): Promise<ElementHandle<HTMLStyleElement | HTMLLinkElement>> { | ||||
|     return this.mainFrame().addStyleTag(options); | ||||
|   } | ||||
| 
 | ||||
|  | @ -2764,13 +2791,13 @@ export class Page extends EventEmitter { | |||
|    * | ||||
|    * // overwrite the `languages` property to use a custom getter
 | ||||
|    * Object.defineProperty(navigator, 'languages', { | ||||
|    * get: function () { | ||||
|    * return ['en-US', 'en', 'bn']; | ||||
|    * }, | ||||
|    *   get: function () { | ||||
|    *     return ['en-US', 'en', 'bn']; | ||||
|    *   }, | ||||
|    * }); | ||||
|    * | ||||
|    * // In your puppeteer script, assuming the preload.js file is
 | ||||
|    * in same folder of our script | ||||
|    * // in same folder of our script.
 | ||||
|    * const preloadFile = fs.readFileSync('./preload.js', 'utf8'); | ||||
|    * await page.evaluateOnNewDocument(preloadFile); | ||||
|    * ``` | ||||
|  | @ -2989,7 +3016,12 @@ export class Page extends EventEmitter { | |||
|     const result = await this.#client.send('Page.captureScreenshot', { | ||||
|       format, | ||||
|       quality: options.quality, | ||||
|       clip, | ||||
|       clip: clip | ||||
|         ? { | ||||
|             ...clip, | ||||
|             scale: clip.scale === undefined ? 1 : clip.scale, | ||||
|           } | ||||
|         : undefined, | ||||
|       captureBeyondViewport, | ||||
|       fromSurface, | ||||
|     }); | ||||
|  | @ -3021,14 +3053,12 @@ export class Page extends EventEmitter { | |||
|     } | ||||
|     return buffer; | ||||
| 
 | ||||
|     function processClip( | ||||
|       clip: ScreenshotClip | ||||
|     ): ScreenshotClip & {scale: number} { | ||||
|     function processClip(clip: ScreenshotClip): ScreenshotClip { | ||||
|       const x = Math.round(clip.x); | ||||
|       const y = Math.round(clip.y); | ||||
|       const width = Math.round(clip.width + clip.x - x); | ||||
|       const height = Math.round(clip.height + clip.y - y); | ||||
|       return {x, y, width, height, scale: 1}; | ||||
|       return {x, y, width, height, scale: clip.scale}; | ||||
|     } | ||||
|   } | ||||
| 
 | ||||
|  | @ -3393,7 +3423,7 @@ export class Page extends EventEmitter { | |||
|    */ | ||||
|   async waitForSelector<Selector extends string>( | ||||
|     selector: Selector, | ||||
|     options: Exclude<WaitForSelectorOptions, 'root'> = {} | ||||
|     options: WaitForSelectorOptions = {} | ||||
|   ): Promise<ElementHandle<NodeFor<Selector>> | null> { | ||||
|     return await this.mainFrame().waitForSelector(selector, options); | ||||
|   } | ||||
|  |  | |||
|  | @ -15,9 +15,13 @@ | |||
|  */ | ||||
| 
 | ||||
| import {ariaHandler} from './AriaQueryHandler.js'; | ||||
| import {IsolatedWorld, WaitForSelectorOptions} from './IsolatedWorld.js'; | ||||
| import {ElementHandle} from './ElementHandle.js'; | ||||
| import {JSHandle} from './JSHandle.js'; | ||||
| import {Frame} from './Frame.js'; | ||||
| import { | ||||
|   MAIN_WORLD, | ||||
|   PUPPETEER_WORLD, | ||||
|   WaitForSelectorOptions, | ||||
| } from './IsolatedWorld.js'; | ||||
| 
 | ||||
| /** | ||||
|  * @public | ||||
|  | @ -55,25 +59,13 @@ export interface InternalQueryHandler { | |||
|     element: ElementHandle<Node>, | ||||
|     selector: string | ||||
|   ) => Promise<Array<ElementHandle<Node>>>; | ||||
|   /** | ||||
|    * Queries for multiple nodes given a selector and {@link ElementHandle}. | ||||
|    * Unlike {@link queryAll}, this returns a handle to a node array. | ||||
|    * | ||||
|    * Akin to {@link Window.prototype.querySelectorAll}. | ||||
|    */ | ||||
|   queryAllArray?: ( | ||||
|     element: ElementHandle<Node>, | ||||
|     selector: string | ||||
|   ) => Promise<JSHandle<Node[]>>; | ||||
| 
 | ||||
|   /** | ||||
|    * Waits until a single node appears for a given selector and | ||||
|    * {@link ElementHandle}. | ||||
|    * | ||||
|    * Akin to {@link Window.prototype.querySelectorAll}. | ||||
|    */ | ||||
|   waitFor?: ( | ||||
|     isolatedWorld: IsolatedWorld, | ||||
|     elementOrFrame: ElementHandle<Node> | Frame, | ||||
|     selector: string, | ||||
|     options: WaitForSelectorOptions | ||||
|   ) => Promise<ElementHandle<Node> | null>; | ||||
|  | @ -95,12 +87,34 @@ function internalizeCustomQueryHandler( | |||
|       await jsHandle.dispose(); | ||||
|       return null; | ||||
|     }; | ||||
|     internalHandler.waitFor = ( | ||||
|       domWorld: IsolatedWorld, | ||||
|       selector: string, | ||||
|       options: WaitForSelectorOptions | ||||
|     ) => { | ||||
|       return domWorld._waitForSelectorInPage(queryOne, selector, options); | ||||
|     internalHandler.waitFor = async (elementOrFrame, selector, options) => { | ||||
|       let frame: Frame; | ||||
|       let element: ElementHandle<Node> | undefined; | ||||
|       if (elementOrFrame instanceof Frame) { | ||||
|         frame = elementOrFrame; | ||||
|       } else { | ||||
|         frame = elementOrFrame.frame; | ||||
|         element = await frame.worlds[PUPPETEER_WORLD].adoptHandle( | ||||
|           elementOrFrame | ||||
|         ); | ||||
|       } | ||||
|       const result = await frame.worlds[PUPPETEER_WORLD]._waitForSelectorInPage( | ||||
|         queryOne, | ||||
|         element, | ||||
|         selector, | ||||
|         options | ||||
|       ); | ||||
|       if (element) { | ||||
|         await element.dispose(); | ||||
|       } | ||||
|       if (!result) { | ||||
|         return null; | ||||
|       } | ||||
|       if (!(result instanceof ElementHandle)) { | ||||
|         await result.dispose(); | ||||
|         return null; | ||||
|       } | ||||
|       return frame.worlds[MAIN_WORLD].transferHandle(result); | ||||
|     }; | ||||
|   } | ||||
| 
 | ||||
|  | @ -119,16 +133,6 @@ function internalizeCustomQueryHandler( | |||
|       } | ||||
|       return result; | ||||
|     }; | ||||
|     internalHandler.queryAllArray = async (element, selector) => { | ||||
|       const resultHandle = (await element.evaluateHandle( | ||||
|         queryAll, | ||||
|         selector | ||||
|       )) as JSHandle<Element[] | NodeListOf<Element>>; | ||||
|       const arrayHandle = await resultHandle.evaluateHandle(res => { | ||||
|         return Array.from(res); | ||||
|       }); | ||||
|       return arrayHandle; | ||||
|     }; | ||||
|   } | ||||
| 
 | ||||
|   return internalHandler; | ||||
|  |  | |||
|  | @ -13,12 +13,9 @@ | |||
|  * See the License for the specific language governing permissions and | ||||
|  * limitations under the License. | ||||
|  */ | ||||
| import {assert} from './assert.js'; | ||||
| import { | ||||
|   getReadableAsBuffer, | ||||
|   getReadableFromProtocolStream, | ||||
|   isErrorLike, | ||||
| } from './util.js'; | ||||
| import {assert} from '../util/assert.js'; | ||||
| import {getReadableAsBuffer, getReadableFromProtocolStream} from './util.js'; | ||||
| import {isErrorLike} from '../util/ErrorLike.js'; | ||||
| import {CDPSession} from './Connection.js'; | ||||
| 
 | ||||
| /** | ||||
|  |  | |||
|  | @ -21,6 +21,7 @@ import {EventEmitter} from './EventEmitter.js'; | |||
| import {ExecutionContext} from './ExecutionContext.js'; | ||||
| import {JSHandle} from './JSHandle.js'; | ||||
| import {debugError} from './util.js'; | ||||
| import {createDeferredPromise} from '../util/DeferredPromise.js'; | ||||
| 
 | ||||
| /** | ||||
|  * @internal | ||||
|  | @ -38,8 +39,6 @@ export type ExceptionThrownCallback = ( | |||
|   details: Protocol.Runtime.ExceptionDetails | ||||
| ) => void; | ||||
| 
 | ||||
| type JSHandleFactory = (obj: Protocol.Runtime.RemoteObject) => JSHandle; | ||||
| 
 | ||||
| /** | ||||
|  * This class represents a | ||||
|  * {@link https://developer.mozilla.org/en-US/docs/Web/API/Web_Workers_API | WebWorker}.
 | ||||
|  | @ -67,10 +66,10 @@ type JSHandleFactory = (obj: Protocol.Runtime.RemoteObject) => JSHandle; | |||
|  * @public | ||||
|  */ | ||||
| export class WebWorker extends EventEmitter { | ||||
|   #executionContext = createDeferredPromise<ExecutionContext>(); | ||||
| 
 | ||||
|   #client: CDPSession; | ||||
|   #url: string; | ||||
|   #executionContextPromise: Promise<ExecutionContext>; | ||||
|   #executionContextCallback!: (value: ExecutionContext) => void; | ||||
| 
 | ||||
|   /** | ||||
|    * @internal | ||||
|  | @ -84,32 +83,34 @@ export class WebWorker extends EventEmitter { | |||
|     super(); | ||||
|     this.#client = client; | ||||
|     this.#url = url; | ||||
|     this.#executionContextPromise = new Promise<ExecutionContext>(x => { | ||||
|       return (this.#executionContextCallback = x); | ||||
|     }); | ||||
| 
 | ||||
|     let jsHandleFactory: JSHandleFactory; | ||||
|     this.#client.once('Runtime.executionContextCreated', async event => { | ||||
|       // eslint-disable-next-line @typescript-eslint/explicit-function-return-type
 | ||||
|       jsHandleFactory = remoteObject => { | ||||
|         return new JSHandle(executionContext, client, remoteObject); | ||||
|       }; | ||||
|       const executionContext = new ExecutionContext(client, event.context); | ||||
|       this.#executionContextCallback(executionContext); | ||||
|       const context = new ExecutionContext(client, event.context); | ||||
|       this.#executionContext.resolve(context); | ||||
|     }); | ||||
| 
 | ||||
|     // This might fail if the target is closed before we receive all execution contexts.
 | ||||
|     this.#client.send('Runtime.enable').catch(debugError); | ||||
|     this.#client.on('Runtime.consoleAPICalled', event => { | ||||
|     this.#client.on('Runtime.consoleAPICalled', async event => { | ||||
|       const context = await this.#executionContext; | ||||
|       return consoleAPICalled( | ||||
|         event.type, | ||||
|         event.args.map(jsHandleFactory), | ||||
|         event.args.map((object: Protocol.Runtime.RemoteObject) => { | ||||
|           return new JSHandle(context, object); | ||||
|         }), | ||||
|         event.stackTrace | ||||
|       ); | ||||
|     }); | ||||
|     this.#client.on('Runtime.exceptionThrown', exception => { | ||||
|       return exceptionThrown(exception.exceptionDetails); | ||||
|     }); | ||||
| 
 | ||||
|     // This might fail if the target is closed before we receive all execution contexts.
 | ||||
|     this.#client.send('Runtime.enable').catch(debugError); | ||||
|   } | ||||
| 
 | ||||
|   /** | ||||
|    * @internal | ||||
|    */ | ||||
|   async executionContext(): Promise<ExecutionContext> { | ||||
|     return this.#executionContext; | ||||
|   } | ||||
| 
 | ||||
|   /** | ||||
|  | @ -119,14 +120,6 @@ export class WebWorker extends EventEmitter { | |||
|     return this.#url; | ||||
|   } | ||||
| 
 | ||||
|   /** | ||||
|    * Returns the ExecutionContext the WebWorker runs in | ||||
|    * @returns The ExecutionContext the web worker runs in. | ||||
|    */ | ||||
|   async executionContext(): Promise<ExecutionContext> { | ||||
|     return this.#executionContextPromise; | ||||
|   } | ||||
| 
 | ||||
|   /** | ||||
|    * If the function passed to the `worker.evaluate` returns a Promise, then | ||||
|    * `worker.evaluate` would wait for the promise to resolve and return its | ||||
|  | @ -148,10 +141,8 @@ export class WebWorker extends EventEmitter { | |||
|     pageFunction: Func | string, | ||||
|     ...args: Params | ||||
|   ): Promise<Awaited<ReturnType<Func>>> { | ||||
|     return (await this.#executionContextPromise).evaluate( | ||||
|       pageFunction, | ||||
|       ...args | ||||
|     ); | ||||
|     const context = await this.#executionContext; | ||||
|     return context.evaluate(pageFunction, ...args); | ||||
|   } | ||||
| 
 | ||||
|   /** | ||||
|  | @ -173,9 +164,7 @@ export class WebWorker extends EventEmitter { | |||
|     pageFunction: Func | string, | ||||
|     ...args: Params | ||||
|   ): Promise<HandleFor<Awaited<ReturnType<Func>>>> { | ||||
|     return (await this.#executionContextPromise).evaluateHandle( | ||||
|       pageFunction, | ||||
|       ...args | ||||
|     ); | ||||
|     const context = await this.#executionContext; | ||||
|     return context.evaluateHandle(pageFunction, ...args); | ||||
|   } | ||||
| } | ||||
|  |  | |||
|  | @ -17,7 +17,8 @@ | |||
| import {Protocol} from 'devtools-protocol'; | ||||
| import type {Readable} from 'stream'; | ||||
| import {isNode} from '../environment.js'; | ||||
| import {assert} from './assert.js'; | ||||
| import {assert} from '../util/assert.js'; | ||||
| import {isErrorLike} from '../util/ErrorLike.js'; | ||||
| import {CDPSession} from './Connection.js'; | ||||
| import {debug} from './Debug.js'; | ||||
| import {ElementHandle} from './ElementHandle.js'; | ||||
|  | @ -216,19 +217,10 @@ export function createJSHandle( | |||
|   context: ExecutionContext, | ||||
|   remoteObject: Protocol.Runtime.RemoteObject | ||||
| ): JSHandle | ElementHandle<Node> { | ||||
|   const frame = context.frame(); | ||||
|   if (remoteObject.subtype === 'node' && frame) { | ||||
|     const frameManager = frame._frameManager; | ||||
|     return new ElementHandle( | ||||
|       context, | ||||
|       context._client, | ||||
|       remoteObject, | ||||
|       frame, | ||||
|       frameManager.page(), | ||||
|       frameManager | ||||
|     ); | ||||
|   if (remoteObject.subtype === 'node' && context._world) { | ||||
|     return new ElementHandle(context, remoteObject, context._world.frame()); | ||||
|   } | ||||
|   return new JSHandle(context, context._client, remoteObject); | ||||
|   return new JSHandle(context, remoteObject); | ||||
| } | ||||
| 
 | ||||
| /** | ||||
|  | @ -341,7 +333,7 @@ export function pageBindingDeliverErrorValueString( | |||
|  */ | ||||
| export function makePredicateString( | ||||
|   predicate: Function, | ||||
|   predicateQueryHandler?: Function | ||||
|   predicateQueryHandler: Function | ||||
| ): string { | ||||
|   function checkWaitForOptions( | ||||
|     node: Node | null, | ||||
|  | @ -371,12 +363,10 @@ export function makePredicateString( | |||
|       return !!(rect.top || rect.bottom || rect.width || rect.height); | ||||
|     } | ||||
|   } | ||||
|   const predicateQueryHandlerDef = predicateQueryHandler | ||||
|     ? `const predicateQueryHandler = ${predicateQueryHandler};` | ||||
|     : ''; | ||||
| 
 | ||||
|   return ` | ||||
|     (() => { | ||||
|       ${predicateQueryHandlerDef} | ||||
|       const predicateQueryHandler = ${predicateQueryHandler}; | ||||
|       const checkWaitForOptions = ${checkWaitForOptions}; | ||||
|       return (${predicate})(...args) | ||||
|     })() `;
 | ||||
|  | @ -496,104 +486,3 @@ export async function getReadableFromProtocolStream( | |||
|     }, | ||||
|   }); | ||||
| } | ||||
| 
 | ||||
| /** | ||||
|  * @internal | ||||
|  */ | ||||
| export interface ErrorLike extends Error { | ||||
|   name: string; | ||||
|   message: string; | ||||
| } | ||||
| 
 | ||||
| /** | ||||
|  * @internal | ||||
|  */ | ||||
| export function isErrorLike(obj: unknown): obj is ErrorLike { | ||||
|   return ( | ||||
|     typeof obj === 'object' && obj !== null && 'name' in obj && 'message' in obj | ||||
|   ); | ||||
| } | ||||
| 
 | ||||
| /** | ||||
|  * @internal | ||||
|  */ | ||||
| export function isErrnoException(obj: unknown): obj is NodeJS.ErrnoException { | ||||
|   return ( | ||||
|     isErrorLike(obj) && | ||||
|     ('errno' in obj || 'code' in obj || 'path' in obj || 'syscall' in obj) | ||||
|   ); | ||||
| } | ||||
| 
 | ||||
| /** | ||||
|  * @internal | ||||
|  */ | ||||
| export interface DeferredPromise<T> extends Promise<T> { | ||||
|   resolved: () => boolean; | ||||
|   resolve: (_: T) => void; | ||||
|   reject: (_: Error) => void; | ||||
| } | ||||
| 
 | ||||
| /** | ||||
|  * Creates an returns a promise along with the resolve/reject functions. | ||||
|  * | ||||
|  * If the promise has not been resolved/rejected withing the `timeout` period, | ||||
|  * the promise gets rejected with a timeout error. | ||||
|  * | ||||
|  * @internal | ||||
|  */ | ||||
| export function createDeferredPromiseWithTimer<T>( | ||||
|   timeoutMessage: string, | ||||
|   timeout = 5000 | ||||
| ): DeferredPromise<T> { | ||||
|   let isResolved = false; | ||||
|   let resolver = (_: T): void => {}; | ||||
|   let rejector = (_: Error) => {}; | ||||
|   const taskPromise = new Promise<T>((resolve, reject) => { | ||||
|     resolver = resolve; | ||||
|     rejector = reject; | ||||
|   }); | ||||
|   const timeoutId = setTimeout(() => { | ||||
|     rejector(new TimeoutError(timeoutMessage)); | ||||
|   }, timeout); | ||||
|   return Object.assign(taskPromise, { | ||||
|     resolved: () => { | ||||
|       return isResolved; | ||||
|     }, | ||||
|     resolve: (value: T) => { | ||||
|       clearTimeout(timeoutId); | ||||
|       isResolved = true; | ||||
|       resolver(value); | ||||
|     }, | ||||
|     reject: (err: Error) => { | ||||
|       clearTimeout(timeoutId); | ||||
|       rejector(err); | ||||
|     }, | ||||
|   }); | ||||
| } | ||||
| 
 | ||||
| /** | ||||
|  * Creates an returns a promise along with the resolve/reject functions. | ||||
|  * | ||||
|  * @internal | ||||
|  */ | ||||
| export function createDeferredPromise<T>(): DeferredPromise<T> { | ||||
|   let isResolved = false; | ||||
|   let resolver = (_: T): void => {}; | ||||
|   let rejector = (_: Error) => {}; | ||||
|   const taskPromise = new Promise<T>((resolve, reject) => { | ||||
|     resolver = resolve; | ||||
|     rejector = reject; | ||||
|   }); | ||||
|   return Object.assign(taskPromise, { | ||||
|     resolved: () => { | ||||
|       return isResolved; | ||||
|     }, | ||||
|     resolve: (value: T) => { | ||||
|       isResolved = true; | ||||
|       resolver(value); | ||||
|     }, | ||||
|     reject: (err: Error) => { | ||||
|       rejector(err); | ||||
|     }, | ||||
|   }); | ||||
| } | ||||
|  |  | |||
|  | @ -18,3 +18,12 @@ | |||
|  * @internal | ||||
|  */ | ||||
| export const isNode = !!(typeof process !== 'undefined' && process.version); | ||||
| 
 | ||||
| /** | ||||
|  * @internal | ||||
|  */ | ||||
| export const DEFERRED_PROMISE_DEBUG_TIMEOUT = | ||||
|   typeof process !== 'undefined' && | ||||
|   typeof process.env['PUPPETEER_DEFERRED_PROMISE_DEBUG_TIMEOUT'] !== 'undefined' | ||||
|     ? Number(process.env['PUPPETEER_DEFERRED_PROMISE_DEBUG_TIMEOUT']) | ||||
|     : -1; | ||||
|  |  | |||
|  | @ -1,3 +0,0 @@ | |||
| # Generated Artifacts | ||||
| 
 | ||||
| **Do not edit manually edit any TypeScript files in this folder** All TS files are generated from their respectively named template file (ext. `tmpl`) in the `templates` directory. Edit them there is needed. | ||||
|  | @ -1,4 +1,4 @@ | |||
| /** | ||||
|  * @internal | ||||
|  */ | ||||
| export const packageVersion = '16.1.1'; | ||||
| export const packageVersion = '17.1.2'; | ||||
|  |  | |||
|  | @ -14,18 +14,17 @@ | |||
|  * limitations under the License. | ||||
|  */ | ||||
| 
 | ||||
| import {sync} from 'pkg-dir'; | ||||
| import {Product} from './common/Product.js'; | ||||
| import {rootDirname} from './constants.js'; | ||||
| import {PuppeteerNode} from './node/Puppeteer.js'; | ||||
| import {PUPPETEER_REVISIONS} from './revisions.js'; | ||||
| import {getPackageDirectory} from './util/getPackageDirectory.js'; | ||||
| 
 | ||||
| /** | ||||
|  * @internal | ||||
|  */ | ||||
| export const initializePuppeteer = (packageName: string): PuppeteerNode => { | ||||
|   const isPuppeteerCore = packageName === 'puppeteer-core'; | ||||
|   const puppeteerRootDirectory = sync(rootDirname); | ||||
|   let preferredRevision = PUPPETEER_REVISIONS.chromium; | ||||
|   // puppeteer-core ignores environment variables
 | ||||
|   const productName = !isPuppeteerCore | ||||
|  | @ -39,7 +38,7 @@ export const initializePuppeteer = (packageName: string): PuppeteerNode => { | |||
|   } | ||||
| 
 | ||||
|   return new PuppeteerNode({ | ||||
|     projectRoot: puppeteerRootDirectory, | ||||
|     projectRoot: isPuppeteerCore ? undefined : getPackageDirectory(rootDirname), | ||||
|     preferredRevision, | ||||
|     isPuppeteerCore, | ||||
|     productName, | ||||
|  |  | |||
							
								
								
									
										156
									
								
								remote/test/puppeteer/src/injected/Poller.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										156
									
								
								remote/test/puppeteer/src/injected/Poller.ts
									
									
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,156 @@ | |||
| import { | ||||
|   createDeferredPromise, | ||||
|   DeferredPromise, | ||||
| } from '../util/DeferredPromise.js'; | ||||
| import {assert} from '../util/assert.js'; | ||||
| 
 | ||||
| interface Poller<T> { | ||||
|   start(): Promise<T>; | ||||
|   stop(): Promise<void>; | ||||
|   result(): Promise<T>; | ||||
| } | ||||
| 
 | ||||
| export class MutationPoller<T> implements Poller<T> { | ||||
|   #fn: () => Promise<T>; | ||||
| 
 | ||||
|   #root: Node; | ||||
| 
 | ||||
|   #observer?: MutationObserver; | ||||
|   #promise?: DeferredPromise<T>; | ||||
|   constructor(fn: () => Promise<T>, root: Node) { | ||||
|     this.#fn = fn; | ||||
|     this.#root = root; | ||||
|   } | ||||
| 
 | ||||
|   async start(): Promise<T> { | ||||
|     const promise = (this.#promise = createDeferredPromise<T>()); | ||||
|     const result = await this.#fn(); | ||||
|     if (result) { | ||||
|       promise.resolve(result); | ||||
|       return result; | ||||
|     } | ||||
| 
 | ||||
|     this.#observer = new MutationObserver(async () => { | ||||
|       const result = await this.#fn(); | ||||
|       if (!result) { | ||||
|         return; | ||||
|       } | ||||
|       promise.resolve(result); | ||||
|       await this.stop(); | ||||
|     }); | ||||
|     this.#observer.observe(this.#root, { | ||||
|       childList: true, | ||||
|       subtree: true, | ||||
|       attributes: true, | ||||
|     }); | ||||
| 
 | ||||
|     return this.#promise; | ||||
|   } | ||||
| 
 | ||||
|   async stop(): Promise<void> { | ||||
|     assert(this.#promise, 'Polling never started.'); | ||||
|     if (!this.#promise.finished()) { | ||||
|       this.#promise.reject(new Error('Polling stopped')); | ||||
|     } | ||||
|     if (this.#observer) { | ||||
|       this.#observer.disconnect(); | ||||
|     } | ||||
|   } | ||||
| 
 | ||||
|   result(): Promise<T> { | ||||
|     assert(this.#promise, 'Polling never started.'); | ||||
|     return this.#promise; | ||||
|   } | ||||
| } | ||||
| 
 | ||||
| export class RAFPoller<T> implements Poller<T> { | ||||
|   #fn: () => Promise<T>; | ||||
|   #promise?: DeferredPromise<T>; | ||||
|   constructor(fn: () => Promise<T>) { | ||||
|     this.#fn = fn; | ||||
|   } | ||||
| 
 | ||||
|   async start(): Promise<T> { | ||||
|     const promise = (this.#promise = createDeferredPromise<T>()); | ||||
|     const result = await this.#fn(); | ||||
|     if (result) { | ||||
|       promise.resolve(result); | ||||
|       return result; | ||||
|     } | ||||
| 
 | ||||
|     const poll = async () => { | ||||
|       if (promise.finished()) { | ||||
|         return; | ||||
|       } | ||||
|       const result = await this.#fn(); | ||||
|       if (!result) { | ||||
|         window.requestAnimationFrame(poll); | ||||
|         return; | ||||
|       } | ||||
|       promise.resolve(result); | ||||
|       await this.stop(); | ||||
|     }; | ||||
|     window.requestAnimationFrame(poll); | ||||
| 
 | ||||
|     return this.#promise; | ||||
|   } | ||||
| 
 | ||||
|   async stop(): Promise<void> { | ||||
|     assert(this.#promise, 'Polling never started.'); | ||||
|     if (!this.#promise.finished()) { | ||||
|       this.#promise.reject(new Error('Polling stopped')); | ||||
|     } | ||||
|   } | ||||
| 
 | ||||
|   result(): Promise<T> { | ||||
|     assert(this.#promise, 'Polling never started.'); | ||||
|     return this.#promise; | ||||
|   } | ||||
| } | ||||
| 
 | ||||
| export class IntervalPoller<T> implements Poller<T> { | ||||
|   #fn: () => Promise<T>; | ||||
|   #ms: number; | ||||
| 
 | ||||
|   #interval?: NodeJS.Timer; | ||||
|   #promise?: DeferredPromise<T>; | ||||
|   constructor(fn: () => Promise<T>, ms: number) { | ||||
|     this.#fn = fn; | ||||
|     this.#ms = ms; | ||||
|   } | ||||
| 
 | ||||
|   async start(): Promise<T> { | ||||
|     const promise = (this.#promise = createDeferredPromise<T>()); | ||||
|     const result = await this.#fn(); | ||||
|     if (result) { | ||||
|       promise.resolve(result); | ||||
|       return result; | ||||
|     } | ||||
| 
 | ||||
|     this.#interval = setInterval(async () => { | ||||
|       const result = await this.#fn(); | ||||
|       if (!result) { | ||||
|         return; | ||||
|       } | ||||
|       promise.resolve(result); | ||||
|       await this.stop(); | ||||
|     }, this.#ms); | ||||
| 
 | ||||
|     return this.#promise; | ||||
|   } | ||||
| 
 | ||||
|   async stop(): Promise<void> { | ||||
|     assert(this.#promise, 'Polling never started.'); | ||||
|     if (!this.#promise.finished()) { | ||||
|       this.#promise.reject(new Error('Polling stopped')); | ||||
|     } | ||||
|     if (this.#interval) { | ||||
|       clearInterval(this.#interval); | ||||
|     } | ||||
|   } | ||||
| 
 | ||||
|   result(): Promise<T> { | ||||
|     assert(this.#promise, 'Polling never started.'); | ||||
|     return this.#promise; | ||||
|   } | ||||
| } | ||||
							
								
								
									
										5
									
								
								remote/test/puppeteer/src/injected/README.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										5
									
								
								remote/test/puppeteer/src/injected/README.md
									
									
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,5 @@ | |||
| # Injected | ||||
| 
 | ||||
| This folder contains code that is injected into every Puppeteer execution context. Each file is transpiled using esbuild into a script in `src/generated` which is then imported into server code. | ||||
| 
 | ||||
| See `utils/generate_injected.ts` for more information. | ||||
							
								
								
									
										14
									
								
								remote/test/puppeteer/src/injected/injected.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										14
									
								
								remote/test/puppeteer/src/injected/injected.ts
									
									
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,14 @@ | |||
| import {createDeferredPromise} from '../util/DeferredPromise.js'; | ||||
| import * as Poller from './Poller.js'; | ||||
| import * as util from './util.js'; | ||||
| 
 | ||||
| Object.assign( | ||||
|   self, | ||||
|   Object.freeze({ | ||||
|     InjectedUtil: { | ||||
|       ...Poller, | ||||
|       ...util, | ||||
|       createDeferredPromise, | ||||
|     }, | ||||
|   }) | ||||
| ); | ||||
							
								
								
									
										18
									
								
								remote/test/puppeteer/src/injected/util.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										18
									
								
								remote/test/puppeteer/src/injected/util.ts
									
									
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,18 @@ | |||
| const createdFunctions = new Map<string, (...args: unknown[]) => unknown>(); | ||||
| 
 | ||||
| /** | ||||
|  * Creates a function from a string. | ||||
|  */ | ||||
| export const createFunction = ( | ||||
|   functionValue: string | ||||
| ): ((...args: unknown[]) => unknown) => { | ||||
|   let fn = createdFunctions.get(functionValue); | ||||
|   if (fn) { | ||||
|     return fn; | ||||
|   } | ||||
|   fn = new Function(`return ${functionValue}`)() as ( | ||||
|     ...args: unknown[] | ||||
|   ) => unknown; | ||||
|   createdFunctions.set(functionValue, fn); | ||||
|   return fn; | ||||
| }; | ||||
|  | @ -33,7 +33,7 @@ import createHttpsProxyAgent, { | |||
|   HttpsProxyAgentOptions, | ||||
| } from 'https-proxy-agent'; | ||||
| import {getProxyForUrl} from 'proxy-from-env'; | ||||
| import {assert} from '../common/assert.js'; | ||||
| import {assert} from '../util/assert.js'; | ||||
| 
 | ||||
| import tar from 'tar-fs'; | ||||
| import bzip from 'unbzip2-stream'; | ||||
|  |  | |||
|  | @ -20,18 +20,17 @@ import * as path from 'path'; | |||
| import * as readline from 'readline'; | ||||
| import removeFolder from 'rimraf'; | ||||
| import {promisify} from 'util'; | ||||
| import {assert} from '../common/assert.js'; | ||||
| import {assert} from '../util/assert.js'; | ||||
| import {Connection} from '../common/Connection.js'; | ||||
| import {debug} from '../common/Debug.js'; | ||||
| import {TimeoutError} from '../common/Errors.js'; | ||||
| import { | ||||
|   debugError, | ||||
|   addEventListener, | ||||
|   isErrnoException, | ||||
|   isErrorLike, | ||||
|   PuppeteerEventListener, | ||||
|   removeEventListeners, | ||||
| } from '../common/util.js'; | ||||
| import {isErrnoException, isErrorLike} from '../util/ErrorLike.js'; | ||||
| import {Product} from '../common/Product.js'; | ||||
| import {NodeWebSocketTransport as WebSocketTransport} from '../node/NodeWebSocketTransport.js'; | ||||
| import {LaunchOptions} from './LaunchOptions.js'; | ||||
|  |  | |||
|  | @ -1,6 +1,6 @@ | |||
| import fs from 'fs'; | ||||
| import path from 'path'; | ||||
| import {assert} from '../common/assert.js'; | ||||
| import {assert} from '../util/assert.js'; | ||||
| import {Browser} from '../common/Browser.js'; | ||||
| import {Product} from '../common/Product.js'; | ||||
| import {BrowserRunner} from './BrowserRunner.js'; | ||||
|  |  | |||
|  | @ -1,7 +1,7 @@ | |||
| import fs from 'fs'; | ||||
| import os from 'os'; | ||||
| import path from 'path'; | ||||
| import {assert} from '../common/assert.js'; | ||||
| import {assert} from '../util/assert.js'; | ||||
| import {Browser} from '../common/Browser.js'; | ||||
| import {Product} from '../common/Product.js'; | ||||
| import {BrowserFetcher} from './BrowserFetcher.js'; | ||||
|  |  | |||
|  | @ -16,27 +16,12 @@ | |||
| import NodeWebSocket from 'ws'; | ||||
| import {ConnectionTransport} from '../common/ConnectionTransport.js'; | ||||
| import {packageVersion} from '../generated/version.js'; | ||||
| import {promises as dns} from 'dns'; | ||||
| import {URL} from 'url'; | ||||
| 
 | ||||
| /** | ||||
|  * @internal | ||||
|  */ | ||||
| export class NodeWebSocketTransport implements ConnectionTransport { | ||||
|   static async create(urlString: string): Promise<NodeWebSocketTransport> { | ||||
|     // TODO(jrandolf): Starting in Node 17, IPv6 is favoured over IPv4 due to a change
 | ||||
|     // in a default option:
 | ||||
|     // - https://github.com/nodejs/node/issues/40537,
 | ||||
|     // Due to this, for Firefox, we must parse and resolve the `localhost` hostname
 | ||||
|     // manually with the previous behavior according to:
 | ||||
|     // - https://nodejs.org/api/dns.html#dnslookuphostname-options-callback
 | ||||
|     // because of https://bugzilla.mozilla.org/show_bug.cgi?id=1769994.
 | ||||
|     const url = new URL(urlString); | ||||
|     if (url.hostname === 'localhost') { | ||||
|       const {address} = await dns.lookup(url.hostname, {verbatim: false}); | ||||
|       url.hostname = address; | ||||
|     } | ||||
| 
 | ||||
|   static create(url: string): Promise<NodeWebSocketTransport> { | ||||
|     return new Promise((resolve, reject) => { | ||||
|       const ws = new NodeWebSocket(url, [], { | ||||
|         followRedirects: true, | ||||
|  |  | |||
|  | @ -13,7 +13,7 @@ | |||
|  * See the License for the specific language governing permissions and | ||||
|  * limitations under the License. | ||||
|  */ | ||||
| import {assert} from '../common/assert.js'; | ||||
| import {assert} from '../util/assert.js'; | ||||
| import {ConnectionTransport} from '../common/ConnectionTransport.js'; | ||||
| import { | ||||
|   addEventListener, | ||||
|  |  | |||
|  | @ -18,6 +18,6 @@ | |||
|  * @internal | ||||
|  */ | ||||
| export const PUPPETEER_REVISIONS = Object.freeze({ | ||||
|   chromium: '1022525', | ||||
|   chromium: '1036745', | ||||
|   firefox: 'latest', | ||||
| }); | ||||
|  |  | |||
							
								
								
									
										3
									
								
								remote/test/puppeteer/src/templates/README.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										3
									
								
								remote/test/puppeteer/src/templates/README.md
									
									
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,3 @@ | |||
| # Templated Artifacts | ||||
| 
 | ||||
| These files are generated as TypeScript files in the `src/generated` folder. | ||||
							
								
								
									
										10
									
								
								remote/test/puppeteer/src/templates/injected.ts.tmpl
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										10
									
								
								remote/test/puppeteer/src/templates/injected.ts.tmpl
									
									
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,10 @@ | |||
| import {createDeferredPromise} from '../util/DeferredPromise.js'; | ||||
| 
 | ||||
| declare global { | ||||
|   const InjectedUtil: { | ||||
|     createDeferredPromise: typeof createDeferredPromise; | ||||
|   }; | ||||
| } | ||||
| 
 | ||||
| /** @internal */ | ||||
| export const source = SOURCE_CODE; | ||||
|  | @ -8,5 +8,6 @@ | |||
|   "references": [ | ||||
|     {"path": "../vendor/tsconfig.cjs.json"}, | ||||
|     {"path": "../compat/cjs/tsconfig.json"} | ||||
|   ] | ||||
|   ], | ||||
|   "exclude": ["injected/injected.ts"] | ||||
| } | ||||
|  |  | |||
|  | @ -8,5 +8,6 @@ | |||
|   "references": [ | ||||
|     {"path": "../vendor/tsconfig.esm.json"}, | ||||
|     {"path": "../compat/esm/tsconfig.json"} | ||||
|   ] | ||||
|   ], | ||||
|   "exclude": ["injected/injected.ts"] | ||||
| } | ||||
|  |  | |||
|  | @ -1,13 +1,5 @@ | |||
| // AUTOGENERATED - Use `utils/export_all.js` to regenerate.
 | ||||
| // AUTOGENERATED - Use `npm run generate:sources` to regenerate.
 | ||||
| 
 | ||||
| export * from './compat.d.js'; | ||||
| export * from './constants.js'; | ||||
| export * from './environment.js'; | ||||
| export * from './initializePuppeteer.js'; | ||||
| export * from './puppeteer.js'; | ||||
| export * from './revisions.js'; | ||||
| 
 | ||||
| // Exports from `common`
 | ||||
| export * from './common/Accessibility.js'; | ||||
| export * from './common/AriaQueryHandler.js'; | ||||
| export * from './common/Browser.js'; | ||||
|  | @ -26,8 +18,10 @@ export * from './common/EmulationManager.js'; | |||
| export * from './common/Errors.js'; | ||||
| export * from './common/EventEmitter.js'; | ||||
| export * from './common/ExecutionContext.js'; | ||||
| export * from './common/fetch.js'; | ||||
| export * from './common/FileChooser.js'; | ||||
| export * from './common/FirefoxTargetManager.js'; | ||||
| export * from './common/Frame.js'; | ||||
| export * from './common/FrameManager.js'; | ||||
| export * from './common/HTTPRequest.js'; | ||||
| export * from './common/HTTPResponse.js'; | ||||
|  | @ -38,8 +32,8 @@ export * from './common/LifecycleWatcher.js'; | |||
| export * from './common/NetworkConditions.js'; | ||||
| export * from './common/NetworkEventManager.js'; | ||||
| export * from './common/NetworkManager.js'; | ||||
| export * from './common/PDFOptions.js'; | ||||
| export * from './common/Page.js'; | ||||
| export * from './common/PDFOptions.js'; | ||||
| export * from './common/Product.js'; | ||||
| export * from './common/Puppeteer.js'; | ||||
| export * from './common/PuppeteerViewport.js'; | ||||
|  | @ -50,25 +44,31 @@ export * from './common/TargetManager.js'; | |||
| export * from './common/TaskQueue.js'; | ||||
| export * from './common/TimeoutSettings.js'; | ||||
| export * from './common/Tracing.js'; | ||||
| export * from './common/USKeyboardLayout.js'; | ||||
| export * from './common/WebWorker.js'; | ||||
| export * from './common/assert.js'; | ||||
| export * from './common/fetch.js'; | ||||
| export * from './common/types.js'; | ||||
| export * from './common/USKeyboardLayout.js'; | ||||
| export * from './common/util.js'; | ||||
| 
 | ||||
| // Exports from `node`
 | ||||
| export * from './common/WebWorker.js'; | ||||
| export * from './compat.d.js'; | ||||
| export * from './constants.js'; | ||||
| export * from './environment.js'; | ||||
| export * from './generated/injected.js'; | ||||
| export * from './generated/version.js'; | ||||
| export * from './initializePuppeteer.js'; | ||||
| export * from './node/BrowserFetcher.js'; | ||||
| export * from './node/BrowserRunner.js'; | ||||
| export * from './node/ChromeLauncher.js'; | ||||
| export * from './node/FirefoxLauncher.js'; | ||||
| export * from './node/install.js'; | ||||
| export * from './node/LaunchOptions.js'; | ||||
| export * from './node/NodeWebSocketTransport.js'; | ||||
| export * from './node/PipeTransport.js'; | ||||
| export * from './node/ProductLauncher.js'; | ||||
| export * from './node/Puppeteer.js'; | ||||
| export * from './node/install.js'; | ||||
| export * from './node/util.js'; | ||||
| 
 | ||||
| // Exports from `generated`
 | ||||
| export * from './generated/version.js'; | ||||
| export * from './puppeteer.js'; | ||||
| export * from './revisions.js'; | ||||
| export * from './util/assert.js'; | ||||
| export * from './util/DebuggableDeferredPromise.js'; | ||||
| export * from './util/DeferredPromise.js'; | ||||
| export * from './util/ErrorLike.js'; | ||||
| export * from './util/getPackageDirectory.js'; | ||||
|  |  | |||
							
								
								
									
										20
									
								
								remote/test/puppeteer/src/util/DebuggableDeferredPromise.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										20
									
								
								remote/test/puppeteer/src/util/DebuggableDeferredPromise.ts
									
									
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,20 @@ | |||
| import {DEFERRED_PROMISE_DEBUG_TIMEOUT} from '../environment.js'; | ||||
| import {DeferredPromise, createDeferredPromise} from './DeferredPromise.js'; | ||||
| 
 | ||||
| /** | ||||
|  * Creates and returns a deferred promise using DEFERRED_PROMISE_DEBUG_TIMEOUT | ||||
|  * if it's specified or a normal deferred promise otherwise. | ||||
|  * | ||||
|  * @internal | ||||
|  */ | ||||
| export function createDebuggableDeferredPromise<T>( | ||||
|   message: string | ||||
| ): DeferredPromise<T> { | ||||
|   if (DEFERRED_PROMISE_DEBUG_TIMEOUT > 0) { | ||||
|     return createDeferredPromise({ | ||||
|       message, | ||||
|       timeout: DEFERRED_PROMISE_DEBUG_TIMEOUT, | ||||
|     }); | ||||
|   } | ||||
|   return createDeferredPromise(); | ||||
| } | ||||
							
								
								
									
										68
									
								
								remote/test/puppeteer/src/util/DeferredPromise.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										68
									
								
								remote/test/puppeteer/src/util/DeferredPromise.ts
									
									
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,68 @@ | |||
| import {TimeoutError} from '../common/Errors.js'; | ||||
| 
 | ||||
| /** | ||||
|  * @internal | ||||
|  */ | ||||
| export interface DeferredPromise<T> extends Promise<T> { | ||||
|   finished: () => boolean; | ||||
|   resolved: () => boolean; | ||||
|   resolve: (_: T) => void; | ||||
|   reject: (_: Error) => void; | ||||
| } | ||||
| 
 | ||||
| /** | ||||
|  * @internal | ||||
|  */ | ||||
| export interface DeferredPromiseOptions { | ||||
|   message: string; | ||||
|   timeout: number; | ||||
| } | ||||
| 
 | ||||
| /** | ||||
|  * Creates and returns a promise along with the resolve/reject functions. | ||||
|  * | ||||
|  * If the promise has not been resolved/rejected within the `timeout` period, | ||||
|  * the promise gets rejected with a timeout error. `timeout` has to be greater than 0 or | ||||
|  * it is ignored. | ||||
|  * | ||||
|  * @internal | ||||
|  */ | ||||
| export function createDeferredPromise<T>( | ||||
|   opts?: DeferredPromiseOptions | ||||
| ): DeferredPromise<T> { | ||||
|   let isResolved = false; | ||||
|   let isRejected = false; | ||||
|   let resolver = (_: T): void => {}; | ||||
|   let rejector = (_: Error) => {}; | ||||
|   const taskPromise = new Promise<T>((resolve, reject) => { | ||||
|     resolver = resolve; | ||||
|     rejector = reject; | ||||
|   }); | ||||
|   const timeoutId = | ||||
|     opts && opts.timeout > 0 | ||||
|       ? setTimeout(() => { | ||||
|           isRejected = true; | ||||
|           rejector(new TimeoutError(opts.message)); | ||||
|         }, opts.timeout) | ||||
|       : undefined; | ||||
|   return Object.assign(taskPromise, { | ||||
|     resolved: () => { | ||||
|       return isResolved; | ||||
|     }, | ||||
|     finished: () => { | ||||
|       return isResolved || isRejected; | ||||
|     }, | ||||
|     resolve: (value: T) => { | ||||
|       if (timeoutId) { | ||||
|         clearTimeout(timeoutId); | ||||
|       } | ||||
|       isResolved = true; | ||||
|       resolver(value); | ||||
|     }, | ||||
|     reject: (err: Error) => { | ||||
|       clearTimeout(timeoutId); | ||||
|       isRejected = true; | ||||
|       rejector(err); | ||||
|     }, | ||||
|   }); | ||||
| } | ||||
							
								
								
									
										27
									
								
								remote/test/puppeteer/src/util/ErrorLike.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										27
									
								
								remote/test/puppeteer/src/util/ErrorLike.ts
									
									
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,27 @@ | |||
| /** | ||||
|  * @internal | ||||
|  */ | ||||
| 
 | ||||
| export interface ErrorLike extends Error { | ||||
|   name: string; | ||||
|   message: string; | ||||
| } | ||||
| /** | ||||
|  * @internal | ||||
|  */ | ||||
| 
 | ||||
| export function isErrorLike(obj: unknown): obj is ErrorLike { | ||||
|   return ( | ||||
|     typeof obj === 'object' && obj !== null && 'name' in obj && 'message' in obj | ||||
|   ); | ||||
| } | ||||
| /** | ||||
|  * @internal | ||||
|  */ | ||||
| 
 | ||||
| export function isErrnoException(obj: unknown): obj is NodeJS.ErrnoException { | ||||
|   return ( | ||||
|     isErrorLike(obj) && | ||||
|     ('errno' in obj || 'code' in obj || 'path' in obj || 'syscall' in obj) | ||||
|   ); | ||||
| } | ||||
							
								
								
									
										18
									
								
								remote/test/puppeteer/src/util/getPackageDirectory.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										18
									
								
								remote/test/puppeteer/src/util/getPackageDirectory.ts
									
									
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,18 @@ | |||
| import {existsSync} from 'fs'; | ||||
| import {dirname, join, parse} from 'path'; | ||||
| 
 | ||||
| /** | ||||
|  * @internal | ||||
|  */ | ||||
| export const getPackageDirectory = (from: string): string => { | ||||
|   let found = existsSync(join(from, 'package.json')); | ||||
|   const root = parse(from).root; | ||||
|   while (!found) { | ||||
|     if (from === root) { | ||||
|       throw new Error('Cannot find package directory'); | ||||
|     } | ||||
|     from = dirname(from); | ||||
|     found = existsSync(join(from, 'package.json')); | ||||
|   } | ||||
|   return from; | ||||
| }; | ||||
|  | @ -6,6 +6,7 @@ function foo() { | |||
|     console.log(2); | ||||
|   let x = 1 > 2 ? 'foo' : 'bar'; | ||||
|   let y = 1 < 2 ? 'foo' : 'bar'; | ||||
|   let p = {a:1 > 2?function(){console.log('unused');}:function(){console.log('unused');}}; | ||||
|   let z = () => {}; | ||||
|   let q = () => {}; | ||||
|   q(); | ||||
|  |  | |||
|  | @ -0,0 +1 @@ | |||
| // empty
 | ||||
|  | @ -0,0 +1,9 @@ | |||
| { | ||||
|   "name": "Simple extension", | ||||
|   "version": "0.1", | ||||
|   "background": { | ||||
|     "service_worker": "background.js" | ||||
|   }, | ||||
|   "permissions": ["background", "activeTab"], | ||||
|   "manifest_version": 3 | ||||
| } | ||||
|  | @ -16,13 +16,21 @@ | |||
|       }, | ||||
|       { | ||||
|         "start": 148, | ||||
|         "end": 160 | ||||
|         "end": 168 | ||||
|       }, | ||||
|       { | ||||
|         "start": 168, | ||||
|         "end": 207 | ||||
|         "start": 203, | ||||
|         "end": 204 | ||||
|       }, | ||||
|       { | ||||
|         "start": 238, | ||||
|         "end": 251 | ||||
|       }, | ||||
|       { | ||||
|         "start": 259, | ||||
|         "end": 298 | ||||
|       } | ||||
|     ], | ||||
|     "text": "\nfunction foo() {\n  if (1 > 2)\n    console.log(1);\n  if (1 < 2)\n    console.log(2);\n  let x = 1 > 2 ? 'foo' : 'bar';\n  let y = 1 < 2 ? 'foo' : 'bar';\n  let z = () => {};\n  let q = () => {};\n  q();\n}\n\nfoo();\n" | ||||
|     "text": "\nfunction foo() {\n  if (1 > 2)\n    console.log(1);\n  if (1 < 2)\n    console.log(2);\n  let x = 1 > 2 ? 'foo' : 'bar';\n  let y = 1 < 2 ? 'foo' : 'bar';\n  let p = {a:1 > 2?function(){console.log('unused');}:function(){console.log('unused');}};\n  let z = () => {};\n  let q = () => {};\n  q();\n}\n\nfoo();\n" | ||||
|   } | ||||
| ] | ||||
										
											Binary file not shown.
										
									
								
							| After Width: | Height: | Size: 8.3 KiB | 
										
											Binary file not shown.
										
									
								
							| Before Width: | Height: | Size: 79 KiB After Width: | Height: | Size: 79 KiB | 
										
											Binary file not shown.
										
									
								
							| After Width: | Height: | Size: 8.3 KiB | 
|  | @ -22,7 +22,7 @@ import { | |||
|   setupTestPageAndContextHooks, | ||||
|   describeChromeOnly, | ||||
| } from './mocha-utils.js'; | ||||
| import {isErrorLike} from '../../lib/cjs/puppeteer/common/util.js'; | ||||
| import {isErrorLike} from '../../lib/cjs/puppeteer/util/ErrorLike.js'; | ||||
| 
 | ||||
| describeChromeOnly('Target.createCDPSession', function () { | ||||
|   setupTestBrowserHooks(); | ||||
|  |  | |||
|  | @ -23,7 +23,7 @@ import { | |||
| } from '../../lib/cjs/puppeteer/common/NetworkManager.js'; | ||||
| import {HTTPRequest} from '../../lib/cjs/puppeteer/common/HTTPRequest.js'; | ||||
| import {EventEmitter} from '../../lib/cjs/puppeteer/common/EventEmitter.js'; | ||||
| import {Frame} from '../../lib/cjs/puppeteer/common/FrameManager.js'; | ||||
| import {Frame} from '../../lib/cjs/puppeteer/common/Frame.js'; | ||||
| import {HTTPResponse} from '../../lib/cjs/puppeteer/common/HTTPResponse.js'; | ||||
| 
 | ||||
| class MockCDPSession extends EventEmitter { | ||||
|  |  | |||
|  | @ -334,7 +334,7 @@ describeChromeOnly('AriaQueryHandler', () => { | |||
|       await otherFrame!.evaluate(addElement, 'button'); | ||||
|       await page.evaluate(addElement, 'button'); | ||||
|       const elementHandle = await watchdog; | ||||
|       expect(elementHandle!.executionContext().frame()).toBe(page.mainFrame()); | ||||
|       expect(elementHandle!.frame).toBe(page.mainFrame()); | ||||
|     }); | ||||
| 
 | ||||
|     it('should run in specified frame', async () => { | ||||
|  | @ -350,7 +350,7 @@ describeChromeOnly('AriaQueryHandler', () => { | |||
|       await frame1!.evaluate(addElement, 'button'); | ||||
|       await frame2!.evaluate(addElement, 'button'); | ||||
|       const elementHandle = await waitForSelectorPromise; | ||||
|       expect(elementHandle!.executionContext().frame()).toBe(frame2); | ||||
|       expect(elementHandle!.frame).toBe(frame2); | ||||
|     }); | ||||
| 
 | ||||
|     it('should throw when frame is detached', async () => { | ||||
|  | @ -687,7 +687,7 @@ describeChromeOnly('AriaQueryHandler', () => { | |||
|       const {page} = getTestState(); | ||||
|       const found = await page.$$('aria/title'); | ||||
|       const ids = await getIds(found); | ||||
|       expect(ids).toEqual(['shown', 'hidden']); | ||||
|       expect(ids).toEqual(['shown']); | ||||
|     }); | ||||
|   }); | ||||
| }); | ||||
|  |  | |||
|  | @ -104,9 +104,11 @@ describe('Coverage specs', function () { | |||
|       const coverage = await page.coverage.stopJSCoverage(); | ||||
|       expect(coverage.length).toBe(1); | ||||
|       const entry = coverage[0]!; | ||||
|       expect(entry.ranges.length).toBe(1); | ||||
|       const range = entry.ranges[0]!; | ||||
|       expect(entry.text.substring(range.start, range.end)).toBe( | ||||
|       expect(entry.ranges.length).toBe(2); | ||||
|       const range1 = entry.ranges[0]!; | ||||
|       expect(entry.text.substring(range1.start, range1.end)).toBe('\n'); | ||||
|       const range2 = entry.ranges[1]!; | ||||
|       expect(entry.text.substring(range2.start, range2.end)).toBe( | ||||
|         `console.log('used!');` | ||||
|       ); | ||||
|     }); | ||||
|  |  | |||
|  | @ -16,7 +16,7 @@ | |||
| 
 | ||||
| import expect from 'expect'; | ||||
| import {CDPSession} from '../../lib/cjs/puppeteer/common/Connection.js'; | ||||
| import {Frame} from '../../lib/cjs/puppeteer/common/FrameManager.js'; | ||||
| import {Frame} from '../../lib/cjs/puppeteer/common/Frame.js'; | ||||
| import { | ||||
|   getTestState, | ||||
|   setupTestBrowserHooks, | ||||
|  | @ -41,8 +41,8 @@ describe('Frame specs', function () { | |||
|       expect(context1).toBeTruthy(); | ||||
|       expect(context2).toBeTruthy(); | ||||
|       expect(context1 !== context2).toBeTruthy(); | ||||
|       expect(context1.frame()).toBe(frame1); | ||||
|       expect(context2.frame()).toBe(frame2); | ||||
|       expect(context1._world?.frame()).toBe(frame1); | ||||
|       expect(context2._world?.frame()).toBe(frame2); | ||||
| 
 | ||||
|       await Promise.all([ | ||||
|         context1.evaluate(() => { | ||||
|  |  | |||
|  | @ -36,6 +36,13 @@ const mkdtempAsync = promisify(fs.mkdtemp); | |||
| const TMP_FOLDER = path.join(os.tmpdir(), 'pptr_tmp_folder-'); | ||||
| 
 | ||||
| const extensionPath = path.join(__dirname, '../assets', 'simple-extension'); | ||||
| const serviceWorkerExtensionPath = path.join( | ||||
|   __dirname, | ||||
|   '..', | ||||
|   'assets', | ||||
|   'serviceworkers', | ||||
|   'extension' | ||||
| ); | ||||
| 
 | ||||
| describeChromeOnly('headful tests', function () { | ||||
|   /* These tests fire up an actual browser so let's | ||||
|  | @ -120,7 +127,7 @@ describeChromeOnly('headful tests', function () { | |||
|       ); | ||||
|       const page = await browserWithExtension.newPage(); | ||||
|       const backgroundPageTarget = await browserWithExtension.waitForTarget( | ||||
|         (target: {type: () => string}) => { | ||||
|         target => { | ||||
|           return target.type() === 'background_page'; | ||||
|         } | ||||
|       ); | ||||
|  | @ -128,6 +135,26 @@ describeChromeOnly('headful tests', function () { | |||
|       await browserWithExtension.close(); | ||||
|       expect(backgroundPageTarget).toBeTruthy(); | ||||
|     }); | ||||
|     it('service_worker target type should be available', async () => { | ||||
|       const {puppeteer, defaultBrowserOptions} = getTestState(); | ||||
|       const browserWithExtension = await launchBrowser(puppeteer, { | ||||
|         ...defaultBrowserOptions, | ||||
|         headless: false, | ||||
|         args: [ | ||||
|           `--disable-extensions-except=${serviceWorkerExtensionPath}`, | ||||
|           `--load-extension=${serviceWorkerExtensionPath}`, | ||||
|         ], | ||||
|       }); | ||||
|       const page = await browserWithExtension.newPage(); | ||||
|       const serviceWorkerTarget = await browserWithExtension.waitForTarget( | ||||
|         target => { | ||||
|           return target.type() === 'service_worker'; | ||||
|         } | ||||
|       ); | ||||
|       await page.close(); | ||||
|       await browserWithExtension.close(); | ||||
|       expect(serviceWorkerTarget).toBeTruthy(); | ||||
|     }); | ||||
|     it('target.page() should return a background_page', async function () { | ||||
|       const {puppeteer} = getTestState(); | ||||
|       const browserWithExtension = await launchBrowser( | ||||
|  | @ -135,7 +162,7 @@ describeChromeOnly('headful tests', function () { | |||
|         extensionOptions | ||||
|       ); | ||||
|       const backgroundPageTarget = await browserWithExtension.waitForTarget( | ||||
|         (target: {type: () => string}) => { | ||||
|         target => { | ||||
|           return target.type() === 'background_page'; | ||||
|         } | ||||
|       ); | ||||
|  |  | |||
							
								
								
									
										40
									
								
								remote/test/puppeteer/test/src/injected.spec.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										40
									
								
								remote/test/puppeteer/test/src/injected.spec.ts
									
									
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,40 @@ | |||
| /** | ||||
|  * Copyright 2022 Google Inc. All rights reserved. | ||||
|  * | ||||
|  * 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. | ||||
|  */ | ||||
| 
 | ||||
| import expect from 'expect'; | ||||
| import '../../lib/cjs/puppeteer/generated/injected.js'; | ||||
| import {PUPPETEER_WORLD} from '../../lib/cjs/puppeteer/common/IsolatedWorld.js'; | ||||
| import { | ||||
|   getTestState, | ||||
|   setupTestBrowserHooks, | ||||
|   setupTestPageAndContextHooks, | ||||
| } from './mocha-utils.js'; | ||||
| 
 | ||||
| describe('InjectedUtil tests', function () { | ||||
|   setupTestBrowserHooks(); | ||||
|   setupTestPageAndContextHooks(); | ||||
| 
 | ||||
|   it('should work', async () => { | ||||
|     const {page} = getTestState(); | ||||
| 
 | ||||
|     const handle = await page | ||||
|       .mainFrame() | ||||
|       .worlds[PUPPETEER_WORLD].evaluate(() => { | ||||
|         return typeof InjectedUtil === 'object'; | ||||
|       }); | ||||
|     expect(handle).toBeTruthy(); | ||||
|   }); | ||||
| }); | ||||
|  | @ -375,9 +375,7 @@ describe('Launcher specs', function () { | |||
|         // This might throw. See https://github.com/puppeteer/puppeteer/issues/2778
 | ||||
|         await rmAsync(userDataDir).catch(() => {}); | ||||
|       }); | ||||
|       // This mysteriously fails on Windows on AppVeyor. See
 | ||||
|       // https://github.com/puppeteer/puppeteer/issues/4111
 | ||||
|       xit('userDataDir option should restore cookies', async () => { | ||||
|       it('userDataDir option should restore cookies', async () => { | ||||
|         const {server, puppeteer, defaultBrowserOptions} = getTestState(); | ||||
| 
 | ||||
|         const userDataDir = await mkdtempAsync(TMP_FOLDER); | ||||
|  |  | |||
|  | @ -26,7 +26,7 @@ import { | |||
|   BrowserContext, | ||||
| } from '../../lib/cjs/puppeteer/common/Browser.js'; | ||||
| import {Page} from '../../lib/cjs/puppeteer/common/Page.js'; | ||||
| import {isErrorLike} from '../../lib/cjs/puppeteer/common/util.js'; | ||||
| import {isErrorLike} from '../../lib/cjs/puppeteer/util/ErrorLike.js'; | ||||
| import { | ||||
|   PuppeteerLaunchOptions, | ||||
|   PuppeteerNode, | ||||
|  |  | |||
|  | @ -22,21 +22,21 @@ import { | |||
|   setupTestPageAndContextHooks, | ||||
| } from './mocha-utils.js'; | ||||
| import os from 'os'; | ||||
| import { ServerResponse } from 'http'; | ||||
| import { HTTPRequest } from '../../lib/cjs/puppeteer/common/HTTPRequest.js'; | ||||
| import {ServerResponse} from 'http'; | ||||
| import {HTTPRequest} from '../../lib/cjs/puppeteer/common/HTTPRequest.js'; | ||||
| 
 | ||||
| describe('navigation', function () { | ||||
|   setupTestBrowserHooks(); | ||||
|   setupTestPageAndContextHooks(); | ||||
|   describe('Page.goto', function () { | ||||
|     it('should work', async () => { | ||||
|       const { page, server } = getTestState(); | ||||
|       const {page, server} = getTestState(); | ||||
| 
 | ||||
|       await page.goto(server.EMPTY_PAGE); | ||||
|       expect(page.url()).toBe(server.EMPTY_PAGE); | ||||
|     }); | ||||
|     it('should work with anchor navigation', async () => { | ||||
|       const { page, server } = getTestState(); | ||||
|       const {page, server} = getTestState(); | ||||
| 
 | ||||
|       await page.goto(server.EMPTY_PAGE); | ||||
|       expect(page.url()).toBe(server.EMPTY_PAGE); | ||||
|  | @ -46,7 +46,7 @@ describe('navigation', function () { | |||
|       expect(page.url()).toBe(server.EMPTY_PAGE + '#bar'); | ||||
|     }); | ||||
|     it('should work with redirects', async () => { | ||||
|       const { page, server } = getTestState(); | ||||
|       const {page, server} = getTestState(); | ||||
| 
 | ||||
|       server.setRedirect('/redirect/1.html', '/redirect/2.html'); | ||||
|       server.setRedirect('/redirect/2.html', '/empty.html'); | ||||
|  | @ -54,19 +54,19 @@ describe('navigation', function () { | |||
|       expect(page.url()).toBe(server.EMPTY_PAGE); | ||||
|     }); | ||||
|     it('should navigate to about:blank', async () => { | ||||
|       const { page } = getTestState(); | ||||
|       const {page} = getTestState(); | ||||
| 
 | ||||
|       const response = await page.goto('about:blank'); | ||||
|       expect(response).toBe(null); | ||||
|     }); | ||||
|     it('should return response when page changes its URL after load', async () => { | ||||
|       const { page, server } = getTestState(); | ||||
|       const {page, server} = getTestState(); | ||||
| 
 | ||||
|       const response = (await page.goto(server.PREFIX + '/historyapi.html'))!; | ||||
|       expect(response.status()).toBe(200); | ||||
|     }); | ||||
|     it('should work with subframes return 204', async () => { | ||||
|       const { page, server } = getTestState(); | ||||
|       const {page, server} = getTestState(); | ||||
| 
 | ||||
|       server.setRoute('/frames/frame.html', (_req, res) => { | ||||
|         res.statusCode = 204; | ||||
|  | @ -75,20 +75,20 @@ describe('navigation', function () { | |||
|       let error!: Error; | ||||
|       await page | ||||
|         .goto(server.PREFIX + '/frames/one-frame.html') | ||||
|         .catch((error_) => { | ||||
|         .catch(error_ => { | ||||
|           return (error = error_); | ||||
|         }); | ||||
|       expect(error).toBeUndefined(); | ||||
|     }); | ||||
|     it('should fail when server returns 204', async () => { | ||||
|       const { page, server, isChrome } = getTestState(); | ||||
|       const {page, server, isChrome} = getTestState(); | ||||
| 
 | ||||
|       server.setRoute('/empty.html', (_req, res) => { | ||||
|         res.statusCode = 204; | ||||
|         res.end(); | ||||
|       }); | ||||
|       let error!: Error; | ||||
|       await page.goto(server.EMPTY_PAGE).catch((error_) => { | ||||
|       await page.goto(server.EMPTY_PAGE).catch(error_ => { | ||||
|         return (error = error_); | ||||
|       }); | ||||
|       expect(error).not.toBe(null); | ||||
|  | @ -99,7 +99,7 @@ describe('navigation', function () { | |||
|       } | ||||
|     }); | ||||
|     it('should navigate to empty page with domcontentloaded', async () => { | ||||
|       const { page, server } = getTestState(); | ||||
|       const {page, server} = getTestState(); | ||||
| 
 | ||||
|       const response = await page.goto(server.EMPTY_PAGE, { | ||||
|         waitUntil: 'domcontentloaded', | ||||
|  | @ -107,7 +107,7 @@ describe('navigation', function () { | |||
|       expect(response!.status()).toBe(200); | ||||
|     }); | ||||
|     it('should work when page calls history API in beforeunload', async () => { | ||||
|       const { page, server } = getTestState(); | ||||
|       const {page, server} = getTestState(); | ||||
| 
 | ||||
|       await page.goto(server.EMPTY_PAGE); | ||||
|       await page.evaluate(() => { | ||||
|  | @ -122,27 +122,33 @@ describe('navigation', function () { | |||
|       const response = await page.goto(server.PREFIX + '/grid.html'); | ||||
|       expect(response!.status()).toBe(200); | ||||
|     }); | ||||
|     it('should navigate to empty page with networkidle0', async () => { | ||||
|       const { page, server } = getTestState(); | ||||
|     it( | ||||
|       'should navigate to empty page with networkidle0', | ||||
|       async () => { | ||||
|         const {page, server} = getTestState(); | ||||
| 
 | ||||
|       const response = await page.goto(server.EMPTY_PAGE, { | ||||
|         waitUntil: 'networkidle0', | ||||
|       }); | ||||
|       expect(response!.status()).toBe(200); | ||||
|     }); | ||||
|     it('should navigate to empty page with networkidle2', async () => { | ||||
|       const { page, server } = getTestState(); | ||||
|         const response = await page.goto(server.EMPTY_PAGE, { | ||||
|           waitUntil: 'networkidle0', | ||||
|         }); | ||||
|         expect(response!.status()).toBe(200); | ||||
|       } | ||||
|     ); | ||||
|     it( | ||||
|       'should navigate to empty page with networkidle2', | ||||
|       async () => { | ||||
|         const {page, server} = getTestState(); | ||||
| 
 | ||||
|       const response = await page.goto(server.EMPTY_PAGE, { | ||||
|         waitUntil: 'networkidle2', | ||||
|       }); | ||||
|       expect(response!.status()).toBe(200); | ||||
|     }); | ||||
|         const response = await page.goto(server.EMPTY_PAGE, { | ||||
|           waitUntil: 'networkidle2', | ||||
|         }); | ||||
|         expect(response!.status()).toBe(200); | ||||
|       } | ||||
|     ); | ||||
|     it('should fail when navigating to bad url', async () => { | ||||
|       const { page, isChrome } = getTestState(); | ||||
|       const {page, isChrome} = getTestState(); | ||||
| 
 | ||||
|       let error!: Error; | ||||
|       await page.goto('asdfasdf').catch((error_) => { | ||||
|       await page.goto('asdfasdf').catch(error_ => { | ||||
|         return (error = error_); | ||||
|       }); | ||||
|       if (isChrome) { | ||||
|  | @ -153,7 +159,7 @@ describe('navigation', function () { | |||
|     }); | ||||
| 
 | ||||
|     function getExpectedSSLCertMessage(): string { | ||||
|       const { headless } = getTestState(); | ||||
|       const {headless} = getTestState(); | ||||
|       /** | ||||
|        * If you are running this on pre-Catalina versions of macOS this will fail | ||||
|        * locally. Mac OSX Catalina outputs a different message than other | ||||
|  | @ -168,7 +174,7 @@ describe('navigation', function () { | |||
|     } | ||||
| 
 | ||||
|     it('should fail when navigating to bad SSL', async () => { | ||||
|       const { page, httpsServer, isChrome } = getTestState(); | ||||
|       const {page, httpsServer, isChrome} = getTestState(); | ||||
| 
 | ||||
|       // Make sure that network events do not emit 'undefined'.
 | ||||
|       // @see https://crbug.com/750469
 | ||||
|  | @ -184,7 +190,7 @@ describe('navigation', function () { | |||
|       }); | ||||
| 
 | ||||
|       let error!: Error; | ||||
|       await page.goto(httpsServer.EMPTY_PAGE).catch((error_) => { | ||||
|       await page.goto(httpsServer.EMPTY_PAGE).catch(error_ => { | ||||
|         return (error = error_); | ||||
|       }); | ||||
|       if (isChrome) { | ||||
|  | @ -198,43 +204,27 @@ describe('navigation', function () { | |||
|       expect(requests[1]!).toBe('requestfailed'); | ||||
|     }); | ||||
|     it('should fail when navigating to bad SSL after redirects', async () => { | ||||
|       const { page, server, httpsServer, isChrome } = getTestState(); | ||||
|       const {page, server, httpsServer, isChrome} = getTestState(); | ||||
| 
 | ||||
|       server.setRedirect('/redirect/1.html', '/redirect/2.html'); | ||||
|       server.setRedirect('/redirect/2.html', '/empty.html'); | ||||
|       let error!: Error; | ||||
|       await page | ||||
|         .goto(httpsServer.PREFIX + '/redirect/1.html') | ||||
|         .catch((error_) => { | ||||
|           return (error = error_); | ||||
|         }); | ||||
|       await page.goto(httpsServer.PREFIX + '/redirect/1.html').catch(error_ => { | ||||
|         return (error = error_); | ||||
|       }); | ||||
|       if (isChrome) { | ||||
|         expect(error.message).toContain(getExpectedSSLCertMessage()); | ||||
|       } else { | ||||
|         expect(error.message).toContain('SSL_ERROR_UNKNOWN'); | ||||
|       } | ||||
|     }); | ||||
|     it('should throw if networkidle is passed as an option', async () => { | ||||
|       const { page, server } = getTestState(); | ||||
| 
 | ||||
|       let error!: Error; | ||||
|       await page | ||||
|         // @ts-expect-error purposefully passing an old option
 | ||||
|         .goto(server.EMPTY_PAGE, { waitUntil: 'networkidle' }) | ||||
|         .catch((error_) => { | ||||
|           return (error = error_); | ||||
|         }); | ||||
|       expect(error.message).toContain( | ||||
|         '"networkidle" option is no longer supported' | ||||
|       ); | ||||
|     }); | ||||
|     it('should fail when main resources failed to load', async () => { | ||||
|       const { page, isChrome } = getTestState(); | ||||
|       const {page, isChrome} = getTestState(); | ||||
| 
 | ||||
|       let error!: Error; | ||||
|       await page | ||||
|         .goto('http://localhost:44123/non-existing-url') | ||||
|         .catch((error_) => { | ||||
|         .catch(error_ => { | ||||
|           return (error = error_); | ||||
|         }); | ||||
|       if (isChrome) { | ||||
|  | @ -244,61 +234,61 @@ describe('navigation', function () { | |||
|       } | ||||
|     }); | ||||
|     it('should fail when exceeding maximum navigation timeout', async () => { | ||||
|       const { page, server, puppeteer } = getTestState(); | ||||
|       const {page, server, puppeteer} = getTestState(); | ||||
| 
 | ||||
|       // Hang for request to the empty.html
 | ||||
|       server.setRoute('/empty.html', () => {}); | ||||
|       let error!: Error; | ||||
|       await page | ||||
|         .goto(server.PREFIX + '/empty.html', { timeout: 1 }) | ||||
|         .catch((error_) => { | ||||
|         .goto(server.PREFIX + '/empty.html', {timeout: 1}) | ||||
|         .catch(error_ => { | ||||
|           return (error = error_); | ||||
|         }); | ||||
|       expect(error.message).toContain('Navigation timeout of 1 ms exceeded'); | ||||
|       expect(error).toBeInstanceOf(puppeteer.errors.TimeoutError); | ||||
|     }); | ||||
|     it('should fail when exceeding default maximum navigation timeout', async () => { | ||||
|       const { page, server, puppeteer } = getTestState(); | ||||
|       const {page, server, puppeteer} = getTestState(); | ||||
| 
 | ||||
|       // Hang for request to the empty.html
 | ||||
|       server.setRoute('/empty.html', () => {}); | ||||
|       let error!: Error; | ||||
|       page.setDefaultNavigationTimeout(1); | ||||
|       await page.goto(server.PREFIX + '/empty.html').catch((error_) => { | ||||
|       await page.goto(server.PREFIX + '/empty.html').catch(error_ => { | ||||
|         return (error = error_); | ||||
|       }); | ||||
|       expect(error.message).toContain('Navigation timeout of 1 ms exceeded'); | ||||
|       expect(error).toBeInstanceOf(puppeteer.errors.TimeoutError); | ||||
|     }); | ||||
|     it('should fail when exceeding default maximum timeout', async () => { | ||||
|       const { page, server, puppeteer } = getTestState(); | ||||
|       const {page, server, puppeteer} = getTestState(); | ||||
| 
 | ||||
|       // Hang for request to the empty.html
 | ||||
|       server.setRoute('/empty.html', () => {}); | ||||
|       let error!: Error; | ||||
|       page.setDefaultTimeout(1); | ||||
|       await page.goto(server.PREFIX + '/empty.html').catch((error_) => { | ||||
|       await page.goto(server.PREFIX + '/empty.html').catch(error_ => { | ||||
|         return (error = error_); | ||||
|       }); | ||||
|       expect(error.message).toContain('Navigation timeout of 1 ms exceeded'); | ||||
|       expect(error).toBeInstanceOf(puppeteer.errors.TimeoutError); | ||||
|     }); | ||||
|     it('should prioritize default navigation timeout over default timeout', async () => { | ||||
|       const { page, server, puppeteer } = getTestState(); | ||||
|       const {page, server, puppeteer} = getTestState(); | ||||
| 
 | ||||
|       // Hang for request to the empty.html
 | ||||
|       server.setRoute('/empty.html', () => {}); | ||||
|       let error!: Error; | ||||
|       page.setDefaultTimeout(0); | ||||
|       page.setDefaultNavigationTimeout(1); | ||||
|       await page.goto(server.PREFIX + '/empty.html').catch((error_) => { | ||||
|       await page.goto(server.PREFIX + '/empty.html').catch(error_ => { | ||||
|         return (error = error_); | ||||
|       }); | ||||
|       expect(error.message).toContain('Navigation timeout of 1 ms exceeded'); | ||||
|       expect(error).toBeInstanceOf(puppeteer.errors.TimeoutError); | ||||
|     }); | ||||
|     it('should disable timeout when its set to 0', async () => { | ||||
|       const { page, server } = getTestState(); | ||||
|       const {page, server} = getTestState(); | ||||
| 
 | ||||
|       let error!: Error; | ||||
|       let loaded = false; | ||||
|  | @ -306,34 +296,34 @@ describe('navigation', function () { | |||
|         return (loaded = true); | ||||
|       }); | ||||
|       await page | ||||
|         .goto(server.PREFIX + '/grid.html', { timeout: 0, waitUntil: ['load'] }) | ||||
|         .catch((error_) => { | ||||
|         .goto(server.PREFIX + '/grid.html', {timeout: 0, waitUntil: ['load']}) | ||||
|         .catch(error_ => { | ||||
|           return (error = error_); | ||||
|         }); | ||||
|       expect(error).toBeUndefined(); | ||||
|       expect(loaded).toBe(true); | ||||
|     }); | ||||
|     it('should work when navigating to valid url', async () => { | ||||
|       const { page, server } = getTestState(); | ||||
|       const {page, server} = getTestState(); | ||||
| 
 | ||||
|       const response = (await page.goto(server.EMPTY_PAGE))!; | ||||
|       expect(response.ok()).toBe(true); | ||||
|     }); | ||||
|     it('should work when navigating to data url', async () => { | ||||
|       const { page } = getTestState(); | ||||
|       const {page} = getTestState(); | ||||
| 
 | ||||
|       const response = (await page.goto('data:text/html,hello'))!; | ||||
|       expect(response.ok()).toBe(true); | ||||
|     }); | ||||
|     it('should work when navigating to 404', async () => { | ||||
|       const { page, server } = getTestState(); | ||||
|       const {page, server} = getTestState(); | ||||
| 
 | ||||
|       const response = (await page.goto(server.PREFIX + '/not-found'))!; | ||||
|       expect(response.ok()).toBe(false); | ||||
|       expect(response.status()).toBe(404); | ||||
|     }); | ||||
|     it('should return last response in redirect chain', async () => { | ||||
|       const { page, server } = getTestState(); | ||||
|       const {page, server} = getTestState(); | ||||
| 
 | ||||
|       server.setRedirect('/redirect/1.html', '/redirect/2.html'); | ||||
|       server.setRedirect('/redirect/2.html', '/redirect/3.html'); | ||||
|  | @ -342,84 +332,90 @@ describe('navigation', function () { | |||
|       expect(response.ok()).toBe(true); | ||||
|       expect(response.url()).toBe(server.EMPTY_PAGE); | ||||
|     }); | ||||
|     it('should wait for network idle to succeed navigation', async () => { | ||||
|       const { page, server } = getTestState(); | ||||
|     it( | ||||
|       'should wait for network idle to succeed navigation', | ||||
|       async () => { | ||||
|         const {page, server} = getTestState(); | ||||
| 
 | ||||
|       let responses: ServerResponse[] = []; | ||||
|       // Hold on to a bunch of requests without answering.
 | ||||
|       server.setRoute('/fetch-request-a.js', (_req, res) => { | ||||
|         return responses.push(res); | ||||
|       }); | ||||
|       server.setRoute('/fetch-request-b.js', (_req, res) => { | ||||
|         return responses.push(res); | ||||
|       }); | ||||
|       server.setRoute('/fetch-request-c.js', (_req, res) => { | ||||
|         return responses.push(res); | ||||
|       }); | ||||
|       server.setRoute('/fetch-request-d.js', (_req, res) => { | ||||
|         return responses.push(res); | ||||
|       }); | ||||
|       const initialFetchResourcesRequested = Promise.all([ | ||||
|         server.waitForRequest('/fetch-request-a.js'), | ||||
|         server.waitForRequest('/fetch-request-b.js'), | ||||
|         server.waitForRequest('/fetch-request-c.js'), | ||||
|       ]); | ||||
|       const secondFetchResourceRequested = server.waitForRequest( | ||||
|         '/fetch-request-d.js' | ||||
|       ); | ||||
|         let responses: ServerResponse[] = []; | ||||
|         // Hold on to a bunch of requests without answering.
 | ||||
|         server.setRoute('/fetch-request-a.js', (_req, res) => { | ||||
|           return responses.push(res); | ||||
|         }); | ||||
|         server.setRoute('/fetch-request-b.js', (_req, res) => { | ||||
|           return responses.push(res); | ||||
|         }); | ||||
|         server.setRoute('/fetch-request-c.js', (_req, res) => { | ||||
|           return responses.push(res); | ||||
|         }); | ||||
|         server.setRoute('/fetch-request-d.js', (_req, res) => { | ||||
|           return responses.push(res); | ||||
|         }); | ||||
|         const initialFetchResourcesRequested = Promise.all([ | ||||
|           server.waitForRequest('/fetch-request-a.js'), | ||||
|           server.waitForRequest('/fetch-request-b.js'), | ||||
|           server.waitForRequest('/fetch-request-c.js'), | ||||
|         ]); | ||||
|         const secondFetchResourceRequested = server.waitForRequest( | ||||
|           '/fetch-request-d.js' | ||||
|         ); | ||||
| 
 | ||||
|       // Navigate to a page which loads immediately and then does a bunch of
 | ||||
|       // requests via javascript's fetch method.
 | ||||
|       const navigationPromise = page.goto(server.PREFIX + '/networkidle.html', { | ||||
|         waitUntil: 'networkidle0', | ||||
|       }); | ||||
|       // Track when the navigation gets completed.
 | ||||
|       let navigationFinished = false; | ||||
|       navigationPromise.then(() => { | ||||
|         return (navigationFinished = true); | ||||
|       }); | ||||
|         // Navigate to a page which loads immediately and then does a bunch of
 | ||||
|         // requests via javascript's fetch method.
 | ||||
|         const navigationPromise = page.goto( | ||||
|           server.PREFIX + '/networkidle.html', | ||||
|           { | ||||
|             waitUntil: 'networkidle0', | ||||
|           } | ||||
|         ); | ||||
|         // Track when the navigation gets completed.
 | ||||
|         let navigationFinished = false; | ||||
|         navigationPromise.then(() => { | ||||
|           return (navigationFinished = true); | ||||
|         }); | ||||
| 
 | ||||
|       // Wait for the page's 'load' event.
 | ||||
|       await new Promise((fulfill) => { | ||||
|         return page.once('load', fulfill); | ||||
|       }); | ||||
|       expect(navigationFinished).toBe(false); | ||||
|         // Wait for the page's 'load' event.
 | ||||
|         await new Promise(fulfill => { | ||||
|           return page.once('load', fulfill); | ||||
|         }); | ||||
|         expect(navigationFinished).toBe(false); | ||||
| 
 | ||||
|       // Wait for the initial three resources to be requested.
 | ||||
|       await initialFetchResourcesRequested; | ||||
|         // Wait for the initial three resources to be requested.
 | ||||
|         await initialFetchResourcesRequested; | ||||
| 
 | ||||
|       // Expect navigation still to be not finished.
 | ||||
|       expect(navigationFinished).toBe(false); | ||||
|         // Expect navigation still to be not finished.
 | ||||
|         expect(navigationFinished).toBe(false); | ||||
| 
 | ||||
|       // Respond to initial requests.
 | ||||
|       for (const response of responses) { | ||||
|         response.statusCode = 404; | ||||
|         response.end(`File not found`); | ||||
|         // Respond to initial requests.
 | ||||
|         for (const response of responses) { | ||||
|           response.statusCode = 404; | ||||
|           response.end(`File not found`); | ||||
|         } | ||||
| 
 | ||||
|         // Reset responses array
 | ||||
|         responses = []; | ||||
| 
 | ||||
|         // Wait for the second round to be requested.
 | ||||
|         await secondFetchResourceRequested; | ||||
|         // Expect navigation still to be not finished.
 | ||||
|         expect(navigationFinished).toBe(false); | ||||
| 
 | ||||
|         // Respond to requests.
 | ||||
|         for (const response of responses) { | ||||
|           response.statusCode = 404; | ||||
|           response.end(`File not found`); | ||||
|         } | ||||
| 
 | ||||
|         const response = (await navigationPromise)!; | ||||
|         // Expect navigation to succeed.
 | ||||
|         expect(response.ok()).toBe(true); | ||||
|       } | ||||
| 
 | ||||
|       // Reset responses array
 | ||||
|       responses = []; | ||||
| 
 | ||||
|       // Wait for the second round to be requested.
 | ||||
|       await secondFetchResourceRequested; | ||||
|       // Expect navigation still to be not finished.
 | ||||
|       expect(navigationFinished).toBe(false); | ||||
| 
 | ||||
|       // Respond to requests.
 | ||||
|       for (const response of responses) { | ||||
|         response.statusCode = 404; | ||||
|         response.end(`File not found`); | ||||
|       } | ||||
| 
 | ||||
|       const response = (await navigationPromise)!; | ||||
|       // Expect navigation to succeed.
 | ||||
|       expect(response.ok()).toBe(true); | ||||
|     }); | ||||
|     ); | ||||
|     it('should not leak listeners during navigation', async () => { | ||||
|       const { page, server } = getTestState(); | ||||
|       const {page, server} = getTestState(); | ||||
| 
 | ||||
|       let warning = null; | ||||
|       const warningHandler: NodeJS.WarningListener = (w) => { | ||||
|       const warningHandler: NodeJS.WarningListener = w => { | ||||
|         return (warning = w); | ||||
|       }; | ||||
|       process.on('warning', warningHandler); | ||||
|  | @ -430,10 +426,10 @@ describe('navigation', function () { | |||
|       expect(warning).toBe(null); | ||||
|     }); | ||||
|     it('should not leak listeners during bad navigation', async () => { | ||||
|       const { page } = getTestState(); | ||||
|       const {page} = getTestState(); | ||||
| 
 | ||||
|       let warning = null; | ||||
|       const warningHandler: NodeJS.WarningListener = (w) => { | ||||
|       const warningHandler: NodeJS.WarningListener = w => { | ||||
|         return (warning = w); | ||||
|       }; | ||||
|       process.on('warning', warningHandler); | ||||
|  | @ -446,10 +442,10 @@ describe('navigation', function () { | |||
|       expect(warning).toBe(null); | ||||
|     }); | ||||
|     it('should not leak listeners during navigation of 11 pages', async () => { | ||||
|       const { context, server } = getTestState(); | ||||
|       const {context, server} = getTestState(); | ||||
| 
 | ||||
|       let warning = null; | ||||
|       const warningHandler: NodeJS.WarningListener = (w) => { | ||||
|       const warningHandler: NodeJS.WarningListener = w => { | ||||
|         return (warning = w); | ||||
|       }; | ||||
|       process.on('warning', warningHandler); | ||||
|  | @ -463,41 +459,47 @@ describe('navigation', function () { | |||
|       process.removeListener('warning', warningHandler); | ||||
|       expect(warning).toBe(null); | ||||
|     }); | ||||
|     it('should navigate to dataURL and fire dataURL requests', async () => { | ||||
|       const { page } = getTestState(); | ||||
|     it( | ||||
|       'should navigate to dataURL and fire dataURL requests', | ||||
|       async () => { | ||||
|         const {page} = getTestState(); | ||||
| 
 | ||||
|       const requests: HTTPRequest[] = []; | ||||
|       page.on('request', (request) => { | ||||
|         return !utils.isFavicon(request) && requests.push(request); | ||||
|       }); | ||||
|       const dataURL = 'data:text/html,<div>yo</div>'; | ||||
|       const response = (await page.goto(dataURL))!; | ||||
|       expect(response.status()).toBe(200); | ||||
|       expect(requests.length).toBe(1); | ||||
|       expect(requests[0]!.url()).toBe(dataURL); | ||||
|     }); | ||||
|     it('should navigate to URL with hash and fire requests without hash', async () => { | ||||
|       const { page, server } = getTestState(); | ||||
|         const requests: HTTPRequest[] = []; | ||||
|         page.on('request', request => { | ||||
|           return !utils.isFavicon(request) && requests.push(request); | ||||
|         }); | ||||
|         const dataURL = 'data:text/html,<div>yo</div>'; | ||||
|         const response = (await page.goto(dataURL))!; | ||||
|         expect(response.status()).toBe(200); | ||||
|         expect(requests.length).toBe(1); | ||||
|         expect(requests[0]!.url()).toBe(dataURL); | ||||
|       } | ||||
|     ); | ||||
|     it( | ||||
|       'should navigate to URL with hash and fire requests without hash', | ||||
|       async () => { | ||||
|         const {page, server} = getTestState(); | ||||
| 
 | ||||
|       const requests: HTTPRequest[] = []; | ||||
|       page.on('request', (request) => { | ||||
|         return !utils.isFavicon(request) && requests.push(request); | ||||
|       }); | ||||
|       const response = (await page.goto(server.EMPTY_PAGE + '#hash'))!; | ||||
|       expect(response.status()).toBe(200); | ||||
|       expect(response.url()).toBe(server.EMPTY_PAGE); | ||||
|       expect(requests.length).toBe(1); | ||||
|       expect(requests[0]!.url()).toBe(server.EMPTY_PAGE); | ||||
|     }); | ||||
|         const requests: HTTPRequest[] = []; | ||||
|         page.on('request', request => { | ||||
|           return !utils.isFavicon(request) && requests.push(request); | ||||
|         }); | ||||
|         const response = (await page.goto(server.EMPTY_PAGE + '#hash'))!; | ||||
|         expect(response.status()).toBe(200); | ||||
|         expect(response.url()).toBe(server.EMPTY_PAGE); | ||||
|         expect(requests.length).toBe(1); | ||||
|         expect(requests[0]!.url()).toBe(server.EMPTY_PAGE); | ||||
|       } | ||||
|     ); | ||||
|     it('should work with self requesting page', async () => { | ||||
|       const { page, server } = getTestState(); | ||||
|       const {page, server} = getTestState(); | ||||
| 
 | ||||
|       const response = (await page.goto(server.PREFIX + '/self-request.html'))!; | ||||
|       expect(response.status()).toBe(200); | ||||
|       expect(response.url()).toContain('self-request.html'); | ||||
|     }); | ||||
|     it('should fail when navigating and show the url at the error message', async () => { | ||||
|       const { page, httpsServer } = getTestState(); | ||||
|       const {page, httpsServer} = getTestState(); | ||||
| 
 | ||||
|       const url = httpsServer.PREFIX + '/redirect/1.html'; | ||||
|       let error!: Error; | ||||
|  | @ -509,7 +511,7 @@ describe('navigation', function () { | |||
|       expect(error.message).toContain(url); | ||||
|     }); | ||||
|     it('should send referer', async () => { | ||||
|       const { page, server } = getTestState(); | ||||
|       const {page, server} = getTestState(); | ||||
| 
 | ||||
|       const [request1, request2] = await Promise.all([ | ||||
|         server.waitForRequest('/grid.html'), | ||||
|  | @ -526,7 +528,7 @@ describe('navigation', function () { | |||
| 
 | ||||
|   describe('Page.waitForNavigation', function () { | ||||
|     it('should work', async () => { | ||||
|       const { page, server } = getTestState(); | ||||
|       const {page, server} = getTestState(); | ||||
| 
 | ||||
|       await page.goto(server.EMPTY_PAGE); | ||||
|       const [response] = await Promise.all([ | ||||
|  | @ -539,7 +541,7 @@ describe('navigation', function () { | |||
|       expect(response!.url()).toContain('grid.html'); | ||||
|     }); | ||||
|     it('should work with both domcontentloaded and load', async () => { | ||||
|       const { page, server } = getTestState(); | ||||
|       const {page, server} = getTestState(); | ||||
| 
 | ||||
|       let response!: ServerResponse; | ||||
|       server.setRoute('/one-style.css', (_req, res) => { | ||||
|  | @ -567,7 +569,7 @@ describe('navigation', function () { | |||
|       await navigationPromise; | ||||
|     }); | ||||
|     it('should work with clicking on anchor links', async () => { | ||||
|       const { page, server } = getTestState(); | ||||
|       const {page, server} = getTestState(); | ||||
| 
 | ||||
|       await page.goto(server.EMPTY_PAGE); | ||||
|       await page.setContent(`<a href='#foobar'>foobar</a>`); | ||||
|  | @ -579,7 +581,7 @@ describe('navigation', function () { | |||
|       expect(page.url()).toBe(server.EMPTY_PAGE + '#foobar'); | ||||
|     }); | ||||
|     it('should work with history.pushState()', async () => { | ||||
|       const { page, server } = getTestState(); | ||||
|       const {page, server} = getTestState(); | ||||
| 
 | ||||
|       await page.goto(server.EMPTY_PAGE); | ||||
|       await page.setContent(` | ||||
|  | @ -596,7 +598,7 @@ describe('navigation', function () { | |||
|       expect(page.url()).toBe(server.PREFIX + '/wow.html'); | ||||
|     }); | ||||
|     it('should work with history.replaceState()', async () => { | ||||
|       const { page, server } = getTestState(); | ||||
|       const {page, server} = getTestState(); | ||||
| 
 | ||||
|       await page.goto(server.EMPTY_PAGE); | ||||
|       await page.setContent(` | ||||
|  | @ -612,11 +614,13 @@ describe('navigation', function () { | |||
|       expect(response).toBe(null); | ||||
|       expect(page.url()).toBe(server.PREFIX + '/replaced.html'); | ||||
|     }); | ||||
|     it('should work with DOM history.back()/history.forward()', async () => { | ||||
|       const { page, server } = getTestState(); | ||||
|     it( | ||||
|       'should work with DOM history.back()/history.forward()', | ||||
|       async () => { | ||||
|         const {page, server} = getTestState(); | ||||
| 
 | ||||
|       await page.goto(server.EMPTY_PAGE); | ||||
|       await page.setContent(` | ||||
|         await page.goto(server.EMPTY_PAGE); | ||||
|         await page.setContent(` | ||||
|         <a id=back onclick='javascript:goBack()'>back</a> | ||||
|         <a id=forward onclick='javascript:goForward()'>forward</a> | ||||
|         <script> | ||||
|  | @ -626,47 +630,51 @@ describe('navigation', function () { | |||
|           history.pushState({}, '', '/second.html'); | ||||
|         </script> | ||||
|       `);
 | ||||
|       expect(page.url()).toBe(server.PREFIX + '/second.html'); | ||||
|       const [backResponse] = await Promise.all([ | ||||
|         page.waitForNavigation(), | ||||
|         page.click('a#back'), | ||||
|       ]); | ||||
|       expect(backResponse).toBe(null); | ||||
|       expect(page.url()).toBe(server.PREFIX + '/first.html'); | ||||
|       const [forwardResponse] = await Promise.all([ | ||||
|         page.waitForNavigation(), | ||||
|         page.click('a#forward'), | ||||
|       ]); | ||||
|       expect(forwardResponse).toBe(null); | ||||
|       expect(page.url()).toBe(server.PREFIX + '/second.html'); | ||||
|     }); | ||||
|     it('should work when subframe issues window.stop()', async () => { | ||||
|       const { page, server } = getTestState(); | ||||
|         expect(page.url()).toBe(server.PREFIX + '/second.html'); | ||||
|         const [backResponse] = await Promise.all([ | ||||
|           page.waitForNavigation(), | ||||
|           page.click('a#back'), | ||||
|         ]); | ||||
|         expect(backResponse).toBe(null); | ||||
|         expect(page.url()).toBe(server.PREFIX + '/first.html'); | ||||
|         const [forwardResponse] = await Promise.all([ | ||||
|           page.waitForNavigation(), | ||||
|           page.click('a#forward'), | ||||
|         ]); | ||||
|         expect(forwardResponse).toBe(null); | ||||
|         expect(page.url()).toBe(server.PREFIX + '/second.html'); | ||||
|       } | ||||
|     ); | ||||
|     it( | ||||
|       'should work when subframe issues window.stop()', | ||||
|       async () => { | ||||
|         const {page, server} = getTestState(); | ||||
| 
 | ||||
|       server.setRoute('/frames/style.css', () => {}); | ||||
|       const navigationPromise = page.goto( | ||||
|         server.PREFIX + '/frames/one-frame.html' | ||||
|       ); | ||||
|       const frame = await utils.waitEvent(page, 'frameattached'); | ||||
|       await new Promise<void>((fulfill) => { | ||||
|         page.on('framenavigated', (f) => { | ||||
|           if (f === frame) { | ||||
|             fulfill(); | ||||
|           } | ||||
|         server.setRoute('/frames/style.css', () => {}); | ||||
|         const navigationPromise = page.goto( | ||||
|           server.PREFIX + '/frames/one-frame.html' | ||||
|         ); | ||||
|         const frame = await utils.waitEvent(page, 'frameattached'); | ||||
|         await new Promise<void>(fulfill => { | ||||
|           page.on('framenavigated', f => { | ||||
|             if (f === frame) { | ||||
|               fulfill(); | ||||
|             } | ||||
|           }); | ||||
|         }); | ||||
|       }); | ||||
|       await Promise.all([ | ||||
|         frame.evaluate(() => { | ||||
|           return window.stop(); | ||||
|         }), | ||||
|         navigationPromise, | ||||
|       ]); | ||||
|     }); | ||||
|         await Promise.all([ | ||||
|           frame.evaluate(() => { | ||||
|             return window.stop(); | ||||
|           }), | ||||
|           navigationPromise, | ||||
|         ]); | ||||
|       } | ||||
|     ); | ||||
|   }); | ||||
| 
 | ||||
|   describe('Page.goBack', function () { | ||||
|     it('should work', async () => { | ||||
|       const { page, server } = getTestState(); | ||||
|       const {page, server} = getTestState(); | ||||
| 
 | ||||
|       await page.goto(server.EMPTY_PAGE); | ||||
|       await page.goto(server.PREFIX + '/grid.html'); | ||||
|  | @ -683,7 +691,7 @@ describe('navigation', function () { | |||
|       expect(response).toBe(null); | ||||
|     }); | ||||
|     it('should work with HistoryAPI', async () => { | ||||
|       const { page, server } = getTestState(); | ||||
|       const {page, server} = getTestState(); | ||||
| 
 | ||||
|       await page.goto(server.EMPTY_PAGE); | ||||
|       await page.evaluate(() => { | ||||
|  | @ -703,7 +711,7 @@ describe('navigation', function () { | |||
| 
 | ||||
|   describe('Frame.goto', function () { | ||||
|     it('should navigate subframes', async () => { | ||||
|       const { page, server } = getTestState(); | ||||
|       const {page, server} = getTestState(); | ||||
| 
 | ||||
|       await page.goto(server.PREFIX + '/frames/one-frame.html'); | ||||
|       expect(page.frames()[0]!.url()).toContain('/frames/one-frame.html'); | ||||
|  | @ -714,7 +722,7 @@ describe('navigation', function () { | |||
|       expect(response.frame()).toBe(page.frames()[1]!); | ||||
|     }); | ||||
|     it('should reject when frame detaches', async () => { | ||||
|       const { page, server } = getTestState(); | ||||
|       const {page, server} = getTestState(); | ||||
| 
 | ||||
|       await page.goto(server.PREFIX + '/frames/one-frame.html'); | ||||
| 
 | ||||
|  | @ -722,19 +730,19 @@ describe('navigation', function () { | |||
|       const navigationPromise = page | ||||
|         .frames()[1]! | ||||
|         .goto(server.EMPTY_PAGE) | ||||
|         .catch((error_) => { | ||||
|         .catch(error_ => { | ||||
|           return error_; | ||||
|         }); | ||||
|       await server.waitForRequest('/empty.html'); | ||||
| 
 | ||||
|       await page.$eval('iframe', (frame) => { | ||||
|       await page.$eval('iframe', frame => { | ||||
|         return frame.remove(); | ||||
|       }); | ||||
|       const error = await navigationPromise; | ||||
|       expect(error.message).toBe('Navigating frame was detached'); | ||||
|     }); | ||||
|     it('should return matching responses', async () => { | ||||
|       const { page, server } = getTestState(); | ||||
|       const {page, server} = getTestState(); | ||||
| 
 | ||||
|       // Disable cache: otherwise, chromium will cache similar requests.
 | ||||
|       await page.setCacheEnabled(false); | ||||
|  | @ -768,7 +776,7 @@ describe('navigation', function () { | |||
| 
 | ||||
|   describe('Frame.waitForNavigation', function () { | ||||
|     it('should work', async () => { | ||||
|       const { page, server } = getTestState(); | ||||
|       const {page, server} = getTestState(); | ||||
| 
 | ||||
|       await page.goto(server.PREFIX + '/frames/one-frame.html'); | ||||
|       const frame = page.frames()[1]!; | ||||
|  | @ -784,14 +792,14 @@ describe('navigation', function () { | |||
|       expect(page.url()).toContain('/frames/one-frame.html'); | ||||
|     }); | ||||
|     it('should fail when frame detaches', async () => { | ||||
|       const { page, server } = getTestState(); | ||||
|       const {page, server} = getTestState(); | ||||
| 
 | ||||
|       await page.goto(server.PREFIX + '/frames/one-frame.html'); | ||||
|       const frame = page.frames()[1]!; | ||||
| 
 | ||||
|       server.setRoute('/empty.html', () => {}); | ||||
|       let error!: Error; | ||||
|       const navigationPromise = frame.waitForNavigation().catch((error_) => { | ||||
|       const navigationPromise = frame.waitForNavigation().catch(error_ => { | ||||
|         return (error = error_); | ||||
|       }); | ||||
|       await Promise.all([ | ||||
|  | @ -800,7 +808,7 @@ describe('navigation', function () { | |||
|           return ((window as any).location = '/empty.html'); | ||||
|         }), | ||||
|       ]); | ||||
|       await page.$eval('iframe', (frame) => { | ||||
|       await page.$eval('iframe', frame => { | ||||
|         return frame.remove(); | ||||
|       }); | ||||
|       await navigationPromise; | ||||
|  | @ -810,7 +818,7 @@ describe('navigation', function () { | |||
| 
 | ||||
|   describe('Page.reload', function () { | ||||
|     it('should work', async () => { | ||||
|       const { page, server } = getTestState(); | ||||
|       const {page, server} = getTestState(); | ||||
| 
 | ||||
|       await page.goto(server.EMPTY_PAGE); | ||||
|       await page.evaluate(() => { | ||||
|  |  | |||
|  | @ -469,7 +469,6 @@ describe('network', function () { | |||
|       }); | ||||
|       (await page.goto(server.EMPTY_PAGE))!; | ||||
|       expect(responses.length).toBe(1); | ||||
|       console.log('timing',responses[0]!.timing()) | ||||
|       expect(responses[0]!.timing()!.receiveHeadersEnd).toBeGreaterThan(0); | ||||
|     }); | ||||
|   }); | ||||
|  | @ -854,10 +853,13 @@ describe('network', function () { | |||
|           res.end(); | ||||
|         }); | ||||
|         await page.goto(httpsServer.PREFIX + '/setcookie.html'); | ||||
| 
 | ||||
|         const url = httpsServer.CROSS_PROCESS_PREFIX + '/setcookie.html'; | ||||
|         const response = await new Promise<HTTPResponse>(resolve => { | ||||
|           page.on('response', resolve); | ||||
|           const url = httpsServer.CROSS_PROCESS_PREFIX + '/setcookie.html'; | ||||
|           page.on('response', response => { | ||||
|             if (response.url() === url) { | ||||
|               resolve(response); | ||||
|             } | ||||
|           }); | ||||
|           page.evaluate(src => { | ||||
|             const xhr = new XMLHttpRequest(); | ||||
|             xhr.open('GET', src); | ||||
|  |  | |||
|  | @ -433,32 +433,29 @@ describe('Page', function () { | |||
|         }) | ||||
|       ).toEqual(['prompt', 'denied', 'granted', 'prompt']); | ||||
|     }); | ||||
|     it( | ||||
|       'should isolate permissions between browser contexts', | ||||
|       async () => { | ||||
|         const {page, server, context, browser} = getTestState(); | ||||
|     it('should isolate permissions between browser contexts', async () => { | ||||
|       const {page, server, context, browser} = getTestState(); | ||||
| 
 | ||||
|         await page.goto(server.EMPTY_PAGE); | ||||
|         const otherContext = await browser.createIncognitoBrowserContext(); | ||||
|         const otherPage = await otherContext.newPage(); | ||||
|         await otherPage.goto(server.EMPTY_PAGE); | ||||
|         expect(await getPermission(page, 'geolocation')).toBe('prompt'); | ||||
|         expect(await getPermission(otherPage, 'geolocation')).toBe('prompt'); | ||||
|       await page.goto(server.EMPTY_PAGE); | ||||
|       const otherContext = await browser.createIncognitoBrowserContext(); | ||||
|       const otherPage = await otherContext.newPage(); | ||||
|       await otherPage.goto(server.EMPTY_PAGE); | ||||
|       expect(await getPermission(page, 'geolocation')).toBe('prompt'); | ||||
|       expect(await getPermission(otherPage, 'geolocation')).toBe('prompt'); | ||||
| 
 | ||||
|         await context.overridePermissions(server.EMPTY_PAGE, []); | ||||
|         await otherContext.overridePermissions(server.EMPTY_PAGE, [ | ||||
|           'geolocation', | ||||
|         ]); | ||||
|         expect(await getPermission(page, 'geolocation')).toBe('denied'); | ||||
|         expect(await getPermission(otherPage, 'geolocation')).toBe('granted'); | ||||
|       await context.overridePermissions(server.EMPTY_PAGE, []); | ||||
|       await otherContext.overridePermissions(server.EMPTY_PAGE, [ | ||||
|         'geolocation', | ||||
|       ]); | ||||
|       expect(await getPermission(page, 'geolocation')).toBe('denied'); | ||||
|       expect(await getPermission(otherPage, 'geolocation')).toBe('granted'); | ||||
| 
 | ||||
|         await context.clearPermissionOverrides(); | ||||
|         expect(await getPermission(page, 'geolocation')).toBe('prompt'); | ||||
|         expect(await getPermission(otherPage, 'geolocation')).toBe('granted'); | ||||
|       await context.clearPermissionOverrides(); | ||||
|       expect(await getPermission(page, 'geolocation')).toBe('prompt'); | ||||
|       expect(await getPermission(otherPage, 'geolocation')).toBe('granted'); | ||||
| 
 | ||||
|         await otherContext.close(); | ||||
|       } | ||||
|     ); | ||||
|       await otherContext.close(); | ||||
|     }); | ||||
|     it('should grant persistent-storage', async () => { | ||||
|       const {page, server, context} = getTestState(); | ||||
| 
 | ||||
|  | @ -1630,7 +1627,7 @@ describe('Page', function () { | |||
|         error = error_ as Error; | ||||
|       } | ||||
|       expect(error.message).toBe( | ||||
|         'Provide an object with a `url`, `path` or `content` property' | ||||
|         'Exactly one of `url`, `path`, or `content` must be specified.' | ||||
|       ); | ||||
|     }); | ||||
| 
 | ||||
|  | @ -1692,7 +1689,7 @@ describe('Page', function () { | |||
|     }); | ||||
| 
 | ||||
|     it('should throw an error if loading from url fail', async () => { | ||||
|       const {page, server} = getTestState(); | ||||
|       const {page, server, isFirefox} = getTestState(); | ||||
| 
 | ||||
|       await page.goto(server.EMPTY_PAGE); | ||||
|       let error!: Error; | ||||
|  | @ -1701,7 +1698,11 @@ describe('Page', function () { | |||
|       } catch (error_) { | ||||
|         error = error_ as Error; | ||||
|       } | ||||
|       expect(error.message).toBe('Loading script from /nonexistfile.js failed'); | ||||
|       if (isFirefox) { | ||||
|         expect(error.message).toBeTruthy(); | ||||
|       } else { | ||||
|         expect(error.message).toContain('Could not load script'); | ||||
|       } | ||||
|     }); | ||||
| 
 | ||||
|     it('should work with a path', async () => { | ||||
|  | @ -1796,7 +1797,7 @@ describe('Page', function () { | |||
|         error = error_ as Error; | ||||
|       } | ||||
|       expect(error.message).toBe( | ||||
|         'Provide an object with a `url`, `path` or `content` property' | ||||
|         'Exactly one of `url`, `path`, or `content` must be specified.' | ||||
|       ); | ||||
|     }); | ||||
| 
 | ||||
|  | @ -1814,7 +1815,7 @@ describe('Page', function () { | |||
|     }); | ||||
| 
 | ||||
|     it('should throw an error if loading from url fail', async () => { | ||||
|       const {page, server} = getTestState(); | ||||
|       const {page, server, isFirefox} = getTestState(); | ||||
| 
 | ||||
|       await page.goto(server.EMPTY_PAGE); | ||||
|       let error!: Error; | ||||
|  | @ -1823,7 +1824,11 @@ describe('Page', function () { | |||
|       } catch (error_) { | ||||
|         error = error_ as Error; | ||||
|       } | ||||
|       expect(error.message).toBe('Loading style from /nonexistfile.js failed'); | ||||
|       if (isFirefox) { | ||||
|         expect(error.message).toBeTruthy(); | ||||
|       } else { | ||||
|         expect(error.message).toContain('Could not load style'); | ||||
|       } | ||||
|     }); | ||||
| 
 | ||||
|     it('should work with a path', async () => { | ||||
|  | @ -1849,7 +1854,7 @@ describe('Page', function () { | |||
|         path: path.join(__dirname, '../assets/injectedstyle.css'), | ||||
|       }); | ||||
|       const styleHandle = (await page.$('style'))!; | ||||
|       const styleContent = await page.evaluate((style: HTMLStyleElement) => { | ||||
|       const styleContent = await page.evaluate(style => { | ||||
|         return style.innerHTML; | ||||
|       }, styleHandle); | ||||
|       expect(styleContent).toContain(path.join('assets', 'injectedstyle.css')); | ||||
|  | @ -1870,21 +1875,18 @@ describe('Page', function () { | |||
|       ).toBe('rgb(0, 128, 0)'); | ||||
|     }); | ||||
| 
 | ||||
|     it( | ||||
|       'should throw when added with content to the CSP page', | ||||
|       async () => { | ||||
|         const {page, server} = getTestState(); | ||||
|     it('should throw when added with content to the CSP page', async () => { | ||||
|       const {page, server} = getTestState(); | ||||
| 
 | ||||
|         await page.goto(server.PREFIX + '/csp.html'); | ||||
|         let error!: Error; | ||||
|         await page | ||||
|           .addStyleTag({content: 'body { background-color: green; }'}) | ||||
|           .catch(error_ => { | ||||
|             return (error = error_); | ||||
|           }); | ||||
|         expect(error).toBeTruthy(); | ||||
|       } | ||||
|     ); | ||||
|       await page.goto(server.PREFIX + '/csp.html'); | ||||
|       let error!: Error; | ||||
|       await page | ||||
|         .addStyleTag({content: 'body { background-color: green; }'}) | ||||
|         .catch(error_ => { | ||||
|           return (error = error_); | ||||
|         }); | ||||
|       expect(error).toBeTruthy(); | ||||
|     }); | ||||
| 
 | ||||
|     it('should throw when added with URL to the CSP page', async () => { | ||||
|       const {page, server} = getTestState(); | ||||
|  | @ -1953,23 +1955,20 @@ describe('Page', function () { | |||
|       ]); | ||||
|       expect(nonCachedRequest.headers['if-modified-since']).toBe(undefined); | ||||
|     }); | ||||
|     it( | ||||
|       'should stay disabled when toggling request interception on/off', | ||||
|       async () => { | ||||
|         const {page, server} = getTestState(); | ||||
|     it('should stay disabled when toggling request interception on/off', async () => { | ||||
|       const {page, server} = getTestState(); | ||||
| 
 | ||||
|         await page.setCacheEnabled(false); | ||||
|         await page.setRequestInterception(true); | ||||
|         await page.setRequestInterception(false); | ||||
|       await page.setCacheEnabled(false); | ||||
|       await page.setRequestInterception(true); | ||||
|       await page.setRequestInterception(false); | ||||
| 
 | ||||
|         await page.goto(server.PREFIX + '/cached/one-style.html'); | ||||
|         const [nonCachedRequest] = await Promise.all([ | ||||
|           server.waitForRequest('/cached/one-style.html'), | ||||
|           page.reload(), | ||||
|         ]); | ||||
|         expect(nonCachedRequest.headers['if-modified-since']).toBe(undefined); | ||||
|       } | ||||
|     ); | ||||
|       await page.goto(server.PREFIX + '/cached/one-style.html'); | ||||
|       const [nonCachedRequest] = await Promise.all([ | ||||
|         server.waitForRequest('/cached/one-style.html'), | ||||
|         page.reload(), | ||||
|       ]); | ||||
|       expect(nonCachedRequest.headers['if-modified-since']).toBe(undefined); | ||||
|     }); | ||||
|   }); | ||||
| 
 | ||||
|   describe('printing to PDF', function () { | ||||
|  | @ -2214,29 +2213,26 @@ describe('Page', function () { | |||
|       expect(error.message).toContain('Values must be strings'); | ||||
|     }); | ||||
|     // @see https://github.com/puppeteer/puppeteer/issues/3327
 | ||||
|     it( | ||||
|       'should work when re-defining top-level Event class', | ||||
|       async () => { | ||||
|         const {page, server} = getTestState(); | ||||
|     it('should work when re-defining top-level Event class', async () => { | ||||
|       const {page, server} = getTestState(); | ||||
| 
 | ||||
|         await page.goto(server.PREFIX + '/input/select.html'); | ||||
|       await page.goto(server.PREFIX + '/input/select.html'); | ||||
|       await page.evaluate(() => { | ||||
|         // @ts-expect-error Expected.
 | ||||
|         return (window.Event = undefined); | ||||
|       }); | ||||
|       await page.select('select', 'blue'); | ||||
|       expect( | ||||
|         await page.evaluate(() => { | ||||
|           // @ts-expect-error Expected.
 | ||||
|           return (window.Event = undefined); | ||||
|         }); | ||||
|         await page.select('select', 'blue'); | ||||
|         expect( | ||||
|           await page.evaluate(() => { | ||||
|             return (globalThis as any).result.onInput; | ||||
|           }) | ||||
|         ).toEqual(['blue']); | ||||
|         expect( | ||||
|           await page.evaluate(() => { | ||||
|             return (globalThis as any).result.onChange; | ||||
|           }) | ||||
|         ).toEqual(['blue']); | ||||
|       } | ||||
|     ); | ||||
|           return (globalThis as any).result.onInput; | ||||
|         }) | ||||
|       ).toEqual(['blue']); | ||||
|       expect( | ||||
|         await page.evaluate(() => { | ||||
|           return (globalThis as any).result.onChange; | ||||
|         }) | ||||
|       ).toEqual(['blue']); | ||||
|     }); | ||||
|   }); | ||||
| 
 | ||||
|   describe('Page.Events.Close', function () { | ||||
|  |  | |||
|  | @ -18,8 +18,8 @@ import expect from 'expect'; | |||
| import http from 'http'; | ||||
| import os from 'os'; | ||||
| import { | ||||
|   describeFailsFirefox, | ||||
|   getTestState, | ||||
|   describeFailsFirefox, | ||||
|   itFailsWindows, | ||||
| } from './mocha-utils.js'; | ||||
| import type {Server, IncomingMessage, ServerResponse} from 'http'; | ||||
|  | @ -125,7 +125,7 @@ describeFailsFirefox('request proxy', () => { | |||
|     const response = (await page.goto(emptyPageUrl))!; | ||||
| 
 | ||||
|     expect(response.ok()).toBe(true); | ||||
|     console.log('test',{proxiedRequestUrls, emptyPageUrl}) | ||||
| 
 | ||||
|     expect(proxiedRequestUrls).toEqual([emptyPageUrl]); | ||||
|   }); | ||||
| 
 | ||||
|  |  | |||
|  | @ -614,10 +614,11 @@ describe('request interception', function () { | |||
|         return request.continue(); | ||||
|       }); | ||||
| 
 | ||||
|       await page.goto(server.PREFIX + '/cached/one-style-font.html'); | ||||
|       await page.waitForResponse(r => { | ||||
|       const responsePromise = page.waitForResponse(r => { | ||||
|         return r.url().endsWith('/one-style.woff'); | ||||
|       }); | ||||
|       await page.goto(server.PREFIX + '/cached/one-style-font.html'); | ||||
|       await responsePromise; | ||||
|     }); | ||||
|   }); | ||||
| 
 | ||||
|  |  | |||
|  | @ -51,6 +51,22 @@ describe('Screenshots', function () { | |||
|       }); | ||||
|       expect(screenshot).toBeGolden('screenshot-clip-rect.png'); | ||||
|     }); | ||||
|     it('should use scale for clip', async () => { | ||||
|       const {page, server} = getTestState(); | ||||
| 
 | ||||
|       await page.setViewport({width: 500, height: 500}); | ||||
|       await page.goto(server.PREFIX + '/grid.html'); | ||||
|       const screenshot = await page.screenshot({ | ||||
|         clip: { | ||||
|           x: 50, | ||||
|           y: 100, | ||||
|           width: 150, | ||||
|           height: 100, | ||||
|           scale: 2, | ||||
|         }, | ||||
|       }); | ||||
|       expect(screenshot).toBeGolden('screenshot-clip-rect-scale2.png'); | ||||
|     }); | ||||
|     it( | ||||
|       'should get screenshot bigger than the viewport', | ||||
|       async () => { | ||||
|  |  | |||
|  | @ -99,13 +99,10 @@ describe('Target', function () { | |||
|         return window.open(url); | ||||
|       }, server.CROSS_PROCESS_PREFIX + '/empty.html'), | ||||
|     ]); | ||||
| 
 | ||||
|     expect(otherPage!.url()).toEqual( | ||||
|       server.CROSS_PROCESS_PREFIX + '/empty.html' | ||||
|     ); | ||||
|     expect(page).not.toEqual(otherPage); | ||||
| 
 | ||||
|     await otherPage!.close(); | ||||
|   }); | ||||
|   it( | ||||
|     'should report when a new page is created and closed', | ||||
|  |  | |||
|  | @ -16,7 +16,7 @@ | |||
| 
 | ||||
| import expect from 'expect'; | ||||
| import path from 'path'; | ||||
| import {Frame} from '../../lib/cjs/puppeteer/common/FrameManager.js'; | ||||
| import {Frame} from '../../lib/cjs/puppeteer/common/Frame.js'; | ||||
| import {Page} from '../../lib/cjs/puppeteer/common/Page.js'; | ||||
| import {EventEmitter} from '../../lib/cjs/puppeteer/common/EventEmitter.js'; | ||||
| import {compare} from './golden-utils.js'; | ||||
|  |  | |||
|  | @ -15,7 +15,7 @@ | |||
|  */ | ||||
| 
 | ||||
| import expect from 'expect'; | ||||
| import {isErrorLike} from '../../lib/cjs/puppeteer/common/util.js'; | ||||
| import {isErrorLike} from '../../lib/cjs/puppeteer/util/ErrorLike.js'; | ||||
| import { | ||||
|   getTestState, | ||||
|   setupTestBrowserHooks, | ||||
|  | @ -476,7 +476,7 @@ describe('waittask specs', function () { | |||
|         await otherFrame.evaluate(addElement, 'div'); | ||||
|         await page.evaluate(addElement, 'div'); | ||||
|         const eHandle = await watchdog; | ||||
|         expect(eHandle?.executionContext().frame()).toBe(page.mainFrame()); | ||||
|         expect(eHandle?.frame).toBe(page.mainFrame()); | ||||
|       } | ||||
|     ); | ||||
| 
 | ||||
|  | @ -491,7 +491,7 @@ describe('waittask specs', function () { | |||
|       await frame1.evaluate(addElement, 'div'); | ||||
|       await frame2.evaluate(addElement, 'div'); | ||||
|       const eHandle = await waitForSelectorPromise; | ||||
|       expect(eHandle?.executionContext().frame()).toBe(frame2); | ||||
|       expect(eHandle?.frame).toBe(frame2); | ||||
|     }); | ||||
| 
 | ||||
|     it('should throw when frame is detached', async () => { | ||||
|  | @ -748,7 +748,7 @@ describe('waittask specs', function () { | |||
|       await frame1.evaluate(addElement, 'div'); | ||||
|       await frame2.evaluate(addElement, 'div'); | ||||
|       const eHandle = await waitForXPathPromise; | ||||
|       expect(eHandle?.executionContext().frame()).toBe(frame2); | ||||
|       expect(eHandle?.frame).toBe(frame2); | ||||
|     }); | ||||
|     it('should throw when frame is detached', async () => { | ||||
|       const {page, server} = getTestState(); | ||||
|  |  | |||
|  | @ -9,7 +9,7 @@ | |||
|   }, | ||||
|   "include": ["src"], | ||||
|   "references": [ | ||||
|     {"path": "../src/tsconfig.cjs.json"}, | ||||
|     {"path": "../tsconfig.lib.json"}, | ||||
|     {"path": "../utils/testserver/tsconfig.json"} | ||||
|   ] | ||||
| } | ||||
|  |  | |||
							
								
								
									
										16
									
								
								remote/test/puppeteer/tsconfig.lib.json
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										16
									
								
								remote/test/puppeteer/tsconfig.lib.json
									
									
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,16 @@ | |||
| /** | ||||
|  * This configuration only exists for the API Extractor tool and for VSCode to use. It is NOT the tsconfig used for compilation. | ||||
|  * For CJS builds, `tsconfig.cjs.json` is used, and for ESM, it's `tsconfig.esm.json`. | ||||
|  * See the details in CONTRIBUTING.md that describes our TypeScript setup. | ||||
|  */ | ||||
| { | ||||
|   "extends": "./tsconfig.base.json", | ||||
|   "compilerOptions": { | ||||
|     "composite": true | ||||
|   }, | ||||
|   "references": [ | ||||
|     {"path": "src/tsconfig.esm.json"}, | ||||
|     {"path": "src/tsconfig.cjs.json"} | ||||
|   ], | ||||
|   "exclude": ["**/*"] | ||||
| } | ||||
|  | @ -45,7 +45,7 @@ const fileExists = async filePath => { | |||
|  * place. | ||||
|  */ | ||||
| async function compileTypeScript() { | ||||
|   return exec('npm run build:tsc').catch(error => { | ||||
|   return exec('npm run build').catch(error => { | ||||
|     console.error('Error running TypeScript', error); | ||||
|     process.exit(1); | ||||
|   }); | ||||
|  |  | |||
|  | @ -1,30 +0,0 @@ | |||
| const {readdirSync, writeFileSync} = require('fs'); | ||||
| const {join, basename} = require('path'); | ||||
| 
 | ||||
| const EXCLUDE_FILES = ['puppeteer-core.ts']; | ||||
| 
 | ||||
| let typesTs = '// AUTOGENERATED - Use `utils/export_all.js` to regenerate.\n'; | ||||
| 
 | ||||
| typesTs += `\n`; | ||||
| for (const file of readdirSync(join(__dirname, `../src`)).filter(filename => { | ||||
|   return ( | ||||
|     filename.endsWith('ts') && | ||||
|     !filename.startsWith('types') && | ||||
|     !EXCLUDE_FILES.includes(filename) | ||||
|   ); | ||||
| })) { | ||||
|   typesTs += `export * from './${basename(file, '.ts')}.js';\n`; | ||||
| } | ||||
| 
 | ||||
| for (const folder of ['common', 'node', 'generated']) { | ||||
|   typesTs += `\n// Exports from \`${folder}\`\n`; | ||||
|   for (const file of readdirSync(join(__dirname, `../src/${folder}`)).filter( | ||||
|     filename => { | ||||
|       return filename.endsWith('ts') && !EXCLUDE_FILES.includes(filename); | ||||
|     } | ||||
|   )) { | ||||
|     typesTs += `export * from './${folder}/${basename(file, '.ts')}.js';\n`; | ||||
|   } | ||||
| } | ||||
| 
 | ||||
| writeFileSync(join(__dirname, '../src/types.ts'), typesTs); | ||||
							
								
								
									
										28
									
								
								remote/test/puppeteer/utils/generate_artifacts.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										28
									
								
								remote/test/puppeteer/utils/generate_artifacts.ts
									
									
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,28 @@ | |||
| #!/usr/bin/env node | ||||
| import {writeFile} from 'fs/promises'; | ||||
| import {job} from './internal/job.js'; | ||||
| import {spawnAndLog} from './internal/util.js'; | ||||
| 
 | ||||
| (async () => { | ||||
|   job('', async ({outputs}) => { | ||||
|     await writeFile(outputs[0]!, '{"type": "module"}'); | ||||
|   }) | ||||
|     .outputs(['lib/esm/package.json']) | ||||
|     .build(); | ||||
| 
 | ||||
|   job('', async ({outputs}) => { | ||||
|     spawnAndLog('api-extractor', 'run', '--local'); | ||||
|     spawnAndLog( | ||||
|       'eslint', | ||||
|       '--ext=ts', | ||||
|       '--no-ignore', | ||||
|       '--no-eslintrc', | ||||
|       '-c=.eslintrc.types.cjs', | ||||
|       '--fix', | ||||
|       outputs[0]! | ||||
|     ); | ||||
|   }) | ||||
|     .inputs(['lib/esm/puppeteer/types.d.ts']) | ||||
|     .outputs(['lib/types.d.ts', 'docs/puppeteer.api.json']) | ||||
|     .build(); | ||||
| })(); | ||||
|  | @ -14,15 +14,11 @@ | |||
|  * limitations under the License. | ||||
|  */ | ||||
| 
 | ||||
| import {readFileSync, writeFileSync} from 'fs'; | ||||
| import {join} from 'path'; | ||||
| import {chdir} from 'process'; | ||||
| import {readFile, rm, writeFile} from 'fs/promises'; | ||||
| import semver from 'semver'; | ||||
| import {versionsPerRelease} from '../versions.js'; | ||||
| import versionsArchived from '../website/versionsArchived.json'; | ||||
| 
 | ||||
| // eslint-disable-next-line import/extensions
 | ||||
| import {generateDocs} from './internal/custom_markdown_action'; | ||||
| import {generateDocs} from './internal/custom_markdown_action.js'; | ||||
| import {job} from './internal/job.js'; | ||||
| import {spawnAndLog} from './internal/util.js'; | ||||
| 
 | ||||
| function getOffsetAndLimit( | ||||
|   sectionName: string, | ||||
|  | @ -49,56 +45,73 @@ function spliceIntoSection( | |||
|   return lines.join('\n'); | ||||
| } | ||||
| 
 | ||||
| // Change to root directory
 | ||||
| chdir(join(__dirname, '..')); | ||||
| (async () => { | ||||
|   const job1 = job('', async ({inputs, outputs}) => { | ||||
|     const content = await readFile(inputs[0]!, 'utf-8'); | ||||
|     const sectionContent = ` | ||||
| --- | ||||
| sidebar_position: 1 | ||||
| --- | ||||
| `;
 | ||||
|     await writeFile(outputs[0]!, sectionContent + content); | ||||
|   }) | ||||
|     .inputs(['README.md']) | ||||
|     .outputs(['docs/index.md']) | ||||
|     .build(); | ||||
| 
 | ||||
| // README
 | ||||
| { | ||||
|   const content = readFileSync('README.md', 'utf-8'); | ||||
|   const sectionContent = ` | ||||
|  --- | ||||
|  sidebar_position: 1 | ||||
|  --- | ||||
|   // Chrome Versions
 | ||||
|   const job2 = job('', async ({inputs, outputs}) => { | ||||
|     let content = await readFile(inputs[2]!, {encoding: 'utf8'}); | ||||
|     const {versionsPerRelease} = await import(inputs[0]!); | ||||
|     const versionsArchived = JSON.parse(await readFile(inputs[1]!, 'utf8')); | ||||
| 
 | ||||
|  `;
 | ||||
|   writeFileSync('docs/index.md', sectionContent + content); | ||||
| } | ||||
| 
 | ||||
| // Chrome Versions
 | ||||
| { | ||||
|   const filename = 'docs/chromium-support.md'; | ||||
|   let content = readFileSync(filename, {encoding: 'utf8'}); | ||||
| 
 | ||||
|   // Generate versions
 | ||||
|   const buffer: string[] = []; | ||||
|   for (const [chromiumVersion, puppeteerVersion] of versionsPerRelease) { | ||||
|     if (puppeteerVersion === 'NEXT') { | ||||
|       continue; | ||||
|     // Generate versions
 | ||||
|     const buffer: string[] = []; | ||||
|     for (const [chromiumVersion, puppeteerVersion] of versionsPerRelease) { | ||||
|       if (puppeteerVersion === 'NEXT') { | ||||
|         continue; | ||||
|       } | ||||
|       if (versionsArchived.includes(puppeteerVersion.substring(1))) { | ||||
|         buffer.push( | ||||
|           `  * Chromium ${chromiumVersion} - [Puppeteer ${puppeteerVersion}](https://github.com/puppeteer/puppeteer/blob/${puppeteerVersion}/docs/api/index.md)` | ||||
|         ); | ||||
|       } else if (semver.lt(puppeteerVersion, '15.0.0')) { | ||||
|         buffer.push( | ||||
|           `  * Chromium ${chromiumVersion} - [Puppeteer ${puppeteerVersion}](https://github.com/puppeteer/puppeteer/blob/${puppeteerVersion}/docs/api.md)` | ||||
|         ); | ||||
|       } else if (semver.gte(puppeteerVersion, '15.3.0')) { | ||||
|         buffer.push( | ||||
|           `  * Chromium ${chromiumVersion} - [Puppeteer ${puppeteerVersion}](https://pptr.dev/${puppeteerVersion.slice( | ||||
|             1 | ||||
|           )})` | ||||
|         ); | ||||
|       } else { | ||||
|         buffer.push( | ||||
|           `  * Chromium ${chromiumVersion} - Puppeteer ${puppeteerVersion}` | ||||
|         ); | ||||
|       } | ||||
|     } | ||||
|     if (versionsArchived.includes(puppeteerVersion.substring(1))) { | ||||
|       buffer.push( | ||||
|         `  * Chromium ${chromiumVersion} - [Puppeteer ${puppeteerVersion}](https://github.com/puppeteer/puppeteer/blob/${puppeteerVersion}/docs/api/index.md)` | ||||
|       ); | ||||
|     } else if (semver.lt(puppeteerVersion, '15.0.0')) { | ||||
|       buffer.push( | ||||
|         `  * Chromium ${chromiumVersion} - [Puppeteer ${puppeteerVersion}](https://github.com/puppeteer/puppeteer/blob/${puppeteerVersion}/docs/api.md)` | ||||
|       ); | ||||
|     } else if (semver.gte(puppeteerVersion, '15.3.0')) { | ||||
|       buffer.push( | ||||
|         `  * Chromium ${chromiumVersion} - [Puppeteer ${puppeteerVersion}](https://pptr.dev/${puppeteerVersion.slice( | ||||
|           1 | ||||
|         )})` | ||||
|       ); | ||||
|     } else { | ||||
|       buffer.push( | ||||
|         `  * Chromium ${chromiumVersion} - Puppeteer ${puppeteerVersion}` | ||||
|       ); | ||||
|     } | ||||
|   } | ||||
|   content = spliceIntoSection('version', content, buffer.join('\n')); | ||||
|     content = spliceIntoSection('version', content, buffer.join('\n')); | ||||
| 
 | ||||
|   writeFileSync(filename, content); | ||||
| } | ||||
|     await writeFile(outputs[0]!, content); | ||||
|   }) | ||||
|     .inputs([ | ||||
|       'versions.js', | ||||
|       'website/versionsArchived.json', | ||||
|       'docs/chromium-support.md', | ||||
|     ]) | ||||
|     .outputs(['docs/chromium-support.md']) | ||||
|     .build(); | ||||
| 
 | ||||
| // Generate documentation
 | ||||
| generateDocs('docs/puppeteer.api.json', 'docs/api'); | ||||
|   await Promise.all([job1, job2]); | ||||
| 
 | ||||
|   // Generate documentation
 | ||||
|   job('', async ({inputs, outputs}) => { | ||||
|     await rm(outputs[0]!, {recursive: true, force: true}); | ||||
|     generateDocs(inputs[0]!, outputs[0]!); | ||||
|     spawnAndLog('prettier', '--ignore-path', 'none', '--write', 'docs'); | ||||
|   }) | ||||
|     .inputs(['docs/puppeteer.api.json']) | ||||
|     .outputs(['docs/api']) | ||||
|     .build(); | ||||
| })(); | ||||
|  |  | |||
							
								
								
									
										106
									
								
								remote/test/puppeteer/utils/generate_sources.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										106
									
								
								remote/test/puppeteer/utils/generate_sources.ts
									
									
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,106 @@ | |||
| #!/usr/bin/env node | ||||
| import {createHash} from 'crypto'; | ||||
| import esbuild from 'esbuild'; | ||||
| import {mkdir, mkdtemp, readFile, rm, writeFile} from 'fs/promises'; | ||||
| import {sync as glob} from 'glob'; | ||||
| import path from 'path'; | ||||
| import {job} from './internal/job.js'; | ||||
| 
 | ||||
| const INCLUDED_FOLDERS = ['common', 'node', 'generated', 'util']; | ||||
| 
 | ||||
| (async () => { | ||||
|   await job('', async ({outputs}) => { | ||||
|     await Promise.all( | ||||
|       outputs.map(outputs => { | ||||
|         return mkdir(outputs, {recursive: true}); | ||||
|       }) | ||||
|     ); | ||||
|   }) | ||||
|     .outputs(['src/generated']) | ||||
|     .build(); | ||||
| 
 | ||||
|   await job('', async ({name, inputs, outputs}) => { | ||||
|     const input = inputs.find(input => { | ||||
|       return input.endsWith('injected.ts'); | ||||
|     })!; | ||||
|     const template = await readFile( | ||||
|       inputs.find(input => { | ||||
|         return input.includes('injected.ts.tmpl'); | ||||
|       })!, | ||||
|       'utf8' | ||||
|     ); | ||||
|     const tmp = await mkdtemp(name); | ||||
|     await esbuild.build({ | ||||
|       entryPoints: [input], | ||||
|       bundle: true, | ||||
|       outdir: tmp, | ||||
|       format: 'cjs', | ||||
|       platform: 'browser', | ||||
|       target: 'ES2019', | ||||
|     }); | ||||
|     const baseName = path.basename(input); | ||||
|     const content = await readFile( | ||||
|       path.join(tmp, baseName.replace('.ts', '.js')), | ||||
|       'utf-8' | ||||
|     ); | ||||
|     const scriptContent = template.replace( | ||||
|       'SOURCE_CODE', | ||||
|       JSON.stringify(content) | ||||
|     ); | ||||
|     await writeFile(outputs[0]!, scriptContent); | ||||
|     await rm(tmp, {recursive: true, force: true}); | ||||
|   }) | ||||
|     .inputs(['src/templates/injected.ts.tmpl', 'src/injected/**/*.ts']) | ||||
|     .outputs(['src/generated/injected.ts']) | ||||
|     .build(); | ||||
| 
 | ||||
|   const sources = glob( | ||||
|     `src/{@(${INCLUDED_FOLDERS.join('|')})/*.ts,!(types|puppeteer-core).ts}` | ||||
|   ); | ||||
|   await job('', async ({outputs}) => { | ||||
|     let types = | ||||
|       '// AUTOGENERATED - Use `npm run generate:sources` to regenerate.\n\n'; | ||||
|     for (const input of sources.map(source => { | ||||
|       return `.${source.slice(3)}`; | ||||
|     })) { | ||||
|       types += `export * from '${input.replace('.ts', '.js')}';\n`; | ||||
|     } | ||||
|     await writeFile(outputs[0]!, types); | ||||
|   }) | ||||
|     .value( | ||||
|       sources | ||||
|         .reduce((hmac, value) => { | ||||
|           return hmac.update(value); | ||||
|         }, createHash('sha256')) | ||||
|         .digest('hex') | ||||
|     ) | ||||
|     .outputs(['src/types.ts']) | ||||
|     .build(); | ||||
| 
 | ||||
|   if (process.env['PUBLISH']) { | ||||
|     job('', async ({inputs}) => { | ||||
|       const version = JSON.parse(await readFile(inputs[0]!, 'utf8')).version; | ||||
|       await writeFile( | ||||
|         inputs[1]!, | ||||
|         ( | ||||
|           await readFile(inputs[1]!, { | ||||
|             encoding: 'utf-8', | ||||
|           }) | ||||
|         ).replace("'NEXT'", `'v${version}'`) | ||||
|       ); | ||||
|     }) | ||||
|       .inputs(['package.json', 'versions.js']) | ||||
|       .build(); | ||||
|   } | ||||
| 
 | ||||
|   job('', async ({inputs, outputs}) => { | ||||
|     const version = JSON.parse(await readFile(inputs[0]!, 'utf8')).version; | ||||
|     await writeFile( | ||||
|       outputs[0]!, | ||||
|       (await readFile(inputs[1]!, 'utf8')).replace('PACKAGE_VERSION', version) | ||||
|     ); | ||||
|   }) | ||||
|     .inputs(['package.json', 'src/templates/version.ts.tmpl']) | ||||
|     .outputs(['src/generated/version.ts']) | ||||
|     .build(); | ||||
| })(); | ||||
|  | @ -1,18 +0,0 @@ | |||
| const {writeFileSync, readFileSync} = require('fs'); | ||||
| const {join} = require('path'); | ||||
| 
 | ||||
| const version = require('../package.json').version; | ||||
| 
 | ||||
| writeFileSync( | ||||
|   join(__dirname, '../src/generated/version.ts'), | ||||
|   readFileSync(join(__dirname, '../src/templates/version.ts.tmpl'), { | ||||
|     encoding: 'utf-8', | ||||
|   }).replace('PACKAGE_VERSION', version) | ||||
| ); | ||||
| 
 | ||||
| writeFileSync( | ||||
|   join(__dirname, '../versions.js'), | ||||
|   readFileSync(join(__dirname, '../versions.js'), { | ||||
|     encoding: 'utf-8', | ||||
|   }).replace('NEXT', `v${version}`) | ||||
| ); | ||||
|  | @ -15,9 +15,7 @@ | |||
|  */ | ||||
| 
 | ||||
| import {ApiModel} from '@microsoft/api-extractor-model'; | ||||
| 
 | ||||
| // eslint-disable-next-line import/extensions
 | ||||
| import {MarkdownDocumenter} from './custom_markdown_documenter'; | ||||
| import {MarkdownDocumenter} from './custom_markdown_documenter.js'; | ||||
| 
 | ||||
| export const generateDocs = (jsonPath: string, outputDir: string): void => { | ||||
|   const apiModel = new ApiModel(); | ||||
|  |  | |||
							
								
								
									
										161
									
								
								remote/test/puppeteer/utils/internal/job.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										161
									
								
								remote/test/puppeteer/utils/internal/job.ts
									
									
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,161 @@ | |||
| import {createHash} from 'crypto'; | ||||
| import {existsSync, Stats} from 'fs'; | ||||
| import {mkdir, readFile, stat, writeFile} from 'fs/promises'; | ||||
| import {glob} from 'glob'; | ||||
| import {tmpdir} from 'os'; | ||||
| import {dirname, join, resolve} from 'path'; | ||||
| import {chdir} from 'process'; | ||||
| 
 | ||||
| const packageRoot = resolve(join(__dirname, '..', '..')); | ||||
| chdir(packageRoot); | ||||
| 
 | ||||
| interface JobContext { | ||||
|   name: string; | ||||
|   inputs: string[]; | ||||
|   outputs: string[]; | ||||
| } | ||||
| 
 | ||||
| class JobBuilder { | ||||
|   #inputs: string[] = []; | ||||
|   #outputs: string[] = []; | ||||
|   #callback: (ctx: JobContext) => Promise<void>; | ||||
|   #name: string; | ||||
|   #value = ''; | ||||
|   #force = false; | ||||
| 
 | ||||
|   constructor(name: string, callback: (ctx: JobContext) => Promise<void>) { | ||||
|     this.#name = name; | ||||
|     this.#callback = callback; | ||||
|   } | ||||
| 
 | ||||
|   get jobHash(): string { | ||||
|     return createHash('sha256').update(this.#name).digest('hex'); | ||||
|   } | ||||
| 
 | ||||
|   force() { | ||||
|     this.#force = true; | ||||
|     return this; | ||||
|   } | ||||
| 
 | ||||
|   value(value: string) { | ||||
|     this.#value = value; | ||||
|     return this; | ||||
|   } | ||||
| 
 | ||||
|   inputs(inputs: string[]): JobBuilder { | ||||
|     this.#inputs = inputs.flatMap(value => { | ||||
|       if (glob.hasMagic(value)) { | ||||
|         return glob.sync(value).map(value => { | ||||
|           // Glob doesn't support `\` on Windows, so we join here.
 | ||||
|           return join(packageRoot, value); | ||||
|         }); | ||||
|       } | ||||
|       return join(packageRoot, value); | ||||
|     }); | ||||
|     return this; | ||||
|   } | ||||
| 
 | ||||
|   outputs(outputs: string[]): JobBuilder { | ||||
|     if (!this.#name) { | ||||
|       this.#name = outputs.join(' and '); | ||||
|     } | ||||
| 
 | ||||
|     this.#outputs = outputs.map(value => { | ||||
|       return join(packageRoot, value); | ||||
|     }); | ||||
|     return this; | ||||
|   } | ||||
| 
 | ||||
|   async build(): Promise<void> { | ||||
|     console.log(`Running job ${this.#name}...`); | ||||
|     // For debugging.
 | ||||
|     if (this.#force) { | ||||
|       return this.#run(); | ||||
|     } | ||||
|     // In case we deleted an output file on purpose.
 | ||||
|     if (!this.getOutputStats()) { | ||||
|       return this.#run(); | ||||
|     } | ||||
|     // Run if the job has a value, but it changes.
 | ||||
|     if (this.#value) { | ||||
|       if (!(await this.isValueDifferent())) { | ||||
|         return; | ||||
|       } | ||||
|       return this.#run(); | ||||
|     } | ||||
|     // Always run when there is no output.
 | ||||
|     if (!this.#outputs.length) { | ||||
|       return this.#run(); | ||||
|     } | ||||
|     // Make-like comparator.
 | ||||
|     if (!(await this.areInputsNewer())) { | ||||
|       return; | ||||
|     } | ||||
|     return this.#run(); | ||||
|   } | ||||
| 
 | ||||
|   async isValueDifferent(): Promise<boolean> { | ||||
|     const file = join(tmpdir(), `puppeteer/${this.jobHash}.txt`); | ||||
|     await mkdir(dirname(file), {recursive: true}); | ||||
|     if (!existsSync(file)) { | ||||
|       await writeFile(file, this.#value); | ||||
|       return true; | ||||
|     } | ||||
|     return this.#value !== (await readFile(file, 'utf8')); | ||||
|   } | ||||
| 
 | ||||
|   #outputStats?: Stats[]; | ||||
|   async getOutputStats(): Promise<Stats[] | undefined> { | ||||
|     if (this.#outputStats) { | ||||
|       return this.#outputStats; | ||||
|     } | ||||
|     try { | ||||
|       this.#outputStats = await Promise.all( | ||||
|         this.#outputs.map(output => { | ||||
|           return stat(output); | ||||
|         }) | ||||
|       ); | ||||
|     } catch {} | ||||
|     return this.#outputStats; | ||||
|   } | ||||
| 
 | ||||
|   async areInputsNewer(): Promise<boolean> { | ||||
|     const inputStats = await Promise.all( | ||||
|       this.#inputs.map(input => { | ||||
|         return stat(input); | ||||
|       }) | ||||
|     ); | ||||
|     const outputStats = await this.getOutputStats(); | ||||
|     if ( | ||||
|       outputStats && | ||||
|       outputStats.reduce(reduceMinTime, Infinity) > | ||||
|         inputStats.reduce(reduceMaxTime, 0) | ||||
|     ) { | ||||
|       return false; | ||||
|     } | ||||
|     return true; | ||||
|   } | ||||
| 
 | ||||
|   #run(): Promise<void> { | ||||
|     return this.#callback({ | ||||
|       name: this.#name, | ||||
|       inputs: this.#inputs, | ||||
|       outputs: this.#outputs, | ||||
|     }); | ||||
|   } | ||||
| } | ||||
| 
 | ||||
| export const job = ( | ||||
|   name: string, | ||||
|   callback: (ctx: JobContext) => Promise<void> | ||||
| ): JobBuilder => { | ||||
|   return new JobBuilder(name, callback); | ||||
| }; | ||||
| 
 | ||||
| const reduceMaxTime = (time: number, stat: Stats) => { | ||||
|   return time < stat.mtimeMs ? stat.mtimeMs : time; | ||||
| }; | ||||
| 
 | ||||
| const reduceMinTime = (time: number, stat: Stats) => { | ||||
|   return time > stat.mtimeMs ? stat.mtimeMs : time; | ||||
| }; | ||||
							
								
								
									
										14
									
								
								remote/test/puppeteer/utils/internal/util.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										14
									
								
								remote/test/puppeteer/utils/internal/util.ts
									
									
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,14 @@ | |||
| import {spawnSync} from 'child_process'; | ||||
| 
 | ||||
| export const spawnAndLog = (...args: string[]): void => { | ||||
|   const {stdout, stderr} = spawnSync(args[0]!, args.slice(1), { | ||||
|     encoding: 'utf-8', | ||||
|     shell: true, | ||||
|   }); | ||||
|   if (stdout) { | ||||
|     console.log(stdout); | ||||
|   } | ||||
|   if (stderr) { | ||||
|     console.error(stderr); | ||||
|   } | ||||
| }; | ||||
							
								
								
									
										12
									
								
								remote/test/puppeteer/utils/tsconfig.json
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										12
									
								
								remote/test/puppeteer/utils/tsconfig.json
									
									
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,12 @@ | |||
| /** | ||||
|  * This configuration only exists for the API Extractor tool and for VSCode to use. It is NOT the tsconfig used for compilation. | ||||
|  * For CJS builds, `tsconfig.cjs.json` is used, and for ESM, it's `tsconfig.esm.json`. | ||||
|  * See the details in CONTRIBUTING.md that describes our TypeScript setup. | ||||
|  */ | ||||
| { | ||||
|   "extends": "../tsconfig.base.json", | ||||
|   "compilerOptions": { | ||||
|     "noEmit": true, | ||||
|     "module": "CommonJS" | ||||
|   } | ||||
| } | ||||
|  | @ -16,7 +16,8 @@ | |||
| 
 | ||||
| const versionsPerRelease = new Map([ | ||||
|   // This is a mapping from Chromium version => Puppeteer version.
 | ||||
|   // In Chromium roll patches, use 'v16.1.1' for the Puppeteer version.
 | ||||
|   // In Chromium roll patches, use `NEXT` for the Puppeteer version.
 | ||||
|   ['106.0.5249.0', 'v17.1.0'], | ||||
|   ['105.0.5173.0', 'v15.5.0'], | ||||
|   ['104.0.5109.0', 'v15.1.0'], | ||||
|   ['103.0.5059.0', 'v14.2.0'], | ||||
|  |  | |||
		Loading…
	
		Reference in a new issue
	
	 Alexandra Borovova
						Alexandra Borovova