Package com.addthis.codec

Source Code of com.addthis.codec.CodecJSON

/*
* 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 com.addthis.codec;

import java.lang.reflect.Array;
import java.lang.reflect.InvocationTargetException;
import java.lang.reflect.Modifier;

import java.util.Collection;
import java.util.HashSet;
import java.util.Iterator;
import java.util.List;
import java.util.Map;
import java.util.Map.Entry;
import java.util.concurrent.atomic.AtomicBoolean;
import java.util.concurrent.atomic.AtomicInteger;
import java.util.concurrent.atomic.AtomicLong;

import com.addthis.basis.util.Bytes;

import com.addthis.maljson.JSONArray;
import com.addthis.maljson.JSONException;
import com.addthis.maljson.JSONObject;
import com.addthis.maljson.LineNumberInfo;

public class CodecJSON extends Codec {

    private static final boolean quoteDefault = System.getProperty("json.quoting", "0") != "0";

    public static interface JSONCodable extends Codable {

        public JSONObject toJSONObject() throws Exception;

        public void fromJSONObject(JSONObject jo) throws Exception;
    }

    public CodecJSON() {
        this(false);
    }

    public CodecJSON(boolean quoteKeys) {
        this.quoteKeys = quoteKeys;
    }

    private final boolean quoteKeys;

    @Override
    public byte[] encode(Object obj) throws Exception {
        return Bytes.toBytes(encodeString(obj, !quoteKeys));
    }

    @Override
    public CodableStatistics statistics(Object obj) {
        throw new UnsupportedOperationException();
    }


    @Override
    public <T> T decode(T shell, byte data[]) throws CodecExceptionLineNumber, JSONException {
        return decodeString(shell, Bytes.toString(data));
    }

    public <T> T decode(T shell, byte data[], List<CodecExceptionLineNumber> warnings) throws Exception {
        return decodeString(shell, Bytes.toString(data), warnings);
    }

    @Override
    public boolean storesNull(byte data[]) {
        throw new UnsupportedOperationException();
    }

    public static JSONObject encodeJSON(Object object) throws Exception {
        return (JSONObject) encodeObject(object);
    }

    public static String encodeString(Object object) {
        return encodeString(object, 0, !quoteDefault);
    }

    public static String encodeString(Object object, int nest) {
        return encodeString(object, nest, !quoteDefault);
    }

    public static String encodeString(Object object, boolean noQuoteKeys) {
        return encodeString(object, 0, noQuoteKeys);
    }

    private static String encodeString(Object object, int nest, boolean noQuoteKeys) {
        try {
            Object ret = encodeObject(object);
            if (ret instanceof JSONObject) {
                return nest > 0 ? ((JSONObject) ret).toString(nest) : ((JSONObject) ret).toString();
            } else {
                return ret.toString();
            }
        } catch (RuntimeException ex) {
            throw ex;
        } catch (Exception ex) {
            throw new RuntimeException(ex);
        }
    }

    private static Object encodeArray(Object value) throws Exception {
        JSONArray arr = new JSONArray();
        int len = Array.getLength(value);
        for (int i = 0; i < len; i++) {
            arr.put(encodeObject(Array.get(value, i)));
        }
        return arr;
    }

    private static Object encodeObject(Object object) throws Exception {
        if (object == null) {
            return JSONObject.NULL;
        }
        if (object.getClass().isArray()) {
            return encodeArray(object);
        }
        if (object instanceof JSONCodable) {
            return ((JSONCodable) object).toJSONObject();
        }
        JSONObject obj = null;
        boolean lock = object instanceof Codec.ConcurrentCodable;
        if (lock && !((Codec.ConcurrentCodable) object).encodeLock()) {
            throw new Exception("Unable to acquire encoding lock on " + object);
        }
        try {
            if (object instanceof SuperCodable) {
                ((SuperCodable) object).preEncode();
            }
            CodableClassInfo classInfo = getClassFieldMap(object.getClass());
            if (classInfo.size() == 0 && !(object instanceof Codable)) {
                return object;
            }
            obj = new JSONObject();
            String altType = classInfo.getClassName(object);
            if (altType != null) {
                obj.put(classInfo.getClassField(), altType);
            }
            for (Iterator<CodableFieldInfo> fields = classInfo.values().iterator(); fields.hasNext();) {
                CodableFieldInfo field = fields.next();
                Object value = field.get(object);
                if (value == null || value == JSONObject.NULL || field.isReadOnly()) {
                    continue;
                }
                if (CodecJSON.JSONCodable.class.isAssignableFrom(field.getType())) {
                    value = ((CodecJSON.JSONCodable) value).toJSONObject();
                    obj.put(field.getName(), value);
                } else if (field.isArray()) {
                    obj.put(field.getName(), encodeArray(value));
                } else if (field.isMap()) {
                    Map<?, ?> map = (Map<?, ?>) value;
                    JSONObject jmap = new JSONObject();
                    for (Entry<?, ?> entry : map.entrySet()) {
                        Object mval = entry.getValue();
                        // TODO fails with null keys
                        jmap.put(entry.getKey().toString(), encodeObject(mval));
                    }
                    obj.put(field.getName(), jmap);
                } else if (field.isCollection()) {
                    JSONArray jarr = new JSONArray();
                    for (Iterator<?> iter = ((Collection<?>) value).iterator(); iter.hasNext();) {
                        jarr.put(encodeObject(iter.next()));
                    }
                    obj.put(field.getName(), jarr);
                } else if (field.isCodable()) {
                    obj.put(field.getName(), encodeObject(value));
                } else if (field.isEnum()) {
                    obj.put(field.getName(), value.toString());
                } else if (field.isNative()) {
                    obj.put(field.getName(), value);
                } else {
                    System.out.println("unmatched field '" + field.getName() + "' = " + field);
                }
            }
        } finally {
            if (lock) {
                ((Codec.ConcurrentCodable) object).encodeUnlock();
            }
        }
        return obj;
    }

    public static <T> T decodeString(T object, String json)
            throws CodecExceptionLineNumber, JSONException {

        return decodeJSON(object, new JSONObject(json));
    }

    public static <T> T decodeString(T object, String json, List<CodecExceptionLineNumber> warnings)
            throws CodecExceptionLineNumber, JSONException {

        JSONObject jsonObj = new JSONObject(json);
        return decodeJSONInternal(getClassFieldMap(object.getClass()), object, jsonObj, warnings);
    }

    public static <T> T decodeArray(Class<T> type, Object object)
            throws CodecExceptionLineNumber, JSONException {

        return decodeArrayInternal(type, object, LineNumberInfo.MissingInfo, null);
    }

    public static <T> T decodeArray(Class<T> type, Object object, List<CodecExceptionLineNumber> warnings)
            throws CodecExceptionLineNumber, JSONException {

        return decodeArrayInternal(type, object, LineNumberInfo.MissingInfo, warnings);
    }

    private static <T> T decodeArrayInternal(Class<T> type, Object object, LineNumberInfo info, List<CodecExceptionLineNumber> warnings)
            throws CodecExceptionLineNumber, JSONException {

        if (object == null || object == JSONObject.NULL) {
            return null;
        }
        if (object.getClass() != JSONArray.class) {
            throw new CodecExceptionLineNumber(object.toString() + " not an instance of JSONArray for class " + type, info);
        }
        JSONArray array = (JSONArray) object;
        T value = (T) Array.newInstance(type, array.length());
        if (type == byte.class || type == Byte.class) {
            for (int i = 0; i < array.length(); i++) {
                Array.set(value, i, (byte) array.getInt(i));
            }
        } else {
            for (int i = 0; i < array.length(); i++) {
                Array.set(value, i, decodeObjectInternal(type, array.opt(i), array.getLineNumber(i), warnings));
            }
        }
        return value;
    }

    @SuppressWarnings("unchecked")
    public static <T> T decodeObject(Class<T> type, Object json)
            throws CodecExceptionLineNumber, JSONException {

        return decodeObjectInternal(type, json, LineNumberInfo.MissingInfo, null);
    }

    @SuppressWarnings("unchecked")
    public static <T> T decodeObject(Class<T> type, Object json, List<CodecExceptionLineNumber> warnings)
            throws CodecExceptionLineNumber, JSONException {

        return decodeObjectInternal(type, json, LineNumberInfo.MissingInfo, warnings);
    }

    @SuppressWarnings("unchecked")
    public static <T> T decodeObjectInternal(Class<T> type, Object json,
            LineNumberInfo info, List<CodecExceptionLineNumber> warnings)
            throws CodecExceptionLineNumber, JSONException {

        if (json == null || json == JSONObject.NULL) {
            return null;
        }
        if (isNative(type) || !(json instanceof JSONObject)) {
            if (type != json.getClass()) {
                if (Number.class.isAssignableFrom(type)) {
                    Number num = (Number) json;
                    if (type == Short.class) {
                        json = new Short(num.shortValue());
                    } else if (type == Integer.class) {
                        json = new Integer(num.intValue());
                    } else if (type == Long.class) {
                        json = new Long(num.longValue());
                    } else if (type == Float.class) {
                        json = new Float(num.floatValue());
                    } else if (type == Double.class) {
                        json = new Double(num.doubleValue());
                    } else if (type == AtomicInteger.class) {
                        json = new AtomicInteger(num.intValue());
                    } else if (type == AtomicLong.class) {
                        json = new AtomicLong(num.longValue());
                    }
                } else if (type.isEnum()) {
                    /**
                     * Attempting to invoke {@link Enum#valueOf(Class, String)}
                     * fails in the downstream assignment on
                     * {@link Array#set(Object, int, Object)} so reflection is used
                     * instead.
                     */
                    try {
                        json = type.getMethod("valueOf", String.class).invoke(null, json.toString().toUpperCase());
                    } catch (InvocationTargetException ex) {
                        throw new CodecExceptionLineNumber("Could not convert the string \"" + json.toString() +
                                                           "\" to the Enum type " + type.getName(), info);
                    } catch (NoSuchMethodException ex) {
                        throw new IllegalStateException("Attempted to decode enum type", ex);
                    } catch (IllegalAccessException ex) {
                        throw new IllegalStateException("Attempted to decode enum type", ex);
                    }
                }
            }
            return (T) json;
        } else {
            JSONObject jsonObj = (JSONObject) json;

            if (info == LineNumberInfo.MissingInfo) {
                info = jsonObj.getLineNumberInfo();
            }

            CodableClassInfo classInfo = getClassFieldMap(type);
            String classField = classInfo.getClassField();
            String stype = jsonObj.optString(classField, null);
            Class<?> atype;
            try {
                atype = stype != null ? classInfo.getClass(stype) : type;
            } catch (Exception ex) {
                throw new CodecExceptionLineNumber(ex, jsonObj.getValLineNumber(classField));
            }
            if (atype != null && atype != type) {
                classInfo = getClassFieldMap(atype);
                type = (Class<T>) atype;
            }
            if (classField != null) {
                jsonObj.remove(classField);
            }
            try {
                return decodeJSONInternal(classInfo, type.newInstance(), jsonObj, warnings);
            } catch (InstantiationException ex) {
                CodecExceptionLineNumber celn = translateInstantiationException(type, info);
                throw celn;
            } catch (IllegalAccessException ex) {
                throw new CodecExceptionLineNumber("Could not access either the type or the constructor of " +
                                                   type.getName(), info);
            }
        }
    }

    private static <T> CodecExceptionLineNumber translateInstantiationException(Class<T> type, LineNumberInfo info) {
        if (type.isInterface()) {
            String msg = "Perhaps you failed to specify what type of object to create. " +
                         "Could not instantiate the interface " + type.getName() + ". ";
            return new CodecExceptionLineNumber(msg, info);
        } else if (Modifier.isAbstract(type.getModifiers())) {
            String msg = "Perhaps you failed to specify what type of object to create. " +
                         "Could not instantiate the abstract class " + type.getName() + ". ";
            return new CodecExceptionLineNumber(msg, info);
        } else {
            String msg = "Could not instantiate an instance of " + type.getName();
            return new CodecExceptionLineNumber(msg, info);
        }
    }

    @SuppressWarnings("unchecked")
    public static <T> T decodeJSON(T object, JSONObject json) throws CodecExceptionLineNumber, JSONException {
        return decodeJSONInternal(getClassFieldMap(object.getClass()), object, json, null);
    }

    @SuppressWarnings("unchecked")
    public static <T> T decodeJSON(CodableClassInfo classInfo, T object, JSONObject json)
            throws CodecExceptionLineNumber, JSONException {
        return decodeJSONInternal(classInfo, object, json, null);
    }

    private static <T> T decodeJSONInternal(CodableClassInfo classInfo, T object,
            JSONObject json, List<CodecExceptionLineNumber> warnings)
            throws CodecExceptionLineNumber, JSONException {

        if (object instanceof JSONCodable) {
            try {
                ((JSONCodable) object).fromJSONObject(json);
            } catch (Exception ex) {
                throw new CodecExceptionLineNumber(ex, json.getLineNumberInfo());
            }
            return object;
        }
        java.util.Set<String> unknownFields = new HashSet<String>(json.keySet());
        for (Iterator<CodableFieldInfo> fields = classInfo.values().iterator(); fields.hasNext();) {
            CodableFieldInfo field = fields.next();
            if (field.isWriteOnly()) {
                continue;
            }
            String fieldName = field.getName();
            unknownFields.remove(fieldName);
            Class type = field.getType();
            Object value = json.opt(fieldName);
            if (value == null) {
                field.set(object, json.getLineNumberInfo(), value, LineNumberInfo.MissingInfo);
                continue;
            }
            LineNumberInfo keyInfo = json.getKeyLineNumber(fieldName);
            LineNumberInfo valInfo = json.getValLineNumber(fieldName);

            if (CodecJSON.JSONCodable.class.isAssignableFrom(type)) {
                Object oValue = value;
                try {
                    value = type.newInstance();
                } catch (InstantiationException ex) {
                    CodecExceptionLineNumber celn = translateInstantiationException(type, valInfo);
                    throw celn;
                } catch (IllegalAccessException ex) {
                    throw new CodecExceptionLineNumber("Could not access the type or the constructor of " +
                                                       type.getName(), valInfo);
                }
                try {
                    ((CodecJSON.JSONCodable) value).fromJSONObject(new JSONObject(oValue.toString()));
                } catch (Exception ex) {
                    throw new CodecExceptionLineNumber(ex, valInfo);
                }
            } else if (field.isArray()) {
                value = decodeArrayInternal(type, value, valInfo, warnings);
            } else if (field.isNative()) {
                if (value.getClass() != type) {
                    if (value.getClass() == Integer.class || value.getClass() == int.class) {
                        // upconvert integer values to long if the field requires long
                        if (type == Long.class || type == long.class || type == AtomicLong.class) {
                            value = new Long(((Integer) value).longValue());
                        }
                        // upconvert integer values to double if the field requires double
                        if (type == Double.class || type == double.class) {
                            value = new Double(((Integer) value));
                        }
                        // downconvert integer to short if the field requires short
                        if (type == Short.class || type == short.class) {
                            value = new Short(((Integer) value).shortValue());
                        }
                    }
                    // downconvert double to float if the field requires a float
                    if ((value.getClass() == double.class || value.getClass() == Double.class) &&
                        (type == float.class || type == Float.class)) {
                        value = new Float(((Double) value));
                    }
                    // upconvert long values to double if the field requires double
                    if ((value.getClass() == long.class || value.getClass() == Long.class) &&
                        (type == double.class || type == Double.class)) {
                        value = new Double(((Integer) value));
                    }
                    // upconvert float values to double if the field requires double
                    if ((value.getClass() == float.class || value.getClass() == Float.class) &&
                        (type == double.class || type == Double.class)) {
                        value = new Double(((Float) value));
                    }
                    if (value.getClass() == String.class) {
                        try {

                            // convert String values to int if the field requires int
                            if (type == Integer.class || type == int.class || type == AtomicInteger.class) {
                                value = Integer.parseInt((String) value);
                            }

                            // convert String values to long if the field requires long
                            if (type == long.class || type == Long.class || type == AtomicLong.class) {
                                value = Long.parseLong((String) value);
                            }

                            // convert String values to double if the field requires double
                            if (type == double.class || type == Double.class) {
                                value = Double.parseDouble((String) value);
                            }

                            // convert String values to boolean if the field requires boolean
                            if (type == boolean.class || type == Boolean.class || type == AtomicBoolean.class) {
                                value = Boolean.parseBoolean((String) value);
                            }
                        } catch (NumberFormatException ex) {
                            if (type == Integer.class || type == int.class || type == AtomicInteger.class) {
                                throw new CodecExceptionLineNumber("cannot convert the string to an integer", valInfo);
                            } else if (type == long.class || type == Long.class || type == AtomicLong.class) {
                                throw new CodecExceptionLineNumber("cannot convert the string to a long", valInfo);
                            } else if (type == double.class || type == Double.class) {
                                throw new CodecExceptionLineNumber("cannot convert the string to a double", valInfo);
                            } else {
                                throw new IllegalStateException("unhandled case in the NumberFormatException");
                            }
                        }
                    }
                    if (type == AtomicInteger.class) {
                        value = new AtomicInteger((Integer) value);
                    } else if (type == AtomicLong.class) {
                        value = new AtomicLong((Long) value);

                    } else if (type == AtomicBoolean.class) {
                        value = new AtomicBoolean((Boolean) value);
                    }
                }
                // this space left intentionally blank
            } else if (field.isMap()) {
                Map map;
                try {
                    map = (Map) type.newInstance();
                } catch (Exception ex) {
                    throw new CodecExceptionLineNumber(ex, keyInfo);
                }
                JSONObject jmap = (JSONObject) value;
                Class vc = (Class) field.getGenericTypes()[1];
                boolean va = field.isMapValueArray();
                for (Iterator<String> iter = jmap.keys(); iter.hasNext();) {
                    String key = iter.next();
                    if (field.isInterned()) {
                        key = key.intern();
                    }
                    map.put(key, va ? decodeArrayInternal(vc, jmap.get(key), jmap.getKeyLineNumber(key), warnings)
                                    : decodeObjectInternal(vc, jmap.get(key), jmap.getKeyLineNumber(key), warnings));
                }
                value = map;
            } else if (field.isCollection()) {
                Collection col;
                JSONArray jarr;
                try {
                    col = (Collection) type.newInstance();
                } catch (Exception ex) {
                    throw new CodecExceptionLineNumber(ex, keyInfo);
                }
                try {
                    jarr = (JSONArray) value;
                } catch (Exception ex) {
                    throw new CodecExceptionLineNumber(ex, valInfo);
                }
                Class vc = field.getCollectionClass();
                boolean ar = field.isCollectionArray();
                for (int i = 0; i < jarr.length(); i++) {
                    col.add(ar ? decodeArrayInternal(vc, jarr.get(i), jarr.getLineNumber(i), warnings) :
                            decodeObjectInternal(vc, jarr.get(i), jarr.getLineNumber(i), warnings));
                }
                value = col;
            } else if (field.isEnum()) {
                try {
                    String valString = value.toString();
                    if (valString != "") {
                        value = Enum.valueOf(type, valString.toUpperCase());
                    }
                } catch (Exception ex) {
                    throw new CodecExceptionLineNumber(ex, valInfo);
                }
            } else if (field.isCodable()) {
                value = decodeObjectInternal(type, value, valInfo, warnings);
            }
            field.set(object, json.getLineNumberInfo(), value, valInfo);
        }
        if (object instanceof SuperCodable) {
            ((SuperCodable) object).postDecode();
        }
        if (warnings != null) {
            for (String fieldName : unknownFields) {
                String msg = "Unrecognized field '" + fieldName + "' in class " + object.getClass().getName();
                warnings.add(new UnrecognizedFieldException(msg, json.getKeyLineNumber(fieldName),
                        fieldName, object.getClass()));
            }
        }
        return object;
    }

}
TOP

Related Classes of com.addthis.codec.CodecJSON

TOP
Copyright © 2018 www.massapi.com. All rights reserved.
All source code are property of their respective owners. Java is a trademark of Sun Microsystems, Inc and owned by ORACLE Inc. Contact coftware#gmail.com.