mirror of https://github.com/renovatebot/renovate
204 lines
6.6 KiB
TypeScript
204 lines
6.6 KiB
TypeScript
import is from '@sindresorhus/is';
|
|
import { dequal } from 'dequal';
|
|
import { logger } from '../logger';
|
|
import { clone } from '../util/clone';
|
|
import { regEx } from '../util/regex';
|
|
import { MigrationsService } from './migrations';
|
|
import { getOptions } from './options';
|
|
import type {
|
|
MigratedConfig,
|
|
MigratedRenovateConfig,
|
|
PackageRule,
|
|
RenovateConfig,
|
|
RenovateOptions,
|
|
} from './types';
|
|
import { mergeChildConfig } from './utils';
|
|
|
|
const options = getOptions();
|
|
export function fixShortHours(input: string): string {
|
|
return input.replace(regEx(/( \d?\d)((a|p)m)/g), '$1:00$2');
|
|
}
|
|
|
|
let optionTypes: Record<string, RenovateOptions['type']>;
|
|
// Returns a migrated config
|
|
export function migrateConfig(
|
|
config: RenovateConfig,
|
|
parentKey?: string,
|
|
): MigratedConfig {
|
|
try {
|
|
if (!optionTypes) {
|
|
optionTypes = {};
|
|
options.forEach((option) => {
|
|
optionTypes[option.name] = option.type;
|
|
});
|
|
}
|
|
const newConfig = MigrationsService.run(config, parentKey);
|
|
const migratedConfig = clone(newConfig) as MigratedRenovateConfig;
|
|
|
|
for (const [key, val] of Object.entries(newConfig)) {
|
|
if (is.string(val) && val.includes('{{baseDir}}')) {
|
|
migratedConfig[key] = val.replace(
|
|
regEx(/{{baseDir}}/g),
|
|
'{{packageFileDir}}',
|
|
);
|
|
} else if (is.string(val) && val.includes('{{lookupName}}')) {
|
|
migratedConfig[key] = val.replace(
|
|
regEx(/{{lookupName}}/g),
|
|
'{{packageName}}',
|
|
);
|
|
} else if (is.string(val) && val.includes('{{depNameShort}}')) {
|
|
migratedConfig[key] = val.replace(
|
|
regEx(/{{depNameShort}}/g),
|
|
'{{depName}}',
|
|
);
|
|
} else if (is.string(val) && val.startsWith('{{semanticPrefix}}')) {
|
|
migratedConfig[key] = val.replace(
|
|
'{{semanticPrefix}}',
|
|
'{{#if semanticCommitType}}{{semanticCommitType}}{{#if semanticCommitScope}}({{semanticCommitScope}}){{/if}}: {{/if}}',
|
|
);
|
|
} else if (optionTypes[key] === 'object' && is.boolean(val)) {
|
|
migratedConfig[key] = { enabled: val };
|
|
} else if (optionTypes[key] === 'boolean') {
|
|
if (val === 'true') {
|
|
migratedConfig[key] = true;
|
|
} else if (val === 'false') {
|
|
migratedConfig[key] = false;
|
|
}
|
|
} else if (
|
|
optionTypes[key] === 'string' &&
|
|
is.array(val) &&
|
|
val.length === 1
|
|
) {
|
|
migratedConfig[key] = String(val[0]);
|
|
} else if (is.array(val)) {
|
|
if (is.array(migratedConfig?.[key])) {
|
|
const newArray = [];
|
|
for (const item of migratedConfig[key]) {
|
|
if (is.object(item) && !is.array(item)) {
|
|
const arrMigrate = migrateConfig(item as RenovateConfig);
|
|
newArray.push(arrMigrate.migratedConfig);
|
|
} else {
|
|
newArray.push(item);
|
|
}
|
|
}
|
|
migratedConfig[key] = newArray;
|
|
}
|
|
} else if (is.object(val)) {
|
|
const subMigrate = migrateConfig(
|
|
migratedConfig[key] as RenovateConfig,
|
|
key,
|
|
);
|
|
if (subMigrate.isMigrated) {
|
|
migratedConfig[key] = subMigrate.migratedConfig;
|
|
}
|
|
}
|
|
|
|
const migratedTemplates = {
|
|
fromVersion: 'currentVersion',
|
|
newValueMajor: 'newMajor',
|
|
newValueMinor: 'newMinor',
|
|
newVersionMajor: 'newMajor',
|
|
newVersionMinor: 'newMinor',
|
|
toVersion: 'newVersion',
|
|
};
|
|
if (is.string(migratedConfig[key])) {
|
|
for (const [from, to] of Object.entries(migratedTemplates)) {
|
|
migratedConfig[key] = (migratedConfig[key] as string).replace(
|
|
regEx(from, 'g'),
|
|
to,
|
|
);
|
|
}
|
|
}
|
|
}
|
|
const languages = [
|
|
'docker',
|
|
'dotnet',
|
|
'golang',
|
|
'java',
|
|
'js',
|
|
'node',
|
|
'php',
|
|
'python',
|
|
'ruby',
|
|
'rust',
|
|
];
|
|
for (const language of languages) {
|
|
if (is.nonEmptyObject(migratedConfig[language])) {
|
|
migratedConfig.packageRules ??= [];
|
|
const currentContent = migratedConfig[language] as any;
|
|
const packageRule = {
|
|
matchCategories: [language],
|
|
...currentContent,
|
|
};
|
|
migratedConfig.packageRules.unshift(packageRule);
|
|
delete migratedConfig[language];
|
|
}
|
|
}
|
|
// Migrate nested packageRules
|
|
if (is.nonEmptyArray(migratedConfig.packageRules)) {
|
|
const existingRules = migratedConfig.packageRules;
|
|
migratedConfig.packageRules = [];
|
|
for (const packageRule of existingRules) {
|
|
if (is.array(packageRule.packageRules)) {
|
|
logger.debug('Flattening nested packageRules');
|
|
// merge each subrule and add to the parent list
|
|
for (const subrule of packageRule.packageRules) {
|
|
// TODO: fix types #22198
|
|
const combinedRule = mergeChildConfig(
|
|
packageRule,
|
|
subrule as PackageRule,
|
|
);
|
|
delete combinedRule.packageRules;
|
|
migratedConfig.packageRules.push(combinedRule);
|
|
}
|
|
} else {
|
|
migratedConfig.packageRules.push(packageRule);
|
|
}
|
|
}
|
|
}
|
|
if (
|
|
is.nonEmptyObject(migratedConfig['pip-compile']) &&
|
|
is.nonEmptyArray(migratedConfig['pip-compile'].fileMatch)
|
|
) {
|
|
migratedConfig['pip-compile'].fileMatch = migratedConfig[
|
|
'pip-compile'
|
|
].fileMatch.map((fileMatch) => {
|
|
const match = fileMatch as string;
|
|
if (match.endsWith('.in')) {
|
|
return match.replace(/\.in$/, '.txt');
|
|
}
|
|
return match.replace(/\.in\$$/, '.txt$');
|
|
});
|
|
}
|
|
if (is.nonEmptyArray(migratedConfig.matchManagers)) {
|
|
if (migratedConfig.matchManagers.includes('gradle-lite')) {
|
|
if (!migratedConfig.matchManagers.includes('gradle')) {
|
|
migratedConfig.matchManagers.push('gradle');
|
|
}
|
|
migratedConfig.matchManagers = migratedConfig.matchManagers.filter(
|
|
(manager) => manager !== 'gradle-lite',
|
|
);
|
|
}
|
|
}
|
|
if (is.nonEmptyObject(migratedConfig['gradle-lite'])) {
|
|
migratedConfig.gradle = mergeChildConfig(
|
|
migratedConfig.gradle ?? {},
|
|
migratedConfig['gradle-lite'],
|
|
);
|
|
}
|
|
delete migratedConfig['gradle-lite'];
|
|
const isMigrated = !dequal(config, migratedConfig);
|
|
if (isMigrated) {
|
|
// recursive call in case any migrated configs need further migrating
|
|
return {
|
|
isMigrated,
|
|
migratedConfig: migrateConfig(migratedConfig).migratedConfig,
|
|
};
|
|
}
|
|
return { isMigrated, migratedConfig };
|
|
} catch (err) /* istanbul ignore next */ {
|
|
logger.debug({ config, err }, 'migrateConfig() error');
|
|
throw err;
|
|
}
|
|
}
|