refactor(config): Generalize allowedVersions regex (#7683)

This commit is contained in:
Sergio Zharinov 2020-11-10 13:12:03 +04:00 committed by GitHub
parent 772f924887
commit fe83cf914e
No known key found for this signature in database
GPG key ID: 4AEE18F83AFDEB23
4 changed files with 45 additions and 49 deletions

View file

@ -1,6 +1,6 @@
import is from '@sindresorhus/is'; import is from '@sindresorhus/is';
import { getManagerList } from '../manager'; import { getManagerList } from '../manager';
import { regEx } from '../util/regex'; import { configRegexPredicate, isConfigRegex, regEx } from '../util/regex';
import * as template from '../util/template'; import * as template from '../util/template';
import { hasValidSchedule, hasValidTimezone } from '../workers/branch/schedule'; import { hasValidSchedule, hasValidTimezone } from '../workers/branch/schedule';
import { RenovateConfig, ValidationMessage } from './common'; import { RenovateConfig, ValidationMessage } from './common';
@ -144,31 +144,8 @@ export async function validateConfig(
message: `Invalid ${currentPath}: \`${errorMessage}\``, message: `Invalid ${currentPath}: \`${errorMessage}\``,
}); });
} }
} else if ( } else if (key === 'allowedVersions' && isConfigRegex(val)) {
key === 'allowedVersions' && if (!configRegexPredicate(val)) {
is.string(val) &&
val.length > 1 &&
val.startsWith('/') &&
val.endsWith('/')
) {
try {
regEx(val.slice(1, -1));
} catch (err) {
errors.push({
depName: 'Configuration Error',
message: `Invalid regExp for ${currentPath}: \`${val}\``,
});
}
} else if (
key === 'allowedVersions' &&
is.string(val) &&
val.length > 2 &&
val.startsWith('!/') &&
val.endsWith('/')
) {
try {
regEx(val.slice(2, -1));
} catch (err) {
errors.push({ errors.push({
depName: 'Configuration Error', depName: 'Configuration Error',
message: `Invalid regExp for ${currentPath}: \`${val}\``, message: `Invalid regExp for ${currentPath}: \`${val}\``,

View file

@ -1,3 +1,4 @@
import is from '@sindresorhus/is';
import { CONFIG_VALIDATION } from '../constants/error-messages'; import { CONFIG_VALIDATION } from '../constants/error-messages';
import { logger } from '../logger'; import { logger } from '../logger';
@ -29,3 +30,38 @@ export function regEx(pattern: string, flags?: string): RegExp {
export function escapeRegExp(input: string): string { export function escapeRegExp(input: string): string {
return input.replace(/[.*+\-?^${}()|[\]\\]/g, '\\$&'); // $& means the whole matched string return input.replace(/[.*+\-?^${}()|[\]\\]/g, '\\$&'); // $& means the whole matched string
} }
const configValStart = /^!?\//;
const configValEnd = /\/$/;
export function isConfigRegex(input: unknown): input is string {
return (
is.string(input) && configValStart.test(input) && configValEnd.test(input)
);
}
function parseConfigRegex(input: string): RegExp | null {
try {
const regexString = input
.replace(configValStart, '')
.replace(configValEnd, '');
return regEx(regexString);
} catch (err) {
// no-op
}
return null;
}
type ConfigRegexPredicate = (string) => boolean;
export function configRegexPredicate(input: string): ConfigRegexPredicate {
const configRegex = parseConfigRegex(input);
if (configRegex) {
const isPositive = !input.startsWith('!');
return (x: string): boolean => {
const res = configRegex.test(x);
return isPositive ? res : !res;
};
}
return null;
}

View file

@ -2,7 +2,7 @@ import * as semver from 'semver';
import { CONFIG_VALIDATION } from '../../../../constants/error-messages'; import { CONFIG_VALIDATION } from '../../../../constants/error-messages';
import { Release } from '../../../../datasource'; import { Release } from '../../../../datasource';
import { logger } from '../../../../logger'; import { logger } from '../../../../logger';
import { regEx } from '../../../../util/regex'; import { configRegexPredicate, isConfigRegex } from '../../../../util/regex';
import * as allVersioning from '../../../../versioning'; import * as allVersioning from '../../../../versioning';
import * as npmVersioning from '../../../../versioning/npm'; import * as npmVersioning from '../../../../versioning/npm';
import * as pep440 from '../../../../versioning/pep440'; import * as pep440 from '../../../../versioning/pep440';
@ -18,8 +18,6 @@ export interface FilterConfig {
versioning: string; versioning: string;
} }
const regexes: Record<string, RegExp> = {};
export function filterVersions( export function filterVersions(
config: FilterConfig, config: FilterConfig,
fromVersion: string, fromVersion: string,
@ -74,25 +72,10 @@ export function filterVersions(
} }
if (allowedVersions) { if (allowedVersions) {
if ( if (isConfigRegex(allowedVersions)) {
allowedVersions.length > 1 && const isAllowed = configRegexPredicate(allowedVersions);
allowedVersions.startsWith('/') && filteredVersions = filteredVersions.filter(({ version }) =>
allowedVersions.endsWith('/') isAllowed(version)
) {
regexes[allowedVersions] =
regexes[allowedVersions] || regEx(allowedVersions.slice(1, -1));
filteredVersions = filteredVersions.filter((v) =>
regexes[allowedVersions].test(v.version)
);
} else if (
allowedVersions.length > 2 &&
allowedVersions.startsWith('!/') &&
allowedVersions.endsWith('/')
) {
regexes[allowedVersions] =
regexes[allowedVersions] || regEx(allowedVersions.slice(2, -1));
filteredVersions = filteredVersions.filter(
(v) => !regexes[allowedVersions].test(v.version)
); );
} else if (versioning.isValid(allowedVersions)) { } else if (versioning.isValid(allowedVersions)) {
filteredVersions = filteredVersions.filter((v) => filteredVersions = filteredVersions.filter((v) =>

View file

@ -161,7 +161,7 @@ describe('workers/repository/process/lookup', () => {
nock('https://registry.npmjs.org').get('/q').reply(200, qJson); nock('https://registry.npmjs.org').get('/q').reply(200, qJson);
expect((await lookup.lookupUpdates(config)).updates).toHaveLength(1); expect((await lookup.lookupUpdates(config)).updates).toHaveLength(1);
}); });
it('enforces allowedVersions with negativee regex', async () => { it('enforces allowedVersions with negative regex', async () => {
config.currentValue = '0.4.0'; config.currentValue = '0.4.0';
config.allowedVersions = '!/^1/'; config.allowedVersions = '!/^1/';
config.depName = 'q'; config.depName = 'q';