renovate/lib/config/presets.ts

261 lines
8 KiB
TypeScript

import is from '@sindresorhus/is';
import { logger } from '../logger';
import * as massage from './massage';
import * as migration from './migration';
import * as github from '../datasource/github';
import * as npm from '../datasource/npm';
import * as gitlab from '../datasource/gitlab';
import { RenovateConfig } from './common';
import { mergeChildConfig } from './utils';
import { regEx } from '../util/regex';
import {
CONFIG_VALIDATION,
DATASOURCE_FAILURE,
PLATFORM_FAILURE,
} from '../constants/error-messages';
import {
DATASOURCE_GITHUB,
DATASOURCE_GITLAB,
DATASOURCE_NPM,
} from '../constants/data-binary-source';
const datasources = {
github,
npm,
gitlab,
};
export function replaceArgs(
obj: string | string[] | object | object[],
argMapping: Record<string, any>
): any {
if (is.string(obj)) {
let returnStr = obj;
for (const [arg, argVal] of Object.entries(argMapping)) {
const re = regEx(`{{${arg}}}`, 'g');
returnStr = returnStr.replace(re, argVal);
}
return returnStr;
}
if (is.array(obj)) {
const returnArray = [];
for (const item of obj) {
returnArray.push(replaceArgs(item, argMapping));
}
return returnArray;
}
if (is.object(obj)) {
const returnObj = {};
for (const [key, val] of Object.entries(obj)) {
returnObj[key] = replaceArgs(val, argMapping);
}
return returnObj;
}
return obj;
}
export function parsePreset(input: string): ParsedPreset {
let str = input;
let datasource: string;
let packageName: string;
let presetName: string;
let params: string[];
if (str.startsWith('github>')) {
datasource = DATASOURCE_GITHUB;
str = str.substring('github>'.length);
} else if (str.startsWith('gitlab>')) {
datasource = DATASOURCE_GITLAB;
str = str.substring('gitlab>'.length);
}
str = str.replace(/^npm>/, '');
datasource = datasource || DATASOURCE_NPM;
if (str.includes('(')) {
params = str
.slice(str.indexOf('(') + 1, -1)
.split(',')
.map(elem => elem.trim());
str = str.slice(0, str.indexOf('('));
}
if (str[0] === ':') {
// default namespace
packageName = 'renovate-config-default';
presetName = str.slice(1);
} else if (str[0] === '@') {
// scoped namespace
[, packageName] = str.match(/(@.*?)(:|$)/);
str = str.slice(packageName.length);
if (!packageName.includes('/')) {
packageName += '/renovate-config';
}
if (str === '') {
presetName = 'default';
} else {
presetName = str.slice(1);
}
} else {
// non-scoped namespace
[, packageName] = str.match(/(.*?)(:|$)/);
presetName = str.slice(packageName.length + 1);
if (
datasource === DATASOURCE_NPM &&
!packageName.startsWith('renovate-config-')
) {
packageName = `renovate-config-${packageName}`;
}
if (presetName === '') {
presetName = 'default';
}
}
return { datasource, packageName, presetName, params };
}
export async function getPreset(preset: string): Promise<RenovateConfig> {
logger.trace(`getPreset(${preset})`);
const { datasource, packageName, presetName, params } = parsePreset(preset);
let presetConfig = await datasources[datasource].getPreset(
packageName,
presetName
);
logger.trace({ presetConfig }, `Found preset ${preset}`);
if (params) {
const argMapping = {};
for (const [index, value] of params.entries()) {
argMapping[`arg${index}`] = value;
}
presetConfig = replaceArgs(presetConfig, argMapping);
}
logger.trace({ presetConfig }, `Applied params to preset ${preset}`);
const presetKeys = Object.keys(presetConfig);
if (
presetKeys.length === 2 &&
presetKeys.includes('description') &&
presetKeys.includes('extends')
) {
// preset is just a collection of other presets
delete presetConfig.description;
}
const packageListKeys = [
'description',
'packageNames',
'excludePackageNames',
'packagePatterns',
'excludePackagePatterns',
];
if (presetKeys.every(key => packageListKeys.includes(key))) {
delete presetConfig.description;
}
const { migratedConfig } = migration.migrateConfig(presetConfig);
return massage.massageConfig(migratedConfig);
}
export async function resolveConfigPresets(
inputConfig: RenovateConfig,
ignorePresets?: string[],
existingPresets: string[] = []
): Promise<RenovateConfig> {
if (!ignorePresets) {
ignorePresets = inputConfig.ignorePresets || []; // eslint-disable-line
}
logger.trace(
{ config: inputConfig, existingPresets },
'resolveConfigPresets'
);
let config: RenovateConfig = {};
// First, merge all the preset configs from left to right
if (inputConfig.extends && inputConfig.extends.length) {
for (const preset of inputConfig.extends) {
// istanbul ignore if
if (existingPresets.includes(preset)) {
logger.info(`Already seen preset ${preset} in ${existingPresets}`);
} else if (ignorePresets.includes(preset)) {
// istanbul ignore next
logger.info(`Ignoring preset ${preset} in ${existingPresets}`);
} else {
logger.trace(`Resolving preset "${preset}"`);
let fetchedPreset;
try {
fetchedPreset = await getPreset(preset);
} catch (err) {
logger.debug({ err }, 'Preset fetch error');
// istanbul ignore if
if (
err.message === PLATFORM_FAILURE ||
err.message === DATASOURCE_FAILURE
) {
throw err;
}
const error = new Error(CONFIG_VALIDATION);
if (err.message === 'dep not found') {
error.validationError = `Cannot find preset's package (${preset})`;
} else if (err.message === 'preset renovate-config not found') {
// istanbul ignore next
error.validationError = `Preset package is missing a renovate-config entry (${preset})`;
} else if (err.message === 'preset not found') {
error.validationError = `Preset name not found within published preset config (${preset})`;
}
// istanbul ignore if
if (existingPresets.length) {
error.validationError +=
'. Note: this is a *nested* preset so please contact the preset author if you are unable to fix it yourself.';
}
logger.info('Throwing preset error');
throw error;
}
const presetConfig = await resolveConfigPresets(
fetchedPreset,
ignorePresets,
existingPresets.concat([preset])
);
// istanbul ignore if
if (
inputConfig &&
inputConfig.ignoreDeps &&
inputConfig.ignoreDeps.length === 0
) {
delete presetConfig.description;
}
config = mergeChildConfig(config, presetConfig);
}
}
}
logger.trace({ config }, `Post-preset resolve config`);
// Now assign "regular" config on top
config = mergeChildConfig(config, inputConfig);
delete config.extends;
delete config.ignorePresets;
logger.trace({ config }, `Post-merge resolve config`);
for (const [key, val] of Object.entries(config)) {
const ignoredKeys = ['content', 'onboardingConfig'];
if (is.array(val)) {
// Resolve nested objects inside arrays
config[key] = [];
for (const element of val) {
if (is.object(element)) {
config[key].push(
await resolveConfigPresets(element, ignorePresets, existingPresets)
);
} else {
config[key].push(element);
}
}
} else if (is.object(val) && !ignoredKeys.includes(key)) {
// Resolve nested objects
logger.trace(`Resolving object "${key}"`);
config[key] = await resolveConfigPresets(
val,
ignorePresets,
existingPresets
);
}
}
logger.trace({ config: inputConfig }, 'Input config');
logger.trace({ config }, 'Resolved config');
return config;
}
export interface ParsedPreset {
datasource: string;
packageName: string;
presetName: string;
params?: string[];
}