ujson.AsyncParser.scala Maven / Gradle / Ivy
package ujson
import scala.annotation.{switch, tailrec}
import scala.math.max
import scala.collection.mutable
import scala.util.control
import java.nio.ByteBuffer
object AsyncParser {
sealed abstract class Mode(val start: Int, val value: Int)
case object UnwrapArray extends Mode(-5, 1)
case object ValueStream extends Mode(-1, 0)
case object SingleValue extends Mode(-1, -1)
def apply[J](mode: Mode = SingleValue): AsyncParser[J] =
new AsyncParser(state = mode.start, curr = 0, stack = Nil,
data = new Array[Byte](131072), len = 0, allocated = 131072,
offset = 0, done = false, streamMode = mode.value)
}
/**
* AsyncParser is able to parse chunks of data (encoded as
* Option[ByteBuffer] instances) and parse asynchronously. You can
* use the factory methods in the companion object to instantiate an
* async parser.
*
* The async parser's fields are described below:
*
* The (state, curr, stack) triple is used to save and restore parser
* state between async calls. State also helps encode extra
* information when streaming or unwrapping an array.
*
* The (data, len, allocated) triple is used to manage the underlying
* data the parser is keeping track of. As new data comes in, data may
* be expanded if not enough space is available.
*
* The offset parameter is used to drive the outer async parsing. It
* stores similar information to curr but is kept separate to avoid
* "corrupting" our snapshot.
*
* The done parameter is used internally to help figure out when the
* atEof() parser method should return true. This will be set when
* apply(None) is called.
*
* The streamMode parameter controls how the asynchronous parser will
* be handling multiple values. There are three states:
*
* 1: An array is being unwrapped. Normal JSON array rules apply
* (Note that if the outer value observed is not an array, this
* mode will toggle to the -1 mode).
*
* 0: A stream of individual JSON elements separated by whitespace
* are being parsed. We can return each complete element as we
* parse it.
*
* -1: No streaming is occuring. Only a single JSON value is
* allowed.
*/
final class AsyncParser[J] protected[ujson](
protected[ujson] var state: Int,
protected[ujson] var curr: Int,
protected[ujson] var stack: List[ObjArrVisitor[_, J]],
protected[ujson] var data: Array[Byte],
protected[ujson] var len: Int,
protected[ujson] var allocated: Int,
protected[ujson] var offset: Int,
protected[ujson] var done: Boolean,
protected[ujson] var streamMode: Int
) extends ByteBasedParser[J] {
protected[this] var line = 0
protected[this] var pos = 0
protected[this] final def newline(i: Int) { line += 1; pos = i + 1 }
protected[this] final def column(i: Int) = i - pos
final def copy() =
new AsyncParser(state, curr, stack, data.clone, len, allocated, offset, done, streamMode)
final def absorb(buf: ByteBuffer, facade: Visitor[_, J]): Either[ParsingFailedException, Seq[J]] = {
done = false
val buflen = buf.limit - buf.position
val need = len + buflen
resizeIfNecessary(need)
buf.get(data, len, buflen)
len = need
churn(facade)
}
final def absorb(bytes: Array[Byte], facade: Visitor[_, J]): Either[ParsingFailedException, Seq[J]] =
absorb(ByteBuffer.wrap(bytes), facade)
final def absorb(s: String, facade: Visitor[_, J]): Either[ParsingFailedException, Seq[J]] =
absorb(ByteBuffer.wrap(s.getBytes(utf8)), facade)
final def finish(facade: Visitor[_, J]): Either[ParsingFailedException, Seq[J]] = {
done = true
try churn(facade)
catch{case e: ParsingFailedException => Left(e)}
}
protected[this] final def resizeIfNecessary(need: Int): Unit = {
// if we don't have enough free space available we'll need to grow our
// data array. we never shrink the data array, assuming users will call
// feed with similarly-sized buffers.
if (need > allocated) {
val doubled = if (allocated < 0x40000000) allocated * 2 else Int.MaxValue
val newsize = max(need, doubled)
val newdata = new Array[Byte](newsize)
System.arraycopy(data, 0, newdata, 0, len)
data = newdata
allocated = newsize
}
}
/**
* Explanation of the new synthetic states. The parser machinery
* uses positive integers for states while parsing json values. We
* use these negative states to keep track of the async parser's
* status between json values.
*
* ASYNC_PRESTART: We haven't seen any non-whitespace yet. We
* could be parsing an array, or not. We are waiting for valid
* JSON.
*
* ASYNC_START: We've seen an array and have begun unwrapping
* it. We could see a ] if the array is empty, or valid JSON.
*
* ASYNC_END: We've parsed an array and seen the final ]. At this
* point we should only see whitespace or an EOF.
*
* ASYNC_POSTVAL: We just parsed a value from inside the array. We
* expect to see whitespace, a comma, or a ].
*
* ASYNC_PREVAL: We are in an array and we just saw a comma. We
* expect to see whitespace or a JSON value.
*/
@inline private[this] final def ASYNC_PRESTART = -5
@inline private[this] final def ASYNC_START = -4
@inline private[this] final def ASYNC_END = -3
@inline private[this] final def ASYNC_POSTVAL = -2
@inline private[this] final def ASYNC_PREVAL = -1
protected[ujson] def churn(facade: Visitor[_, J]): Either[ParsingFailedException, Seq[J]] = {
// accumulates json values
val results = mutable.ArrayBuffer.empty[J]
// we rely on exceptions to tell us when we run out of data
try {
while (true) {
if (state < 0) {
(at(offset): @switch) match {
case '\n' =>
newline(offset)
offset += 1
case ' ' | '\t' | '\r' =>
offset += 1
case '[' =>
if (state == ASYNC_PRESTART) {
offset += 1
state = ASYNC_START
} else if (state == ASYNC_END) {
die(offset, "expected eof")
} else if (state == ASYNC_POSTVAL) {
die(offset, "expected , or ]")
} else {
state = 0
}
case ',' =>
if (state == ASYNC_POSTVAL) {
offset += 1
state = ASYNC_PREVAL
} else if (state == ASYNC_END) {
die(offset, "expected eof")
} else {
die(offset, "expected json value")
}
case ']' =>
if (state == ASYNC_POSTVAL || state == ASYNC_START) {
if (streamMode > 0) {
offset += 1
state = ASYNC_END
} else {
die(offset, "expected json value or eof")
}
} else if (state == ASYNC_END) {
die(offset, "expected eof")
} else {
die(offset, "expected json value")
}
case c =>
if (state == ASYNC_END) {
die(offset, "expected eof")
} else if (state == ASYNC_POSTVAL) {
die(offset, "expected ] or ,")
} else {
if (state == ASYNC_PRESTART && streamMode > 0) streamMode = -1
state = 0
}
}
} else {
// jump straight back into rparse
offset = reset(offset)
val (value, j) = if (state <= 0) {
parse(offset, facade)
} else {
rparse(state, curr, stack, Nil)
}
if (streamMode > 0) {
state = ASYNC_POSTVAL
} else if (streamMode == 0) {
state = ASYNC_PREVAL
} else {
state = ASYNC_END
}
curr = j
offset = j
stack = Nil
results.append(value)
}
}
Right(results)
} catch {
case e: AsyncException =>
if (done) {
// if we are done, make sure we ended at a good stopping point
if (state == ASYNC_PREVAL || state == ASYNC_END) Right(results)
else Left(IncompleteParseException("exhausted input", e))
} else {
// we ran out of data, so return what we have so far
Right(results)
}
case e: ParseException =>
// we hit a parser error, so return that error and results so far
Left(e)
}
}
// every 1M we shift our array back to the beginning.
protected[this] final def reset(i: Int): Int = {
if (offset >= 1048576) {
val diff = offset
curr -= diff
len -= diff
offset = 0
pos -= diff
System.arraycopy(data, diff, data, 0, len)
i - diff
} else {
i
}
}
/**
* We use this to keep track of the last recoverable place we've
* seen. If we hit an AsyncException, we can later resume from this
* point.
*
* This method is called during every loop of rparse, and the
* arguments are the exact arguments we can pass to rparse to
* continue where we left off.
*/
protected[this] final def checkpoint(state: Int, i: Int, stack: List[ObjArrVisitor[_, J]]) {
this.state = state
this.curr = i
this.stack = stack
}
/**
* This is a specialized accessor for the case where our underlying data are
* bytes not chars.
*/
protected[this] final def byte(i: Int): Byte =
if (i >= len) throw new AsyncException else data(i)
// we need to signal if we got out-of-bounds
protected[this] final def at(i: Int): Char =
if (i >= len) throw new AsyncException else data(i).toChar
/**
* Access a byte range as a string.
*
* Since the underlying data are UTF-8 encoded, i and k must occur on unicode
* boundaries. Also, the resulting String is not guaranteed to have length
* (k - i).
*/
protected[this] final def at(i: Int, k: Int): CharSequence = {
if (k > len) throw new AsyncException
val size = k - i
val arr = new Array[Byte](size)
System.arraycopy(data, i, arr, 0, size)
new String(arr, utf8)
}
// the basic idea is that we don't signal EOF until done is true, which means
// the client explicitly send us an EOF.
protected[this] final def atEof(i: Int): Boolean =
if (done) i >= len else false
// we don't have to do anything special on close.
protected[this] final def close(): Unit = ()
}
/**
* This class is used internally by AsyncParser to signal that we've
* reached the end of the particular input we were given.
*/
private[ujson] class AsyncException extends Exception with control.NoStackTrace
/**
* This is a more prosaic exception which indicates that we've hit a
* parsing error.
*/
private[ujson] class FailureException extends Exception
© 2015 - 2025 Weber Informatics LLC | Privacy Policy