io.circe.Printer.scala Maven / Gradle / Ivy
package io.circe
import java.io.Serializable
import java.lang.StringBuilder
import java.nio.{ ByteBuffer, CharBuffer }
import java.nio.charset.{ Charset, StandardCharsets }
import java.util.concurrent.CopyOnWriteArrayList
import scala.annotation.switch
/**
* A pretty-printer for JSON values.
*
* @author Travis Brown
* @author Tony Morris
*
* @param indent The indentation to use if any format strings contain a new line.
* @param lbraceLeft Spaces to insert to left of a left brace.
* @param lbraceRight Spaces to insert to right of a left brace.
* @param rbraceLeft Spaces to insert to left of a right brace.
* @param rbraceRight Spaces to insert to right of a right brace.
* @param lbracketLeft Spaces to insert to left of a left bracket.
* @param lbracketRight Spaces to insert to right of a left bracket.
* @param rbracketLeft Spaces to insert to left of a right bracket.
* @param rbracketRight Spaces to insert to right of a right bracket.
* @param lrbracketsEmpty Spaces to insert for an empty array.
* @param arrayCommaLeft Spaces to insert to left of a comma in an array.
* @param arrayCommaRight Spaces to insert to right of a comma in an array.
* @param objectCommaLeft Spaces to insert to left of a comma in an object.
* @param objectCommaRight Spaces to insert to right of a comma in an object.
* @param colonLeft Spaces to insert to left of a colon.
* @param colonRight Spaces to insert to right of a colon.
* @param dropNullValues Determines if object fields with values of null are dropped from the output.
* @param reuseWriters Determines whether the printer will reuse Appendables via thread-local
* storage.
* @param predictSize Uses an adaptive size predictor to avoid grow-and-copy steps while printing
* into a binary output.
* @param escapeNonAscii Unicode-escape any non-ASCII characters in strings.
* @param sortKeys Determines whether the fields should be sorted.
*/
final case class Printer(
dropNullValues: Boolean,
indent: String,
lbraceLeft: String = "",
lbraceRight: String = "",
rbraceLeft: String = "",
rbraceRight: String = "",
lbracketLeft: String = "",
lbracketRight: String = "",
rbracketLeft: String = "",
rbracketRight: String = "",
lrbracketsEmpty: String = "",
arrayCommaLeft: String = "",
arrayCommaRight: String = "",
objectCommaLeft: String = "",
objectCommaRight: String = "",
colonLeft: String = "",
colonRight: String = "",
reuseWriters: Boolean = false,
predictSize: Boolean = false,
escapeNonAscii: Boolean = false,
sortKeys: Boolean = false
) {
private[this] final val openBraceText = "{"
private[this] final val closeBraceText = "}"
private[this] final val openArrayText = "["
private[this] final val closeArrayText = "]"
private[this] final val commaText = ","
private[this] final val colonText = ":"
private[this] final class StringBuilderFolder(
writer: StringBuilder
) extends Printer.PrintingFolder(writer, pieces, dropNullValues, escapeNonAscii, sortKeys) {
final def onBoolean(value: Boolean): Unit = writer.append(value)
final def onNumber(value: JsonNumber): Unit = value.appendToStringBuilder(writer)
}
private[this] final class AppendableByteBufferFolder(
writer: Printer.AppendableByteBuffer
) extends Printer.PrintingFolder(writer, pieces, dropNullValues, escapeNonAscii, sortKeys) {
final def onBoolean(value: Boolean): Unit = writer.append(java.lang.Boolean.toString(value))
final def onNumber(value: JsonNumber): Unit = writer.append(value.toString)
}
private[this] final def concat(left: String, text: String, right: String): String = {
val builder = new StringBuilder()
builder.append(left)
builder.append(text)
builder.append(right)
builder.toString
}
private[this] final val pieces: Printer.PiecesAtDepth =
if (indent.isEmpty)
new Printer.ConstantPieces(
Printer.Pieces(
concat(lbraceLeft, openBraceText, lbraceRight),
concat(rbraceRight, closeBraceText, rbraceLeft),
concat(lbracketLeft, openArrayText, lbracketRight),
concat(rbracketLeft, closeArrayText, rbracketRight),
concat(openArrayText, lrbracketsEmpty, closeArrayText),
concat(arrayCommaLeft, commaText, arrayCommaRight),
concat(objectCommaLeft, commaText, objectCommaRight),
concat(colonLeft, colonText, colonRight)
)
)
else
new Printer.MemoizedPieces(indent) {
final def compute(i: Int): Printer.Pieces = {
val builder = new StringBuilder()
addIndentation(builder, lbraceLeft, i)
builder.append(openBraceText)
addIndentation(builder, lbraceRight, i + 1)
val lBraces = builder.toString
builder.setLength(0)
addIndentation(builder, rbraceLeft, i)
builder.append(closeBraceText)
addIndentation(builder, rbraceRight, i + 1)
val rBraces = builder.toString
builder.setLength(0)
addIndentation(builder, lbracketLeft, i)
builder.append(openArrayText)
addIndentation(builder, lbracketRight, i + 1)
val lBrackets = builder.toString
builder.setLength(0)
addIndentation(builder, rbracketLeft, i)
builder.append(closeArrayText)
addIndentation(builder, rbracketRight, i + 1)
val rBrackets = builder.toString
builder.setLength(0)
builder.append(openArrayText)
addIndentation(builder, lrbracketsEmpty, i)
builder.append(closeArrayText)
val lrEmptyBrackets = builder.toString
builder.setLength(0)
addIndentation(builder, arrayCommaLeft, i + 1)
builder.append(commaText)
addIndentation(builder, arrayCommaRight, i + 1)
val arrayCommas = builder.toString
builder.setLength(0)
addIndentation(builder, objectCommaLeft, i + 1)
builder.append(commaText)
addIndentation(builder, objectCommaRight, i + 1)
val objectCommas = builder.toString
builder.setLength(0)
addIndentation(builder, colonLeft, i + 1)
builder.append(colonText)
addIndentation(builder, colonRight, i + 1)
val colons = builder.toString
Printer.Pieces(lBraces, rBraces, lBrackets, rBrackets, lrEmptyBrackets, arrayCommas, objectCommas, colons)
}
}
@transient
private[this] final val stringWriter: ThreadLocal[StringBuilder] = new ThreadLocal[StringBuilder] {
override final def initialValue: StringBuilder = new StringBuilder()
}
/**
* Returns a string representation of a pretty-printed JSON value.
*/
final def pretty(json: Json): String = {
val writer = if (reuseWriters && stringWriter.ne(null)) {
val w = stringWriter.get()
w.setLength(0)
w
} else new StringBuilder()
val folder = new StringBuilderFolder(writer)
json.foldWith(folder)
writer.toString
}
@transient
private[this] final val sizePredictor: ThreadLocal[Printer.SizePredictor] =
new ThreadLocal[Printer.SizePredictor] {
override final def initialValue: Printer.SizePredictor =
new Printer.AdaptiveSizePredictor()
}
final def prettyByteBuffer(json: Json, cs: Charset): ByteBuffer = {
val predictor =
if (predictSize && sizePredictor.ne(null)) sizePredictor.get()
else Printer.NoSizePredictor
val writer = new Printer.AppendableByteBuffer(cs, predictor)
val folder = new AppendableByteBufferFolder(writer)
json.foldWith(folder)
writer.toByteBuffer
}
final def prettyByteBuffer(json: Json): ByteBuffer =
prettyByteBuffer(json, StandardCharsets.UTF_8)
/**
* The same pretty-printer configuration that outputs fields in sorted order.
*/
final def withSortedKeys: Printer = copy(sortKeys = true)
}
object Printer {
/**
* A pretty-printer configuration that inserts no spaces.
*/
final val noSpaces: Printer = Printer(
dropNullValues = false,
indent = ""
)
final val noSpacesSortKeys: Printer = Printer(
dropNullValues = false,
indent = "",
sortKeys = true
)
/**
* A pretty-printer configuration that indents by the given spaces.
*/
final def indented(indent: String, sortKeys: Boolean = false): Printer = Printer(
dropNullValues = false,
indent = indent,
lbraceRight = "\n",
rbraceLeft = "\n",
lbracketRight = "\n",
rbracketLeft = "\n",
lrbracketsEmpty = "\n",
arrayCommaRight = "\n",
objectCommaRight = "\n",
colonLeft = " ",
colonRight = " ",
sortKeys = sortKeys
)
/**
* A pretty-printer configuration that indents by two spaces.
*/
final val spaces2: Printer = indented(" ")
/**
* A pretty-printer configuration that indents by two spaces and outputs fields in sorted order.
*/
final val spaces2SortKeys: Printer = indented(" ", true)
/**
* A pretty-printer configuration that indents by four spaces.
*/
final val spaces4: Printer = indented(" ")
/**
* A pretty-printer configuration that indents by four spaces and outputs fields in sorted order.
*/
final val spaces4SortKeys: Printer = indented(" ", true)
private[this] final def writeEscapedChar(writer: Appendable, c: Char): Unit =
writer
.append('u')
.append(toHex((c >> 12) & 15))
.append(toHex((c >> 8) & 15))
.append(toHex((c >> 4) & 15))
.append(toHex(c & 15))
private[this] final def toHex(nibble: Int): Char = (nibble + (if (nibble >= 10) 87 else 48)).toChar
private[circe] abstract class PrintingFolder(
private[circe] val writer: Appendable,
private[circe] val pieces: PiecesAtDepth,
private[circe] val dropNullValues: Boolean,
private[circe] val escapeNonAscii: Boolean,
private[circe] val sortKeys: Boolean
) extends Json.Folder[Unit] {
private[circe] var depth: Int = 0
final def onNull: Unit = writer.append("null")
final def onString(value: String): Unit = {
writer.append('"')
var i = 0
var offset = 0
while (i < value.length) {
val c = value.charAt(i)
val esc = (c: @switch) match {
case '"' => '"'
case '\\' => '\\'
case '\b' => 'b'
case '\f' => 'f'
case '\n' => 'n'
case '\r' => 'r'
case '\t' => 't'
case _ => (if ((escapeNonAscii && c.toInt > 127) || Character.isISOControl(c)) 1 else 0).toChar
}
if (esc != 0) {
writer.append(value, offset, i).append('\\')
if (esc != 1) writer.append(esc) else writeEscapedChar(writer, c)
offset = i + 1
}
i += 1
}
if (offset < i) writer.append(value, offset, i)
writer.append('"')
}
final def onArray(value: Vector[Json]): Unit = {
val orig = depth
val p = pieces(depth)
if (value.isEmpty) writer.append(p.lrEmptyBrackets)
else {
val iterator = value.iterator
writer.append(p.lBrackets)
depth += 1
iterator.next().foldWith(this)
depth = orig
while (iterator.hasNext) {
writer.append(p.arrayCommas)
depth += 1
iterator.next().foldWith(this)
depth = orig
}
writer.append(p.rBrackets)
}
}
final def onObject(value: JsonObject): Unit = value.appendToFolder(this)
}
private[circe] final case class Pieces(
lBraces: String,
rBraces: String,
lBrackets: String,
rBrackets: String,
lrEmptyBrackets: String,
arrayCommas: String,
objectCommas: String,
colons: String
) extends Serializable
private[this] final val maxMemoizationDepth = 128
private[circe] abstract class PiecesAtDepth extends Serializable {
def apply(i: Int): Pieces
}
private[circe] final class ConstantPieces(pieces: Pieces) extends PiecesAtDepth {
def apply(i: Int): Pieces = pieces
}
private[circe] abstract class MemoizedPieces(indent: String) extends PiecesAtDepth {
def compute(i: Int): Pieces
private[this] final val known = new CopyOnWriteArrayList[Pieces](
new Array[Pieces](maxMemoizationDepth)
)
protected[this] final def addIndentation(builder: StringBuilder, s: String, depth: Int): Unit = {
val lastNewLineIndex = s.lastIndexOf('\n')
if (lastNewLineIndex == -1) builder.append(s)
else {
builder.append(s, 0, lastNewLineIndex + 1)
var i = 0
while (i < depth) {
builder.append(indent)
i += 1
}
builder.append(s, lastNewLineIndex + 1, s.length)
}
}
final def apply(i: Int): Pieces = if (i >= maxMemoizationDepth) compute(i)
else {
val res = known.get(i)
if (res.ne(null)) res
else {
val tmp = compute(i)
known.set(i, tmp)
tmp
}
}
}
// We use these sizes as an initial buffers size: 32 bytes to 32 megabytes.
private[this] val SizeTable: Array[Int] = Array(
32, 48, 64, 80, 96, 112, 128, 144, 160, 176, 192, 208, 224, 240, 256, 272, 288, 304, 320, 336, 352, 368, 384, 400,
416, 432, 448, 464, 480, 496, 512, 1024, 2048, 4096, 8192, 16384, 32768, 65536, 131072, 262144, 524288, 1048576,
2097152, 4194304, 8388608, 16777216, 33554432
)
private abstract class SizePredictor {
def recordSize(size: Int): Unit
def predictSize: Int
}
/**
* Predicts an output buffer size based on the history of allocations. Highly inspired by
* Netty's AdaptiveRecvByteBufAllocator.
*
* @note This class isn't thread-safe. Any access to it should be synchronized externally.
*/
private final class AdaptiveSizePredictor extends SizePredictor {
private[this] var index: Int = 0
private[this] var nextSize: Int = 32
private[this] var decreasing: Boolean = false
def recordSize(size: Int): Unit =
if (size <= SizeTable(math.max(0, index - 2))) {
if (decreasing) {
index = math.max(0, index - 1)
nextSize = SizeTable(index)
decreasing = false
} else {
decreasing = true
}
} else if (size >= nextSize) {
index = math.min(SizeTable.length - 1, index + 4)
nextSize = SizeTable(index)
decreasing = false
}
def predictSize: Int = nextSize
}
private object NoSizePredictor extends SizePredictor {
def recordSize(size: Int): Unit = ()
def predictSize: Int = 32
}
/**
* Very bare-bones and fast [[Appendable]] that can produce a [[ByteBuffer]].
*
* The implementation is pretty much a regular growing char buffer that assumes all given
* [[CharSequence]]s are just [[String]]s so both `toString` and `charAt` methods are cheap.
*/
private[circe] final class AppendableByteBuffer(
cs: Charset,
sizePredictor: SizePredictor
) extends Appendable {
private[this] var index = 0
private[this] var chars = new Array[Char](sizePredictor.predictSize)
private[this] def ensureToFit(n: Int): Unit = {
val required = index + n
if (required > chars.length) {
val copy = new Array[Char](math.max(required, chars.length * 2))
System.arraycopy(chars, 0, copy, 0, index)
chars = copy
}
}
def append(csq: CharSequence): Appendable = {
ensureToFit(csq.length)
csq.toString.getChars(0, csq.length, chars, index)
index += csq.length
this
}
def append(csq: CharSequence, start: Int, end: Int): Appendable = {
ensureToFit(end - start)
var j = start
while (j < end) {
chars(index) = csq.charAt(j)
j += 1
index += 1
}
this
}
def append(c: Char): Appendable = {
ensureToFit(1)
chars(index) = c
index += 1
this
}
def toByteBuffer: ByteBuffer = {
sizePredictor.recordSize(index)
cs.encode(CharBuffer.wrap(chars, 0, index))
}
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy