puppeteer-core
Version:
A high-level API to control headless Chrome over the DevTools Protocol
387 lines • 10.6 kB
JavaScript
/**
* @license
* Copyright 2017 Google Inc.
* SPDX-License-Identifier: Apache-2.0
*/
import { filter, from, fromEvent, map, mergeMap, NEVER, Observable, timer, } from '../../third_party/rxjs/rxjs.js';
import { environment } from '../environment.js';
import { packageVersion } from '../generated/version.js';
import { assert } from '../util/assert.js';
import { mergeUint8Arrays } from '../util/encoding.js';
import { debug } from './Debug.js';
import { TimeoutError } from './Errors.js';
import { paperFormats } from './PDFOptions.js';
/**
* @internal
*/
export const debugError = debug('puppeteer:error');
/**
* @internal
*/
export const DEFAULT_VIEWPORT = Object.freeze({ width: 800, height: 600 });
/**
* @internal
*/
const SOURCE_URL = Symbol('Source URL for Puppeteer evaluation scripts');
/**
* @internal
*/
export class PuppeteerURL {
static INTERNAL_URL = 'pptr:internal';
static fromCallSite(functionName, site) {
const url = new PuppeteerURL();
url.#functionName = functionName;
url.#siteString = site.toString();
return url;
}
static parse = (url) => {
url = url.slice('pptr:'.length);
const [functionName = '', siteString = ''] = url.split(';');
const puppeteerUrl = new PuppeteerURL();
puppeteerUrl.#functionName = functionName;
puppeteerUrl.#siteString = decodeURIComponent(siteString);
return puppeteerUrl;
};
static isPuppeteerURL = (url) => {
return url.startsWith('pptr:');
};
#functionName;
#siteString;
get functionName() {
return this.#functionName;
}
get siteString() {
return this.#siteString;
}
toString() {
return `pptr:${[
this.#functionName,
encodeURIComponent(this.#siteString),
].join(';')}`;
}
}
/**
* @internal
*/
export const withSourcePuppeteerURLIfNone = (functionName, object) => {
if (Object.prototype.hasOwnProperty.call(object, SOURCE_URL)) {
return object;
}
const original = Error.prepareStackTrace;
Error.prepareStackTrace = (_, stack) => {
// First element is the function.
// Second element is the caller of this function.
// Third element is the caller of the caller of this function
// which is precisely what we want.
return stack[2];
};
const site = new Error().stack;
Error.prepareStackTrace = original;
return Object.assign(object, {
[SOURCE_URL]: PuppeteerURL.fromCallSite(functionName, site),
});
};
/**
* @internal
*/
export const getSourcePuppeteerURLIfAvailable = (object) => {
if (Object.prototype.hasOwnProperty.call(object, SOURCE_URL)) {
return object[SOURCE_URL];
}
return undefined;
};
/**
* @internal
*/
export const isString = (obj) => {
return typeof obj === 'string' || obj instanceof String;
};
/**
* @internal
*/
export const isNumber = (obj) => {
return typeof obj === 'number' || obj instanceof Number;
};
/**
* @internal
*/
export const isPlainObject = (obj) => {
return typeof obj === 'object' && obj?.constructor === Object;
};
/**
* @internal
*/
export const isRegExp = (obj) => {
return typeof obj === 'object' && obj?.constructor === RegExp;
};
/**
* @internal
*/
export const isDate = (obj) => {
return typeof obj === 'object' && obj?.constructor === Date;
};
/**
* @internal
*/
export function evaluationString(
// eslint-disable-next-line @typescript-eslint/no-unsafe-function-type
fun, ...args) {
if (isString(fun)) {
assert(args.length === 0, 'Cannot evaluate a string with arguments');
return fun;
}
function serializeArgument(arg) {
if (Object.is(arg, undefined)) {
return 'undefined';
}
return JSON.stringify(arg);
}
return `(${fun})(${args.map(serializeArgument).join(',')})`;
}
/**
* @internal
*/
export async function getReadableAsTypedArray(readable, path) {
const buffers = [];
const reader = readable.getReader();
if (path) {
const fileHandle = await environment.value.fs.promises.open(path, 'w+');
try {
while (true) {
const { done, value } = await reader.read();
if (done) {
break;
}
buffers.push(value);
await fileHandle.writeFile(value);
}
}
finally {
await fileHandle.close();
}
}
else {
while (true) {
const { done, value } = await reader.read();
if (done) {
break;
}
buffers.push(value);
}
}
try {
const concat = mergeUint8Arrays(buffers);
if (concat.length === 0) {
return null;
}
return concat;
}
catch (error) {
debugError(error);
return null;
}
}
/**
* @internal
*/
/**
* @internal
*/
export async function getReadableFromProtocolStream(client, handle) {
return new ReadableStream({
async pull(controller) {
function getUnit8Array(data, isBase64) {
if (isBase64) {
return Uint8Array.from(atob(data), m => {
return m.codePointAt(0);
});
}
const encoder = new TextEncoder();
return encoder.encode(data);
}
const { data, base64Encoded, eof } = await client.send('IO.read', {
handle,
});
controller.enqueue(getUnit8Array(data, base64Encoded ?? false));
if (eof) {
await client.send('IO.close', { handle });
controller.close();
}
},
});
}
/**
* @internal
*/
export function validateDialogType(type) {
let dialogType = null;
const validDialogTypes = new Set([
'alert',
'confirm',
'prompt',
'beforeunload',
]);
if (validDialogTypes.has(type)) {
dialogType = type;
}
assert(dialogType, `Unknown javascript dialog type: ${type}`);
return dialogType;
}
/**
* @internal
*/
export function timeout(ms, cause) {
return ms === 0
? NEVER
: timer(ms).pipe(map(() => {
throw new TimeoutError(`Timed out after waiting ${ms}ms`, { cause });
}));
}
/**
* @internal
*/
export const UTILITY_WORLD_NAME = '__puppeteer_utility_world__' + packageVersion;
/**
* @internal
*/
export const SOURCE_URL_REGEX = /^[\x20\t]*\/\/[@#] sourceURL=\s{0,10}(\S*?)\s{0,10}$/m;
/**
* @internal
*/
export function getSourceUrlComment(url) {
return `//# sourceURL=${url}`;
}
/**
* @internal
*/
export const NETWORK_IDLE_TIME = 500;
/**
* @internal
*/
export function parsePDFOptions(options = {}, lengthUnit = 'in') {
const defaults = {
scale: 1,
displayHeaderFooter: false,
headerTemplate: '',
footerTemplate: '',
printBackground: false,
landscape: false,
pageRanges: '',
preferCSSPageSize: false,
omitBackground: false,
outline: false,
tagged: true,
waitForFonts: true,
};
let width = 8.5;
let height = 11;
if (options.format) {
const format = paperFormats[options.format.toLowerCase()][lengthUnit];
assert(format, 'Unknown paper format: ' + options.format);
width = format.width;
height = format.height;
}
else {
width = convertPrintParameterToInches(options.width, lengthUnit) ?? width;
height =
convertPrintParameterToInches(options.height, lengthUnit) ?? height;
}
const margin = {
top: convertPrintParameterToInches(options.margin?.top, lengthUnit) || 0,
left: convertPrintParameterToInches(options.margin?.left, lengthUnit) || 0,
bottom: convertPrintParameterToInches(options.margin?.bottom, lengthUnit) || 0,
right: convertPrintParameterToInches(options.margin?.right, lengthUnit) || 0,
};
// Quirk https://bugs.chromium.org/p/chromium/issues/detail?id=840455#c44
if (options.outline) {
options.tagged = true;
}
return {
...defaults,
...options,
width,
height,
margin,
};
}
/**
* @internal
*/
export const unitToPixels = {
px: 1,
in: 96,
cm: 37.8,
mm: 3.78,
};
function convertPrintParameterToInches(parameter, lengthUnit = 'in') {
if (typeof parameter === 'undefined') {
return undefined;
}
let pixels;
if (isNumber(parameter)) {
// Treat numbers as pixel values to be aligned with phantom's paperSize.
pixels = parameter;
}
else if (isString(parameter)) {
const text = parameter;
let unit = text.substring(text.length - 2).toLowerCase();
let valueText = '';
if (unit in unitToPixels) {
valueText = text.substring(0, text.length - 2);
}
else {
// In case of unknown unit try to parse the whole parameter as number of pixels.
// This is consistent with phantom's paperSize behavior.
unit = 'px';
valueText = text;
}
const value = Number(valueText);
assert(!isNaN(value), 'Failed to parse parameter value: ' + text);
pixels = value * unitToPixels[unit];
}
else {
throw new Error('page.pdf() Cannot handle parameter type: ' + typeof parameter);
}
return pixels / unitToPixels[lengthUnit];
}
/**
* @internal
*/
export function fromEmitterEvent(emitter, eventName) {
return new Observable(subscriber => {
const listener = (event) => {
subscriber.next(event);
};
emitter.on(eventName, listener);
return () => {
emitter.off(eventName, listener);
};
});
}
/**
* @internal
*/
export function fromAbortSignal(signal, cause) {
return signal
? fromEvent(signal, 'abort').pipe(map(() => {
if (signal.reason instanceof Error) {
signal.reason.cause = cause;
throw signal.reason;
}
throw new Error(signal.reason, { cause });
}))
: NEVER;
}
/**
* @internal
*/
export function filterAsync(predicate) {
return mergeMap((value) => {
return from(Promise.resolve(predicate(value))).pipe(filter(isMatch => {
return isMatch;
}), map(() => {
return value;
}));
});
}
//# sourceMappingURL=util.js.map