Skip to content
This repository was archived by the owner on Apr 12, 2024. It is now read-only.

Commit 4f69d38

Browse files
gkalpakpetebacondarwin
authored andcommittedFeb 24, 2017
fix($sanitize): prevent clobbered elements from freezing the browser
Closes #15699
1 parent ebe9005 commit 4f69d38

File tree

3 files changed

+50
-4
lines changed

3 files changed

+50
-4
lines changed
 
+11
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,11 @@
1+
@ngdoc error
2+
@name $sanitize:elclob
3+
@fullName Failed to sanitize html because the element is clobbered
4+
@description
5+
6+
This error occurs when `$sanitize` sanitizer is unable to traverse the HTML because one or more of the elements in the
7+
HTML have been "clobbered". This could be a sign that the payload contains code attempting to cause a DoS attack on the
8+
browser.
9+
10+
Typically clobbering breaks the `nextSibling` property on an element so that it points to one of its child nodes. This
11+
makes it impossible to walk the HTML tree without getting stuck in an infinite loop, which causes the browser to freeze.

‎src/ngSanitize/sanitize.js

+19-4
Original file line numberDiff line numberDiff line change
@@ -18,6 +18,7 @@ var forEach;
1818
var isDefined;
1919
var lowercase;
2020
var noop;
21+
var nodeContains;
2122
var htmlParser;
2223
var htmlSanitizeWriter;
2324

@@ -218,6 +219,11 @@ function $SanitizeProvider() {
218219
htmlParser = htmlParserImpl;
219220
htmlSanitizeWriter = htmlSanitizeWriterImpl;
220221

222+
nodeContains = window.Node.prototype.contains || /** @this */ function(arg) {
223+
// eslint-disable-next-line no-bitwise
224+
return !!(this.compareDocumentPosition(arg) & 16);
225+
};
226+
221227
// Regular Expressions for parsing tags and attributes
222228
var SURROGATE_PAIR_REGEXP = /[\uD800-\uDBFF][\uDC00-\uDFFF]/g,
223229
// Match everything outside of normal chars and " (quote character)
@@ -381,12 +387,12 @@ function $SanitizeProvider() {
381387
if (node.nodeType === 1) {
382388
handler.end(node.nodeName.toLowerCase());
383389
}
384-
nextNode = node.nextSibling;
390+
nextNode = getNonDescendant('nextSibling', node);
385391
if (!nextNode) {
386392
while (nextNode == null) {
387-
node = node.parentNode;
393+
node = getNonDescendant('parentNode', node);
388394
if (node === inertBodyElement) break;
389-
nextNode = node.nextSibling;
395+
nextNode = getNonDescendant('nextSibling', node);
390396
if (node.nodeType === 1) {
391397
handler.end(node.nodeName.toLowerCase());
392398
}
@@ -518,8 +524,17 @@ function $SanitizeProvider() {
518524
stripCustomNsAttrs(nextNode);
519525
}
520526

521-
node = node.nextSibling;
527+
node = getNonDescendant('nextSibling', node);
528+
}
529+
}
530+
531+
function getNonDescendant(propName, node) {
532+
// An element is clobbered if its `propName` property points to one of its descendants
533+
var nextNode = node[propName];
534+
if (nextNode && nodeContains.call(node, nextNode)) {
535+
throw $sanitizeMinErr('elclob', 'Failed to sanitize html because the element is clobbered: {0}', node.outerHTML || node.outerText);
522536
}
537+
return nextNode;
523538
}
524539
}
525540

‎test/ngSanitize/sanitizeSpec.js

+20
Original file line numberDiff line numberDiff line change
@@ -246,6 +246,26 @@ describe('HTML', function() {
246246
.toEqual('<p>text1text2</p>');
247247
});
248248

249+
it('should remove clobbered elements', function() {
Has conversations. Original line has conversations.
250+
inject(function($sanitize) {
251+
expect(function() {
252+
$sanitize('<form><input name="parentNode" /></form>');
253+
}).toThrowMinErr('$sanitize', 'elclob');
254+
255+
expect(function() {
256+
$sanitize('<form><div><div><input name="parentNode" /></div></div></form>');
257+
}).toThrowMinErr('$sanitize', 'elclob');
258+
259+
expect(function() {
260+
$sanitize('<form><input name="nextSibling" /></form>');
261+
}).toThrowMinErr('$sanitize', 'elclob');
262+
263+
expect(function() {
264+
$sanitize('<form><div><div><input name="nextSibling" /></div></div></form>');
265+
}).toThrowMinErr('$sanitize', 'elclob');
266+
});
267+
});
268+
249269

250270
describe('SVG support', function() {
251271

0 commit comments

Comments
 (0)
This repository has been archived.