Cancellation
This example shows how long running effects can be made cancellable, either by triggering the same effect again before it has completed, or explicitly by, say, clicking a button. We illustrate this with a simple HTTP GET request to Open Meteo.
State
The state holds the most recently fetched result and the loading state, which indicates whether a request is currently running.
case class State(
temperature: Option[Double] = None,
loading: Boolean = false
)
The JSON API response is modeled using a case class with a circe decoder instance.
import io.circe.*
import io.circe.generic.semiauto.*
case class OpenMeteoData(temperature_2m: Double)
case class OpenMeteoApiResponse(current: OpenMeteoData)
object OpenMeteoData:
given Decoder[OpenMeteoData] = deriveDecoder
object OpenMeteoApiResponse:
given Decoder[OpenMeteoApiResponse] = deriveDecoder
Actions
enum Action:
case Cancel
case GetTemperature
case SetTemperature(temp: Option[Double])
case SetLoading(loading: Boolean)
Store
The interesting bit is the construction of the store.
By wrapping the data fetching effect with store.withCancellationKey
, we can cancel it using store.cancel
.
By wrapping it with store.withRunningState
, we can observe whether it is running using store.runningState
.
Finally, we keep the loading
state in sync by subscribing to changes of store.runningState
.
import cats.effect.*
import cats.effect.implicits.*
import cats.syntax.all.*
import scala.concurrent.duration.*
object Store:
private val cancelKey = "get-temperature"
private val loadingKey = "loading"
def apply[F[_]](using
F: Async[F]
): Resource[F, ff4s.Store[F, State, Action]] =
for
store <- ff4s.Store[F, State, Action](State()): store =>
case (Action.SetTemperature(temp), state) =>
state.copy(temperature = temp) -> F.unit
case (Action.SetLoading(loading), state) =>
state.copy(loading = loading) -> F.unit
case (Action.Cancel, state) => state -> store.cancel(cancelKey)
case (Action.GetTemperature, state) =>
(
state.copy(temperature = none),
store
.withCancellationKey(cancelKey)(
store.withRunningState(loadingKey)(
F.sleep(
1.second // pretend that this is really long running
) *>
ff4s
.HttpClient[F]
.get[OpenMeteoApiResponse]("https://api.open-meteo.com/v1/forecast?latitude=47.3667&longitude=8.55¤t=temperature_2m")
.flatMap(r =>
store.dispatch(Action.SetTemperature(r.current.temperature_2m.some))
)
)
)
)
_ <- store
.runningState(loadingKey)
.discrete
.evalMap(loading => store.dispatch(Action.SetLoading(loading)))
.compile
.drain
.background
yield store
View
trait View:
self: ff4s.Dsl[State, Action] =>
val view =
import html._
useState: state =>
div(
button(
"Get current temperature",
onClick := (_ => Action.GetTemperature.some)
),
button("Cancel", onClick := (_ => Action.Cancel.some)),
if (state.loading) div("loading...")
else div(s"${state.temperature.fold("")(t => s"${t}°C")}")
)