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

fix($compile): Resolve various potential memory leaks (v1.2.x) #9326

Closed
wants to merge 2 commits into from
Closed
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
9 changes: 6 additions & 3 deletions src/ng/compile.js
Original file line number Diff line number Diff line change
Expand Up @@ -976,11 +976,11 @@ function $CompileProvider($provide, $$sanitizeUriProvider) {

function createBoundTranscludeFn(scope, transcludeFn, previousBoundTranscludeFn) {

var boundTranscludeFn = function(transcludedScope, cloneFn, controllers) {
var boundTranscludeFn = function(transcludedScope, cloneFn, controllers, containingScope) {
var scopeCreated = false;

if (!transcludedScope) {
transcludedScope = scope.$new();
transcludedScope = scope.$new(false, containingScope);
transcludedScope.$$transcluded = true;
scopeCreated = true;
}
Expand Down Expand Up @@ -1592,7 +1592,7 @@ function $CompileProvider($provide, $$sanitizeUriProvider) {
transcludeControllers = elementControllers;
}

return boundTranscludeFn(scope, cloneAttachFn, transcludeControllers);
return boundTranscludeFn(scope, cloneAttachFn, transcludeControllers, scopeToChild);
}
}
}
Expand Down Expand Up @@ -1754,6 +1754,8 @@ function $CompileProvider($provide, $$sanitizeUriProvider) {
boundTranscludeFn = linkQueue.shift(),
linkNode = $compileNode[0];

if (scope.$$destroyed) continue;

if (beforeTemplateLinkNode !== beforeTemplateCompileNode) {
var oldClasses = beforeTemplateLinkNode.className;

Expand Down Expand Up @@ -1784,6 +1786,7 @@ function $CompileProvider($provide, $$sanitizeUriProvider) {

return function delayedNodeLinkFn(ignoreChildLinkFn, scope, node, rootElement, boundTranscludeFn) {
var childBoundTranscludeFn = boundTranscludeFn;
if (scope.$$destroyed) return;
if (linkQueue) {
linkQueue.push(scope);
linkQueue.push(node);
Expand Down
24 changes: 21 additions & 3 deletions src/ng/rootScope.js
Original file line number Diff line number Diff line change
Expand Up @@ -182,12 +182,17 @@ function $RootScopeProvider(){
* When creating widgets, it is useful for the widget to not accidentally read parent
* state.
*
* @param {Scope} [parent=this] The {@link ng.$rootScope.Scope `Scope`} that will contain this
* the newly created scope. Defaults to `this` scope if not provided.
* This is used to ensure that $destroy events are handled correctly.
*
* @returns {Object} The newly created child scope.
*
*/
$new: function(isolate) {
var ChildScope,
child;
$new: function(isolate, parent) {
var child;

parent = parent || this;

if (isolate) {
child = new Scope();
Expand Down Expand Up @@ -220,6 +225,19 @@ function $RootScopeProvider(){
} else {
this.$$childHead = this.$$childTail = child;
}

// When the new scope is not isolated or we inherit from `this`, and
// the parent scope is destroyed, the property `$$destroyed` is inherited
// prototypically. In all other cases, this property needs to be set
// when the parent scope is destroyed.
// The listener needs to be added after the parent is set
if (isolate || parent != this) child.$on('$destroy', destroyChild);


function destroyChild() {
child.$$destroyed = true;
}

return child;
},

Expand Down
3 changes: 2 additions & 1 deletion test/.jshintrc
Original file line number Diff line number Diff line change
Expand Up @@ -162,6 +162,7 @@
"provideLog": false,
"spyOnlyCallsWithArgs": false,
"createMockStyleSheet": false,
"browserTrigger": false
"browserTrigger": false,
"jqLiteCacheSize": false
}
}
9 changes: 9 additions & 0 deletions test/helpers/testabilityPatch.js
Original file line number Diff line number Diff line change
Expand Up @@ -118,6 +118,15 @@ function dealoc(obj) {
}
}


function jqLiteCacheSize() {
var size = 0;
forEach(jqLite.cache, function() { size++; });
return size - jqLiteCacheSize.initSize;
}
jqLiteCacheSize.initSize = 0;


/**
* @param {DOMElement} element
* @param {boolean=} showNgClass
Expand Down
Loading