org.jibx.schema.codegen.CodeGen Maven / Gradle / Ivy
/*
* Copyright (c) 2007-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.io.BufferedReader;
import java.io.BufferedWriter;
import java.io.File;
import java.io.FileReader;
import java.io.FileWriter;
import java.io.IOException;
import java.net.URL;
import java.util.ArrayList;
import java.util.HashMap;
import java.util.HashSet;
import java.util.Iterator;
import java.util.LinkedList;
import java.util.List;
import java.util.Map;
import java.util.Set;
import org.apache.log4j.Level;
import org.apache.log4j.Logger;
import org.eclipse.jdt.core.dom.AST;
import org.jibx.binding.model.BindingElement;
import org.jibx.binding.model.BindingHolder;
import org.jibx.binding.model.BindingOrganizer;
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.NamespaceElement;
import org.jibx.binding.model.StructureElement;
import org.jibx.runtime.JiBXException;
import org.jibx.runtime.QName;
import org.jibx.runtime.Utility;
import org.jibx.schema.INamed;
import org.jibx.schema.IReference;
import org.jibx.schema.ISchemaResolver;
import org.jibx.schema.SchemaContextTracker;
import org.jibx.schema.SchemaUtils;
import org.jibx.schema.SchemaVisitor;
import org.jibx.schema.TreeWalker;
import org.jibx.schema.UrlResolver;
import org.jibx.schema.codegen.custom.ComponentExtension;
import org.jibx.schema.codegen.custom.GlobalExtension;
import org.jibx.schema.codegen.custom.SchemaCustom;
import org.jibx.schema.codegen.custom.SchemaExtension;
import org.jibx.schema.codegen.custom.SchemasetCustom;
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.AttributeGroupRefElement;
import org.jibx.schema.elements.CommonTypeDerivation;
import org.jibx.schema.elements.ComplexTypeElement;
import org.jibx.schema.elements.ElementElement;
import org.jibx.schema.elements.GroupRefElement;
import org.jibx.schema.elements.OpenAttrBase;
import org.jibx.schema.elements.SchemaBase;
import org.jibx.schema.elements.SchemaElement;
import org.jibx.schema.elements.SimpleRestrictionElement;
import org.jibx.schema.elements.SimpleTypeElement;
import org.jibx.schema.elements.UnionElement;
import org.jibx.schema.support.SchemaTypes;
import org.jibx.schema.validation.ProblemConsoleLister;
import org.jibx.schema.validation.ProblemHandler;
import org.jibx.schema.validation.ProblemLogLister;
import org.jibx.schema.validation.ProblemMultiHandler;
import org.jibx.schema.validation.ValidationContext;
import org.jibx.schema.validation.ValidationProblem;
import org.jibx.schema.validation.ValidationUtils;
import org.jibx.util.ClasspathUrlExtender;
import org.jibx.util.DummyClassLocator;
import org.jibx.util.IClassLocator;
import org.jibx.util.InsertionOrderedSet;
import org.jibx.util.LazyList;
import org.jibx.util.ResourceMatcher;
/**
* Code generator from schema definition. Although many of the methods in this class use public
access,
* they are intended for use only by the JiBX developers and may change from one release to the next. To make use of
* this class from your own code, call the {@link #main(String[])} method with an appropriate argument list.
*
* @author Dennis M. Sosnoski
*/
public class CodeGen
{
/** Logger for class. */
private static final Logger s_logger = Logger.getLogger(CodeGen.class.getName());
/** Default type replacements applied. */
private static final QName[] DEFAULT_REPLACEMENTS =
new QName[] {
SchemaTypes.ANY_URI.getQName(), SchemaTypes.STRING.getQName(),
SchemaTypes.DURATION.getQName(), SchemaTypes.STRING.getQName(),
SchemaTypes.ENTITIES.getQName(), SchemaTypes.STRING.getQName(),
SchemaTypes.ENTITY.getQName(), SchemaTypes.STRING.getQName(),
SchemaTypes.GDAY.getQName(), SchemaTypes.STRING.getQName(),
SchemaTypes.GMONTH.getQName(), SchemaTypes.STRING.getQName(),
SchemaTypes.GMONTHDAY.getQName(), SchemaTypes.STRING.getQName(),
SchemaTypes.GYEAR.getQName(), SchemaTypes.STRING.getQName(),
SchemaTypes.GYEARMONTH.getQName(), SchemaTypes.STRING.getQName(),
SchemaTypes.ID.getQName(), SchemaTypes.STRING.getQName(),
SchemaTypes.IDREF.getQName(), SchemaTypes.STRING.getQName(),
SchemaTypes.IDREFS.getQName(), SchemaTypes.STRING.getQName(),
SchemaTypes.LANGUAGE.getQName(), SchemaTypes.STRING.getQName(),
SchemaTypes.NAME.getQName(), SchemaTypes.STRING.getQName(),
SchemaTypes.NEGATIVE_INTEGER.getQName(), SchemaTypes.INTEGER.getQName(),
SchemaTypes.NON_NEGATIVE_INTEGER.getQName(), SchemaTypes.INTEGER.getQName(),
SchemaTypes.NON_POSITIVE_INTEGER.getQName(), SchemaTypes.INTEGER.getQName(),
SchemaTypes.NORMALIZED_STRING.getQName(), SchemaTypes.STRING.getQName(),
SchemaTypes.NCNAME.getQName(), SchemaTypes.STRING.getQName(),
SchemaTypes.NMTOKEN.getQName(), SchemaTypes.STRING.getQName(),
SchemaTypes.NMTOKENS.getQName(), SchemaTypes.STRING.getQName(),
SchemaTypes.NOTATION.getQName(), SchemaTypes.STRING.getQName(),
SchemaTypes.POSITIVE_INTEGER.getQName(), SchemaTypes.STRING.getQName(),
SchemaTypes.TOKEN.getQName(), SchemaTypes.STRING.getQName(),
SchemaTypes.UNSIGNED_BYTE.getQName(), SchemaTypes.BYTE.getQName(),
SchemaTypes.UNSIGNED_INT.getQName(), SchemaTypes.INT.getQName(),
SchemaTypes.UNSIGNED_LONG.getQName(), SchemaTypes.LONG.getQName(),
SchemaTypes.UNSIGNED_SHORT.getQName(), SchemaTypes.SHORT.getQName()
};
/** Mask for schema elements which derive from a type. */
private static final long TYPE_DERIVE_MASK =
SchemaBase.ELEMENT_MASKS[SchemaBase.EXTENSION_TYPE] | SchemaBase.ELEMENT_MASKS[SchemaBase.RESTRICTION_TYPE];
/** Mask for schema elements which define a type. */
private static final long TYPE_DEFINE_MASK =
SchemaBase.ELEMENT_MASKS[SchemaBase.COMPLEXTYPE_TYPE] | SchemaBase.ELEMENT_MASKS[SchemaBase.SIMPLETYPE_TYPE];
/** Mask for schema elements which block name inheritance downward. */
private static final long BLOCK_NAME_INHERIT_MASK =
TYPE_DERIVE_MASK | SchemaBase.ELEMENT_MASKS[SchemaBase.UNION_TYPE];
/** Code generation customizations. */
private final SchemasetCustom m_global;
/** Target directory for code generation. */
private final File m_targetDir;
/** Context for loading and processing schemas. */
private final ValidationContext m_validationContext;
/** Package directory for generated classes. */
private PackageOrganizer m_packageDirectory;
/** Directory for constructed bindings. */
private BindingOrganizer m_bindingDirectory;
/** Root binding definition holder (set by {@link #writeBindings(String, String, List, ProblemHandler)}). */
private BindingHolder m_rootHolder;
/**
* Constructor.
*
* @param global schema customization root element
* @param root URL for base of schema paths
* @param target destination directory for code generation
*/
public CodeGen(SchemasetCustom global, URL root, File target) {
m_global = global;
addDefaultSubstitutions(m_global);
m_targetDir = target;
m_validationContext = new ValidationContext();
}
/**
* Constructor used by tests. This uses supplied schemas and skips writing to the file system.
*
* @param custom
* @param vctx
*/
public CodeGen(SchemasetCustom custom, ValidationContext vctx) {
m_global = custom;
addDefaultSubstitutions(custom);
m_targetDir = null;
m_validationContext = vctx;
}
/**
* Get the validation context used for processing schemas.
*
* @return context
*/
public ValidationContext getSchemaValidationContext() {
return m_validationContext;
}
/**
* Add default type substitutions to set currently defined.
*
* @param custom
*/
private static void addDefaultSubstitutions(SchemasetCustom custom) {
QName[] subs = custom.getSubstitutions();
if (subs == null) {
custom.setSubstitutions(DEFAULT_REPLACEMENTS);
} else {
QName[] newsubs = new QName[subs.length + DEFAULT_REPLACEMENTS.length];
System.arraycopy(subs, 0, newsubs, 0, subs.length);
System.arraycopy(DEFAULT_REPLACEMENTS, 0, newsubs, subs.length, DEFAULT_REPLACEMENTS.length);
custom.setSubstitutions(newsubs);
}
}
/**
* Find the most specific schemaset owning a schema. If multiple matches are found which are not in line of
* containment the first match is returned and the conflict is reported as an error.
*
* @param schema
* @param custom schema set customization
* @return owning schemaset, null
if none
*/
public SchemasetCustom findSchemaset(SchemaElement schema, SchemasetCustom custom) {
LazyList childs = custom.getChildren();
SchemasetCustom owner = null;
String name = schema.getResolver().getName();
for (int i = 0; i < childs.size(); i++) {
Object child = childs.get(i);
if (child instanceof SchemasetCustom) {
SchemasetCustom schemaset = (SchemasetCustom)child;
if (schemaset.isInSet(name, schema)) {
SchemasetCustom match = findSchemaset(schema, schemaset);
if (match != null) {
if (owner == null) {
owner = match;
} else {
m_validationContext.addError("schema-set overlap on schema " + name + " (first match "
+ ValidationProblem.componentDescription(owner) + ')', match);
}
}
}
}
}
return owner == null ? custom : owner;
}
/**
* Scan schemas to find the default prefixes used for namespaces.
*
* @param iter schema iterator
*/
public void setDefaultPrefixes(Iterator iter) {
while (iter.hasNext()) {
SchemaElement schema = (SchemaElement)iter.next();
ArrayList decls = schema.getNamespaceDeclarations();
for (int i = 0; i < decls.size(); i++) {
String prefix = (String)decls.get(i++);
m_bindingDirectory.addDefaultPrefix((String)decls.get(i), prefix);
}
}
}
/**
* Validate and apply customizations to loaded schemas.
*
* @param pack package to be used by default for no-namespaced schemas (non-null
)
* @param handler validation problem handler
* @return true
if successful, false
if error
*/
public boolean customizeSchemas(String pack, ProblemHandler handler) {
// TODO: remove this once list and union handling fully implemented
SchemaVisitor visitor = new SchemaVisitor() {
private void replaceSimpleType(OpenAttrBase node) {
OpenAttrBase parent = node.getParent();
int count = parent.getChildCount();
for (int i = 0; i < count; i++) {
if (parent.getChild(i) == node) {
SimpleRestrictionElement empty = new SimpleRestrictionElement();
empty.setBase(SchemaTypes.STRING.getQName());
parent.replaceChild(i, empty);
return;
}
}
}
/* public void exit(ListElement node) {
replaceSimpleType(node);
} */
public void exit(UnionElement node) {
replaceSimpleType(node);
}
};
TreeWalker wlkr = new TreeWalker(null, new SchemaContextTracker());
for (Iterator iter = m_validationContext.iterateSchemas(); iter.hasNext();) {
wlkr.walkElement((SchemaElement)iter.next(), visitor);
}
// // TODO: test code for annotating schema
// try {
// DocumentBuilder builder = DocumentBuilderFactory.newInstance().newDocumentBuilder();
// final Document document = builder.newDocument();
// visitor = new SchemaVisitor() {
//
// public void exit(AnnotatedBase node) {
// if (node.getAnnotation() == null) {
// StringBuffer buff = new StringBuffer();
// buff.append(node.name());
// if (node instanceof INamed && ((INamed)node).getName() != null) {
// buff.append(" name='");
// buff.append(((INamed)node).getName());
// buff.append('\'');
// } else if (node instanceof IReference && ((IReference)node).getRef() != null) {
// buff.append(" ref='");
// buff.append(((IReference)node).getRef().getName());
// buff.append('\'');
// }
// AnnotationElement anno = new AnnotationElement();
// node.setAnnotation(anno);
// DocumentationElement doc = new DocumentationElement();
// doc.addContent(document.createTextNode(buff.toString()));
// anno.getItemsList().add(doc);
// }
// }
//
// };
// for (Iterator iter = m_validationContext.iterateSchemas(); iter.hasNext();) {
// wlkr.walkElement((SchemaElement)iter.next(), visitor);
// }
// } catch (ParserConfigurationException e) {
// /* ignore any error */
// }
// validate the customizations
m_global.validate(m_validationContext);
// create the package directory, using default package from root schemaset
String dfltpack = m_global.getPackage();
if (dfltpack == null) {
dfltpack = pack;
}
m_packageDirectory = new PackageOrganizer(m_targetDir, dfltpack);
// link each schema to a customization, creating a default customization if necessary
int count = 0;
for (Iterator iter = m_validationContext.iterateSchemas(); iter.hasNext();) {
SchemaElement schema = (SchemaElement)iter.next();
ISchemaResolver resolver = schema.getResolver();
s_logger.debug("Assigning customization for schema " + ++count + ": " + resolver.getName());
SchemasetCustom owner = findSchemaset(schema, m_global);
SchemaCustom custom = owner.forceCustomization(resolver.getName(), resolver.getId(), schema,
m_validationContext);
custom.validate(m_validationContext);
String pname = custom.getPackage();
PackageHolder holder = null;
if (pname == null) {
String uri = schema.getEffectiveNamespace();
if (uri == null) {
uri = "";
}
holder = m_packageDirectory.getPackageForUri(uri);
} else {
holder = m_packageDirectory.getPackage(pname);
}
custom.extend(holder, m_validationContext);
}
// check all the customizations
m_global.checkSchemas(m_validationContext);
return !m_validationContext.reportProblems(handler);
}
/**
* Process substitutions and deletions defined by extensions. This builds the cross-reference information for the
* global definition components of the schemas while removing references to deleted components.
*
* @return true
if any changes to the schemas, false
if not
*/
private boolean processExtensions() {
// first clear all the cross reference information
for (Iterator iter = m_validationContext.iterateSchemas(); iter.hasNext();) {
SchemaElement schema = (SchemaElement)iter.next();
int count = schema.getChildCount();
for (int i = 0; i < count; i++) {
SchemaBase child = schema.getChild(i);
Object obj = child.getExtension();
if (obj instanceof GlobalExtension) {
((GlobalExtension)obj).resetDependencies();
}
}
}
// process each loaded schema for deletions and cross referencing
int index = 0;
m_validationContext.clearTraversed();
boolean modified = false;
// Level level = TreeWalker.setLogging(s_logger.getLevel());
for (Iterator iter = m_validationContext.iterateSchemas(); iter.hasNext();) {
SchemaElement schema = (SchemaElement)iter.next();
m_validationContext.enterSchema(schema);
s_logger.debug("Applying extensions to schema " + ++index + ": " + schema.getResolver().getName());
int count = schema.getChildCount();
boolean instmod = false;
for (int i = 0; i < count; i++) {
SchemaBase child = schema.getChild(i);
Object obj = child.getExtension();
if (obj instanceof GlobalExtension) {
// apply extension to global definition element
ComponentExtension exten = (ComponentExtension)obj;
if (exten.isRemoved()) {
// just eliminate this definition from the schema
schema.detachChild(i);
instmod = true;
} else {
// process the definition to remove references to deleted components
exten.applyAndCountUsage(m_validationContext);
}
}
}
if (instmod) {
schema.compactChildren();
modified = true;
}
m_validationContext.exitSchema();
}
// TreeWalker.setLogging(level);
return modified;
}
/**
* Apply extensions and normalize all schemas. This may be a multipass process, since applying extensions may create
* the opportunity for further normalizations and vice versa.
*/
public void applyAndNormalize() {
// loop until no modifications, with at least one pass of extensions and normalizations
boolean modified = true;
while (processExtensions() || modified) {
// normalize all the schema definitions
modified = false;
for (Iterator iter = m_validationContext.iterateSchemas(); iter.hasNext();) {
SchemaElement schema = (SchemaElement)iter.next();
int count = schema.getChildCount();
boolean instmod = false;
for (int i = 0; i < count; i++) {
SchemaBase child = schema.getChild(i);
Object obj = child.getExtension();
if (obj instanceof GlobalExtension) {
GlobalExtension global = (GlobalExtension)obj;
global.normalize();
if (global.isRemoved() ||
(!global.isIncluded() && global.isPreferInline() && global.getOverrideType() != null)) {
// just eliminate this definition from the schema
schema.detachChild(i);
instmod = true;
}
}
}
if (instmod) {
schema.compactChildren();
modified = true;
}
}
}
// finish by flagging global definitions requiring separate classes
TreeWalker wlkr = new TreeWalker(null, new SchemaContextTracker());
for (Iterator iter = m_validationContext.iterateSchemas(); iter.hasNext();) {
SchemaElement schema = (SchemaElement)iter.next();
SchemaCustom custom = ((SchemaExtension)schema.getExtension()).getCustom();
if (custom.isGenerateAll()) {
int count = schema.getChildCount();
for (int i = 0; i < count; i++) {
// check if this global needs a class
SchemaBase comp = schema.getChild(i);
boolean include = false;
switch (comp.type()) {
/* case SchemaBase.ATTRIBUTE_TYPE:
include = !custom.isAttributeInlined();
break; */
case SchemaBase.ATTRIBUTEGROUP_TYPE:
case SchemaBase.GROUP_TYPE:
// for attribute group or group, only force separate class if multiple components
ValueCountVisitor visitor = new ValueCountVisitor();
wlkr.walkChildren(comp, visitor);
include = visitor.getCount() > 1;
break;
case SchemaBase.COMPLEXTYPE_TYPE:
case SchemaBase.ELEMENT_TYPE:
include = true;
break;
case SchemaBase.SIMPLETYPE_TYPE:
// for simpleType definition, only need separate class if an enumeration
if (comp.getChildCount() > 0) {
SchemaBase deriv = ((SimpleTypeElement)comp).getDerivation();
include = deriv != null && deriv.type() == SchemaBase.RESTRICTION_TYPE &&
((SimpleRestrictionElement)deriv).getFacetsList().size() > 0;
}
break;
}
if (include) {
// flag separate class for global
Object obj = comp.getExtension();
if (obj instanceof GlobalExtension) {
((GlobalExtension)obj).setIncluded(true);
if (s_logger.isDebugEnabled()) {
s_logger.debug("Set include for definition " + SchemaUtils.describeComponent(comp));
}
}
}
}
}
}
}
/**
* Processes the schemas to remove unused global definitions.
*/
public void pruneDefinitions() {
// start by recursively checking for removable global definitions
int index = 0;
for (Iterator iter = m_validationContext.iterateSchemas(); iter.hasNext();) {
SchemaElement schema = (SchemaElement)iter.next();
s_logger.debug("Checking for unused definitions in schema " + ++index + ": "
+ schema.getResolver().getName());
int count = schema.getChildCount();
for (int i = 0; i < count; i++) {
SchemaBase child = schema.getChild(i);
Object exten = child.getExtension();
if (exten instanceof GlobalExtension) {
// check if global definition is unused and not specifically required
((GlobalExtension)exten).checkRemovable();
}
}
}
// next remove all the definitions flagged in the first step
index = 0;
for (Iterator iter = m_validationContext.iterateSchemas(); iter.hasNext();) {
SchemaElement schema = (SchemaElement)iter.next();
s_logger.debug("Deleting unused definitions in schema " + ++index + ": " + schema.getResolver().getName());
int count = schema.getChildCount();
boolean modified = false;
for (int i = 0; i < count; i++) {
SchemaBase child = schema.getChild(i);
Object exten = child.getExtension();
if (exten instanceof GlobalExtension && ((ComponentExtension)exten).isRemoved()) {
// remove the definition from schema
schema.detachChild(i);
modified = true;
if (s_logger.isDebugEnabled()) {
s_logger.debug(" Removed definition " + ((INamed)child).getQName());
}
}
}
if (modified) {
schema.compactChildren();
}
}
}
/**
* Check if an item has an associated name. If the component associated with the item has a name, this just returns
* that name converted to base name form. The only exception is for inlined global type definitions, which are
* treated as unnamed.
*
* @param item
* @return name associated name, or null
if none
*/
private String checkDirectName(Item item) {
AnnotatedBase comp = item.getSchemaComponent();
if (comp instanceof INamed || comp instanceof IReference) {
// check for an inlined global type definition
boolean usename = true;
if (comp.isGlobal()) {
if ((comp.bit() & TYPE_DEFINE_MASK) != 0 && !(item instanceof DefinitionItem)) {
usename = false;
}
}
if (usename) {
// use name from schema component
String name = null;
if (comp instanceof INamed) {
name = ((INamed)comp).getName();
}
if (name == null && comp instanceof IReference) {
name = ((IReference)comp).getRef().getName();
}
if (name != null) {
NameConverter nconv = item.getComponentExtension().getGlobal().getNameConverter();
return nconv.toBaseName(name);
}
}
}
return null;
}
/**
* Derive the base name for an item. If not forced, the only time a name will be returned is when the item is a
* reference to a non-type definition. If forced, this will try other alternatives for names including the text
* "Enumeration" for an enumeration group, the base type name for a type derivation, the schema type name for a
* value of a schema type, or finally the schema component element name.
*
* @param item
* @param force name forced flag
* @return name (null
if to use inherited name when force == false
)
*/
private String deriveName(Item item, boolean force) {
// try alternatives, in decreasing preference order
AnnotatedBase comp = item.getSchemaComponent();
String text = null;
if (force) {
if (item instanceof ReferenceItem) {
text = ((ReferenceItem)item).getDefinition().getName();
} else if (item instanceof GroupItem && ((GroupItem)item).isEnumeration()) {
text = "Enumeration";
} else if ((TYPE_DERIVE_MASK & comp.bit()) != 0 && ((CommonTypeDerivation)comp).getBase() != null) {
text = ((CommonTypeDerivation)comp).getBase().getName();
} else if (item instanceof ValueItem) {
text = ((ValueItem)item).getSchemaType().getName();
} else {
text = comp.name();
}
} else if (item instanceof ReferenceItem && (TYPE_DEFINE_MASK & comp.type()) == 0) {
// use name from definition in the case of anything except a type definition
text = ((ReferenceItem)item).getDefinition().getName();
}
if (text == null) {
return null;
} else {
return item.getComponentExtension().getGlobal().getNameConverter().toBaseName(text);
}
}
/**
* Compact group structures. This eliminates redundant groupings, in the form of groups with only one child, which
* child is a group referencing the same schema component as the parent group, from the data structure
* representation.
*
* @param group
*/
private void compactGroups(GroupItem group) {
Item child;
while (group.getChildCount() == 1 && (child = group.getFirstChild()) instanceof GroupItem &&
!child.isTopmost()) {
group.adoptChildren((GroupItem)child);
}
for (child = group.getFirstChild(); child != null; child = child.getNext()) {
if (child instanceof GroupItem) {
compactGroups((GroupItem)child);
}
}
}
/**
* Assemble a name from a base name and an optional prefix. If the prefix is supplied, this first converts the base
* name to embedded form, then appends the prefix.
*
* @param prefix
* @param base
* @return name
*/
private String assembleName(String prefix, String base) {
if (prefix == null) {
return base;
} else {
return prefix + NameUtils.toNameWord(base);
}
}
/**
* Set the basic names to be used for a structure of items. For named components of the schema definition the names
* used are simply the converted XML local names, for other components more complex rules apply (see {@link
* #deriveName(Item,boolean)}. This method calls itself recursively to handle nested groups.
*
* @param prefix text to be prefixed to names within inlined group (null
if none)
* @param group item group to be assigned
* @param force force name derivation flag
*/
private void assignNames(String prefix, GroupItem group, boolean force) {
// use existing name if set, otherwise derive from context if necessary
String name = group.getName();
AnnotatedBase comp = group.getSchemaComponent();
int type = comp.type();
boolean propagate = group.getChildCount() == 1 &&
(type == SchemaBase.ELEMENT_TYPE || type == SchemaBase.ATTRIBUTE_TYPE);
if (name == null) {
name = checkDirectName(group);
if (name == null && force) {
propagate = false;
name = deriveName(group, true);
}
}
// set name needed for this structure (as either value or class)
if (name != null) {
if (group.getClassName() == null) {
NameConverter conv = group.getComponentExtension().getGlobal().getNameConverter();
group.setClassName(conv.toJavaClassName(name));
}
if (group.getName() == null) {
group.setName(NameUtils.toNameLead(assembleName(prefix, name)));
}
}
// set passed-down prefix to this name if inline group with element name and more than one child, or multiple
// nested children and no passed-in prefix
if (group.isInline() && name != null) {
if (comp.type() == SchemaBase.ELEMENT_TYPE && SchemaUtils.isNamed(comp)) {
boolean passprefix = false;
if (group.getChildCount() > 1) {
passprefix = true;
} else if (prefix == null) {
passprefix = true;
GroupItem childgroup = group;
while (childgroup.getChildCount() == 1) {
Item childchild = childgroup.getFirstChild();
if (childchild instanceof GroupItem) {
childgroup = (GroupItem)childchild;
} else {
passprefix = false;
break;
}
}
}
if (passprefix) {
prefix = assembleName(prefix, name);
}
}
}
// propagate name downward if group is inline and single nested item without its own name
Item head = group.getFirstChild();
if (propagate) {
// name can be inherited, but continue recursion for child group
if (head instanceof GroupItem) {
GroupItem childgroup = (GroupItem)head;
assignNames(childgroup.isInline() ? prefix : null, childgroup, false);
}
} else {
// process all child items with definite name assignments
for (Item item = head; item != null; item = item.getNext()) {
if (item instanceof GroupItem) {
GroupItem childgroup = (GroupItem)item;
assignNames(childgroup.isInline() ? prefix : null, childgroup, true);
} else {
if (item.getName() == null) {
String childname = checkDirectName(item);
if (childname == null) {
childname = deriveName(item, true);
}
item.setName(NameUtils.toNameLead(assembleName(prefix, childname)));
}
}
}
}
}
/**
* Compute the complexity of a structure. In order to find the complexity of a structure all items of the structure
* must first be checked for inlining, which in turn requires checking their complexity. That makes this method
* mutually recursive with {@link #checkInline(DefinitionItem, int, List)}.
*
* @param group
* @param depth nesting depth
* @param defs list of generated definitions
* @return complexity (0, 1, or 2 for anything more than a single value)
*/
private int computeComplexity(GroupItem group, int depth, List defs) {
if (s_logger.isDebugEnabled()) {
s_logger.debug(SchemaUtils.getIndentation(depth) + "counting values for "
+ (group instanceof DefinitionItem ? "definition " : "group ")
+ SchemaUtils.describeComponent(group.getSchemaComponent()));
}
// count the actual values in the structure
int count = 0;
for (Item item = group.getFirstChild(); item != null; item = item.getNext()) {
// get the schema customization information
SchemaCustom custom = ((SchemaExtension)item.getSchemaComponent().getSchema().getExtension()).getCustom();
// handle inlining of references
if (item instanceof ReferenceItem) {
// first make sure the definition has been checked for inlining
ReferenceItem reference = (ReferenceItem)item;
DefinitionItem definition = reference.getDefinition();
checkInline(definition, depth + 1, defs);
if (definition.isInline()) {
// convert the reference to an inline copy of the definition
item = reference.inlineReference();
if (s_logger.isDebugEnabled()) {
s_logger.debug(SchemaUtils.getIndentation(depth) + "converted reference to "
+ SchemaUtils.describeComponent(definition.getSchemaComponent()) + " to inline group");
}
}
}
// handle actual item count, and inlining of child group (may be new, from converted reference)
if (item instanceof GroupItem) {
// check count for nested group
GroupItem grpitem = (GroupItem)item;
int grpcount = computeComplexity(grpitem, depth + 1, defs);
// avoid inlining if an enumeration, or an extension reference; or the nested group is optional or a
// collection and has more than one item (or a single item which is itself optional or a collection,
// which will be counted as multiple items); or the main group is a choice, and the nested group is a
// compositor with more than one element, and the first element is optional (or the first element child
// of that element, if inlined) - but allow override from customizations
boolean inline = true;
ComponentExtension exten = grpitem.getComponentExtension();
if (exten.isSeparateClass()) {
inline = false;
} else {
if (grpitem.isEnumeration()) {
inline = false;
} else if (grpitem.isCollection() || grpitem.isOptional()) {
if (grpcount > 1) {
inline = false;
} else {
// must be single child, but block inlining if that child is optional or an element or
// attribute with simple value (since we need an object for the optional part, separate
// from the simple value)
Item child;
GroupItem childgrp = grpitem;
boolean named = false;
while ((child = childgrp.getFirstChild()) instanceof GroupItem) {
childgrp = (GroupItem)child;
if (childgrp.isOptional()) {
inline = false;
break;
}
if (!named && childgrp.isTopmost()) {
int type = childgrp.getSchemaComponent().type();
named = type == SchemaBase.ATTRIBUTE_TYPE || type == SchemaBase.ELEMENT_TYPE;
}
}
if (named && child instanceof ValueItem) {
inline = false;
}
}
} else if (grpcount > 1 && group.getSchemaComponent().type() == SchemaBase.CHOICE_TYPE) {
// assume no inlining, but dig into structure to make sure first non-inlined element is required
inline = false;
Item child = grpitem.getFirstChild();
while (!child.isOptional()) {
if (child.getSchemaComponent().type() == SchemaBase.ELEMENT_TYPE &&
!(child instanceof GroupItem && ((GroupItem)child).isInline())) {
// required element with simple value or separate class, safe to inline
inline = true;
break;
} else if (child instanceof GroupItem) {
child = ((GroupItem)child).getFirstChild();
if (child == null) {
// empty group, safe to inline
inline = true;
break;
}
} else {
// required reference item, safe to inline
inline = true;
break;
}
}
}
}
if (inline) {
// inline the group
if (!grpitem.isInline()) {
grpitem.setInline(true);
}
count += grpcount;
if (s_logger.isDebugEnabled()) {
s_logger.debug(SchemaUtils.getIndentation(depth) + "inlining "
+ (grpitem instanceof DefinitionItem ? "definition " : "group ")
+ SchemaUtils.describeComponent(grpitem.getSchemaComponent()) + " with item count " +
count);
}
} else {
// force separate class for group
grpitem.setInline(false);
count++;
}
} else {
count++;
}
// bump up the complexity if the item is repeated (optionally inlining collection wrappers)
if (!custom.isNullCollectionAllowed() && item.isCollection()) {
count++;
}
}
return count > 1 ? 2 : count;
}
/**
* Check if a group consists only of a single non-repeating item, which is not an enumeration.
*
* @param group
* @return true
if simple group, false
if repeated, multiple, or enumeration
*/
private boolean isSimple(GroupItem group) {
if (group.isEnumeration() || group.getChildCount() > 1) {
return false;
} else {
for (Item item = group.getFirstChild(); item != null; item = item.getNext()) {
if (item.isCollection()) {
return false;
} else if (item instanceof GroupItem && !isSimple((GroupItem)item)) {
return false;
}
}
return true;
}
}
/**
* Convert nested groups which are not inlined to freestanding definitions. This calls itself recursively to process
* nested groups, except those nested within groups converted to definitions.
*
* @param group
* @param defs list of generated definitions
*/
private void convertToDefinitions(GroupItem group, List defs) {
Item item = group.getFirstChild();
while (item != null) {
if (item instanceof GroupItem) {
GroupItem childgrp = (GroupItem)item;
if (childgrp.isInline()) {
convertToDefinitions(childgrp, defs);
} else {
DefinitionItem def = childgrp.convertToDefinition();
def.setChecked(true);
OpenAttrBase ancestor = group.getSchemaComponent();
while (!ancestor.isGlobal()) {
ancestor = ancestor.getParent();
}
GlobalExtension global = (GlobalExtension)ancestor.getExtension();
PackageHolder pack = global.getPackage();
String clasname = def.getClassName();
NameConverter nconv = global.getNameConverter();
boolean useinner = global.isUseInnerClasses();
ClassDecorator[] decorators = global.getClassDecorators();
BindingHolder holder = m_bindingDirectory.getRequiredBinding(def.getSchemaComponent().getSchema());
ClassHolder clas = pack.addClass(clasname, nconv, decorators, useinner,
childgrp.isEnumeration(), holder);
def.setGenerateClass(clas);
defs.add(def);
s_logger.debug("Added definition class " + clas.getFullName());
}
}
item = item.getNext();
}
}
/**
* Check if a global definition structure is to be inlined. This method is mutually recursive with {@link
* #computeComplexity(GroupItem, int, List)}. The two methods together determine the inlining status of all items.
*
* @param def
* @param depth nesting depth
* @param defs list of generated definitions
*/
private void checkInline(DefinitionItem def, int depth, List defs) {
if (def.isChecked()) {
def.setReferenced(true);
} else if (def.isInline()) {
def.setChecked(true);
} else {
// flag checked and clear referenced to detect circular references
def.setChecked(true);
def.setReferenced(false);
if (s_logger.isDebugEnabled()) {
s_logger.debug(SchemaUtils.getIndentation(depth) + "checking inlining of definition "
+ SchemaUtils.describeComponent(def.getSchemaComponent())
+ (def.isInlineBlocked() ? " (inlining blocked)" : ""));
}
// inline references where appropriate, and count the values defined
int count = computeComplexity(def, depth, defs);
// determine the representation based on customizations or complexity
GlobalExtension exten = (GlobalExtension)def.getComponentExtension();
if (exten.isInlined()) {
if (def.isInlineBlocked()) {
s_logger.error("Cannot inline " + SchemaUtils.describeComponent(def.getSchemaComponent()) + " because of use as collection value, type for global element, or other usage");
} else {
def.setInline(true);
}
} else if (!exten.isSeparateClass() && !def.isInlineBlocked() && (exten.isPushInline() || count == 0 ||
(count == 1 && isSimple(def)) || (exten.isPreferInline() && def.getReferenceCount() == 1))) {
if (def.isReferenced()) {
s_logger.debug(SchemaUtils.getIndentation(depth) + "blocking inlining of self-referencing definition " +
SchemaUtils.describeComponent(def.getSchemaComponent()));
} else {
// set state as inlined
def.setInline(true);
if (s_logger.isDebugEnabled()) {
s_logger.debug(SchemaUtils.getIndentation(depth) + "inlining definition "
+ SchemaUtils.describeComponent(def.getSchemaComponent()) +
" with item count " + count);
}
// convert non-inlined child components to definitions if multiple use
if (def.getReferenceCount() > 1) {
convertToDefinitions(def, defs);
}
}
}
}
}
/**
* Set the name and namespace URI for a concrete <mapping> binding component. This is the same logic as used in
* the {@link StructureClassHolder} equivalent.
*
* @param qname qualified name to be set (null
if none)
* @param mapping concrete mapping definition
* @param holder binding containing the mapping
*/
private void setName(QName qname, MappingElementBase mapping, BindingHolder holder) {
if (qname != null) {
String name = qname.getName();
mapping.setName(name);
String uri = qname.getUri();
holder.addNamespaceUsage(uri);
if (!Utility.safeEquals(uri, holder.getElementDefaultNamespace())) {
mapping.setUri(uri);
}
}
}
/**
* Build the structure of items to be used in code generation for each global definition.
*
* @return constructed definition items
*/
private ArrayList buildItemStructures() {
ArrayList items = new ArrayList();
ItemVisitor itembuilder = new ItemVisitor();
int index = 0;
for (Iterator iter = m_validationContext.iterateSchemas(); iter.hasNext();) {
SchemaElement schema = (SchemaElement)iter.next();
m_validationContext.enterSchema(schema);
s_logger.info("Building item structure for schema " + ++index + ": " + schema.getResolver().getName());
int count = schema.getChildCount();
for (int i = 0; i < count; i++) {
SchemaBase child = schema.getChild(i);
if (child.getExtension() instanceof GlobalExtension) {
// create the definition
GlobalExtension global = (GlobalExtension)child.getExtension();
DefinitionItem definition = global.getDefinition();
if (definition == null) {
definition = itembuilder.buildGlobal((AnnotatedBase)child);
if (s_logger.isInfoEnabled()) {
s_logger.info("Constructed item structure for " + SchemaUtils.describeComponent(child)
+ ":\n" + definition.describe());
}
} else if (s_logger.isInfoEnabled()) {
s_logger.info("Found existing item structure for " + SchemaUtils.describeComponent(child)
+ ":\n" + definition.describe());
}
items.add(definition);
// set the names on the definition so they'll be available for inlining
NameConverter nconv = global.getNameConverter();
String dfltname = nconv.toBaseName(((INamed)child).getName());
if (!definition.isFixedName()) {
String name = global.getBaseName();
if (name == null) {
name = NameUtils.toNameLead(dfltname);
}
definition.setName(name);
}
if (!definition.isFixedClassName()) {
String name = global.getClassName();
if (name == null) {
name = global.getNameConverter().toJavaClassName(dfltname);
}
definition.setClassName(name);
}
// force class generation if required
if (global.isIncluded()) {
definition.setInlineBlocked(true);
if (s_logger.isDebugEnabled()) {
s_logger.debug("Forcing class generation for " + SchemaUtils.describeComponent(child));
}
}
}
}
}
return items;
}
/**
* Convert the item model used for global elements, if they're the only reference to a global type. This just
* prevents generating two separate classes, one for the type and one for the element.
*
* @param items global definition items
* @return map from base type definition to singleton element definition using that type
*/
private Map convertTypeIsomorphicElements(ArrayList items) {
// check for special (but common) case of single global element using global complexType
ArrayList checkitems = new ArrayList();
for (int i = 0; i < items.size(); i++) {
DefinitionItem definition = (DefinitionItem)items.get(i);
if (!definition.isPregenerated()) {
// record all simple element definition items separately for next pass
AnnotatedBase child = definition.getSchemaComponent();
if (child.type() == SchemaBase.ELEMENT_TYPE && definition.getChildCount() == 1) {
Item item = definition.getFirstChild();
if (item instanceof ReferenceItem) {
DefinitionItem basedef = ((ReferenceItem)item).getDefinition();
if (!basedef.isPregenerated()) {
AnnotatedBase comp = basedef.getSchemaComponent();
if (comp.type() != SchemaBase.SIMPLETYPE_TYPE) {
checkitems.add(definition);
}
}
}
}
}
}
// build map from type definition used by element to flag for single vs. multiple usage
Map usemap = new HashMap();
for (int i = 0; i < checkitems.size(); i++) {
// make sure the type definition is not using a pregenerated class (since can't be sure won't be reused)
ReferenceItem reference = (ReferenceItem)((DefinitionItem)checkitems.get(i)).getFirstChild();
DefinitionItem basedef = reference.getDefinition();
if (!basedef.hasDirectGenerateClass()) {
Boolean usedirect = (Boolean)usemap.get(basedef);
if (usedirect == null) {
// first time type was referenced, flag to use directly
usemap.put(basedef, Boolean.TRUE);
} else if (usedirect.booleanValue()) {
// second time type was referenced, flag to use separate classes
usemap.put(basedef, Boolean.FALSE);
}
}
}
// flag type-isomorphic elements as inlined, building map from type to isomorphic element definition
HashMap typeinstmap = new HashMap();
for (int i = 0; i < checkitems.size(); i++) {
DefinitionItem elemdef = (DefinitionItem)checkitems.get(i);
ReferenceItem reference = (ReferenceItem)elemdef.getFirstChild();
DefinitionItem basedef = reference.getDefinition();
if (!basedef.hasDirectGenerateClass() && ((Boolean)usemap.get(basedef)).booleanValue()) {
// single element definition using type, make sure abstract state matches on element and type
ElementElement element = (ElementElement)elemdef.getSchemaComponent();
AnnotatedBase refcomp = basedef.getSchemaComponent();
if (!(refcomp instanceof ComplexTypeElement) ||
((ComplexTypeElement)refcomp).isAbstract() == element.isAbstract()) {
// abstract state matches, force class for type but none for element
basedef.setInlineBlocked(true);
elemdef.setInlineBlocked(true);
typeinstmap.put(basedef, elemdef);
elemdef.setTypeIsomorphic(true);
}
}
}
// flag type-isomorphic elements as inlined, building map from type to isomorphic element definition
/* HashMap typeinstmap = new HashMap();
for (int i = 0; i < checkitems.size(); i++) {
DefinitionItem elemdef = (DefinitionItem)checkitems.get(i);
ReferenceItem reference = (ReferenceItem)elemdef.getFirstChild();
DefinitionItem basedef = reference.getDefinition();
if (!basedef.hasDirectGenerateClass() && basedef.getReferenceCount() == 1) {
// single element definition using type, force class for type but none for element
elemdef.setInlineBlocked(false);
elemdef.setInline(true);
typeinstmap.put(basedef, elemdef);
if (s_logger.isDebugEnabled()) {
s_logger.debug("Forcing inlining of type-isomorphic "
+ SchemaUtils.describeComponent(elemdef.getSchemaComponent()));
}
}
} */
return typeinstmap;
}
/**
* Build list of definitions to be generated. Where possible, references are converted to inline definitions. Where
* they can't be converted, new definitions are created for separate class generation.
*
* @param items
* @return definitions to be processed
*/
private ArrayList inlineReferences(ArrayList items) {
ArrayList defs = new ArrayList();
for (int i = 0; i < items.size(); i++) {
// make sure definition has been checked for inlining (may add new definitions directly to list)
DefinitionItem def = (DefinitionItem)items.get(i);
checkInline(def, 1, defs);
// add definition to list if not inlined or pregenerated or type-isomorphic
if (!def.isInline() && !def.hasDirectGenerateClass() && !def.isTypeIsomorphic()) {
GlobalExtension global = (GlobalExtension)def.getComponentExtension();
defs.add(def);
PackageHolder pack = global.getPackage();
String cname = def.getClassName();
NameConverter nconv = global.getNameConverter();
ClassDecorator[] decorators = global.getClassDecorators();
boolean userinner = global.isUseInnerClasses();
BindingHolder holder = m_bindingDirectory.getRequiredBinding(def.getSchemaComponent().getSchema());
ClassHolder clas = pack.addClass(cname, nconv, decorators, userinner, def.isEnumeration(), holder);
def.setGenerateClass(clas);
}
}
return defs;
}
/**
* Check if no-namespace namespace is used in any of the schemas.
*
* @return true
if no-namespace used, false
if not
*/
private boolean checkNoNamespacedUsed() {
SchemaNameVisitor visitor = new SchemaNameVisitor();
TreeWalker walker = new TreeWalker(null, new SchemaContextTracker());
for (Iterator iter = m_validationContext.iterateSchemas(); iter.hasNext();) {
walker.walkSchema((SchemaElement)iter.next(), visitor);
}
return visitor.isNoNamespaceUsed();
}
/**
* Build classes and bindings for all definitions, except those for singleton element definitions using a type.
*
* @param defs definitions to be generated
* @param typeinst map from base type definition to singleton element definition using that type
*/
private void buildClassesAndBindings(ArrayList defs, Map typeinst) {
// process all type definitions
for (int i = 0; i < defs.size(); i++) {
// compact again after inlining and converting extension references
DefinitionItem definition = (DefinitionItem)defs.get(i);
compactGroups(definition);
// build the binding component for this definition
ClassHolder clas = (ClassHolder)definition.getGenerateClass();
OpenAttrBase comp = definition.getSchemaComponent();
SchemaElement schema = comp.getSchema();
BindingHolder holder = m_bindingDirectory.getRequiredBinding(schema);
if (definition.isEnumeration()) {
// construct format and add to binding
FormatElement format = new FormatElement();
format.setTypeName(clas.getBindingName());
format.setQName(definition.getQName());
((EnumerationClassHolder)clas).setBinding(format);
m_bindingDirectory.addFormat(format);
} else {
// construct mapping element
MappingElementBase mapping = new MappingElement();
mapping.setClassName(clas.getBindingName());
if (comp.type() == SchemaBase.ELEMENT_TYPE) {
// abstract or concrete mapping for element
ElementElement element = (ElementElement)comp;
setName(element.getEffectiveQName(), mapping, holder);
mapping.setAbstract(element.isAbstract());
QName group = element.getSubstitutionGroup();
if (group != null) {
ElementElement base = m_validationContext.findElement(group);
DefinitionItem basedef = ((GlobalExtension)base.getExtension()).getDefinition();
mapping.setExtendsName(basedef.getGenerateClass().getFullName());
}
} else {
// abstract mapping for type definition or group
mapping.setAbstract(true);
QName qname = definition.getQName();
mapping.setTypeQName(qname);
String uri = qname.getUri();
if (uri != null) {
m_bindingDirectory.addTypeNameReference(holder, uri, schema);
}
}
// add the mapping to binding and set on class
holder.addMapping(mapping);
((StructureClassHolder)clas).setBinding(mapping);
DefinitionItem elementdef = (DefinitionItem)typeinst.get(definition);
if (elementdef != null) {
// create mapping for element name linked to type
ElementElement element = (ElementElement)elementdef.getSchemaComponent();
SchemaElement elschema = element.getSchema();
MappingElementBase elmapping = new MappingElement();
elmapping.setClassName(clas.getBindingName());
elmapping.setAbstract(element.isAbstract());
// handle linking to substitution group head using extends mapping
QName group = element.getSubstitutionGroup();
if (group != null) {
ElementElement base = m_validationContext.findElement(group);
DefinitionItem basedef = ((GlobalExtension)base.getExtension()).getDefinition();
elmapping.setExtendsName(basedef.getGenerateClass().getFullName());
}
// create single structure child invoking the type mapping
holder = m_bindingDirectory.getRequiredBinding(elschema);
setName(element.getEffectiveQName(), elmapping, holder);
StructureElement struct = new StructureElement();
QName qname = mapping.getTypeQName();
String uri = qname.getUri();
if (uri != null) {
m_bindingDirectory.addTypeNameReference(holder, uri, schema);
}
struct.setMapAsQName(qname);
elmapping.addChild(struct);
holder.addMapping(elmapping);
}
}
// set the definition for the class (and create any required secondary classes)
clas.buildDataStructure(definition, holder);
}
}
/**
* Initialize the bindings to be generated. This configures the binding directory, then creates the required
* bindings and sets their prefixes.
*/
private void initializeBindings() {
m_bindingDirectory = new BindingOrganizer(false, false, false, true, true, true);
Map namebindings = new HashMap();
Map nsbindings = new HashMap();
for (Iterator iter = m_validationContext.iterateSchemas(); iter.hasNext();) {
// check binding handling for schema
SchemaElement schema = (SchemaElement)iter.next();
SchemaExtension exten = (SchemaExtension)schema.getExtension();
String uri = schema.getEffectiveNamespace();
String prefix = exten.getPrefix();
BindingHolder holder;
String name = exten.getBindingFileName();
boolean dflt = schema.isElementQualifiedDefault() || uri == null;
if (name != null) {
// specific binding name, create the binding or link to existing
holder = (BindingHolder)namebindings.get(name);
if (holder == null) {
holder = m_bindingDirectory.getBinding(schema);
if (holder == null) {
holder = m_bindingDirectory.addBinding(schema, uri, prefix, dflt);
}
holder.setFileName(name);
namebindings.put(name, holder);
} else {
m_bindingDirectory.addBindingObject(schema, holder);
}
} else if (exten.isForceBinding()) {
// binding needed for this schema, create it
holder = m_bindingDirectory.addBinding(schema, uri, prefix, dflt);
} else {
// need binding for namespace, create or link to existing
holder = (BindingHolder)nsbindings.get(uri);
if (holder == null) {
holder = m_bindingDirectory.addBinding(schema, uri, prefix, dflt);
nsbindings.put(uri, holder);
} else {
m_bindingDirectory.addBindingObject(schema, holder);
}
}
}
}
/**
* Generate the data model. This first builds a representation of all the data items from the schema definitions,
* then determines which items can be inlined and which need separate class representations, and finally builds the
* actual data model class and binding definitions.
*
* @param verbose verbose output flag
* @param usenns no-namespace namespace used flag
* @param typemap map from qualified name of type (or attributeGroup, or group) to pregenerated mapping definition
* @param elemmap map from qualified name of element to pregenerated mapping definition
* @return root package for binding
*/
public String buildDataModel(boolean verbose, boolean usenns, Map elemmap, Map typemap) {
// build the item structure for each definition and link to pregenerated classes
ArrayList items = buildItemStructures();
for (int i = 0; i < items.size(); i++) {
DefinitionItem definition = (DefinitionItem)items.get(i);
QName qname = definition.getQName();
TypeData clas = null;
switch (definition.getSchemaComponent().type()) {
case SchemaBase.ATTRIBUTEGROUP_TYPE:
case SchemaBase.COMPLEXTYPE_TYPE:
case SchemaBase.GROUP_TYPE:
case SchemaBase.SIMPLETYPE_TYPE:
{
clas = (TypeData)typemap.get(qname);
break;
}
case SchemaBase.ELEMENT_TYPE:
{
clas = (TypeData)elemmap.get(qname);
break;
}
}
if (clas != null) {
definition.setGenerateClass(clas);
definition.setInlineBlocked(true);
if (s_logger.isDebugEnabled()) {
s_logger.debug("Found class " + clas.getFullName() + " for " + qname);
}
}
}
// convert elements based on types and compact all items
Map typeinstmap = convertTypeIsomorphicElements(items);
for (int i = 0; i < items.size(); i++) {
compactGroups((DefinitionItem)items.get(i));
}
// inline references where appropriate
initializeBindings();
ArrayList defs = inlineReferences(items);
// assign class and property names for all items
for (int i = 0; i < defs.size(); i++) {
DefinitionItem def = (DefinitionItem)defs.get(i);
compactGroups(def);
assignNames(null, def, true);
if (s_logger.isInfoEnabled()) {
s_logger.info("After assigning names for "
+ SchemaUtils.describeComponent(def.getSchemaComponent()) + ":\n" + def.describe());
}
}
// convert extension references for all global type definitions
for (int i = 0; i < defs.size(); i++) {
((DefinitionItem)defs.get(i)).convertTypeReference();
}
// classify all the items by form of content
for (int i = 0; i < items.size(); i++) {
DefinitionItem def = (DefinitionItem)items.get(i);
def.classifyContent();
if (s_logger.isInfoEnabled()) {
s_logger.info("After inlining and classification for " +
SchemaUtils.describeComponent(def.getSchemaComponent())
+ ":\n" + def.describe());
}
}
// build the actual class and binding structure
setDefaultPrefixes(m_validationContext.iterateSchemas());
buildClassesAndBindings(defs, typeinstmap);
// build the actual classes
AST ast = AST.newAST(AST.JLS3);
ArrayList packs = m_packageDirectory.getPackages();
PackageHolder rootpack = null;
for (int i = 0; i < packs.size(); i++) {
PackageHolder pack = ((PackageHolder)packs.get(i));
if (pack.getClassCount() > 0) {
if (rootpack == null) {
PackageHolder scan = pack;
while (scan != null) {
if (scan.getClassCount() > 0 || scan.getSubpackageCount() > 1) {
rootpack = scan;
}
scan = scan.getParent();
}
}
pack.generate(verbose, ast, m_bindingDirectory);
}
}
// return the root package name
if (rootpack == null) {
rootpack = m_packageDirectory.getPackage("");
}
return rootpack.getName();
}
/**
* Get the binding directory.
*
* @return directory
*/
public BindingOrganizer getBindingDirectory() {
return m_bindingDirectory;
}
/**
* Write the binding definitions file(s). This method can only be used after
* {@link #buildDataModel(boolean, boolean, Map, Map)} is called.
*
* @param name root binding definition file name (use customization, or default, if null
)
* @param pack target package for binding (null
if unspecified)
* @param pregens pregenerated bindings to be included in root binding
* @param handler validation error and code generation problem handler
* @throws JiBXException
* @throws IOException
*/
public void writeBindings(String name, String pack, List pregens, ProblemHandler handler)
throws JiBXException, IOException {
if (name == null) {
name = m_global.getBindingFileName();
if (name == null) {
name = "binding.xml";
}
}
m_rootHolder = m_bindingDirectory.configureFiles(name, pack, pregens);
IClassLocator iloc = new DummyClassLocator();
org.jibx.binding.model.ValidationContext vctx = new org.jibx.binding.model.ValidationContext(iloc);
if (m_bindingDirectory.validateBindings(m_rootHolder, m_targetDir, vctx)) {
m_bindingDirectory.writeBindings(m_targetDir);
} else {
reportBindingProblems(vctx, handler);
throw new JiBXException("Terminating due to errors in bindings");
}
}
/**
* Get the root binding definition. This is only allowed after the call to
* {@link #writeBindings(String, String, List, ProblemHandler)}.
*
* @return root binding element
*/
public BindingElement getRootBinding() {
return m_rootHolder.getBinding();
}
/**
* Add the schemas specified by customizations to the set to be loaded.
*
* @param base root URL for schemas
* @param dir root directory for schemas
* @param custom schema set customization
* @param fileset set of schema files to be loaded
* @throws MalformedURLException
*/
/* private static void addCustomizedSchemas(URL base, File dir, SchemasetCustom custom, InsertionOrderedSet fileset)
throws MalformedURLException {
// first check for name match patterns supplied
String[] names = custom.getNames();
if (names != null) {
for (int i = 0; i < names.length; i++) {
SchemaNameFilter filter = new SchemaNameFilter();
String name = names[i];
filter.setPattern(name);
s_logger.debug("Matching file names to schemaset pattern '" + name + '\'');
String[] matches = dir.list(filter);
for (int j = 0; j < matches.length; j++) {
String match = matches[j];
fileset.add(new UrlResolver(match, new URL(base, match)));
s_logger.debug("Added schema from schemaset pattern match: " + match);
}
}
}
// next check all child customizations
LazyList childs = custom.getChildren();
for (int i = 0; i < childs.size(); i++) {
Object child = childs.get(i);
if (child instanceof SchemaCustom) {
String name = ((SchemaCustom)child).getName();
if (name != null) {
try {
fileset.add(new UrlResolver(name, new URL(base, name)));
s_logger.debug("Added schema from customizations: " + name);
} catch (MalformedURLException e) {
System.out.println("Error adding schema from customizations: " + name);
}
}
} else if (child instanceof SchemasetCustom) {
addCustomizedSchemas(base, dir, (SchemasetCustom)child, fileset);
}
}
}
*/
/**
* Get the package directory used for code generation.
*
* @return directory
*/
public PackageOrganizer getPackageDirectory() {
return m_packageDirectory;
}
/**
* Generate code from a list of schemas.
*
* @param verbose verbose output flag
* @param usens namespace to be used when no schemas with namespaces are being generated (null
if not
* specified)
* @param dfltpkg default package for no-namespace schemas (null
if not specified)
* @param bindname name for root binding (null
if not specified)
* @param fileset list of resolvers for schemas to be generated
* @param inclpaths list of paths for bindings to be used for matching schema definitions (empty if none)
* @param model file to be used for dumping generated data model (null
if none)
* @param handler validation error and code generation problem handler
* @return true
if successful, false
if failure
* @throws JiBXException
* @throws IOException
*/
public boolean generate(boolean verbose, String usens, String dfltpkg, String bindname, List fileset,
List inclpaths, File model, ProblemHandler handler) throws JiBXException, IOException {
// load the full set of schemas
SchemaElement[] schemas = ValidationUtils.load(fileset, usens, m_validationContext);
if (!m_validationContext.reportProblems(handler)) {
if (dfltpkg == null) {
dfltpkg = "dflt";
}
if (customizeSchemas(dfltpkg, handler)) {
// report on schemas loaded
StringBuffer buff = new StringBuffer();
buff.append("Loaded and validated " + fileset.size() + " specified schema(s)");
int refcount = m_validationContext.getSchemaCount() - fileset.size();
if (refcount > 0) {
buff.append(" and " + refcount + " referenced schema(s)");
}
if (verbose) {
buff.append(':');
handler.report(buff.toString());
listSchemas(schemas, m_validationContext, handler);
} else {
handler.report(buff.toString());
}
// apply the schema customizations and revalidate modified schema
applyAndNormalize();
pruneDefinitions();
ValidationUtils.validateSchemas(schemas, m_validationContext);
if (!m_validationContext.reportProblems(handler)) {
// load any precompiled bindings
Map elemmap = new HashMap();
Map typemap = new HashMap();
List includes = new ArrayList();
boolean usenns = false;
for (Iterator iter = inclpaths.iterator(); iter.hasNext();) {
String path = (String)iter.next();
URL url;
File file = null;
if (ClasspathUrlExtender.isClasspathUrl(path)) {
url = ClasspathUrlExtender.buildURL(null, path);
} else {
file = new File(path);
if (!file.isAbsolute())
file = new File(m_targetDir, path);
url = file.toURI().toURL();
}
try {
BindingElement binding = processPregeneratedBinding(url, elemmap, typemap, handler);
if (!usenns) {
usenns = checkNoNamespace(binding);
}
try {
org.jibx.binding.model.IncludeElement include =
new org.jibx.binding.model.IncludeElement();
if (file == null) {
include.setIncludePath(path);
} else {
include.setIncludePath(relativeFilePath(m_targetDir, file));
}
include.setPrecompiled(true);
includes.add(include);
} catch (IOException e) {
handler.terminate("Error processing path " + path, e);
return false;
}
} catch (IOException e) {
handler.terminate("Pregenerated binding not found with path: " + path);
return false;
}
}
// generate code and bindings, using any existing bindings supplied
buildDataModel(verbose, usenns, elemmap, typemap);
List packages = m_packageDirectory.getPackages();
PackageHolder top = null;
for (Iterator iter = packages.iterator(); iter.hasNext();) {
PackageHolder holder = (PackageHolder)iter.next();
if (holder.getClassCount() > 0) {
if (top == null) {
top = holder;
} else {
// find common parent package of all packages so far processed
outer: while (true) {
PackageHolder scan = holder;
while (true) {
if (scan == top) {
break outer;
} else if (scan.getName().length() < top.getName().length()) {
top = top.getParent();
break;
} else {
scan = scan.getParent();
}
}
}
}
}
}
String rootpack = top == null ? null : top.getName();
writeBindings(bindname, rootpack, includes, handler);
// print the total number of classes generated
listGeneratedPackages(packages, handler);
// check if prior data model dump file provided for comparison
if (model != null) {
if (model.exists()) {
// read the data model file
BufferedReader reader = new BufferedReader(new FileReader(model));
StringObjectPair[] image = DataModelUtils.readImage(reader);
String diff = DataModelUtils.imageDiff(image, DataModelUtils.getImage(m_packageDirectory));
if (diff == null) {
handler.report("No difference found from data model file " + model.getPath());
} else {
handler.report(diff);
}
} else {
handler.report("Difference data model file " + model.getPath() + " not found");
}
}
// check if dump file to be output
if (model != null) {
// delete the file if it already exists
if (model.exists()) {
model.delete();
}
model.createNewFile();
// now print out the class details by package
BufferedWriter writer = new BufferedWriter(new FileWriter(model));
DataModelUtils.writeImage(m_packageDirectory, writer);
writer.close();
}
} else {
handler.terminate("Terminating due to errors in normalized schemas - please report the problem");
return false;
}
} else {
handler.terminate("Terminating due to errors in customized schemas");
return false;
}
} else {
handler.terminate("Terminating due to errors in input schemas");
return false;
}
return true;
}
/**
* List the schemas in use.
*
* @param schemas
* @param vctx
* @param handler
*/
private static void listSchemas(SchemaElement[] schemas, ValidationContext vctx, ProblemHandler handler) {
Set topset = new HashSet();
for (int i = 0; i < schemas.length; i++) {
SchemaElement schema = schemas[i];
topset.add(schema);
handler.report(" " + "top-level schema " + schema.getResolver().getName());
}
for (Iterator iter = vctx.iterateSchemas(); iter.hasNext();) {
SchemaElement schema = (SchemaElement)iter.next();
if (!topset.contains(schema)) {
handler.report(" " + "referenced schema " + schema.getResolver().getName());
}
}
}
/**
* List the number of classes in each package, and the totals.
*
* @param packages
* @param handler
*/
private static void listGeneratedPackages(List packages, ProblemHandler handler) {
int top = 0;
int total = 0;
StringBuffer buff = new StringBuffer();
for (Iterator iter = packages.iterator(); iter.hasNext();) {
PackageHolder hold = (PackageHolder)iter.next();
int count = hold.getClassCount();
if (count > 0) {
int topcount = hold.getTopClassCount();
top += topcount;
buff.setLength(0);
buff.append("Generated " + topcount + " top-level classes ");
if (count != topcount) {
buff.append("(plus " + (count-topcount) + " inner classes) ");
}
buff.append("in package " + hold.getName());
handler.report(buff.toString());
total += count;
}
}
if (total == top) {
handler.report("Total classes in model: " + top);
} else {
handler.report("Total top-level classes in model: " + top);
handler.report("Total classes (including inner classes) in model: " + total);
}
}
/**
* Build class data for pregenerated class.
*
* @param cname class name
* @param simple simple value flag
* @return data
*/
private static TypeData buildClassData(String cname, boolean simple) {
String bname = cname;
String fname = bname.replace('$', '.');
return new TypeData(fname, bname, true, simple);
}
/**
* Accumulate all format and mapping definitions, including those found in included bindings. For each abstract
* mapping or named format found, the type name is associated with the class data in the type map; for each concrete
* mapping found, the element name (and namespace) is associated with the class data in the element map. Included
* bindings are handled with recursive calls.
*
* @param binding binding definition root
* @param elemmap map from element qualified name to class data
* @param typemap map from type qualified name to class data
*/
private static void accumulateBindingDefinitions(BindingElement binding, Map elemmap, Map typemap) {
ArrayList childs = binding.topChildren();
for (int i = 0; i < childs.size(); i++) {
ElementBase element = (ElementBase)childs.get(i);
if (element.type() == ElementBase.INCLUDE_ELEMENT) {
// use recursive call to add nested definitions in included binding
BindingElement inclbinding = ((org.jibx.binding.model.IncludeElement)element).getBinding();
if (inclbinding != null) {
accumulateBindingDefinitions(inclbinding, elemmap,
typemap);
}
} else if (element.type() == ElementBase.MAPPING_ELEMENT) {
// handle mapping as type if abstract with type name, or as element if concrete
MappingElementBase mapping = (MappingElementBase)element;
String cname = mapping.getClassName();
TypeData data = buildClassData(cname, false);
if (mapping.isAbstract()) {
QName qname = mapping.getTypeQName();
if (qname != null) {
typemap.put(qname, data);
if (s_logger.isDebugEnabled()) {
s_logger.debug("Added class " + cname + " for type " + qname);
}
}
} else {
QName qname = new QName(mapping.getNamespace().getUri(), mapping.getName());
elemmap.put(qname, data);
if (s_logger.isDebugEnabled()) {
s_logger.debug("Added class " + cname + " for element " + qname);
}
}
} else if (element.type() == ElementBase.FORMAT_ELEMENT) {
// add named format as simple type definition
FormatElement format = (FormatElement)element;
String name = format.getTypeName();
if (name != null) {
TypeData data = buildClassData(name, true);
typemap.put(format.getQName(), data);
}
}
}
}
/**
* Load and validate binding and process all format and mapping definitions, including those in included bindings.
*
* @param url binding definition path
* @param elemmap map from element qualified name to class data
* @param typemap map from type qualified name to class data
* @param handler validation error and problem handler
* @return binding
* @throws JiBXException
* @throws IOException
*/
public static BindingElement processPregeneratedBinding(URL url, Map elemmap, Map typemap, ProblemHandler handler)
throws JiBXException, IOException {
// get binding definition file name from path
String name = "";
String path = url.getPath();
if (path != null) {
name = path;
int split = Math.max(name.lastIndexOf('/'), name.lastIndexOf('\\'));
if (split > 0) {
name = name.substring(split+1);
}
split = name.lastIndexOf('.');
name = name.substring(0, split);
}
// construct object model for binding
org.jibx.binding.model.ValidationContext vctx =
new org.jibx.binding.model.ValidationContext(new DummyClassLocator());
BindingElement binding = BindingElement.readBinding(url.openStream(), name, null, true,
vctx);
binding.setBaseUrl(url);
vctx.setBindingRoot(binding);
// validate the binding definition
binding.runValidation(vctx);
// list validation errors
if (vctx.getProblems().size() > 0) {
reportBindingProblems(vctx, handler);
if (vctx.getErrorCount() > 0 || vctx.getFatalCount() > 0) {
throw new JiBXException("Errors in binding");
}
}
// add all the mapping and format definitions in binding to qualified name maps
accumulateBindingDefinitions(binding, elemmap, typemap);
return binding;
}
/**
* Report problems found in binding.
*
* @param vctx
* @param handler
*/
private static void reportBindingProblems(org.jibx.binding.model.ValidationContext vctx, ProblemHandler handler) {
StringBuffer buff = new StringBuffer();
ArrayList problems = vctx.getProblems();
for (int i = 0; i < problems.size(); i++) {
org.jibx.binding.model.ValidationProblem prob =
(org.jibx.binding.model.ValidationProblem)problems.get(i);
buff.setLength(0);
buff.append(prob.getSeverity() >= org.jibx.binding.model.ValidationProblem.ERROR_LEVEL ?
"Error: " : "Warning: ");
buff.append(prob.getDescription());
handler.report(buff.toString());
}
}
/**
* Find the steps in the canonical path to a file.
*
* @param file
* @return steps
* @throws IOException
*/
public static List findPathSteps(File file) throws IOException {
LinkedList steps = new LinkedList();
steps.addFirst(file.getName());
File parent = file;
while ((parent = parent.getParentFile()) != null) {
steps.addFirst(parent.getName());
}
return steps;
}
/**
* Construct a relative file path.
*
* @param dir start directory for path
* @param file supplied file path
* @return relative file path
* @throws IOException
*/
public static String relativeFilePath(File dir, File file) throws IOException {
// first find any common portion of the paths
List dirsteps = findPathSteps(dir);
List filesteps = findPathSteps(file);
// scan past any common portion of the paths
Iterator diriter = dirsteps.iterator();
Iterator fileiter = filesteps.iterator();
boolean abs = dir.isAbsolute();
boolean diff = false;
String filestep = null;
while (diriter.hasNext() && fileiter.hasNext()) {
String dirstep = (String)diriter.next();
filestep = (String)fileiter.next();
if (!dirstep.equals(filestep)) {
diff = true;
break;
} else {
abs = false;
}
}
if (!diriter.hasNext())
filestep = (String)fileiter.next(); // Relative to target directory
// check whether relative path can be used
if (abs) {
return file.getCanonicalPath();
} else {
// create relative path to common ancestor directory
StringBuffer buff = new StringBuffer();
if (diff) {
buff.append("..");
buff.append(File.separatorChar);
}
while (diriter.hasNext()) {
buff.append("..");
buff.append(File.separatorChar);
diriter.next();
}
// extend path to the actual file
buff.append(filestep);
while (fileiter.hasNext()) {
buff.append(File.separatorChar);
buff.append(fileiter.next());
}
return buff.toString();
}
}
/**
* Check if a binding definition uses the no-namespace namespace. This calls itself to recursively check on included
* bindings.
*
* @param binding
* @return true
if no-namespace namespace used, false
if not
*/
public static boolean checkNoNamespace(BindingElement binding) {
loop: for (Iterator childiter = binding.topChildIterator(); childiter.hasNext();) {
ElementBase child = (ElementBase)childiter.next();
switch (child.type()) {
case ElementBase.NAMESPACE_ELEMENT:
if (((NamespaceElement)child).getPrefix() == null) {
return true;
}
break;
case ElementBase.INCLUDE_ELEMENT:
if (checkNoNamespace(((org.jibx.binding.model.IncludeElement)child).getBinding())) {
return true;
}
break;
case ElementBase.MAPPING_ELEMENT:
break loop;
}
}
return false;
}
/**
* Run the binding generation using command line parameters.
*
* @param args
* @throws Exception
*/
public static void main(String[] args) throws Exception {
TreeWalker.setLogging(Level.ERROR);
CodeGenCommandLine parms = new CodeGenCommandLine();
if (args.length > 0 && parms.processArgs(args)) {
// build set of schemas specified on command line (including via wildcards)
final InsertionOrderedSet fileset = new InsertionOrderedSet();
URL base = parms.getSchemaRoot();
File basedir = parms.getSchemaDir();
List errors = ResourceMatcher.matchPaths(basedir, base, parms.getExtraArgs(),
new ResourceMatcher.ReportMatch() {
public void foundMatch(String path, URL url) {
fileset.add(new UrlResolver(path, url));
}
});
if (errors.size() > 0) {
for (Iterator iter = errors.iterator(); iter.hasNext();) {
s_logger.error(iter.next());
}
System.exit(2);
}
// handle code generation from schemas
ProblemMultiHandler handler = new ProblemMultiHandler();
handler.addHandler(new ProblemConsoleLister());
handler.addHandler(new ProblemLogLister(s_logger));
CodeGen inst = new CodeGen(parms.getCustomRoot(), parms.getSchemaRoot(),
parms.getGeneratePath());
inst.generate(parms.isVerbose(), parms.getUsingNamespace(), parms.getNonamespacePackage(),
parms.getBindingName(), fileset.asList(), parms.getIncludePaths(), parms.getModelFile(), handler);
} else {
if (args.length > 0) {
s_logger.error("Terminating due to command line or customization errors");
} else {
parms.printUsage();
}
System.exit(1);
}
}
/**
* Visitor to count the number of values in a definition.
*/
private static class ValueCountVisitor extends SchemaVisitor
{
// count of attributes, elements, atttributeGroup references, and group references
private int m_count;
public int getCount() {
return m_count;
}
public boolean visit(AttributeElement node) {
m_count++;
return false;
}
public boolean visit(AttributeGroupRefElement node) {
m_count++;
return false;
}
public boolean visit(ElementElement node) {
m_count++;
return false;
}
public boolean visit(GroupRefElement node) {
m_count++;
return false;
}
}
/**
* Visitor for checking element namespace usage in schema definitions. This just accumulates the set of namespaces
* used by element definitions.
*/
private static class SchemaNameVisitor extends SchemaVisitor
{
/** Set of namespace URIs used by element definitions. */
private final Set m_uris;
/**
* Constructor.
*/
public SchemaNameVisitor() {
m_uris = new HashSet();
}
/**
* Check if a single namespace is used for all element definitions.
*
* @return true
if single namespace, false
if not
*/
public boolean isSingleNamespace() {
return m_uris.size() <= 1;
}
/**
* Check if the no-namespace namespace is used by one or more elements.
*
* @return true
if no-namespace used, false
if not
*/
public boolean isNoNamespaceUsed() {
return m_uris.contains(null);
}
/**
* Accumulate namespace used by element definition.
*
* @param node
*/
public void exit(ElementElement node) {
QName qname = node.getQName();
if (qname != null) {
m_uris.add(qname.getUri());
}
super.exit(node);
}
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy