org.jibx.schema.codegen.StructureClassHolder Maven / Gradle / Ivy
/*
* Copyright (c) 2006-2010, 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.schema.codegen;
import java.util.ArrayList;
import java.util.HashMap;
import java.util.Map;
import org.apache.log4j.Logger;
import org.eclipse.jdt.core.dom.Modifier;
import org.eclipse.jdt.core.dom.SingleVariableDeclaration;
import org.eclipse.jdt.core.dom.Type;
import org.eclipse.jdt.core.dom.InfixExpression.Operator;
import org.jibx.binding.model.BindingHolder;
import org.jibx.binding.model.BuiltinFormats;
import org.jibx.binding.model.CollectionElement;
import org.jibx.binding.model.ContainerElementBase;
import org.jibx.binding.model.ElementBase;
import org.jibx.binding.model.FormatElement;
import org.jibx.binding.model.MappingElement;
import org.jibx.binding.model.MappingElementBase;
import org.jibx.binding.model.NestingAttributes;
import org.jibx.binding.model.PropertyAttributes;
import org.jibx.binding.model.StructureElement;
import org.jibx.binding.model.StructureElementBase;
import org.jibx.binding.model.ValueElement;
import org.jibx.runtime.QName;
import org.jibx.runtime.Utility;
import org.jibx.schema.SchemaUtils;
import org.jibx.schema.codegen.custom.NestingCustomBase;
import org.jibx.schema.codegen.custom.SchemaRootBase;
import org.jibx.schema.codegen.extend.ClassDecorator;
import org.jibx.schema.codegen.extend.NameConverter;
import org.jibx.schema.elements.AnnotatedBase;
import org.jibx.schema.elements.AttributeElement;
import org.jibx.schema.elements.CommonCompositorBase;
import org.jibx.schema.elements.ComplexTypeElement;
import org.jibx.schema.elements.ElementElement;
import org.jibx.schema.elements.SchemaBase;
import org.jibx.util.Types;
import org.jibx.util.UniqueNameSet;
/**
* Information for a data class to be included in code generated from schema.
*
* @author Dennis M. Sosnoski
*/
public class StructureClassHolder extends ClassHolder
{
private static final String LIST_DESERIALIZE_PREFIX = "deserialize";
private static final String LIST_SERIALIZE_PREFIX = "serialize";
/** Logger for class. */
private static final Logger s_logger = Logger.getLogger(StructureClassHolder.class.getName());
/** Default format definitions map. */
private static final Map s_formatMap;
static {
s_formatMap = new HashMap();
for (int i = 0; i < BuiltinFormats.s_builtinFormats.length; i++) {
FormatElement format = BuiltinFormats.s_builtinFormats[i];
s_formatMap.put(format.getTypeName(), format);
}
}
/** Flag for collection present in class. */
private boolean m_collectionPresent;
/** Root node for data structure of class. */
private ParentNode m_dataRoot;
/** Binding definition element for this class. */
private ContainerElementBase m_bindingElement;
/** Selection property names used in class (lazy create, null
if none). */
protected UniqueNameSet m_selectSet;
/**
* Constructor.
*
* @param name class name
* @param base base class name
* @param pack package information
* @param holder binding holder
* @param nconv name converter
* @param decorators class decorators
* @param inner use inner classes for substructures
*/
public StructureClassHolder(String name, String base, PackageHolder pack, BindingHolder holder, NameConverter nconv,
ClassDecorator[] decorators, boolean inner) {
super(name, base, pack, holder, nconv, decorators, inner, false);
}
/**
* Constructor for creating a child inner class definition.
*
* @param name class name
* @param context parent class
*/
private StructureClassHolder(String name, StructureClassHolder context) {
super(name, context, false);
}
/**
* Derive group names from the containing group prefix and the simple name of the group.
*
* @param group
* @param container (null
if none)
* @return name
*/
// static String deriveGroupName(GroupItem group, Group container) {
// String prefix = null;
// if (container != null) {
// prefix = group.getClassName();
// String prior = container.getPrefix();
// if (prior == null) {
// prefix = NameConverter.toNameLead(prefix);
// } else {
// prefix = prior + NameConverter.toNameWord(prefix);
// }
// prefix = container.uniqueChildPrefix(prefix);
// }
// return prefix;
// }
/**
* Populate a class data representation tree based on a supplied item tree. The mapping between the two trees is not
* one-to-one since item groupings may be ignored where irrelevant.
*
* @param struct root item in tree
* @param supertext schema documentation passed in for item tree
* @param parent containing data node
* @param bindhold associated binding definition holder
*/
private void addToTree(GroupItem struct, String supertext, ParentNode parent, BindingHolder bindhold) {
if (struct.getChildCount() > 1) {
supertext = null;
}
for (Item item = struct.getFirstChild(); item != null; item = item.getNext()) {
// first check for ignored group
if (item.isIgnored()) {
new ParentNode((GroupItem)item, parent);
} else {
// check for documentation available on schema component
String doctext = supertext;
if (item.isTopmost()) {
String curtext = extractDocumentation(item.getSchemaComponent());
if (curtext != null) {
doctext = curtext;
}
}
if (item.isCollection()) {
m_collectionPresent = true;
}
if (item instanceof GroupItem) {
GroupItem group = (GroupItem)item;
if (group.isInline()) {
if (group.getChildCount() > 0) {
// create a new group for an inlined compositor only if it's or or nested
ParentNode into = parent;
AnnotatedBase comp = item.getSchemaComponent();
if (comp instanceof CommonCompositorBase) {
if (comp.type() == SchemaBase.ALL_TYPE || comp.type() == SchemaBase.CHOICE_TYPE
|| comp.getParent() instanceof CommonCompositorBase) {
into = new ParentNode(group, parent);
into.setDocumentation(doctext);
doctext = null;
}
} else {
// create a new group for a non-compositor only if
// different schema component
if (struct.getSchemaComponent() != comp
&& group.getFirstChild().getSchemaComponent() != comp) {
into = new ParentNode(group, parent);
into.setDocumentation(doctext);
doctext = null;
}
}
addToTree(group, doctext, into, bindhold);
into.adjustName();
} else {
// just create parent node with no children matching group for optional empty grouping (so
// that it can be generated using a presence flag)
new ParentNode(group, parent);
if (s_logger.isDebugEnabled()) {
s_logger.debug("Added empty parent for " +
SchemaUtils.describeComponent(group.getSchemaComponent()));
}
}
} else {
// create a new class and populate that
ClassHolder child;
String text = group.getEffectiveClassName();
if (m_useInnerClasses) {
if (m_nameSet.contains(text)) {
StructureClassHolder outer = this;
while (outer != null) {
if (outer.getName().equals(text)) {
text += "Inner";
break;
} else {
outer = (StructureClassHolder)outer.m_outerClass;
}
}
}
text = m_nameSet.add(text);
child = group.isEnumeration() ? new EnumerationClassHolder(text, this)
: (ClassHolder)new StructureClassHolder(text, this);
m_inners.add(child);
if (s_logger.isDebugEnabled()) {
s_logger.debug("Added inner class " + child.getFullName());
}
} else {
String fullname = m_baseName + text;
child = m_package.addClass(fullname, m_baseName, m_nameConverter, m_decorators,
group.isEnumeration(), m_holder);
m_importsTracker.addImport(child.getFullName(), true);
text = child.getName();
if (group.isEnumeration()) {
FormatElement format = new FormatElement();
format.setTypeName(child.getBindingName());
((EnumerationClassHolder)child).setBinding(format);
bindhold.addFormat(format);
}
if (s_logger.isDebugEnabled()) {
s_logger.debug("Added derived class " + child.getFullName());
}
}
if (!group.isFixedClassName()) {
group.setClassName(text);
}
group.setGenerateClass(child);
DataNode value = new LeafNode(group, parent);
value.setDocumentation(doctext);
if (!group.isEnumeration()) {
group.convertTypeReference();
importValueType(value);
}
child.buildDataStructure(group, bindhold);
}
} else {
DataNode value = new LeafNode(item, parent);
value.setDocumentation(doctext);
importValueType(value);
}
}
}
}
/**
* Convert an item structure to a class representation. This may include creating subsidiary classes (either as
* inner classes, or as separate standalone classes), where necessary.
*
* @param group item group
* @param bindhold associated binding definition holder
*/
public void buildDataStructure(GroupItem group, BindingHolder bindhold) {
super.buildDataStructure(group, bindhold);
if (group.isEnumeration()) {
throw new IllegalArgumentException("Internal error - group is an enumeration");
} else {
// populate the actual definition structure
m_dataRoot = new ParentNode(group, null);
m_dataRoot.setDocumentation(extractDocumentation(group.getSchemaComponent()));
addToTree(group, null, m_dataRoot, bindhold);
// import the list type if needed
int reptype = getSchemaCustom().getRepeatType();
if ((reptype == SchemaRootBase.REPEAT_LIST || reptype == SchemaRootBase.REPEAT_TYPED) &&
m_collectionPresent) {
m_importsTracker.addImport(COLLECTION_VARIABLE_TYPE, false);
m_importsTracker.addImport(m_listImplClass, false);
}
}
}
/**
* Set the binding component linked to this class.
*
* @param container binding definition element (<mapping> or <structure>)
*/
public void setBinding(ContainerElementBase container) {
m_bindingElement = container;
}
/**
* Recursively add all inner enumeration classes as formats to a <mapping> definition. This is used to create the
* <format> elements for all nested enumerations, which need to be direct children of the <mapping> element
* for the top-level class.
*
* @param mapping
*/
private void addInnerFormats(MappingElementBase mapping) {
for (int i = 0; i < m_inners.size(); i++) {
ClassHolder inner = (ClassHolder)m_inners.get(i);
if (inner.isSimpleValue()) {
FormatElement format = new FormatElement();
format.setTypeName(inner.getBindingName());
((EnumerationClassHolder)inner).setBinding(format);
mapping.addTopChild(format);
} else {
((StructureClassHolder)inner).addInnerFormats(mapping);
}
}
}
/**
* Add all fixed names in a group to the set of names defined for this class. This calls itself recursively to
* handle nested groups.
*
* @param wrapper
*/
private void addFixedNames(ParentNode wrapper) {
ArrayList values = wrapper.getChildren();
for (int i = 0; i < values.size(); i++) {
DataNode value = (DataNode)values.get(i);
Item item = value.getItem();
boolean addname = item.isFixedName();
if (value instanceof ParentNode) {
ParentNode childgrp = (ParentNode)value;
addFixedNames(childgrp);
addname = addname && (childgrp.isSelectorNeeded() || wrapper.isSelectorNeeded());
}
if (addname) {
String name = item.getEffectiveName();
if (!m_nameSet.add(name).equals(name)) {
// TODO: pass in the validation context, create an error
throw new IllegalStateException("Name '" + name + "' cannot be used twice in same context");
}
}
}
}
/**
* Convert base name to collection name. If using a java.util.List
representation the name is converted
* to singular form and "List" is appended; if using an array representation the name is converted to plural form.
*
* @param base
* @param item
*/
private void setCollectionName(String base, Item item) {
String name;
if (getSchemaCustom().getRepeatType() == SchemaRootBase.REPEAT_ARRAY) {
name = m_nameConverter.pluralize(base);
if (!name.equals(base)) {
s_logger.debug("Converted name " + base + " to " + name);
}
} else {
String singular = m_nameConverter.depluralize(base);
if (!singular.equals(base)) {
s_logger.debug("Converted name " + base + " to " + singular);
}
name = singular + "List";
}
item.setName(m_nameSet.add(name));
}
/**
* Handle value name assignments for a group within this class. This calls itself recursively to handle nested
* groups.
*
* @param parent
* @param innamed flag for parent group name already fixed
*/
private void fixFlexibleNames(ParentNode parent, boolean innamed) {
// check for group which uses a selector (choice or union)
String suffix = null;
ArrayList nodes = parent.getChildren();
if (parent.isSelectorNeeded()) {
// add the actual variable name used to record current state
Item item = parent.getItem();
item.setName(m_nameSet.add(m_nameConverter.toBaseName(item.getEffectiveName()) + "Select"));
// create value name set if first time used
if (m_selectSet == null) {
m_selectSet = new UniqueNameSet();
}
// generate constant for each child value
if (parent.getSchemaComponent().type() == SchemaBase.UNION_TYPE) {
suffix = "_Form";
} else {
suffix = "_Choice";
}
}
// handle name conversions and recording
for (int i = 0; i < nodes.size(); i++) {
DataNode node = (DataNode)nodes.get(i);
Item item = node.getItem();
String name = null;
if (node instanceof ParentNode) {
// use recursive call to set child group names (adopting name for group if same as first child, for
// group inside choice, in order to avoid adding the same name twice for non-conflicting usages); also
// adopt the child name if this is a collection, or if the first child is an implicit value, or if the
// group item is unnamed and there's only one child
ArrayList childvals = ((ParentNode)node).getChildren();
if (childvals.size() > 0) {
if (!item.isFixedName() && innamed) {
item.setName(parent.getItem().getName());
}
boolean passname = false;
if (childvals.size() == 1) {
// want to set the name at this level and pass it down when this is an element or attribute and
// there's only one child component all the way down to a leaf
passname = item.isFixedName() || innamed;
int type = item.getSchemaComponent().type();
if (!passname && (type == SchemaBase.ATTRIBUTE_TYPE ||
type == SchemaBase.ELEMENT_TYPE)) {
// check for single value all the way down
ArrayList childs = childvals;
boolean collect = node.isCollection();
while (childs.size() == 1) {
DataNode child = (DataNode)childs.get(0);
if (child.isCollection()) {
collect = true;
}
if (child instanceof LeafNode) {
// reached leaf with single path, so set name at current level and pass down
name = item.getEffectiveName();
if (collect) {
setCollectionName(name, item);
} else {
item.setName(m_nameSet.add(name));
}
passname = true;
break;
} else {
childs = ((ParentNode)child).getChildren();
}
}
}
}
fixFlexibleNames((ParentNode)node, passname);
}
} else if (item.isFixedName()) {
name = item.getName();
} else if (innamed) {
if (!item.isFixedName()) {
item.setName(parent.getItem().getName());
}
name = item.getName();
} else {
// convert and add the value name
name = item.getEffectiveName();
if (node.isCollection()) {
setCollectionName(name, item);
name = item.getName();
} else {
item.setName(m_nameSet.add(name));
}
}
// handle selection naming, if needed
if (parent.isSelectorNeeded()) {
if (name == null) {
name = item.getEffectiveName();
if (node.isCollection()) {
// pluralize name for collection, as will be done with the actual property name
// TODO: really need to use name structure and references, to avoid duplicated efforts like this
name = m_nameConverter.pluralize(name);
}
}
name = m_selectSet.add(NameUtils.toNameWord(name));
node.setSelectPropName(name);
node.setSelectConstName(m_nameConverter.toConstantName(name + suffix));
}
}
}
/**
* Generate the code to check and set the selection on any containing selector group. This should be used when
* setting any value, including inside selector methods (if used), since selector groups may be nested.
*
* @param value
* @param block
* @param builder
*/
private void generateSelectorSet(DataNode value, BlockBuilder block, ClassBuilder builder) {
ParentNode group;
while ((group = value.getParent()) != null) {
if (group.isSelectorNeeded()) {
int type = group.getSelectorType();
if (type == NestingCustomBase.SELECTION_CHECKEDSET || type == NestingCustomBase.SELECTION_CHECKEDBOTH) {
// when using select method call, just call that method (it will call containing group method, if
// any)
InvocationBuilder call = builder.createMemberMethodCall(group.getSelectSetMethod());
call.addVariableOperand(value.getSelectConstName());
block.addCall(call);
break;
} else {
// if setting directly, set this one and continue up to next containing group
block.addAssignVariableToField(value.getSelectConstName(), group.getSelectField());
}
}
value = group;
}
}
/**
* Generate the code to check the selection on any containing selector group. This should be used when getting any
* value, including inside selector methods (if used), since selector groups may be nested.
*
* @param value
* @param block
* @param builder
*/
private void generateSelectorCheck(DataNode value, BlockBuilder block, ClassBuilder builder) {
ParentNode group;
while ((group = value.getParent()) != null) {
if (group.isSelectorNeeded()) {
int type = group.getSelectorType();
if (type == NestingCustomBase.SELECTION_CHECKEDBOTH ||
type == NestingCustomBase.SELECTION_OVERRIDEBOTH) {
// when using select method call, just call that method (it will call containing group method, if
// any)
InvocationBuilder call = builder.createMemberMethodCall(group.getSelectCheckMethod());
call.addVariableOperand(value.getSelectConstName());
block.addCall(call);
break;
}
}
value = group;
}
}
/**
* Generate a test method for a value, if it's part of a group with a selector.
*
* @param node
* @param seldesc containing group description
* @param valdesc description of this value within group
* @param builder
*/
private void checkIfMethod(DataNode node, String seldesc, String valdesc, ClassBuilder builder) {
if (node.getParent().isSelectorNeeded()) {
MethodBuilder ifmeth = builder.addMethod("if" + node.getSelectPropName(), "boolean");
ifmeth.setPublic();
ifmeth.addSourceComment("Check if " + valdesc + " is current selection for " + seldesc + '.');
ifmeth.addSourceComment("");
ifmeth.addSourceComment("@return", " true
if selection, false
if not");
InfixExpressionBuilder testexpr = builder.buildNameOp(node.getParent().getSelectField(), Operator.EQUALS);
testexpr.addVariableOperand(node.getSelectConstName());
ifmeth.createBlock().addReturnExpression(testexpr);
}
}
/**
* Set the optional state of a structure or collection element in the binding. The name for the
* structure or collection must be set before calling this method, since the presence or absence of
* a name determines whether optional status is passed down from a parent.
*
* @param value node
* @param force optional state forced flag
* @param struct binding structure
*/
private void setStructureOptional(DataNode value, boolean force, StructureElementBase struct) {
boolean optional = value.isOptional();
if (!optional && struct.getName() == null) {
// no name for structure, so see if we can make the whole structure optional
Item item = value.getItem();
if (item instanceof GroupItem && item.isTopmost()) {
optional = ((GroupItem)item).isAllOptional();
} else if (getSchemaCustom().isStructureOptional() && item instanceof ReferenceItem) {
optional = ((ReferenceItem)item).getDefinition().isAllOptional();
}
}
if (optional) {
struct.setUsage(PropertyAttributes.OPTIONAL_USAGE);
}
}
/**
* Set the name and namespace URI for a <value> binding component. The value style must be set before making this
* call, since element and attribute elements are handled differently.
*
* @param qname qualified name to be set (null
if none)
* @param holder binding containing the value definition
* @param value binding component
*/
private static void setName(QName qname, BindingHolder holder, ValueElement value) {
if (qname != null) {
if (value.getName() != null) {
throw new IllegalStateException("Internal error - trying to overwrite name '" + value.getName() + "' with '" + qname.getName() + '\'');
}
value.setName(qname.getName());
String uri = qname.getUri();
int style = value.getStyle();
if (style == NestingAttributes.ATTRIBUTE_STYLE) {
if (uri != null) {
holder.addNamespaceUsage(uri);
}
value.setUri(uri);
} else if (style == NestingAttributes.ELEMENT_STYLE) {
holder.addNamespaceUsage(uri);
if (!Utility.safeEquals(uri, holder.getElementDefaultNamespace())) {
value.setUri(uri);
}
}
}
}
/**
* Set the name and namespace URI for a <structure> or <collection> binding component.
*
* @param qname qualified name to be set (null
if none)
* @param holder binding containing the structure or collection definition
* @param struct binding component
*/
private static void setName(QName qname, BindingHolder holder, StructureElementBase struct) {
if (qname != null) {
if (struct.getName() != null) {
throw new IllegalStateException("Internal error - trying to overwrite name '" + struct.getName() + "' with '" + qname.getName() + '\'');
}
String name = qname.getName();
String uri = qname.getUri();
if (name.charAt(0) == '{') {
int split = name.indexOf('}');
uri = name.substring(1, split);
name = name.substring(split+1);
}
struct.setName(name);
holder.addNamespaceUsage(uri);
if (!Utility.safeEquals(uri, holder.getElementDefaultNamespace())) {
struct.setUri(uri);
}
}
}
/**
* Build a <value> binding component for a field.
*
* @param node
* @param wrapname
* @param gname
* @param sname
* @param holder
* @return constructed binding component
*/
private ValueElement buildValueBinding(DataNode node, QName wrapname, String gname, String sname,
BindingHolder holder) {
// add element to binding structure for simple (primitive or text) value
ValueElement value = new ValueElement();
if (gname == null) {
value.setDeclaredType(node.getBindingType());
} else {
value.setGetName(gname);
value.setSetName(sname);
}
setValueHandlingOptions(node.getItem(), value, holder);
// set test method if needed to pick between alternatives
ParentNode wrapper = node.getParent();
if (wrapper.isSelectorNeeded()) {
value.setTestName("if" + node.getSelectPropName());
value.setUsage(PropertyAttributes.OPTIONAL_USAGE);
} else if (node.isOptional()) {
value.setUsage(PropertyAttributes.OPTIONAL_USAGE);
}
// get the schema component supplying an element or attribute name
AnnotatedBase comp = node.getSchemaComponent();
if (wrapname == null) {
// scan up structure to find parent linked to a different schema component
// TODO: find a better way to handle this in the node or item structures
ParentNode parent = wrapper;
while (parent != null && parent.getSchemaComponent() == comp) {
parent = parent.getParent();
}
// set name from node and style from component type
if (parent == null) {
// all parents linked to same component, treat as special case with name already set
value.setEffectiveStyle(ValueElement.TEXT_STYLE);
} else {
// name will have been passed down from containing element
// (since attributes handled directly)
if (comp.type() == SchemaBase.ELEMENT_TYPE) {
// value is an element, set the name directly
value.setEffectiveStyle(NestingAttributes.ELEMENT_STYLE);
ElementElement elem = (ElementElement)comp;
if (SchemaUtils.isOptionalElement(elem)) {
// TODO: this is needed because the optional status doesn't inherit downward for embedded items,
// as when a simpleType is nested inside an optional attribute. should the code be changed to
// inherit instead?
value.setUsage(PropertyAttributes.OPTIONAL_USAGE);
}
} else if (comp.type() == SchemaBase.ATTRIBUTE_TYPE) {
// value is an attribute, set the name directly
value.setEffectiveStyle(NestingAttributes.ATTRIBUTE_STYLE);
AttributeElement attr = (AttributeElement)comp;
if (SchemaUtils.isOptionalAttribute(attr)) {
// TODO: this is needed because the optional status doesn't inherit downward for embedded items, as
// when a simpleType is nested inside an optional attribute. should the code be changed to
// inherit instead?
value.setUsage(PropertyAttributes.OPTIONAL_USAGE);
}
} else {
value.setEffectiveStyle(ValueElement.TEXT_STYLE);
}
setName(node.getQName(), holder, value);
}
} else {
value.setEffectiveStyle(NestingAttributes.ELEMENT_STYLE);
setName(wrapname, holder, value);
}
return value;
}
/**
* Set the field and get/set access method names for a property.
*
* @param basename
* @param node
*/
private void setMemberNames(String basename, DataNode node) {
// define the field and get/set names
String propname = node.getPropName();
s_logger.debug("Adding property " + propname);
node.setFieldName(m_nameConverter.toFieldName(m_nameConverter.toBaseName(propname)));
String getpref = "boolean".equals(node.getType()) ? "is" : "get";
node.setGetMethodName(getpref + propname);
node.setSetMethodName("set" + propname);
}
/**
* Add a simple property to the class. This adds the actual field definition, along with the appropriate access
* methods.
*
* @param basename
* @param node
* @param builder
*/
private void addSimpleProperty(String basename, DataNode node, ClassBuilder builder) {
// set the field and method names
setMemberNames(basename, node);
// make sure the type is defined
String type = node.getType();
if (type == null) {
// type can be null in case of xs:any with discard handling, but otherwise invalid
if (node.isAny()) {
return;
} else {
throw new IllegalStateException("Internal error - no type for property");
}
}
if (s_logger.isDebugEnabled()) {
s_logger.debug("Found type " + type + " for property " + basename);
}
// generate the field as a simple value
String fname = node.getFieldName();
FieldBuilder field = builder.addField(fname, type);
field.setPrivate();
// add get method definition (unchecked, but result meaningless if not the selected group item)
MethodBuilder getmeth = builder.addMethod(node.getGetMethodName(), type);
getmeth.setPublic();
StringBuffer buff = new StringBuffer();
buff.append("Get the ");
String descript = describe(node);
buff.append(descript);
buff.append(" value.");
String document = findDocumentation(false, node);
if (document != null) {
buff.append(' ');
buff.append(document);
}
getmeth.addSourceComment(buff.toString());
getmeth.addSourceComment("");
getmeth.addSourceComment("@return", " value");
BlockBuilder block = getmeth.createBlock();
generateSelectorCheck(node, block, builder);
block.addReturnNamed(fname);
// add the set method definition
MethodBuilder setmeth = builder.addMethod(node.getSetMethodName(), "void");
setmeth.setPublic();
buff.replace(0, 3, "Set");
setmeth.addSourceComment(buff.toString());
setmeth.addSourceComment("");
String nonres = NameUtils.convertReserved(basename);
setmeth.addSourceComment("@param", " " + nonres);
setmeth.addParameter(nonres, type);
block = setmeth.createBlock();
generateSelectorSet(node, block, builder);
block.addAssignVariableToField(nonres, fname);
// call decorators for added value processing
for (int i = 0; i < m_decorators.length; i++) {
m_decorators[i].valueAdded(nonres, false, type, field.getDeclaration(), getmeth.getDeclaration(),
setmeth.getDeclaration(), descript, this);
}
}
/**
* Add a multiple-valued property to the class. This adds the actual field definition, along with the appropriate
* access methods.
*
* @param basename
* @param node
* @param builder
*/
private void addRepeatedProperty(String basename, DataNode node, ClassBuilder builder) {
// set up the member names
String propname = node.getPropName();
setMemberNames(basename, node);
// find the basic value type (which may require digging down to the leaf node, for a complex collection)
String basetype;
int collcount = node.isCollection() ? 1 : 0;
DataNode nested = node;
while ((basetype = nested.getType()) == null && nested.isInterior() &&
((ParentNode)nested).getChildren().size() > 0) {
nested = (DataNode)((ParentNode)nested).getChildren().get(0);
if (nested.isCollection()) {
collcount++;
}
}
if (basetype == null) {
// type can be null in case of xs:any with discard handling, but otherwise invalid
if (node.isAny()) {
return;
} else {
basetype = "java.lang.String";
}
}
if (s_logger.isDebugEnabled()) {
s_logger.debug("Found type " + basetype + " (" + collcount + " layers of collection nesting) for property "
+ propname);
}
// find the types to be used for field and actual instance
String type;
Type fieldtype;
Type insttype;
int reptype = getSchemaCustom().getRepeatType();
if (reptype == SchemaRootBase.REPEAT_TYPED) {
fieldtype = builder.createParameterizedType(COLLECTION_VARIABLE_TYPE, basetype);
insttype = builder.createType(basetype);
while (--collcount > 0) {
fieldtype = builder.createParameterizedType(COLLECTION_VARIABLE_TYPE, fieldtype);
insttype = builder.createParameterizedType(COLLECTION_VARIABLE_TYPE, insttype);
}
type = basetype;
insttype = builder.createParameterizedType(m_listImplClass, insttype);
} else if (reptype == SchemaRootBase.REPEAT_LIST || node.isAny()) {
fieldtype = builder.createType(COLLECTION_VARIABLE_TYPE);
insttype = builder.createType(m_listImplClass);
if (collcount > 1) {
type = COLLECTION_VARIABLE_TYPE;
} else {
type = basetype;
}
} else {
type = basetype;
while (--collcount > 0) {
type += "[]";
}
fieldtype = builder.createType(type + "[]");
insttype = null;
}
// get documentation and description text
String document = findDocumentation(true, node);
String descript = describe(node);
String term = getSchemaCustom().getRepeatType() == SchemaRootBase.REPEAT_ARRAY ? "array" : "list";
// generate the field as a collection
String fname = node.getFieldName();
FieldBuilder field = builder.addField(fname, fieldtype);
if (insttype != null) {
field.setInitializer(builder.newInstance(insttype));
}
field.setPrivate();
// add get method definition (unchecked, but result meaningless if not the selected group item)
MethodBuilder getmeth = builder.addMethod(node.getGetMethodName(), (Type)builder.clone(fieldtype));
getmeth.setPublic();
StringBuffer buff = new StringBuffer();
buff.append("Get the ");
buff.append(term);
buff.append(" of ");
buff.append(descript);
buff.append(" items.");
if (document != null) {
buff.append(' ');
buff.append(document);
}
getmeth.addSourceComment(buff.toString());
getmeth.addSourceComment("");
getmeth.addSourceComment("@return", " " + term);
getmeth.createBlock().addReturnNamed(fname);
// add the set method definition
MethodBuilder setmeth = builder.addMethod(node.getSetMethodName(), "void");
setmeth.setPublic();
buff.replace(0, 3, "Set");
setmeth.addSourceComment(buff.toString());
setmeth.addSourceComment("");
setmeth.addSourceComment("@param", " " + COLLECTION_VARIABLE_NAME);
setmeth.addParameter(COLLECTION_VARIABLE_NAME, (Type)builder.clone(fieldtype));
BlockBuilder block = setmeth.createBlock();
generateSelectorSet(node, block, builder);
block.addAssignVariableToField(COLLECTION_VARIABLE_NAME, fname);
// process list and collection differently for binding
Item item = node.getItem();
if (!node.isCollection()) {
// determine format conversion handling for type
String valsername = null;
String valdesername = null;
String valuename = null;
FormatElement format = (FormatElement)s_formatMap.get(type);
boolean passctx = false;
if (format != null) {
valsername = format.getSerializerName();
valdesername = format.getDeserializerName();
if (valsername == null && !"java.lang.String".equals(type)) {
valuename = "toString";
}
passctx = "org.jibx.runtime.QName".equals(type);
} else if (item instanceof ReferenceItem) {
DefinitionItem def = ((ReferenceItem)item).getDefinition();
if (def.isEnumeration()) {
EnumerationClassHolder genclas = (EnumerationClassHolder)def.getGenerateClass();
valsername = EnumerationClassHolder.CONVERTFORCE_METHOD;
valuename = genclas.getName() + ".toString";
}
} else {
throw new IllegalStateException("Internal error - invalid list type");
}
// add list serializer method to class
String sername = LIST_SERIALIZE_PREFIX + propname;
MethodBuilder sermeth = builder.addMethod(sername, "java.lang.String");
sermeth.addParameter("values", (Type)builder.clone(fieldtype));
if (passctx) {
SingleVariableDeclaration decl = sermeth.addParameter("ictx", "org.jibx.runtime.IMarshallingContext");
decl.modifiers().add(decl.getAST().newModifier(Modifier.ModifierKeyword.FINAL_KEYWORD));
sermeth.addThrows("org.jibx.runtime.JiBXException");
}
sermeth.setPublicStatic();
sermeth.addSourceComment("Serializer for " + descript + ' ' + term + '.');
sermeth.addSourceComment("");
sermeth.addSourceComment("@param", " values");
sermeth.addSourceComment("@return", " text");
// create a simple null return for null parameter string
BlockBuilder nullblock = builder.newBlock();
nullblock.addReturnNull();
// create block for actual serialization when parameter non-null
BlockBuilder serblock = builder.newBlock();
NewInstanceBuilder newbuff = builder.newInstance("java.lang.StringBuffer");
serblock.addLocalVariableDeclaration("java.lang.StringBuffer", "buff", newbuff);
// create body of loop to handle the conversion
BlockBuilder forblock = builder.newBlock();
// append space to buffer unless empty
InfixExpressionBuilder lengthexpr = builder.buildInfix(Operator.GREATER);
lengthexpr.addOperand(builder.createNormalMethodCall("buff", "length"));
lengthexpr.addNumberLiteralOperand("0");
InvocationBuilder appendcall = builder.createNormalMethodCall("buff", "append");
appendcall.addCharacterLiteralOperand(' ');
BlockBuilder spaceblock = builder.newBlock();
spaceblock.addExpressionStatement(appendcall);
forblock.addIfStatement(lengthexpr, spaceblock);
// load the current value from array
if (reptype == SchemaRootBase.REPEAT_TYPED) {
forblock.addLocalVariableDeclaration(type, "value", builder.createNormalMethodCall("iter", "next"));
} else if (reptype == SchemaRootBase.REPEAT_LIST) {
CastBuilder castexpr = builder.buildCast(type);
castexpr.addOperand(builder.createNormalMethodCall("iter", "next"));
forblock.addLocalVariableDeclaration(type, "value", castexpr);
} else {
forblock.addLocalVariableDeclaration(type, "value", builder.buildArrayIndexAccess("values", "index"));
}
// append the current value to the buffer
appendcall = builder.createNormalMethodCall("buff", "append");
if (valuename != null) {
appendcall.addOperand(builder.createNormalMethodCall("value", valuename));
} else if (valdesername != null) {
InvocationBuilder sercall = builder.createStaticMethodCall(valsername);
sercall.addVariableOperand("value");
if (passctx) {
sercall.addVariableOperand("ictx");
}
appendcall.addOperand(sercall);
} else {
appendcall.addVariableOperand("value");
}
forblock.addExpressionStatement(appendcall);
// build the for loop around the conversion
if (reptype == SchemaRootBase.REPEAT_TYPED) {
Type itertype = builder.createParameterizedType("java.util.Iterator", type);
serblock.addIteratedForStatement("iter", itertype,
builder.createNormalMethodCall("values", "iterator"), forblock);
} else if (reptype == SchemaRootBase.REPEAT_LIST) {
serblock.addIteratedForStatement("iter", builder.createType("java.util.Iterator"),
builder.createNormalMethodCall("values", "iterator"), forblock);
} else {
serblock.addIndexedForStatement("index", "values", forblock);
}
// finish non-null serialization block with buffer conversion
serblock.addReturnExpression(builder.createNormalMethodCall("buff", "toString"));
// finish with the if statement that decides which to execute
InfixExpressionBuilder iftest = builder.buildNameOp("values", Operator.EQUALS);
iftest.addNullOperand();
sermeth.createBlock().addIfElseStatement(iftest, nullblock, serblock);
// add list deserializer method to class
String desername = LIST_DESERIALIZE_PREFIX + propname;
MethodBuilder desermeth = builder.addMethod(desername, (Type)builder.clone(fieldtype));
desermeth.addParameter("text", "java.lang.String");
if (passctx) {
SingleVariableDeclaration decl = desermeth.addParameter("ictx",
"org.jibx.runtime.IUnmarshallingContext");
decl.modifiers().add(decl.getAST().newModifier(Modifier.ModifierKeyword.FINAL_KEYWORD));
}
desermeth.setPublicStatic();
desermeth.addSourceComment("Deserializer for " + descript + ' ' + term + '.');
desermeth.addSourceComment("");
desermeth.addSourceComment("@param", " text");
desermeth.addSourceComment("@return", " values");
desermeth.addSourceComment("@throws", " org.jibx.runtime.JiBXException on conversion error");
desermeth.addThrows("org.jibx.runtime.JiBXException");
block = desermeth.createBlock();
// build instance creation for anonymous inner class to handle deserialization
NewInstanceBuilder newinst = builder.newInstance("org.jibx.runtime.IListItemDeserializer");
ClassBuilder anonclas = newinst.addAnonymousInnerClass();
MethodBuilder innermeth = anonclas.addMethod("deserialize", "java.lang.Object");
if (passctx) {
innermeth.addThrows("org.jibx.runtime.JiBXException");
}
innermeth.addParameter("text", "java.lang.String");
innermeth.setPublic();
BlockBuilder innerblock = innermeth.createBlock();
if (valdesername == null) {
innerblock.addReturnNamed("text");
} else {
InvocationBuilder desercall = builder.createStaticMethodCall(valdesername);
desercall.addVariableOperand("text");
if (passctx) {
desercall.addVariableOperand("ictx");
}
innerblock.addReturnExpression(desercall);
}
block.addLocalVariableDeclaration("org.jibx.runtime.IListItemDeserializer", "ldser", newinst);
// build call using anonymous inner class to deserialize to untyped collection
InvocationBuilder desercall = builder.createStaticMethodCall("org.jibx.runtime.Utility.deserializeList");
desercall.addVariableOperand("text");
desercall.addVariableOperand("ldser");
// handle the return as appropriate
if (reptype == SchemaRootBase.REPEAT_TYPED) {
CastBuilder castexpr = builder.buildCast((Type)builder.clone(fieldtype));
castexpr.addOperand(desercall);
block.addReturnExpression(castexpr);
} else if (reptype == SchemaRootBase.REPEAT_LIST) {
block.addReturnExpression(desercall);
} else {
// save deserialization result list to local variable
block.addLocalVariableDeclaration("java.util.List", "list", desercall);
// create null return block
BlockBuilder ifnull = builder.newBlock();
ifnull.addReturnNull();
// create non-null return with conversion to array
BlockBuilder ifnonnull = builder.newBlock();
InvocationBuilder toarraycall = builder.createNormalMethodCall("list", "toArray");
NewArrayBuilder newarray = builder.newArrayBuilder(type);
newarray.setSize(builder.createNormalMethodCall("list", "size"));
toarraycall.addOperand(newarray);
CastBuilder castexpr = builder.buildCast((Type)builder.clone(fieldtype));
castexpr.addOperand(toarraycall);
ifnonnull.addReturnExpression(castexpr);
// finish with the if statement that decides which to execute
iftest = builder.buildNameOp("list", Operator.EQUALS);
iftest.addNullOperand();
block.addIfElseStatement(iftest, ifnull, ifnonnull);
}
}
// call decorators for added value processing
for (int i = 0; i < m_decorators.length; i++) {
m_decorators[i].valueAdded(basename, true, basetype, field.getDeclaration(), getmeth.getDeclaration(),
setmeth.getDeclaration(), descript, this);
}
}
/**
* Add a flag property to the class. This adds the actual field definition, along with the appropriate access
* methods.
*
* @param basename
* @param node
* @param builder
*/
private void addFlagProperty(String basename, DataNode node, ClassBuilder builder) {
// define the flag field and method names
String propname = node.getPropName();
s_logger.debug("Adding property " + propname);
String fname = m_nameConverter.toFieldName(basename);
node.setFieldName(fname);
node.setTestMethodName("if" + propname + "Present");
node.setFlagMethodName("flag" + propname + "Present");
String nonres = NameUtils.convertReserved(basename);
// generate the field as a simple boolean value
FieldBuilder field = builder.addField(fname, "boolean");
field.setPrivate();
// add test method definition
MethodBuilder testmeth = builder.addMethod(node.getTestMethodName(), "boolean");
testmeth.setPublic();
StringBuffer buff = new StringBuffer();
buff.append("Check if the ");
String descript = describe(node);
buff.append(descript);
buff.append(" is present.");
String document = findDocumentation(false, node);
if (document != null) {
buff.append(' ');
buff.append(document);
}
testmeth.addSourceComment(buff.toString());
testmeth.addSourceComment("");
testmeth.addSourceComment("@return", " true
if present, false
if not");
BlockBuilder block = testmeth.createBlock();
generateSelectorCheck(node, block, builder);
block.addReturnNamed(fname);
// add the flag method definition
MethodBuilder flagmeth = builder.addMethod(node.getFlagMethodName(), "void");
flagmeth.setPublic();
buff.setLength(0);
buff.append("Set flag for ");
buff.append(descript);
buff.append(" present.");
if (document != null) {
buff.append(' ');
buff.append(document);
}
flagmeth.addSourceComment(buff.toString());
flagmeth.addSourceComment("");
flagmeth.addSourceComment("@param", " " + nonres);
flagmeth.addParameter(nonres, "boolean");
block = flagmeth.createBlock();
generateSelectorSet(node, block, builder);
block.addAssignVariableToField(nonres, fname);
// call decorators for added value processing
for (int i = 0; i < m_decorators.length; i++) {
m_decorators[i].valueAdded(nonres, false, "boolean", field.getDeclaration(), null, null, descript, this);
}
}
/**
* Find the schema documentation associated with a data node. If the node has documentation set and does not have a
* separate class, this just returns the documentation from that node. Otherwise, it moves up the node tree until it
* finds a documented node, terminating if any parent has more than one child or when it reaches the node matching
* the root of the class data structure.
*
* @param top use topmost documentation found flag
* @param node starting node
* @return documentation
*/
private String findDocumentation(boolean top, DataNode node) {
// first check if separate class for node (in which case documentation goes with class)
Item item = node.getItem();
if (item instanceof GroupItem && ((GroupItem)item).getGenerateClass() != null) {
node = node.getParent();
}
// scan for documentation which can be used for node
String text = null;
while (node != m_dataRoot) {
String thisdoc = node.getDocumentation();
if (thisdoc != null) {
text = thisdoc;
if (!top) {
break;
}
}
ParentNode parent = node.getParent();
if (parent.getChildren().size() == 1) {
node = parent;
} else {
break;
}
}
return text;
}
/**
* Generate the fields and methods for a wrapper around one or more properties. This calls itself recursively to
* handle nested wrappers.
*
* @param parent
*/
private void addToClass(ParentNode parent) {
// first check if grouping requires a selector field
ClassBuilder builder = getBuilder();
ArrayList nodes = parent.getChildren();
Item grpitem = parent.getItem();
String seldesc = null;
if (parent.isSelectorNeeded()) {
// build the selector field
String basename = grpitem.getEffectiveName();
String fieldname = m_nameConverter.toFieldName(basename);
parent.setSelectField(fieldname);
builder.addIntField(fieldname, "-1").setPrivate();
// build the basic description information
AnnotatedBase comp = parent.getSchemaComponent();
StringBuffer buff = new StringBuffer();
if (parent.getQName() != null) {
buff.append(parent.getQName().getName());
buff.append(' ');
}
String descript;
if (comp.type() == SchemaBase.UNION_TYPE) {
descript = "form";
buff.append("union");
} else {
descript = "choice";
buff.append("choice");
}
seldesc = buff.toString();
buff.insert(0, "Clear the ");
buff.append(" selection.");
// create constants for each alternative value
String namesuffix = NameUtils.toNameWord(basename);
boolean expose = parent.isSelectorExposed();
int index = 0;
for (int i = 0; i < nodes.size(); i++) {
DataNode node = (DataNode)nodes.get(i);
if (!node.isIgnored()) {
FieldBuilder field = builder.addIntField(node.getSelectConstName(), Integer.toString(index++));
if (expose) {
field.setPublicStaticFinal();
field.addSourceComment(namesuffix + " value when " + node.getItem().getEffectiveName() +
" is set");
} else {
field.setPrivateStaticFinal();
}
}
}
// check for selector set methods used
int seltype = parent.getSelectorType();
// add the selector set method
String selectname = "set" + namesuffix;
String resetname = "clear" + namesuffix;
parent.setSelectSetMethod(selectname);
MethodBuilder setmeth = builder.addMethod(selectname, "void");
setmeth.setPrivate();
BlockBuilder block = setmeth.createBlock();
setmeth.addParameter(descript, "int");
// start by setting any containing selectors
generateSelectorSet(parent, block, builder);
// check for state check needed on set
if (seltype == NestingCustomBase.SELECTION_OVERRIDESET ||
seltype == NestingCustomBase.SELECTION_OVERRIDEBOTH) {
// set overrides prior state, just set new state directly
block.addAssignVariableToField(descript, fieldname);
} else {
// create the set block for when there's no current choice
BlockBuilder assignblock = builder.newBlock();
assignblock.addAssignVariableToField(descript, fieldname);
// create the exception thrown when choice does not match current setting
BlockBuilder throwblock = builder.newBlock();
throwblock.addThrowException("IllegalStateException", "Need to call " + resetname
+ "() before changing existing " + descript);
// finish with the if statement that decides which to execute
InfixExpressionBuilder iftest = builder.buildNameOp(fieldname, Operator.EQUALS);
iftest.addNumberLiteralOperand("-1");
InfixExpressionBuilder elsetest = builder.buildNameOp(fieldname, Operator.NOT_EQUALS);
elsetest.addVariableOperand(descript);
block.addIfElseIfStatement(iftest, elsetest, assignblock, throwblock);
}
// check for state check needed on get
if (seltype == NestingCustomBase.SELECTION_CHECKEDBOTH ||
seltype == NestingCustomBase.SELECTION_OVERRIDEBOTH) {
// add the selector check method
String checkname = "check" + namesuffix;
parent.setSelectCheckMethod(checkname);
MethodBuilder checkmeth = builder.addMethod(checkname, "void");
checkmeth.setPrivate();
block = checkmeth.createBlock();
checkmeth.addParameter(descript, "int");
// start by setting any containing selectors
generateSelectorCheck(parent, block, builder);
// create the exception thrown when current state is wrong
BlockBuilder throwblock = builder.newBlock();
InfixExpressionBuilder strcat = builder.buildStringConcatenation("State mismatch when accessing " +
descript + " value: current state is ");
strcat.addVariableOperand(fieldname);
throwblock.addThrowException("IllegalStateException", strcat);
// finish with the if statement that decides which to execute
InfixExpressionBuilder unequaltest = builder.buildNameOp(fieldname, Operator.NOT_EQUALS);
unequaltest.addVariableOperand(descript);
InfixExpressionBuilder unsettest = builder.buildNameOp(fieldname, Operator.NOT_EQUALS);
unsettest.addNumberLiteralOperand("-1");
InfixExpressionBuilder iftest = builder.buildInfix(Operator.CONDITIONAL_AND);
iftest.addOperand(unequaltest);
iftest.addOperand(unsettest);
block.addIfStatement(iftest, throwblock);
}
// add selector clear method (public, so documented)
MethodBuilder resetmeth = builder.addMethod(resetname, "void");
resetmeth.setPublic();
resetmeth.addSourceComment(buff.toString());
block = resetmeth.createBlock();
block.addAssignToName(block.numberLiteral("-1"), fieldname);
// add state check method if needed
if (expose) {
MethodBuilder statemethod = builder.addMethod("state" + namesuffix, "int");
statemethod.setPublic();
statemethod.addSourceComment("Get the current " + seldesc + " state.");
statemethod.addSourceComment("@return", " state");
statemethod.createBlock().addReturnNamed(fieldname);
}
if (s_logger.isDebugEnabled()) {
s_logger.debug("Created selector for grouping component "
+ SchemaUtils.describeComponent(grpitem.getSchemaComponent()) + " in class " + getFullName());
}
}
// generate all values in group
for (int i = 0; i < nodes.size(); i++) {
// check if there's a separate value for this child node
DataNode node = (DataNode)nodes.get(i);
Item item = node.getItem();
if (!node.isIgnored() && !item.isImplicit()) {
// get the base name and property name to be used for item
boolean repeat = node.isCollection() || node.isList();
String basename = item.getEffectiveName();
String workingname = basename;
if (repeat) {
String plural = m_nameConverter.pluralize(workingname);
if (!plural.equals(workingname)) {
workingname = m_nameSet.add(plural);
}
}
String propname = NameUtils.toNameWord(workingname);
node.setPropName(propname);
// generate test method, if inside selector group
checkIfMethod(node, seldesc, propname, builder);
// handle wrapper or simple property as appropriate
if (node.isInterior()) {
ParentNode nestedparent = (ParentNode)node;
if (nestedparent.isCollection()) {
// add property matching this node
addRepeatedProperty(basename, nestedparent, builder);
} else if (nestedparent.getChildren().size() == 0) {
if (nestedparent.isOptional() || parent.isSelectorNeeded()) {
// empty branch of tree, just add a boolean flag to indicate presence
addFlagProperty(basename, nestedparent, builder);
}
} else {
// just process nesting directly
addToClass(nestedparent);
}
} else {
// add property matching this node
if (repeat) {
addRepeatedProperty(basename, node, builder);
} else {
addSimpleProperty(basename, node, builder);
}
}
}
}
}
/**
* Build the binding structure element for a reference to a class.
*
* @param leaf reference node
* @param def target definition
* @param single flag for single child
* @param holder holder for binding definition
* @param bindcomp containing binding element
* @return binding structure
*/
private StructureElement addReferenceStructure(LeafNode leaf, DefinitionItem def, boolean single,
BindingHolder holder, ContainerElementBase bindcomp) {
// first check if a new structure element is needed in the binding for the reference
StructureElement struct = null;
boolean keep = false;
int type = leaf.getSchemaComponent().type();
if (single && bindcomp instanceof StructureElement) {
struct = (StructureElement)bindcomp;
keep = struct.getGetName() == null && struct.getSetName() == null && struct.getFieldName() == null &&
!struct.isChoice() && struct.getDeclaredType() == null && struct.getMapAsQName() == null &&
(struct.getName() == null || (type != SchemaBase.ATTRIBUTE_TYPE && type != SchemaBase.ELEMENT_TYPE));
}
if (!keep) {
struct = new StructureElement();
}
if (def.getSchemaComponent().type() == SchemaBase.ELEMENT_TYPE) {
// element definition reference, invoke concrete mapping by class name
struct.setDeclaredType(leaf.getBindingType());
} else {
// not an element reference, so map by abstract mapping 'type' name (which may or may not be a type)
QName qname = def.getQName();
String uri = qname.getUri();
if (uri != null) {
holder.addTypeNameReference(uri, def.getSchemaComponent().getSchema());
}
struct.setMapAsQName(qname);
}
return struct;
}
/**
* Create a new collection element for the binding. This initializes the create and declared types of the
* collection as appropriate, along with the wrapper name and optional status.
*
* @param wrapname name to be used for wrapper collection or structure, null
if none
* @param wrapopt wrapper element optional flag (should be false
if wrapname is null
)
* @param holder binding definition tracking information
* @param node data node associated with collection
* @return collection element
*/
private CollectionElement newCollection(QName wrapname, boolean wrapopt, BindingHolder holder, DataNode node) {
CollectionElement collect = new CollectionElement();
int reptype = getSchemaCustom().getRepeatType();
boolean list = reptype == SchemaRootBase.REPEAT_LIST || reptype == SchemaRootBase.REPEAT_TYPED;
if (list) {
collect.setCreateType(m_listImplClass);
}
if (wrapname == null) {
if (node.isOptional()) {
collect.setUsage(PropertyAttributes.OPTIONAL_USAGE);
}
} else {
setName(wrapname, holder, collect);
if (wrapopt) {
collect.setUsage(PropertyAttributes.OPTIONAL_USAGE);
}
}
return collect;
}
/**
* Set serializer/deserializer options for a <value> component of the binding. If the item defining the item is a
* reference, this uses the definition type name as the format and makes sure the definition namespace is defined
* within the binding being generated. If the item defining the item is a builtin type, this sets the format and/or
* serializer/deserializer methods based on the type definition.
*
* @param item
* @param value
* @param holder
*/
private void setValueHandlingOptions(Item item, ValueElement value, BindingHolder holder) {
if (item instanceof ReferenceItem) {
ReferenceItem refitem = (ReferenceItem)item;
DefinitionItem defitem = refitem.getDefinition();
QName qname = defitem.getQName();
if (qname != null) {
value.setFormatQName(qname);
String uri = qname.getUri();
if (uri != null) {
holder.addTypeNameReference(uri, defitem.getSchemaComponent().getSchema());
}
}
} else if (item instanceof ValueItem) {
ValueItem valitem = (ValueItem)item;
JavaType jtype = valitem.getType();
value.setFormatName(jtype.getFormat());
value.setSerializerName(jtype.getSerializerMethod());
value.setDeserializerName(jtype.getDeserializerMethod());
}
}
/**
* Generate the binding for a parent node of the data structure tree. This calls itself recursively to handle nested
* subtrees.
*
* TODO: This needs a more structured approach to creating the binding, which probably involves trying to merge the
* binding components down a particular branch of the tree as long as there's only one child (creating new
* structures as needed when the child has a name and there's already a name, or the child has a property and
* there's already a property, etc.)
*
* @param parent node to be added to binding
* @param wrapname name to be used for wrapper collection or structure, null
if none
* @param wrapopt wrapper element optional flag (should be false
if wrapname is null
)
* @param single parent node binding component can be modified by child flag (single path from parent)
* @param bindcomp binding definition component corresponding to the parent node
* @param holder binding definition tracking information
*/
private void addToBinding(ParentNode parent, QName wrapname, boolean wrapopt, boolean single,
ContainerElementBase bindcomp, BindingHolder holder) {
// generate for each child node in turn
ArrayList children = parent.getChildren();
for (int i = 0; i < children.size(); i++) {
DataNode child = (DataNode)children.get(i);
Item item = child.getItem();
String propname = child.getPropName();
if (child.isIgnored()) {
// create structure for element to be ignored in unmarshalling
StructureElement struct = new StructureElement();
setName(child.getQName(), holder, struct);
struct.setUsage(PropertyAttributes.OPTIONAL_USAGE);
bindcomp.addChild(struct);
} else if (item.isImplicit()) {
// implicit item uses superclass data, not a field, so just handle with map-as structure
DefinitionItem def = ((ReferenceItem)item).getDefinition();
QName qname = def.getQName();
AnnotatedBase comp = def.getSchemaComponent();
int type = comp.type();
StructureElement struct = new StructureElement();
if (type == SchemaBase.ELEMENT_TYPE) {
// reference to global element definition, just set it directly as concrete mapping reference
struct.setMapAsName(getSuperClass().getBindingName());
} else {
// set reference to abstract mapping without name, since this is an implicit reference
String uri = qname.getUri();
if (uri != null) {
holder.addTypeNameReference(uri, def.getSchemaComponent().getSchema());
}
struct.setMapAsQName(qname);
}
bindcomp.addChild(struct);
} else if (child instanceof ParentNode) {
// set up for binding generation alternatives
ParentNode subparent = (ParentNode)child;
boolean empty = subparent.getChildren().size() == 0;
boolean recurse = true;
ContainerElementBase wrapcomp = bindcomp;
StructureElementBase newcomp = null;
QName newname = null;
boolean newopt = false;
if (subparent.isCollection() && !empty) {
// always create a new element for the binding to match a collection parent node
CollectionElement collect = newCollection(wrapname, wrapopt, holder, subparent);
wrapcomp.addChild(collect);
newcomp = collect;
newname = subparent.getQName();
} else {
// check for wrapper element needed (with nested name, or multiple values, or all)
boolean all = item.getSchemaComponent().type() == SchemaBase.ALL_TYPE;
boolean multi = subparent.getChildren().size() > 1;
if ((wrapname != null && (subparent.isNamed() || multi)) || (all && multi)) {
StructureElement struct = new StructureElement();
struct.setOrdered(!all);
setName(wrapname, holder, struct);
if (wrapopt) {
struct.setUsage(PropertyAttributes.OPTIONAL_USAGE);
}
wrapcomp.addChild(struct);
wrapcomp = struct;
if (!empty && bindcomp.type() == ElementBase.COLLECTION_ELEMENT &&
getSchemaCustom().getRepeatType() != SchemaRootBase.REPEAT_ARRAY) {
// dig into child node(s) to find the item type
DataNode nested = subparent;
String type = null;
while (((ParentNode)nested).getChildren().size() > 0) {
nested = (DataNode)((ParentNode)nested).getChildren().get(0);
if (nested.isCollection() || nested instanceof LeafNode) {
type = nested.getBindingType();
break;
}
}
struct.setDeclaredType(type);
}
} else {
newname = wrapname;
newopt = wrapopt;
}
// check for name associated with this node
if (subparent.isNamed()) {
// check if this is an attribute
AnnotatedBase comp = item.getSchemaComponent();
if (comp.type() == SchemaBase.ATTRIBUTE_TYPE) {
// handle attribute with embedded definition
ValueElement value = new ValueElement();
value.setEffectiveStyle(NestingAttributes.ATTRIBUTE_STYLE);
setName(subparent.getQName(), holder, value);
DataNode nested = subparent;
while ((nested = (DataNode)((ParentNode)nested).getChildren().get(0)).isInterior());
value.setGetName(((LeafNode)nested).getGetMethodName());
value.setSetName(((LeafNode)nested).getSetMethodName());
setValueHandlingOptions(item, value, holder);
wrapcomp.addChild(value);
if (SchemaUtils.isOptionalAttribute((AttributeElement)comp)) {
value.setUsage(PropertyAttributes.OPTIONAL_USAGE);
}
if (nested.isList()) {
String nestname = nested.getPropName();
value.setSerializerName(getBindingName() + '.' + LIST_SERIALIZE_PREFIX + nestname);
value.setDeserializerName(getBindingName() + '.' + LIST_DESERIALIZE_PREFIX + nestname);
} else if (getSchemaCustom().isForceTypes()) {
value.setDeclaredType(nested.getBindingType());
}
recurse = false;
} else if (subparent.getChildren().size() == 1) {
// wrapper for an embedded structure or value, just pass name
newname = subparent.getQName();
newopt = subparent.isOptional();
} else {
// create a element, using the name supplied
StructureElement struct = new StructureElement();
setName(subparent.getQName(), holder, struct);
wrapcomp.addChild(struct);
newcomp = struct;
newname = null;
newopt = false;
}
}
}
// set 'if' method and optional if inside a choice
if (parent.isSelectorNeeded()) {
if (newcomp == null) {
newcomp = new StructureElement();
setName(newname, holder, newcomp);
newname = null;
wrapcomp.addChild(newcomp);
}
newcomp.setTestName("if" + child.getSelectPropName());
newcomp.setUsage(PropertyAttributes.OPTIONAL_USAGE);
newopt = false;
}
// handle parent with no children as flag-only value
if (empty) {
// make sure there's a structure element
StructureElementBase struct = newcomp;
if (struct == null) {
// create a element, using the wrapping name supplied
if (newname == null) {
throw new IllegalStateException("Internal error - no wrapping name for empty structure");
} else {
struct = new StructureElement();
setName(newname, holder, struct);
wrapcomp.addChild(struct);
newcomp = struct;
}
}
// set flag and test methods on structure
struct.setFlagName(subparent.getFlagMethodName());
struct.setTestName(subparent.getTestMethodName());
setStructureOptional(subparent, newopt, struct);
} else {
// add choice handling for this structure
if (subparent.isSelectorNeeded()) {
if (newcomp == null) {
newcomp = new StructureElement();
setStructureOptional(subparent, false, newcomp);
wrapcomp.addChild(newcomp);
}
newcomp.setChoice(true);
newcomp.setOrdered(false);
}
// check for new binding component created for this node
if (recurse) {
if (newcomp == null) {
addToBinding(subparent, newname, newopt, single && children.size() == 1, wrapcomp, holder);
} else {
newcomp.setGetName(subparent.getGetMethodName());
newcomp.setSetName(subparent.getSetMethodName());
if (getSchemaCustom().isForceTypes()) {
newcomp.setDeclaredType(subparent.getBindingType());
}
addToBinding(subparent, newname, newopt, true, newcomp, holder);
}
}
}
} else {
LeafNode leaf = (LeafNode)child;
String gname = leaf.getGetMethodName();
String sname = leaf.getSetMethodName();
if (leaf.isAny()) {
// add structure binding with details determined by xs:any handling
int anytype = item.getComponentExtension().getAnyType();
StructureElementBase struct = (leaf.isCollection() && anytype != NestingCustomBase.ANY_DOM) ?
(StructureElementBase)new CollectionElement() : (StructureElementBase)new StructureElement();
String mapper;
switch (anytype) {
case NestingCustomBase.ANY_DISCARD:
// use discard mapper to skip past arbitrary element(s) when unmarshalling
mapper = leaf.isCollection() ? "org.jibx.extras.DiscardListMapper" :
"org.jibx.extras.DiscardElementMapper";
struct.setDeclaredType("java.lang.Object");
gname = sname = null;
break;
case NestingCustomBase.ANY_DOM:
// use DOM mapper to marshal/unmarshal arbitrary element(s)
mapper = leaf.isCollection() ? "org.jibx.extras.DomListMapper" :
"org.jibx.extras.DomElementMapper";
break;
case NestingCustomBase.ANY_MAPPED:
// create item child for case of list, otherwise just handle directly
mapper = null;
if (leaf.isCollection()) {
StructureElement itemstruct = new StructureElement();
itemstruct.setDeclaredType("java.lang.Object");
struct.addChild(itemstruct);
struct.setCreateType(m_listImplClass);
}
break;
default:
throw new IllegalStateException("Internal error - unknown xs:any handling");
}
if (leaf.isOptional()) {
struct.setUsage(PropertyAttributes.OPTIONAL_USAGE);
}
struct.setGetName(gname);
struct.setSetName(sname);
struct.setMarshallerName(mapper);
struct.setUnmarshallerName(mapper);
bindcomp.addChild(struct);
} else {
// set the names to be used for value
if (leaf.isCollection() || leaf.isList()) {
// process list and collection differently for binding
if (leaf.isCollection()) {
// create a new collection element
CollectionElement collect = newCollection(wrapname, wrapopt, holder, leaf);
bindcomp.addChild(collect);
// fill in the collection details
collect.setGetName(gname);
collect.setSetName(sname);
if (parent.isSelectorNeeded()) {
collect.setUsage(PropertyAttributes.OPTIONAL_USAGE);
collect.setTestName("if" + leaf.getSelectPropName());
}
int reptype = getSchemaCustom().getRepeatType();
if (reptype == SchemaRootBase.REPEAT_LIST || reptype == SchemaRootBase.REPEAT_TYPED) {
collect.setCreateType(m_listImplClass);
if (gname == null) {
collect.setDeclaredType(COLLECTION_VARIABLE_TYPE);
}
}
// check the content (if any) for
boolean usevalue = true;
String usetype = leaf.getType();
if (item instanceof ReferenceItem) {
DefinitionItem def = ((ReferenceItem)item).getDefinition();
TypeData defclas = def.getGenerateClass();
if (defclas.isSimpleValue()) {
usetype = defclas.getBindingName();
} else {
// reference to mapped class, configure to handle it properly
usevalue = false;
if (def.getSchemaComponent().type() == SchemaBase.ELEMENT_TYPE) {
// must be a non-abstract , so use it directly
collect.setItemTypeName(defclas.getBindingName());
} else {
// abstract mapping reference, create child with map-as type
StructureElement struct = new StructureElement();
QName qname = def.getQName();
String uri = qname.getUri();
if (uri != null) {
holder.addTypeNameReference(uri, def.getSchemaComponent().getSchema());
}
struct.setMapAsQName(qname);
if (leaf.isNamed()) {
setName(leaf.getQName(), holder, struct);
}
collect.addChild(struct);
}
}
} else if (item instanceof GroupItem) {
// handle group directly if a structure class, else just as
TypeData groupclas = ((GroupItem)item).getGenerateClass();
if (groupclas.isSimpleValue()) {
usetype = groupclas.getBindingName();
} else {
// add element to be filled in by inner class generation
usevalue = false;
StructureClassHolder classholder = ((StructureClassHolder)groupclas);
StructureElement struct = new StructureElement();
struct.setDeclaredType(classholder.getBindingName());
setName(leaf.getQName(), holder, struct);
// set component for dependent class generation
classholder.setBinding(struct);
collect.addChild(struct);
}
}
if (usevalue) {
// add element to collection for simple (primitive or text) value
ValueElement value = new ValueElement();
value.setEffectiveStyle(NestingAttributes.ELEMENT_STYLE);
if (leaf.isNamed()) {
setName(leaf.getQName(), holder, value);
}
setValueHandlingOptions(item, value, holder);
value.setDeclaredType(usetype);
collect.addChild(value);
}
} else {
// handle list serialization and deserialization directly
ValueElement value = buildValueBinding(leaf, null, gname, sname, holder);
value.setSerializerName(getBindingName() + '.' + LIST_SERIALIZE_PREFIX + propname);
value.setDeserializerName(getBindingName() + '.' + LIST_DESERIALIZE_PREFIX + propname);
bindcomp.addChild(value);
}
} else {
// add wrapper if name passed in with name on value
ContainerElementBase contain = bindcomp;
boolean consing = single && children.size() == 1;
if (wrapname != null && (leaf.isNamed() || leaf.isReference())) {
StructureElement struct = new StructureElement();
setName(wrapname, holder, struct);
String type = leaf.getBindingType();
if (gname == null) {
struct.setDeclaredType(type);
} else if (!Types.isSimpleValue(type)) {
// apply access methods to wrapper only if this is a complex value
struct.setGetName(gname);
struct.setSetName(sname);
gname = sname = null;
if (getSchemaCustom().isForceTypes()) {
struct.setDeclaredType(type);
}
}
if (wrapopt) {
struct.setUsage(PropertyAttributes.OPTIONAL_USAGE);
}
contain.addChild(struct);
contain = struct;
consing = true;
wrapname = null;
wrapopt = false;
}
// build the appropriate binding representation
StructureElement struct = null;
if (item instanceof ReferenceItem) {
// handle reference directly if a structure class, else just as value
DefinitionItem def = ((ReferenceItem)item).getDefinition();
TypeData defclas = def.getGenerateClass();
if (!defclas.isSimpleValue()) {
struct = addReferenceStructure(leaf, def, consing, holder, contain);
}
} else if (item instanceof GroupItem) {
// handle group directly if a structure class, else just as value
TypeData groupclas = ((GroupItem)item).getGenerateClass();
if (!groupclas.isSimpleValue()) {
// create a new element for reference
struct = new StructureElement();
StructureClassHolder refclas = (StructureClassHolder)groupclas;
if (gname == null) {
struct.setDeclaredType(refclas.getBindingName());
}
// set the binding component to be filled in by inner class generation
refclas.setBinding(struct);
}
}
if (struct == null) {
// add simple binding for field
ValueElement value = buildValueBinding(leaf, wrapname, gname, sname, holder);
if (getSchemaCustom().isForceTypes()) {
value.setDeclaredType(leaf.getBindingType());
}
if (wrapopt) {
value.setUsage(PropertyAttributes.OPTIONAL_USAGE);
}
contain.addChild(value);
} else {
// fill in structure attributes
struct.setGetName(gname);
struct.setSetName(sname);
if (getSchemaCustom().isForceTypes()) {
struct.setDeclaredType(leaf.getBindingType());
}
// set the name and optional status from wrapper or item
setName(wrapname == null ? leaf.getQName() : wrapname, holder, struct);
setStructureOptional(leaf, wrapopt, struct);
// common handling for structure
if (parent.isSelectorNeeded()) {
struct.setUsage(PropertyAttributes.OPTIONAL_USAGE);
struct.setTestName("if" + leaf.getSelectPropName());
}
// add to containing binding component
if (struct != contain) {
contain.addChild(struct);
}
}
}
}
}
}
}
/**
* Generate this class.
*
* @param verbose
* @param builder class source file builder
*/
public void generate(boolean verbose, SourceBuilder builder) {
// setup the class builder
String name = getName();
ClassBuilder clasbuilder;
if (m_outerClass == null) {
clasbuilder = builder.newMainClass(name, false);
} else {
clasbuilder = builder.newInnerClass(name, m_outerClass.getBuilder(), false);
}
// handle the common initialization
initClass(verbose, clasbuilder, m_dataRoot);
// add nested definitions to
if (m_bindingElement instanceof MappingElement) {
addInnerFormats((MappingElementBase)m_bindingElement);
}
// add choice handling if needed
if (m_dataRoot.isSelectorNeeded()) {
m_bindingElement.setChoice(true);
m_bindingElement.setOrdered(false);
}
// fix all the value names
String fullname = getFullName();
if (s_logger.isInfoEnabled()) {
s_logger.info("Generating class " + fullname + ":\n" + m_dataRoot.describe(0));
}
addFixedNames(m_dataRoot);
fixFlexibleNames(m_dataRoot, false);
// make class abstract if appropriate
AnnotatedBase comp = m_dataRoot.getSchemaComponent();
boolean abs = false;
int type = comp.type();
if (type == SchemaBase.ELEMENT_TYPE) {
abs = ((ElementElement)comp).isAbstract();
} else if (type == SchemaBase.COMPLEXTYPE_TYPE) {
abs = ((ComplexTypeElement)comp).isAbstract();
}
if (abs) {
getBuilder().setAbstract();
}
// generate the class data structure
addToClass(m_dataRoot);
if (s_logger.isInfoEnabled()) {
s_logger.info("Class " + fullname + " after fields generated:\n" + m_dataRoot.describe(0));
}
addToBinding(m_dataRoot, null, false, false, m_bindingElement, m_holder);
// generate the binding code
// m_classBuilder.addInterface("org.jibx.v2.MappedStructure");
// finish with subclass generation
generateInner(verbose, builder);
finishClass(m_bindingElement);
// check for empty elements to be deleted from
if (m_bindingElement instanceof MappingElement) {
MappingElementBase mapping = (MappingElementBase)m_bindingElement;
ArrayList childs = mapping.topChildren();
int fill = 0;
for (int i = 0; i < childs.size(); i++) {
Object child = childs.get(i);
boolean keep = true;
if (child instanceof FormatElement) {
FormatElement format = (FormatElement)child;
keep = format.getDefaultText() != null || format.getDeserializerName() != null ||
format.getEnumValueName() != null || format.getSerializerName() != null;
}
if (keep) {
childs.set(fill++, child);
}
}
while (fill < childs.size()) {
childs.remove(childs.size()-1);
}
}
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy