/*
* Copyright (C) 2012 Jason Gedge <http://www.gedge.ca>
*
* This file is part of the OpGraph project.
*
* This program is free software: you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation, either version 3 of the License, or
* (at your option) any later version.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with this program. If not, see <http://www.gnu.org/licenses/>.
*/
package ca.gedge.opgraph.io.xml;
import static ca.gedge.opgraph.io.xml.XMLSerializerFactory.*;
import java.io.IOException;
import javax.xml.XMLConstants;
import javax.xml.namespace.QName;
import org.w3c.dom.Document;
import org.w3c.dom.Element;
import org.w3c.dom.Node;
import org.w3c.dom.NodeList;
import ca.gedge.opgraph.ContextualItem;
import ca.gedge.opgraph.InputField;
import ca.gedge.opgraph.OpGraph;
import ca.gedge.opgraph.OutputField;
import ca.gedge.opgraph.extensions.Extendable;
/**
* A default serializer for reading/writing {@link InputField}s and
* {@link OutputField}s to/from XML.
*/
public class DefaultFieldXMLSerializer implements XMLSerializer {
// qualified names
static final QName INPUT_QNAME = new QName(DEFAULT_NAMESPACE, "input", XMLConstants.DEFAULT_NS_PREFIX);
static final QName OUTPUT_QNAME = new QName(DEFAULT_NAMESPACE, "output", XMLConstants.DEFAULT_NS_PREFIX);
@Override
public void write(XMLSerializerFactory serializerFactory, Document doc, Element parentElem, Object obj)
throws IOException
{
if(obj == null)
throw new IOException("Null object given to serializer");
if(obj instanceof InputField) {
final InputField field = (InputField)obj;
// Only write if field is non-fixed, or fixed but with extensions
if(!field.isFixed() || field.getExtensionClasses().size() > 0) {
final Element fieldElem = doc.createElementNS(INPUT_QNAME.getNamespaceURI(), INPUT_QNAME.getLocalPart());
fieldElem.setAttribute("key", field.getKey());
fieldElem.setAttribute("optional", Boolean.toString(field.isOptional()));
fieldElem.setTextContent(field.getDescription());
// XXX Store type validators?
// Extensions
if(field.getExtensionClasses().size() > 0) {
final XMLSerializer serializer = serializerFactory.getHandler(Extendable.class);
if(serializer == null)
throw new IOException("No XML serializer for extensions");
serializer.write(serializerFactory, doc, fieldElem, field);
}
parentElem.appendChild(fieldElem);
}
} else if(obj instanceof OutputField) {
final OutputField field = (OutputField)obj;
// Only write if field is non-fixed, or fixed but with extensions
if(!field.isFixed() || field.getExtensionClasses().size() > 0) {
final Element fieldElem = doc.createElementNS(OUTPUT_QNAME.getNamespaceURI(), OUTPUT_QNAME.getLocalPart());
fieldElem.setAttribute("key", field.getKey());
fieldElem.setAttribute("type", field.getOutputType().getName());
fieldElem.setTextContent(field.getDescription());
// Extensions
if(field.getExtensionClasses().size() > 0) {
final XMLSerializer serializer = serializerFactory.getHandler(Extendable.class);
if(serializer == null)
throw new IOException("No XML serializer for extensions");
serializer.write(serializerFactory, doc, fieldElem, field);
}
parentElem.appendChild(fieldElem);
}
} else {
throw new IOException(DefaultFieldXMLSerializer.class.getName() + " cannot write objects of type " + obj.getClass().getName());
}
}
@Override
public Object read(XMLSerializerFactory serializerFactory, OpGraph graph, Object parent, Document doc, Element elem)
throws IOException
{
ContextualItem item = null;
if(INPUT_QNAME.equals(XMLSerializerFactory.getQName(elem))) {
// Create
final String key = elem.getAttribute( "key");
final String description = elem.getTextContent();
final InputField field = new InputField(key, description);
if(elem.hasAttribute("optional"))
field.setOptional(Boolean.parseBoolean(elem.getAttributeNS(DEFAULT_NAMESPACE, "optional")));
// XXX Read type validator?
// Read children
final NodeList children = elem.getChildNodes();
for(int childIndex = 0; childIndex < children.getLength(); ++childIndex) {
final Node node = children.item(childIndex);
if(node instanceof Element) {
final Element childElem = (Element)node;
final QName name = XMLSerializerFactory.getQName(childElem);
// Get a handler for the element
final XMLSerializer serializer = serializerFactory.getHandler(name);
if(serializer == null)
throw new IOException("Could not get handler for element: " + name);
}
}
item = field;
} else if(OUTPUT_QNAME.equals(XMLSerializerFactory.getQName(elem))) {
// Create
Class<?> outputType = Object.class;
if(elem.hasAttribute("type")) {
final String outputTypeClassName = elem.getAttribute("type");
try {
outputType = Class.forName(outputTypeClassName);
} catch(ClassNotFoundException exc) {
throw new IOException("Unknown output type for field: " + outputTypeClassName);
}
}
final String key = elem.getAttribute("key");
final String description = elem.getTextContent();
final OutputField field = new OutputField(key, description, false, outputType);
// Read children
final NodeList children = elem.getChildNodes();
for(int childIndex = 0; childIndex < children.getLength(); ++childIndex) {
final Node node = children.item(childIndex);
if(node instanceof Element) {
final Element childElem = (Element)node;
final QName name = XMLSerializerFactory.getQName(childElem);
// Get a handler for the element
final XMLSerializer serializer = serializerFactory.getHandler(name);
if(serializer == null)
throw new IOException("Could not get handler for element: " + name);
}
}
item = field;
}
return item;
}
@Override
public boolean handles(Class<?> cls) {
return ((cls == InputField.class) || (cls == OutputField.class));
}
@Override
public boolean handles(QName name) {
return (INPUT_QNAME.equals(name) || OUTPUT_QNAME.equals(name));
}
}