renovate/test/util.ts

139 lines
4 KiB
TypeScript
Raw Normal View History

import crypto from 'crypto';
import { expect, jest } from '@jest/globals';
import type { Plugin } from 'pretty-format';
import upath from 'upath';
2020-05-01 16:03:48 +00:00
import { getConfig } from '../lib/config/defaults';
2021-12-22 08:27:24 +00:00
import type { RenovateConfig } from '../lib/config/types';
import * as _logger from '../lib/logger';
import { Platform, platform as _platform } from '../lib/modules/platform';
import * as _env from '../lib/util/exec/env';
import * as _fs from '../lib/util/fs';
import * as _git from '../lib/util/git';
import * as _hostRules from '../lib/util/host-rules';
import { regEx } from '../lib/util/regex';
2019-12-09 11:42:55 +00:00
/**
* Simple wrapper for getting mocked version of a module
* @param module module which is mocked by `jest.mock`
*/
export function mocked<T extends object>(module: T): jest.MockedObject<T> {
return module as jest.MockedObject<T>;
}
/**
* Simple wrapper for getting mocked version of a function
* @param func function which is mocked by `jest.mock`
*/
export function mockedFunction<T extends (...args: any[]) => any>(
func: T
): jest.MockedFunction<T> {
return func as jest.MockedFunction<T>;
}
2019-12-09 11:42:55 +00:00
2020-02-03 20:19:42 +00:00
/**
* Simply wrapper to create partial mocks.
* @param obj Object to cast to final type
*/
export function partial<T>(obj: Partial<T>): T;
export function partial<T>(obj: Partial<T>[]): T[];
export function partial(obj: unknown): unknown {
return obj;
2020-02-03 20:19:42 +00:00
}
export const fs = mocked(_fs);
export const git = mocked(_git);
// TODO: fix types, jest / typescript is using wrong overload (#7154)
export const platform = mocked(partial<Required<Platform>>(_platform));
export const env = mocked(_env);
export const hostRules = mocked(_hostRules);
export const logger = mocked(_logger);
2019-12-09 11:42:55 +00:00
2021-12-22 08:27:24 +00:00
export type { RenovateConfig };
export { getConfig };
2020-03-07 10:27:10 +00:00
function getCallerFileName(): string | null {
let result: string | null = null;
const prepareStackTrace = Error.prepareStackTrace;
const stackTraceLimit = Error.stackTraceLimit;
Error.prepareStackTrace = (_err, stack) => stack;
Error.stackTraceLimit = 5; // max calls inside this file + 1
try {
const err = new Error();
const stack = err.stack as unknown as NodeJS.CallSite[];
let currentFile: string | null = null;
for (const frame of stack) {
const fileName = frame.getFileName();
if (!currentFile) {
currentFile = fileName;
} else if (currentFile !== fileName) {
result = fileName;
break;
}
}
} catch (e) {
// no-op
}
Error.prepareStackTrace = prepareStackTrace;
Error.stackTraceLimit = stackTraceLimit;
return result;
}
2021-05-22 12:32:19 +00:00
export function getFixturePath(fixtureFile: string, fixtureRoot = '.'): string {
const callerDir = upath.dirname(getCallerFileName()!);
2021-05-22 12:32:19 +00:00
return upath.join(callerDir, fixtureRoot, '__fixtures__', fixtureFile);
}
/**
* Can be used to search and replace strings in jest snapshots.
* @example
* expect.addSnapshotSerializer(
* replacingSerializer(upath.toUnix(gradleDir.path), 'localDir')
* );
*/
export const replacingSerializer = (
search: string,
replacement: string
): Plugin => ({
test: (value) => typeof value === 'string' && value.includes(search),
serialize: (val, config, indent, depth, refs, printer) => {
const replaced = (val as string).replace(search, replacement);
return printer(replaced, config, indent, depth, refs);
},
});
export function addReplacingSerializer(from: string, to: string): void {
expect.addSnapshotSerializer(replacingSerializer(from, to));
}
function toHash(buf: Buffer): string {
return crypto.createHash('sha256').update(buf).digest('hex');
}
const bufferSerializer: Plugin = {
test: (value) => Buffer.isBuffer(value),
serialize: (val, config, indent, depth, refs, printer) => {
const replaced = toHash(val);
return printer(replaced, config, indent, depth, refs);
},
};
export function addBufferSerializer(): void {
expect.addSnapshotSerializer(bufferSerializer);
}
export function regexMatches(target: string, patterns: string[]): boolean {
return patterns.some((patt: string) => {
const re = regEx(patt);
return re.test(target);
});
}