org.joda.beans.ser.xml.JodaBeanXmlWriter Maven / Gradle / Ivy
/*
* Copyright 2001-2015 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.xml;
import static org.joda.beans.ser.xml.JodaBeanXml.BEAN;
import static org.joda.beans.ser.xml.JodaBeanXml.COL;
import static org.joda.beans.ser.xml.JodaBeanXml.COLS;
import static org.joda.beans.ser.xml.JodaBeanXml.COUNT;
import static org.joda.beans.ser.xml.JodaBeanXml.ENTRY;
import static org.joda.beans.ser.xml.JodaBeanXml.ITEM;
import static org.joda.beans.ser.xml.JodaBeanXml.KEY;
import static org.joda.beans.ser.xml.JodaBeanXml.METATYPE;
import static org.joda.beans.ser.xml.JodaBeanXml.NULL;
import static org.joda.beans.ser.xml.JodaBeanXml.ROW;
import static org.joda.beans.ser.xml.JodaBeanXml.ROWS;
import static org.joda.beans.ser.xml.JodaBeanXml.TYPE;
import java.util.HashMap;
import java.util.Map;
import org.joda.beans.Bean;
import org.joda.beans.MetaProperty;
import org.joda.beans.ser.JodaBeanSer;
import org.joda.beans.ser.SerCategory;
import org.joda.beans.ser.SerIterator;
import org.joda.beans.ser.SerOptional;
import org.joda.beans.ser.SerTypeMapper;
import org.joda.convert.StringConverter;
/**
* Provides the ability for a Joda-Bean to be written to XML.
*
* This class contains mutable state and cannot be used from multiple threads.
* A new instance must be created for each message.
*
* The XML consists of a root level 'bean' element with a 'type' attribute.
* At each subsequent level, a bean is output using the property name.
* Where necessary, the 'type' attribute is used to clarify a type.
*
* Simple types, defined by Joda-Convert, are output as strings.
* Beans are output recursively within the parent property element.
* Collections are output using 'item' elements within the property element.
* The 'item' elements will use 'key' for map keys, 'count' for multiset counts
* and 'null=true' for null entries. Note that map keys must be simple types.
*
* If a collection contains a collection then more information is written.
* A 'metatype' attribute is added to define the high level type, such as List.
* At this level, the data read back may not be identical to that written.
*
* Type names are shortened by the package of the root type if possible.
* Certain basic types are also handled, such as String, Integer, File and URI.
*
* @author Stephen Colebourne
*/
public class JodaBeanXmlWriter {
/**
* The settings to use.
*/
private final JodaBeanSer settings;
/**
* The string builder.
*/
private final StringBuilder builder;
/**
* The root bean.
*/
private Bean rootBean;
/**
* The base package including the trailing dot.
*/
private String basePackage;
/**
* The known types.
*/
private Map, String> knownTypes = new HashMap, String>();
/**
* Creates an instance.
*
* @param settings the settings to use, not null
*/
public JodaBeanXmlWriter(final JodaBeanSer settings) {
this(settings, new StringBuilder(1024));
}
/**
* Creates an instance.
*
* @param settings the settings to use, not null
* @param builder the builder to output to, not null
*/
public JodaBeanXmlWriter(final JodaBeanSer settings, final StringBuilder builder) {
this.settings = settings;
this.builder = builder;
}
//-----------------------------------------------------------------------
/**
* Writes the bean to a string.
*
* The type of the bean will be set in the message.
*
* @param bean the bean to output, not null
* @return the XML, not null
*/
public String write(final Bean bean) {
return write(bean, true);
}
/**
* Writes the bean to a string.
*
* @param bean the bean to output, not null
* @param rootType true to output the root type
* @return the XML, not null
*/
public String write(final Bean bean, final boolean rootType) {
return writeToBuilder(bean, rootType).toString();
}
/**
* Writes the bean to the {@code StringBuilder}.
*
* The type of the bean will be set in the message.
*
* @param bean the bean to output, not null
* @return the builder, not null
*/
public StringBuilder writeToBuilder(final Bean bean) {
return writeToBuilder(bean, true);
}
/**
* Writes the bean to the {@code StringBuilder}.
*
* @param bean the bean to output, not null
* @param rootType true to output the root type
* @return the builder, not null
*/
public StringBuilder writeToBuilder(final Bean bean, final boolean rootType) {
if (bean == null) {
throw new NullPointerException("bean");
}
this.rootBean = bean;
this.basePackage = (rootType ? bean.getClass().getPackage().getName() + "." : null);
String type = rootBean.getClass().getName();
writeHeader();
builder.append('<').append(BEAN);
if (rootType) {
appendAttribute(builder, TYPE, type);
}
builder.append('>').append(settings.getNewLine());
writeBean(rootBean, settings.getIndent());
builder.append('<').append('/').append(BEAN).append('>').append(settings.getNewLine());
return builder;
}
private void writeHeader() {
builder.append("").append(settings.getNewLine());
}
//-----------------------------------------------------------------------
private boolean writeBean(final Bean bean, final String currentIndent) {
boolean output = false;
for (MetaProperty> prop : bean.metaBean().metaPropertyIterable()) {
if (prop.style().isSerializable()) {
output = true;
Object value = SerOptional.extractValue(prop, bean);
if (value != null) {
String propName = prop.name();
Class> propType = SerOptional.extractType(prop, bean.getClass());
if (value instanceof Bean) {
if (settings.getConverter().isConvertible(value.getClass())) {
writeSimple(currentIndent, propName, new StringBuilder(), propType, value);
} else {
writeBean(currentIndent, propName, new StringBuilder(), propType, (Bean) value);
}
} else {
SerIterator itemIterator = settings.getIteratorFactory().create(value, prop, bean.getClass());
if (itemIterator != null) {
writeElements(currentIndent, propName, new StringBuilder(), itemIterator);
} else {
writeSimple(currentIndent, propName, new StringBuilder(), propType, value);
}
}
}
}
}
return output;
}
//-----------------------------------------------------------------------
private void writeBean(final String currentIndent, final String tagName, final StringBuilder attrs, final Class> propType, final Bean value) {
if (value == null) {
throw new IllegalArgumentException("Bean cannot be null");
}
builder.append(currentIndent).append('<').append(tagName).append(attrs);
if (value.getClass() != propType) {
String typeStr = SerTypeMapper.encodeType(value.getClass(), settings, basePackage, knownTypes);
appendAttribute(builder, TYPE, typeStr);
}
int insertPoint = builder.length();
builder.append('>').append(settings.getNewLine());
if (writeBean(value, currentIndent + settings.getIndent())) {
builder.append(currentIndent).append('<').append('/').append(tagName).append('>').append(settings.getNewLine());
} else {
builder.insert(insertPoint, '/');
}
}
//-----------------------------------------------------------------------
private void writeElements(final String currentIndent, final String tagName, final StringBuilder attrs, final SerIterator itemIterator) {
if (itemIterator.metaTypeRequired()) {
appendAttribute(attrs, METATYPE, itemIterator.metaTypeName());
}
if (itemIterator.category() == SerCategory.GRID) {
appendAttribute(attrs, ROWS, Integer.toString(itemIterator.dimensionSize(0)));
appendAttribute(attrs, COLS, Integer.toString(itemIterator.dimensionSize(1)));
}
if (itemIterator.size() == 0) {
builder.append(currentIndent).append('<').append(tagName).append(attrs).append('/').append('>').append(settings.getNewLine());
} else {
builder.append(currentIndent).append('<').append(tagName).append(attrs).append('>').append(settings.getNewLine());
writeElements(currentIndent + settings.getIndent(), itemIterator);
builder.append(currentIndent).append('<').append('/').append(tagName).append('>').append(settings.getNewLine());
}
}
private void writeElements(final String currentIndent, final SerIterator itemIterator) {
// find converter once for performance, and before checking if key is bean
StringConverter