UNPKG

@luminati-io/webdriverio8

Version:

Next-gen browser and mobile automation test framework for Node.js

268 lines 24.2 kB
import fs from 'node:fs/promises'; import path from 'node:path'; import logger from '@wdio/logger'; import Interception from './index.js'; import { containsHeaderObject } from '../index.js'; import { ERROR_REASON } from '../../constants.js'; import { CDP_SESSIONS, SESSION_MOCKS } from '../../commands/browser/mock.js'; const log = logger('webdriverio'); export default class DevtoolsInterception extends Interception { restored = false; static handleRequestInterception(client, mocks) { return async (event) => { // responseHeaders and responseStatusCode are only present in Response stage // https://chromedevtools.github.io/devtools-protocol/tot/Fetch/#event-requestPaused const isRequest = !event.responseHeaders && !event.responseErrorReason; event.responseStatusCode ||= event.responseErrorReason ? 0 : 200; event.responseHeaders ||= []; const responseHeaders = event.responseHeaders.reduce((headers, { name, value }) => { headers[name] = value; return headers; }, {}); let mockResponded = false; for (const mock of mocks) { /** * skip response mocks in Request stage */ if (isRequest && (mock.respondOverwrites.length === 0 || // nothing to do in Request stage (!mock.respondOverwrites[0].errorReason && // skip if not going to abort a request // or want to fetch response mock.respondOverwrites[0].params && mock.respondOverwrites[0].params.fetchResponse !== false))) { continue; } /** * match mock url */ if (!Interception.isMatchingRequest(mock.url, event.request.url)) { continue; } /** * Add statusCode and responseHeaders to request to be used in expect-webdriverio */ event.request.statusCode = event.responseStatusCode; event.request.responseHeaders = { ...responseHeaders }; /** * match filter options */ if (filterMethod(event.request.method, mock.filterOptions.method) || filterHeaders(event.request.headers, mock.filterOptions.requestHeaders) || filterHeaders(responseHeaders, mock.filterOptions.headers) || filterRequest(event.request.postData, mock.filterOptions.postData) || filterStatusCode(event.responseStatusCode, mock.filterOptions.statusCode)) { continue; } mock.emit('request', event); const { requestId, request, responseStatusCode } = event; const { body, base64Encoded = undefined } = isRequest ? { body: '' } : await client.send('Fetch.getResponseBody', { requestId }).catch(/* istanbul ignore next */ () => ({})); request.body = base64Encoded ? Buffer.from(body, 'base64').toString('utf8') : body; const contentTypeHeader = Object.keys(responseHeaders).find(h => h.toLowerCase() === 'content-type') || ''; const responseContentType = responseHeaders[contentTypeHeader]; request.body = responseContentType && responseContentType.includes('application/json') ? tryParseJson(request.body) : request.body; mock.matches.push(request); /** * no stubbing if no overwrites were defined */ if (mockResponded || mock.respondOverwrites.length === 0) { mock.emit('match', request); mock.emit('continue', requestId); continue; } const { errorReason, overwrite, params = {} } = mock.respondOverwrites[0].sticky ? mock.respondOverwrites[0] : mock.respondOverwrites.shift() || {}; /** * when response is modified */ if (overwrite !== undefined) { let newBody = overwrite; if (typeof overwrite === 'function') { newBody = await overwrite(request, client); } const isBodyUndefined = typeof newBody === 'undefined'; if (isBodyUndefined) { newBody = ''; } if (typeof newBody !== 'string') { newBody = JSON.stringify(newBody); } let responseCode = typeof params.statusCode === 'function' ? params.statusCode(request) : params.statusCode || responseStatusCode; let responseHeaders = [ ...event.responseHeaders, ...Object.entries(typeof params.headers === 'function' ? params.headers(request) : params.headers || {}).map(([name, value]) => ({ name, value })) ]; /** * check if local file and load it */ const responseFilePath = path.isAbsolute(newBody) ? newBody : path.join(process.cwd(), newBody); const responseFileAccessible = await fs.access(responseFilePath).then(() => true, () => false); if (newBody.length > 0 && responseFileAccessible) { newBody = (await fs.readFile(responseFilePath)); } else if (newBody.startsWith('http')) { responseCode = 301; /** * filter out possible available location header */ responseHeaders = responseHeaders.filter(({ name }) => name.toLowerCase() !== 'location'); responseHeaders.push({ name: 'Location', value: newBody }); } request.mockedResponse = newBody; mock.emit('match', request); const overwriteData = { requestId, responseCode, responseHeaders, body: isBodyUndefined ? undefined : newBody }; mock.emit('overwrite', overwriteData); mockResponded = true; const body = typeof overwriteData.body === 'undefined' ? undefined : (overwriteData.body instanceof Buffer ? overwriteData.body : Buffer.from(overwriteData.body, 'utf8')).toString('base64'); await client.send('Fetch.fulfillRequest', { ...overwriteData, body }).catch(/* istanbul ignore next */ logFetchError); continue; } /** * when request is aborted */ if (errorReason) { const failData = { requestId, errorReason }; mock.emit('fail', failData); mockResponded = true; await client.send('Fetch.failRequest', failData).catch(/* istanbul ignore next */ logFetchError); continue; } } if (!mockResponded) { return client.send('Fetch.continueRequest', { requestId: event.requestId }).catch(/* istanbul ignore next */ logFetchError); } }; } /** * allows access to all requests made with given pattern */ get calls() { return this.matches; } /** * Resets all information stored in the `mock.calls` set. */ clear() { this.matches = []; } /** * Does everything that `mock.clear()` does, and also * removes any mocked return values or implementations. * Restored mock does not emit events and could not mock responses */ async restore(sessionMocks = SESSION_MOCKS, cdpSessions = CDP_SESSIONS) { this.clear(); this.respondOverwrites = []; this.restored = true; const handle = await this.browser.getWindowHandle(); log.trace(`Restoring mock for ${handle}`); sessionMocks[handle].delete(this); if (sessionMocks[handle].size) { return; } log.trace(`Disabling fetch domain for ${handle}`); return cdpSessions[handle].send('Fetch.disable') .then(() => { delete sessionMocks[handle]; delete cdpSessions[handle]; }).catch(/* istanbul ignore next */ logFetchError); } /** * Always respond with same overwrite * @param {*} overwrites payload to overwrite the response * @param {*} params additional respond parameters to overwrite */ respond(overwrite, params = {}) { this.ensureNotRestored(); this.respondOverwrites.push({ overwrite, params, sticky: true }); } /** * Respond request once with given overwrite * @param {*} overwrites payload to overwrite the response * @param {*} params additional respond parameters to overwrite */ respondOnce(overwrite, params = {}) { this.ensureNotRestored(); this.respondOverwrites.push({ overwrite, params }); } /** * Abort the request with an error code * @param {string} errorCode error code of the response */ abort(errorReason, sticky = true) { this.ensureNotRestored(); if (typeof errorReason !== 'string' || !ERROR_REASON.includes(errorReason)) { throw new Error(`Invalid value for errorReason, allowed are: ${ERROR_REASON.join(', ')}`); } this.respondOverwrites.push({ errorReason, sticky }); } /** * Abort the request once with an error code * @param {string} errorReason error code of the response */ abortOnce(errorReason) { this.abort(errorReason, false); } ensureNotRestored() { if (this.restored) { throw new Error('This can\'t be done on restored mock'); } } } const filterMethod = (method, expected) => { if (typeof expected === 'undefined') { return false; } if (typeof expected === 'function') { return expected(method) !== true; } return expected.toLowerCase() !== method.toLowerCase(); }; const filterHeaders = (responseHeaders, expected) => { if (typeof expected === 'undefined') { return false; } if (typeof expected === 'function') { return expected(responseHeaders) !== true; } return !containsHeaderObject(responseHeaders, expected); }; const filterRequest = (postData, expected) => { if (typeof expected === 'undefined') { return false; } if (typeof expected === 'function') { return expected(postData) !== true; } return postData !== expected; }; const filterStatusCode = (statusCode, expected) => { if (typeof expected === 'undefined') { return false; } if (typeof expected === 'function') { return expected(statusCode) !== true; } return statusCode !== expected; }; const tryParseJson = (body) => { try { return JSON.parse(body) || body; } catch { return body; } }; const logFetchError = (err) => { /* istanbul ignore next */ log.debug(err?.message); }; //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"devtools.js","sourceRoot":"","sources":["../../../src/utils/interception/devtools.ts"],"names":[],"mappings":"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"}