blob: 7a1859dbf74bfdbcb2c1cf9a74acd791d38e01ae [file] [log] [blame]
/*
* Licensed to the Apache Software Foundation (ASF) under one
* or more contributor license agreements. See the NOTICE file
* distributed with this work for additional information
* regarding copyright ownership. The ASF licenses this file
* to you 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.apache.uima.resource.metadata.impl;
import java.beans.IntrospectionException;
import java.beans.Introspector;
import java.beans.PropertyDescriptor;
import java.io.File;
import java.io.IOException;
import java.io.OutputStream;
import java.io.Writer;
import java.lang.reflect.Array;
import java.lang.reflect.Constructor;
import java.lang.reflect.Method;
import java.net.MalformedURLException;
import java.net.URL;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.Iterator;
import java.util.List;
import java.util.Map;
import java.util.Set;
import org.apache.uima.UIMARuntimeException;
import org.apache.uima.UIMA_IllegalArgumentException;
import org.apache.uima.UIMA_UnsupportedOperationException;
import org.apache.uima.internal.util.XMLUtils;
import org.apache.uima.resource.metadata.MetaDataObject;
import org.apache.uima.util.InvalidXMLException;
import org.apache.uima.util.NameClassPair;
import org.apache.uima.util.XMLParser;
import org.apache.uima.util.XMLSerializer;
import org.apache.uima.util.XMLizable;
import org.w3c.dom.Element;
import org.w3c.dom.Node;
import org.w3c.dom.NodeList;
import org.xml.sax.Attributes;
import org.xml.sax.ContentHandler;
import org.xml.sax.SAXException;
import org.xml.sax.helpers.AttributesImpl;
/**
* Abstract base class for all MetaDataObjects in the reference implementation. Provides basic
* support for getting and setting property values given their names, by storing all attribute
* values in a HashMap keyed on attribute name.
* <p>
* Also provides the ability to write objects to XML and build objects from their DOM
* representation, as required to implement the {@link XMLizable} interface, which is a
* superinterface of {@link MetaDataObject}. In future versions, this could be replaced by a
* non-proprietary XML binding solution such as JAXB or EMF.
* <p>
*
* The implementation for getting and setting property values uses the JavaBeans introspection API.
* Therefore subclasses of this class must be valid JavaBeans and either use the standard naming
* conventions for getters and setters or else provide a BeanInfo class. See <a
* href="http://java.sun.com/docs/books/tutorial/javabeans/"> The Java Beans Tutorial</a> for more
* information.
*
*
*/
public abstract class MetaDataObject_impl implements MetaDataObject {
static final long serialVersionUID = 5876728533863334480L;
private static String PROP_NAME_SOURCE_URL = "sourceUrl";
private static final Attributes EMPTY_ATTRIBUTES = new AttributesImpl();
private transient PropertyDescriptor[] mPropertyDescriptors = null;
private transient URL mSourceUrl;
/**
* Creates a new <code>MetaDataObject_impl</code> with null attribute values
*/
public MetaDataObject_impl() {
}
/**
* Returns a list of <code>NameClassPair</code> objects indicating the attributes of this object
* and the Classes of the attributes' values. For primitive types, the wrapper classes will be
* returned (e.g. <code>java.lang.Integer</code> instead of int).
*
* @see org.apache.uima.resource.MetaDataObject#listAttributes()
*/
public List listAttributes() {
try {
PropertyDescriptor[] props = getPropertyDescriptors();
List resultList = new ArrayList(props.length);
for (int i = 0; i < props.length; i++) {
// only list properties with read and write methods,
// and don't include the SourceUrl property, which is for
// internal bookkeeping and shouldn't affect object equality
if (props[i].getReadMethod() != null && props[i].getWriteMethod() != null
&& !props[i].getName().equals(PROP_NAME_SOURCE_URL)) {
String propName = props[i].getName();
Class propClass = props[i].getPropertyType();
// translate primitive types (int, boolean, etc.) to wrapper classes
if (propClass.isPrimitive()) {
propClass = getWrapperClass(propClass);
}
resultList.add(new NameClassPair(propName, propClass.getName()));
}
}
return resultList;
} catch (IntrospectionException e) {
throw new UIMARuntimeException(e);
}
}
/**
* @see org.apache.uima.resource.MetaDataObject#getAttributeValue(String)
*/
public Object getAttributeValue(String aName) {
try {
PropertyDescriptor[] props = getPropertyDescriptors();
for (int i = 0; i < props.length; i++) {
if (props[i].getName().equals(aName)) {
Method reader = props[i].getReadMethod();
if (reader != null) {
return reader.invoke(this, new Object[0]);
}
}
}
return null;
} catch (Exception e) {
throw new UIMARuntimeException(e);
}
}
/**
* Gets the Class of the given attribue's value. For primitive types, the wrapper classes will be
* returned (e.g. <code>java.lang.Integer</code> instead of int).
*
* @param aName
* name of an attribute
*
* @return Class of value that may be assigned to the named attribute. Returns <code>null</code>
* if there is no attribute with the given name.
*/
public Class getAttributeClass(String aName) {
try {
List attrList = listAttributes();
Iterator it = attrList.iterator();
while (it.hasNext()) {
NameClassPair ncp = (NameClassPair) it.next();
if (ncp.getName().equals(aName)) {
return Class.forName(ncp.getClassName());
}
}
return null;
} catch (Exception e) {
throw new UIMARuntimeException(e);
}
}
/**
* Returns whether this object is modifiable. MetaDataObjects are modifiable by default.
*
* @see org.apache.uima.resource.MetaDataObject#isModifiable()
*/
public boolean isModifiable() {
return true;
}
/**
* @see org.apache.uima.resource.MetaDataObject#setAttributeValue(String, Object)
*/
public void setAttributeValue(String aName, Object aValue) {
try {
PropertyDescriptor[] props = getPropertyDescriptors();
for (int i = 0; i < props.length; i++) {
if (props[i].getName().equals(aName)) {
Method writer = props[i].getWriteMethod();
if (writer != null) {
try {
writer.invoke(this, new Object[] { aValue });
} catch (IllegalArgumentException e) {
throw new UIMA_IllegalArgumentException(
UIMA_IllegalArgumentException.METADATA_ATTRIBUTE_TYPE_MISMATCH, new Object[] {
aValue, aName }, e);
}
} else {
throw new UIMA_UnsupportedOperationException(
UIMA_UnsupportedOperationException.NOT_MODIFIABLE, new Object[] { aName,
this.getClass().getName() });
}
}
}
} catch (UIMA_IllegalArgumentException e) {
throw e;
} catch (Exception e) {
throw new UIMARuntimeException(e);
}
}
/**
* Gets the relative path base used to resolve imports. This is equal to the sourceUrl of this
* object, if known (i.e. if the object was parsed from an XML file or if setSourceUrl was
* explicitly called). If the source URL is not known, the value of the user.dir System property
* is returned.
*
* @return the base URL for resolving relative paths in this object
*/
public URL getRelativePathBase() {
if (mSourceUrl != null) {
return mSourceUrl;
} else {
try {
return new File(System.getProperty("user.dir")).toURL();
} catch (MalformedURLException e) {
try {
return new URL("file:/");
} catch (MalformedURLException e1) {
// assert false;
return null;
}
}
}
}
/**
* Gets the URL from which this object was parsed. When this object is parsed from an XML file,
* this is set by the parser to the URL of the source file XML file. If the object has been
* created by some other method, the source URL will not be known, and this method will return
* null.
* <p>
* This setting is used to resolve imports and is also included in exception messages to indicate
* the source of the problem.
*
* @return the source URL from which this object was parsed
*/
public URL getSourceUrl() {
return mSourceUrl;
}
/**
* If the sourceURL of this object is non-null, returns its string representation. If it is null,
* returns "&lt;unknown>". Useful for error messages.
*
* @return the source URL as a string, or "&lt;unknown>"
*/
public String getSourceUrlString() {
return mSourceUrl != null ? mSourceUrl.toString() : "<unknown>";
}
/**
* Sets the source URL of this object, only if that URL is currently set to null. This is used
* internally to update null relative base paths before doing import resolution, without
* overriding user-specified settings.
*
* @param aUrl
* the location of the XML file from which this object was parsed
*/
public void setSourceUrlIfNull(URL aUrl) {
if (mSourceUrl == null) {
setSourceUrl(aUrl);
}
}
/**
* Sets the URL from which this object was parsed. Typically only the XML parser sets this. This
* recursively sets the source URL of all descendants of this object.
*
* @param aUrl
* the location of the XML file from which this object was parsed
*/
public void setSourceUrl(URL aUrl) {
mSourceUrl = aUrl;
// set recursively on subobjects
List attrs = listAttributes();
Iterator i = attrs.iterator();
while (i.hasNext()) {
String attrName = ((NameClassPair) i.next()).getName();
Object val = getAttributeValue(attrName);
if (val instanceof MetaDataObject_impl) {
((MetaDataObject_impl) val).setSourceUrl(aUrl);
} else if (val != null && val.getClass().isArray()) {
int len = Array.getLength(val);
for (int j = 0; j < len; j++) {
Object arrayElem = Array.get(val, j);
if (arrayElem instanceof MetaDataObject_impl) {
((MetaDataObject_impl) arrayElem).setSourceUrl(aUrl);
}
}
} else if (val instanceof Map) {
Iterator valIter = ((Map) val).values().iterator();
while (valIter.hasNext()) {
Object mapVal = valIter.next();
if (mapVal instanceof MetaDataObject_impl) {
((MetaDataObject_impl) mapVal).setSourceUrl(aUrl);
}
}
}
}
}
/**
* @see org.apache.uima.resource.MetaDataObject#clone()
*/
public Object clone() {
// System.out.println("MetaDataObject_impl: clone");
MetaDataObject_impl clone = null;
try {
// do the default cloning
clone = (MetaDataObject_impl) super.clone();
} catch (CloneNotSupportedException e) {
// assert false : "All MetaDataObjects are Cloneable";
throw new UIMARuntimeException(e);
}
// now clone all values that are MetaDataObjects
List attrs = listAttributes();
Iterator i = attrs.iterator();
while (i.hasNext()) {
String attrName = ((NameClassPair) i.next()).getName();
Object val = getAttributeValue(attrName);
if (val instanceof MetaDataObject) {
Object clonedVal = ((MetaDataObject) val).clone();
clone.setAttributeValue(attrName, clonedVal);
} else if (val != null && val.getClass().isArray()) {
// clone the array, and clone any MetaDataObjects in the array
Class componentType = val.getClass().getComponentType();
int length = Array.getLength(val);
Object arrayClone = Array.newInstance(componentType, length);
for (int j = 0; j < length; j++) {
Object component = Array.get(val, j);
if (component instanceof MetaDataObject) {
component = ((MetaDataObject) component).clone();
}
Array.set(arrayClone, j, component);
}
clone.setAttributeValue(attrName, arrayClone);
}
}
return clone;
}
/**
* Dump this metadata object's attributes and values to a String. This is useful for debugging.
*/
public String toString() {
StringBuffer buf = new StringBuffer();
buf.append(getClass().getName()).append(": \n");
List attrList = listAttributes();
Iterator i = attrList.iterator();
while (i.hasNext()) {
NameClassPair ncp = (NameClassPair) i.next();
buf.append(ncp.getName() + " = ");
Object val = getAttributeValue(ncp.getName());
if (val == null)
buf.append("NULL");
else if (val instanceof Object[]) {
Object[] array = (Object[]) val;
buf.append("Array{");
for (int j = 0; j < array.length; j++) {
buf.append(j).append(": ").append(array[j].toString()).append("\n");
}
buf.append("}\n");
} else
buf.append(val.toString());
buf.append("\n");
}
return buf.toString();
}
/**
* Determines if this object is equal to another. Two MetaDataObjects are equivalent if they share
* the same attributes and the same values for those attributes.
*
* @param aObj
* object with which to compare this object
*
* @return true if and only if this object is equal to <code>aObj</code>
*/
public boolean equals(Object aObj) {
if (!(aObj instanceof MetaDataObject)) {
return false;
}
MetaDataObject mdo = (MetaDataObject) aObj;
// get the attributes (and classes) for the two objects
List theseAttrs = this.listAttributes();
List thoseAttrs = mdo.listAttributes();
// attribute lists must be same length
if (theseAttrs.size() != thoseAttrs.size()) {
return false;
}
// iterate through all attributes in this object
Iterator i = theseAttrs.iterator();
while (i.hasNext()) {
NameClassPair ncp = (NameClassPair) i.next();
// other object must contain this attribute
if (!thoseAttrs.contains(ncp)) {
return false;
}
// get values and test equivalency
Object val1 = this.getAttributeValue(ncp.getName());
Object val2 = mdo.getAttributeValue(ncp.getName());
if (!valuesEqual(val1, val2)) {
return false;
}
}
// if we get this far, objects are equal
return true;
}
/**
* Compare 2 values for equality.
* Reason val1.equals(val2) is not used:
* If val1 is of type Object[], the equal test is object identity equality, not
* element by element identity.
* So we use Arrays.equals or deepEquals instead.
*
* @param val1
* @param val2
* @return
*/
private boolean valuesEqual(Object val1, Object val2) {
if (val1 == null) {
return val2 == null;
}
if (val1.getClass().isArray()) {
if (val2.getClass() != val1.getClass()) {
return false;
}
// some of this may not be necessary - it depends what kind of array values are actually used
// The "if" statements below are in guessed order of frequency of occurance
if (val1 instanceof String[]) return Arrays.equals((String[])val1, (String[])val2);
// deepEquals handles arrays whose elements are arrays
if (val1 instanceof Object[]) return Arrays.deepEquals((Object[])val1, (Object[])val2);
if (val1 instanceof int[]) return Arrays.equals((int[])val1, (int[])val2);
if (val1 instanceof float[]) return Arrays.equals((float[])val1, (float[])val2);
if (val1 instanceof double[]) return Arrays.equals((double[])val1, (double[])val2);
if (val1 instanceof boolean[]) return Arrays.equals((boolean[])val1, (boolean[])val2);
if (val1 instanceof byte[]) return Arrays.equals((byte[])val1, (byte[])val2);
if (val1 instanceof short[]) return Arrays.equals((short[])val1, (short[])val2);
if (val1 instanceof long[]) return Arrays.equals((long[])val1, (long[])val2);
return Arrays.equals((char[])val1, (char[])val2);
}
if (val1 instanceof Map) {// only need this to handle Maps w/ array vals
if (!(val2 instanceof Map)) {
return false;
}
if (((Map)val1).size() != ((Map)val2).size()) {
return false;
}
if (val1.getClass() != val2.getClass()) {
return false;
}
Set entrySet1 = ((Map) val1).entrySet();
Iterator it = entrySet1.iterator();
while (it.hasNext()) {
Map.Entry entry = (Map.Entry) it.next();
Object subval1 = ((Map) val1).get(entry.getKey());
Object subval2 = ((Map) val2).get(entry.getKey());
if (!valuesEqual(subval1, subval2)) {
return false;
}
}
// for Map values, get here if all values in the map are equal
return true;
}
// not an instance of Map
return val1.equals(val2);
}
/**
* Gets the hash code for this object. The hash codes of two NameClassPairs <code>x</code> and
* </code>y</code> must be equal if <code>x.equals(y)</code> returns true;
*
* @return the hash code for this object
*/
public int hashCode() {
int hashCode = 0;
// add the hash codes of all attributes
List attrs = listAttributes();
Iterator i = attrs.iterator();
while (i.hasNext()) {
String attrName = ((NameClassPair) i.next()).getName();
Object val = getAttributeValue(attrName);
if (val != null) {
if (val instanceof Object[]) {
Object[] arr = (Object[]) val;
for (int j = 0; j < arr.length; j++) {
if (arr[j] != null) {
hashCode += arr[j].hashCode();
}
}
} else if (val instanceof Map) // only need to do this to handle Maps w/ array vals
{
Set entrySet = ((Map) val).entrySet();
Iterator it = entrySet.iterator();
while (it.hasNext()) {
Map.Entry entry = (Map.Entry) it.next();
hashCode += entry.getKey().hashCode();
Object subval = entry.getValue();
if (subval instanceof Object[]) {
Object[] arr = (Object[]) subval;
for (int j = 0; j < arr.length; j++) {
if (arr[j] != null) {
hashCode += arr[j].hashCode();
}
}
} else {
hashCode += subval.hashCode();
}
}
} else {
hashCode += val.hashCode();
}
}
}
return hashCode;
}
/**
* Writes out this object's XML representation.
*
* @param aWriter
* a Writer to which the XML string will be written
*/
public void toXML(Writer aWriter) throws SAXException, IOException {
XMLSerializer sax2xml = new XMLSerializer(aWriter);
ContentHandler contentHandler = sax2xml.getContentHandler();
contentHandler.startDocument();
toXML(sax2xml.getContentHandler(), true);
contentHandler.endDocument();
}
/**
* Writes out this object's XML representation.
*
* @param aOutputStream
* an OutputStream to which the XML string will be written
*/
public void toXML(OutputStream aOutputStream) throws SAXException, IOException {
XMLSerializer sax2xml = new XMLSerializer(aOutputStream);
ContentHandler contentHandler = sax2xml.getContentHandler();
contentHandler.startDocument();
toXML(sax2xml.getContentHandler(), true);
contentHandler.endDocument();
}
/**
* @see org.apache.uima.util.XMLizable#toXML(java.net.ContentHandler)
*/
public void toXML(ContentHandler aContentHandler) throws SAXException {
toXML(aContentHandler, false);
}
/**
* @see org.apache.uima.util.XMLizable#toXML(org.xml.sax.ContentHandler, boolean)
*/
public void toXML(ContentHandler aContentHandler, boolean aWriteDefaultNamespaceAttribute)
throws SAXException {
XmlizationInfo inf = getXmlizationInfo();
// write the element's start tag
// get attributes (can be provided by subclasses)
AttributesImpl attrs = getXMLAttributes();
// add default namespace attr if desired
if (aWriteDefaultNamespaceAttribute) {
if (inf.namespace != null) {
attrs.addAttribute("", "xmlns", "xmlns", null, inf.namespace);
}
}
// start element
aContentHandler.startElement(inf.namespace, inf.elementTagName, inf.elementTagName, attrs);
// write child elements
for (int i = 0; i < inf.propertyInfo.length; i++) {
PropertyXmlInfo propInf = inf.propertyInfo[i];
writePropertyAsElement(propInf, inf.namespace, aContentHandler);
}
// end element
aContentHandler.endElement(inf.namespace, inf.elementTagName, inf.elementTagName);
}
/**
* Called by the {@link toXML(Writer,String)} method to get the XML attributes that will be
* written as part of the element's tag. By default this method returns an empty Attributes
* object. Subclasses may override it in order to write attributes to the XML.
*
* @return an object defining the attributes to be written to the XML
*/
protected AttributesImpl getXMLAttributes() {
return new AttributesImpl();
}
/**
* To be implemented by subclasses to return information describing how to represent this object
* in XML.
*
* @return information defining this object's XML representation
*/
protected abstract XmlizationInfo getXmlizationInfo();
/**
* Looks in this class's XmlizationInfo for a property with the given XML element name.
*
* @param aXmlElementName
* the unqualified name of an XML element
*
* @return information on the property that corresponds to the given element name,
* <code>null</code> if none.
*/
protected PropertyXmlInfo getPropertyXmlInfo(String aXmlElementName) {
PropertyXmlInfo[] inf = getXmlizationInfo().propertyInfo;
for (int i = 0; i < inf.length; i++) {
if (aXmlElementName.equals(inf[i].xmlElementName))
return inf[i];
}
return null;
}
/**
* Utility method used to write a property out as an XML element.
*
* @param aPropInfo
* information on how to represent the property in XML
* @param aNamespace
* XML namespace URI for this object
* @param aContentHandler
* content handler to which this object will send events that describe its XML
* representation
*/
protected void writePropertyAsElement(PropertyXmlInfo aPropInfo, String aNamespace,
ContentHandler aContentHandler) throws SAXException {
// get value of property
Object val = getAttributeValue(aPropInfo.propertyName);
// if null or empty array, check to see if we're supposed to omit it
if (aPropInfo.omitIfNull
&& (val == null || (val.getClass().isArray() && ((Object[]) val).length == 0)))
return;
// if XML element name was supplied, write a tag
if (aPropInfo.xmlElementName != null) {
aContentHandler.startElement(aNamespace, aPropInfo.xmlElementName, aPropInfo.xmlElementName,
EMPTY_ATTRIBUTES);
}
// get class of property
Class propClass = getAttributeClass(aPropInfo.propertyName);
// if value is null then write nothing
if (val != null) {
// if value is an array then we have to treat that specially
if (val.getClass().isArray()) {
writeArrayPropertyAsElement(aPropInfo.propertyName, propClass, val,
aPropInfo.arrayElementTagName, aNamespace, aContentHandler);
} else {
// if value is an XMLizable object, call its toXML method
if (val instanceof XMLizable) {
((XMLizable) val).toXML(aContentHandler);
}
// else, if property's class is java.lang.Object, attempt to write
// it as a primitive
else if (propClass == Object.class) {
XMLUtils.writePrimitiveValue(val, aContentHandler);
} else {
// assume attribute's class is known (e.g. String, Integer), so it
// is not necessary to write the class name to the XML. Just write
// the string representation of the object
// XMLUtils.writeNormalizedString(val.toString(), aWriter, true);
String valStr = val.toString();
aContentHandler.characters(valStr.toCharArray(), 0, valStr.length());
}
}
}
// if XML element name was supplied, end the element that we started
if (aPropInfo.xmlElementName != null) {
aContentHandler.endElement(aNamespace, aPropInfo.xmlElementName, aPropInfo.xmlElementName);
}
}
/**
* Utility method used to write an array property out as an XML element.
*
* @param aPropName
* name of the attribute
* @param aPropClass
* class of the attribute
* @param aValue
* value (guaranteed to be an array and non-null)
* @param aArrayElementTagName
* name of tag to be assigned to each element of the array. May be <code>null</code>,
* in which case each element will be assigned a value appropriate to its class.
* @param aNamespace
* the XML namespace URI for this object
* @param aContentHandler
* the ContentHandler to which this object will write events that describe its XML
* representation
*/
protected void writeArrayPropertyAsElement(String aPropName, Class aPropClass, Object aValue,
String aArrayElementTagName, String aNamespace, ContentHandler aContentHandler)
throws SAXException {
// if aPropClass is generic Object, reader won't know whether to expect
// an array, so we tell it be writing an "array" element here.
if (aPropClass == Object.class) {
aContentHandler.startElement(aNamespace, "array", "array", EMPTY_ATTRIBUTES);
}
// iterate through elements of the array (at this point we don't allow
// nested arrays here
int len = ((Object[]) aValue).length;
for (int i = 0; i < len; i++) {
Object curElem = Array.get(aValue, i);
// if a particular array element tag has been specified, write it
if (aArrayElementTagName != null) {
aContentHandler.startElement(aNamespace, aArrayElementTagName, aArrayElementTagName,
EMPTY_ATTRIBUTES);
}
// if attribute's value is an XMLizable object, call its toXML method
if (curElem instanceof XMLizable) {
((XMLizable) curElem).toXML(aContentHandler);
}
// else, attempt to write it as a primitive
else {
if (aArrayElementTagName == null) {
// need to include the type, e.g. <string>
XMLUtils.writePrimitiveValue(curElem, aContentHandler);
} else {
// don't include the type - just write the value
String valStr = curElem.toString();
aContentHandler.characters(valStr.toCharArray(), 0, valStr.length());
}
}
// if we started an element, end it
if (aArrayElementTagName != null) {
aContentHandler.endElement(aNamespace, aArrayElementTagName, aArrayElementTagName);
}
}
// if we started an "Array" element, end it
if (aPropClass == Object.class) {
aContentHandler.endElement(aNamespace, "array", "array");
}
}
/**
* Utility method for writing to XML an property whose value is a <code>Map</code> with
* <code>String</code> keys and <code>XMLizable</code> values.
*
* @param aPropName
* name of the property to write to XML
* @param aXmlElementName
* name of the XML element for the property, <code>null</code> if none
* @param aKeyXmlAttributeName
* name of the XML attribute for the key
* @param aValueTagName
* XML element tag name to use for each entry in the Map
* @param aOmitIfNull
* if true, null or empty map will not be written at all, if false, null or empty map
* will be written as an empty element
* @param aNamespace
* namespace for this object
* @param aContentHandler
* ContentHandler to which this object will send events describing its XML representation
*/
protected void writeMapPropertyToXml(String aPropName, String aXmlElementName,
String aKeyXmlAttribute, String aValueTagName, boolean aOmitIfNull, String aNamespace,
ContentHandler aContentHandler) throws SAXException {
// get map
Map theMap = (Map) getAttributeValue(aPropName);
// if map is empty handle appropriately
if (theMap == null || theMap.isEmpty()) {
if (!aOmitIfNull && aXmlElementName != null) {
aContentHandler
.startElement(aNamespace, aXmlElementName, aXmlElementName, EMPTY_ATTRIBUTES);
aContentHandler.endElement(aNamespace, aXmlElementName, aXmlElementName);
}
} else {
// write start tag for attribute if desired
if (aXmlElementName != null) {
aContentHandler
.startElement(aNamespace, aXmlElementName, aXmlElementName, EMPTY_ATTRIBUTES);
}
// iterate over entries in the Map
Set entries = theMap.entrySet();
Iterator i = entries.iterator();
while (i.hasNext()) {
Map.Entry curEntry = (Map.Entry) i.next();
String key = (String) curEntry.getKey();
// write a tag for the value, with a "key" attribute
AttributesImpl attrs = new AttributesImpl();
attrs.addAttribute("", aKeyXmlAttribute, aKeyXmlAttribute, null, key); // are these nulls
// OK?
aContentHandler.startElement(aNamespace, aValueTagName, aValueTagName, attrs);
// write the value (must be XMLizable or an array of XMLizable)
Object val = curEntry.getValue();
if (val.getClass().isArray()) {
Object[] arr = (Object[]) val;
for (int j = 0; j < arr.length; j++) {
XMLizable elem = (XMLizable) arr[j];
elem.toXML(aContentHandler);
}
} else {
((XMLizable) val).toXML(aContentHandler);
}
// write end tag for the value
aContentHandler.endElement(aNamespace, aValueTagName, aValueTagName);
}
// if we wrote start tag for attribute, now write end tag
if (aXmlElementName != null) {
aContentHandler.endElement(aNamespace, aXmlElementName, aXmlElementName);
}
}
}
/**
* Initializes this object from its XML DOM representation. This method is typically called from
* the {@link XMLParser}.
*
* @param aElement
* the XML element that represents this object.
* @param aParser
* a reference to the UIMA <code>XMLParser</code>. The
* {@link XMLParser#buildObject(Element)} method can be used to construct sub-objects.
*
* @throws InvalidXMLException
* if the input XML element does not specify a valid object
*/
public final void buildFromXMLElement(Element aElement, XMLParser aParser)
throws InvalidXMLException {
buildFromXMLElement(aElement, aParser, new XMLParser.ParsingOptions(true));
}
/**
* Initializes this object from its XML DOM representation. This method is typically called from
* the {@link XMLParser}.
*
* @param aElement
* the XML element that represents this object.
* @param aParser
* a reference to the UIMA <code>XMLParser</code>. The
* {@link XMLParser#buildObject(Element)} method can be used to construct sub-objects.
* @param aOptions
* option settings
*
* @throws InvalidXMLException
* if the input XML element does not specify a valid object
*/
public void buildFromXMLElement(Element aElement, XMLParser aParser,
XMLParser.ParsingOptions aOptions) throws InvalidXMLException {
// check element type
if (!aElement.getTagName().equals(getXmlizationInfo().elementTagName))
throw new InvalidXMLException(InvalidXMLException.INVALID_ELEMENT_TYPE, new Object[] {
getXmlizationInfo().elementTagName, aElement.getTagName() });
// get child elements, each of which represents a property
List foundProperties = new ArrayList();
NodeList childNodes = aElement.getChildNodes();
for (int i = 0; i < childNodes.getLength(); i++) {
Node curNode = childNodes.item(i);
if (curNode instanceof Element) {
Element curElem = (Element) curNode;
String elemName = curElem.getTagName();
// look up this name in this class's XmlizationInfo
PropertyXmlInfo propXmlInfo = getPropertyXmlInfo(elemName);
if (propXmlInfo != null) {
// read the property's value from this element
readPropertyValueFromXMLElement(propXmlInfo, curElem, aParser, aOptions);
foundProperties.add(elemName);
} else {
// There is no property that matches the XML element.
// We have a couple of special cases to check before concluding that
// the XML is invalid.
// (1) if this class has only one property, we do not require
// specifying the property name as an XML element.
if (getXmlizationInfo().propertyInfo.length == 1) {
readPropertyValueFromXMLElement(getXmlizationInfo().propertyInfo[0], aElement, aParser,
aOptions);
} else {
// (2) if an object can be constructed from the unknown element,
// attempt to assign this object to any null-valued property
// that can accept it.
readUnknownPropertyValueFromXMLElement(curElem, aParser, aOptions, foundProperties);
}
}
}
}
}
/**
* Utility method to read an attribute's value from its DOM representation.
*
* @param aPropXmlInfo
* information about the property to read
* @param aElement
* DOM element to read from
* @param aParser
* parser to use to construct complex values
* @param aOptions
* option settings
*/
protected void readPropertyValueFromXMLElement(PropertyXmlInfo aPropXmlInfo, Element aElement,
XMLParser aParser, XMLParser.ParsingOptions aOptions) throws InvalidXMLException {
String propName = aPropXmlInfo.propertyName;
// get class of the property
Class propClass = getAttributeClass(propName);
// if we expect an array, treat that specially
if (propClass.isArray()) {
readArrayPropertyValueFromXMLElement(aPropXmlInfo, propClass, aElement, aParser, aOptions);
} else if (propClass == String.class) // special processing to handle env vars
{
String text = XMLUtils.getText(aElement, aOptions.expandEnvVarRefs);
setAttributeValue(propName, text);
} else {
// attempt to get the first child element, which represents the object
// that is the value of this attribute (there may not be one for a
// primitive type)
Element objElem = XMLUtils.getFirstChildElement(aElement);
if (objElem != null) {
// is it an array? (This can happen if aAttrClass is generic Object)
if (objElem.getTagName().equals("array")) {
readArrayPropertyValueFromXMLElement(aPropXmlInfo, Object.class, objElem, aParser,
aOptions);
} else {
setAttributeValue(propName, aParser.buildObjectOrPrimitive(objElem, aOptions));
}
} else {
// get text of the element and check for null
String text = XMLUtils.getText(aElement, aOptions.expandEnvVarRefs);
if (!text.equals("")) {
try {
// not null - attempt to construct an appropriate object from this String
// class must have a constructor that takes a String parameter
Constructor constructor = propClass.getConstructor(new Class[] { String.class });
// construct the object
Object val = constructor.newInstance(new Object[] { text });
setAttributeValue(propName, val);
} catch (Exception e) {
throw new InvalidXMLException(InvalidXMLException.UNKNOWN_ELEMENT,
new Object[] { aElement.getTagName() }, e);
}
}
}
}
}
/**
* Utility method to read an array property's value from its DOM representation.
*
* @param aPropXmlInfo
* information about the property to read
* @param aPropClass
* class of the property's value
* @param aElement
* DOM element representing the entire array
* @param aParser
* parser to use to construct complex values
* @param aOptions
* option settings
*/
protected void readArrayPropertyValueFromXMLElement(PropertyXmlInfo aPropXmlInfo,
Class aPropClass, Element aElement, XMLParser aParser, XMLParser.ParsingOptions aOptions)
throws InvalidXMLException {
Constructor primitiveElementStringConstructor = null; // may be used to build objects from
// strings
// get all child nodes (note not all may be elements)
NodeList elems = aElement.getChildNodes();
int numChildren = elems.getLength();
// iterate through children, and for each element construct a value,
// adding it to a list
List valueList = new ArrayList();
for (int i = 0; i < numChildren; i++) {
Node curNode = elems.item(i);
if (curNode instanceof Element) {
Element curElem = (Element) curNode;
// does the PropertyXmlInfo specify the expected tag name?
if (aPropXmlInfo.arrayElementTagName != null) {
if (curElem.getTagName().equals(aPropXmlInfo.arrayElementTagName)) {
// get text of element
String elemText = XMLUtils.getText(curElem, aOptions.expandEnvVarRefs);
// create appropriate primitive value
try {
// must have a constructor that takes a String parameter
if (primitiveElementStringConstructor == null) {
primitiveElementStringConstructor = aPropClass.getComponentType().getConstructor(
new Class[] { String.class });
}
// construct the object and add to list
valueList.add(primitiveElementStringConstructor
.newInstance(new Object[] { elemText }));
} catch (Exception e) {
throw new InvalidXMLException(e);
}
} else
// element type does not match
throw new InvalidXMLException(InvalidXMLException.INVALID_ELEMENT_TYPE, new Object[] {
aPropXmlInfo.arrayElementTagName, curElem.getTagName() });
} else {
// array element type is not specified, try defaults
valueList.add(aParser.buildObjectOrPrimitive(curElem, aOptions));
}
}
}
// initialize an appropriate array of the same length as the valueList,
// and copy the values
Class componentType = Object.class;
if (!(aPropClass == Object.class)) {
componentType = aPropClass.getComponentType();
// verify that objects are of appropriate type
Iterator i = valueList.iterator();
while (i.hasNext()) {
Object curObj = i.next();
if (!componentType.isAssignableFrom(curObj.getClass())) {
throw new InvalidXMLException(InvalidXMLException.INVALID_CLASS, new Object[] {
componentType, curObj.getClass() });
}
}
} else {
// attribute class is generic object, so we don't know what type of
// array to create. Try to infer it from the values - if they are all
// of the same type, use that, else use Object[].
Iterator i = valueList.iterator();
while (i.hasNext()) {
Object curObj = i.next();
if (componentType == Object.class) {
componentType = curObj.getClass();
} else if (componentType != curObj.getClass()) {
componentType = Object.class;
break;
}
}
}
Object array = Array.newInstance(componentType, valueList.size());
valueList.toArray((Object[]) array);
// assign this array as the value of the attribute
setAttributeValue(aPropXmlInfo.propertyName, array);
}
/**
* Utility method that attempts to read a property value from an XML element even though it is not
* known to which property the value should be assigned. If an object can be constructed from the
* XML element, it will be assigned to any unasigned property that can accept it.
*
* @param aElement
* DOM element to read from
* @param aParser
* parser to use to construct complex values
* @param aKnownPropertyNames
* List of propertiees that we've already values for (these values will not be
* overwritten)
*
* @throws InvalidXMLException
* if no acceptable object is described by aElement
*/
protected void readUnknownPropertyValueFromXMLElement(Element aElement, XMLParser aParser,
XMLParser.ParsingOptions aOptions, List aKnownPropertyNames) throws InvalidXMLException {
boolean success = false;
try {
Object valueObj = aParser.buildObjectOrPrimitive(aElement, aOptions);
if (valueObj != null) {
// find a property that can accept this type, which we have not already
// read a value for
PropertyXmlInfo[] props = getXmlizationInfo().propertyInfo;
for (int i = 0; i < props.length; i++) {
String propName = props[i].propertyName;
Class propClass = getAttributeClass(propName);
if (propClass.isAssignableFrom(valueObj.getClass())) {
// check if we have already read a value for this attribute
if (!aKnownPropertyNames.contains(propName)) {
// set value
setAttributeValue(propName, valueObj);
success = true;
break;
}
} else if (propClass.isArray()
&& propClass.getComponentType().isAssignableFrom(valueObj.getClass())) {
// it's an array - get current value and append
Object curVal = getAttributeValue(propName);
int curLen = curVal == null ? 0 : Array.getLength(curVal);
Object newVal = Array.newInstance(propClass.getComponentType(), curLen + 1);
if (curLen > 0) {
System.arraycopy(curVal, 0, newVal, 0, curLen);
}
Array.set(newVal, curLen, valueObj);
// set value
setAttributeValue(propName, newVal);
success = true;
break;
}
}
}
} catch (Exception e) {
if (e instanceof InvalidXMLException) {
throw (InvalidXMLException) e;
} else {
throw new InvalidXMLException(e);
}
}
// throw exception if we did not succeed
if (!success) {
throw new InvalidXMLException(InvalidXMLException.UNKNOWN_ELEMENT, new Object[] { aElement
.getTagName() });
}
}
/**
* Utility method for reading from XML an attribute whose value is a <code>Map</code> with
* <code>String</code> keys and <code>XMLizable</code> values.
*
* @param aPropName
* name of the property to read from XML
* @param aElement
* element to read from
* @param aKeyXmlAttribute
* XML attribute for the key
* @param aValueTagName
* XML element tag name for each entry in the map
* @param aParser
* parser to use to build sub-objects
* @param aOptions
* parsing option settings
* @param aValueIsArray
* true if the value of the map entires is an array. This method only supports
* homogeneous arrays.
*/
protected void readMapPropertyFromXml(String aPropName, Element aElement,
String aKeyXmlAttribute, String aValueTagName, XMLParser aParser,
XMLParser.ParsingOptions aOptions, boolean aValueIsArray) throws InvalidXMLException {
// get the Map to which we add entries (it should already exist)
Map theMap = (Map) getAttributeValue(aPropName);
// get all child nodes
NodeList childNodes = aElement.getChildNodes();
for (int i = 0; i < childNodes.getLength(); i++) {
Node curNode = childNodes.item(i);
if (curNode instanceof Element) {
Element curElem = (Element) curNode;
// check element tag name
if (!curElem.getTagName().equals(aValueTagName)) {
throw new InvalidXMLException(InvalidXMLException.INVALID_ELEMENT_TYPE, new Object[] {
aValueTagName, curElem.getTagName() });
}
// get the key attribute
String key = curElem.getAttribute(aKeyXmlAttribute);
if (key.equals("")) {
throw new InvalidXMLException(InvalidXMLException.REQUIRED_ATTRIBUTE_MISSING,
new Object[] { "key", aValueTagName });
}
// build value object
Object val = null;
if (!aValueIsArray) {
Element valElem = XMLUtils.getFirstChildElement(curElem);
if (valElem == null) {
throw new InvalidXMLException(InvalidXMLException.ELEMENT_NOT_FOUND, new Object[] {
"(any)", aValueTagName });
}
val = aParser.buildObject(valElem, aOptions);
} else // array
{
ArrayList vals = new ArrayList();
NodeList arrayNodes = curElem.getChildNodes();
for (int j = 0; j < arrayNodes.getLength(); j++) {
Node curArrayNode = arrayNodes.item(j);
if (curArrayNode instanceof Element) {
Element valElem = (Element) curArrayNode;
vals.add(aParser.buildObject(valElem));
}
}
if (!vals.isEmpty()) {
val = Array.newInstance(vals.get(0).getClass(), vals.size());
vals.toArray((Object[]) val);
}
}
// add object to the map
theMap.put(key, val);
}
}
}
/**
* Gets the wrapper class corresponding to the given primitive type. For example,
* <code>java.lang.Integer</code> is the wrapper class for the primitive type <code>int</code>.
*
* @param aPrimitiveType
* <code>Class</code> object representing a primitive type
*
* @return <code>Class</object> object representing the wrapper type for
* <code>PrimitiveType</code>. If <code>aPrimitiveType</code> is not
* a primitive type, it is itself returned.
*/
protected Class getWrapperClass(Class aPrimitiveType) {
if (Integer.TYPE.equals(aPrimitiveType))
return Integer.class;
else if (Short.TYPE.equals(aPrimitiveType))
return Short.class;
else if (Long.TYPE.equals(aPrimitiveType))
return Long.class;
else if (Byte.TYPE.equals(aPrimitiveType))
return Byte.class;
else if (Character.TYPE.equals(aPrimitiveType))
return Character.class;
else if (Float.TYPE.equals(aPrimitiveType))
return Float.class;
else if (Double.TYPE.equals(aPrimitiveType))
return Double.class;
else if (Boolean.TYPE.equals(aPrimitiveType))
return Boolean.class;
else
return aPrimitiveType;
}
/**
* Utility method that introspects this bean and returns a list of <code>PropertyDescriptor</code>s
* for its properties.
* <p>
* The JavaBeans introspector is used, with the IGNORE_ALL_BEANINFO flag. This saves on
* initialization time by preventing the introspector from searching for nonexistent BeanInfo
* classes for all the MetaDataObjects.
*
* @return the <code>PropertyDescriptors</code> for all properties introduced by subclasses of
* <code>MetaDataObject_impl</code>.
*
* @throw IntrospectionException if introspection fails
*/
protected PropertyDescriptor[] getPropertyDescriptors() throws IntrospectionException {
if (mPropertyDescriptors == null) {
mPropertyDescriptors = Introspector.getBeanInfo(this.getClass(),
Introspector.IGNORE_ALL_BEANINFO).getPropertyDescriptors();
}
return mPropertyDescriptors;
}
}