All Downloads are FREE. Search and download functionalities are using the official Maven repository.

cqrs.Backend.scala Maven / Gradle / Ivy

The newest version!
/*
 * Copyright 2021 Hossein Naderi
 *
 * Licensed under the Apache License, Version 2.0 (the "License");
 * you may not use this file except in compliance with the License.
 * You may obtain a copy of the License at
 *
 *     http://www.apache.org/licenses/LICENSE-2.0
 *
 * Unless required by applicable law or agreed to in writing, software
 * distributed under the License is distributed on an "AS IS" BASIS,
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 * See the License for the specific language governing permissions and
 * limitations under the License.
 */

package edomata.backend
package cqrs

import cats.effect.kernel.Async
import cats.effect.kernel.Resource
import cats.effect.std.Random
import edomata.core.*

import scala.concurrent.duration.*

trait Backend[F[_], S, R, N] {
  def compile: CommandHandler[F, S, N]
  def outbox: OutboxReader[F, N]
  def repository: RepositoryReader[F, S]
  def updates: NotificationsConsumer[F]
}

final class BackendBuilder[
    F[_]: Async,
    Codec[_],
    Handler[_],
    C,
    S,
    R,
    N
] private[cqrs] (
    driver: Resource[F, StorageDriver[F, Codec, Handler]],
    domain: CQRSDomain[C, S, R, N],
    commandCache: Option[Resource[F, CommandStore[F]]],
    notifHandler: Option[Handler[N]] = None,
    val maxRetry: Int = 5,
    val retryInitialDelay: FiniteDuration = 2.seconds,
    val stateCacheSize: Int = 1000
)(using StateModelTC[S]) {
  private def copy(
      driver: Resource[F, StorageDriver[F, Codec, Handler]] = driver,
      domain: CQRSDomain[C, S, R, N] = domain,
      commandCache: Option[Resource[F, CommandStore[F]]] = commandCache,
      notifHandler: Option[Handler[N]] = notifHandler,
      maxRetry: Int = maxRetry,
      retryInitialDelay: FiniteDuration = retryInitialDelay,
      stateCacheSize: Int = stateCacheSize
  ) = new BackendBuilder(
    driver = driver,
    domain = domain,
    maxRetry = maxRetry,
    commandCache = commandCache,
    notifHandler = notifHandler,
    retryInitialDelay = retryInitialDelay,
    stateCacheSize = stateCacheSize
  )

  def disableCache: BackendBuilder[F, Codec, Handler, C, S, R, N] =
    copy(commandCache = None)

  def withCommandCache(
      cache: Resource[F, CommandStore[F]]
  ): BackendBuilder[F, Codec, Handler, C, S, R, N] =
    copy(commandCache = Some(cache))

  def withCommandCache(
      cache: CommandStore[F]
  ): BackendBuilder[F, Codec, Handler, C, S, R, N] = withCommandCache(
    Resource.pure(cache)
  )

  def withCommandCache(
      cache: F[CommandStore[F]]
  ): BackendBuilder[F, Codec, Handler, C, S, R, N] = withCommandCache(
    Resource.eval(cache)
  )

  def withCommandCacheSize(
      maxCommandsToCache: Int
  ): BackendBuilder[F, Codec, Handler, C, S, R, N] = copy(commandCache =
    Some(Resource.eval(CommandStore.inMem(maxCommandsToCache)))
  )

  def withStateCacheSize(
      size: Int
  ): BackendBuilder[F, Codec, Handler, C, S, R, N] = copy(stateCacheSize = size)

  def withRetryConfig(
      maxRetry: Int = maxRetry,
      retryInitialDelay: FiniteDuration = retryInitialDelay
  ): BackendBuilder[F, Codec, Handler, C, S, R, N] =
    copy(maxRetry = maxRetry, retryInitialDelay = retryInitialDelay)

  def withEventHandler(h: Handler[N]) =
    copy(notifHandler = Some(h))

  def build(using
      state: Codec[S],
      notifs: Codec[N]
  ): Resource[F, Backend[F, S, R, N]] = for {
    d <- driver
    storage <- notifHandler.fold(d.build[S, N, R])(d.build[S, N, R](_))
    repo <- commandCache.fold(Resource.pure(storage.repository))(
      _.evalMap(CachedRepository(storage.repository, _, size = stateCacheSize))
    )
    given Random[F] <- Resource.eval(Random.scalaUtilRandom[F])
  } yield new Backend[F, S, R, N] {

    override def compile: CommandHandler[F, S, N] =
      CommandHandler.withRetry(repo, maxRetry, retryInitialDelay)

    override def outbox: OutboxReader[F, N] = storage.outbox

    override def repository: RepositoryReader[F, S] = repo

    override def updates: NotificationsConsumer[F] = storage.updates

  }
}

final class PartialBackendBuilder[C, S, R, N] private[backend] (
    domain: CQRSDomain[C, S, R, N]
)(using model: StateModelTC[S]) {
  def use[F[_]: Async, Codec[_], Handler[_]](
      driver: StorageDriver[F, Codec, Handler]
  ): BackendBuilder[F, Codec, Handler, C, S, R, N] =
    from(Resource.pure(driver))

  def use[F[_]: Async, Codec[_], Handler[_], D <: StorageDriver[
    F,
    Codec,
    Handler
  ]](
      driver: F[D]
  ): BackendBuilder[F, Codec, Handler, C, S, R, N] =
    from(Resource.eval(driver))

  def from[F[_]: Async, Codec[_], Handler[_], D <: StorageDriver[
    F,
    Codec,
    Handler
  ]](
      driver: Resource[F, D]
  ): BackendBuilder[F, Codec, Handler, C, S, R, N] =
    new BackendBuilder(
      driver,
      domain,
      commandCache = Some(Resource.eval(CommandStore.inMem(1000)))
    )
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy