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

kinesis4cats.instances.ciris.scala Maven / Gradle / Ivy

There is a newer version: 0.0.32
Show newest version
/*
 * Copyright 2023-2023 etspaceman
 *
 * 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 kinesis4cats.instances

import scala.concurrent.duration.{Duration, FiniteDuration}
import scala.jdk.CollectionConverters._
import scala.util.Try

import java.time.Instant

import _root_.ciris.{ConfigDecoder, ConfigError}
import cats.syntax.all._

import kinesis4cats.compat.DurationConverters._
import kinesis4cats.models.{AwsRegion, ConsumerArn}
import kinesis4cats.syntax.string._

/** Common [[https://cir.is/docs/configurations#decoders ConfigDecoder]]
  * instances
  */
object ciris {
  implicit val awsRegionConfigDecoder: ConfigDecoder[String, AwsRegion] =
    ConfigDecoder[String, String].mapEither { case (_, x) =>
      Either.fromOption(
        AwsRegion.values.find(r => x === r.name),
        ConfigError(s"$x is not a known region")
      )
    }

  implicit val consumerArnConfigDecoder: ConfigDecoder[String, ConsumerArn] =
    ConfigDecoder[String].mapEither { case (_, value) =>
      ConsumerArn.fromArn(value).leftMap(e => ConfigError(e))
    }

  implicit val durationConfigDecoder: ConfigDecoder[String, Duration] =
    ConfigDecoder[String].mapEither { case (_, value) =>
      Try(Duration(value)).toEither.leftMap(e =>
        ConfigError(s"Could not decode duration $value: ${e}")
      )
    }
  implicit val finiteDurationDecoder: ConfigDecoder[String, FiniteDuration] =
    durationConfigDecoder.map(x => FiniteDuration(x.length, x.unit))

  implicit val javaDurationDecoder: ConfigDecoder[String, java.time.Duration] =
    finiteDurationDecoder.map(_.toJava)

  implicit def seqDecoder[A](implicit
      CDA: ConfigDecoder[String, A]
  ): ConfigDecoder[String, Seq[A]] = ConfigDecoder[String].mapEither {
    case (configKey, value) =>
      value.asList.traverse(x => CDA.decode(configKey, x))
  }

  implicit def listDecoder[A](implicit
      CDA: ConfigDecoder[String, A]
  ): ConfigDecoder[String, List[A]] = seqDecoder[A].map(_.toList)

  implicit def setDecoder[A](implicit
      CDA: ConfigDecoder[String, A]
  ): ConfigDecoder[String, Set[A]] = seqDecoder[A].map(_.toSet)

  implicit def javaSetDecoder[A](implicit
      CDA: ConfigDecoder[String, A]
  ): ConfigDecoder[String, java.util.Set[A]] = setDecoder[A].map(_.asJava)

  implicit def javaHashSetDecoder[A](implicit
      CDA: ConfigDecoder[String, A]
  ): ConfigDecoder[String, java.util.HashSet[A]] = setDecoder[A].map { x =>
    val hs = new java.util.HashSet[A]
    hs.addAll(x.asJava)
    hs
  }

  implicit def mapConfigDecoder[A, B](implicit
      CDA: ConfigDecoder[String, A],
      CDB: ConfigDecoder[String, B]
  ): ConfigDecoder[String, Map[A, B]] = ConfigDecoder[String].mapEither {
    case (configKey, x) =>
      x.asMap
        .leftMap(ConfigError(_))
        .flatMap(
          _.toList
            .traverse { case (keyStr, valueStr) =>
              for {
                key <- CDA.decode(configKey, keyStr)
                value <- CDB.decode(configKey, valueStr)
              } yield key -> value
            }
            .map(_.toMap)
        )
  }

  implicit def javaMapConfigDecoder[A, B](implicit
      CDA: ConfigDecoder[String, A],
      CDB: ConfigDecoder[String, B]
  ): ConfigDecoder[String, java.util.Map[A, B]] =
    mapConfigDecoder[A, B].map(_.asJava)

  implicit val instantConfigDecoder: ConfigDecoder[String, Instant] =
    ConfigDecoder[String].mapEither { case (_, value) =>
      Try(Instant.parse(value)).toEither.leftMap(e =>
        ConfigError(s"Unable to parse timestamp value $value: ${e.getMessage}")
      )
    }

  implicit val javaUtilDateConfigDecoder
      : ConfigDecoder[String, java.util.Date] =
    ConfigDecoder[String].mapEither { case (configKey, value) =>
      instantConfigDecoder
        .decode(configKey, value)
        .map(java.util.Date.from)
    }

  implicit val javaIntegerConfigDecoder
      : ConfigDecoder[String, java.lang.Integer] =
    ConfigDecoder[String, Int].map(java.lang.Integer.valueOf)

  implicit val javaLongConfigDecoder: ConfigDecoder[String, java.lang.Long] =
    ConfigDecoder[String, Long].map(java.lang.Long.valueOf)
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy