2019-08-22 18:09:23 +00:00
|
|
|
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>.*))?$')
|
2019-08-23 03:44:51 +00:00
|
|
|
// * matches the versioning scheme used by the Python images on DockerHub:
|
2019-08-22 18:09:23 +00:00
|
|
|
// 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 {
|
2019-08-23 21:01:20 +00:00
|
|
|
return ltr(asSemver(this._parse(version)), asSemver(this._parse(range)));
|
2019-08-22 18:09:23 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
maxSatisfyingVersion(versions: string[], range: string): string | null {
|
2019-08-23 21:01:20 +00:00
|
|
|
return maxSatisfying(
|
|
|
|
versions.map(v => asSemver(this._parse(v))),
|
|
|
|
asSemver(this._parse(range))
|
|
|
|
);
|
2019-08-22 18:09:23 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
minSatisfyingVersion(versions: string[], range: string): string | null {
|
2019-08-23 21:01:20 +00:00
|
|
|
return minSatisfying(
|
|
|
|
versions.map(v => asSemver(this._parse(v))),
|
|
|
|
asSemver(this._parse(range))
|
|
|
|
);
|
2019-08-22 18:09:23 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
matches(version: string, range: string) {
|
2019-08-23 21:01:20 +00:00
|
|
|
return satisfies(
|
|
|
|
asSemver(this._parse(version)),
|
|
|
|
asSemver(this._parse(range))
|
|
|
|
);
|
2019-08-22 18:09:23 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// 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;
|