ECMAScript Proposal, specs, and reference implementation for Promise.prototype.finally
Spec drafted by @ljharb, following the lead from the cancelable promise proposal.
This proposal is currently stage 0 of the process.
Many promise libraries have a "finally" method, for registering a callback to be invoked when a promise is settled (either fulfilled, or rejected). The essential use case here is cleanup - I want to hide the "loading" spinner on my AJAX request, or I want to close any file handles I’ve opened, or I want to log that an operation has completed regardless of whether it succeeded or not.
promise.finally(func)
is similar to promise.then(func, func)
, but is different in a few critical ways:
- When creating a function inline, you can pass it once, instead of being forced to either declare it twice, or create a variable for it
- A
finally
callback will not receive any argument, since there's no reliably means of determining if the promise was fulfilled or rejected. This use case is for precisely when you do not care about the rejection reason, or the fulfillment value, and so there's no need to provide it. - Unlike
Promise.resolve(2).then(() => {}, () => {})
(which will be resolved withundefined
),Promise.resolve(2).finally(() => {})
will be resolved with2
. - Similarly, unlike
Promise.reject(3).then(() => {}, () => {})
(which will be resolved withundefined
),Promise.reject(3).finally(() => {})
will be rejected with3
.
However, please note: just like a syntactic finally
, a non-undefined return
in the finally
callback will resolve the new promise to that value, and a throw
(or returning a rejected promise) in the finally
callback will reject the new promise with that rejection reason.
The reasons to stick with finally
are straightforward: just like catch
, finally
would be an analog to the similarly-named syntactic forms from try
/catch
/finally
(try
, of course, not really having an analog any closer than Promise.resolve().then
). Just like syntactic finally
, Promise#finally
would not be able to modify the return value, except by creating an abrupt completion by throwing an exception (ie, rejecting the promise).
I’d briefly considered always
as an alternative, since that wouldn’t imply ordering, but I think the parallels to the syntactic variation are compelling.
You can view the spec in markdown format or rendered as HTML.