PlotSquared/src/main/java/com/intellectualcrafters/json/JSONTokener.java

404 lines
12 KiB
Java
Raw Normal View History

2014-11-08 20:27:09 +01:00
package com.intellectualcrafters.json;
2014-11-05 04:42:08 +01:00
2015-01-13 17:38:15 +01:00
import java.io.BufferedReader;
import java.io.IOException;
import java.io.InputStream;
import java.io.InputStreamReader;
import java.io.Reader;
import java.io.StringReader;
/**
2014-12-18 03:15:11 +01:00
* A JSONTokener takes a source string and extracts characters and tokens from it. It is used by the JSONObject and
* JSONArray constructors to parse JSON source strings.
2014-11-05 04:42:08 +01:00
*
* @author JSON.org
* @version 2014-05-03
*/
public class JSONTokener {
2014-11-08 20:35:00 +01:00
private final Reader reader;
2014-12-18 03:15:11 +01:00
private long character;
private boolean eof;
private long index;
private long line;
private char previous;
private boolean usePrevious;
2015-02-23 02:32:27 +01:00
/**
* Construct a JSONTokener from a Reader.
*
2014-12-18 03:15:11 +01:00
* @param reader A reader.
*/
2014-11-05 04:42:08 +01:00
public JSONTokener(final Reader reader) {
this.reader = reader.markSupported() ? reader : new BufferedReader(reader);
this.eof = false;
this.usePrevious = false;
this.previous = 0;
this.index = 0;
this.character = 1;
this.line = 1;
}
2015-02-23 02:32:27 +01:00
/**
* Construct a JSONTokener from an InputStream.
2014-11-05 04:42:08 +01:00
*
2014-12-18 03:15:11 +01:00
* @param inputStream The source.
*/
2014-11-05 04:42:08 +01:00
public JSONTokener(final InputStream inputStream) throws JSONException {
this(new InputStreamReader(inputStream));
}
2015-02-23 02:32:27 +01:00
/**
* Construct a JSONTokener from a string.
*
2014-12-18 03:15:11 +01:00
* @param s A source string.
*/
2014-11-05 04:42:08 +01:00
public JSONTokener(final String s) {
this(new StringReader(s));
}
2015-02-23 02:32:27 +01:00
/**
* Get the hex value of a character (base16).
2014-11-05 04:42:08 +01:00
*
2014-12-18 03:15:11 +01:00
* @param c A character between '0' and '9' or between 'A' and 'F' or between 'a' and 'f'.
*
2014-11-05 04:42:08 +01:00
* @return An int between 0 and 15, or -1 if c was not a hex digit.
*/
2014-11-05 04:42:08 +01:00
public static int dehexchar(final char c) {
if ((c >= '0') && (c <= '9')) {
return c - '0';
}
2014-11-05 04:42:08 +01:00
if ((c >= 'A') && (c <= 'F')) {
return c - ('A' - 10);
}
2014-11-05 04:42:08 +01:00
if ((c >= 'a') && (c <= 'f')) {
return c - ('a' - 10);
}
return -1;
}
2015-02-23 02:32:27 +01:00
2014-11-08 20:35:00 +01:00
/**
2014-12-18 03:15:11 +01:00
* Back up one character. This provides a sort of lookahead capability, so that you can test for a digit or letter
* before attempting to parse the next number or identifier.
2014-11-08 20:35:00 +01:00
*/
public void back() throws JSONException {
if (this.usePrevious || (this.index <= 0)) {
throw new JSONException("Stepping back two steps is not supported");
}
this.index -= 1;
this.character -= 1;
this.usePrevious = true;
this.eof = false;
}
2015-02-23 02:32:27 +01:00
public boolean end() {
return this.eof && !this.usePrevious;
}
2015-02-23 02:32:27 +01:00
/**
2014-12-18 03:15:11 +01:00
* Determine if the source string still contains characters that next() can consume.
2014-11-05 04:42:08 +01:00
*
* @return true if not yet at the end of the source.
*/
public boolean more() throws JSONException {
this.next();
if (this.end()) {
return false;
}
this.back();
return true;
}
2015-02-23 02:32:27 +01:00
/**
* Get the next character in the source string.
*
* @return The next character, or 0 if past the end of the source string.
*/
public char next() throws JSONException {
int c;
if (this.usePrevious) {
this.usePrevious = false;
c = this.previous;
2014-12-18 03:15:11 +01:00
} else {
try {
c = this.reader.read();
2014-12-18 03:15:11 +01:00
} catch (final IOException exception) {
throw new JSONException(exception);
}
if (c <= 0) { // End of stream
this.eof = true;
c = 0;
}
}
this.index += 1;
if (this.previous == '\r') {
this.line += 1;
this.character = c == '\n' ? 0 : 1;
2014-12-18 03:15:11 +01:00
} else if (c == '\n') {
this.line += 1;
this.character = 0;
2014-12-18 03:15:11 +01:00
} else {
this.character += 1;
}
this.previous = (char) c;
return this.previous;
}
2015-02-23 02:32:27 +01:00
/**
2014-12-18 03:15:11 +01:00
* Consume the next character, and check that it matches a specified character.
*
* @param c The character to match.
2014-11-05 04:42:08 +01:00
*
* @return The character.
2014-12-18 03:15:11 +01:00
*
* @throws JSONException if the character does not match.
*/
2014-11-05 04:42:08 +01:00
public char next(final char c) throws JSONException {
final char n = this.next();
if (n != c) {
2014-11-05 04:42:08 +01:00
throw this.syntaxError("Expected '" + c + "' and instead saw '" + n + "'");
}
return n;
}
2015-02-23 02:32:27 +01:00
/**
* Get the next n characters.
*
2014-12-18 03:15:11 +01:00
* @param n The number of characters to take.
*
2014-11-05 04:42:08 +01:00
* @return A string of n characters.
2014-12-18 03:15:11 +01:00
*
* @throws JSONException Substring bounds error if there are not n characters remaining in the source string.
*/
2014-11-05 04:42:08 +01:00
public String next(final int n) throws JSONException {
if (n == 0) {
return "";
}
final char[] chars = new char[n];
int pos = 0;
while (pos < n) {
chars[pos] = this.next();
if (this.end()) {
throw this.syntaxError("Substring bounds error");
}
pos += 1;
}
return new String(chars);
}
2015-02-23 02:32:27 +01:00
/**
* Get the next char in the string, skipping whitespace.
2014-11-05 04:42:08 +01:00
*
* @return A character, or 0 if there are no more characters.
2014-12-18 03:15:11 +01:00
*
2014-11-08 20:27:09 +01:00
* @throws JSONException
*/
public char nextClean() throws JSONException {
2015-02-20 07:34:19 +01:00
for (;;) {
2014-11-05 04:42:08 +01:00
final char c = this.next();
if ((c == 0) || (c > ' ')) {
return c;
}
}
}
2015-02-23 02:32:27 +01:00
/**
2014-12-18 03:15:11 +01:00
* Return the characters up to the next close quote character. Backslash processing is done. The formal JSON format
* does not allow strings in single quotes, but an implementation is allowed to accept them.
*
* @param quote The quoting character, either <code>"</code> &nbsp;<small>(double quote)</small> or <code>'</code>
* &nbsp;<small>(single quote)</small>.
2014-11-05 04:42:08 +01:00
*
* @return A String.
2014-12-18 03:15:11 +01:00
*
* @throws JSONException Unterminated string.
*/
2014-11-05 04:42:08 +01:00
public String nextString(final char quote) throws JSONException {
char c;
2014-11-05 04:42:08 +01:00
final StringBuilder sb = new StringBuilder();
2015-02-20 07:34:19 +01:00
for (;;) {
c = this.next();
switch (c) {
2014-11-05 04:42:08 +01:00
case 0:
case '\n':
case '\r':
throw this.syntaxError("Unterminated string");
case '\\':
2014-11-05 04:42:08 +01:00
c = this.next();
switch (c) {
case 'b':
sb.append('\b');
break;
case 't':
sb.append('\t');
break;
case 'n':
sb.append('\n');
break;
case 'f':
sb.append('\f');
break;
case 'r':
sb.append('\r');
break;
case 'u':
sb.append((char) Integer.parseInt(this.next(4), 16));
break;
case '"':
case '\'':
case '\\':
case '/':
sb.append(c);
break;
default:
throw this.syntaxError("Illegal escape.");
}
break;
default:
2014-11-05 04:42:08 +01:00
if (c == quote) {
return sb.toString();
}
sb.append(c);
}
}
}
2015-02-23 02:32:27 +01:00
/**
2014-12-18 03:15:11 +01:00
* Get the text up but not including the specified character or the end of line, whichever comes first.
*
* @param delimiter A delimiter character.
2014-11-05 04:42:08 +01:00
*
* @return A string.
*/
2014-11-05 04:42:08 +01:00
public String nextTo(final char delimiter) throws JSONException {
final StringBuilder sb = new StringBuilder();
2015-02-20 07:34:19 +01:00
for (;;) {
2014-11-05 04:42:08 +01:00
final char c = this.next();
if ((c == delimiter) || (c == 0) || (c == '\n') || (c == '\r')) {
if (c != 0) {
this.back();
}
return sb.toString().trim();
}
sb.append(c);
}
}
2015-02-23 02:32:27 +01:00
/**
2014-12-18 03:15:11 +01:00
* Get the text up but not including one of the specified delimiter characters or the end of line, whichever comes
* first.
*
* @param delimiters A set of delimiter characters.
2014-11-05 04:42:08 +01:00
*
* @return A string, trimmed.
*/
2014-11-05 04:42:08 +01:00
public String nextTo(final String delimiters) throws JSONException {
char c;
2014-11-05 04:42:08 +01:00
final StringBuilder sb = new StringBuilder();
2015-02-20 07:34:19 +01:00
for (;;) {
c = this.next();
2014-11-05 04:42:08 +01:00
if ((delimiters.indexOf(c) >= 0) || (c == 0) || (c == '\n') || (c == '\r')) {
if (c != 0) {
this.back();
}
return sb.toString().trim();
}
sb.append(c);
}
}
2015-02-23 02:32:27 +01:00
/**
2014-12-18 03:15:11 +01:00
* Get the next value. The value can be a Boolean, Double, Integer, JSONArray, JSONObject, Long, or String, or the
* JSONObject.NULL object.
2014-11-05 04:42:08 +01:00
*
* @return An object.
2014-12-18 03:15:11 +01:00
*
* @throws JSONException If syntax error.
*/
public Object nextValue() throws JSONException {
char c = this.nextClean();
String string;
switch (c) {
case '"':
case '\'':
return this.nextString(c);
case '{':
this.back();
return new JSONObject(this);
case '[':
this.back();
return new JSONArray(this);
}
/*
* Handle unquoted text. This could be the values true, false, or
* null, or it can be a number. An implementation (such as this one)
* is allowed to also accept non-standard forms.
* Accumulate characters until we reach the end of the text or a
* formatting character.
*/
2014-11-05 04:42:08 +01:00
final StringBuilder sb = new StringBuilder();
while ((c >= ' ') && (",:]}/\\\"[{;=#".indexOf(c) < 0)) {
sb.append(c);
c = this.next();
}
this.back();
string = sb.toString().trim();
if ("".equals(string)) {
throw this.syntaxError("Missing value");
}
return JSONObject.stringToValue(string);
}
2015-02-23 02:32:27 +01:00
/**
2014-12-18 03:15:11 +01:00
* Skip characters until the next character is the requested character. If the requested character is not found, no
* characters are skipped.
*
* @param to A character to skip to.
2014-11-05 04:42:08 +01:00
*
2014-12-18 03:15:11 +01:00
* @return The requested character, or zero if the requested character is not found.
*/
2014-11-05 04:42:08 +01:00
public char skipTo(final char to) throws JSONException {
char c;
try {
2014-11-05 04:42:08 +01:00
final long startIndex = this.index;
final long startCharacter = this.character;
final long startLine = this.line;
this.reader.mark(1000000);
do {
c = this.next();
if (c == 0) {
this.reader.reset();
this.index = startIndex;
this.character = startCharacter;
this.line = startLine;
return c;
}
2014-12-18 03:15:11 +01:00
} while (c != to);
} catch (final IOException exception) {
throw new JSONException(exception);
}
this.back();
return c;
}
2015-02-23 02:32:27 +01:00
/**
* Make a JSONException to signal a syntax error.
*
2014-12-18 03:15:11 +01:00
* @param message The error message.
*
2014-11-05 04:42:08 +01:00
* @return A JSONException object, suitable for throwing
*/
2014-11-05 04:42:08 +01:00
public JSONException syntaxError(final String message) {
return new JSONException(message + this.toString());
}
2015-02-23 02:32:27 +01:00
/**
* Make a printable string of this JSONTokener.
*
* @return " at {index} [character {character} line {line}]"
*/
2014-11-05 04:42:08 +01:00
@Override
public String toString() {
2014-11-05 04:42:08 +01:00
return " at " + this.index + " [character " + this.character + " line " + this.line + "]";
}
}