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

camundala.bpmn.CamundaVariable.scala Maven / Gradle / Ivy

There is a newer version: 1.30.23
Show newest version
package camundala
package bpmn

import camundala.domain.*
import io.circe.*
import io.circe.Json.*

import java.time.LocalDateTime
import scala.annotation.tailrec
import java.time.LocalDate

sealed trait CamundaVariable:
  
  def value: Any

  def toJson: Json =
    import CamundaVariable.*
    this match
      case CNull =>
        Json.Null
      case CString(value, _) =>
        Json.fromString(value)
      case CInteger(value, _) =>
        Json.fromInt(value)
      case CLong(value, _) =>
        Json.fromLong(value)
      case CBoolean(value, _) =>
        Json.fromBoolean(value)
      case CDouble(value, _) =>
        Json.fromDoubleOrNull(value)
      case file: CFile =>
        file.asJson
      case CJson(value, _) =>
        parser.parse(value).getOrElse(Json.obj())
end CamundaVariable

object CamundaVariable:

  given InOutEncoder[CamundaVariable] =
    Encoder.instance {
      case v: CString => v.asJson
      case v: CInteger => v.asJson
      case v: CLong => v.asJson
      case v: CDouble => v.asJson
      case v: CBoolean => v.asJson
      case v: CFile => v.asJson
      case v: CJson => v.asJson.deepDropNullValues
      case CNull => Json.Null
    }

  given ApiSchema[CamundaVariable] =
    deriveApiSchema

  given ApiSchema[CString] = deriveApiSchema
  given InOutCodec[CString] = deriveInOutCodec

  given ApiSchema[CInteger] = deriveApiSchema
  given InOutCodec[CInteger] = deriveInOutCodec

  given ApiSchema[CLong] = deriveApiSchema
  given InOutCodec[CLong] = deriveInOutCodec

  given ApiSchema[CDouble] = deriveApiSchema
  given InOutCodec[CDouble] = deriveInOutCodec

  given ApiSchema[CBoolean] = deriveApiSchema
  given InOutCodec[CBoolean] = deriveInOutCodec

  given ApiSchema[CFile] = deriveApiSchema
  given InOutCodec[CFile] = deriveInOutCodec

  given ApiSchema[CFileValueInfo] = deriveApiSchema
  given InOutCodec[CFileValueInfo] = deriveInOutCodec

  given ApiSchema[CJson] = deriveApiSchema
  given InOutCodec[CJson] = deriveInOutCodec

  def toCamunda[T <: Product: InOutEncoder](
      products: Seq[T]
  ): Seq[Map[String, CamundaVariable]] =
    products.map(toCamunda)

  def toCamunda[T <: Product: InOutEncoder](
      product: T
  ): Map[String, CamundaVariable] =
    product.productElementNames
      .zip(product.productIterator)
      .map:
        case k -> v =>
          k -> objectToCamunda(product, k, v)
      .toMap

  @tailrec
  def objectToCamunda[T <: Product: InOutEncoder](
      product: T,
      key: String,
      value: Any
  ): CamundaVariable =
    value match
      case None => CNull
      case Some(v) => objectToCamunda(product, key, v)
      case f @ FileInOut(fileName, _, mimeType) =>
        CFile(
          f.contentAsBase64,
          CFileValueInfo(
            fileName,
            mimeType
          )
        )
      case v: (Product | Iterable[?] | Map[?, ?]) =>
        product.asJson.deepDropNullValues.hcursor
          .downField(key)
          .as[Json] match
          case Right(str) if str.isString => CString(str.asString.get) // Pure Enum!
          case Right(v) => CJson(v.toString)
          case Left(ex) => throwErr(s"$key of $v could NOT be Parsed to a JSON!\n$ex")

      case v =>
        valueToCamunda(v)

  def valueToCamunda(value: Any): CamundaVariable =
    value match
      case v: String =>
        CString(v)
      case v: Int =>
        CInteger(v)
      case v: Long =>
        CLong(v)
      case v: Boolean =>
        CBoolean(v)
      case v: Float =>
        CDouble(v.toDouble)
      case v: Double =>
        CDouble(v)
      case v: scala.reflect.Enum =>
        CString(v.toString)
      case ld: LocalDate =>
        CString(ld.toString)
      case ldt: LocalDateTime =>
        CString(ldt.toString)
      case other if other == null =>
        CNull
      case v: Json =>
        CJson(v.toString)
      case other =>
        throwErr(s"Unexpected Value to map to CamundaVariable: $other")

  case object CNull extends CamundaVariable:
    val value: Null = null

    private val `type`: String = "String"
  end CNull
  case class CString(value: String, private val `type`: String = "String")
      extends CamundaVariable
  case class CInteger(value: Int, private val `type`: String = "Integer")
      extends CamundaVariable
  case class CLong(value: Long, private val `type`: String = "Long")
      extends CamundaVariable
  case class CBoolean(value: Boolean, private val `type`: String = "Boolean")
      extends CamundaVariable

  case class CDouble(value: Double, private val `type`: String = "Double")
      extends CamundaVariable

  case class CFile(
      @description("The File's content as Base64 encoded String.")
      value: String,
      valueInfo: CFileValueInfo,
      private val `type`: String = "File"
  ) extends CamundaVariable

  case class CFileValueInfo(
      filename: String,
      mimetype: Option[String]
  )

  case class CJson(value: String, private val `type`: String = "Json")
      extends CamundaVariable

  given InOutDecoder[CamundaVariable] =
    (c: HCursor) =>
      for
        valueType <- c.downField("type").as[String]
        anyValue = c.downField("value")
        value <- decodeValue(valueType, anyValue, c.downField("valueInfo"))
      yield value

  def decodeValue(
      valueType: String,
      anyValue: ACursor,
      valueInfo: ACursor
  ): Either[DecodingFailure, CamundaVariable] =
    valueType match
      case "Null" => Right(CNull)
      case "Boolean" => anyValue.as[Boolean].map(CBoolean(_))
      case "Integer" => anyValue.as[Int].map(CInteger(_))
      case "Long" => anyValue.as[Long].map(CLong(_))
      case "Double" => anyValue.as[Double].map(CDouble(_))
      case "Json" => anyValue.as[String].map(CJson(_))
      case "File" =>
        valueInfo.as[CFileValueInfo].map(vi => CFile("not_set", vi))
      case _ => anyValue.as[String].map(CString(_))

  type JsonToCamundaValue = CamundaVariable | Map[String, CamundaVariable] |
    Seq[Any]

  def jsonToCamundaValue(json: Json): JsonToCamundaValue =

    val folder: Json.Folder[JsonToCamundaValue] =
      new Json.Folder[JsonToCamundaValue]:
        def onNull: CamundaVariable = CNull

        def onBoolean(value: Boolean): CamundaVariable = CBoolean(value)

        def onNumber(value: JsonNumber): CamundaVariable =
          value.toBigDecimal
            .map {
              case v if v.isValidInt => CInteger(v.intValue)
              case v if v.isValidLong => CLong(v.longValue)
              case v => CDouble(v.doubleValue)
            }
            .getOrElse(CDouble(value.toDouble))

        def onString(value: String): CamundaVariable = CString(value)

        def onArray(value: Vector[Json]): JsonToCamundaValue =
          value.collect {
            case v if !v.isNull => v.foldWith(this)
          }

        def onObject(value: JsonObject): JsonToCamundaValue =
          jsonObjectToProcessVariables(value)
    json.foldWith(folder)

  end jsonToCamundaValue

  def jsonObjectToProcessVariables(obj: JsonObject): Map[String, CamundaVariable] =
    obj
      .filter { case (_, v) => !v.isNull }
      .toMap
      .map((k, v) =>
        k -> (jsonToCamundaValue(v) match
          case cv: CamundaVariable => cv
          case _: (Map[?, ?] | Seq[?]) => CJson(v.toString)
        )
      )

end CamundaVariable




© 2015 - 2025 Weber Informatics LLC | Privacy Policy