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

io.circe.jawn.JawnParser.scala Maven / Gradle / Ivy

There is a newer version: 0.15.0-M1
Show newest version
package io.circe.jawn

import cats.data.ValidatedNel
import io.circe.{ Decoder, Error, Json, Parser, ParsingFailure }
import java.io.File
import java.nio.ByteBuffer
import scala.util.{ Failure, Success, Try }

object JawnParser {

  /**
   * Returns a parser that fails on:
   * 
    *
  • JSON strings, object keys, or numbers that exceed a given length
  • *
  • encountering duplicate keys as per JSONlint
  • *
* * In some cases excessively long values (e.g. JSON numbers with millions of * digits) may support denial-of-service attacks. For example, the string * constructor for Java's `BigInteger` is quadratic with the length of the * input, and decoding a ten-million digit JSON number into a `BigInteger` may * take minutes. * * If `allowDuplicateKeys` is set to `true`, the parser will fail if it encounters an object * containing duplicate keys. Note that duplicate keys are not prohibited by the JSON * specification, but many linters and other processors do not handle them. */ def apply(maxValueSize: Int, allowDuplicateKeys: Boolean): JawnParser = new JawnParser(Some(maxValueSize), allowDuplicateKeys) /** * Returns a parser that fails on: *
    *
  • JSON strings, object keys, or numbers that exceed a given length
  • *
  • encountering duplicate keys as per JSONlint
  • *
* * In some cases excessively long values (e.g. JSON numbers with millions of * digits) may support denial-of-service attacks. For example, the string * constructor for Java's `BigInteger` is quadratic with the length of the * input, and decoding a ten-million digit JSON number into a `BigInteger` may * take minutes. */ def apply(maxValueSize: Int): JawnParser = JawnParser(maxValueSize, true) /** * If `allowDuplicateKeys` is set to `true`, the parser will fail if it encounters an object * containing duplicate keys. Note that duplicate keys are not prohibited by the JSON * specification, but many linters and other processors do not handle them. */ def apply(allowDuplicateKeys: Boolean): JawnParser = new JawnParser(None, allowDuplicateKeys) } class JawnParser(maxValueSize: Option[Int], allowDuplicateKeys: Boolean) extends Parser { def this() = this(None, true) private[this] final val supportParser: CirceSupportParser = maxValueSize match { case Some(_) => new CirceSupportParser(maxValueSize, allowDuplicateKeys) case None => new CirceSupportParser(None, allowDuplicateKeys) } private[this] final def fromTry(t: Try[Json]): Either[ParsingFailure, Json] = t match { case Success(json) => Right(json) case Failure(error) => Left(ParsingFailure(error.getMessage, error)) } final def parse(input: String): Either[ParsingFailure, Json] = fromTry(supportParser.parseFromString(input)) final def parseFile(file: File): Either[ParsingFailure, Json] = fromTry(supportParser.parseFromFile(file)) final def parseByteBuffer(buffer: ByteBuffer): Either[ParsingFailure, Json] = fromTry(supportParser.parseFromByteBuffer(buffer)) final def decodeByteBuffer[A: Decoder](buffer: ByteBuffer): Either[Error, A] = finishDecode[A](parseByteBuffer(buffer)) final def decodeByteBufferAccumulating[A: Decoder](buffer: ByteBuffer): ValidatedNel[Error, A] = finishDecodeAccumulating[A](parseByteBuffer(buffer)) final def decodeFile[A: Decoder](file: File): Either[Error, A] = finishDecode[A](parseFile(file)) final def decodeFileAccumulating[A: Decoder](file: File): ValidatedNel[Error, A] = finishDecodeAccumulating[A](parseFile(file)) }




© 2015 - 2025 Weber Informatics LLC | Privacy Policy