Package clj_json

Source Code of clj_json.JsonExt

package clj_json;

import org.codehaus.jackson.JsonToken;
import org.codehaus.jackson.JsonParser;
import org.codehaus.jackson.JsonGenerator;
import java.math.BigInteger;
import java.util.Iterator;
import java.util.Map;
import java.util.Map.Entry;
import java.util.Set;
import clojure.lang.IFn;
import clojure.lang.Keyword;
import clojure.lang.PersistentArrayMap;
import clojure.lang.PersistentVector;
import clojure.lang.ITransientMap;
import clojure.lang.ITransientCollection;
import clojure.lang.Seqable;
import clojure.lang.BigInt;

public class JsonExt {
    public static class Generator {
        final JsonGenerator jg;
        final Map coercions;

        public Generator(JsonGenerator jg, Map coercions){
            this.jg = jg;
            this.coercions = coercions;
        }

        public Object coercionKey(Object obj){
            if (obj != null) {
                return obj.getClass();
            }
            return null;
        }

        public void generate(Object obj) throws Exception{
            if (this.coercions != null) {
                IFn fn;
                while ((fn = (IFn)coercions.get(this.coercionKey(obj))) != null){
                    obj = fn.invoke(obj);
                }
            }

            if (obj instanceof String) {
                jg.writeString((String) obj);
            } else if (obj instanceof Number) {
                if (obj instanceof Integer) {
                    jg.writeNumber((Integer) obj);
                } else if (obj instanceof Long) {
                    jg.writeNumber((Long) obj);
                } else if (obj instanceof BigInteger) {
                    jg.writeNumber((BigInteger) obj);
                } else if (obj instanceof BigInt) {
                  jg.writeNumber(((BigInt) obj).toBigInteger());
                } else if (obj instanceof Double) {
                    jg.writeNumber((Double) obj);
                } else if (obj instanceof Float) {
                    jg.writeNumber((Float) obj);
                }
            } else if (obj instanceof Boolean) {
                jg.writeBoolean((Boolean) obj);
            } else if (obj == null) {
                jg.writeNull();
            } else if (obj instanceof Keyword) {
                jg.writeString(((Keyword) obj).getName());
            } else if (obj instanceof Map) {
                Map<?,?> map = (Map<?,?>) obj;
                jg.writeStartObject();
                for (Map.Entry<?,?> me : map.entrySet()) {
                    Object key = me.getKey();
                    if (key instanceof Keyword) {
                        jg.writeFieldName(((Keyword) key).getName());
                    } else {
                        jg.writeFieldName(key.toString());
                    }
                    generate(me.getValue());
                }
                jg.writeEndObject();
            } else if (obj instanceof Iterable) {
                jg.writeStartArray();
                for (Object o : (Iterable)obj){
                    generate(o);
                }
                jg.writeEndArray();
            }
            else {
                throw new Exception("Cannot generate " + obj);
            }
        }
    }

    public static void generate(JsonGenerator jg, Map coercions, Object obj) throws Exception {
        Generator g = new Generator(jg, coercions);
        g.generate(obj);
    }

    public static Object parse(JsonParser jp, boolean first, boolean keywords, Object eofValue) throws Exception {
        if (first) {
            jp.nextToken();
            if (jp.getCurrentToken() == null) {
                return eofValue;
            }
        }
        switch (jp.getCurrentToken()) {
        case START_OBJECT:
            ITransientMap map = PersistentArrayMap.EMPTY.asTransient();
            jp.nextToken();
            while (jp.getCurrentToken() != JsonToken.END_OBJECT) {
                String keyStr = jp.getText();
                jp.nextToken();
                Object key = keywords ? Keyword.intern(keyStr) : keyStr;
                map = map.assoc(key, parse(jp, false, keywords, eofValue));
                jp.nextToken();
            }
            return map.persistent();
        case START_ARRAY:
            ITransientCollection vec = PersistentVector.EMPTY.asTransient();
            jp.nextToken();
            while (jp.getCurrentToken() != JsonToken.END_ARRAY) {
                vec = vec.conj(parse(jp, false, keywords, eofValue));
                jp.nextToken();
            }
            return vec.persistent();
        case VALUE_STRING:
            return jp.getText();
        case VALUE_NUMBER_INT:
            return jp.getNumberValue();
        case VALUE_NUMBER_FLOAT:
            return jp.getDoubleValue();
        case VALUE_TRUE:
            return Boolean.TRUE;
        case VALUE_FALSE:
            return Boolean.FALSE;
        case VALUE_NULL:
            return null;
        default:
            throw new Exception("Cannot parse " + jp.getCurrentToken());
        }
    }
}
TOP

Related Classes of clj_json.JsonExt

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.