2020-03-29 16:22:08 +00:00
|
|
|
export enum TokenType {
|
|
|
|
Number = 1,
|
|
|
|
String,
|
|
|
|
}
|
|
|
|
|
|
|
|
type Token = {
|
|
|
|
type: TokenType;
|
|
|
|
val: string | number;
|
|
|
|
};
|
|
|
|
|
|
|
|
function iterateChars(str: string, cb: (p: string, n: string) => void): void {
|
|
|
|
let prev = null;
|
|
|
|
let next = null;
|
|
|
|
for (let i = 0; i < str.length; i += 1) {
|
|
|
|
next = str.charAt(i);
|
|
|
|
cb(prev, next);
|
|
|
|
prev = next;
|
|
|
|
}
|
|
|
|
cb(prev, null);
|
|
|
|
}
|
|
|
|
|
|
|
|
function isSeparator(char: string): boolean {
|
|
|
|
return /^[-._+]$/i.test(char);
|
|
|
|
}
|
|
|
|
|
|
|
|
function isDigit(char: string): boolean {
|
|
|
|
return /^\d$/.test(char);
|
|
|
|
}
|
|
|
|
|
|
|
|
function isLetter(char: string): boolean {
|
|
|
|
return !isSeparator(char) && !isDigit(char);
|
|
|
|
}
|
|
|
|
|
|
|
|
function isTransition(prevChar: string, nextChar: string): boolean {
|
|
|
|
return (
|
|
|
|
(isDigit(prevChar) && isLetter(nextChar)) ||
|
|
|
|
(isLetter(prevChar) && isDigit(nextChar))
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
export function tokenize(versionStr: string): Token[] | null {
|
|
|
|
let result = [];
|
|
|
|
let currentVal = '';
|
|
|
|
|
|
|
|
function yieldToken(): void {
|
|
|
|
if (currentVal === '') {
|
|
|
|
result = null;
|
|
|
|
}
|
|
|
|
if (result) {
|
|
|
|
const val = currentVal;
|
|
|
|
if (/^\d+$/.test(val)) {
|
|
|
|
result.push({
|
|
|
|
type: TokenType.Number,
|
|
|
|
val: parseInt(val, 10),
|
|
|
|
});
|
|
|
|
} else {
|
|
|
|
result.push({
|
|
|
|
type: TokenType.String,
|
|
|
|
val,
|
|
|
|
});
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
iterateChars(versionStr, (prevChar, nextChar) => {
|
|
|
|
if (nextChar === null) {
|
|
|
|
yieldToken();
|
|
|
|
} else if (isSeparator(nextChar)) {
|
|
|
|
yieldToken();
|
|
|
|
currentVal = '';
|
|
|
|
} else if (prevChar !== null && isTransition(prevChar, nextChar)) {
|
|
|
|
yieldToken();
|
|
|
|
currentVal = nextChar;
|
|
|
|
} else {
|
|
|
|
currentVal = currentVal.concat(nextChar);
|
|
|
|
}
|
|
|
|
});
|
|
|
|
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
2020-03-30 12:48:48 +00:00
|
|
|
export enum QualifierRank {
|
|
|
|
Dev = -1,
|
|
|
|
Default = 0,
|
|
|
|
RC,
|
2021-07-31 05:33:30 +00:00
|
|
|
Snapshot,
|
2020-03-30 12:48:48 +00:00
|
|
|
Final,
|
2021-07-31 05:33:30 +00:00
|
|
|
GA,
|
|
|
|
Release,
|
|
|
|
SP,
|
2020-03-29 16:22:08 +00:00
|
|
|
}
|
|
|
|
|
2020-03-30 12:48:48 +00:00
|
|
|
export function qualifierRank(input: string): number {
|
|
|
|
const val = input.toLowerCase();
|
|
|
|
if (val === 'dev') {
|
|
|
|
return QualifierRank.Dev;
|
2020-03-29 16:22:08 +00:00
|
|
|
}
|
2020-03-30 12:48:48 +00:00
|
|
|
if (val === 'rc' || val === 'cr') {
|
|
|
|
return QualifierRank.RC;
|
2020-03-29 16:22:08 +00:00
|
|
|
}
|
2021-07-31 05:33:30 +00:00
|
|
|
if (val === 'snapshot') {
|
|
|
|
return QualifierRank.Snapshot;
|
|
|
|
}
|
|
|
|
if (val === 'ga') {
|
|
|
|
return QualifierRank.GA;
|
2020-03-30 12:48:48 +00:00
|
|
|
}
|
|
|
|
if (val === 'final') {
|
|
|
|
return QualifierRank.Final;
|
2020-03-29 16:22:08 +00:00
|
|
|
}
|
2021-07-31 05:33:30 +00:00
|
|
|
if (val === 'release' || val === 'latest' || val === 'sr') {
|
|
|
|
return QualifierRank.Release;
|
|
|
|
}
|
|
|
|
if (val === 'sp') {
|
|
|
|
return QualifierRank.SP;
|
|
|
|
}
|
2020-03-30 12:48:48 +00:00
|
|
|
return QualifierRank.Default;
|
2020-03-29 16:22:08 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
function stringTokenCmp(left: string, right: string): number {
|
2020-03-30 12:48:48 +00:00
|
|
|
const leftRank = qualifierRank(left);
|
|
|
|
const rightRank = qualifierRank(right);
|
|
|
|
if (leftRank === 0 && rightRank === 0) {
|
2021-07-31 05:33:30 +00:00
|
|
|
if (left < right) {
|
|
|
|
return -1;
|
|
|
|
}
|
2020-03-29 16:22:08 +00:00
|
|
|
|
2021-07-31 05:33:30 +00:00
|
|
|
if (left > right) {
|
|
|
|
return 1;
|
2020-03-29 16:22:08 +00:00
|
|
|
}
|
|
|
|
} else {
|
2020-03-30 12:48:48 +00:00
|
|
|
if (leftRank < rightRank) {
|
2020-03-29 16:22:08 +00:00
|
|
|
return -1;
|
|
|
|
}
|
2020-03-30 12:48:48 +00:00
|
|
|
if (leftRank > rightRank) {
|
2020-03-29 16:22:08 +00:00
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
function tokenCmp(left: Token | null, right: Token | null): number {
|
|
|
|
if (left === null) {
|
|
|
|
if (right.type === TokenType.String) {
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (right === null) {
|
|
|
|
if (left.type === TokenType.String) {
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (left.type === TokenType.Number && right.type === TokenType.Number) {
|
|
|
|
if (left.val < right.val) {
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
if (left.val > right.val) {
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
} else if (typeof left.val === 'string' && typeof right.val === 'string') {
|
|
|
|
return stringTokenCmp(left.val, right.val);
|
|
|
|
} else if (right.type === TokenType.Number) {
|
|
|
|
return -1;
|
|
|
|
} else if (left.type === TokenType.Number) {
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
export function compare(left: string, right: string): number {
|
|
|
|
const leftTokens = tokenize(left);
|
|
|
|
const rightTokens = tokenize(right);
|
|
|
|
const length = Math.max(leftTokens.length, rightTokens.length);
|
|
|
|
for (let idx = 0; idx < length; idx += 1) {
|
|
|
|
const leftToken = leftTokens[idx] || null;
|
|
|
|
const rightToken = rightTokens[idx] || null;
|
|
|
|
const cmpResult = tokenCmp(leftToken, rightToken);
|
|
|
|
if (cmpResult !== 0) {
|
|
|
|
return cmpResult;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
export function isVersion(input: string): boolean {
|
|
|
|
if (!input) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!/^[-._+a-zA-Z0-9]+$/i.test(input)) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (/^latest\.?/i.test(input)) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
const tokens = tokenize(input);
|
|
|
|
return !!tokens && !!tokens.length;
|
|
|
|
}
|
|
|
|
|
|
|
|
interface PrefixRange {
|
|
|
|
tokens: Token[];
|
|
|
|
}
|
|
|
|
|
|
|
|
export enum RangeBound {
|
|
|
|
Inclusive = 1,
|
|
|
|
Exclusive,
|
|
|
|
}
|
|
|
|
|
|
|
|
interface MavenBasedRange {
|
|
|
|
leftBound: RangeBound;
|
|
|
|
leftBoundStr: string;
|
|
|
|
leftVal: string | null;
|
|
|
|
separator: string;
|
|
|
|
rightBound: RangeBound;
|
|
|
|
rightBoundStr: string;
|
|
|
|
rightVal: string | null;
|
|
|
|
}
|
|
|
|
|
|
|
|
export function parsePrefixRange(input: string): PrefixRange | null {
|
|
|
|
if (!input) {
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (input.trim() === '+') {
|
|
|
|
return { tokens: [] };
|
|
|
|
}
|
|
|
|
|
|
|
|
const postfixRegex = /[-._]\+$/;
|
|
|
|
if (postfixRegex.test(input)) {
|
|
|
|
const prefixValue = input.replace(/[-._]\+$/, '');
|
|
|
|
const tokens = tokenize(prefixValue);
|
|
|
|
return tokens ? { tokens } : null;
|
|
|
|
}
|
|
|
|
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
2021-05-17 08:06:24 +00:00
|
|
|
const mavenBasedRangeRegex =
|
|
|
|
/^(?<leftBoundStr>[[\](]\s*)(?<leftVal>[-._+a-zA-Z0-9]*?)(?<separator>\s*,\s*)(?<rightVal>[-._+a-zA-Z0-9]*?)(?<rightBoundStr>\s*[[\])])$/;
|
2020-03-29 16:22:08 +00:00
|
|
|
|
|
|
|
export function parseMavenBasedRange(input: string): MavenBasedRange | null {
|
|
|
|
if (!input) {
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
|
|
|
const match = mavenBasedRangeRegex.exec(input);
|
|
|
|
if (match) {
|
|
|
|
const { leftBoundStr, separator, rightBoundStr } = match.groups;
|
|
|
|
let { leftVal, rightVal } = match.groups;
|
|
|
|
if (!leftVal) {
|
|
|
|
leftVal = null;
|
|
|
|
}
|
|
|
|
if (!rightVal) {
|
|
|
|
rightVal = null;
|
|
|
|
}
|
|
|
|
const isVersionLeft = isVersion(leftVal);
|
|
|
|
const isVersionRight = isVersion(rightVal);
|
|
|
|
if (
|
|
|
|
(leftVal === null || isVersionLeft) &&
|
|
|
|
(rightVal === null || isVersionRight)
|
|
|
|
) {
|
|
|
|
if (isVersionLeft && isVersionRight && compare(leftVal, rightVal) === 1) {
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
const leftBound =
|
|
|
|
leftBoundStr.trim() === '['
|
|
|
|
? RangeBound.Inclusive
|
|
|
|
: RangeBound.Exclusive;
|
|
|
|
const rightBound =
|
|
|
|
rightBoundStr.trim() === ']'
|
|
|
|
? RangeBound.Inclusive
|
|
|
|
: RangeBound.Exclusive;
|
|
|
|
return {
|
|
|
|
leftBound,
|
|
|
|
leftBoundStr,
|
|
|
|
leftVal,
|
|
|
|
separator,
|
|
|
|
rightBound,
|
|
|
|
rightBoundStr,
|
|
|
|
rightVal,
|
|
|
|
};
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
|
|
|
export function isValid(str: string): boolean {
|
|
|
|
if (!str) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
return (
|
|
|
|
isVersion(str) || !!parsePrefixRange(str) || !!parseMavenBasedRange(str)
|
|
|
|
);
|
|
|
|
}
|