/* * The MIT License (MIT) * * Copyright (c) 2015 Ziver Koc * * Permission is hereby granted, free of charge, to any person obtaining a copy * of this software and associated documentation files (the "Software"), to deal * in the Software without restriction, including without limitation the rights * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell * copies of the Software, and to permit persons to whom the Software is * furnished to do so, subject to the following conditions: * * The above copyright notice and this permission notice shall be included in * all copies or substantial portions of the Software. * * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN * THE SOFTWARE. */ package zutil.parser.json; import sun.reflect.generics.reflectiveObjects.NotImplementedException; import zutil.ClassUtil; import zutil.log.LogUtil; import zutil.parser.Base64Decoder; import zutil.parser.DataNode; import javax.activation.UnsupportedDataTypeException; import java.io.*; import java.lang.reflect.*; import java.util.HashMap; import java.util.Iterator; import java.util.List; import java.util.Map; import java.util.logging.Level; import java.util.logging.Logger; public class JSONObjectInputStream extends InputStream implements ObjectInput, Closeable{ private static final Logger logger = LogUtil.getLogger(); protected static final String MD_OBJECT_ID = "@object_id"; protected static final String MD_CLASS = "@class"; private JSONParser parser; private HashMap registeredClasses; private HashMap objectCache; private JSONObjectInputStream() { this.registeredClasses = new HashMap<>(); this.objectCache = new HashMap<>(); } public JSONObjectInputStream(Reader in) { this(); this.parser = new JSONParser(in); } public JSONObjectInputStream(InputStream in) { this(); this.parser = new JSONParser(in); } /** * If no metadata is available in the stream then this * class will be instantiated and assigned data from the received JSON. * * @param c the Class that will be instantiated for the root JSON. */ public void registerRootClass(Class c){ registeredClasses.put(null, c); } /** * A object instance of the given class will be created if the specific key is seen. * This can be used for streams that do not have any class metadata. * NOTE: any meta-data in the stream will override the registered classes. * * @param key a String key that will be looked for in the InputStream * @param c the Class that will be instantiated for the specific key. */ public void registerClass(String key, Class c){ registeredClasses.put(key, c); } /** * @return the object read from the stream */ @Override public Object readObject() throws IOException { return readObject(null); } /** * @param is a simple cast to this type * @return the object read from the stream */ public T readGenericObject() throws IOException { return readObject(null); } /** * @param c will override the registered root class and use this value instead * @param * @return the object read from the stream */ public synchronized T readObject(Class c) throws IOException { try{ DataNode root = parser.read(); if(root != null){ return (T)readObject(c, null, root); } } catch (ClassNotFoundException e) { logger.log(Level.WARNING, null, e); } catch (NoSuchFieldException e) { logger.log(Level.WARNING, null, e); } catch (InstantiationException e) { logger.log(Level.WARNING, null, e); } catch (IllegalAccessException e) { logger.log(Level.WARNING, null, e); } finally { objectCache.clear(); } return null; } @SuppressWarnings({ "rawtypes", "unchecked" }) protected Object readType(Class type, Class[] genType, String key, DataNode json) throws IllegalAccessException, ClassNotFoundException, InstantiationException, UnsupportedDataTypeException, NoSuchFieldException { if(json == null || type == null) return null; // Field type is a primitive? if(type.isPrimitive() || String.class.isAssignableFrom(type)){ return readPrimitive(type, json); } else if(type.isArray()){ if(type.getComponentType() == Byte.class) return Base64Decoder.decodeToByte(json.getString()); else{ Object array = Array.newInstance(type.getComponentType(), json.size()); for(int i=0; i=1? genType[0] : null), null, key, json.get(i))); } return list; } else if(Map.class.isAssignableFrom(type)){ if(genType == null || genType.length < 2) genType = ClassUtil.getGenericClasses(type, Map.class); Map map = (Map)type.newInstance(); for(Iterator it=json.keyIterator(); it.hasNext();){ String subKey = it.next(); map.put( subKey, readType((genType.length>=2? genType[1] : null), null, subKey, json.get(subKey))); } return map; } // Field is a new Object else{ return readObject(type, key, json); } } protected Object readObject(Class type, String key, DataNode json) throws IllegalAccessException, InstantiationException, ClassNotFoundException, IllegalArgumentException, UnsupportedDataTypeException, NoSuchFieldException { // Only parse if json is a map if(json == null || !json.isMap()) return null; // See if the Object id is in the cache before continuing if(json.getString("@object_id") != null && objectCache.containsKey(json.getInt(MD_OBJECT_ID))) return objectCache.get(json.getInt(MD_OBJECT_ID)); // Resolve the class Object obj = null; // Try using explicit class if(type != null){ obj = type.newInstance(); } // Try using metadata else if(json.getString(MD_CLASS) != null) { Class objClass = Class.forName(json.getString(MD_CLASS)); obj = objClass.newInstance(); } // Search for registered classes else if(registeredClasses.containsKey(key)){ Class objClass = registeredClasses.get(key); obj = objClass.newInstance(); } // Unknown class else { logger.warning("Unknown type for key: '"+key+"'"); return null; } // Read all fields from the new object instance for(Field field : obj.getClass().getDeclaredFields()){ if((field.getModifiers() & Modifier.STATIC) == 0 && (field.getModifiers() & Modifier.TRANSIENT) == 0 && json.get(field.getName()) != null){ // Parse field field.setAccessible(true); field.set(obj, readType( field.getType(), ClassUtil.getGenericClasses(field), field.getName(), json.get(field.getName()))); } } // Add object to the cache if(json.getString(MD_OBJECT_ID) != null) objectCache.put(json.getInt(MD_OBJECT_ID), obj); return obj; } protected static Object readPrimitive(Class type, DataNode json){ if (type == int.class || type == Integer.class) return json.getInt(); else if(type == long.class || type == Long.class) return json.getLong(); else if(type == double.class || type == Double.class) return json.getDouble(); else if(type == boolean.class || type == Boolean.class) return json.getBoolean(); else if(type == String.class) return json.getString(); return null; } @Override public void readFully(byte[] b) throws IOException { throw new NotImplementedException(); } @Override public void readFully(byte[] b, int off, int len) throws IOException { throw new NotImplementedException(); } @Override public int skipBytes(int n) throws IOException { throw new NotImplementedException(); } @Override public boolean readBoolean() throws IOException { throw new NotImplementedException(); } @Override public byte readByte() throws IOException { throw new NotImplementedException(); } @Override public int readUnsignedByte() throws IOException { throw new NotImplementedException(); } @Override public short readShort() throws IOException { throw new NotImplementedException(); } @Override public int readUnsignedShort() throws IOException { throw new NotImplementedException(); } @Override public char readChar() throws IOException { throw new NotImplementedException(); } @Override public int readInt() throws IOException { throw new NotImplementedException(); } @Override public long readLong() throws IOException { throw new NotImplementedException(); } @Override public float readFloat() throws IOException { throw new NotImplementedException(); } @Override public double readDouble() throws IOException { throw new NotImplementedException(); } @Override public String readLine() throws IOException { throw new NotImplementedException(); } @Override public String readUTF() throws IOException { throw new NotImplementedException(); } @Override public int read() throws IOException { throw new NotImplementedException(); } }