-
Notifications
You must be signed in to change notification settings - Fork 165
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
chain of never resolved promises create memory leaks #179
Comments
This is part of the reason for having 2.3.2 in the Promise Resolution Procedure:
This is separate from the specification of the resolution procedure for thenables. Because it doesn't specify how that state is adopted, we allow the promise to use some internal means to unravel the stack (providing all promises come from the same implementation). This probably hasn't been implemented in practice though, and the spec does not explicitly state that a reference should not be held. I'll have a go at testing and fixing this in then/promise and post back here to let you know how that goes. |
I have made something that I think fixes the memory leak in then/promise (then/promise#67) It would be great if you could give it a more thorough testing to see when/if it fails. I'm particularly concerned that it might have created memory leaks where there was none before. |
We implement a job queue using this exact pattern, should we be worried? Seem like a mistake a lot of developer will make, eg. http://stackoverflow.com/questions/15027192/how-do-i-stop-memory-leaks-with-recursive-javascript-promises Please let me know if I missed the point... |
@bitinn yes, unfortunately that will leak memory (until the promise chain unravells) with most current promise implementations. It should work fine with the fix linked to above (using then/promise rather than Q). The solution is very similar to the concept of tail recursion in synchronous code. |
Fixes the memory described at promises-aplus/promises-spec#179
seem like if I do this it won't leak (return
|
@bitinn much simpler would just be leaving out the return if you don't need it var i = 0;
function run() {
new Promise(function(resolve) {
if (i % 10000 === 0) {
global.gc();
console.log(process.memoryUsage());
}
i++;
setImmediate(resolve);
}).then(run);
}
run(); |
@petkaantonov got it, cheers, we kinda need it to return promise as we called it with |
var p2 = new Promise(function() {
return p;
}); is equivalent to: var p2 = new Promise(function() {
}); because the return value of the factory function is ignored. You probably meant to do: var p2 = new Promise(function(resolve) {
resolve(p);
}); which would have still leaked. For the original example of doing something asynchronous continuously: function run() {
return doAsyncWork().then(run);
}
run(); you could fix the memory leak in all promise implementations by breaking the chain of promises: function run() {
return new Promise(function (resolve, reject) {
function recurse() {
doAsyncWork().then(recurse, reject);
}
recurse();
});
}
run(); It's a little dirty, but by having an inner |
An interesting question, should we specify that promises are not permitted to leak memory in this use case? What would such a specification look like. |
Well the whole 2.3.2 needs to be redone. As far as I can tell we both fixed the memory leak in our implementations by turning This is different from what the spec currently says, In the case |
The spec as written today does allow implementors to avoid this leak. This leads me to believe that it may not be a spec concern rather the concern of implementors. I suppose one could argue that if many implementations exhibit this behavior, the spec wording may need to be refined. That being said, this feels like the responsibility of a supplementary compliance test suite. |
@stefanpenner As suggested by @petkaantonov, we haven't really done what the spec says to do. In order to fix the leak we had to make promise into a proxy for x, which is not quite the same thing as making promise adopt the state of x. The difference is not observable to the outside world, but we aren't really complying with what the spec says to do. |
I believe the spec intends to describe the observable behavior, not implementation. I do believe the wording can be improved, but I am nervous of the spec dictating implementation details. |
What I just specified does change observable behavior var resolveA;
var a = new Promise(function() {
resolveA = arguments[0];
});
a.then(function() {
console.log("first");
});
var resolveB;
var b = new Promise(function() {
resolveB = arguments[0];
});
b.then(function() {
console.log("second");
});
resolveA(b);
b.then(function() {
console.log("third");
});
resolveB(); |
Bluebird before the fix (and e.g. Q, both leak) will log second, third, first While after the fix it logs second, first, third. |
@petkaantonov ah, since during followers merge some ordering information is lost. |
i wonder if the 2.3.2 |
I suspect the solution is to refine and elaborate on the "adoption" algorithm |
This may by true, but now their is an observable difference between situations were all promises are ownPromises, vs a mixture of foreign and ownPromises. This seems quite bad. |
Yes you will get different order if you do this: var Promise = require("bluebird");
var Thenable = require("rsvp").Promise;
var resolveA;
var a = new Promise(function() {
resolveA = arguments[0];
});
a.then(function() {
console.log("first");
});
var resolveB;
var b = new Thenable(function() {
resolveB = arguments[0];
});
b.then(function() {
console.log("second");
});
resolveA(b);
b.then(function() {
console.log("third");
});
resolveB(); However, 2 of the |
Up until this discussion I would have expected the interaction between 2 spec compliant implementations to not be observably different from the same interactions happening between promises of 1 implementation. If it wasn't for the observable differences when mixing 2 implementations, I would prefer the ordering of the "fixed" adoption strategy. Unfortunately with this difference, I am now unsure. Illustrating the differences [https://gist.github.com/stefanpenner/10afbee137180c2e7862] rsvp (pre fix) x bluebird (post fix)
rsvp (post fix) x bluebird (post fix)
rsvp (pre fix) x native
rsvp(pre fix) x then/promise (post fix)
rsvp(pre fix) x then/promise (pre fix)
RSVP (pre fix) X when
[edit: added when] RSVP (pre fix) X when (master)
[edit added: when#master] |
If one treats all promises as "foreign" (i.e., ignore 2.3.2 entirely and only use 2.3.3.3) then the order is second-third-first, if I'm not mistaken. Should that be considered the canonical order |
Conceptually, it seems reasonable to say that since A is resolved to B (making A and B effectively indistinguishable), resolving B causes A and B to become fulfilled simultaneously. IOW, there is no time when B is fulfilled and A is not, and vice versa. Then, |
It is great to see when can achieve both reasonably here. |
In that case, should 2.3.2 just be dropped from the spec, along with the concept of own-vs-foreign promises? Let all promises be handled by the generic thenable case? Implementations can still implement 2.3.2 on their own as an optimization, of course. Also, does this affect |
@Arnavion The point is that if you drop 2.3.2, implementations would not be allowed to implement that optimisation while adhering to the spec. The spec would then be forcing them to implement it in the way that leaks memory. Also, we should probably specify that implementations should not leak memory in this case (which requires the optimisation).
|
I don't think implementations would be prohibited from implementing 2.3.2 just because it's not in the spec. They only need to implement it in such a way that it gives the same result as 2.3.3.3 would have. The spec shouldn't be expected to be translated line-by-line to code. Of course, we can also just keep 2.3.2 and modify it to read something like that instead. Eg. "If x is a promise, then adopt its state in such a way that it gives the same results as 2.3.3.3" |
I think the best way to think of this is as being a request for tail recursion on promises. i.e. function run(i) {
return i < 99999999 ? run(i + 1) : i;
}
console.log(run(0)); would run out of stack and cause an exception. Similarly: function run(i) {
return new Promise(function(resolve) {
setImmediate(resolve);
}).then(function () {
return i < 99999999 ? run(i + 1) : i;
});
}
run(0).then(function (result) {
console.log(result);
}); Maintains an ever increasing stack of promises, resulting in an out of memory error. |
One issue, in the case that
The third option is to make use of weak-references in order to squash the chain in both directions. It works something like this:
At the end of that process, the head and tail ( |
As for test case, The only way I've figured out of testing that this is working is to profile the memory as in https://github.com/then/promise/blob/master/test/memory-leak.js The ordering change is that for: var A = new Promise();
var B = new Promise();
RESOLVE(B, A);
B.then(function () {
console.log('B');
});
A.then(function () {
console.log('A');
});
RESOLVE(A, null); would log B, A instead of A, B unless you do some clever workaround. |
@domenic Let me try to put this down: Nothing needs to change in the current Promises/A+ spec. There might be amendments to make, of varying strength, to the "adopt state" step, as point 2.3.2.4:
(and respectively for rejection and The first specifies basically what would happen when a thenable would be assimilated, where a We further might want to nail down the behaviour (callback ordering) when A test case could be derived from the examples given above (without the Going to look at the ES6 spec now. |
Ah thanks at @ForbesLindesay I obviously hadn't seen your post. Yes, that's exactly what we want to get. This third option with the "weak references" is just what I have implemented in my own Promise library. And indeed, my library does do |
@bergus how do you do weak references? there is just a WeakMap. Are you using that? |
@petkaantonov no, I use an "indirect reference". Promises that adopt each other share a common handle that contains a reference to innermost adopted promise (somewhere around https://github.com/bergus/F-Promise/blob/b66e917e2bf8836dab824875e77170f088e8361c/Promise.js#L151). This handle is |
I don't think anything specific needs to change. Except, if we wanted to spec tail-adoption-optimisation, probably everything would need to change (promise capabilities getting resolved lazily etc). |
That basically gives me no new information. Do we need to spec tail-adoption-optimization to avoid the memory leak? If so what are the observable consequences---just ordering changes? |
Need? Probably not. I think the spec does already allow optimised implementations. Do we want to specify it? Maybe, I can't tell. That's a design decision, just as optimised tail recursion was.
Yes, if ordering was changed this could simplify a few things. I'm not sure whether it is required at all to avoid the memory leak, but I guess it would need some very clever implementation strategies to avoid the memory leak and guarantee the ordering currently specced by the PromiseJobs. If this was changed, a possibly different ordering of callbacks would be the only observable difference (and your app not breaking, of course), the rest of the user-visible methods would behave the same. If someone subclassed promises however, this might break the optimisation. Not sure whether we can have both. |
OK, so my summary of where we are now with regard to ES6 is that people think it might be possible to maintain the current observable semantics with no spec changes and also avoid memory leaks, but they are unsure, and think that it would probably be easier to avoid memory leaks if we changed the ordering. |
That seems like a reasonable summary. At the moment, A+ doesn't even specify this particular ordering. I'm not sure about ES6 though. I do think we should find a way to specify that this optimisation must be performed, because code that assumes it's implemented will break in really hard to debug ways if the implementation doesn't support it. |
@domenic OK I reviewed this and found something that will need to change semantically (besides possibly the ordering)
see https://mail.mozilla.org/pipermail/es-discuss/2015-April/042517.html |
I think the following example is also symptomatic of the problem, without any recursion involved: const x0 = new Promise(r => global.resolveX0 = r);
const x1 = Promise.resolve(x0);
const x2 = Promise.resolve(x1); The reference chain here, even after GC, becomes This occurs because Obviously you can add arbitrary intermediate levels to the chain. |
@ForbesLindesay I do not understand (probably because I am missing something obvious) how your solution in tc39/proposal-cancelable-promises#1 will work in the following case: const x0 = new Promise(r => global.resolveX0 = r);
const x1 = Promise.resolve(x0);
const x2 = Promise.resolve(x1);
x2.then(v => console.log(v)); If I read your solution correctly, the reference graph set up here is
However after a GC run, all non-rooted chains will go away, resulting in
i.e. the handler will never be executed. But this must be wrong, because otherwise we have created a way of observing GC behavior in JavaScript, which is not otherwise possible. So I am missing something obvious. Please help :) |
Oh, I see, I missed https://github.com/domenic/cancelable-promise/pull/1/files#diff-6230eb7fbf1dd2186bf1be6042c27f55R389, which essentially forwards the call global.resolveX0 -> x0 -> (v => console.log(v))
x1 -> x0
x2 -> x0 |
This version accomodates overwritten .then's, but breaks a few Promises/A+ tests. It strategy for the scenario at promises-aplus/promises-spec#179 (comment) is: - Add a [[PromiseFollowing]] slot to all promises - Resolving x1 to x0 notices that x0 is a promise without any value for its [[PromiseFollowing]] internal slot, so: - Do the normal thing, i.e. `x0.then(resolveX1, rejectX1)` - Save x1@[[PromiseFollowing]] = x0 - Resolving x2 to x1 notices that x1 is a promise with a value for its [[PromiseFollowing]] internal slot, so: - Follow the chain of [[PromiseFollowing]] to get a "real" resolution value of x0 - Now do as before, i.e. `x0.then(resolveX2, rejectX2)` and x2@[[PromiseFollowing]] = x0. Thus the reference graph is ... fuck, it's just x0 -> x1, x2 so x1 stays in memory. This breaks some Promises/A+ tests which are coded in terms of a given thenable only having its `then` property accessed once.
The issue title is a bit misleading, the tail-recursive code used for promise chain setup should be the problem. I've done some test and the full comment with node test code is at: nodejs/node#6673 (comment). And here's some quote:
|
Fixes the memory described at promises-aplus/promises-spec#179
is this the right place to ask this?
tj/co#180 (comment)
The text was updated successfully, but these errors were encountered: