Skip to content

Commit 11494f1

Browse files
authoredMar 25, 2022
fix: properly escape dots in GTE0 regexes (#432)
Previously the dots were not properly escaped causing them to match any character. This caused ranges like `>=09090` to evaulate to `*` after they were incorrectly matched by the `GTE0` regex. This only happened in strict mode since in loose mode the leading 0 is allowed and parsed into `>=9090.0.0` before the `GTE0` check. After this fix, this range now will throw an error. This also affected prerelease versions in both strict and loose mode.
1 parent 9ab7b71 commit 11494f1

File tree

5 files changed

+22
-13
lines changed

5 files changed

+22
-13
lines changed
 

‎classes/range.js

+12-7
Original file line numberDiff line numberDiff line change
@@ -94,7 +94,7 @@ class Range {
9494
debug('hyphen replace', range)
9595
// `> 1.2.3 < 1.2.5` => `>1.2.3 <1.2.5`
9696
range = range.replace(re[t.COMPARATORTRIM], comparatorTrimReplace)
97-
debug('comparator trim', range, re[t.COMPARATORTRIM])
97+
debug('comparator trim', range)
9898

9999
// `~ 1.2.3` => `~1.2.3`
100100
range = range.replace(re[t.TILDETRIM], tildeTrimReplace)
@@ -108,24 +108,29 @@ class Range {
108108
// At this point, the range is completely trimmed and
109109
// ready to be split into comparators.
110110

111-
const compRe = loose ? re[t.COMPARATORLOOSE] : re[t.COMPARATOR]
112-
const rangeList = range
111+
let rangeList = range
113112
.split(' ')
114113
.map(comp => parseComparator(comp, this.options))
115114
.join(' ')
116115
.split(/\s+/)
117116
// >=0.0.0 is equivalent to *
118117
.map(comp => replaceGTE0(comp, this.options))
118+
119+
if (loose) {
119120
// in loose mode, throw out any that are not valid comparators
120-
.filter(this.options.loose ? comp => !!comp.match(compRe) : () => true)
121-
.map(comp => new Comparator(comp, this.options))
121+
rangeList = rangeList.filter(comp => {
122+
debug('loose invalid filter', comp, this.options)
123+
return !!comp.match(re[t.COMPARATORLOOSE])
124+
})
125+
}
126+
debug('range list', rangeList)
122127

123128
// if any comparators are the null set, then replace with JUST null set
124129
// if more than one comparator, remove any * comparators
125130
// also, don't include the same comparator more than once
126-
const l = rangeList.length
127131
const rangeMap = new Map()
128-
for (const comp of rangeList) {
132+
const comparators = rangeList.map(comp => new Comparator(comp, this.options))
133+
for (const comp of comparators) {
129134
if (isNullSet(comp))
130135
return [comp]
131136
rangeMap.set(comp.value, comp)

‎internal/re.js

+3-3
Original file line numberDiff line numberDiff line change
@@ -10,7 +10,7 @@ let R = 0
1010

1111
const createToken = (name, value, isGlobal) => {
1212
const index = R++
13-
debug(index, value)
13+
debug(name, index, value)
1414
t[name] = index
1515
src[index] = value
1616
re[index] = new RegExp(value, isGlobal ? 'g' : undefined)
@@ -178,5 +178,5 @@ createToken('HYPHENRANGELOOSE', `^\\s*(${src[t.XRANGEPLAINLOOSE]})` +
178178
// Star ranges basically just allow anything at all.
179179
createToken('STAR', '(<|>)?=?\\s*\\*')
180180
// >=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*$')
181+
createToken('GTE0', '^\\s*>=\\s*0\\.0\\.0\\s*$')
182+
createToken('GTE0PRE', '^\\s*>=\\s*0\\.0\\.0-0\\s*$')

‎test/classes/range.js

+2-2
Original file line numberDiff line numberDiff line change
@@ -17,9 +17,9 @@ test('range tests', t => {
1717

1818
test('range parsing', t => {
1919
t.plan(rangeParse.length)
20-
rangeParse.forEach(([range, expect, options]) => t.test(`${range} ${expect}`, t => {
20+
rangeParse.forEach(([range, expect, options]) => t.test(`${range} ${expect} ${JSON.stringify(options)}`, t => {
2121
if (expect === null)
22-
t.throws(() => new Range(range), TypeError, `invalid range: ${range}`)
22+
t.throws(() => new Range(range, options), TypeError, `invalid range: ${range} ${JSON.stringify(options)}`)
2323
else {
2424
t.equal(new Range(range, options).range || '*', expect,
2525
`${range} => ${expect}`)

‎test/fixtures/range-parse.js

+4
Original file line numberDiff line numberDiff line change
@@ -89,4 +89,8 @@ module.exports = [
8989
['<X', '<0.0.0-0'],
9090
['<x <* || >* 2.x', '<0.0.0-0'],
9191
['>x 2.x || * || <x', '*'],
92+
['>=09090', null],
93+
['>=09090', '>=9090.0.0', true],
94+
['>=09090-0', null, { includePrerelease: true }],
95+
['>=09090-0', null, { loose: true, includePrerelease: true }],
9296
]

‎test/ranges/valid.js

+1-1
Original file line numberDiff line numberDiff line change
@@ -8,5 +8,5 @@ test('valid range test', (t) => {
88
t.plan(rangeParse.length)
99
rangeParse.forEach(([pre, wanted, options]) =>
1010
t.equal(validRange(pre, options), wanted,
11-
`validRange(${pre}) === ${wanted}`))
11+
`validRange(${pre}) === ${wanted} ${JSON.stringify(options)}`))
1212
})

0 commit comments

Comments
 (0)
Please sign in to comment.