[6a3a178] | 1 | const { MAX_SAFE_COMPONENT_LENGTH } = require('./constants')
|
---|
| 2 | const debug = require('./debug')
|
---|
| 3 | exports = module.exports = {}
|
---|
| 4 |
|
---|
| 5 | // The actual regexps go on exports.re
|
---|
| 6 | const re = exports.re = []
|
---|
| 7 | const src = exports.src = []
|
---|
| 8 | const t = exports.t = {}
|
---|
| 9 | let R = 0
|
---|
| 10 |
|
---|
| 11 | const createToken = (name, value, isGlobal) => {
|
---|
| 12 | const index = R++
|
---|
| 13 | debug(index, value)
|
---|
| 14 | t[name] = index
|
---|
| 15 | src[index] = value
|
---|
| 16 | re[index] = new RegExp(value, isGlobal ? 'g' : undefined)
|
---|
| 17 | }
|
---|
| 18 |
|
---|
| 19 | // The following Regular Expressions can be used for tokenizing,
|
---|
| 20 | // validating, and parsing SemVer version strings.
|
---|
| 21 |
|
---|
| 22 | // ## Numeric Identifier
|
---|
| 23 | // A single `0`, or a non-zero digit followed by zero or more digits.
|
---|
| 24 |
|
---|
| 25 | createToken('NUMERICIDENTIFIER', '0|[1-9]\\d*')
|
---|
| 26 | createToken('NUMERICIDENTIFIERLOOSE', '[0-9]+')
|
---|
| 27 |
|
---|
| 28 | // ## Non-numeric Identifier
|
---|
| 29 | // Zero or more digits, followed by a letter or hyphen, and then zero or
|
---|
| 30 | // more letters, digits, or hyphens.
|
---|
| 31 |
|
---|
| 32 | createToken('NONNUMERICIDENTIFIER', '\\d*[a-zA-Z-][a-zA-Z0-9-]*')
|
---|
| 33 |
|
---|
| 34 | // ## Main Version
|
---|
| 35 | // Three dot-separated numeric identifiers.
|
---|
| 36 |
|
---|
| 37 | createToken('MAINVERSION', `(${src[t.NUMERICIDENTIFIER]})\\.` +
|
---|
| 38 | `(${src[t.NUMERICIDENTIFIER]})\\.` +
|
---|
| 39 | `(${src[t.NUMERICIDENTIFIER]})`)
|
---|
| 40 |
|
---|
| 41 | createToken('MAINVERSIONLOOSE', `(${src[t.NUMERICIDENTIFIERLOOSE]})\\.` +
|
---|
| 42 | `(${src[t.NUMERICIDENTIFIERLOOSE]})\\.` +
|
---|
| 43 | `(${src[t.NUMERICIDENTIFIERLOOSE]})`)
|
---|
| 44 |
|
---|
| 45 | // ## Pre-release Version Identifier
|
---|
| 46 | // A numeric identifier, or a non-numeric identifier.
|
---|
| 47 |
|
---|
| 48 | createToken('PRERELEASEIDENTIFIER', `(?:${src[t.NUMERICIDENTIFIER]
|
---|
| 49 | }|${src[t.NONNUMERICIDENTIFIER]})`)
|
---|
| 50 |
|
---|
| 51 | createToken('PRERELEASEIDENTIFIERLOOSE', `(?:${src[t.NUMERICIDENTIFIERLOOSE]
|
---|
| 52 | }|${src[t.NONNUMERICIDENTIFIER]})`)
|
---|
| 53 |
|
---|
| 54 | // ## Pre-release Version
|
---|
| 55 | // Hyphen, followed by one or more dot-separated pre-release version
|
---|
| 56 | // identifiers.
|
---|
| 57 |
|
---|
| 58 | createToken('PRERELEASE', `(?:-(${src[t.PRERELEASEIDENTIFIER]
|
---|
| 59 | }(?:\\.${src[t.PRERELEASEIDENTIFIER]})*))`)
|
---|
| 60 |
|
---|
| 61 | createToken('PRERELEASELOOSE', `(?:-?(${src[t.PRERELEASEIDENTIFIERLOOSE]
|
---|
| 62 | }(?:\\.${src[t.PRERELEASEIDENTIFIERLOOSE]})*))`)
|
---|
| 63 |
|
---|
| 64 | // ## Build Metadata Identifier
|
---|
| 65 | // Any combination of digits, letters, or hyphens.
|
---|
| 66 |
|
---|
| 67 | createToken('BUILDIDENTIFIER', '[0-9A-Za-z-]+')
|
---|
| 68 |
|
---|
| 69 | // ## Build Metadata
|
---|
| 70 | // Plus sign, followed by one or more period-separated build metadata
|
---|
| 71 | // identifiers.
|
---|
| 72 |
|
---|
| 73 | createToken('BUILD', `(?:\\+(${src[t.BUILDIDENTIFIER]
|
---|
| 74 | }(?:\\.${src[t.BUILDIDENTIFIER]})*))`)
|
---|
| 75 |
|
---|
| 76 | // ## Full Version String
|
---|
| 77 | // A main version, followed optionally by a pre-release version and
|
---|
| 78 | // build metadata.
|
---|
| 79 |
|
---|
| 80 | // Note that the only major, minor, patch, and pre-release sections of
|
---|
| 81 | // the version string are capturing groups. The build metadata is not a
|
---|
| 82 | // capturing group, because it should not ever be used in version
|
---|
| 83 | // comparison.
|
---|
| 84 |
|
---|
| 85 | createToken('FULLPLAIN', `v?${src[t.MAINVERSION]
|
---|
| 86 | }${src[t.PRERELEASE]}?${
|
---|
| 87 | src[t.BUILD]}?`)
|
---|
| 88 |
|
---|
| 89 | createToken('FULL', `^${src[t.FULLPLAIN]}$`)
|
---|
| 90 |
|
---|
| 91 | // like full, but allows v1.2.3 and =1.2.3, which people do sometimes.
|
---|
| 92 | // also, 1.0.0alpha1 (prerelease without the hyphen) which is pretty
|
---|
| 93 | // common in the npm registry.
|
---|
| 94 | createToken('LOOSEPLAIN', `[v=\\s]*${src[t.MAINVERSIONLOOSE]
|
---|
| 95 | }${src[t.PRERELEASELOOSE]}?${
|
---|
| 96 | src[t.BUILD]}?`)
|
---|
| 97 |
|
---|
| 98 | createToken('LOOSE', `^${src[t.LOOSEPLAIN]}$`)
|
---|
| 99 |
|
---|
| 100 | createToken('GTLT', '((?:<|>)?=?)')
|
---|
| 101 |
|
---|
| 102 | // Something like "2.*" or "1.2.x".
|
---|
| 103 | // Note that "x.x" is a valid xRange identifer, meaning "any version"
|
---|
| 104 | // Only the first item is strictly required.
|
---|
| 105 | createToken('XRANGEIDENTIFIERLOOSE', `${src[t.NUMERICIDENTIFIERLOOSE]}|x|X|\\*`)
|
---|
| 106 | createToken('XRANGEIDENTIFIER', `${src[t.NUMERICIDENTIFIER]}|x|X|\\*`)
|
---|
| 107 |
|
---|
| 108 | createToken('XRANGEPLAIN', `[v=\\s]*(${src[t.XRANGEIDENTIFIER]})` +
|
---|
| 109 | `(?:\\.(${src[t.XRANGEIDENTIFIER]})` +
|
---|
| 110 | `(?:\\.(${src[t.XRANGEIDENTIFIER]})` +
|
---|
| 111 | `(?:${src[t.PRERELEASE]})?${
|
---|
| 112 | src[t.BUILD]}?` +
|
---|
| 113 | `)?)?`)
|
---|
| 114 |
|
---|
| 115 | createToken('XRANGEPLAINLOOSE', `[v=\\s]*(${src[t.XRANGEIDENTIFIERLOOSE]})` +
|
---|
| 116 | `(?:\\.(${src[t.XRANGEIDENTIFIERLOOSE]})` +
|
---|
| 117 | `(?:\\.(${src[t.XRANGEIDENTIFIERLOOSE]})` +
|
---|
| 118 | `(?:${src[t.PRERELEASELOOSE]})?${
|
---|
| 119 | src[t.BUILD]}?` +
|
---|
| 120 | `)?)?`)
|
---|
| 121 |
|
---|
| 122 | createToken('XRANGE', `^${src[t.GTLT]}\\s*${src[t.XRANGEPLAIN]}$`)
|
---|
| 123 | createToken('XRANGELOOSE', `^${src[t.GTLT]}\\s*${src[t.XRANGEPLAINLOOSE]}$`)
|
---|
| 124 |
|
---|
| 125 | // Coercion.
|
---|
| 126 | // Extract anything that could conceivably be a part of a valid semver
|
---|
| 127 | createToken('COERCE', `${'(^|[^\\d])' +
|
---|
| 128 | '(\\d{1,'}${MAX_SAFE_COMPONENT_LENGTH}})` +
|
---|
| 129 | `(?:\\.(\\d{1,${MAX_SAFE_COMPONENT_LENGTH}}))?` +
|
---|
| 130 | `(?:\\.(\\d{1,${MAX_SAFE_COMPONENT_LENGTH}}))?` +
|
---|
| 131 | `(?:$|[^\\d])`)
|
---|
| 132 | createToken('COERCERTL', src[t.COERCE], true)
|
---|
| 133 |
|
---|
| 134 | // Tilde ranges.
|
---|
| 135 | // Meaning is "reasonably at or greater than"
|
---|
| 136 | createToken('LONETILDE', '(?:~>?)')
|
---|
| 137 |
|
---|
| 138 | createToken('TILDETRIM', `(\\s*)${src[t.LONETILDE]}\\s+`, true)
|
---|
| 139 | exports.tildeTrimReplace = '$1~'
|
---|
| 140 |
|
---|
| 141 | createToken('TILDE', `^${src[t.LONETILDE]}${src[t.XRANGEPLAIN]}$`)
|
---|
| 142 | createToken('TILDELOOSE', `^${src[t.LONETILDE]}${src[t.XRANGEPLAINLOOSE]}$`)
|
---|
| 143 |
|
---|
| 144 | // Caret ranges.
|
---|
| 145 | // Meaning is "at least and backwards compatible with"
|
---|
| 146 | createToken('LONECARET', '(?:\\^)')
|
---|
| 147 |
|
---|
| 148 | createToken('CARETTRIM', `(\\s*)${src[t.LONECARET]}\\s+`, true)
|
---|
| 149 | exports.caretTrimReplace = '$1^'
|
---|
| 150 |
|
---|
| 151 | createToken('CARET', `^${src[t.LONECARET]}${src[t.XRANGEPLAIN]}$`)
|
---|
| 152 | createToken('CARETLOOSE', `^${src[t.LONECARET]}${src[t.XRANGEPLAINLOOSE]}$`)
|
---|
| 153 |
|
---|
| 154 | // A simple gt/lt/eq thing, or just "" to indicate "any version"
|
---|
| 155 | createToken('COMPARATORLOOSE', `^${src[t.GTLT]}\\s*(${src[t.LOOSEPLAIN]})$|^$`)
|
---|
| 156 | createToken('COMPARATOR', `^${src[t.GTLT]}\\s*(${src[t.FULLPLAIN]})$|^$`)
|
---|
| 157 |
|
---|
| 158 | // An expression to strip any whitespace between the gtlt and the thing
|
---|
| 159 | // it modifies, so that `> 1.2.3` ==> `>1.2.3`
|
---|
| 160 | createToken('COMPARATORTRIM', `(\\s*)${src[t.GTLT]
|
---|
| 161 | }\\s*(${src[t.LOOSEPLAIN]}|${src[t.XRANGEPLAIN]})`, true)
|
---|
| 162 | exports.comparatorTrimReplace = '$1$2$3'
|
---|
| 163 |
|
---|
| 164 | // Something like `1.2.3 - 1.2.4`
|
---|
| 165 | // Note that these all use the loose form, because they'll be
|
---|
| 166 | // checked against either the strict or loose comparator form
|
---|
| 167 | // later.
|
---|
| 168 | createToken('HYPHENRANGE', `^\\s*(${src[t.XRANGEPLAIN]})` +
|
---|
| 169 | `\\s+-\\s+` +
|
---|
| 170 | `(${src[t.XRANGEPLAIN]})` +
|
---|
| 171 | `\\s*$`)
|
---|
| 172 |
|
---|
| 173 | createToken('HYPHENRANGELOOSE', `^\\s*(${src[t.XRANGEPLAINLOOSE]})` +
|
---|
| 174 | `\\s+-\\s+` +
|
---|
| 175 | `(${src[t.XRANGEPLAINLOOSE]})` +
|
---|
| 176 | `\\s*$`)
|
---|
| 177 |
|
---|
| 178 | // Star ranges basically just allow anything at all.
|
---|
| 179 | createToken('STAR', '(<|>)?=?\\s*\\*')
|
---|
| 180 | // >=0.0.0 is like a star
|
---|
| 181 | createToken('GTE0', '^\\s*>=\\s*0\.0\.0\\s*$')
|
---|
| 182 | createToken('GTE0PRE', '^\\s*>=\\s*0\.0\.0-0\\s*$')
|
---|