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

Commit 06f479c

Browse files
committed
fix(testability): findBindings should escape binding expressions before performing regexp
Move the function to escape regexps to Angular.js, fix the link, and use it in the $$testability service. See #9595
1 parent 0f6aa10 commit 06f479c

File tree

4 files changed

+27
-10
lines changed

4 files changed

+27
-10
lines changed

src/Angular.js

+9
Original file line numberDiff line numberDiff line change
@@ -49,6 +49,7 @@
4949
isBoolean: true,
5050
isPromiseLike: true,
5151
trim: true,
52+
escapeForRegexp: true,
5253
isElement: true,
5354
makeMap: true,
5455
size: true,
@@ -586,6 +587,14 @@ var trim = function(value) {
586587
return isString(value) ? value.trim() : value;
587588
};
588589

590+
// Copied from:
591+
// http://docs.closure-library.googlecode.com/git/local_closure_goog_string_string.js.source.html#line1021
592+
// Prereq: s is a string.
593+
var escapeForRegexp = function(s) {
594+
return s.replace(/([-()\[\]{}+?*.$\^|,:#<!\\])/g, '\\$1').
595+
replace(/\x08/g, '\\x08');
596+
};
597+
589598

590599
/**
591600
* @ngdoc function

src/ng/sce.js

-9
Original file line numberDiff line numberDiff line change
@@ -14,15 +14,6 @@ var SCE_CONTEXTS = {
1414

1515
// Helper functions follow.
1616

17-
// Copied from:
18-
// http://docs.closure-library.googlecode.com/git/closure_goog_string_string.js.source.html#line962
19-
// Prereq: s is a string.
20-
function escapeForRegexp(s) {
21-
return s.replace(/([-()\[\]{}+?*.$\^|,:#<!\\])/g, '\\$1').
22-
replace(/\x08/g, '\\x08');
23-
}
24-
25-
2617
function adjustMatcher(matcher) {
2718
if (matcher === 'self') {
2819
return matcher;

src/ng/testability.js

+1-1
Original file line numberDiff line numberDiff line change
@@ -34,7 +34,7 @@ function $$TestabilityProvider() {
3434
if (dataBinding) {
3535
forEach(dataBinding, function(bindingName) {
3636
if (opt_exactMatch) {
37-
var matcher = new RegExp('(^|\\s)' + expression + '(\\s|\\||$)');
37+
var matcher = new RegExp('(^|\\s)' + escapeForRegexp(expression) + '(\\s|\\||$)');
3838
if (matcher.test(bindingName)) {
3939
matches.push(binding);
4040
}

test/ng/testabilitySpec.js

+17
Original file line numberDiff line numberDiff line change
@@ -88,6 +88,23 @@ describe('$$testability', function() {
8888
expect(names[0]).toBe(element.find('li')[0]);
8989
});
9090

91+
it('should find bindings with allowed special characters', function() {
92+
element =
93+
'<div>' +
94+
' <span>{{$index}}</span>' +
95+
' <span>{{foo.bar}}</span>' +
96+
' <span>{{foonbar}}</span>' +
97+
'</div>';
98+
element = $compile(element)(scope);
99+
var indexes = $$testability.findBindings(element[0], '$index', true);
100+
expect(indexes.length).toBe(1);
101+
expect(indexes[0]).toBe(element.find('span')[0]);
102+
103+
var foobars = $$testability.findBindings(element[0], 'foo.bar', true);
104+
expect(foobars.length).toBe(1); // it should not match {{foonbar}}
105+
expect(foobars[0]).toBe(element.find('span')[1]);
106+
});
107+
91108
it('should find partial models', function() {
92109
element =
93110
'<div>' +

0 commit comments

Comments
 (0)