io.circe.MapDecoder.scala Maven / Gradle / Ivy
package io.circe
import cats.data.{ NonEmptyList, Validated }
import io.circe.cursor.ObjectCursor
import scala.collection.Map
import scala.collection.mutable.Builder
private[circe] abstract class MapDecoder[K, V, M[K, V] <: Map[K, V]](
decodeK: KeyDecoder[K],
decodeV: Decoder[V]
) extends Decoder[M[K, V]] {
private[this] val alwaysDecodeK = decodeK match {
case decodeK: KeyDecoder.AlwaysKeyDecoder[K] => decodeK
case _ => null
}
protected def createBuilder(): Builder[(K, V), M[K, V]]
final def apply(c: HCursor): Decoder.Result[M[K, V]] = c.value match {
case Json.JObject(obj) => decodeJsonObject(c, obj)
case _ => MapDecoder.failureResult[M[K, V]](c)
}
private[this] final def createObjectCursor(c: HCursor, obj: JsonObject, key: String): HCursor =
new ObjectCursor(obj, key, c, false)(c, CursorOp.DownField(key))
private[this] final def handleResult(key: K, c: HCursor, builder: Builder[(K, V), M[K, V]]): DecodingFailure =
decodeV(c) match {
case Right(value) => builder += ((key, value)); null
case Left(error) => error
}
private[this] final def decodeJsonObject(c: HCursor, obj: JsonObject): Decoder.Result[M[K, V]] = {
val it = obj.keys.iterator
val builder = createBuilder()
var failed: DecodingFailure = null
while (failed.eq(null) && it.hasNext) {
val key = it.next
val atH = createObjectCursor(c, obj, key)
failed = if (alwaysDecodeK.ne(null)) {
handleResult(alwaysDecodeK.decodeSafe(key), atH, builder)
} else {
decodeK(key) match {
case None => MapDecoder.failure(atH)
case Some(k) => handleResult(k, atH, builder)
}
}
}
if (failed.eq(null)) Right(builder.result) else Left(failed)
}
final override def decodeAccumulating(c: HCursor): Decoder.AccumulatingResult[M[K, V]] = c.value match {
case Json.JObject(obj) =>
val it = obj.keys.iterator
val builder = createBuilder()
var failed = false
val failures = List.newBuilder[DecodingFailure]
while (it.hasNext) {
val key = it.next
val atH = createObjectCursor(c, obj, key)
if (alwaysDecodeK.ne(null)) {
decodeV.decodeAccumulating(atH) match {
case Validated.Valid(value) => if (!failed) builder += ((alwaysDecodeK.decodeSafe(key), value))
case Validated.Invalid(es) =>
failed = true
failures += es.head
failures ++= es.tail
}
} else {
decodeK(key) match {
case Some(k) =>
decodeV.decodeAccumulating(atH) match {
case Validated.Valid(value) => if (!failed) builder += ((k, value))
case Validated.Invalid(es) =>
failed = true
failures += es.head
failures ++= es.tail
}
case None =>
failed = true
failures += MapDecoder.failure(atH)
}
}
}
if (!failed) Validated.valid(builder.result)
else {
failures.result match {
case h :: t => Validated.invalid(NonEmptyList(h, t))
case Nil => Validated.valid(builder.result)
}
}
case _ => MapDecoder.failureAccumulatingResult[M[K, V]](c)
}
}
private[circe] object MapDecoder {
final def failure(c: HCursor): DecodingFailure = DecodingFailure("[K, V]Map[K, V]", c.history)
final def failureResult[A](c: HCursor): Decoder.Result[A] = Left[DecodingFailure, A](failure(c))
final def failureAccumulatingResult[A](c: HCursor): Decoder.AccumulatingResult[A] =
Validated.invalidNel[DecodingFailure, A](failure(c))
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy