org.jibx.custom.classes.GlobalCustom Maven / Gradle / Ivy
/*
* Copyright (c) 2007-2009, Dennis M. Sosnoski. All rights reserved.
*
* Redistribution and use in source and binary forms, with or without modification, are permitted provided that the
* following conditions are met:
*
* Redistributions of source code must retain the above copyright notice, this list of conditions and the following
* disclaimer. Redistributions in binary form must reproduce the above copyright notice, this list of conditions and the
* following disclaimer in the documentation and/or other materials provided with the distribution. Neither the name of
* JiBX nor the names of its contributors may be used to endorse or promote products derived from this software without
* specific prior written permission.
*
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES,
* INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
* DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
* SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR
* SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY,
* WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
* OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
*/
package org.jibx.custom.classes;
import java.util.ArrayList;
import java.util.Collections;
import java.util.HashMap;
import java.util.Iterator;
import java.util.List;
import java.util.Map;
import org.jibx.binding.classes.ClassCache;
import org.jibx.runtime.EnumSet;
import org.jibx.runtime.IUnmarshaller;
import org.jibx.runtime.IUnmarshallingContext;
import org.jibx.runtime.JiBXException;
import org.jibx.runtime.impl.UnmarshallingContext;
import org.jibx.util.IClass;
import org.jibx.util.IClassLocator;
import org.jibx.util.StringArray;
/**
* Global customization information. This includes some options specific to the <binding> element of the definition,
* as well as controls for structuring of the generated binding(s). It handles the binding customization child elements
* directly, by invoking the abstract unmarshallers for the child elements to process the content. It also allows for
* extension elements which are not part of the binding customization structure, as long as the binding in use defines
* the unmarshalling for these elements.
*
* @author Dennis M. Sosnoski
*/
public class GlobalCustom extends NestingBase
{
/** Enumeration of allowed attribute names */
public static final StringArray s_allowedAttributes =
new StringArray(new String[] { "add-constructors", "direction", "force-classes", "track-source" },
NestingBase.s_allowedAttributes);
/** Element name in XML customization file. */
public static final String ELEMENT_NAME = "custom";
//
// Value set information
public static final int IN_BINDING = 0;
public static final int OUT_BINDING = 1;
public static final int BOTH_BINDING = 2;
/* package */static final EnumSet s_directionEnum =
new EnumSet(IN_BINDING, new String[] { "input", "output", "both" });
//
// Instance data
// structure for package hierarchy
private Map m_packageMap;
// class locator
private final IClassLocator m_classLocator;
// extension elements
private List m_extensionChildren;
// values applied directly to element (or to structure)
private boolean m_addConstructors;
private boolean m_forceClasses;
private boolean m_trackSource;
private boolean m_namespaceModular;
private boolean m_isInput;
private boolean m_isOutput;
private ArrayList m_unmarshalledClasses;
/**
* Constructor with class locator supplied.
*
* @param loc
*/
public GlobalCustom(IClassLocator loc) {
super(null);
m_packageMap = new HashMap();
m_classLocator = loc;
PackageCustom dflt = new PackageCustom("", "", this);
m_packageMap.put("", dflt);
m_isInput = true;
m_isOutput = true;
m_unmarshalledClasses = new ArrayList();
}
/**
* Constructor. This always creates the default package as the only direct child, since other packages will be
* treated as children of the default package.
*/
public GlobalCustom() {
this(new ClassCache.ClassCacheLocator());
}
/**
* Make sure all attributes are defined.
*
* @param uctx unmarshalling context
*/
private void preSet(IUnmarshallingContext uctx) {
validateAttributes(uctx, s_allowedAttributes);
}
/**
* Get global customizations root.
*
* @return global customization
*/
public GlobalCustom getGlobal() {
return this;
}
/**
* Get list of unmarshalled classes. This list is populated by the custom unmarshalling code as the customizations
* document is unmarshalled.
*
* @return list
*/
public ArrayList getUnmarshalledClasses() {
return m_unmarshalledClasses;
}
//
// Access methods for values applied only at top level
/**
* Get 'add-constructors' setting.
*
* @return 'add-constructors' value
*/
public boolean isAddConstructors() {
return m_addConstructors;
}
/**
* Set 'add-constructors' value.
*
* @param add 'add-constructors' value
*/
public void setAddConstructors(boolean add) {
m_addConstructors = add;
}
/**
* Get 'force-classes' setting.
*
* @return 'force-classes' value
*/
public boolean isForceClasses() {
return m_forceClasses;
}
/**
* Set 'force-classes' value.
*
* @param force 'force-classes' value
*/
public void setForceClasses(boolean force) {
m_forceClasses = force;
}
/**
* Get 'track-source' attribute value.
*
* @return 'track-source' value
*/
public boolean isTrackSource() {
return m_trackSource;
}
/**
* Set 'track-source' value.
*
* @param track 'track-source' value
*/
public void setTrackSource(boolean track) {
m_trackSource = track;
}
/**
* Check for an input binding.
*
* @return input flag
*/
public boolean isInput() {
return m_isInput;
}
/**
* Set input binding flag.
*
* @param input
*/
public void setInput(boolean input) {
m_isInput = input;
}
/**
* Check for an output binding.
*
* @return output flag
*/
public boolean isOutput() {
return m_isOutput;
}
/**
* Set output binding falg.
*
* @param output
*/
public void setOutput(boolean output) {
m_isOutput = output;
}
/**
* Get class locator.
*
* @return locator
*/
protected IClassLocator getClassLocator() {
return m_classLocator;
}
/**
* Get class information.
*
* @param type fully-qualified class name
* @return information, or null
if unable to load
*/
public IClass getClassInfo(String type) {
return m_classLocator.getClassInfo(type);
}
/**
* Get the extension elements used in this customization. This does not include the <package> or <class> child
* elements, which are added directly to the customization structures.
*
* @return child list
*/
public List getExtensionChildren() {
if (m_extensionChildren == null) {
return Collections.EMPTY_LIST;
} else {
return m_extensionChildren;
}
}
/**
* Internal method used during unmarshalling to add a child extension element.
*
* @param child
*/
protected void internalAddExtensionChild(Object child) {
if (m_extensionChildren == null) {
m_extensionChildren = new ArrayList();
}
m_extensionChildren.add(child);
}
/**
* Add a child extension element. This both adds the child to the list and invokes the extension element's
* {@link IApply#apply(IClassLocator)} method, if present.
*
* @param child
*/
public void addExtensionChild(Object child) {
internalAddExtensionChild(child);
if (child instanceof IApply) {
((IApply)child).apply(m_classLocator);
}
}
/**
* Check if a class is included in the customization information. This method does not alter the structures in any
* way, it only checks if the class customization information is part of the existing structure.
*
* @param type fully qualified class name
* @return true
if class includes, false
if not
*/
public boolean isClassUsed(String type) {
int split = type.lastIndexOf('.');
PackageCustom pack = (PackageCustom)m_packageMap.get(split < 0 ? "" : type.substring(0, split));
if (pack == null) {
return false;
} else {
return pack.getClassCustomization(type.substring(split + 1)) != null;
}
}
/**
* Get class customization information.
*
* @param type fully qualified class name
* @return class information (null
if not defined)
*/
public ClassCustom getClassCustomization(String type) {
int split = type.lastIndexOf('.');
PackageCustom pack = (PackageCustom)m_packageMap.get(split < 0 ? "" : type.substring(0, split));
if (pack == null) {
return null;
} else {
return pack.getClassCustomization(type.substring(split + 1));
}
}
/**
* Build new class customization information. This creates the customization information and adds it to the internal
* structures, initializing all values based on the settings inherited from <package> and <global> elements of
* the structure. This method should only be used after first calling {@link #getClassCustomization(String)} and
* obtaining a null
result.
*
* @param type fully qualified class name
* @return class information
*/
private ClassCustom buildClassCustomization(String type) {
int split = type.lastIndexOf('.');
PackageCustom pack = split < 0 ? getPackage("") : getPackage(type.substring(0, split));
ClassCustom clas = pack.addClassCustomization(type.substring(split + 1));
return clas;
}
/**
* Get class customization information, creating it if it doesn't already exist. This internal method supplies the
* class information in uninitialized form, so that data can be unmarshalled before initialization.
*
* @param type fully qualified class name
* @return class information
*/
private ClassCustom forceClassCustomization(String type) {
ClassCustom custom = getClassCustomization(type);
if (custom == null) {
custom = buildClassCustomization(type);
}
return custom;
}
/**
* Get initialized class customization information, creating it if it doesn't already exist.
*
* @param type fully qualified class name
* @return class information
*/
public ClassCustom addClassCustomization(String type) {
ClassCustom custom = getClassCustomization(type);
if (custom == null) {
custom = buildClassCustomization(type);
((PackageCustom)custom.getParent()).fixNamespace();
custom.apply(m_classLocator);
}
return custom;
}
/**
* Check if type represents a known mapping.
*
* @param type fully qualified class name
* @return known mapping flag
*/
public boolean isKnownMapping(String type) {
// TODO: add known mappings for org.w3c.dom.*, etc.
return false;
}
/**
* Direction set text method. This is intended for use during unmarshalling. TODO: add validation
*
* @param text
* @param ictx
*/
private void setDirectionText(String text, IUnmarshallingContext ictx) {
int direct = s_directionEnum.getValue(text);
if (direct < 0) {
throw new IllegalArgumentException("Value '" + text + "' not recognized for 'direction' attribute");
} else {
m_isInput = direct != OUT_BINDING;
m_isOutput = direct != IN_BINDING;
}
}
/**
* Direction get text method. This is intended for use during marshalling.
*
* @return text
*/
private String getDirectionText() {
if (m_isInput && m_isOutput) {
return s_directionEnum.getName(BOTH_BINDING);
} else if (m_isInput) {
return s_directionEnum.getName(IN_BINDING);
} else {
return s_directionEnum.getName(OUT_BINDING);
}
}
/**
* Initialize the global default namespace, along with special classes with built-in defaults. This needs to be done
* as a separate step before unmarshalling, so that the special classes are available for use.
*/
public void initClasses() {
// set default namespace
if (getNamespace() == null) {
setNamespace(getSpecifiedNamespace());
}
// create information for some special classes
ClassCustom custom = forceClassCustomization("java.util.List");
if (custom.getCreateType() == null && custom.getFactoryMethod() == null) {
custom.setCreateType("java.util.ArrayList");
}
custom = forceClassCustomization("java.util.Set");
if (custom.getCreateType() == null && custom.getFactoryMethod() == null) {
custom.setCreateType("java.util.HashSet");
}
custom = forceClassCustomization("java.util.Collection");
if (custom.getCreateType() == null && custom.getFactoryMethod() == null) {
custom.setCreateType("java.util.ArrayList");
}
}
/**
* Fills in class information based on inspection of the actual class data. This needs to be done as a separate step
* following unmarshalling, so that the full details of the unmarshalled customizations are available.
*/
public void fillClasses() {
// fill in details for all packages
for (Iterator iter = m_packageMap.values().iterator(); iter.hasNext();) {
PackageCustom pack = (PackageCustom)iter.next();
pack.fixNamespace();
pack.apply(m_classLocator);
}
// fill in details for any extension elements
if (m_extensionChildren != null) {
for (int i = 0; i < m_extensionChildren.size(); i++) {
Object child = m_extensionChildren.get(i);
if (child instanceof IApply) {
((IApply)child).apply(m_classLocator);
}
}
}
}
//
// Package structure support
/**
* Get package customizations. If the requested package is already defined the existing instance will be returned,
* otherwise a new instance will be created (along with any ancestor packages) and added to the structure.
*
* @param name
* @return package
*/
public PackageCustom getPackage(String name) {
// find existing package information
PackageCustom pack = (PackageCustom)m_packageMap.get(name);
if (pack == null) {
// create new package information
PackageCustom parent = null;
int split = name.lastIndexOf('.');
String simple = name;
String full = name;
if (split > 0) {
parent = getPackage(name.substring(0, split));
simple = name.substring(split + 1);
full = parent.getName() + '.' + simple;
} else if (name.length() > 0) {
parent = getPackage("");
}
pack = new PackageCustom(simple, full, parent);
m_packageMap.put(name, pack);
}
return pack;
}
/**
* Unmarshaller implementation for class. This handles the nested structure of packages and classes, using the
* abstract mappings defined by the binding to handle all the actual details.
*/
public static class Mapper implements IUnmarshaller
{
public boolean isPresent(IUnmarshallingContext ictx) throws JiBXException {
return true;
}
/**
* Build the fully-qualified name for a package or class by appending the supplied name attribute value to the
* fully-qualified name of the containing package.
*
* @param contain
* @param ctx
* @throws JiBXException
*/
private String buildFullName(PackageCustom contain, UnmarshallingContext ctx) throws JiBXException {
String lead = "";
if (contain != null) {
lead = contain.getName() + '.';
}
return lead + ctx.attributeText(null, "name");
}
/**
* Unmarshal package element. This calls itself recursively to handle nested package elements, and
* calls {@link #unmarshalClass(GlobalCustom, PackageCustom, UnmarshallingContext)} to handle nested
* class elements.
*
* @param global root customizations
* @param contain containing package
* @param ctx unmarshalling context
* @return unmarshalled package data
* @throws JiBXException
*/
private PackageCustom unmarshalPackage(GlobalCustom global, PackageCustom contain, UnmarshallingContext ctx)
throws JiBXException {
// create class instance and populate mapped values
PackageCustom inst = global.getPackage(buildFullName(contain, ctx));
ctx.getUnmarshaller("package").unmarshal(inst, ctx);
inst.fixNamespace();
// handle nested package and class information
while (ctx.isStart()) {
String element = ctx.getName();
if (PackageCustom.ELEMENT_NAME.equals(element)) {
unmarshalPackage(global, inst, ctx);
} else if (ClassCustom.ELEMENT_NAME.equals(element)) {
unmarshalClass(global, inst, ctx);
}
}
ctx.parsePastEndTag(null, PackageCustom.ELEMENT_NAME);
return inst;
}
/**
* Unmarshal class element. This calls itself recursively to handle nested class elements, and
* calls {@link #unmarshalClass(GlobalCustom, PackageCustom, UnmarshallingContext)} to handle nested
* class elements.
*
* @param global root customizations
* @param contain containing package
* @param ctx unmarshalling context
* @return unmarshalled class data
* @throws JiBXException
*/
private ClassCustom unmarshalClass(GlobalCustom global, PackageCustom contain, UnmarshallingContext ctx)
throws JiBXException {
// create class instance and populate mapped values
String name = buildFullName(contain, ctx);
int split = name.lastIndexOf('.');
PackageCustom pack = global.getPackage(split < 0 ? "" : name.substring(0, split));
pack.fixNamespace();
String simple = name.substring(split + 1);
ClassCustom inst = pack.getClassCustomization(simple);
if (inst == null) {
inst = pack.addClassCustomization(simple);
}
ctx.getUnmarshaller("class").unmarshal(inst, ctx);
ctx.parsePastEndTag(null, ClassCustom.ELEMENT_NAME);
// add to list unmarshalled
global.getUnmarshalledClasses().add(inst);
return inst;
}
/**
* Unmarshal root element of customizations. This expects to handle the actual root element of the binding
* directly, meaning it should always be invoked by using the
* {@link org.jibx.runtime.IUnmarshallable#unmarshal(IUnmarshallingContext)} method. The actual root element may
* be anything, allowing the unmarshaller to be used for subclasses (with different names) of the outer class.
*
* @param obj root element object (must be an instance of the GlobalCustom type)
* @param ictx unmarshalling context
* @return unmarshalled root object element
* @throws JiBXException
*/
public Object unmarshal(Object obj, IUnmarshallingContext ictx) throws JiBXException {
// initialize namespace and special classes
GlobalCustom global = (GlobalCustom)obj;
global.initClasses();
// unmarshal the wrapper element directly
UnmarshallingContext ctx = (UnmarshallingContext)ictx;
while (ctx.isStart()) {
// check type of child present
String element = ctx.getName();
if (PackageCustom.ELEMENT_NAME.equals(element)) {
unmarshalPackage(global, null, ctx);
} else if (ClassCustom.ELEMENT_NAME.equals(element)) {
unmarshalClass(global, null, ctx);
} else {
global.internalAddExtensionChild(ctx.unmarshalElement());
}
}
return global;
}
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy