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

org.joda.beans.ser.json.JodaBeanJsonReader Maven / Gradle / Ivy

/*
 *  Copyright 2001-2014 Stephen Colebourne
 *
 *  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.joda.beans.ser.json;

import static org.joda.beans.ser.json.JodaBeanJsonWriter.BEAN;
import static org.joda.beans.ser.json.JodaBeanJsonWriter.META;
import static org.joda.beans.ser.json.JodaBeanJsonWriter.TYPE;
import static org.joda.beans.ser.json.JodaBeanJsonWriter.VALUE;

import java.io.Reader;
import java.io.StringReader;
import java.util.HashMap;
import java.util.Map;

import org.joda.beans.Bean;
import org.joda.beans.BeanBuilder;
import org.joda.beans.JodaBeanUtils;
import org.joda.beans.MetaBean;
import org.joda.beans.MetaProperty;
import org.joda.beans.ser.JodaBeanSer;
import org.joda.beans.ser.SerCategory;
import org.joda.beans.ser.SerDeserializer;
import org.joda.beans.ser.SerIterable;
import org.joda.beans.ser.SerIteratorFactory;
import org.joda.beans.ser.SerOptional;
import org.joda.beans.ser.SerTypeMapper;

/**
 * Provides the ability for a Joda-Bean to read from JSON.
 * 

* The JSON format is defined by {@link JodaBeanJsonWriter}. *

* This class contains mutable state and cannot be used from multiple threads. * A new instance must be created for each message. */ public class JodaBeanJsonReader { /** * Settings. */ private final JodaBeanSer settings; /** * The reader. */ private JsonInput input; /** * The base package including the trailing dot. */ private String basePackage; /** * The known types. */ private Map> knownTypes = new HashMap>(); /** * Creates an instance. * * @param settings the settings, not null */ public JodaBeanJsonReader(final JodaBeanSer settings) { JodaBeanUtils.notNull(settings, "settings"); this.settings = settings; } //----------------------------------------------------------------------- /** * Reads and parses to a bean. * * @param input the input string, not null * @return the bean, not null */ public Bean read(String input) { return read(input, Bean.class); } /** * Reads and parses to a bean. * * @param the root type * @param input the input string, not null * @param rootType the root type, not null * @return the bean, not null */ public T read(String input, Class rootType) { JodaBeanUtils.notNull(input, "input"); return read(new StringReader(input), rootType); } /** * Reads and parses to a bean. * * @param input the input reader, not null * @return the bean, not null */ public Bean read(Reader input) { return read(input, Bean.class); } /** * Reads and parses to a bean. * * @param the root type * @param input the input reader, not null * @param rootType the root type, not null * @return the bean, not null */ public T read(Reader input, Class rootType) { JodaBeanUtils.notNull(input, "input"); JodaBeanUtils.notNull(rootType, "rootType"); try { this.input = new JsonInput(input); return parseRoot(rootType); } catch (RuntimeException ex) { throw ex; } catch (Exception ex) { throw new RuntimeException(ex); } } //----------------------------------------------------------------------- /** * Parses the root bean. * * @param rootType the root type, not null * @return the bean, not null * @throws Exception if an error occurs */ private T parseRoot(Class declaredType) throws Exception { Object parsed = parseObject(input.acceptEvent(JsonEvent.OBJECT), declaredType, null, null, null, true); return declaredType.cast(parsed); } // parse a bean, event after object start passed in private Object parseBean(JsonEvent event, Class beanType) throws Exception { String propName = ""; try { SerDeserializer deser = settings.getDeserializers().findDeserializer(beanType); MetaBean metaBean = deser.findMetaBean(beanType); BeanBuilder builder = deser.createBuilder(beanType, metaBean); while (event != JsonEvent.OBJECT_END) { // property name propName = input.acceptObjectKey(event); MetaProperty metaProp = deser.findMetaProperty(beanType, metaBean, propName); if (metaProp == null) { input.skipData(); } else { Object value = parseObject(input.readEvent(), SerOptional.extractType(metaProp, beanType), metaProp, beanType, null, false); deser.setValue(builder, metaProp, SerOptional.wrapValue(metaProp, beanType, value)); } propName = ""; event = input.acceptObjectSeparator(); } return deser.build(beanType, builder); } catch (Exception ex) { throw new IllegalArgumentException( "Error parsing bean: " + beanType.getName() + "::" + propName + ", " + ex.getMessage(), ex); } } // parse object, event passed in private Object parseObject( JsonEvent event, Class declaredType, MetaProperty metaProp, Class beanType, SerIterable parentIterable, boolean rootType) throws Exception { // establish type if (event == JsonEvent.OBJECT) { event = input.readEvent(); if (event == JsonEvent.STRING) { String key = input.parseObjectKey(); if (key.equals(BEAN)) { return parseTypedBean(declaredType, rootType); } else if (key.equals(TYPE)) { return parseTypedSimple(declaredType); } else if (key.equals(META)) { return parseTypedMeta(); } else { input.pushBack('"'); input.pushBackObjectKey(key); event = JsonEvent.OBJECT; } } else if (event == JsonEvent.OBJECT_END) { input.pushBack('}'); event = JsonEvent.OBJECT; } else { throw new IllegalArgumentException("Invalid JSON data: Expected JSON object end but found " + event); } } // parse based on type if (Bean.class.isAssignableFrom(declaredType)) { if (event == JsonEvent.OBJECT) { return parseBean(input.readEvent(), declaredType); } else { return parseSimple(event, declaredType); } } else { if (event == JsonEvent.OBJECT || event == JsonEvent.ARRAY) { SerIterable childIterable = null; if (metaProp != null) { childIterable = SerIteratorFactory.INSTANCE.createIterable(metaProp, beanType); } else if (parentIterable != null) { childIterable = SerIteratorFactory.INSTANCE.createIterable(parentIterable); } if (childIterable == null) { throw new IllegalArgumentException("Invalid JSON data: Invalid object or array"); } return parseIterable(event, childIterable); } else { return parseSimple(event, declaredType); } } } private Object parseTypedBean(Class declaredType, boolean rootType) throws Exception { String typeStr = input.acceptString(); Class effectiveType = SerTypeMapper.decodeType(typeStr, settings, basePackage, knownTypes); if (rootType) { if (Bean.class.isAssignableFrom(effectiveType) == false) { throw new IllegalArgumentException("Root type is not a Joda-Bean: " + effectiveType.getName()); } basePackage = effectiveType.getPackage().getName() + "."; } if (declaredType.isAssignableFrom(effectiveType) == false) { throw new IllegalArgumentException("Specified type is incompatible with declared type: " + declaredType.getName() + " and " + effectiveType.getName()); } JsonEvent event = input.readEvent(); if (event == JsonEvent.COMMA) { event = input.readEvent(); } return parseBean(event, effectiveType); } private Object parseTypedSimple(Class declaredType) throws Exception { String typeStr = input.acceptString(); Class effectiveType = SerTypeMapper.decodeType(typeStr, settings, basePackage, knownTypes); if (declaredType.isAssignableFrom(effectiveType) == false) { throw new IllegalArgumentException("Specified type is incompatible with declared type: " + declaredType.getName() + " and " + effectiveType.getName()); } input.acceptEvent(JsonEvent.COMMA); String valueKey = input.acceptObjectKey(input.readEvent()); if (valueKey.equals(VALUE) == false) { throw new IllegalArgumentException("Invalid JSON data: Expected 'value' key but found " + valueKey); } Object result = parseSimple(input.readEvent(), effectiveType); input.acceptEvent(JsonEvent.OBJECT_END); return result; } private Object parseTypedMeta() throws Exception { String metaType = input.acceptString(); SerIterable childIterable = SerIteratorFactory.INSTANCE.createIterable(metaType, settings, knownTypes); input.acceptEvent(JsonEvent.COMMA); String valueKey = input.acceptObjectKey(input.readEvent()); if (valueKey.equals(VALUE) == false) { throw new IllegalArgumentException("Invalid JSON data: Expected 'value' key but found " + valueKey); } Object result = parseIterable(input.readEvent(), childIterable); input.acceptEvent(JsonEvent.OBJECT_END); return result; } private Object parseIterable(JsonEvent event, SerIterable iterable) throws Exception { if (iterable.category() == SerCategory.MAP) { return parseIterableMap(event, iterable); } else if (iterable.category() == SerCategory.COUNTED) { return parseIterableCounted(event, iterable); } else if (iterable.category() == SerCategory.TABLE) { return parseIterableTable(event, iterable); } else if (iterable.category() == SerCategory.GRID) { return parseIterableGrid(event, iterable); } else { return parseIterableArray(event, iterable); } } private Object parseIterableMap(JsonEvent event, SerIterable iterable) throws Exception { if (event == JsonEvent.OBJECT) { event = input.readEvent(); while (event != JsonEvent.OBJECT_END) { String keyStr = input.acceptObjectKey(event); Object key = settings.getConverter().convertFromString(iterable.keyType(), keyStr); Object value = parseObject(input.readEvent(), iterable.valueType(), null, null, iterable, false); iterable.add(key, null, value, 1); event = input.acceptObjectSeparator(); } } else if (event == JsonEvent.ARRAY) { event = input.readEvent(); while (event != JsonEvent.ARRAY_END) { input.ensureEvent(event, JsonEvent.ARRAY); Object key = parseObject(input.readEvent(), iterable.keyType(), null, null, null, false); input.acceptEvent(JsonEvent.COMMA); Object value = parseObject(input.readEvent(), iterable.valueType(), null, null, iterable, false); input.acceptEvent(JsonEvent.ARRAY_END); iterable.add(key, null, value, 1); event = input.acceptArraySeparator(); } return iterable.build(); } else { throw new IllegalArgumentException("Invalid JSON data: Expected array or object but found " + event); } return iterable.build(); } private Object parseIterableTable(JsonEvent event, SerIterable iterable) throws Exception { input.ensureEvent(event, JsonEvent.ARRAY); event = input.readEvent(); while (event != JsonEvent.ARRAY_END) { input.ensureEvent(event, JsonEvent.ARRAY); Object key = parseObject(input.readEvent(), iterable.keyType(), null, null, null, false); input.acceptEvent(JsonEvent.COMMA); Object column = parseObject(input.readEvent(), iterable.columnType(), null, null, null, false); input.acceptEvent(JsonEvent.COMMA); Object value = parseObject(input.readEvent(), iterable.valueType(), null, null, iterable, false); iterable.add(key, column, value, 1); input.acceptEvent(JsonEvent.ARRAY_END); event = input.acceptArraySeparator(); } return iterable.build(); } private Object parseIterableGrid(JsonEvent event, SerIterable iterable) throws Exception { input.ensureEvent(event, JsonEvent.ARRAY); input.acceptEvent(JsonEvent.NUMBER_INTEGRAL); int rows = (int) input.parseNumberIntegral(); input.acceptEvent(JsonEvent.COMMA); input.acceptEvent(JsonEvent.NUMBER_INTEGRAL); int columns = (int) input.parseNumberIntegral(); iterable.dimensions(new int[] {rows, columns}); event = input.acceptArraySeparator(); while (event != JsonEvent.ARRAY_END) { input.ensureEvent(event, JsonEvent.ARRAY); input.acceptEvent(JsonEvent.NUMBER_INTEGRAL); int row = (int) input.parseNumberIntegral(); input.acceptEvent(JsonEvent.COMMA); input.acceptEvent(JsonEvent.NUMBER_INTEGRAL); int column = (int) input.parseNumberIntegral(); input.acceptEvent(JsonEvent.COMMA); Object value = parseObject(input.readEvent(), iterable.valueType(), null, null, iterable, false); input.acceptEvent(JsonEvent.ARRAY_END); iterable.add(row, column, value, 1); event = input.acceptArraySeparator(); } return iterable.build(); } private Object parseIterableCounted(JsonEvent event, SerIterable iterable) throws Exception { input.ensureEvent(event, JsonEvent.ARRAY); event = input.readEvent(); while (event != JsonEvent.ARRAY_END) { input.ensureEvent(event, JsonEvent.ARRAY); Object value = parseObject(input.readEvent(), iterable.valueType(), null, null, iterable, false); input.acceptEvent(JsonEvent.COMMA); input.acceptEvent(JsonEvent.NUMBER_INTEGRAL); iterable.add(null, null, value, (int) input.parseNumberIntegral()); input.acceptEvent(JsonEvent.ARRAY_END); event = input.acceptArraySeparator(); } return iterable.build(); } private Object parseIterableArray(JsonEvent event, SerIterable iterable) throws Exception { input.ensureEvent(event, JsonEvent.ARRAY); event = input.readEvent(); while (event != JsonEvent.ARRAY_END) { Object value = parseObject(event, iterable.valueType(), null, null, iterable, false); iterable.add(null, null, value, 1); event = input.acceptArraySeparator(); } return iterable.build(); } private Object parseSimple(JsonEvent event, Class type) throws Exception { switch (event) { case STRING: { String text = input.parseString(); if (type == String.class || type == Object.class) { return text; } return settings.getConverter().convertFromString(type, text); } case NUMBER_INTEGRAL: { long value = input.parseNumberIntegral(); if (type == Long.class || type == long.class) { return Long.valueOf(value); } else if (type == Short.class || type == short.class) { if (value < Short.MIN_VALUE || value > Short.MAX_VALUE) { throw new IllegalArgumentException("Invalid JSON data: Expected short, but was " + value); } return Short.valueOf((short) value); } else if (type == Byte.class || type == byte.class) { if (value < Byte.MIN_VALUE || value > Byte.MAX_VALUE) { throw new IllegalArgumentException("Invalid JSON data: Expected byte, but was " + value); } return Byte.valueOf((byte) value); } else { if (value < Integer.MIN_VALUE || value > Integer.MAX_VALUE) { throw new IllegalArgumentException("Invalid JSON data: Expected int, but was " + value); } return Integer.valueOf((int) value); } } case NUMBER_FLOATING: { double value = input.parseNumberFloating(); if (type == Float.class || type == float.class) { return Float.valueOf((float) value); } else { return Double.valueOf(value); } } case NULL: { if (type == double.class) { return Double.NaN; // leniently accept null for NaN } else if (type == float.class) { return Float.NaN; // leniently accept null for NaN } else { return null; } } case TRUE: return Boolean.TRUE; case FALSE: return Boolean.FALSE; default: throw new IllegalArgumentException("Invalid JSON data: Expected simple type but found " + event); } } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy