renovate/lib/versioning/regex/index.ts
Kevin James f27c53f4c8 fix(versioner): fixup regex rollback logic (#4358)
As described on Slack, the regex versioner is currently causing bad PRs
such as [this](https://github.com/TheKevJames/experiments/pull/8) since
it can not properly track the current version. By parsing the `range`
values with the same logic as the other version strings, eveverything
should match up properly.
2019-08-23 23:01:20 +02:00

136 lines
4.4 KiB
TypeScript

import { compare, satisfies, ltr, minSatisfying, maxSatisfying } from 'semver';
import RE2 from 're2';
import { VersioningApiConstructor } from '../common';
import { GenericVersion, GenericVersioningApi } from '../loose/generic';
import { logger } from '../../logger';
export interface RegExpVersion extends GenericVersion {
/** prereleases are treated in the standard semver manner, if present */
prerelease: string;
/**
* compatibility, if present, are treated as a compatibility layer: we will
* never try to update to a version with a different compatibility.
*/
compatibility: string;
}
export class RegExpVersioningApi extends GenericVersioningApi<RegExpVersion> {
// config is expected to be overridden by a user-specified RegExp value
// sample values:
//
// * emulates the "semver" configuration:
// RegExp('^(?<major>\\d+)\\.(?<minor>\\d+)\\.(?<patch>\\d+)(-(?<prerelease>.*))?$')
// * emulates the "docker" configuration:
// RegExp('^(?<major>\\d+)\\.(?<minor>\\d+)\\.(?<patch>\\d+)(-(?<compatibility>.*))?$')
// * matches the versioning scheme used by the Python images on DockerHub:
// RegExp('^(?<major>\\d+)\\.(?<minor>\\d+)\\.(?<patch>\\d+)(?<prerelease>[^.-]+)?(-(?<compatibility>.*))?$');
private _config: RegExp = null;
constructor(new_config: string) {
super();
if (!new_config)
// eslint-disable-next-line no-param-reassign
new_config = '^(?<major>\\d+)?$';
// without at least one of {major, minor, patch} specified in the regex,
// this versioner will not work properly
if (
!new_config.includes('<major>') &&
!new_config.includes('<minor>') &&
!new_config.includes('<patch>')
) {
const error = new Error('config-validation');
error.configFile = new_config;
error.validationError =
'regex versionScheme needs at least one major, minor or patch group defined';
throw error;
}
// TODO: should we validate the user has not added extra unsupported
// capture groups?
try {
this._config = new RE2(new_config);
} catch (e) {
logger.debug({ err: e }, 'regex error');
const error = new Error('config-validation');
error.configFile = new_config;
error.validationError = 'Invalid regex versionSheme found';
throw error;
}
}
protected _compare(version: string, other: string): number {
return compare(
asSemver(this._parse(version)),
asSemver(this._parse(other))
);
}
// convenience method for passing a string into a Version given current config.
protected _parse(version: string): RegExpVersion | null {
const match = version.match(this._config);
if (match === null) {
return null;
}
const groups = match.groups;
return {
release: [
typeof groups.major === 'undefined' ? 0 : Number(groups.major),
typeof groups.minor === 'undefined' ? 0 : Number(groups.minor),
typeof groups.patch === 'undefined' ? 0 : Number(groups.patch),
],
prerelease: groups.prerelease,
compatibility: groups.compatibility,
};
}
isCompatible(version: string, range: string): boolean {
return (
this._parse(version).compatibility === this._parse(range).compatibility
);
}
isStable(version: string): boolean {
return typeof this._parse(version).prerelease === 'undefined';
}
isLessThanRange(version: string, range: string): boolean {
return ltr(asSemver(this._parse(version)), asSemver(this._parse(range)));
}
maxSatisfyingVersion(versions: string[], range: string): string | null {
return maxSatisfying(
versions.map(v => asSemver(this._parse(v))),
asSemver(this._parse(range))
);
}
minSatisfyingVersion(versions: string[], range: string): string | null {
return minSatisfying(
versions.map(v => asSemver(this._parse(v))),
asSemver(this._parse(range))
);
}
matches(version: string, range: string) {
return satisfies(
asSemver(this._parse(version)),
asSemver(this._parse(range))
);
}
}
// convenience method for passing a Version object into any semver.* method.
function asSemver(version: RegExpVersion): string {
let vstring = `${version.release[0]}.${version.release[1]}.${version.release[2]}`;
if (typeof version.prerelease !== 'undefined') {
vstring += `-${version.prerelease}`;
}
return vstring;
}
export const api: VersioningApiConstructor = RegExpVersioningApi;
export default api;