@browserstack/testcafe
Version:
Automated browser testing for the modern web development stack.
131 lines • 21.8 kB
JavaScript
;
var __importDefault = (this && this.__importDefault) || function (mod) {
return (mod && mod.__esModule) ? mod : { "default": mod };
};
Object.defineProperty(exports, "__esModule", { value: true });
const path_1 = __importDefault(require("path"));
const os_1 = __importDefault(require("os"));
const chrome_remote_interface_1 = __importDefault(require("chrome-remote-interface"));
const client_functions_1 = require("../../../utils/client-functions");
const DOWNLOADS_DIR = path_1.default.join(os_1.default.homedir(), 'Downloads');
async function getActiveTab(cdpPort, browserId) {
const tabs = await chrome_remote_interface_1.default.listTabs({ port: cdpPort });
return tabs.filter(t => t.type === 'page' && t.url.includes(browserId))[0];
}
async function setEmulationBounds({ client, config, viewportSize, emulatedDevicePixelRatio }) {
await setDeviceMetricsOverride(client, viewportSize.width, viewportSize.height, emulatedDevicePixelRatio, config.mobile);
await client.Emulation.setVisibleSize({ width: viewportSize.width, height: viewportSize.height });
}
async function setEmulation(runtimeInfo) {
const { client, config } = runtimeInfo;
if (config.userAgent !== void 0)
await client.Network.setUserAgentOverride({ userAgent: config.userAgent });
if (config.touch !== void 0) {
const touchConfig = {
enabled: config.touch,
configuration: config.mobile ? 'mobile' : 'desktop',
maxTouchPoints: 1
};
if (client.Emulation.setEmitTouchEventsForMouse)
await client.Emulation.setEmitTouchEventsForMouse(touchConfig);
if (client.Emulation.setTouchEmulationEnabled)
await client.Emulation.setTouchEmulationEnabled(touchConfig);
}
await resizeWindow({ width: config.width, height: config.height }, runtimeInfo);
}
async function enableDownloads({ client }) {
await client.Page.setDownloadBehavior({
behavior: 'allow',
downloadPath: DOWNLOADS_DIR
});
}
async function getScreenshotData({ client, config, emulatedDevicePixelRatio }, fullPage) {
let viewportWidth = 0;
let viewportHeight = 0;
if (fullPage) {
const { contentSize, visualViewport } = await client.Page.getLayoutMetrics();
await setDeviceMetricsOverride(client, Math.ceil(contentSize.width), Math.ceil(contentSize.height), emulatedDevicePixelRatio, config.mobile);
viewportWidth = visualViewport.clientWidth;
viewportHeight = visualViewport.clientHeight;
}
const screenshotData = await client.Page.captureScreenshot({});
if (fullPage) {
if (config.emulation) {
await setDeviceMetricsOverride(client, config.width || viewportWidth, config.height || viewportHeight, emulatedDevicePixelRatio, config.mobile);
}
else
await client.Emulation.clearDeviceMetricsOverride();
}
return Buffer.from(screenshotData.data, 'base64');
}
exports.getScreenshotData = getScreenshotData;
async function setDeviceMetricsOverride(client, width, height, deviceScaleFactor, mobile) {
await client.Emulation.setDeviceMetricsOverride({
width,
height,
deviceScaleFactor,
mobile,
// @ts-ignore
fitWindow: false
});
}
async function createClient(runtimeInfo) {
const { browserId, config, cdpPort } = runtimeInfo;
let tab = null;
let client = null;
try {
tab = await getActiveTab(cdpPort, browserId);
if (!tab)
return;
client = await chrome_remote_interface_1.default({ target: tab, port: cdpPort });
}
catch (e) {
return;
}
runtimeInfo.tab = tab;
runtimeInfo.client = client;
await client.Page.enable();
await client.Network.enable({});
await client.Runtime.enable();
const devicePixelRatioQueryResult = await client.Runtime.evaluate({ expression: 'window.devicePixelRatio' });
runtimeInfo.originalDevicePixelRatio = devicePixelRatioQueryResult.result.value;
runtimeInfo.emulatedDevicePixelRatio = config.scaleFactor || runtimeInfo.originalDevicePixelRatio;
if (config.emulation)
await setEmulation(runtimeInfo);
if (config.headless)
await enableDownloads(runtimeInfo);
}
exports.createClient = createClient;
function isHeadlessTab({ tab, config }) {
return tab && config.headless;
}
exports.isHeadlessTab = isHeadlessTab;
async function closeTab({ tab, cdpPort }) {
await chrome_remote_interface_1.default.closeTab({ id: tab.id, port: cdpPort });
}
exports.closeTab = closeTab;
async function updateMobileViewportSize(runtimeInfo) {
const windowDimensionsQueryResult = await runtimeInfo.client.Runtime.evaluate({
expression: `(${client_functions_1.GET_WINDOW_DIMENSIONS_INFO_SCRIPT})()`,
returnByValue: true
});
const windowDimensions = windowDimensionsQueryResult.result.value;
runtimeInfo.viewportSize.width = windowDimensions.outerWidth;
runtimeInfo.viewportSize.height = windowDimensions.outerHeight;
}
exports.updateMobileViewportSize = updateMobileViewportSize;
async function resizeWindow(newDimensions, runtimeInfo) {
const { browserId, config, viewportSize, providerMethods } = runtimeInfo;
const currentWidth = viewportSize.width;
const currentHeight = viewportSize.height;
const newWidth = newDimensions.width || currentWidth;
const newHeight = newDimensions.height || currentHeight;
if (!config.headless)
await providerMethods.resizeLocalBrowserWindow(browserId, newWidth, newHeight, currentWidth, currentHeight);
viewportSize.width = newWidth;
viewportSize.height = newHeight;
if (config.emulation)
await setEmulationBounds(runtimeInfo);
}
exports.resizeWindow = resizeWindow;
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"cdp.js","sourceRoot":"","sources":["../../../../../../src/browser/provider/built-in/dedicated/chrome/cdp.ts"],"names":[],"mappings":"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","sourcesContent":["import path from 'path';\nimport os from 'os';\nimport remoteChrome from 'chrome-remote-interface';\nimport { GET_WINDOW_DIMENSIONS_INFO_SCRIPT } from '../../../utils/client-functions';\n\ninterface Size {\n    width: number;\n    height: number;\n}\n\ninterface Config {\n    headless: boolean;\n    mobile: boolean;\n    emulation: false;\n    userAgent?: string;\n    touch?: boolean;\n    width: number;\n    height: number;\n    scaleFactor: number;\n}\n\ninterface ProviderMethods {\n    resizeLocalBrowserWindow (browserId: string, newWidth: number, newHeight: number, currentWidth: number, currentHeight: number): Promise<void>;\n}\n\ninterface RuntimeInfo {\n    browserId: string;\n    cdpPort: number;\n    client: remoteChrome.ProtocolApi;\n    tab: remoteChrome.TargetInfo;\n    config: Config;\n    viewportSize: Size;\n    emulatedDevicePixelRatio: number;\n    originalDevicePixelRatio: number;\n    providerMethods: ProviderMethods;\n}\n\ninterface TouchConfigOptions {\n    enabled: boolean;\n    configuration: 'desktop' | 'mobile';\n    maxTouchPoints: number;\n}\n\nconst DOWNLOADS_DIR = path.join(os.homedir(), 'Downloads');\n\nasync function getActiveTab (cdpPort: number, browserId: string): Promise<remoteChrome.TargetInfo> {\n    const tabs = await remoteChrome.listTabs({ port: cdpPort });\n\n    return tabs.filter(t => t.type === 'page' && t.url.includes(browserId))[0];\n}\n\nasync function setEmulationBounds ({ client, config, viewportSize, emulatedDevicePixelRatio }: RuntimeInfo): Promise<void> {\n    await setDeviceMetricsOverride(client, viewportSize.width, viewportSize.height, emulatedDevicePixelRatio, config.mobile);\n\n    await client.Emulation.setVisibleSize({ width: viewportSize.width, height: viewportSize.height });\n}\n\nasync function setEmulation (runtimeInfo: RuntimeInfo): Promise<void> {\n    const { client, config } = runtimeInfo;\n\n    if (config.userAgent !== void 0)\n        await client.Network.setUserAgentOverride({ userAgent: config.userAgent });\n\n    if (config.touch !== void 0) {\n        const touchConfig: TouchConfigOptions = {\n            enabled:        config.touch,\n            configuration:  config.mobile ? 'mobile' : 'desktop',\n            maxTouchPoints: 1\n        };\n\n        if (client.Emulation.setEmitTouchEventsForMouse)\n            await client.Emulation.setEmitTouchEventsForMouse(touchConfig);\n\n        if (client.Emulation.setTouchEmulationEnabled)\n            await client.Emulation.setTouchEmulationEnabled(touchConfig);\n    }\n\n    await resizeWindow({ width: config.width, height: config.height }, runtimeInfo);\n}\n\nasync function enableDownloads ({ client }: RuntimeInfo): Promise<void> {\n    await client.Page.setDownloadBehavior({\n        behavior:     'allow',\n        downloadPath: DOWNLOADS_DIR\n    });\n}\n\nexport async function getScreenshotData ({ client, config, emulatedDevicePixelRatio }: RuntimeInfo, fullPage?: boolean): Promise<Buffer> {\n    let viewportWidth = 0;\n    let viewportHeight = 0;\n\n    if (fullPage) {\n        const { contentSize, visualViewport } = await client.Page.getLayoutMetrics();\n\n        await setDeviceMetricsOverride(\n            client,\n            Math.ceil(contentSize.width),\n            Math.ceil(contentSize.height),\n            emulatedDevicePixelRatio,\n            config.mobile);\n\n        viewportWidth = visualViewport.clientWidth;\n        viewportHeight = visualViewport.clientHeight;\n    }\n\n    const screenshotData = await client.Page.captureScreenshot({});\n\n    if (fullPage) {\n        if (config.emulation) {\n            await setDeviceMetricsOverride(\n                client,\n                config.width || viewportWidth,\n                config.height || viewportHeight,\n                emulatedDevicePixelRatio,\n                config.mobile);\n        }\n        else\n            await client.Emulation.clearDeviceMetricsOverride();\n    }\n\n    return Buffer.from(screenshotData.data, 'base64');\n}\n\nasync function setDeviceMetricsOverride (client: remoteChrome.ProtocolApi, width: number, height: number, deviceScaleFactor: number, mobile: boolean): Promise<void> {\n    await client.Emulation.setDeviceMetricsOverride({\n        width,\n        height,\n        deviceScaleFactor,\n        mobile,\n        // @ts-ignore\n        fitWindow: false\n    });\n}\n\nexport async function createClient (runtimeInfo: RuntimeInfo): Promise<void> {\n    const { browserId, config, cdpPort } = runtimeInfo;\n\n    let tab    = null;\n    let client = null;\n\n    try {\n        tab = await getActiveTab(cdpPort, browserId);\n\n        if (!tab)\n            return;\n\n        client = await remoteChrome({ target: tab, port: cdpPort });\n    }\n    catch (e) {\n        return;\n    }\n\n    runtimeInfo.tab    = tab;\n    runtimeInfo.client = client;\n\n    await client.Page.enable();\n    await client.Network.enable({});\n    await client.Runtime.enable();\n\n    const devicePixelRatioQueryResult = await client.Runtime.evaluate({ expression: 'window.devicePixelRatio' });\n\n    runtimeInfo.originalDevicePixelRatio = devicePixelRatioQueryResult.result.value;\n    runtimeInfo.emulatedDevicePixelRatio = config.scaleFactor || runtimeInfo.originalDevicePixelRatio;\n\n    if (config.emulation)\n        await setEmulation(runtimeInfo);\n\n    if (config.headless)\n        await enableDownloads(runtimeInfo);\n}\n\nexport function isHeadlessTab ({ tab, config }: RuntimeInfo): boolean {\n    return tab && config.headless;\n}\n\nexport async function closeTab ({ tab, cdpPort }: RuntimeInfo): Promise<void> {\n    await remoteChrome.closeTab({ id: tab.id, port: cdpPort });\n}\n\nexport async function updateMobileViewportSize (runtimeInfo: RuntimeInfo): Promise<void> {\n    const windowDimensionsQueryResult = await runtimeInfo.client.Runtime.evaluate({\n        expression:    `(${GET_WINDOW_DIMENSIONS_INFO_SCRIPT})()`,\n        returnByValue: true\n    });\n\n    const windowDimensions = windowDimensionsQueryResult.result.value;\n\n    runtimeInfo.viewportSize.width  = windowDimensions.outerWidth;\n    runtimeInfo.viewportSize.height = windowDimensions.outerHeight;\n}\n\nexport async function resizeWindow (newDimensions: Size, runtimeInfo: RuntimeInfo): Promise<void> {\n    const { browserId, config, viewportSize, providerMethods } = runtimeInfo;\n\n    const currentWidth  = viewportSize.width;\n    const currentHeight = viewportSize.height;\n    const newWidth      = newDimensions.width || currentWidth;\n    const newHeight     = newDimensions.height || currentHeight;\n\n    if (!config.headless)\n        await providerMethods.resizeLocalBrowserWindow(browserId, newWidth, newHeight, currentWidth, currentHeight);\n\n    viewportSize.width  = newWidth;\n    viewportSize.height = newHeight;\n\n    if (config.emulation)\n        await setEmulationBounds(runtimeInfo);\n}\n"]}