All Downloads are FREE. Search and download functionalities are using the official Maven repository.

org.jibx.schema.codegen.GroupItem Maven / Gradle / Ivy

/*
 * Copyright (c) 2007-2009, Dennis M. Sosnoski. All rights reserved.
 * 
 * Redistribution and use in source and binary forms, with or without modification, are permitted provided that the
 * following conditions are met:
 * 
 * Redistributions of source code must retain the above copyright notice, this list of conditions and the following
 * disclaimer. Redistributions in binary form must reproduce the above copyright notice, this list of conditions and the
 * following disclaimer in the documentation and/or other materials provided with the distribution. Neither the name of
 * JiBX nor the names of its contributors may be used to endorse or promote products derived from this software without
 * specific prior written permission.
 * 
 * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES,
 * INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
 * DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
 * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR
 * SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY,
 * WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
 * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
 */

package org.jibx.schema.codegen;

import org.apache.log4j.Logger;
import org.jibx.runtime.QName;
import org.jibx.schema.SchemaUtils;
import org.jibx.schema.codegen.custom.ComponentExtension;
import org.jibx.schema.elements.AnnotatedBase;
import org.jibx.schema.elements.ComplexExtensionElement;
import org.jibx.schema.elements.SchemaBase;

/**
 * Information for a grouping of components (attributes, elements, compositors, and/or wildcards). This is used for
 * both local groupings and global definitions.
 * 
 * @author Dennis M. Sosnoski
 */
public class GroupItem extends Item
{
    /** Logger for class. */
    private static final Logger s_logger = Logger.getLogger(GroupItem.class.getName());
    
    /** Flag for enumeration value. */
    private boolean m_enumeration;
    
    /** Inline references to this structure. */
    private boolean m_inline;
    
    /** Name to be used for generated class (null if inherited). */
    private String m_className;
    
    /** Number of child items in group. */
    private int m_size;
    
    /** First child (null if none). */
    private Item m_head;
    
    /** Last child (null if none). */
    private Item m_tail;

    /** Generated class information (null if inlined). */
    private TypeData m_generateClass;

    /** Flag for all child nodes are optional. In cases where multiple items are associated with the same schema
     component, this is only meaningful for the topmost item. */
    private boolean m_allOptional;

    /** Attribute data present flag. */
    private boolean m_attributePresent;

    /** Element data present flag. */
    private boolean m_elementPresent;

    /** Character data content data present flag. */
    private boolean m_contentPresent;
    
    /**
     * Internal constructor. This is used both for creating a new child group directly, and by the {@link
     * DefinitionItem} subclass.
     * 
     * @param comp schema component (should be the simpleType component in the case of an enumeration)
     * @param parent (null if none)
     */
    protected GroupItem(AnnotatedBase comp, GroupItem parent) {
        super(comp, parent);
        ComponentExtension exten = getComponentExtension();
        m_className = exten.getClassName();
        m_allOptional = true;
    }

    /**
     * Copy constructor. This creates a deep copy with a new parent.
     * 
     * @param original
     * @param ref reference (for overrides to copy; null if none)
     * @param parent (non-null)
     */
    /*package*/ GroupItem(GroupItem original, Item ref, GroupItem parent) {
        super(original, ref, original.getComponentExtension(), parent);
        m_enumeration = original.m_enumeration;
        for (Item child = original.getFirstChild(); child != null; child = child.getNext()) {
            appendChild(child.copy(null, this));
        }
        m_inline = original.m_inline;
        m_className = original.m_className;
        if (parent == null || parent.getSchemaComponent() != getSchemaComponent()) {
            GroupItem top = (GroupItem)original.getTopmost();
            m_allOptional = top.m_allOptional;
            m_attributePresent = top.m_attributePresent;
            m_elementPresent = top.m_elementPresent;
            m_contentPresent = top.m_contentPresent;
        }
    }
    
    /**
     * Constructor from a reference. This is only used for inlining a referenced definition. It merges usage information
     * from the reference with a deep copy of the item structure of the definition.
     * 
     * @param reference
     * @param ext component extension to be linked with inlined definition
     */
    /*package*/ GroupItem(ReferenceItem reference, ComponentExtension ext) {
        super(reference, reference, ext, reference.getParent());
        m_inline = true;
        DefinitionItem definition = reference.getDefinition();
        int type = reference.getSchemaComponent().type();
        if (type == SchemaBase.ATTRIBUTEGROUP_TYPE || type == SchemaBase.GROUP_TYPE) {
            m_allOptional = definition.isAllOptional();
        }
        m_attributePresent = definition.isAttributePresent();
        m_elementPresent = definition.isElementPresent();
        m_contentPresent = definition.isContentPresent();
        appendChild(new GroupItem(definition, reference, this));
    }

    /**
     * Check if this value represents an enumeration.
     *
     * @return enumeration
     */
    public boolean isEnumeration() {
        return m_enumeration;
    }
    
    /**
     * Set value represents an enumeration flag.
     *
     * @param enumeration
     */
    public void setEnumeration(boolean enumeration) {
        m_enumeration = enumeration;
    }

    /**
     * Append an item to the list of children.
     *
     * @param item
     */
    private void appendChild(Item item) {
        if (m_head == null) {
            m_head = m_tail = item;
        } else {
            m_tail.m_next = item;
            item.m_last = m_tail;
            m_tail = item;
        }
        m_size++;
    }

    /**
     * Add a child grouping structure.
     *
     * @param comp schema component
     * @return structure
     */
    public GroupItem addGroup(AnnotatedBase comp) {
        GroupItem group = new GroupItem(comp, this);
        appendChild(group);
        return group;
    }
    
    /**
     * Add a child reference structure.
     *
     * @param comp schema component
     * @param ref referenced definition item
     * @return reference
     */
    public ReferenceItem addReference(AnnotatedBase comp, DefinitionItem ref) {
        ReferenceItem reference = new ReferenceItem(comp, this, ref);
        appendChild(reference);
        return reference;
    }
    
    /**
     * Add a child value.
     *
     * @param comp schema component extension
     * @param type schema type name
     * @param ref schema type equivalent (null if not appropriate)
     * @return value
     */
    public ValueItem addValue(AnnotatedBase comp, QName type, JavaType ref) {
        ValueItem item = new ValueItem(comp, type, ref, this);
        appendChild(item);
        return item;
    }

    /**
     * Add a child any.
     *
     * @param comp schema component
     * @return value
     */
    public AnyItem addAny(AnnotatedBase comp) {
        AnyItem item = new AnyItem(comp, this);
        appendChild(item);
        return item;
    }

    /**
     * Replace an item in this group with another item.
     *
     * @param current
     * @param replace
     */
    /*package*/ void replaceChild(Item current, Item replace) {
        Item last = current.m_last;
        Item next = current.m_next;
        if (last == null) {
            m_head = replace;
        } else {
            last.m_next = replace;
        }
        replace.m_last = last;
        if (next == null) {
            m_tail = replace;
        } else {
            next.m_last = replace;
        }
        replace.m_next = next;
    }
    
    /**
     * Adopt the child items from another group as the child items of this group.
     *
     * @param group
     */
    void adoptChildren(GroupItem group) {
        m_size = group.m_size;
        m_head = group.m_head;
        m_tail = group.m_tail;
        for (Item item = m_head; item != null; item = item.m_next) {
            item.reparent(this);
        }
    }

    /**
     * Check if structure to be inlined.
     *
     * @return inline
     */
    public boolean isInline() {
        return m_inline;
    }

    /**
     * Set structure to be inlined flag.
     *
     * @param inline
     */
    public void setInline(boolean inline) {
        m_inline = inline;
    }

    /**
     * Get effective item name, applying inheritance if necessary.
     * 
     * @return name
     */
    public String getEffectiveClassName() {
        GroupItem item = this;
        while (item.getClassName() == null) {
            item = item.getParent();
            if (item == null) {
                throw new IllegalStateException("Inherited class name with nothing to inherit");
            }
        }
        return item.getClassName();
    }

    /**
     * Get class name set directly for this group.
     *
     * @return name (null if to be inherited)
     */
    public String getClassName() {
        return m_className;
    }
    
    /**
     * Check if the class name is fixed by configuration.
     *
     * @return true if fixed, false if not
     */
    public boolean isFixedClassName() {
        return getComponentExtension().getClassName() != null;
    }
    
    /**
     * Set class name directly for this group. It is an error to call this method if the class name is fixed.
     * 
     * @param name (null if to be inherited)
     */
    public void setClassName(String name) {
        if (isFixedClassName()) {
            throw new IllegalStateException("Internal error - attempt to change configured class name");
        } else {
            m_className = name;
        }
    }

    /**
     * Get the number of items present in the group.
     *
     * @return count
     */
    public int getChildCount() {
        return m_size;
    }

    /**
     * Get head item in list grouped by this structure.
     * 
     * @return item (null if none)
     */
    public Item getFirstChild() {
        return m_head;
    }
    
    /**
     * Get information for class to be generated.
     *
     * @return class
     */
    public TypeData getGenerateClass() {
        return m_generateClass;
    }

    /**
     * Set information for class to be generated. If this group is a complexType extension and the base type is not
     * being inlined, this sets the generated class to extend the base type class.
     *
     * @param clas
     */
    public void setGenerateClass(TypeData clas) {
        m_generateClass = clas;
    }

    /**
     * Check if this group represents an extension reference.
     * 
     * @return true if extension reference, false if not
     */
    public boolean isExtensionReference() {
        return m_head instanceof ReferenceItem && m_head.getSchemaComponent().type() == SchemaBase.EXTENSION_TYPE;
    }

    /**
     * Handle groups which consist of a single type reference, or of an extension type reference, by subclassing the
     * class generated for the reference.
	 * TODO: instead use separate extension test, since this won't be called for embedded types
     */
    public void convertTypeReference() {
        if (m_head instanceof ReferenceItem && (m_head == m_tail ||
            m_head.getSchemaComponent() instanceof ComplexExtensionElement)) {
            DefinitionItem def = ((ReferenceItem)m_head).getDefinition();
            if (isEnumeration() == def.isEnumeration()) {
                TypeData base = def.getGenerateClass();
                if (s_logger.isDebugEnabled()) {
                    s_logger.debug("Setting base class for " + m_generateClass.getFullName() + " to " +
                        base.getFullName());
                }
                ((ClassHolder)m_generateClass).setSuperClass(base);
                m_head.setImplicit(true);
            }
        }
    }

    /**
     * Copy the item under a different parent.
     *
     * @param ref reference (for overrides to copy; null if none)
     * @param parent
     * @return copy
     */
    protected Item copy(Item ref, GroupItem parent) {
        return new GroupItem(this, ref, parent);
    }
    
    /**
     * Set attribute present in group. This cascades the attribute present flag upward through containing groups until
     * one is found which defines an element name.
     */
    protected void forceAttributePresent() {
        if (!m_attributePresent && getSchemaComponent().type() != SchemaBase.ELEMENT_TYPE) {
            GroupItem parent = getParent();
            if (parent != null) {
                parent.forceAttributePresent();
            }
        }
        m_attributePresent = true;
    }
    
    /**
     * Set element present in group. This cascades the element present flag upward through containing groups until
     * one is found which defines an element name.
     */
    protected void forceElementPresent() {
        if (!m_elementPresent && getSchemaComponent().type() != SchemaBase.ELEMENT_TYPE) {
            GroupItem parent = getParent();
            if (parent != null) {
                parent.forceElementPresent();
            }
        }
        m_elementPresent = true;
    }
    
    /**
     * Set character data content present in group. This cascades the content present flag upward through all containing
     * groups until one is found which defines an element name.
     */
    protected void forceContentPresent() {
        if (!m_contentPresent && getSchemaComponent().type() != SchemaBase.ELEMENT_TYPE) {
            GroupItem parent = getParent();
            if (parent != null) {
                parent.forceContentPresent();
            }
        }
        m_contentPresent = true;
    }
    
    /**
     * Set required item present in group. This cascades the required item present flag upward through all containing
     * groups until one is found which defines either a wrapping element or a compositor other than a required sequence
     * (because a required item present within a required sequence means that there will always be something present in
     * the document, while any other type of compositor does not have this meaning).
     */
    protected void forceRequiredPresent() {
        if (m_allOptional) {
            int type = getSchemaComponent().type();
            boolean skip = false;
            switch (type) {
                
                case SchemaBase.ELEMENT_TYPE:
                case SchemaBase.CHOICE_TYPE:
                case SchemaBase.ALL_TYPE:
                    skip = true;
                    break;
                    
                case SchemaBase.SEQUENCE_TYPE:
                    skip = isOptional();
                    break;
                    
            }
            if (!skip) {
                GroupItem parent = getParent();
                if (parent != null) {
                    parent.forceContentPresent();
                }
            }
        }
        m_allOptional = false;
    }
    
    /**
     * Classify the content of this item as attribute, element, and/or character data content. For a group item, this
     * just needs to call the corresponding method for each child item.
     */
    protected void classifyContent() {
        
        // handle basic classification for this component
        super.classifyContent();
        
        // classify each child component
        for (Item item = m_head; item != null; item = item.getNext()) {
            item.classifyContent();
        }
        
        // force off the 'all optional' state regardless of content if a required element or attribute
        if (m_allOptional && !isOptional()) {
            switch (getSchemaComponent().type()) {
                
                case SchemaBase.ANY_TYPE:
                case SchemaBase.ELEMENT_TYPE:
                case SchemaBase.ANYATTRIBUTE_TYPE:
                case SchemaBase.ATTRIBUTE_TYPE:
                    m_allOptional = false;
                    break;
                    
            }
        }
    }
    
    /**
     * Convert an embedded group to a freestanding definition. This creates a definition using a cloned copy of the
     * structure of this group, then replaces this group with a reference to the definition.
     * TODO: just adopt the child items, rather than cloning? minor performance gain.
     *
     * @return definition
     */
    public DefinitionItem convertToDefinition() {
        if (s_logger.isDebugEnabled()) {
            s_logger.debug("Converting " + SchemaUtils.describeComponent(getSchemaComponent()) +
                " to freestanding definition");
        }
        GroupItem parent = this;
        while (! (parent instanceof DefinitionItem)) {
            parent = parent.getParent();
        }
        DefinitionItem def = new DefinitionItem(this);
        if (def.getClassName() == null) {
            def.setClassName(getEffectiveClassName());
        }
        if (def.getName() == null) {
            def.setName(getEffectiveName());
        }
        ReferenceItem ref = new ReferenceItem(this, def);
        getParent().replaceChild(this, ref);
        return def;
    }
    
    /**
     * Build description of nested items.
     *
     * @param depth current nesting depth
     * @param classified include classification details flag
     * @return description
     */
    public String nestedString(int depth, boolean classified) {
        depth++;
        Item child = getFirstChild();
        StringBuffer buff = new StringBuffer(400);
        while (child != null) {
            buff.append(child.describe(depth, classified));
            child = child.getNext();
        }
        return buff.toString();
    }

    /**
     * Check if all immediate child nodes are optional. This is needed when handling code generation for a
     * reference to a group or attributeGroup handled as a separate object, since that object only needs to be
     * present if one or more of the values are present. This method is only meaningful for the topmost item associated
     * with a particular schema component (those for which {@link Item#isTopmost()} returns true).
     *
     * @return true if all child nodes optional, false if not
     */
    public boolean isAllOptional() {
        return m_allOptional;
    }

    /**
     * Check if an attribute is part of this item. This is only true for items corresponding to attribute
     * definitions, and groupings including these items which do not define an element name. This method is only
     * meaningful for the topmost item associated with a particular schema component (those for which {@link
     * Item#isTopmost()} returns true).
     *
     * @return true if attribute
     */
    public boolean isAttributePresent() {
        return m_attributePresent;
    }

    /**
     * Check if a child elements is part of this item. This is true for all items corresponding to element
     * definitions, and all groupings which include such an item. This method is only meaningful for the topmost item
     * associated with a particular schema component (those for which {@link Item#isTopmost()} returns
     * true).
     *
     * @return true if content
     */
    public boolean isElementPresent() {
        return m_elementPresent;
    }

    /**
     * Check if character data content is part of this item. This is true for all items corresponding to
     * simpleContent definitions, and all groupings which include such an item. This method is only meaningful for the
     * topmost item associated with a particular schema component (those for which {@link Item#isTopmost()} returns
     * true).
     *
     * @return true if content
     */
    public boolean isContentPresent() {
        return m_contentPresent;
    }
    
    /**
     * Generate a description of the item, including all nested items.
     *
     * @param depth current nesting depth
     * @param classified include classification details flag
     * @return description
     */
    protected String describe(int depth, boolean classified) {
        StringBuffer buff = new StringBuffer(depth + 50);
        buff.append(leadString(depth));
        if (isInline()) {
            buff.append("inlined ");
        }
        if (m_enumeration) {
            buff.append("enumeration ");
        }
        AnnotatedBase comp = getSchemaComponent();
        if (isTopmost()) {
            if (classified) {
                if (m_attributePresent) {
                    if (m_contentPresent) {
                        buff.append("attribute+content ");
                    } else {
                        buff.append("attribute ");
                    }
                } else if (m_contentPresent) {
                    buff.append("content ");
                }
            }
        } else {
            buff.append("duplicate ");
        }
        buff.append("group with class name ");
        buff.append(getClassName());
        buff.append(" and value name ");
        buff.append(getName());
        if (classified && isTopmost()) {
            buff.append(m_allOptional ? " (all items optional)" : " (not all items optional)");
        }
        buff.append(": ");
        buff.append(SchemaUtils.describeComponent(comp));
        buff.append('\n');
        buff.append(nestedString(depth, classified));
        return buff.toString();
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy