diff options
| author | Peter Bacon Darwin | 2013-10-05 10:49:09 +0100 | 
|---|---|---|
| committer | Vojta Jina | 2013-10-07 09:01:13 -0700 | 
| commit | 7a586e5c19f3d1ecc3fefef084ce992072ee7f60 (patch) | |
| tree | 2690c915adb20d92a065d9ad9d7438766d4620f8 /src/ng/parse.js | |
| parent | fb99f542060d3959d273634c90889788861b5c05 (diff) | |
| download | angular.js-7a586e5c19f3d1ecc3fefef084ce992072ee7f60.tar.bz2 | |
fix(*): protect calls to hasOwnProperty in public API
Objects received from outside AngularJS may have had their `hasOwnProperty`
method overridden with something else. In cases where we can do this without
incurring a performance penalty we call directly on Object.prototype.hasOwnProperty
to ensure that we use the correct method.
Also, we have some internal hash objects, where the keys for the map are provided
from outside AngularJS. In such cases we either prevent `hasOwnProperty` from
being used as a key or provide some other way of preventing our objects from
having their `hasOwnProperty` overridden.
BREAKING CHANGE: Inputs with name equal to "hasOwnProperty" are not allowed inside
form or ngForm directives.
Before, inputs whose name was "hasOwnProperty" were quietly ignored and not added
to the scope.  Now a badname exception is thrown.
Using "hasOwnProperty" for an input name would be very unusual and bad practice.
Either do not include such an input in a `form` or `ngForm` directive or change
the name of the input.
Closes #3331
Diffstat (limited to 'src/ng/parse.js')
| -rw-r--r-- | src/ng/parse.js | 22 | 
1 files changed, 20 insertions, 2 deletions
| diff --git a/src/ng/parse.js b/src/ng/parse.js index 4a1921fc..682b497b 100644 --- a/src/ng/parse.js +++ b/src/ng/parse.js @@ -290,6 +290,7 @@ Lexer.prototype = {        text: ident      }; +    // OPERATORS is our own object so we don't need to use special hasOwnPropertyFn      if (OPERATORS.hasOwnProperty(ident)) {        token.fn = OPERATORS[ident];        token.json = OPERATORS[ident]; @@ -938,6 +939,9 @@ function cspSafeGetterFn(key0, key1, key2, key3, key4, fullExp) {  }  function getterFn(path, csp, fullExp) { +  // Check whether the cache has this getter already. +  // We can use hasOwnProperty directly on the cache because we ensure, +  // see below, that the cache never stores a path called 'hasOwnProperty'    if (getterFnCache.hasOwnProperty(path)) {      return getterFnCache[path];    } @@ -986,7 +990,12 @@ function getterFn(path, csp, fullExp) {      fn.toString = function() { return code; };    } -  return getterFnCache[path] = fn; +  // Only cache the value if it's not going to mess up the cache object +  // This is more performant that using Object.prototype.hasOwnProperty.call +  if (path !== 'hasOwnProperty') { +    getterFnCache[path] = fn; +  } +  return fn;  }  /////////////////////////////////// @@ -1036,6 +1045,7 @@ function $ParseProvider() {      return function(exp) {        var lexer = new Lexer($sniffer.csp);        var parser = new Parser(lexer, $filter, $sniffer.csp); +      var parsedExpression;        switch (typeof exp) {          case 'string': @@ -1043,7 +1053,15 @@ function $ParseProvider() {              return cache[exp];            } -          return cache[exp] = parser.parse(exp, false); +          parsedExpression = parser.parse(exp, false); + +          if (exp !== 'hasOwnProperty') { +            // Only cache the value if it's not going to mess up the cache object +            // This is more performant that using Object.prototype.hasOwnProperty.call +            cache[exp] = parsedExpression; +          } + +          return parsedExpression;          case 'function':            return exp; | 
