247 lines
7.2 KiB
JavaScript
247 lines
7.2 KiB
JavaScript
'use strict';
|
|
|
|
const { getResponseData, buildKey, addMockDispatch } = require('./mock-utils');
|
|
const {
|
|
kDispatches,
|
|
kDispatchKey,
|
|
kDefaultHeaders,
|
|
kDefaultTrailers,
|
|
kContentLength,
|
|
kMockDispatch,
|
|
kIgnoreTrailingSlash,
|
|
} = require('./mock-symbols');
|
|
const { InvalidArgumentError } = require('../core/errors');
|
|
const { serializePathWithQuery } = require('../core/util');
|
|
|
|
/**
|
|
* Defines the scope API for an interceptor reply
|
|
*/
|
|
class MockScope {
|
|
constructor(mockDispatch) {
|
|
this[kMockDispatch] = mockDispatch;
|
|
}
|
|
|
|
/**
|
|
* Delay a reply by a set amount in ms.
|
|
*/
|
|
delay(waitInMs) {
|
|
if (
|
|
typeof waitInMs !== 'number' ||
|
|
!Number.isInteger(waitInMs) ||
|
|
waitInMs <= 0
|
|
) {
|
|
throw new InvalidArgumentError('waitInMs must be a valid integer > 0');
|
|
}
|
|
|
|
this[kMockDispatch].delay = waitInMs;
|
|
return this;
|
|
}
|
|
|
|
/**
|
|
* For a defined reply, never mark as consumed.
|
|
*/
|
|
persist() {
|
|
this[kMockDispatch].persist = true;
|
|
return this;
|
|
}
|
|
|
|
/**
|
|
* Allow one to define a reply for a set amount of matching requests.
|
|
*/
|
|
times(repeatTimes) {
|
|
if (
|
|
typeof repeatTimes !== 'number' ||
|
|
!Number.isInteger(repeatTimes) ||
|
|
repeatTimes <= 0
|
|
) {
|
|
throw new InvalidArgumentError('repeatTimes must be a valid integer > 0');
|
|
}
|
|
|
|
this[kMockDispatch].times = repeatTimes;
|
|
return this;
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Defines an interceptor for a Mock
|
|
*/
|
|
class MockInterceptor {
|
|
constructor(opts, mockDispatches) {
|
|
if (typeof opts !== 'object') {
|
|
throw new InvalidArgumentError('opts must be an object');
|
|
}
|
|
if (typeof opts.path === 'undefined') {
|
|
throw new InvalidArgumentError('opts.path must be defined');
|
|
}
|
|
if (typeof opts.method === 'undefined') {
|
|
opts.method = 'GET';
|
|
}
|
|
// See https://github.com/nodejs/undici/issues/1245
|
|
// As per RFC 3986, clients are not supposed to send URI
|
|
// fragments to servers when they retrieve a document,
|
|
if (typeof opts.path === 'string') {
|
|
if (opts.query) {
|
|
opts.path = serializePathWithQuery(opts.path, opts.query);
|
|
} else {
|
|
// Matches https://github.com/nodejs/undici/blob/main/lib/web/fetch/index.js#L1811
|
|
const parsedURL = new URL(opts.path, 'data://');
|
|
opts.path = parsedURL.pathname + parsedURL.search;
|
|
}
|
|
}
|
|
if (typeof opts.method === 'string') {
|
|
opts.method = opts.method.toUpperCase();
|
|
}
|
|
|
|
this[kDispatchKey] = buildKey(opts);
|
|
this[kDispatches] = mockDispatches;
|
|
this[kIgnoreTrailingSlash] = opts.ignoreTrailingSlash ?? false;
|
|
this[kDefaultHeaders] = {};
|
|
this[kDefaultTrailers] = {};
|
|
this[kContentLength] = false;
|
|
}
|
|
|
|
createMockScopeDispatchData({ statusCode, data, responseOptions }) {
|
|
const responseData = getResponseData(data);
|
|
const contentLength =
|
|
this[kContentLength] ? { 'content-length': responseData.length } : {};
|
|
const headers = {
|
|
...this[kDefaultHeaders],
|
|
...contentLength,
|
|
...responseOptions.headers,
|
|
};
|
|
const trailers = { ...this[kDefaultTrailers], ...responseOptions.trailers };
|
|
|
|
return { statusCode, data, headers, trailers };
|
|
}
|
|
|
|
validateReplyParameters(replyParameters) {
|
|
if (typeof replyParameters.statusCode === 'undefined') {
|
|
throw new InvalidArgumentError('statusCode must be defined');
|
|
}
|
|
if (
|
|
typeof replyParameters.responseOptions !== 'object' ||
|
|
replyParameters.responseOptions === null
|
|
) {
|
|
throw new InvalidArgumentError('responseOptions must be an object');
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Mock an undici request with a defined reply.
|
|
*/
|
|
reply(replyOptionsCallbackOrStatusCode) {
|
|
// Values of reply aren't available right now as they
|
|
// can only be available when the reply callback is invoked.
|
|
if (typeof replyOptionsCallbackOrStatusCode === 'function') {
|
|
// We'll first wrap the provided callback in another function,
|
|
// this function will properly resolve the data from the callback
|
|
// when invoked.
|
|
const wrappedDefaultsCallback = (opts) => {
|
|
// Our reply options callback contains the parameter for statusCode, data and options.
|
|
const resolvedData = replyOptionsCallbackOrStatusCode(opts);
|
|
|
|
// Check if it is in the right format
|
|
if (typeof resolvedData !== 'object' || resolvedData === null) {
|
|
throw new InvalidArgumentError(
|
|
'reply options callback must return an object'
|
|
);
|
|
}
|
|
|
|
const replyParameters = {
|
|
data: '',
|
|
responseOptions: {},
|
|
...resolvedData,
|
|
};
|
|
this.validateReplyParameters(replyParameters);
|
|
// Since the values can be obtained immediately we return them
|
|
// from this higher order function that will be resolved later.
|
|
return {
|
|
...this.createMockScopeDispatchData(replyParameters),
|
|
};
|
|
};
|
|
|
|
// Add usual dispatch data, but this time set the data parameter to function that will eventually provide data.
|
|
const newMockDispatch = addMockDispatch(
|
|
this[kDispatches],
|
|
this[kDispatchKey],
|
|
wrappedDefaultsCallback,
|
|
{ ignoreTrailingSlash: this[kIgnoreTrailingSlash] }
|
|
);
|
|
return new MockScope(newMockDispatch);
|
|
}
|
|
|
|
// We can have either one or three parameters, if we get here,
|
|
// we should have 1-3 parameters. So we spread the arguments of
|
|
// this function to obtain the parameters, since replyData will always
|
|
// just be the statusCode.
|
|
const replyParameters = {
|
|
statusCode: replyOptionsCallbackOrStatusCode,
|
|
data: arguments[1] === undefined ? '' : arguments[1],
|
|
responseOptions: arguments[2] === undefined ? {} : arguments[2],
|
|
};
|
|
this.validateReplyParameters(replyParameters);
|
|
|
|
// Send in-already provided data like usual
|
|
const dispatchData = this.createMockScopeDispatchData(replyParameters);
|
|
const newMockDispatch = addMockDispatch(
|
|
this[kDispatches],
|
|
this[kDispatchKey],
|
|
dispatchData,
|
|
{ ignoreTrailingSlash: this[kIgnoreTrailingSlash] }
|
|
);
|
|
return new MockScope(newMockDispatch);
|
|
}
|
|
|
|
/**
|
|
* Mock an undici request with a defined error.
|
|
*/
|
|
replyWithError(error) {
|
|
if (typeof error === 'undefined') {
|
|
throw new InvalidArgumentError('error must be defined');
|
|
}
|
|
|
|
const newMockDispatch = addMockDispatch(
|
|
this[kDispatches],
|
|
this[kDispatchKey],
|
|
{ error },
|
|
{ ignoreTrailingSlash: this[kIgnoreTrailingSlash] }
|
|
);
|
|
return new MockScope(newMockDispatch);
|
|
}
|
|
|
|
/**
|
|
* Set default reply headers on the interceptor for subsequent replies
|
|
*/
|
|
defaultReplyHeaders(headers) {
|
|
if (typeof headers === 'undefined') {
|
|
throw new InvalidArgumentError('headers must be defined');
|
|
}
|
|
|
|
this[kDefaultHeaders] = headers;
|
|
return this;
|
|
}
|
|
|
|
/**
|
|
* Set default reply trailers on the interceptor for subsequent replies
|
|
*/
|
|
defaultReplyTrailers(trailers) {
|
|
if (typeof trailers === 'undefined') {
|
|
throw new InvalidArgumentError('trailers must be defined');
|
|
}
|
|
|
|
this[kDefaultTrailers] = trailers;
|
|
return this;
|
|
}
|
|
|
|
/**
|
|
* Set reply content length header for replies on the interceptor
|
|
*/
|
|
replyContentLength() {
|
|
this[kContentLength] = true;
|
|
return this;
|
|
}
|
|
}
|
|
|
|
module.exports.MockInterceptor = MockInterceptor;
|
|
module.exports.MockScope = MockScope;
|