303 lines
8.5 KiB
JavaScript
303 lines
8.5 KiB
JavaScript
|
const debug = require('../internal/debug')
|
||
|
const { MAX_LENGTH, MAX_SAFE_INTEGER } = require('../internal/constants')
|
||
|
const { safeRe: re, t } = require('../internal/re')
|
||
|
|
||
|
const parseOptions = require('../internal/parse-options')
|
||
|
const { compareIdentifiers } = require('../internal/identifiers')
|
||
|
class SemVer {
|
||
|
constructor (version, options) {
|
||
|
options = parseOptions(options)
|
||
|
|
||
|
if (version instanceof SemVer) {
|
||
|
if (version.loose === !!options.loose &&
|
||
|
version.includePrerelease === !!options.includePrerelease) {
|
||
|
return version
|
||
|
} else {
|
||
|
version = version.version
|
||
|
}
|
||
|
} else if (typeof version !== 'string') {
|
||
|
throw new TypeError(`Invalid version. Must be a string. Got type "${typeof version}".`)
|
||
|
}
|
||
|
|
||
|
if (version.length > MAX_LENGTH) {
|
||
|
throw new TypeError(
|
||
|
`version is longer than ${MAX_LENGTH} characters`
|
||
|
)
|
||
|
}
|
||
|
|
||
|
debug('SemVer', version, options)
|
||
|
this.options = options
|
||
|
this.loose = !!options.loose
|
||
|
// this isn't actually relevant for versions, but keep it so that we
|
||
|
// don't run into trouble passing this.options around.
|
||
|
this.includePrerelease = !!options.includePrerelease
|
||
|
|
||
|
const m = version.trim().match(options.loose ? re[t.LOOSE] : re[t.FULL])
|
||
|
|
||
|
if (!m) {
|
||
|
throw new TypeError(`Invalid Version: ${version}`)
|
||
|
}
|
||
|
|
||
|
this.raw = version
|
||
|
|
||
|
// these are actually numbers
|
||
|
this.major = +m[1]
|
||
|
this.minor = +m[2]
|
||
|
this.patch = +m[3]
|
||
|
|
||
|
if (this.major > MAX_SAFE_INTEGER || this.major < 0) {
|
||
|
throw new TypeError('Invalid major version')
|
||
|
}
|
||
|
|
||
|
if (this.minor > MAX_SAFE_INTEGER || this.minor < 0) {
|
||
|
throw new TypeError('Invalid minor version')
|
||
|
}
|
||
|
|
||
|
if (this.patch > MAX_SAFE_INTEGER || this.patch < 0) {
|
||
|
throw new TypeError('Invalid patch version')
|
||
|
}
|
||
|
|
||
|
// numberify any prerelease numeric ids
|
||
|
if (!m[4]) {
|
||
|
this.prerelease = []
|
||
|
} else {
|
||
|
this.prerelease = m[4].split('.').map((id) => {
|
||
|
if (/^[0-9]+$/.test(id)) {
|
||
|
const num = +id
|
||
|
if (num >= 0 && num < MAX_SAFE_INTEGER) {
|
||
|
return num
|
||
|
}
|
||
|
}
|
||
|
return id
|
||
|
})
|
||
|
}
|
||
|
|
||
|
this.build = m[5] ? m[5].split('.') : []
|
||
|
this.format()
|
||
|
}
|
||
|
|
||
|
format () {
|
||
|
this.version = `${this.major}.${this.minor}.${this.patch}`
|
||
|
if (this.prerelease.length) {
|
||
|
this.version += `-${this.prerelease.join('.')}`
|
||
|
}
|
||
|
return this.version
|
||
|
}
|
||
|
|
||
|
toString () {
|
||
|
return this.version
|
||
|
}
|
||
|
|
||
|
compare (other) {
|
||
|
debug('SemVer.compare', this.version, this.options, other)
|
||
|
if (!(other instanceof SemVer)) {
|
||
|
if (typeof other === 'string' && other === this.version) {
|
||
|
return 0
|
||
|
}
|
||
|
other = new SemVer(other, this.options)
|
||
|
}
|
||
|
|
||
|
if (other.version === this.version) {
|
||
|
return 0
|
||
|
}
|
||
|
|
||
|
return this.compareMain(other) || this.comparePre(other)
|
||
|
}
|
||
|
|
||
|
compareMain (other) {
|
||
|
if (!(other instanceof SemVer)) {
|
||
|
other = new SemVer(other, this.options)
|
||
|
}
|
||
|
|
||
|
return (
|
||
|
compareIdentifiers(this.major, other.major) ||
|
||
|
compareIdentifiers(this.minor, other.minor) ||
|
||
|
compareIdentifiers(this.patch, other.patch)
|
||
|
)
|
||
|
}
|
||
|
|
||
|
comparePre (other) {
|
||
|
if (!(other instanceof SemVer)) {
|
||
|
other = new SemVer(other, this.options)
|
||
|
}
|
||
|
|
||
|
// NOT having a prerelease is > having one
|
||
|
if (this.prerelease.length && !other.prerelease.length) {
|
||
|
return -1
|
||
|
} else if (!this.prerelease.length && other.prerelease.length) {
|
||
|
return 1
|
||
|
} else if (!this.prerelease.length && !other.prerelease.length) {
|
||
|
return 0
|
||
|
}
|
||
|
|
||
|
let i = 0
|
||
|
do {
|
||
|
const a = this.prerelease[i]
|
||
|
const b = other.prerelease[i]
|
||
|
debug('prerelease compare', i, a, b)
|
||
|
if (a === undefined && b === undefined) {
|
||
|
return 0
|
||
|
} else if (b === undefined) {
|
||
|
return 1
|
||
|
} else if (a === undefined) {
|
||
|
return -1
|
||
|
} else if (a === b) {
|
||
|
continue
|
||
|
} else {
|
||
|
return compareIdentifiers(a, b)
|
||
|
}
|
||
|
} while (++i)
|
||
|
}
|
||
|
|
||
|
compareBuild (other) {
|
||
|
if (!(other instanceof SemVer)) {
|
||
|
other = new SemVer(other, this.options)
|
||
|
}
|
||
|
|
||
|
let i = 0
|
||
|
do {
|
||
|
const a = this.build[i]
|
||
|
const b = other.build[i]
|
||
|
debug('build compare', i, a, b)
|
||
|
if (a === undefined && b === undefined) {
|
||
|
return 0
|
||
|
} else if (b === undefined) {
|
||
|
return 1
|
||
|
} else if (a === undefined) {
|
||
|
return -1
|
||
|
} else if (a === b) {
|
||
|
continue
|
||
|
} else {
|
||
|
return compareIdentifiers(a, b)
|
||
|
}
|
||
|
} while (++i)
|
||
|
}
|
||
|
|
||
|
// preminor will bump the version up to the next minor release, and immediately
|
||
|
// down to pre-release. premajor and prepatch work the same way.
|
||
|
inc (release, identifier, identifierBase) {
|
||
|
switch (release) {
|
||
|
case 'premajor':
|
||
|
this.prerelease.length = 0
|
||
|
this.patch = 0
|
||
|
this.minor = 0
|
||
|
this.major++
|
||
|
this.inc('pre', identifier, identifierBase)
|
||
|
break
|
||
|
case 'preminor':
|
||
|
this.prerelease.length = 0
|
||
|
this.patch = 0
|
||
|
this.minor++
|
||
|
this.inc('pre', identifier, identifierBase)
|
||
|
break
|
||
|
case 'prepatch':
|
||
|
// If this is already a prerelease, it will bump to the next version
|
||
|
// drop any prereleases that might already exist, since they are not
|
||
|
// relevant at this point.
|
||
|
this.prerelease.length = 0
|
||
|
this.inc('patch', identifier, identifierBase)
|
||
|
this.inc('pre', identifier, identifierBase)
|
||
|
break
|
||
|
// If the input is a non-prerelease version, this acts the same as
|
||
|
// prepatch.
|
||
|
case 'prerelease':
|
||
|
if (this.prerelease.length === 0) {
|
||
|
this.inc('patch', identifier, identifierBase)
|
||
|
}
|
||
|
this.inc('pre', identifier, identifierBase)
|
||
|
break
|
||
|
|
||
|
case 'major':
|
||
|
// If this is a pre-major version, bump up to the same major version.
|
||
|
// Otherwise increment major.
|
||
|
// 1.0.0-5 bumps to 1.0.0
|
||
|
// 1.1.0 bumps to 2.0.0
|
||
|
if (
|
||
|
this.minor !== 0 ||
|
||
|
this.patch !== 0 ||
|
||
|
this.prerelease.length === 0
|
||
|
) {
|
||
|
this.major++
|
||
|
}
|
||
|
this.minor = 0
|
||
|
this.patch = 0
|
||
|
this.prerelease = []
|
||
|
break
|
||
|
case 'minor':
|
||
|
// If this is a pre-minor version, bump up to the same minor version.
|
||
|
// Otherwise increment minor.
|
||
|
// 1.2.0-5 bumps to 1.2.0
|
||
|
// 1.2.1 bumps to 1.3.0
|
||
|
if (this.patch !== 0 || this.prerelease.length === 0) {
|
||
|
this.minor++
|
||
|
}
|
||
|
this.patch = 0
|
||
|
this.prerelease = []
|
||
|
break
|
||
|
case 'patch':
|
||
|
// If this is not a pre-release version, it will increment the patch.
|
||
|
// If it is a pre-release it will bump up to the same patch version.
|
||
|
// 1.2.0-5 patches to 1.2.0
|
||
|
// 1.2.0 patches to 1.2.1
|
||
|
if (this.prerelease.length === 0) {
|
||
|
this.patch++
|
||
|
}
|
||
|
this.prerelease = []
|
||
|
break
|
||
|
// This probably shouldn't be used publicly.
|
||
|
// 1.0.0 'pre' would become 1.0.0-0 which is the wrong direction.
|
||
|
case 'pre': {
|
||
|
const base = Number(identifierBase) ? 1 : 0
|
||
|
|
||
|
if (!identifier && identifierBase === false) {
|
||
|
throw new Error('invalid increment argument: identifier is empty')
|
||
|
}
|
||
|
|
||
|
if (this.prerelease.length === 0) {
|
||
|
this.prerelease = [base]
|
||
|
} else {
|
||
|
let i = this.prerelease.length
|
||
|
while (--i >= 0) {
|
||
|
if (typeof this.prerelease[i] === 'number') {
|
||
|
this.prerelease[i]++
|
||
|
i = -2
|
||
|
}
|
||
|
}
|
||
|
if (i === -1) {
|
||
|
// didn't increment anything
|
||
|
if (identifier === this.prerelease.join('.') && identifierBase === false) {
|
||
|
throw new Error('invalid increment argument: identifier already exists')
|
||
|
}
|
||
|
this.prerelease.push(base)
|
||
|
}
|
||
|
}
|
||
|
if (identifier) {
|
||
|
// 1.2.0-beta.1 bumps to 1.2.0-beta.2,
|
||
|
// 1.2.0-beta.fooblz or 1.2.0-beta bumps to 1.2.0-beta.0
|
||
|
let prerelease = [identifier, base]
|
||
|
if (identifierBase === false) {
|
||
|
prerelease = [identifier]
|
||
|
}
|
||
|
if (compareIdentifiers(this.prerelease[0], identifier) === 0) {
|
||
|
if (isNaN(this.prerelease[1])) {
|
||
|
this.prerelease = prerelease
|
||
|
}
|
||
|
} else {
|
||
|
this.prerelease = prerelease
|
||
|
}
|
||
|
}
|
||
|
break
|
||
|
}
|
||
|
default:
|
||
|
throw new Error(`invalid increment argument: ${release}`)
|
||
|
}
|
||
|
this.raw = this.format()
|
||
|
if (this.build.length) {
|
||
|
this.raw += `+${this.build.join('.')}`
|
||
|
}
|
||
|
return this
|
||
|
}
|
||
|
}
|
||
|
|
||
|
module.exports = SemVer
|