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

com.cognite.sdk.scala.v1.fdm.instances.InstanceDefinition.scala Maven / Gradle / Ivy

package com.cognite.sdk.scala.v1.fdm.instances

import cats.implicits._
import com.cognite.sdk.scala.v1.fdm.common.DirectRelationReference
import com.cognite.sdk.scala.v1.fdm.common.properties.{
  ListablePropertyType,
  PrimitivePropType,
  PropertyType
}
import io.circe._
import io.circe.syntax.EncoderOps

import java.time.{LocalDate, ZonedDateTime}

sealed trait InstanceDefinition {
  def space: String
  def externalId: String
  def createdTime: Long
  def lastUpdatedTime: Long
  def deletedTime: Option[Long]
  def version: Option[Long]
  def properties: Option[Map[String, Map[String, Map[String, InstancePropertyValue]]]]

  val instanceType: InstanceType
}

object InstanceDefinition {
  final case class NodeDefinition(
      space: String,
      externalId: String,
      createdTime: Long,
      lastUpdatedTime: Long,
      deletedTime: Option[Long],
      version: Option[Long],
      properties: Option[Map[String, Map[String, Map[String, InstancePropertyValue]]]],
      `type`: Option[DirectRelationReference]
  ) extends InstanceDefinition {
    override val instanceType: InstanceType = InstanceType.Node
  }

  final case class EdgeDefinition(
      `type`: DirectRelationReference,
      space: String,
      externalId: String,
      createdTime: Long,
      lastUpdatedTime: Long,
      deletedTime: Option[Long],
      version: Option[Long],
      properties: Option[Map[String, Map[String, Map[String, InstancePropertyValue]]]],
      startNode: DirectRelationReference,
      endNode: DirectRelationReference
  ) extends InstanceDefinition {
    override val instanceType: InstanceType = InstanceType.Edge
  }

  implicit val nodeDefinitionEncoder: Encoder[NodeDefinition] = Encoder.forProduct8(
    "instanceType",
    "space",
    "externalId",
    "createdTime",
    "lastUpdatedTime",
    "deletedTime",
    "version",
    "properties"
  )((e: NodeDefinition) =>
    (
      e.instanceType,
      e.space,
      e.externalId,
      e.createdTime,
      e.lastUpdatedTime,
      e.deletedTime,
      e.version,
      e.properties
    )
  )

  implicit val edgeDefinitionEncoder: Encoder[EdgeDefinition] = Encoder.forProduct9(
    "instanceType",
    "type",
    "space",
    "externalId",
    "createdTime",
    "lastUpdatedTime",
    "deletedTime",
    "version",
    "properties"
  )((e: EdgeDefinition) =>
    (
      e.instanceType,
      e.`type`,
      e.space,
      e.externalId,
      e.createdTime,
      e.lastUpdatedTime,
      e.deletedTime,
      e.version,
      e.properties
    )
  )

  implicit val instanceDefinitionEncoder: Encoder[InstanceDefinition] =
    Encoder.instance[InstanceDefinition] {
      case e: NodeDefinition => e.asJson
      case e: EdgeDefinition => e.asJson
    }

  def instancePropertyDefinitionBasedInstanceDecoder(
      propertyTypeDefinitionsMap: Option[
        Map[String, Map[String, Map[String, TypePropertyDefinition]]]
      ]
  ): Decoder[InstanceDefinition] = (c: HCursor) =>
    propertyTypeDefinitionsMap match {
      case Some(propDefMap) =>
        c.downField("instanceType").as[InstanceType] match {
          case Left(err) => Left[DecodingFailure, InstanceDefinition](err)
          case Right(InstanceType.Node) =>
            instancePropertyDefinitionBasedNodeDefinitionDecoder(propDefMap).apply(c)
          case Right(InstanceType.Edge) =>
            instancePropertyDefinitionBasedEdgeDefinitionDecoder(propDefMap).apply(c)
        }
      case None =>
        Left[DecodingFailure, InstanceDefinition](
          DecodingFailure(
            """Decoding without typing is not supported,
            |resend the request with includeTyping set to true""".stripMargin,
            c.history
          )
        )
    }

  def instancePropertyDefinitionBasedInstancePropertyTypeDecoder(
      types: Map[String, Map[String, Map[String, TypePropertyDefinition]]]
  ): Decoder[Option[Map[String, Map[String, Map[String, InstancePropertyValue]]]]] = (c: HCursor) =>
    {
      val result = c.value
        .as[Option[Map[String, Map[String, Map[String, Json]]]]]
        .flatMap {
          case Some(propertiesAsJson) =>
            val propertiesTyped = propertiesAsJson.toList
              .traverse { case (spaceName, spacePropsMap) =>
                val spacePropsMapTyped = spacePropsMap.toList
                  .traverse { case (viewOrContainerId, viewOrContainerPropsMap) =>
                    val viewOrContainerPropsMapTyped = viewOrContainerPropsMap.toList
                      .traverse { case (propId, instantPropTypeJson) =>
                        val typeDef = types
                          .get(spaceName)
                          .flatMap(_.get(viewOrContainerId).flatMap(_.get(propId)))
                        val typedInstancePropType = typeDef match {
                          case Some(t) => toInstancePropertyType(instantPropTypeJson, t)
                          case None =>
                            Left[DecodingFailure, InstancePropertyValue](
                              DecodingFailure(
                                s"Couldn't find InstancePropertyDefinition for property $spaceName.$viewOrContainerId.$propId",
                                c.history
                              )
                            )
                        }
                        typedInstancePropType.map(ipt => propId -> ipt)
                      }
                      .map(m => viewOrContainerId -> m.toMap)
                    viewOrContainerPropsMapTyped
                  }
                  .map(m => spaceName -> m.toMap)
                spacePropsMapTyped
              }
              .map(m => Some(m.toMap))

            propertiesTyped

          case None =>
            Right[DecodingFailure, Option[
              Map[String, Map[String, Map[String, InstancePropertyValue]]]
            ]](None)
        }

      result
    }

  private def instancePropertyDefinitionBasedNodeDefinitionDecoder(
      instPropDefMap: Map[String, Map[String, Map[String, TypePropertyDefinition]]]
  ): Decoder[NodeDefinition] = (c: HCursor) =>
    for {
      space <- c.downField("space").as[String]
      externalId <- c.downField("externalId").as[String]
      createdTime <- c.downField("createdTime").as[Long]
      lastUpdatedTime <- c.downField("lastUpdatedTime").as[Long]
      deletedTime <- c.downField("deletedTime").as[Option[Long]]
      version <- c.downField("version").as[Option[Long]]
      properties <- c
        .downField("properties")
        .as[Option[Map[String, Map[String, Map[String, InstancePropertyValue]]]]](
          instancePropertyDefinitionBasedInstancePropertyTypeDecoder(instPropDefMap)
        )
      relation <- c.downField("type").as[Option[DirectRelationReference]]
    } yield NodeDefinition(
      space = space,
      externalId = externalId,
      createdTime = createdTime,
      lastUpdatedTime = lastUpdatedTime,
      deletedTime = deletedTime,
      version = version,
      properties = properties,
      `type` = relation
    )

  private def instancePropertyDefinitionBasedEdgeDefinitionDecoder(
      instPropDefMap: Map[String, Map[String, Map[String, TypePropertyDefinition]]]
  ): Decoder[EdgeDefinition] = (c: HCursor) =>
    for {
      relation <- c.downField("type").as[DirectRelationReference]
      space <- c.downField("space").as[String]
      externalId <- c.downField("externalId").as[String]
      createdTime <- c.downField("createdTime").as[Long]
      lastUpdatedTime <- c.downField("lastUpdatedTime").as[Long]
      deletedTime <- c.downField("deletedTime").as[Option[Long]]
      version <- c.downField("version").as[Option[Long]]
      properties <- c
        .downField("properties")
        .as[Option[Map[String, Map[String, Map[String, InstancePropertyValue]]]]](
          instancePropertyDefinitionBasedInstancePropertyTypeDecoder(instPropDefMap)
        )
      startNode <- c.downField("startNode").as[DirectRelationReference]
      endNode <- c.downField("endNode").as[DirectRelationReference]
    } yield EdgeDefinition(
      `type` = relation,
      space = space,
      externalId = externalId,
      createdTime = createdTime,
      lastUpdatedTime = lastUpdatedTime,
      deletedTime = deletedTime,
      version = version,
      properties = properties,
      startNode = startNode,
      endNode = endNode
    )

  private def toInstancePropertyType(
      propValue: Json,
      t: TypePropertyDefinition
  ): Either[DecodingFailure, InstancePropertyValue] =
    t.`type` match {
      case t: ListablePropertyType if t.isList => toInstancePropertyTypeOfList(propValue, t)
      case t => toInstancePropertyTypeOfNonList(propValue, t)
    }

  private def toInstancePropertyTypeOfList(
      propValue: Json,
      t: PropertyType
  ): Either[DecodingFailure, InstancePropertyValue] =
    t match {
      case PropertyType.TextProperty(Some(true), _) =>
        Decoder[Seq[String]]
          .decodeJson(propValue)
          .map(InstancePropertyValue.StringList.apply)
      case PropertyType.PrimitiveProperty(PrimitivePropType.Boolean, Some(true)) =>
        Decoder[Seq[Boolean]]
          .decodeJson(propValue)
          .map(InstancePropertyValue.BooleanList.apply)
      case PropertyType.PrimitiveProperty(PrimitivePropType.Int32, Some(true)) =>
        Decoder[Seq[Int]]
          .decodeJson(propValue)
          .map(InstancePropertyValue.Int32List.apply)
      case PropertyType.PrimitiveProperty(PrimitivePropType.Int64, Some(true)) =>
        Decoder[Seq[Long]]
          .decodeJson(propValue)
          .map(InstancePropertyValue.Int64List.apply)
      case PropertyType.PrimitiveProperty(PrimitivePropType.Float32, Some(true)) =>
        Decoder[Seq[Float]]
          .decodeJson(propValue)
          .map(InstancePropertyValue.Float32List.apply)
      case PropertyType.PrimitiveProperty(PrimitivePropType.Float64, Some(true)) =>
        Decoder[Seq[Double]]
          .decodeJson(propValue)
          .map(InstancePropertyValue.Float64List.apply)
      case PropertyType.PrimitiveProperty(PrimitivePropType.Date, Some(true)) =>
        Decoder[Seq[LocalDate]]
          .decodeJson(propValue)
          .map(InstancePropertyValue.DateList.apply)
      case PropertyType.PrimitiveProperty(PrimitivePropType.Timestamp, Some(true)) =>
        Decoder[Seq[ZonedDateTime]]
          .decodeJson(propValue)
          .map(InstancePropertyValue.TimestampList.apply)
      case PropertyType.PrimitiveProperty(PrimitivePropType.Json, Some(true)) =>
        Decoder[Seq[Json]]
          .decodeJson(propValue)
          .map(InstancePropertyValue.ObjectList.apply)
      case PropertyType.TimeSeriesReference(Some(true)) =>
        Decoder[Seq[String]]
          .decodeJson(propValue)
          .map(InstancePropertyValue.TimeSeriesReferenceList.apply)
      case PropertyType.FileReference(Some(true)) =>
        Decoder[Seq[String]]
          .decodeJson(propValue)
          .map(InstancePropertyValue.FileReferenceList.apply)
      case PropertyType.SequenceReference(Some(true)) =>
        Decoder[Seq[String]]
          .decodeJson(propValue)
          .map(InstancePropertyValue.SequenceReferenceList.apply)
      case PropertyType.DirectNodeRelationProperty(_, _, Some(true)) =>
        Decoder[Seq[DirectRelationReference]]
          .decodeJson(propValue)
          .map(InstancePropertyValue.ViewDirectNodeRelationList.apply)
      case _ =>
        Left[DecodingFailure, InstancePropertyValue](
          DecodingFailure(
            s"Expected a list type, but found a non list type: ${propValue.noSpaces} as ${t.toString}",
            List.empty
          )
        )
    }

  private def toInstancePropertyTypeOfNonList(
      propValue: Json,
      t: PropertyType
  ): Either[DecodingFailure, InstancePropertyValue] =
    t match {
      case PropertyType.EnumProperty(_, _) =>
        Decoder[String]
          .decodeJson(propValue)
          .map(InstancePropertyValue.Enum.apply)
      case PropertyType.TextProperty(None | Some(false), _) =>
        Decoder[String]
          .decodeJson(propValue)
          .map(InstancePropertyValue.String.apply)
      case PropertyType.PrimitiveProperty(PrimitivePropType.Boolean, _) =>
        Decoder[Boolean]
          .decodeJson(propValue)
          .map(InstancePropertyValue.Boolean.apply)
      case PropertyType.PrimitiveProperty(PrimitivePropType.Int32, None | Some(false)) =>
        Decoder[Int]
          .decodeJson(propValue)
          .map(InstancePropertyValue.Int32.apply)
      case PropertyType.PrimitiveProperty(PrimitivePropType.Int64, None | Some(false)) =>
        Decoder[Long]
          .decodeJson(propValue)
          .map(InstancePropertyValue.Int64.apply)
      case PropertyType.PrimitiveProperty(PrimitivePropType.Float32, None | Some(false)) =>
        Decoder[Float]
          .decodeJson(propValue)
          .map(InstancePropertyValue.Float32.apply)
      case PropertyType.PrimitiveProperty(PrimitivePropType.Float64, None | Some(false)) =>
        Decoder[Double]
          .decodeJson(propValue)
          .map(InstancePropertyValue.Float64.apply)
      case PropertyType.PrimitiveProperty(PrimitivePropType.Date, None | Some(false)) =>
        Decoder[LocalDate]
          .decodeJson(propValue)
          .map(InstancePropertyValue.Date.apply)
      case PropertyType.PrimitiveProperty(PrimitivePropType.Timestamp, None | Some(false)) =>
        Decoder[ZonedDateTime]
          .decodeJson(propValue)
          .map(InstancePropertyValue.Timestamp.apply)
      case PropertyType.PrimitiveProperty(PrimitivePropType.Json, None | Some(false)) =>
        Decoder[Json]
          .decodeJson(propValue)
          .map(InstancePropertyValue.Object.apply)
      case PropertyType.TimeSeriesReference(None | Some(false)) =>
        Decoder[String]
          .decodeJson(propValue)
          .map(InstancePropertyValue.TimeSeriesReference.apply)
      case PropertyType.FileReference(None | Some(false)) =>
        Decoder[String]
          .decodeJson(propValue)
          .map(InstancePropertyValue.FileReference.apply)
      case PropertyType.SequenceReference(None | Some(false)) =>
        Decoder[String]
          .decodeJson(propValue)
          .map(InstancePropertyValue.SequenceReference.apply)
      case PropertyType.DirectNodeRelationProperty(_, _, Some(false)) |
          PropertyType.DirectNodeRelationProperty(_, _, None) =>
        propValue
          .as[Option[DirectRelationReference]]
          .map(InstancePropertyValue.ViewDirectNodeRelation.apply)
      case _ =>
        Left[DecodingFailure, InstancePropertyValue](
          DecodingFailure(
            s"Expected a non list type, but found a list type: ${propValue.noSpaces} as ${t.toString}",
            List.empty
          )
        )
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy