@luminati-io/webdriverio8
Version:
Next-gen browser and mobile automation test framework for Node.js
268 lines • 24.2 kB
JavaScript
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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"}