Replies: 7 comments 23 replies
-
This sounds good. Only thing I'd suggest is changing the name from |
Beta Was this translation helpful? Give feedback.
-
I like this idea of scoping mutations to run them in a queue. However, I agree with @hrastnik on the naming since relating the ID to a queue clearly gives some insight on what's going on. |
Beta Was this translation helpful? Give feedback.
-
I like the idea. A few questions: |
Beta Was this translation helpful? Give feedback.
-
@TkDodo Two things I would like to clarify.
|
Beta Was this translation helpful? Give feedback.
-
@TkDodo We would be quite interested in this enhancement. Do you already know if and when we can expect it to be implemented? |
Beta Was this translation helpful? Give feedback.
-
Can we use it now? @TkDodo |
Beta Was this translation helpful? Give feedback.
-
This version broke our app. When the app goes online after being offline all mutations run in parallel. This breaks our app since we require serial execution after being offline. I guess I can add the same scope to every mutation we have in our app, but this feature is still a breaking change in the library.
The feature itself is a very nice addition! An option to use scope in the mutate call is a great proposal IMO. |
Beta Was this translation helpful? Give feedback.
-
tl;dr: mutations will get a new, opt-in scoping feature. Putting mutations in the same scope will run them in serial rather than in parallel.
The Problem
All our mutations run in parallel, which is generally a good thing. We can't know what our mutations do or which relation they have to each other.
This is a good default behaviour, however, there are sometimes situations, especially when
offline
mutations are involved, where we'd want to fire mutations in serial. As an example, firing to/PATCH
requests to the same resource might result in aConcurrentModificationException
.Also, queuing some mutations that alter the same entity might make the in-between update unnecessary. For example, let's say we're moving a card on a kanban board from column A to B, followed by an immediate move from B to C. In that case, the first mutation is likely unnecessary and should be aborted and / or discarded.
The proposed solution
As a solution, we'd want to introduce
scoping
of mutations. Mutations with the same scope would be treated as being in the samequeue
. This is anopt-in
feature. When not given a dedicatedscope
, mutations get their own uniquescope
where no other mutation can be in. This will result in the mutation running immediately and in parallel with all other mutations.There will be no fuzzy matching of scopes - a scope has to fully match an existing one to be treated as "the same".
Technical implementation
We currently store all mutations as an
Array<Mutation>
in theMutationCache
:query/packages/query-core/src/mutationCache.ts
Line 85 in 4f168d7
The idea is to change this structure into:
where the key of the Map is the scope's id. This will create Mutation Arrays that live "next to each other". When a mutation wants to run, it has to check if its scope allows it. If there is already a running mutation in the scope, we will not run the mutation, but put it into
paused
state instead (status: 'pending', isPaused: true
). Note that paused was, up until now, only used to pause because there was no network connection. However, there is no limitation as to why we pause them - they can be paused for other reasons, too.When a mutation finishes (either successfully or erroneously), it has to kick of the next mutation in its scope. This makes sure that mutations that were previously paused will then run once their time has come.
API
Mutations will get a new, optional property:
If set, mutations in the same scope will be guaranteed to run in serial. The object notation enables to later allow for different scope behaviours. For example, a scope could be implemented to abort previously running mutations when a new mutation is added, and kick of the last mutation immediately. It could also get a
takeLatest
kind of behaviour where mutations still run serially, but in-between mutations get discarded. In the first iteration, scopes will only get the default behaviour of queuing up mutations and starting the next one in the scope when they are finished.Resuming Paused Mutations
While being offline, mutations will still get paused - unless the
networkMode
is set to ignore that. It is therefore not recommended to mix mutations with differentnetworkMode
settings in the same scope. As a an example, adding a mutation that should always run into a scope where we already have paused mutations (because of being offline) will result in this mutation also being queued up until being online again, because it depends on the previous mutations to finish before it can start.When resuming paused mutations after having been offline for some time, only the first mutation of each scope will be continued. The other ones need to wait until their predecessor has finished.
This should fix a couple of bugs around
resumePausedMutations
, especially:Note that we tried to fix #4896 but that fix actually introduced #6825. It also makes all mutations run in serial now, which isn't the same as when those mutations were executed while being online. This proposal tries to fix this holistically - mutations should behave the same no matter if they were added while being offline or not.
Beta Was this translation helpful? Give feedback.
All reactions