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

camundala.worker.EngineContext.scala Maven / Gradle / Ivy

package camundala
package worker

import camundala.domain.*
import camundala.bpmn.*
import camundala.worker.CamundalaWorkerError.ServiceError

import java.time.{LocalDate, LocalDateTime}
import scala.reflect.ClassTag

trait EngineContext:
  def getLogger(clazz: Class[?]): WorkerLogger
  def toEngineObject: Json => Any

  def sendRequest[ServiceIn: InOutEncoder, ServiceOut: InOutDecoder: ClassTag](
      request: RunnableRequest[ServiceIn]
  ): SendRequestType[ServiceOut]

  def jsonObjectToEngineObject(
      json: JsonObject
  ): Map[String, Any] =
    json.toMap
      .map { case (k, v) => k -> jsonToEngineValue(v) }

  def toEngineObject[T <: Product: InOutEncoder](
      product: T
  ): Map[String, Any] =
    product.productElementNames
      .zip(product.productIterator)
      // .filterNot { case _ -> v => v.isInstanceOf[None.type] } // don't send null
      .map { case (k, v) => k -> objectToEngineObject(product, k, v) }
      .toMap

  def toEngineObject(
      variables: Map[String, Json]
  ): Map[String, Any] =
    variables
      .map { case (k, v) => k -> jsonToEngineValue(v) }

  def valuesToEngineObject(
      variables: Map[String, Any]
  ): Map[String, Any] =
    variables
      .map { case (k, v) => k -> valueToEngineObject(v) }

  def objectToEngineObject[T <: Product: InOutEncoder](
      product: T,
      key: String,
      value: Any
  ): Any =
    value match
      case None | null => null
      case Some(v) => objectToEngineObject(product, key, v)
      case v: (Product | Iterable[?] | Map[?, ?]) =>
        product.asJson.hcursor
          .downField(key)
          .as[Json] match
          case Right(v) => jsonToEngineValue(v)
          case Left(ex) =>
            throwErr(s"$key of $v could NOT be Parsed to a JSON!\n$ex")

      case v =>
        valueToEngineObject(v)

  def valueToEngineObject(value: Any): Any =
    value match
      case v: scala.reflect.Enum =>
        v.toString
      case ld: LocalDate =>
        ld.toString
      case ldt: LocalDateTime =>
        ldt.toString
      case other if other == null =>
        null
      case v: Json =>
        jsonToEngineValue(v)
      case other =>
        other

  def domainObjToEngineObject[A <: Product: InOutCodec](variable: A): Any =
    toEngineObject(variable.asJson.deepDropNullValues)

  def jsonToEngineValue(json: Json): Any =
    json match
      case j if j.isNull => null
      case j if j.isNumber =>
        j.asNumber.get.toBigDecimal.get match
          case n if n.isValidInt => n.toInt
          case n if n.isValidLong => n.toLong
          case n => n.toDouble
      case j if j.isBoolean => j.asBoolean.get
      case j if j.isString => j.asString.get
      case j =>
        toEngineObject(j.deepDropNullValues)
  end jsonToEngineValue

end EngineContext

trait WorkerLogger:
  def debug(message: String): Unit
  def info(message: String): Unit
  def warn(message: String): Unit
  def error(err: CamundalaWorkerError): Unit
end WorkerLogger

final case class EngineRunContext(engineContext: EngineContext, generalVariables: GeneralVariables):

  def getLogger(clazz: Class[?]): WorkerLogger = engineContext.getLogger(clazz)

  def sendRequest[ServiceIn: InOutEncoder, ServiceOut: InOutDecoder: ClassTag](
      request: RunnableRequest[ServiceIn]
  ): SendRequestType[ServiceOut] =
    engineContext.sendRequest(request)

  def toEngineObject[T <: Product: InOutEncoder](
      product: T
  ): Map[String, Any] =
    engineContext.toEngineObject(product)

  def toEngineObject(
      variables: Map[String, Json]
  ): Map[String, Any] =
    engineContext.toEngineObject(variables)

  def jsonObjectToEngineObject(
      json: JsonObject
  ): Map[String, Any] =
    engineContext.jsonObjectToEngineObject(json)

end EngineRunContext




© 2015 - 2025 Weber Informatics LLC | Privacy Policy