2020-05-02 08:16:03 +00:00
|
|
|
import { Url } from 'url';
|
2021-05-27 12:13:31 +00:00
|
|
|
import { afterAll, afterEach, beforeAll } from '@jest/globals';
|
2021-06-16 07:13:33 +00:00
|
|
|
// eslint-disable-next-line no-restricted-imports
|
2020-05-02 08:16:03 +00:00
|
|
|
import nock from 'nock';
|
2021-07-21 19:50:13 +00:00
|
|
|
import { makeGraphqlSnapshot } from './graphql-snapshot';
|
2020-05-02 08:16:03 +00:00
|
|
|
|
2021-06-16 07:13:33 +00:00
|
|
|
// eslint-disable-next-line no-restricted-imports
|
|
|
|
export type { Scope, ReplyHeaders } from 'nock';
|
2020-05-02 08:16:03 +00:00
|
|
|
|
|
|
|
interface RequestLogItem {
|
|
|
|
headers: Record<string, string>;
|
|
|
|
method: string;
|
|
|
|
url: string;
|
|
|
|
body?: any;
|
2020-05-05 12:57:05 +00:00
|
|
|
graphql?: any;
|
2020-05-02 08:16:03 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
type BasePath = string | RegExp | Url;
|
|
|
|
|
|
|
|
let requestLog: RequestLogItem[] = [];
|
|
|
|
let missingLog: string[] = [];
|
|
|
|
|
2020-08-19 04:46:00 +00:00
|
|
|
type TestRequest = {
|
|
|
|
method: string;
|
|
|
|
href: string;
|
|
|
|
};
|
|
|
|
|
|
|
|
function onMissing(req: TestRequest, opts?: TestRequest): void {
|
2021-03-04 05:21:55 +00:00
|
|
|
if (opts) {
|
2020-06-07 11:00:49 +00:00
|
|
|
missingLog.push(` ${opts.method} ${opts.href}`);
|
2021-03-04 05:21:55 +00:00
|
|
|
} else {
|
|
|
|
missingLog.push(` ${req.method} ${req.href}`);
|
2020-06-07 11:00:49 +00:00
|
|
|
}
|
2020-05-02 08:16:03 +00:00
|
|
|
}
|
|
|
|
|
2021-05-27 12:13:31 +00:00
|
|
|
export function allUsed(): boolean {
|
|
|
|
return nock.isDone();
|
2020-05-02 08:16:03 +00:00
|
|
|
}
|
|
|
|
|
2021-05-27 12:13:31 +00:00
|
|
|
/**
|
|
|
|
* Clear nock state. Will be called in `afterEach`
|
|
|
|
* @argument throwOnPending Use `false` to simply clear mocks.
|
|
|
|
*/
|
|
|
|
export function clear(throwOnPending = true): void {
|
|
|
|
const isDone = nock.isDone();
|
|
|
|
const pending = nock.pendingMocks();
|
2020-05-02 08:16:03 +00:00
|
|
|
nock.abortPendingRequests();
|
|
|
|
nock.cleanAll();
|
2021-08-09 17:37:28 +00:00
|
|
|
const missing = missingLog;
|
2020-05-02 08:16:03 +00:00
|
|
|
requestLog = [];
|
|
|
|
missingLog = [];
|
2021-05-27 12:13:31 +00:00
|
|
|
if (!isDone && throwOnPending) {
|
|
|
|
throw new Error(`Pending mocks!\n * ${pending.join('\n * ')}`);
|
|
|
|
}
|
2021-08-09 17:37:28 +00:00
|
|
|
if (missing.length && throwOnPending) {
|
|
|
|
throw new Error(`Missing mocks!\n * ${missing.join('\n * ')}`);
|
|
|
|
}
|
2020-07-27 09:24:41 +00:00
|
|
|
}
|
|
|
|
|
2020-05-02 08:16:03 +00:00
|
|
|
export function scope(basePath: BasePath, options?: nock.Options): nock.Scope {
|
|
|
|
return nock(basePath, options).on('request', (req) => {
|
|
|
|
const { headers, method } = req;
|
|
|
|
const url = req.options?.href;
|
|
|
|
const result: RequestLogItem = { headers, method, url };
|
2021-08-06 12:02:37 +00:00
|
|
|
const requestBody = req.requestBodyBuffers?.[0]?.toString();
|
2020-07-10 18:51:40 +00:00
|
|
|
|
2021-08-06 12:02:37 +00:00
|
|
|
if (requestBody && headers['content-type'] === 'application/json') {
|
2020-05-05 12:57:05 +00:00
|
|
|
try {
|
2021-08-06 12:02:37 +00:00
|
|
|
const body = JSON.parse(requestBody);
|
|
|
|
const graphql = makeGraphqlSnapshot(body);
|
|
|
|
if (graphql) {
|
|
|
|
result.graphql = graphql;
|
|
|
|
} else {
|
|
|
|
result.body = body;
|
|
|
|
}
|
|
|
|
} catch (e) {
|
|
|
|
result.body = requestBody;
|
2020-05-05 12:57:05 +00:00
|
|
|
}
|
2020-05-02 08:16:03 +00:00
|
|
|
}
|
|
|
|
requestLog.push(result);
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
export function getTrace(): RequestLogItem[] /* istanbul ignore next */ {
|
|
|
|
const errorLines = [];
|
|
|
|
if (missingLog.length) {
|
|
|
|
errorLines.push('Missing mocks:');
|
|
|
|
errorLines.push(...missingLog);
|
|
|
|
}
|
|
|
|
if (!nock.isDone()) {
|
|
|
|
errorLines.push('Unused mocks:');
|
|
|
|
errorLines.push(...nock.pendingMocks().map((x) => ` ${x}`));
|
|
|
|
}
|
|
|
|
if (errorLines.length) {
|
|
|
|
throw new Error(
|
|
|
|
[
|
|
|
|
'Completed requests:',
|
|
|
|
...requestLog.map(({ method, url }) => ` ${method} ${url}`),
|
|
|
|
...errorLines,
|
|
|
|
].join('\n')
|
|
|
|
);
|
|
|
|
}
|
|
|
|
return requestLog;
|
|
|
|
}
|
2021-05-27 12:13:31 +00:00
|
|
|
|
|
|
|
// init nock
|
|
|
|
beforeAll(() => {
|
|
|
|
nock.emitter.on('no match', onMissing);
|
|
|
|
nock.disableNetConnect();
|
|
|
|
});
|
|
|
|
|
|
|
|
// clean nock to clear memory leack from http module patching
|
|
|
|
afterAll(() => {
|
|
|
|
nock.emitter.removeListener('no match', onMissing);
|
|
|
|
nock.restore();
|
|
|
|
});
|
|
|
|
|
|
|
|
// clear nock state
|
|
|
|
afterEach(() => {
|
|
|
|
clear();
|
|
|
|
});
|