@reatom/effects
v3.11.0
Published
Reatom for effects
Downloads
8,923
Readme
This package is inspired by Sagas and gives you advanced effect management solutions.
included in @reatom/framework
First of all you should know that some effects and async (reatom/async + reatom/hooks) logic uses AbortController under the hood and if some of the controller aborted all nested effects will aborted too! It is a powerful feature for managing async logic which allows you to easily write concurrent logic, like with redux-saga or rxjs, but with the simpler native API.
Before we start, you could find a lot of useful helpers to manage aborts in reatom/utils
The differences between Redux-Saga and Reatom.
- Sagas
take
is liketake
+await
. - Sagas
takeMaybe
- is liketake
WITHOUTawait
. - Sagas
takeEvery
- is likeanAtom.onChange
/anAction.onCall
. - Sagas
takeLatest
- is likeanAtom.onChange
/anAction.onCall
+concurrent
. - Sagas
takeLeading
- is likeanAtom.onChange
+reatomAsync().pipe(withAbort({ strategy: 'first-in-win' }))
. - Sagas
call
is a regular function call with a context +await
. - Sagas
fork
is a regular function call with a context WITHOUTawait
. - Sagas
spawn
isspawn
- Sagas
join
- is justawait
in Reatom. - Sagas
cancel
is likegetTopController(ctx.cause)?.abort()
. - Sagas
cancelled
- is likeonCtxAbort
.
API
concurrent
This is the basic, useful API for performing concurrent async logic. Wrap your function with the concurrent
decorator, and all scheduled tasks of the passed ctx
will throw the abort error when a new request appears.
Main use case for the concurrent API is onChange
handling. Just wrap your function to always get only fresh results, no matter how often the changes occur.
Here, when someAtom
changes for the first time, the hook will be called and start fetching. If someAtom
changes during the fetch execution, the ctx.schedule
of the previous (first) call will throw an AbortError
, and the new fetching will start.
import { concurrent } from '@reatom/effects'
someAtom.onChange(
concurrent(async (ctx, some) => {
const other = await ctx.schedule(() => api.getOther(some))
otherAtom(ctx, other)
}),
)
Another example is how easily you could implement the "debounce" pattern with additional logic. Here is a comparison of the classic "debounce" decorator from "lodash" or any other utility library with the concurrent API. Each of the three examples has the same behavior for the debounce and concurrent examples.
You can see that each new logic addition forces a lot of changes for code with the simple debounce decorator and takes a really small amount of changes for code with the concurrent decorator.
For that example we will use withConcurrency
decorator with an action.
Base debounce.
import { withConcurrency } from '@reatom/effects'
const onChangeDebounce = debounce(
action((ctx, event) => {
inputAtom(ctx, event.currentTarget.value)
}),
500,
)
const onChangeConcurrent = action(async (ctx, event) => {
await ctx.schedule(() => sleep(500))
inputAtom(ctx, event.currentTarget.value)
}).pipe(withConcurrency())
Debounce after some mappings
const _onChangeDebounce = debounce((ctx, value) => {
inputAtom(ctx, value)
}, 500)
const onChangeDebounce = (ctx, event) => {
_onChangeDebounce(ctx, event.currentTarget.value)
}
const onChangeConcurrent = action(async (ctx, event) => {
const { value } = event.currentTarget
await ctx.schedule(() => sleep(500))
inputAtom(ctx, value)
}).pipe(withConcurrency())
Debounce with a condition.
const _onChange = (ctx, value) => {
inputAtom(ctx, value)
}
const _onChangeDebounce = debounce(_onChange, 500)
const onChangeDebounce = (ctx, event) => {
const { value } = event.currentTarget
if (Math.random() > 0.5) _onChange(ctx, value)
else handleDebounceChange(ctx, value)
}
const onChangeConcurrent = action(async (ctx, event) => {
const { value } = event.currentTarget
if (Math.random() > 0.5) await ctx.schedule(() => sleep(500))
inputAtom(ctx, value)
}).pipe(withConcurrency())
And even more! You can change the default "last-in-win" concurrency strategy to "first-in-win" and move the delay in the end of operation to archive a throttle pattern logic.
import { withConcurrency } from '@reatom/effects'
const onChangeThrottle = throttle(
action((ctx, event) => {
inputAtom(ctx, event.currentTarget.value)
}),
500,
)
const onChangeConcurrent = action(async (ctx, event) => {
inputAtom(ctx, event.currentTarget.value)
await ctx.schedule(() => sleep(500))
}).pipe(withConcurrency('first-in-win'))
reaction
This is a base API to describe and perform side effects. It is a combination of action, atom, and "concurrent" functionality. It helps you create an action with a computed function, allowing you to use ctx.spy
within it. The action call creates an atom with the passed function, subscribes to it, and returns the created atom with an unsubscribe
method. Note that the passed computed function is already wrapped by the concurrent function and also tracks the subscription status with an abort context.
This API is a variation of "effect" / "useEffect" API of others libraries. For example, here is how it can be used with reatomComponent
from @reatom/npm-react.
import { atom, reaction } from '@reatom/framework'
import { reatomComponent } from '@reatom/npm-react'
const reatomFilters = () => {
const searchAtom = atom('', 'Filters.searchAtom')
const tagsAtom = atom(0, 'Filters.tagsAtom')
const pageAtom = atom(0, 'Filters.pageAtom')
const changeReaction = reaction(async (ctx, onChange) => {
const search = ctx.spy(searchAtom)
const tags = ctx.spy(tagsAtom)
const page = ctx.spy(pageAtom)
onChange({ search, tags, page })
}, 'Filters.changeReaction')
return { searchAtom, tagsAtom, pageAtom, changeReaction }
}
export const Filters = reatomComponent(({ ctx, onChange }) => {
const model = useMemo(
() => {
const model = reatomFilters()
// Initiate the reaction!
// It will be disposed automatically when the component unmounts,
// as it is a part of `reatomComponent` ctx logic.
model.changeReaction(ctx, onChange)
return model
},
[], // Doesn't work in StrictMode!
)
// This component will not rerender, but it will react to all related changes!
return (
<Layout>
<Search model={model.search} />
<Tags model={model.tags} />
<Page model={model.page} />
</Layout>
)
}, 'Filters')
take
This is the simplest and most powerful API that allows you to wait for an atom update, which is useful for describing certain procedures. It is a shortcut for subscribing to the atom and unsubscribing after the first update. take
respects the main Reatom abort context and will throw AbortError
when the abort occurs. This allows you to describe redux-saga-like procedural logic in synchronous code style with native async/await.
import { action } from '@reatom/core'
import { take } from '@reatom/effects'
export const validateBeforeSubmit = action(async (ctx) => {
let errors = validate(ctx.get(formDataAtom))
while (Object.keys(errors).length) {
formDataAtom.errorsAtom(ctx, errors)
// wait any field change
await take(ctx, formDataAtom)
// recheck validation
errors = validate(ctx.get(formDataAtom))
}
})
You can also await actions!
import { take } from '@reatom/effects'
import { onConnect } from '@reatom/hooks'
import { historyAtom } from '@reatom/npm-history'
import { confirmModalAtom } from '~/features/modal'
// some model logic, doesn't matter
export const formAtom = reatomForm(/* ... */)
onConnect(formAtom, (ctx) => {
// "history" docs: https://github.com/remix-run/history/blob/main/docs/blocking-transitions.md
const unblock = historyAtom.block(ctx, async ({ retry }) => {
if (!ctx.get(formAtom).isSubmitted && !ctx.get(confirmModalAtom).opened) {
confirmModalAtom.open(ctx, 'Are you sure you want to leave?')
const confirmed = await take(ctx, confirmModalAtom.close)
if (confirmed) {
unblock()
retry()
}
}
})
})
take checkpoints
But be aware that take
only starts listening when it's called:
import { take } from '@reatom/effects'
import { reatomAsync } from '@reatom/async'
import { confirmModalAtom } from '~/features/modal'
import { api } from '~/api'
const closeDocument = reatomAsync(async (ctx) => {
confirmModalAtom.open(ctx, 'Are you sure you want to leave?')
const presaveId = await api.presaveDocument()
// ❌ Bug:
// If person's connection is not fast enough or they are too fast
// they can close the modal before we get presaveId and start to listening to modal.close.
// So now we are stuck in loading state forever...
await take(ctx, confirmModalAtom.close)
await api.finalizeDocument(presaveId)
})
You can fix this bug by creating a checkpoint before starting any process:
const closeDocument = reatomAsync(async (ctx) => {
confirmModalAtom.open(ctx, 'Are you sure you want to leave?')
// ✅ Now we listen for changes before we start any process...
const modalConfirmedCheckpoint = take(ctx, confirmModalAtom.close)
const presaveId = await api.presaveDocument()
// ...and we will catch them for sure
await modalConfirmedCheckpoint
await api.finalizeDocument(presaveId)
})
take filter
You can pass the third argument to map the update to the required format.
const input = await take(ctx, onChange, (ctx, event) => event.target.value)
More than that, you can filter unneeded updates by returning the skip
mark from the first argument of your callback.
const input = await take(ctx, onChange, (ctx, event, skip) => {
const { value } = event.target
return value.length < 6 ? skip : value
})
The cool feature of this skip mark is that it helps TypeScript understand the correct type of the returned value, which is hard to achieve with the extra "filter" function. If you have a union type, you could receive the needed data with the correct type easily. It just works.
const someRequest = reatomRequest<{ data: Data } | { error: string }>()
// type-safe destructuring
const { data } = await take(ctx, someRequest, (ctx, payload, skip) => ('error' in payload ? skip : payload))
Note that you can increase your debug experience by passing debug name in the four parameter.
const a = await take(ctx, someAtom, (ctx, data) => data.a, 'a')
const b = await take(ctx, bAtom, undefined, 'b')
takeNested
Allow you to wait all dependent effects, event if they was called in the nested async effect or by spawn.
For example, we have a routing logic for SSR.
// ~/features/some.ts
import { historyAtom } from '@reatom/npm-history'
historyAtom.locationAtom.onChange((ctx, location) => {
if (location.pathname === '/some') {
fetchSomeData(ctx, location.search)
}
})
How to track fetchSomeData
call? We could use takeNested
for this.
// SSR prerender
await takeNested(ctx, (trackedCtx) => {
historyAtom.push(trackedCtx, req.url)
})
render()
You could pass an arguments in the rest params of takeNested
function to pass it to the effect.
await takeNested(ctx, historyAtom.push, req.url)
render()
onCtxAbort
Handle an abort signal from the cause stack. For example, if you want to separate a task from the body of the concurrent handler, you can do it without explicit abort management; all tasks are carried out on top of ctx
.
import { action } from '@reatom/core'
import { reatomAsync, withAbort } from '@reatom/async'
import { onCtxAbort } from '@reatom/effects'
const doLongImportantAsyncWork = action((ctx) =>
ctx.schedule(() => {
const timeoutId = setTimeout(() => {
/* ... */
})
onCtxAbort(ctx, () => clearTimeout(timeoutId))
}),
)
export const handleImportantWork = reatomAsync((ctx) => {
/* ... */
doLongImportantAsyncWork(ctx)
/* ... */
}).pipe(withAbort())
getTopController
This is a simple util to find an abort controller on top of your cause stack. For example, it is useful to stop some async operation inside a regular actions, which are probably called from a concurrent context.
import { action } from '@reatom/core'
import { getTopController } from '@reatom/effects'
import { throwAbort } from '@reatom/utils'
import { onConnect } from '@reatom/hooks'
const doSome = action(async (ctx) => {
const data = await ctx.schedule(() => fetchData())
if (!data) throwAbort('nullable data', getTopController(ctx.cause))
// ... perform data
}, 'doSome')
spawn
This utility allow you to start a function with context which will NOT follow an abort of the cause.
For example, you want to start a fetch when Atom gets a connection, but don't want to abort the fetch when the connection is lost. This is because you want to persist the results.
import { spawn } from '@reatom/effects'
import { onConnect } from '@reatom/hooks'
onConnect(someAtom, (ctx) => {
spawn(ctx, async (spawnCtx) => {
const some = await api.getSome(spawnCtx)
someAtom(spawnCtx, some)
})
})