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

wvlet.airframe.json.JSON.scala Maven / Gradle / Ivy

/*
 * 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 wvlet.airframe.json

import wvlet.log.LogSupport

/**
  */
object JSON extends LogSupport {

  /**
    * Parse JSON object or array
    */
  def parse(s: String): JSONValue = {
    parse(JSONSource.fromString(s))
  }

  /**
    * Parse JSON object or array
    */
  def parse(s: Array[Byte]): JSONValue = {
    parse(JSONSource.fromBytes(s))
  }

  /**
    * Parse JSON object or array
    */
  def parse(s: JSONSource): JSONValue = {
    val b = new JSONValueBuilder().singleContext(s, 0)
    JSONScanner.scan(s, b)
    val j = b.result
    j
  }

  /**
    * Parse any json values including null
    */
  def parseAny(s: String): JSONValue = {
    parseAny(JSONSource.fromString(s))
  }

  /**
    * Parse any json values including null
    */
  def parseAny(s: Array[Byte]): JSONValue = {
    parseAny(JSONSource.fromBytes(s))
  }

  /**
    * Parse any json values including null
    */
  def parseAny(s: JSONSource): JSONValue = {
    val b = new JSONValueBuilder().singleContext(s, 0)
    JSONScanner.scanAny(s, b)
  }

  /**
    * Format JSON value
    */
  def format(v: JSONValue): String = {
    def formatInternal(v: JSONValue, level: Int = 0): String = {
      val s = new StringBuilder()
      v match {
        case x: JSONObject =>
          if (x.v.isEmpty) {
            s.append("{}")
          } else {
            s.append("{\n")
            s.append {
              x.v
                .map { case (k, v: JSONValue) =>
                  val ss = new StringBuilder
                  ss.append("  " * (level + 1))
                  ss.append("\"")
                  ss.append(quoteJSONString(k))
                  ss.append("\": ")
                  ss.append(formatInternal(v, level + 1))
                  ss.result()
                }.mkString("", ",\n", "\n")
            }
            s.append("  " * level)
            s.append("}")
          }
        case x: JSONArray =>
          if (x.v.isEmpty) {
            s.append("[]")
          } else {
            s.append("[\n")
            s.append(
              x.v
                .map { x =>
                  ("  " * (level + 1)) + formatInternal(x, level + 1)
                }.mkString("", ",\n", "\n")
            )
            s.append("  " * level)
            s.append("]")
          }
        case x => s.append(x.toJSON)
      }
      s.result()
    }

    formatInternal(v, 0)
  }

  sealed trait JSONValue {
    override def toString = toJSON
    def toJSON: String
  }

  case object JSONNull extends JSONValue {
    override def toJSON: String = "null"
  }

  final case class JSONBoolean(val v: Boolean) extends JSONValue {
    override def toJSON: String = if (v) "true" else "false"
  }

  val JSONTrue  = JSONBoolean(true)
  val JSONFalse = JSONBoolean(false)

  sealed trait JSONNumber extends JSONValue
  final case class JSONDouble(v: Double) extends JSONNumber {
    override def toJSON: String = v.toString
  }
  final case class JSONLong(v: Long) extends JSONNumber {
    override def toJSON: String = v.toString
  }
  final case class JSONString(v: String) extends JSONValue {
    override def toString = v
    override def toJSON: String = {
      val s = new StringBuilder(v.length + 2)
      s.append("\"")
      s.append(quoteJSONString(v))
      s.append("\"")
      s.result()
    }
  }

  object JSONObject {
    val empty: JSONObject = JSONObject(Seq.empty)
  }

  final case class JSONObject(v: Seq[(String, JSONValue)]) extends JSONValue {
    def isEmpty: Boolean = v.isEmpty
    def size: Int        = v.size
    override def toJSON: String = {
      val s = new StringBuilder
      s.append("{")
      s.append {
        v.map { case (k, v: JSONValue) =>
          val ss = new StringBuilder
          ss.append("\"")
          ss.append(quoteJSONString(k))
          ss.append("\":")
          ss.append(v.toJSON)
          ss.result()
        }.mkString(",")
      }
      s.append("}")
      s.result()
    }
    def get(name: String): Option[JSONValue] = {
      v.collectFirst {
        case (key, value) if key == name =>
          value
      }
    }
  }
  final case class JSONArray(v: IndexedSeq[JSONValue]) extends JSONValue {
    def size: Int = v.length
    override def toJSON: String = {
      val s = new StringBuilder
      s.append("[")
      s.append(v.map(x => x.toJSON).mkString(","))
      s.append("]")
      s.result()
    }

    def apply(i: Int): JSONValue = {
      v.apply(i)
    }
  }

  /**
    * This function can be used to properly quote Strings for JSON output.
    */
  def quoteJSONString(s: String): String = {
    s.map {
      case '"'  => "\\\""
      case '\\' => "\\\\"
//      case '/'  => "\\/" We don't need to escape forward slashes
      case '\b' => "\\b"
      case '\f' => "\\f"
      case '\n' => "\\n"
      case '\r' => "\\r"
      case '\t' => "\\t"
      /* We'll unicode escape any control characters. These include:
       * 0x0 -> 0x1f  : ASCII Control (C0 Control Codes)
       * 0x7f         : ASCII DELETE
       * 0x80 -> 0x9f : C1 Control Codes
       *
       * Per RFC4627, section 2.5, we're not technically required to
       * encode the C1 codes, but we do to be safe.
       */
      case c if ((c >= '\u0000' && c <= '\u001f') || (c >= '\u007f' && c <= '\u009f')) => "\\u%04x".format(c.toInt)
      case c                                                                           => c
    }.mkString
  }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy