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

fix(ng-bind-html): deep watch the actual $sce value, not its toString. #14527

Closed
wants to merge 1 commit into from

Conversation

mprobst
Copy link
Contributor

@mprobst mprobst commented Apr 27, 2016

Custom $sce implementations might not have a toString(), or it might be
compiled away in non-debug mode. Deep watching the object is the correct fix
with Angular's regular watch semantics.

The performance of this should be equivalent - toString() on objects usually
touches all fields, so a deep object watch will be in the same ballpark, except
that it avoids the (potentially big) string allocation.

Fixes #14526.

@mprobst
Copy link
Contributor Author

mprobst commented Apr 27, 2016

@rjamet FYI

$compile.$$addBindingClass(tElement);

return function ngBindHtmlLink(scope, element, attr) {
$compile.$$addBindingInfo(element, attr.ngBindHtml);

scope.$watch(ngBindHtmlWatch, function ngBindHtmlWatchAction() {
scope.$watch(ngBindHtmlGetter, function ngBindHtmlWatchAction(newVal) {
// we re-evaluate the expr because we want a TrustedValueHolderType
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

update the comment please

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

fixed.

@IgorMinar
Copy link
Contributor

IgorMinar commented Apr 27, 2016

lgtm.

@gkalpak, @petebacondarwin can you please help to merge this in?

@mprobst mprobst force-pushed the deep-equal-ng.html branch 2 times, most recently from 3c7d36b to 8063521 Compare April 27, 2016 21:57
@gkalpak
Copy link
Member

gkalpak commented Apr 28, 2016

FWIW, I don't think performance will be the same (in all cases). Running angular.equals() on every digest is more expensive that comparing strings. But ngBindHtml shouldn't be so common, so the overall impact won't make any practical difference.

I wonder whether it would be worth it to use toString() if the built-in $sceDelegate is used or to always use $sce.valueOf() (with swallow $watch) instead of toString() ?

Anyway, I'll merge this once Travis is green and we can make further changes in subsequent PRs.

@gkalpak
Copy link
Member

gkalpak commented Apr 28, 2016

The failures are legitimate. Basically, due to how equals() works (and how $sceDelegate builds trusted types), any vaues returned by $sce.trustAsHtml('...') are considered identical.

Here is a failing testcase:

inject(function($compile, $rootScope, $sce) {
  $rootScope.getHtml = function() { return $sce.trustAsHtml($rootScope.snippet); };
  element = $compile('<div ng-bind-html="getHtml()"></div>')($rootScope);

  $rootScope.$apply('snippet = 1');
  expect(element.text()).toBe('1');

  $rootScope.$apply('snippet = 2');
  expect(element.text()).toBe('2');
      // It is still '1', because
      // `angular.equals($sce.trustAsHtml('1'), $sce.trustAsHtml('2')) === true` 
});

@gkalpak
Copy link
Member

gkalpak commented Apr 28, 2016

We can fix this by exposing the trusted value as a public property on $sceDelegate's wrapper type (so that angular.equals detects them as different), but this change might still break existing implementations of custom $sceDelegates (the same way it broke the built-in one).

@petebacondarwin
Copy link
Contributor

@mprobst can you take a look at @gkalpak's comments?

@rjamet
Copy link
Contributor

rjamet commented Apr 28, 2016

Exposing the internal trusted value is what $sce.valueOf does, and it returns non-sce types as-is. It's part of the public interface of the $sce, so I think it's the right tool here. (code: https://github.com/angular/angular.js/blob/master/src/ng/sce.js#L330)

Custom $sce implementations might not have a `toString()`, or it might be
compiled away in non-debug mode. Deep watching the object is the correct fix
with Angular's regular watch semantics.

The performance of this should be equivalent - `toString()` on objects usually
touches all fields, so a deep object watch will be in the same ballpark, except
that it avoids the (potentially big) string allocation.
@mprobst
Copy link
Contributor Author

mprobst commented Apr 28, 2016

@gkalpak you're right re performance, assuming most of the time the string return is reference identical and we don't need to compare a (potentially long) string, we should be fine. But it should be a wash indeed, and the updated PR should have the same characteristics as before.

@rjamet thanks for the tip with valueOf, that's indeed the right approach. I've amended the PR. I also added a unit test for ngBindHtml updating values on digest loops, i.e. the that the specific watch getter that it uses actually fires.

@mprobst mprobst force-pushed the deep-equal-ng.html branch from 8063521 to f247b0d Compare April 28, 2016 14:51
@gkalpak
Copy link
Member

gkalpak commented Apr 28, 2016

LGTM

gkalpak pushed a commit that referenced this pull request Apr 28, 2016
…instead of `toString()`)

Custom `$sce` implementations might not provide a `toString()` method on the wrapped object, or it
might be compiled away in non-debug mode. Watching the unwrapped value (retrieved using
`$sce.valueOf()`) fixes the problem.

The performance of this should be equivalent - `toString()` on objects usually touches all fields,
plus we will also avoid the (potentially big) string allocation.

Fixes #14526
Closes #14527
@gkalpak gkalpak closed this in 707664a Apr 28, 2016
gkalpak pushed a commit that referenced this pull request Apr 28, 2016
…instead of `toString()`)

Custom `$sce` implementations might not provide a `toString()` method on the wrapped object, or it
might be compiled away in non-debug mode. Watching the unwrapped value (retrieved using
`$sce.valueOf()`) fixes the problem.

The performance of this should be equivalent - `toString()` on objects usually touches all fields,
plus we will also avoid the (potentially big) string allocation.

Fixes #14526
Closes #14527
@gkalpak
Copy link
Member

gkalpak commented Apr 28, 2016

I tweaked the commit message a bit (because it was describing the previous approach with deep-watching) and merged.

@mprobst
Copy link
Contributor Author

mprobst commented Apr 28, 2016

Thanks @gkalpak!

Sign up for free to subscribe to this conversation on GitHub. Already have an account? Sign in.
Labels
Projects
None yet
Development

Successfully merging this pull request may close these issues.

ngBindHtml does not work if SafeHtml types don't have a toString function
6 participants