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';
|
2020-05-05 12:57:05 +00:00
|
|
|
import is from '@sindresorhus/is';
|
|
|
|
import { parse as parseGraphqlQuery } from 'graphql/language';
|
2020-05-02 08:16:03 +00:00
|
|
|
import nock from 'nock';
|
|
|
|
|
|
|
|
export type { Scope } from 'nock';
|
|
|
|
|
|
|
|
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-05-05 12:57:05 +00:00
|
|
|
function simplifyGraphqlAST(tree: any): any {
|
|
|
|
if (!tree || is.emptyArray(tree) || is.emptyObject(tree)) {
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (is.array(tree)) {
|
|
|
|
return tree.map(simplifyGraphqlAST);
|
|
|
|
}
|
|
|
|
if (is.object(tree)) {
|
|
|
|
return [
|
|
|
|
'operation',
|
|
|
|
'definitions',
|
|
|
|
'selectionSet',
|
|
|
|
'arguments',
|
|
|
|
'value',
|
|
|
|
'alias',
|
|
|
|
'directives',
|
|
|
|
].reduce((acc: Record<string, any>, field) => {
|
|
|
|
const value = tree[field];
|
|
|
|
let simplifiedValue;
|
|
|
|
|
|
|
|
if (field === 'definitions') {
|
|
|
|
return (value || []).reduce((defsAcc, def) => {
|
|
|
|
const name = def?.operation;
|
|
|
|
const defValue = simplifyGraphqlAST(def);
|
|
|
|
if (name && defValue) {
|
|
|
|
return { ...defsAcc, [name]: defValue };
|
|
|
|
}
|
|
|
|
return defsAcc;
|
|
|
|
}, {});
|
|
|
|
}
|
|
|
|
|
|
|
|
if (field === 'arguments') {
|
|
|
|
const args = (value || []).reduce((argsAcc, arg) => {
|
|
|
|
const name = arg?.name?.value;
|
|
|
|
const argValue = arg?.value?.value;
|
|
|
|
if (name && argValue) {
|
|
|
|
return { ...argsAcc, [name]: argValue };
|
|
|
|
}
|
|
|
|
return argsAcc;
|
|
|
|
}, {});
|
|
|
|
if (!is.emptyObject(args)) {
|
|
|
|
acc.__args = args;
|
|
|
|
}
|
|
|
|
} else if (field === 'selectionSet') {
|
|
|
|
(value?.selections || []).forEach((selection) => {
|
|
|
|
const name = selection?.name?.value;
|
|
|
|
const selValue = simplifyGraphqlAST(selection);
|
|
|
|
if (name && selValue) {
|
|
|
|
acc[name] = is.emptyObject(selValue) ? null : selValue;
|
|
|
|
}
|
|
|
|
});
|
|
|
|
} else {
|
|
|
|
simplifiedValue = simplifyGraphqlAST(value);
|
|
|
|
if (simplifiedValue) {
|
|
|
|
acc[`__${field}`] = simplifiedValue;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return acc;
|
|
|
|
}, {});
|
|
|
|
}
|
|
|
|
return tree;
|
|
|
|
}
|
|
|
|
|
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();
|
|
|
|
requestLog = [];
|
|
|
|
missingLog = [];
|
2021-05-27 12:13:31 +00:00
|
|
|
if (!isDone && throwOnPending) {
|
|
|
|
throw new Error(`Pending mocks!\n * ${pending.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 };
|
2020-07-10 18:51:40 +00:00
|
|
|
const body = req.requestBodyBuffers?.[0]?.toString();
|
|
|
|
|
2020-05-02 08:16:03 +00:00
|
|
|
if (body) {
|
2020-05-05 12:57:05 +00:00
|
|
|
try {
|
|
|
|
const strQuery = JSON.parse(body).query;
|
|
|
|
const rawQuery = parseGraphqlQuery(strQuery, {
|
|
|
|
noLocation: true,
|
|
|
|
});
|
|
|
|
result.graphql = simplifyGraphqlAST(rawQuery);
|
|
|
|
} catch (ex) {
|
|
|
|
result.body = body;
|
|
|
|
}
|
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();
|
|
|
|
});
|