Skip to content

Commit f9ac319

Browse files
committed
add safeRegex query option
1 parent 933edd5 commit f9ac319

File tree

5 files changed

+44
-1
lines changed

5 files changed

+44
-1
lines changed

CHANGELOG.md

+5
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,10 @@
11
# mongo-url-parser changelog
22

3+
## NEXT
4+
5+
- added `safeRegex` `query` option
6+
+ checks regex query operator values for regex which could be used as a DoS
7+
38
## 1.4.0 (2017/11/22)
49

510
- added support for `$not` query operator in combination with most operators

README.md

+15
Original file line numberDiff line numberDiff line change
@@ -164,6 +164,21 @@ var options = {
164164
mongoUrlUtils({query: 'regex(email,"[email protected]",i)'}, options);
165165
```
166166

167+
#### regex safety checking
168+
The `regex` query operator can be dangerous allowing attackers to create an expensive expression.
169+
Set `safeRegex` to `true` to pass all regexes through [safe-regex](https://github.com/substack/safe-regex)
170+
and throw an error if an unsafe regex is sent.
171+
172+
```js
173+
var options = {
174+
query: {
175+
safeRegex: true
176+
}
177+
};
178+
mongoUrlUtils({query: 'regex(email,"(a+){10}y")'}, options);
179+
// throws Error
180+
```
181+
167182
#### mongo types
168183
The `type()` query operator allows either integer identifiers as per the mongodb documentation. For
169184
convinience it also maps the following types to their ids: `Double`, `String`, `Object`, `Array`,

package.json

+2-1
Original file line numberDiff line numberDiff line change
@@ -28,7 +28,8 @@
2828
"author": "Sean Garner <[email protected]>",
2929
"license": "MIT",
3030
"dependencies": {
31-
"is": "^3.0.1"
31+
"is": "^3.0.1",
32+
"safe-regex": "^1.1.0"
3233
},
3334
"devDependencies": {
3435
"in-publish": "^2.0.0",

src/query.pegjs

+10
Original file line numberDiff line numberDiff line change
@@ -1,8 +1,11 @@
11
{
2+
const safeRegex = require('safe-regex');
3+
24
//TODO: disabled presets (e.g. mongo 2.2/2.6/3.0)
35
//TODO: determine dependencies automatically
46
if (!Array.isArray(options.disabledOperators)) options.disabledOperators = [];
57
if (options.caseInsensitiveOperators === undefined) options.caseInsensitiveOperators = false;
8+
if (options.safeRegex === undefined) options.safeRegex = false;
69

710
function collect(head, tail) {
811
var res = [head];
@@ -58,6 +61,12 @@
5861
throw new Error(dateTimeStr + ' is not a valid date time string');
5962
}
6063
}
64+
65+
function assertSafeRegex(regex) {
66+
if (options.safeRegex && !safeRegex(regex)) {
67+
throw new Error('regex not safe; too many repetitions or star height is above 1');
68+
}
69+
}
6170
}
6271

6372
start
@@ -142,6 +151,7 @@ ArrayComparison
142151
Regex
143152
= "regex(" __ prop:Property __ "," __ pattern:String __ opts:("," __ [imxs]+ __)? ")" {
144153
assertCan('regex');
154+
assertSafeRegex(pattern);
145155
if (opts) return set({}, prop, {$regex: pattern, $options: opts[2].join('')});
146156
return set({}, prop, {$regex: pattern});
147157
}

test/query.js

+12
Original file line numberDiff line numberDiff line change
@@ -141,6 +141,18 @@ describe('query', () => {
141141
});
142142

143143
describe('options', () => {
144+
describe('safeRegex', () => {
145+
it('should default to disabled', () => {
146+
expect(query.bind(null, 'regex(email,"(a+){10}")')).to.not.throw();
147+
});
148+
it('should throw when an unsafe regex is used', () => {
149+
expect(query.bind(null, 'regex(email,"(a+){10}")', {safeRegex:true})).to.throw('regex not safe');
150+
});
151+
it('should not throw when the regex is safe', () => {
152+
expect(query.bind(null, 'regex(email,"(a){10}")', {safeRegex:true})).to.not.throw();
153+
});
154+
});
155+
144156
describe('disabledOperators', () => {
145157
it('take an array of keywords to disable', () => {
146158
expect(query.bind(null, 'regex(email,".*\\\\.gmail\\\\.com")', {disabledOperators:['regex']}))

0 commit comments

Comments
 (0)