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

Commit 0b474b3

Browse files
committed
fix(jqLite): prevent possible XSS due to regex-based HTML replacement
1 parent a31c207 commit 0b474b3

File tree

5 files changed

+138
-14
lines changed

5 files changed

+138
-14
lines changed

src/.eslintrc.json

+1
Original file line numberDiff line numberDiff line change
@@ -100,6 +100,7 @@
100100
"VALIDITY_STATE_PROPERTY": false,
101101
"reloadWithDebugInfo": false,
102102
"stringify": false,
103+
"UNSAFE_restoreLegacyJqLiteXHTMLReplacement": false,
103104

104105
"NODE_TYPE_ELEMENT": false,
105106
"NODE_TYPE_ATTRIBUTE": false,

src/Angular.js

+21
Original file line numberDiff line numberDiff line change
@@ -93,6 +93,7 @@
9393
hasOwnProperty,
9494
createMap,
9595
stringify,
96+
UNSAFE_restoreLegacyJqLiteXHTMLReplacement,
9697
9798
NODE_TYPE_ELEMENT,
9899
NODE_TYPE_ATTRIBUTE,
@@ -1949,6 +1950,26 @@ function bindJQuery() {
19491950
bindJQueryFired = true;
19501951
}
19511952

1953+
/**
1954+
* @ngdoc function
1955+
* @name angular.UNSAFE_restoreLegacyJqLiteXHTMLReplacement
1956+
* @module ng
1957+
* @kind function
1958+
*
1959+
* @description
1960+
* Restores the pre-1.8 behavior of jqLite that turns XHTML-like strings like
1961+
* `<div /><span />` to `<div></div><span></span>` instead of `<div><span></span></div>`.
1962+
* The new behavior is a security fix so if you use this method, please try to adjust
1963+
* to the change & remove the call as soon as possible.
1964+
1965+
* Note that this only patches jqLite. If you use jQuery 3.5.0 or newer, please read
1966+
* [jQuery 3.5 upgrade guide](https://jquery.com/upgrade-guide/3.5/) for more details
1967+
* about the workarounds.
1968+
*/
1969+
function UNSAFE_restoreLegacyJqLiteXHTMLReplacement() {
1970+
JQLite.legacyXHTMLReplacement = true;
1971+
}
1972+
19521973
/**
19531974
* throw error if the argument is falsy.
19541975
*/

src/AngularPublic.js

+1
Original file line numberDiff line numberDiff line change
@@ -156,6 +156,7 @@ function publishExternalAPI(angular) {
156156
'callbacks': {$$counter: 0},
157157
'getTestability': getTestability,
158158
'reloadWithDebugInfo': reloadWithDebugInfo,
159+
'UNSAFE_restoreLegacyJqLiteXHTMLReplacement': UNSAFE_restoreLegacyJqLiteXHTMLReplacement,
159160
'$$minErr': minErr,
160161
'$$csp': csp,
161162
'$$encodeUriSegment': encodeUriSegment,

src/jqLite.js

+31-14
Original file line numberDiff line numberDiff line change
@@ -90,6 +90,16 @@
9090
* - [`val()`](http://api.jquery.com/val/)
9191
* - [`wrap()`](http://api.jquery.com/wrap/)
9292
*
93+
* jqLite also provides a method restoring pre-1.8 insecure treatment of XHTML-like tags
94+
* that makes input like `<div /><span />` turned to `<div></div><span></span>` instead of
95+
* `<div><span></span></div>` like version 1.8 & newer do:
96+
* ```js
97+
* angular.UNSAFE_restoreLegacyJqLiteXHTMLReplacement();
98+
* ```
99+
* Note that this only patches jqLite. If you use jQuery 3.5.0 or newer, please read
100+
* [jQuery 3.5 upgrade guide](https://jquery.com/upgrade-guide/3.5/) for more details
101+
* about the workarounds.
102+
*
93103
* ## jQuery/jqLite Extras
94104
* AngularJS also provides the following additional methods and events to both jQuery and jqLite:
95105
*
@@ -170,19 +180,22 @@ var TAG_NAME_REGEXP = /<([\w:-]+)/;
170180
var XHTML_TAG_REGEXP = /<(?!area|br|col|embed|hr|img|input|link|meta|param)(([\w:-]+)[^>]*)\/>/gi;
171181

172182
var wrapMap = {
173-
'option': [1, '<select multiple="multiple">', '</select>'],
174-
175-
'thead': [1, '<table>', '</table>'],
176-
'col': [2, '<table><colgroup>', '</colgroup></table>'],
177-
'tr': [2, '<table><tbody>', '</tbody></table>'],
178-
'td': [3, '<table><tbody><tr>', '</tr></tbody></table>'],
179-
'_default': [0, '', '']
183+
thead: ['table'],
184+
col: ['colgroup', 'table'],
185+
tr: ['tbody', 'table'],
186+
td: ['tr', 'tbody', 'table']
180187
};
181188

182-
wrapMap.optgroup = wrapMap.option;
183189
wrapMap.tbody = wrapMap.tfoot = wrapMap.colgroup = wrapMap.caption = wrapMap.thead;
184190
wrapMap.th = wrapMap.td;
185191

192+
// Support: IE <=9 only
193+
// IE <=9 replaces <option> tags with their contents when inserted outside of
194+
// the select element.
195+
if (msie < 10) {
196+
wrapMap.optgroup = wrapMap.option = [1, '<select multiple="multiple">', '</select>'];
197+
}
198+
186199

187200
function jqLiteIsTextNode(html) {
188201
return !HTML_REGEXP.test(html);
@@ -214,15 +227,19 @@ function jqLiteBuildFragment(html, context) {
214227
// Convert html into DOM nodes
215228
tmp = fragment.appendChild(context.createElement('div'));
216229
tag = (TAG_NAME_REGEXP.exec(html) || ['', ''])[1].toLowerCase();
217-
wrap = wrapMap[tag] || wrapMap._default;
218-
tmp.innerHTML = wrap[1] + html.replace(XHTML_TAG_REGEXP, '<$1></$2>') + wrap[2];
230+
wrap = wrapMap[tag] || [];
219231

220-
// Descend through wrappers to the right content
221-
i = wrap[0];
222-
while (i--) {
223-
tmp = tmp.lastChild;
232+
// Create wrappers & descend into them.
233+
i = wrap.length;
234+
while (--i > -1) {
235+
tmp.appendChild(window.document.createElement(wrap[i]));
236+
tmp = tmp.firstChild;
224237
}
225238

239+
tmp.innerHTML = JQLite.legacyXHTMLReplacement ?
240+
html.replace(XHTML_TAG_REGEXP, '<$1></$2>') :
241+
html;
242+
226243
nodes = concat(nodes, tmp.childNodes);
227244

228245
tmp = fragment.firstChild;

test/jqLiteSpec.js

+84
Original file line numberDiff line numberDiff line change
@@ -169,6 +169,90 @@ describe('jqLite', function() {
169169
expect(nodes[0].nodeName.toLowerCase()).toBe(name);
170170
});
171171
});
172+
173+
describe('security', function() {
174+
it('shouldn\'t crash at attempts to close the table wrapper', function() {
175+
// jQuery doesn't pass this test yet.
176+
if (!_jqLiteMode) return;
177+
178+
expect(function() {
179+
// This test case attempts to close the tags which wrap input
180+
// based on matching done in wrapMap, escaping the wrapper & thus
181+
// triggering an error when descending.
182+
var el = jqLite('<td></td></tr></tbody></table><td></td>');
183+
expect(el.length).toBe(2);
184+
expect(el[0].nodeName.toLowerCase()).toBe('td');
185+
expect(el[1].nodeName.toLowerCase()).toBe('td');
186+
}).not.toThrow();
187+
});
188+
189+
it('shouldn\'t unsanitize sanitized code', function(done) {
190+
// jQuery <3.5.0 fail those tests.
191+
if (isJQuery2x()) {
192+
done();
193+
return;
194+
}
195+
196+
var counter = 0,
197+
assertCount = 13,
198+
container = jqLite('<div></div>');
199+
200+
function donePartial() {
201+
counter++;
202+
if (counter === assertCount) {
203+
container.remove();
204+
delete window.xss;
205+
done();
206+
}
207+
}
208+
209+
jqLite(document.body).append(container);
210+
window.xss = jasmine.createSpy('xss');
211+
212+
// Thanks to Masato Kinugawa from Cure53 for providing the following test cases.
213+
// Note: below test cases need to invoke the xss function with consecutive
214+
// decimal parameters for the assertions to be correct.
215+
forEach([
216+
'<img alt="<x" title="/><img src=url404 onerror=xss(0)>">',
217+
'<img alt="\n<x" title="/>\n<img src=url404 onerror=xss(1)>">',
218+
'<style><style/><img src=url404 onerror=xss(2)>',
219+
'<xmp><xmp/><img src=url404 onerror=xss(3)>',
220+
'<title><title /><img src=url404 onerror=xss(4)>',
221+
'<iframe><iframe/><img src=url404 onerror=xss(5)>',
222+
'<noframes><noframes/><img src=url404 onerror=xss(6)>',
223+
'<noscript><noscript/><img src=url404 onerror=xss(7)>',
224+
'<foo" alt="" title="/><img src=url404 onerror=xss(8)>">',
225+
'<img alt="<x" title="" src="/><img src=url404 onerror=xss(9)>">',
226+
'<noscript/><img src=url404 onerror=xss(10)>',
227+
'<noembed><noembed/><img src=url404 onerror=xss(11)>',
228+
229+
'<option><style></option></select><img src=url404 onerror=xss(12)></style>'
230+
], function(htmlString, index) {
231+
var element = jqLite('<div></div>');
232+
233+
container.append(element);
234+
element.append(jqLite(htmlString));
235+
236+
window.setTimeout(function() {
237+
expect(window.xss).not.toHaveBeenCalledWith(index);
238+
donePartial();
239+
}, 1000);
240+
});
241+
});
242+
243+
it('should allow to restore legacy insecure behavior', function() {
244+
// jQuery doesn't have this API.
245+
if (!_jqLiteMode) return;
246+
247+
// eslint-disable-next-line new-cap
248+
angular.UNSAFE_restoreLegacyJqLiteXHTMLReplacement();
249+
250+
var elem = jqLite('<div/><span/>');
251+
expect(elem.length).toBe(2);
252+
expect(elem[0].nodeName.toLowerCase()).toBe('div');
253+
expect(elem[1].nodeName.toLowerCase()).toBe('span');
254+
});
255+
});
172256
});
173257

174258
describe('_data', function() {

0 commit comments

Comments
 (0)