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

panda.bind.json.JsonTokener Maven / Gradle / Ivy

package panda.bind.json;

import java.io.IOException;
import java.io.InputStream;
import java.io.InputStreamReader;
import java.io.Reader;

import panda.io.stream.CharSequenceReader;


/**
 * A JSONTokener takes a source string and extracts characters and tokens from it.
 * 
 * @see JSON.org
 */
public class JsonTokener {
	private boolean ignoreComments = true;
	private long character;
	private boolean eof;
	private long index;
	private long line;
	private char previous;
	private Reader reader;
	private boolean usePrevious;

	/**
	 * Construct a JSONTokener from a Reader.
	 * 
	 * @param reader A reader.
	 */
	public JsonTokener(Reader reader) {
		this.reader = reader;
		this.eof = false;
		this.usePrevious = false;
		this.previous = 0;
		this.index = 0;
		this.character = 1;
		this.line = 1;
	}

	/**
	 * Construct a JSONTokener from an InputStream.
	 * 
	 * @param inputStream the input stream
	 * 
	 * @throws JsonException if a parse error occurred
	 */
	public JsonTokener(InputStream inputStream) throws JsonException {
		this(new InputStreamReader(inputStream));
	}

	/**
	 * Construct a JSONTokener from a string.
	 * 
	 * @param s A source string.
	 */
	public JsonTokener(CharSequence s) {
		this(new CharSequenceReader(s));
	}

	/**
	 * set the ignore comments option
	 * @param ignoreComments set the ignore comments option
	 */
	public void setIgnoreComments(boolean ignoreComments) {
		this.ignoreComments = ignoreComments;
	}

	/**
	 * 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.
	 * 
	 * @throws JsonException if a parse error occurred
	 */
	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;
	}

	public boolean end() {
		return this.eof && !this.usePrevious;
	}

	/**
	 * Determine if the source string still contains characters that next() can consume.
	 * 
	 * @return true if not yet at the end of the source.
	 * 
	 * @throws JsonException if a parse error occurred
	 */
	public boolean more() throws JsonException {
		this.next();
		if (this.end()) {
			return false;
		}
		this.back();
		return true;
	}

	/**
	 * Look at the next character in the source string.
	 * 
	 * @return The next character, or 0 if past the end of the source string.
	 */
	public char peek() {
		char c = this.next();
		if (!this.end()) {
			this.back();
		}
		return c;
	}

	/**
	 * Get the next character in the source string.
	 * 
	 * @return The next character, or 0 if past the end of the source string.
	 * 
	 * @throws JsonException if a parse error occurred
	 */
	public char next() throws JsonException {
		int c;
		if (this.usePrevious) {
			this.usePrevious = false;
			c = this.previous;
		}
		else {
			try {
				c = this.reader.read();
			}
			catch (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;
		}
		else if (c == '\n') {
			this.line += 1;
			this.character = 0;
		}
		else {
			this.character += 1;
		}
		this.previous = (char)c;
		return this.previous;
	}

	/**
	 * Consume the next character, and check that it matches a specified character.
	 * 
	 * @param c The character to match.
	 * @return The character.
	 * @throws JsonException if the character does not match.
	 */
	public char next(char c) throws JsonException {
		char n = this.next();
		if (n != c) {
			throw this.syntaxError("Expected '" + c + "' and instead saw '" + n + "'");
		}
		return n;
	}

	/**
	 * Get the next n characters.
	 * 
	 * @param n The number of characters to take.
	 * @return A string of n characters.
	 * @throws JsonException Substring bounds error if there are not n characters remaining in the
	 *             source string.
	 */
	public String next(int n) throws JsonException {
		if (n == 0) {
			return "";
		}

		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);
	}

	/**
	 * Get the next char in the string, skipping whitespace.
	 * 
	 * @return A character, or 0 if there are no more characters.
	 * @throws JsonException if a json syntax error occurs
	 */
	public char nextClean() throws JsonException {
		char c = this.skipBlank();
		if (ignoreComments) {
			while (true) {
				if (c == '/') {
					char n = this.peek();
					if (n == '/') {
						// inline comment
						c = this.skipTo('\n');
					}
					else if (n == '*') {
						// block comment
						c = this.skipBlockComment();
					}
					else {
						throw this.syntaxError("Comment syntax error");
					}
				}
				else if (c == '#') {
					// inline comment
					c = this.skipTo('\n');
				}
				else {
					break;
				}

				if (c == 0) {
					break;
				}
				if (c <= ' ') {
					c = this.skipBlank();
				}
			}
		}
		return c;
	}

	private char skipBlank() throws JsonException {
		for (;;) {
			char c = this.next();
			if (c == 0 || c > ' ') {
				return c;
			}
		}
	}

	private char skipBlockComment() throws JsonException {
		char c = this.next(); // skip '*'
		while (c > 0) {
			c = this.skipTo('*');
			if (c > 0) {
				c = this.next();
				if (c == '/') {
					c = this.next();
					break;
				}
			}
		}
		return c;
	}

	/**
	 * 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 " (double quote)
	 *            or ' (single quote).
	 * @return A String.
	 * @throws JsonException Unterminated string.
	 */
	public String nextString(char quote) throws JsonException {
		char c;
		StringBuilder sb = new StringBuilder();
		for (;;) {
			c = this.next();
			switch (c) {
			case 0:
			case '\n':
			case '\r':
				throw this.syntaxError("Unterminated string");
			case '\\':
				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:
				if (c == quote) {
					return sb.toString();
				}
				sb.append(c);
			}
		}
	}

	/**
	 * Get the text up but not including the '='specified character or the end of line, whichever comes
	 * first.
	 * 
	 * @return A id string.
	 * 
	 * @throws JsonException if a parse error occurred
	 */
	public String nextId() throws JsonException {
		StringBuilder sb = new StringBuilder();
		for (;;) {
			char c = this.next();
			if (c == 0 || !Character.isJavaIdentifierPart(c)) {
				if (c != 0) {
					this.back();
				}
				return sb.toString().trim();
			}
			sb.append(c);
		}
	}

	/**
	 * Get the text up but not including the specified character or the end of line, whichever comes
	 * first.
	 * 
	 * @param delimiter A delimiter character.
	 * @return A string.
	 * 
	 * @throws JsonException if a parse error occurred
	 */
	public String nextTo(char delimiter) throws JsonException {
		StringBuilder sb = new StringBuilder();
		for (;;) {
			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);
		}
	}

	/**
	 * 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.
	 * @return A string, trimmed.
	 * 
	 * @throws JsonException if a parse error occurred
	 */
	public String nextTo(String delimiters) throws JsonException {
		char c;
		StringBuilder sb = new StringBuilder();
		for (;;) {
			c = this.next();
			if (delimiters.indexOf(c) >= 0 || c == 0 || c == '\n' || c == '\r') {
				if (c != 0) {
					this.back();
				}
				return sb.toString().trim();
			}
			sb.append(c);
		}
	}

	/**
	 * Skip characters until the next character is the requested character. 
	 * 
	 * @param to A character to skip to.
	 * @return The requested character, or 0 if past the end of the source string.
	 * 
	 * @throws JsonException if a parse error occurred
	 */
	public char skipTo(char to) throws JsonException {
		char c;
		do {
			c = this.next();
			if (c == 0) {
				return c;
			}
		}
		while (c != to);
		return c;
	}

	/**
	 * Make a JSONException to signal a syntax error.
	 * 
	 * @param message The error message.
	 * @return A JSONException object, suitable for throwing
	 */
	public JsonException syntaxError(String message) {
		return new JsonException(message + this.toString());
	}

	/**
	 * Make a printable string of this JSONTokener.
	 * 
	 * @return " at {index} [character {character} line {line}]"
	 */
	public String toString() {
		return " at " + this.index + " [character " + this.character + " line " + this.line + "]";
	}
}




© 2015 - 2024 Weber Informatics LLC | Privacy Policy