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

org.sfj.JSONOne Maven / Gradle / Ivy

/*
 * Copyright 2020 C. Schanck
 *
 * 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 org.sfj;

import java.io.IOException;
import java.io.Serializable;
import java.io.StringWriter;
import java.io.Writer;
import java.text.NumberFormat;
import java.text.ParseException;
import java.util.AbstractList;
import java.util.AbstractMap;
import java.util.Arrays;
import java.util.LinkedList;
import java.util.Objects;
import java.util.Set;
import java.util.concurrent.ConcurrentHashMap;
import java.util.concurrent.CopyOnWriteArrayList;

/**
 * 

Single class containing classes for representing JSON objects * programmatically. Includes pretty print and parser. NOT like the world * needs another JSON package. But... 1) it was an interesting exercise * 2) perhaps someone needs something to adapt from, etc. *

Very vanilla. Null, Boolean, String, Number, Array and Map objects. *

Parser (well, the scanner actually) is inefficient for large objects; * it expects a single String object for input. So there is that. Would be * not too tough to remedy, but you'd need pushback/lookahead etc. *

Should handle backslash escaping properly, \uabcd unicode, etc. *

Also, in this day and age of awesome parser frameworks, it was * really fun to write a parser by hand. I used: http://www.craftinginterpreters.com/ * as a refresher for how to do old school scanning/parsing. I swear I have an * actual book around somewhere, but I could not find it, and I did not feel * like digging through Knuth this time. * @author cschanck */ public class JSONOne { /** * JSON value types */ public enum Type { NUMBER, STRING, BOOLEAN, MAP, ARRAY, NULL } /** * A JSON object. Asking for the value as the worng type results in a * ClassCastException. */ public interface JObject extends Serializable { default JArray arrayValue() { throw new ClassCastException(); } default boolean boolValue() { throw new ClassCastException(); } JSONOne.Type getType(); default JMap mapValue() { throw new ClassCastException(); } default boolean nullValue() { return false; } default Number numberValue() { throw new ClassCastException(); } Object pojoValue(); default String stringValue() { throw new ClassCastException(); } void print(Writer w, int indent, boolean compact) throws IOException; default String print(boolean compact) throws IOException { return print(0, compact); } default String print() throws IOException { return print(0, true); } default String print(int indent, boolean compact) throws IOException { StringWriter sw = new StringWriter(); print(sw, indent, compact); return sw.toString(); } } private static abstract class AbstractJSONObject implements JObject { private Type type; private Object obj; public AbstractJSONObject() { } public AbstractJSONObject(Type type, Object obj) { this.type = type; this.obj = obj; } @Override public Type getType() { return type; } @Override public Object pojoValue() { return obj; } @Override public boolean equals(Object o) { if (this == o) { return true; } if (o == null || getClass() != o.getClass()) { return false; } AbstractJSONObject object = (AbstractJSONObject) o; return Objects.equals(obj, object.obj) && type == object.type; } @Override public int hashCode() { return Objects.hash(obj, type); } @Override public String toString() { return obj.toString(); } static String indent(int indention) { char[] c = new char[indention * 2]; Arrays.fill(c, ' '); return new String(c); } public void print(Writer w, int indent, boolean compact) throws IOException { w.append(obj.toString()); } } /** * Number object. Represents a Number in JSON. Underlying Value * POJO wil be up converted to either Long or Double, always. */ public static class JNumber extends AbstractJSONObject { private boolean isFixed; public JNumber() { } public JNumber(Number obj) { super(Type.NUMBER, obj instanceof Float ? obj.doubleValue() : obj instanceof Double ? obj : obj.longValue()); this.isFixed = !(obj instanceof Double); } public boolean isFixed() { return isFixed; } @Override public Number numberValue() { return (Number) pojoValue(); } } /** * Holds a string value. */ public static class JString extends AbstractJSONObject { public JString() { } public JString(CharSequence obj) { super(Type.STRING, obj.toString()); Objects.requireNonNull(obj); } @Override public String stringValue() { return (String) pojoValue(); } @Override public void print(Writer w, int indent, boolean compact) throws IOException { w.append('"'); w.append(escapeString(stringValue())); w.append('"'); } } /** * Boolean value */ public static class JBoolean extends AbstractJSONObject { public JBoolean() { } public JBoolean(Boolean obj) { super(Type.BOOLEAN, obj); } @Override public boolean boolValue() { return (Boolean) pojoValue(); } @Override public void print(Writer w, int indent, boolean compact) throws IOException { w.append((boolValue() ? "true" : "false")); } } /** * Null value. */ public static class JNull extends AbstractJSONObject { public JNull() { super(Type.NULL, null); } @Override public boolean nullValue() { return true; } @Override public void print(Writer w, int indent, boolean compact) throws IOException { w.append("null"); } @Override public String toString() { return "null"; } } /** * JSON Array class. List of {@link JSONOne.JObject}s. */ public static class JArray extends AbstractList implements JObject { private CopyOnWriteArrayList list = new CopyOnWriteArrayList<>(); public JArray() { } @Override public JObject get(int index) { return list.get(index); } @Override public JArray arrayValue() { return this; } @Override public Type getType() { return Type.ARRAY; } @Override public Object pojoValue() { return list; } @Override public int size() { return list.size(); } @Override public JObject set(int index, JObject element) { return list.set(index, element); } @Override public void add(int index, JObject element) { list.add(index, element); } @Override public JObject remove(int index) { return list.remove(index); } public JObject setNumber(int index, Number val) { return list.set(index, new JNumber(val)); } public void addNumber(int index, Number val) { list.add(index, new JNumber(val)); } public void addNumber(Number val) { list.add(new JNumber(val)); } public JObject setString(int index, String val) { return list.set(index, new JString(val)); } public void addString(int index, String val) { list.add(index, new JString(val)); } public void addString(String val) { list.add(new JString(val)); } public JObject setBoolean(int index, boolean val) { return list.set(index, new JBoolean(val)); } public void addBoolean(int index, boolean val) { list.add(index, new JBoolean(val)); } public void addBoolean(boolean val) { list.add(new JBoolean(val)); } public JObject setNull(int index) { return list.set(index, new JNull()); } public void addNull(int index) { list.add(index, new JNull()); } public void addNull() { list.add(new JNull()); } @Override public void print(Writer w, int indent, boolean compact) throws IOException { String ind1 = AbstractJSONObject.indent(indent); String ind2 = AbstractJSONObject.indent(indent + 1); w.append('['); boolean first = true; for (JObject obj : this) { if (first) { first = false; } else { w.append(','); } if (!compact) { w.append(System.lineSeparator()); w.append(ind2); } obj.print(w, indent + 1, compact); } if (!compact) { w.append(System.lineSeparator()); w.append(ind1); } w.append(']'); } } /** * JSON Map value. String to {@link JSONOne.JObject}. */ public static class JMap extends AbstractMap implements JObject { private ConcurrentHashMap map = new ConcurrentHashMap<>(); public JMap() { } @Override public JMap mapValue() { return this; } @Override public Set> entrySet() { return map.entrySet(); } @Override public Type getType() { return Type.MAP; } @Override public Object pojoValue() { return map; } @Override public boolean containsValue(Object value) { return map.containsValue(value); } @Override public boolean containsKey(Object key) { return map.containsKey(key); } @Override public JObject get(Object key) { return map.get(key); } @Override public JObject put(String key, JObject value) { return map.put(key, value); } @Override public JObject remove(Object key) { return map.remove(key); } @Override public void clear() { map.clear(); } @Override public Set keySet() { return map.keySet(); } private Object getTyped(String key, Type t, Object defVal) { JObject p = get(key); if (p != null) { if (p.getType().equals(t)) { switch (t) { case MAP: case ARRAY: return p; default: return p.pojoValue(); } } } return defVal; } public Boolean getBoolean(String key, Boolean defValue) { return (Boolean) getTyped(key, Type.BOOLEAN, defValue); } public Number getNumber(String key, Number defValue) { return (Number) getTyped(key, Type.NUMBER, defValue); } public String getString(String key, String defValue) { return (String) getTyped(key, Type.STRING, defValue); } public JArray getArray(String key, JArray defValue) { return (JArray) getTyped(key, Type.ARRAY, defValue); } public JArray getMap(String key, JMap defValue) { return (JArray) getTyped(key, Type.MAP, defValue); } public boolean isNull(String key, boolean defValue) { JObject p = map.get(key); if (p.getType().equals(Type.NULL)) { return true; } return defValue; } public JMap putBoolean(String key, boolean value) { map.put(key, new JBoolean(value)); return this; } public JMap putNumber(String key, Number val) { map.put(key, new JNumber(val)); return this; } public JMap putString(String key, String value) { map.put(key, new JString(value)); return this; } public JMap putNull(String key) { map.put(key, new JNull()); return this; } public JMap putMap(String key, JMap value) { map.put(key, value); return this; } public JMap putArray(String key, JArray value) { map.put(key, value); return this; } @Override public void print(Writer w, int indent, boolean compact) throws IOException { String ind = AbstractJSONObject.indent(indent); w.append('{'); boolean first = true; String ind2 = AbstractJSONObject.indent(indent + 1); for (Entry obj : this.entrySet()) { if (first) { first = false; } else { w.append(','); } if (!compact) { w.append(System.lineSeparator()); w.append(ind2); } w.append('"'); w.append(obj.getKey()); w.append('"'); w.append(compact ? ":" : " : "); obj.getValue().print(w, indent + 1, compact); } if (!compact) { w.append(System.lineSeparator()); w.append(ind); } w.append('}'); } } enum TokenType { LEFT_CURLY, RIGHT_CURLY, LEFT_BRACKET, RIGHT_BACKET, COMMA, NUMBER, STRING, NULL, EOF, TRUE, FALSE, COLON, } static class Token { final TokenType type; final String lexeme; final Object literal; final int line; Token(TokenType type, String lexeme, Object literal, int line) { this.type = type; this.lexeme = lexeme; this.literal = literal; this.line = line; } public String toString() { return type + " " + lexeme + " " + literal; } } /** * Scanner for parsing. */ static class Scanner { private final String input; private int current = 0; private int start = 0; private int line = 1; private LinkedList pushBack = new LinkedList<>(); Scanner(String input) { this.input = input; } ParseException error(int line, String message) { return report(line, "", message); } ParseException report(int line, String where, String message) { return new ParseException("[line " + line + "] Error" + where + ": " + message, line); } public Token nextOrFail(TokenType target) throws ParseException { Token tok = nextToken(); if (tok.type.equals(target)) { return tok; } throw error(line, "mismatch type; looking for " + target); } void pushback(Token t) { pushBack.addFirst(t); } char peek() { if (isAtEnd()) { return '\0'; } return input.charAt(current); } private boolean isAtEnd() { return current >= input.length(); } char advance() { if (isAtEnd()) { return '\0'; } return input.charAt(current++); } private Token token(TokenType type) { return token(type, null); } private Token token(TokenType type, Object literal) { String text = input.substring(start, current); start = current; return new Token(type, text, literal, line); } public Token nextToken() throws ParseException { if (!pushBack.isEmpty()) { return pushBack.removeFirst(); } for (; !isAtEnd(); ) { char ch = advance(); switch (ch) { case '\n': line++; start++; break; case ' ': case '\r': case '\t': start++; break; case ':': return token(TokenType.COLON); case '{': return token(TokenType.LEFT_CURLY); case '}': return token(TokenType.RIGHT_CURLY); case '[': return token(TokenType.LEFT_BRACKET); case ']': return token(TokenType.RIGHT_BACKET); case '"': return stringToken(); case '-': case '+': case '0': case '1': case '2': case '3': case '4': case '5': case '6': case '7': case '8': case '9': return numberToken(ch); case 'n': return advanceMatching("ull", false, TokenType.NULL); case 't': return advanceMatching("rue", false, TokenType.TRUE); case 'f': return advanceMatching("alse", false, TokenType.FALSE); case ',': return token(TokenType.COMMA); } } return token(TokenType.EOF); } private Token numberToken(char ch) { StringBuilder ret = new StringBuilder(); integer(ch, ret); String fract = fraction(); if (fract.length() > 0) { ret.append(fract); digits(ret); } exponent(ret); return token(TokenType.NUMBER, ret.toString()); } private void exponent(StringBuilder ret) { if (Character.toUpperCase(peek()) == 'E') { ret.append(advance()); if (peek() == '-') { ret.append(advance()); } else if (peek() == '+') { ret.append(advance()); } digits(ret); } } private String integer(char first, StringBuilder ret) { ret.append(first); digits(ret); return ret.toString(); } private String fraction() { if (peek() == '.') { advance(); return "."; } return ""; } private String digits(StringBuilder ret) { for (char p = peek(); Character.isDigit(p); p = peek()) { ret.append(advance()); } return ret.toString(); } private Token advanceMatching(String next, boolean caseMatters, TokenType type) throws ParseException { if (!caseMatters) { next = next.toLowerCase(); } for (int i = 0; i < next.length(); i++) { char c = advance(); if (!caseMatters) { c = Character.toLowerCase(c); } if (next.charAt(i) != c) { throw error(line, "expected [" + next + "]"); } } return token(type); } private Token stringToken() { StringBuilder lit = new StringBuilder(); for (char p = advance(); p != '"'; p = advance()) { if (p == '\\') { escape(lit); } else { lit.append(p); } } return token(TokenType.STRING, lit.toString()); } private void escape(StringBuilder lit) { char p = advance(); switch (p) { case 'u': String utf = ((("" + advance()) + advance()) + advance()) + advance(); int value = Integer.parseInt(utf, 16); lit.append(Character.toChars(value)); break; case 'n': lit.append('\n'); break; case 'r': lit.append('\r'); break; case 'b': lit.append('\b'); break; case 't': lit.append('\t'); break; case 'f': lit.append('\f'); break; default: lit.append(p); break; } } } /** * Parser, parses reasonably standard JSON. */ public static class Parser { private final Scanner scanner; private NumberFormat nFormat = NumberFormat.getInstance(); public Parser(String input) { scanner = new Scanner(input); } private ParseException error(Token token, String message) { if (token.type == TokenType.EOF) { return scanner.report(token.line, " at end", message); } return scanner.report(token.line, " at '" + token.lexeme + "'", message); } public JObject singleObject() throws ParseException { Token p = scanner.nextToken(); switch (p.type) { case LEFT_BRACKET: return array(); case LEFT_CURLY: return map(); case FALSE: return new JBoolean(false); case TRUE: return new JBoolean(true); case NULL: return new JNull(); case STRING: return new JString((String) p.literal); case NUMBER: return new JNumber(nFormat.parse(p.lexeme)); case EOF: return null; default: throw error(p, "Unexpected token"); } } private JMap map() throws ParseException { JMap ret = new JMap(); listOfMapEntries(ret); return ret; } private Token peekToken() throws ParseException { Token p = scanner.nextToken(); if (p != null) { scanner.pushback(p); } return p; } private void listOfMapEntries(JMap ret) throws ParseException { if (peekToken().type.equals(TokenType.RIGHT_CURLY)) { scanner.nextToken(); return; } for (; ; ) { String key = (String) scanner.nextOrFail(TokenType.STRING).literal; scanner.nextOrFail(TokenType.COLON); JObject obj = singleObject(); ret.put(key, obj); Token next = scanner.nextToken(); switch (next.type) { case COMMA: if (peekToken().type.equals(TokenType.RIGHT_CURLY)) { scanner.nextToken(); return; } break; case RIGHT_CURLY: return; default: throw error(next, "expected ',' or '}'"); } } } private JArray array() throws ParseException { JArray ret = new JArray(); listOfArrayEntries(ret); return ret; } private void listOfArrayEntries(JArray ret) throws ParseException { if (peekToken().type.equals(TokenType.RIGHT_BACKET)) { scanner.nextToken(); return; } for (; ; ) { JObject obj = singleObject(); ret.add(obj); Token next = scanner.nextToken(); switch (next.type) { case COMMA: if (peekToken().type.equals(TokenType.RIGHT_BACKET)) { scanner.nextToken(); return; } break; case RIGHT_BACKET: return; default: throw error(next, "expected ',' or ']'"); } } } } public static CharSequence unescapeString(CharSequence seq) { StringBuilder w = new StringBuilder(); final int len = seq.length(); for (int pos = 0; pos < len; pos++) { char ch = seq.charAt(pos); if (ch == '\\') { ch = seq.charAt(++pos); switch (ch) { case 'u': String utf = Character.toString(seq.charAt(++pos)) + seq.charAt(++pos) + seq.charAt(++pos) + seq.charAt(++pos); int value = Integer.parseInt(utf, 16); w.append(Character.toChars(value)); break; case 'n': w.append('\n'); break; case 'r': w.append('\r'); break; case 'b': w.append('\b'); break; case 't': w.append('\t'); break; case 'f': w.append('\f'); break; default: w.append(ch); break; } } else { w.append(ch); } } return w; } public static CharSequence escapeString(CharSequence seq) { StringBuilder w = new StringBuilder(); final int len = seq.length(); for (int i = 0; i < len; i++) { char ch = seq.charAt(i); switch (ch) { case '"': w.append("\\\""); break; case '\\': w.append("\\\\"); break; case '\b': w.append("\\b"); break; case '\f': w.append("\\f"); break; case '\n': w.append("\\n"); break; case '\r': w.append("\\r"); break; case '\t': w.append("\\t"); break; default: if ((ch <= '\u001F') || (ch >= '\u007F' && ch <= '\u009F') || (ch >= '\u2000' && ch <= '\u20FF')) { String ss = Integer.toHexString(ch); w.append("\\u"); for (int k = 0; k < 4 - ss.length(); k++) { w.append('0'); } w.append(ss.toUpperCase()); } else { w.append(ch); } } } return w; } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy