Skip to content
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

Result futures strongly retain supplying functions #361

Closed
stevenschlansker opened this issue Mar 2, 2023 · 0 comments · Fixed by #362
Closed

Result futures strongly retain supplying functions #361

stevenschlansker opened this issue Mar 2, 2023 · 0 comments · Fixed by #362

Comments

@stevenschlansker
Copy link
Contributor

stevenschlansker commented Mar 2, 2023

Hi, we use Failsafe to retry network operations that might fail. We had a minor production outage last night where a JVM ran out of heap. Upon inspection of the heap dump, we were surprised to realize that this pseudo-code retains much more memory than you expect:

var retryExec = Failsafe.with(RetryPolicy.builder()
                    .abortOn(InterruptedException.class)
                    .onFailedAttempt(e -> LOG.warn("Index action failed #{}", e.getExecutionCount(), e.getLastException()))
                    .withMaxAttempts(10)
                    .build())
                .with(outerExecutor);
var largeIntermediateState = load10KBOfData();
var key = "something";
var resultFuture = retryExec.getAsync(() -> callRemoteService(key, largeIntermediateState));
static var cacheMap = new ConcurrentHashMap<Key, CompletableFuture<Result>>();
cacheMap.put(key, resultFuture);

If the cacheMap is long lived, it seems that the original supplying function is retained indefinitely even once the result is available.

FailsafeFuture -> AsyncExecutionImpl -> RetryPolicyExecutor.apply$Lambda -> failsafe.Functions$Lambda -> OurSupplier

While the Future is executing, it makes perfect sense to retain the supplying function.
But, after all retries and policies are exhausted, is that still necessary?

It seems that this could be improved with a small change: in AsyncExecutionImpl.complete, set outerFn = null at the very end. In SyncExecutionImpl.executeSync, set outerFn = null right after applying it. Then, the supplying function could be reclaimed but the result or exception would still be available in the Future.

If this cannot be improved, it is probably good to make a documentation note.

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

Successfully merging a pull request may close this issue.

1 participant