org.jibx.schema.codegen.Item Maven / Gradle / Ivy
/*
* Copyright (c) 2007-2008, 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 org.jibx.schema.SchemaUtils;
import org.jibx.schema.codegen.custom.ComponentExtension;
import org.jibx.schema.elements.AnnotatedBase;
import org.jibx.schema.elements.CommonTypeDerivation;
import org.jibx.schema.elements.ElementElement;
import org.jibx.schema.elements.SchemaBase;
/**
* Base class for code generation items. Each instance corresponds to a particular schema component, and this base class
* tracks that schema component (by way of the extension information), along with related details and linkage
* information. The linkage uses embedded list links, which allows replacing one instance with another with minimal
* overhead.
*
* @author Dennis M. Sosnoski
*/
public abstract class Item
{
/** Corresponding schema component extension. */
private final ComponentExtension m_componentExtension;
/** Flag for topmost item associated with a particular schema component. */
private final boolean m_topmost;
/** Flag for an optional item. In cases where multiple items are associated with the same schema component, this
is only meaningful for the topmost item. */
private final boolean m_optional;
/** Flag for a collection item. In cases where multiple items are associated with the same schema component, this
is only meaningful for the topmost item. */
private final boolean m_collection;
/** Flag for a nillable item. In cases where multiple items are associated with the same schema component, this
is only meaningful for the topmost item. */
private final boolean m_nillable;
/** Item is handled by subclassing flag. */
private boolean m_implicit;
/** Containing group item. */
private GroupItem m_parent;
/** Next item in list (null
if none). */
protected Item m_next;
/** Preceding item in list (null
if none). */
protected Item m_last;
/** Actual name to be used for item (null
if to be inherited). */
private String m_name;
/**
* Basic constructor. This uses the schema component to determine all information other than the parent item group,
* including the optional/nillable/collection flags. As a special case, if the parent group is associated with the
* same component this sets all three of these flags false
to avoid redundant handling.
*
* @param comp schema component
* @param parent containing group (null
if a top-level group)
*/
protected Item(AnnotatedBase comp, GroupItem parent) {
// save basic values
m_componentExtension = (ComponentExtension)comp.getExtension();
m_parent = parent;
// set other values based on extension and component information
if (parent == null || parent.getSchemaComponent() != comp) {
m_topmost = true;
m_optional = m_componentExtension.isOptional();
m_collection = m_componentExtension.isRepeated();
m_nillable = comp.type() == SchemaBase.ELEMENT_TYPE && ((ElementElement)comp).isNillable();
} else {
m_topmost = m_optional = m_collection = m_nillable = false;
}
m_name = m_componentExtension.getBaseName();
}
/**
* Copy constructor. This creates a copy with a new parent.
*
* @param original
* @param ref reference (for name override; null
if none)
* @param ext component extension to be linked with copy
* @param parent (non-null
)
*/
protected Item(Item original, Item ref, ComponentExtension ext, GroupItem parent) {
m_componentExtension = ext;
m_parent = parent;
if (parent == null || parent.getSchemaComponent() != m_componentExtension.getComponent()) {
m_topmost = true;
Item top = original.getTopmost();
m_optional = top.m_optional;
m_collection = top.m_collection;
m_nillable = top.m_nillable;
} else {
m_topmost = m_optional = m_collection = m_nillable = false;
}
if (ref == null || original.isFixedName()) {
m_name = original.m_name;
} else {
m_name = ref.m_name;
}
}
/**
* Replace the parent for this item.
*
* @param parent
*/
protected void reparent(GroupItem parent) {
m_parent = parent;
}
/**
* Get schema component corresponding to this item.
*
* @return schema component
*/
public AnnotatedBase getSchemaComponent() {
return (AnnotatedBase)m_componentExtension.getComponent();
}
/**
* Get schema component annotation corresponding to this item.
*
* @return schema component
*/
public ComponentExtension getComponentExtension() {
return m_componentExtension;
}
/**
* Get containing group item.
*
* @return group (null
if a top-level group)
*/
public GroupItem getParent() {
return m_parent;
}
/**
* Check if the name is fixed by configuration.
*
* @return true
if fixed, false
if not
*/
public boolean isFixedName() {
return m_componentExtension.getBaseName() != null;
}
/**
* Get effective item name, applying inheritance if necessary.
*
* @return name
*/
public String getEffectiveName() {
Item item = this;
while (item.m_name == null) {
item = item.m_parent;
if (item == null) {
throw new IllegalStateException("Inherited name with nothing to inherit");
}
}
return item.m_name;
}
/**
* Get name set directly for this item.
*
* @return name (null
if to be inherited)
*/
public String getName() {
return m_name;
}
/**
* Set name directly for this item. It is an error to call this method if the name is fixed.
*
* @param name (null
if to be inherited)
*/
public void setName(String name) {
if (isFixedName()) {
throw new IllegalStateException("Internal error - attempt to change configured name");
} else {
m_name = name;
}
}
/**
* Get next item in list.
*
* @return next
*/
public Item getNext() {
return m_next;
}
/**
* Check if topmost item for a particular schema component. The methods {@link #isCollection()},
* {@link #isOptional()}, {@link GroupItem#isAllOptional()}, {@link GroupItem#isAttributePresent()},
* {@link GroupItem#isContentPresent()}, and {@link GroupItem#isElementPresent()} are all only meaningful for the
* topmost item associated with a schema component.
*
* @return topmost
*/
public boolean isTopmost() {
return m_topmost;
}
/**
* Get the topmost item associated with the same schema component as this item. The methods {@link
* #isCollection()}, {@link #isOptional()}, {@link GroupItem#isAllOptional()}, {@link
* GroupItem#isAttributePresent()}, {@link GroupItem#isContentPresent()}, and {@link GroupItem#isElementPresent()}
* are all only meaningful for the topmost item associated with a schema component.
*
* @return topmost
*/
public Item getTopmost() {
Item top = this;
while (!top.isTopmost()) {
top = top.m_parent;
}
return top;
}
/**
* Check if item is optional. This method is only meaningful for the topmost item associated with a particular
* schema component (those for which {@link #isTopmost()} returns true
).
*
* @return optional
*/
public boolean isOptional() {
return m_optional;
}
/**
* Check if item is ignored. This method is only meaningful for the topmost item associated with a particular
* schema component (those for which {@link #isTopmost()} returns true
).
*
* @return optional
*/
public boolean isIgnored() {
return getComponentExtension().isIgnored();
}
/**
* Check if a collection item. This method is only meaningful for the topmost item associated with a particular
* schema component (those for which {@link #isTopmost()} returns true
).
*
* @return true
if collection
*/
public boolean isCollection() {
return m_collection;
}
/**
* Check if the item is represented implicitly by subclassing.
*
* @return implicit
*/
public boolean isImplicit() {
return m_implicit;
}
/**
* Set item represented implicitly by subclassing flag.
*
* @param implicit
*/
public void setImplicit(boolean implicit) {
m_implicit = implicit;
}
/**
* Copy the item under a different parent. This is intended for replacing a reference with a copy, and allows the
* reference to override settings from the original.
*
* @param ref reference (for overrides to copy; null
if none)
* @param parent
* @return copy
*/
protected abstract Item copy(Item ref, GroupItem parent);
/**
* Find the nearest ancestor group which relates to a different schema component.
*
* @return ancestor with different schema component, or null
if none
*/
protected GroupItem findDisjointParent() {
AnnotatedBase comp = getSchemaComponent();
GroupItem parent = getParent();
while (parent != null && parent.getSchemaComponent() == comp) {
parent = parent.getParent();
}
return parent;
}
/**
* Classify the content of this item as attribute, element, and/or character data content, and as requiring content
* of some form if appropriate. This needs to be done as a separate step after construction in order to handle
* references, which must assume the content of the definition, and also to work after inlining. This base class
* implementation does the classification based solely on the schema component type. Any subclasses which override
* this method should generally call the base class implementation before doing their own classification handling,
* unless they use a substitute component.
*/
protected void classifyContent() {
if (m_topmost) {
// find the parent group with a different schema component
GroupItem parent = findDisjointParent();
if (parent != null) {
// flag content type for that parent group
AnnotatedBase comp = getSchemaComponent();
switch (comp.type()) {
case SchemaBase.ANY_TYPE:
case SchemaBase.ELEMENT_TYPE:
// set required component present if appropriate
if (!m_optional) {
parent.forceRequiredPresent();
}
// fall through for common handling
case SchemaBase.GROUP_TYPE:
parent.forceElementPresent();
break;
case SchemaBase.EXTENSION_TYPE:
CommonTypeDerivation deriv = (CommonTypeDerivation)comp;
if (!deriv.isComplexType()) {
parent.forceContentPresent();
parent.forceRequiredPresent();
}
break;
case SchemaBase.ANYATTRIBUTE_TYPE:
case SchemaBase.ATTRIBUTE_TYPE:
// set required component present if appropriate
if (!m_optional) {
parent.forceRequiredPresent();
}
// fall through for common handling
case SchemaBase.ATTRIBUTEGROUP_TYPE:
parent.forceAttributePresent();
break;
}
}
}
}
/**
* Generate a description of the item. For items with nested items this will show the complete structure.
*
* @param depth current nesting depth
* @param classified include classification details flag
* @return description
*/
protected abstract String describe(int depth, boolean classified);
/**
* Generate the standard leading text for description of the item.
*
* @param depth current nesting depth
* @return leading text for description
*/
protected String leadString(int depth) {
StringBuffer buff = new StringBuffer(SchemaUtils.getIndentation(depth));
if (isOptional()) {
buff.append("optional ");
}
if (isCollection()) {
buff.append("repeating ");
}
return buff.toString();
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy