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

org.jibx.schema.codegen.custom.ComponentExtension Maven / Gradle / Ivy

There is a newer version: 1.4.2
Show newest version
/*
 * 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.custom;

import java.util.ArrayList;
import java.util.HashSet;
import java.util.Set;

import org.apache.log4j.Logger;
import org.jibx.runtime.QName;
import org.jibx.schema.IArity;
import org.jibx.schema.SchemaUtils;
import org.jibx.schema.elements.AttributeElement;
import org.jibx.schema.elements.AttributeGroupRefElement;
import org.jibx.schema.elements.ChoiceElement;
import org.jibx.schema.elements.CommonCompositorDefinition;
import org.jibx.schema.elements.CommonTypeDefinition;
import org.jibx.schema.elements.CommonTypeDerivation;
import org.jibx.schema.elements.ComplexExtensionElement;
import org.jibx.schema.elements.ComplexRestrictionElement;
import org.jibx.schema.elements.ComplexTypeElement;
import org.jibx.schema.elements.ElementElement;
import org.jibx.schema.elements.FilteredSegmentList;
import org.jibx.schema.elements.GroupRefElement;
import org.jibx.schema.elements.ListElement;
import org.jibx.schema.elements.OpenAttrBase;
import org.jibx.schema.elements.SchemaBase;
import org.jibx.schema.elements.SequenceElement;
import org.jibx.schema.elements.SimpleExtensionElement;
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.types.Count;
import org.jibx.schema.validation.ValidationContext;

/**
 * Extension information for all schema components other than the schema element itself. This is the basic extension
 * which associates schema components with values or classes for code generation.
 * 
 * @author Dennis M. Sosnoski
 */
public class ComponentExtension extends BaseExtension
{
    /** Logger for class. */
    private static final Logger s_logger = Logger.getLogger(ComponentExtension.class.getName());
    
    /** Containing global definition extension. */
    private final GlobalExtension m_global;
    
    /** Component dropped from schema definition. */
    private boolean m_removed;
    
    /** Optional component flag. */
    private boolean m_optional;
    
    /** Repeated component flag. */
    private boolean m_repeated;
    
    /** Customization information for this component. */
    private ComponentCustom m_custom;
    
    /** Override for type specified in schema (null if none). */
    private QName m_overrideType;
    
    /** Number of times a component is used in code generation. */
    private int m_useCount;
    
    /**
     * Constructor.
     * 
     * @param comp Component
     * @param global containing global definition extension (null allowed only as special case when
     * calling this constructor from the global extension subclass constructor)
     */
    public ComponentExtension(OpenAttrBase comp, GlobalExtension global) {
        super(comp);
        m_global = global == null ? (GlobalExtension)this : global;
        if (comp instanceof IArity) {
            
            // element can only be optional or repeated when embedded, but no need to check since counts are forbidden
            //  on a global definition
            IArity particle = (IArity)comp;
            m_repeated = SchemaUtils.isRepeated(particle);
            if (comp.type() == SchemaBase.ELEMENT_TYPE) {
                m_optional = SchemaUtils.isOptionalElement((ElementElement)comp);
            } else {
                m_optional = SchemaUtils.isOptional(particle);
            }
            
        } else if (comp.type() == SchemaBase.ATTRIBUTE_TYPE) {
            
            // attribute can never be repeated, and can only be optional when embedded (rather than a global definition)
            m_repeated = false;
            m_optional = global != null && ((AttributeElement)comp).getUse() == AttributeElement.OPTIONAL_USE;
            
        }
    }
    
    /**
     * Check if component to be removed from schema.
     * 
     * @return removed flag
     */
    public boolean isRemoved() {
        return m_removed;
    }
    
    /**
     * Set flag for component to be removed from schema.
     * 
     * @param removed Flag
     */
    public void setRemoved(boolean removed) {
        m_removed = removed;
    }
    
    /**
     * Check if component is to be ignored.
     *
     * @return ignored flag
     */
    public boolean isIgnored() {
        if (m_custom == null) {
            return false;
        } else {
            return m_custom.isIgnored();
        }
    }
    
    /**
     * Check if optional component.
     * 
     * @return optional
     */
    public boolean isOptional() {
        return m_optional;
    }
    
    /**
     * Set optional component.
     * 
     * @param optional Flag
     */
    public void setOptional(boolean optional) {
        m_optional = optional;
    }
    
    /**
     * Check if repeated component.
     * 
     * @return repeated
     */
    public boolean isRepeated() {
        return m_repeated;
    }
    
    /**
     * Set repeated component.
     * 
     * @param repeated Flag
     */
    public void setRepeated(boolean repeated) {
        m_repeated = repeated;
    }
    
    /**
     * Check if schema component is to be generated inline.
     *
     * @return true if inlined, false if not
     */
    public boolean isInlined() {
        if (m_custom == null) {
            return false;
        } else {
            return m_custom.isInlined();
        }
    }
    
    /**
     * Check if schema component is to be generated as a separate class.
     *
     * @return true if separate class, false if not
     */
    public boolean isSeparateClass() {
        if (m_custom == null) {
            return false;
        } else {
            return m_custom.isSeparateClass();
        }
    }

    /**
     * Get the containing global extension.
     *
     * @return global
     */
    public GlobalExtension getGlobal() {
        return m_global;
    }

    /**
     * Get override type.
     * 
     * @return type name (null if none)
     */
    public QName getOverrideType() {
        return m_overrideType;
    }
    
    /**
     * Set override type.
     * 
     * @param qname type name (null if none)
     */
    public void setOverrideType(QName qname) {
        m_overrideType = qname;
    }
    
    /**
     * Increment the use count for the component.
     * 
     * @return incremented use count
     */
    public int incrementUseCount() {
        return ++m_useCount;
    }
    
    /**
     * Get the use count for the component.
     *
     * @return use count
     */
    public int getUseCount() {
        return m_useCount;
    }
    
    /**
     * Get name to be used for generated class.
     *
     * @return class name (null if not set)
     */
    public String getClassName() {
        if (m_custom == null) {
            return null;
        } else {
            return m_custom.getClassName();
        }
    }
    
    /**
     * Get base name for corresponding property.
     *
     * @return property name (null if not set)
     */
    public String getBaseName() {
        if (m_custom == null) {
            return null;
        } else {
            return m_custom.getBaseName();
        }
    }
    
    /**
     * Get customization information for this component.
     *
     * @return custom
     */
    ComponentCustom getCustom() {
        return m_custom;
    }

    /**
     * Set customization information for this component.
     *
     * @param custom
     */
    void setCustom(ComponentCustom custom) {
        m_custom = custom;
    }
    
    /**
     * Get the innermost customization which applies to this component.
     *
     * @return customization
     */
    private NestingCustomBase getContainingCustom() {
        if (m_custom == null) {
            Object extension = getComponent().getParent().getExtension();
            if (extension instanceof ComponentExtension) {
                return ((ComponentExtension)extension).getContainingCustom();
            } else {
                return ((SchemaExtension)extension).getCustom();
            }
        } else {
            return m_custom;
        }
    }
    
    /**
     * Get the xs:any handling type code to be applied for this component. The default value is
     * {@link NestingCustomBase#ANY_DOM} if not overridden at any level.
     * 
     * @return code
     */
    public int getAnyType() {
        return getContainingCustom().getAnyType();
    }
    
    /**
     * Get the xs:choice handling type code to be applied for this component. The default value is
     * {@link NestingCustomBase#SELECTION_CHECKEDSET} if not overridden at any level.
     * 
     * @return code
     */
    public int getChoiceType() {
        return getContainingCustom().getChoiceType();
    }
    
    /**
     * Check if xs:choice selection state should be exposed for this component.
     * 
     * @return exposed flag
     */
    public boolean isChoiceExposed() {
        return getContainingCustom().isChoiceExposed();
    }
    
    /**
     * Get the xs:union handling type code to be applied for this component. The default value is
     * {@link NestingCustomBase#SELECTION_CHECKEDSET} if not overridden at any level.
     * 
     * @return code
     */
    public int getUnionType() {
        return getContainingCustom().getUnionType();
    }
    
    /**
     * Check if xs:union selection state should be exposed for this component.
     * 
     * @return exposed flag
     */
    public boolean isUnionExposed() {
        return getContainingCustom().isUnionExposed();
    }

    /**
     * Check for type substitution on a type reference, then record the reference.
     * TODO: how to handle substitutions across namespaces? the problem here is that the returned name needs to be
     * correct in the use context, which may be using a different namespace for the same definition (due to includes of
     * no-namespace schemas into a namespaced schema). the current code is only an interim fix that will not always work
     *
     * @param type original type
     * @param vctx validation context
     * @return replacement type (may be the same as the original type; null if to be deleted)
     */
    private QName replaceAndReference(QName type, ValidationContext vctx) {
        QName repl = getReplacementType(type);
        if (repl != type && s_logger.isDebugEnabled()) {
            s_logger.debug("Replacing type " + type + " with type " + repl + " in component " +
                SchemaUtils.describeComponent(getComponent()));
        }
        while (repl != null) {
            CommonTypeDefinition def = vctx.findType(repl);
            if (def == null) {
                
                // try substituting the original namespace, to work with no-namespace schemas
                if (repl.getUri() == null && type.getUri() != null) {
                    repl = new QName(type.getUri(), repl.getName());
                    def = vctx.findType(repl);
                }
                if (def == null) {
                    throw new IllegalStateException("Internal error - type definition not found");
                }
            }
            GlobalExtension exten = (GlobalExtension)def.getExtension();
            if (exten == null) {
                if (s_logger.isDebugEnabled()) {
                    s_logger.debug("No extension found for referenced type " + SchemaUtils.describeComponent(def));
                }
                return repl;
            } else {
                if (exten.isRemoved()) {
                    if (s_logger.isDebugEnabled()) {
                        s_logger.debug("Reference to deleted type " + SchemaUtils.describeComponent(def));
                    }
                    return null;
                } else {
                    if (exten.getOverrideType() == null) {
                        exten.addReference(this);
                        m_global.addDependency(exten);
                        return repl;
                    } else {
                        repl = exten.getOverrideType();
                    }
                }
            }
        }
        return repl;
    }
    
    /**
     * Check a reference to a component. If the reference component has been deleted this just returns
     * false. If the component has not been deleted it counts the reference on that component, and records
     * the dependency from this component before returning true. For convenience, this may be called with a
     * null argument, which just returns true.
     *
     * @param comp component (call ignored if null)
     * @return true if reference to be kept, false if deleted
     */
    private boolean checkReference(OpenAttrBase comp) {
        if (comp != null) {
            GlobalExtension exten = (GlobalExtension)comp.getExtension();
            if (exten == null) {
                if (s_logger.isDebugEnabled()) {
                    s_logger.debug("No extension found for referenced component " +
                        SchemaUtils.describeComponent(comp));
                }
            } else {
                if (exten.isRemoved()) {
                    if (s_logger.isDebugEnabled()) {
                        s_logger.debug("Referenced to deleted component " + SchemaUtils.describeComponent(comp));
                    }
                    return false;
                } else {
                    exten.addReference(this);
                    m_global.addDependency(exten);
                }
            }
        }
        return true;
    }

    /**
     * Remove a child element. This checks to make sure the removal is valid, and also handles logging of the change.
     *
     * @param index
     */
    private void removeChild(int index) {
        SchemaBase child = getComponent().getChild(index);
        switch (child.type())
        {
            case SchemaBase.ALL_TYPE:
            case SchemaBase.ANY_TYPE:
            case SchemaBase.ANYATTRIBUTE_TYPE:
            case SchemaBase.ATTRIBUTE_TYPE:
            case SchemaBase.ATTRIBUTEGROUP_TYPE:
            case SchemaBase.CHOICE_TYPE:
            case SchemaBase.COMPLEXTYPE_TYPE:
            case SchemaBase.ELEMENT_TYPE:
            case SchemaBase.ENUMERATION_TYPE:
            case SchemaBase.FRACTIONDIGITS_TYPE:
            case SchemaBase.GROUP_TYPE:
            case SchemaBase.LENGTH_TYPE:
            case SchemaBase.MAXEXCLUSIVE_TYPE:
            case SchemaBase.MAXINCLUSIVE_TYPE:
            case SchemaBase.MAXLENGTH_TYPE:
            case SchemaBase.MINEXCLUSIVE_TYPE:
            case SchemaBase.MININCLUSIVE_TYPE:
            case SchemaBase.MINLENGTH_TYPE:
            case SchemaBase.PATTERN_TYPE:
            case SchemaBase.SEQUENCE_TYPE:
            case SchemaBase.SIMPLETYPE_TYPE:
            case SchemaBase.TOTALDIGITS_TYPE:
            case SchemaBase.WHITESPACE_TYPE:
            {
                // definition component - just delete from the parent structure
                getComponent().detachChild(index);
                if (s_logger.isDebugEnabled()) {
                    s_logger.debug("Removed component " + SchemaUtils.describeComponent(child) +
                        " as per extension");
                }
                break;
            }
            
            default:
                throw new IllegalStateException("Internal error: element type '" + child.name() +
                    "' cannot be deleted");
        }
    }

    /**
     * Apply extensions to schema definition component, deleting components flagged for skipping and substituting types
     * as configured. This code is not intended to handle the deletion of global definition components, which should be
     * removed separately.
     * 
     * @param vctx validation context
     */
    public void applyAndCountUsage(ValidationContext vctx) {
        
        // handle type substitutions for this component
        OpenAttrBase component = getComponent();
        boolean delete = false;
        switch (component.type())
        {
            
            case SchemaBase.ATTRIBUTE_TYPE:
            {
                if (isIgnored()) {
                    delete = true;
                } else {
                    AttributeElement attr = ((AttributeElement)component);
                    if (m_overrideType != null) {
                        attr.setType(m_overrideType);
                    }
                    QName type = attr.getType();
                    if (type == null) {
                        AttributeElement ref = attr.getReference();
                        if (ref != null) {
                            delete = !checkReference(ref);
                        } else if (!attr.isInlineType()) {
                            attr.setType(SchemaTypes.ANY_SIMPLE_TYPE.getQName());
                        }
                    } else {
                        QName repl = replaceAndReference(type, vctx);
                        if (repl == null) {
                            // TODO optionally make sure the attribute is optional?
                            delete = true;
                        } else if (repl != type) {
                            attr.setType(repl);
                        }
                    }
                }
                break;
            }
            
            case SchemaBase.ATTRIBUTEGROUP_TYPE:
            {
                if (component instanceof AttributeGroupRefElement) {
                    delete = !checkReference(((AttributeGroupRefElement)component).getReference());
                }
                break;
            }
            
            case SchemaBase.ELEMENT_TYPE:
            {
                ElementElement elem = ((ElementElement)component);
                if (isIgnored()) {
                    
                    // reference or definition determines handling
                    QName ref = elem.getRef();
                    if (ref == null) {
                        
                        // definition, just delete all content
                        for (int i = 0; i < elem.getChildCount(); i++) {
                            elem.detachChild(i);
                        }
                        elem.compactChildren();
                        
                    } else {
                        
                        // reference may be to other namespace, so convert to qualified name string
                        StringBuffer buff = new StringBuffer();
                        buff.append('{');
                        if (ref.getUri() != null) {
                            buff.append(ref.getUri());
                        }
                        buff.append('}');
                        buff.append(ref.getName());
                        elem.setName(buff.toString());
                        elem.setRef(null);
                        
                    }
                    elem.setType(SchemaTypes.ANY_TYPE.getQName());
                    
                } else {
                    if (m_overrideType != null) {
                        elem.setType(m_overrideType);
                    }
                    QName type = elem.getType();
                    if (type == null) {
                        delete = !checkReference(elem.getReference());
                    } else {
                        QName repl = replaceAndReference(type, vctx);
                        if (repl == null) {
                            // TODO optionally make sure the element is optional?
                            delete = true;
                        } else if (repl != type) {
                            elem.setType(repl);
                        }
                    }
                }
                break;
            }
            
            case SchemaBase.EXTENSION_TYPE:
            case SchemaBase.RESTRICTION_TYPE:
            {
                CommonTypeDerivation deriv = (CommonTypeDerivation)component;
                QName type = deriv.getBase();
                if (type != null) {
                    QName repl = replaceAndReference(type, vctx);
                    if (repl == null) {
                        delete = true;
                    } else if (repl != type) {
                        deriv.setBase(repl);
                    }
                }
                break;
            }
            
            case SchemaBase.GROUP_TYPE:
            {
                if (component instanceof GroupRefElement) {
                    delete = !checkReference(((GroupRefElement)component).getReference());
                }
                break;
            }
                
            case SchemaBase.LIST_TYPE:
            {
                ListElement list = ((ListElement)component);
/*                    // not currently supported - is it needed?
                    if (m_overrideType != null) {
                        list.setItemType(m_overrideType);
                    }   */
                QName type = list.getItemType();
                if (type != null) {
                    QName repl = replaceAndReference(type, vctx);
                    if (repl == null) {
                        delete = true;
                    } else if (repl != type) {
                        list.setItemType(repl);
                    }
                }
                break;
            }
            
            case SchemaBase.UNION_TYPE:
            {
                UnionElement union = ((UnionElement)component);
                QName[] types = union.getMemberTypes();
                if (types != null) {
                    ArrayList repls = new ArrayList();
                    boolean changed = false;
                    for (int i = 0; i < types.length; i++) {
                        QName type = types[i];
                        QName repl = replaceAndReference(type, vctx);
                        changed = changed || repl != type;
                        if (repl != null) {
                            repls.add(repl);
                        }
                    }
                    if (changed) {
                        if (repls.size() > 0) {
                            union.setMemberTypes((QName[])repls.toArray(new QName[repls.size()]));
                        } else {
                            union.setMemberTypes(null);
                        }
                    }
                }
                break;
            }
                
        }
        if (delete) {
            
            // raise deletion to containing removable component
            SchemaBase parent = component;
            SchemaBase remove = null;
            loop: while (parent != null) {
                switch (parent.type())
                {
                    
                    case SchemaBase.ATTRIBUTE_TYPE:
                    case SchemaBase.ATTRIBUTEGROUP_TYPE:
                    case SchemaBase.ELEMENT_TYPE:
                    case SchemaBase.GROUP_TYPE:
                        remove = parent;
                        break loop;
                    
                    case SchemaBase.COMPLEXTYPE_TYPE:
                    case SchemaBase.SIMPLETYPE_TYPE:
                        remove = parent;
                        parent = parent.getParent();
                        break;
                    
                    default:
                        if (remove == null) {
                            parent = parent.getParent();
                            break;
                        } else {
                            break loop;
                        }
                        
                }
            }
            if (remove == null) {
                throw new IllegalStateException("Internal error: no removable ancestor found for component "
                    + SchemaUtils.describeComponent(component));
            } else {
                ((ComponentExtension)remove.getExtension()).setRemoved(true);
            }
            
        } else {
            
            // process all child component extensions
            boolean modified = false;
            for (int i = 0; i < component.getChildCount(); i++) {
                SchemaBase child = component.getChild(i);
                ComponentExtension exten = (ComponentExtension)child.getExtension();
                if (!exten.isRemoved()) {
                    exten.applyAndCountUsage(vctx);
                }
                if (exten.isRemoved()) {
                    removeChild(i);
                    modified = true;
                }
            }
            if (modified) {
                component.compactChildren();
            }
        }
    }

    /**
     * Try to replace type definition with substitute type from derivation. If 
     *
     * @param lead prefix text for indentation of logging messages
     * @param topcomp schema component being normalized
     * @param childcomp current child of schema component being normalized
     * @param derive type derivation supplying substitute type
     * @return true if type modified, false if not
     */
    private boolean substituteTypeDerivation(String lead, OpenAttrBase topcomp, OpenAttrBase childcomp,
        CommonTypeDerivation derive) {
        SchemaBase parent = topcomp;
        while (parent != null) {
            switch (parent.type())
            {
                
                case SchemaBase.ATTRIBUTE_TYPE:
                {
                    // set the attribute type to the specified type
                    AttributeElement attr = (AttributeElement)parent;
                    attr.setType(derive.getBase());
                    return true;
                }
                
                case SchemaBase.COMPLEXTYPE_TYPE:
                {
                    if (parent.isGlobal()) {
                        GlobalExtension global = (GlobalExtension)parent.getExtension();
                        if (!global.isIncluded() && global.isPreferInline()) {
                            ComplexTypeElement type = (ComplexTypeElement)parent;
                            ComponentExtension typeext = (ComponentExtension)type.getExtension();
                            typeext.setOverrideType(derive.getBase());
                            if (s_logger.isDebugEnabled()) {
                                s_logger.debug(lead + "set substitution for global complex type " +
                                    SchemaUtils.componentPath(type) + " to base type from derivation " +
                                    SchemaUtils.componentPath(childcomp));
                            }
                            return true;
                        }
                    }
                    break;
                }
                
                case SchemaBase.ELEMENT_TYPE:
                {
                    // set the element type to the restriction base
                    ElementElement elem = (ElementElement)parent;
                    elem.setTypeDefinition(derive.getBaseType());
                    if (s_logger.isDebugEnabled()) {
                        s_logger.debug(lead + "replaced element " + SchemaUtils.componentPath(elem) +
                            " type with base type from derivation " + SchemaUtils.componentPath(childcomp));
                    }
                    return true;
                }
                
                case SchemaBase.SIMPLETYPE_TYPE:
                {
                    // replace type only if global type definition
                    if (parent.isGlobal()) {
                        GlobalExtension global = (GlobalExtension)parent.getExtension();
                        if (!global.isIncluded()) {
                            SimpleTypeElement type = (SimpleTypeElement)parent;
                            ComponentExtension parentext = (ComponentExtension)type.getExtension();
                            parentext.setOverrideType(derive.getBase());
                            return true;
                        }
                    }
                    break;
                }
                
                case SchemaBase.UNION_TYPE:
                {
                    // this will be handled by union normalization, so just leave in place
                    return false;
                }
            }
            parent = parent.getParent();
        }
        return false;
    }

    /**
     * Normalize the child schema definition. This recursively traverses the schema model tree rooted in the
     * component associated with this extension, normalizing each child component.
     * 
     * TODO: handle revalidation for changed subtrees
     * 
     * @param depth nesting depth for validation
     * @return true if any part of tree under this component modified, false if not
     */
    protected boolean normalize(int depth) {
        
        // initialize debug handling
        String path = null;
        String lead = null;
        OpenAttrBase topcomp = getComponent();
        if (s_logger.isDebugEnabled()) {
            path = SchemaUtils.componentPath(topcomp);
            lead = SchemaUtils.getIndentation(depth);
            s_logger.debug(lead + "entering normalization for node " + path);
        }
        
        // normalize each child component in turn
        int count = topcomp.getChildCount();
        boolean modified = false;
        boolean compact = false;
        for (int i = 0; i < count; i++) {
            SchemaBase child = topcomp.getChild(i);
            if (child instanceof OpenAttrBase) {
                
                // start by normalizing the child component content
                OpenAttrBase childcomp = (OpenAttrBase)child;
                ComponentExtension childext = (ComponentExtension)childcomp.getExtension();
                if (childext.normalize(depth+1)) {
                    modified = true;
                }
                
                // check for child components with special normalization handling
                switch (childcomp.type()) {
                    
                    case SchemaBase.ALL_TYPE:
                    case SchemaBase.CHOICE_TYPE:
                    case SchemaBase.SEQUENCE_TYPE:
                    {
                        // child component is a compositor, so see if it can be deleted or replaced
                        CommonCompositorDefinition compositor = (CommonCompositorDefinition)childcomp;
                        int size = compositor.getParticleList().size();
                        if (size == 0) {
                            
                            // empty compositor, just remove (always safe to do this)
                            childext.setRemoved(true);
                            modified = true;
                            if (s_logger.isDebugEnabled()) {
                                s_logger.debug(lead + "eliminated empty compositor " + path);
                            }
                            
                        } else if (size == 1) {
                            
                            // single component in compositor, first try to convert compositor to singleton
                            OpenAttrBase grandchild = (OpenAttrBase)compositor.getParticleList().get(0);
                            IArity particle = (IArity)grandchild;
                            if (SchemaUtils.isRepeated(compositor)) {
                                if (!SchemaUtils.isRepeated(particle)) {
                                    
                                    // repeated compositor with non-repeated particle, so pass repeat to particle
                                    if (s_logger.isDebugEnabled()) {
                                        s_logger.debug(lead + "passing repeat from compositor " +
                                            SchemaUtils.componentPath(childcomp) + " to only child " +
                                            SchemaUtils.describeComponent(grandchild));
                                    }
                                    particle.setMaxOccurs(compositor.getMaxOccurs());
                                    ((ComponentExtension)grandchild.getExtension()).setRepeated(true);
                                    compositor.setMaxOccurs(null);
                                    ((ComponentExtension)compositor.getExtension()).setRepeated(false);
                                    
                                } else if ((compositor.getMaxOccurs().isUnbounded() &&
                                    particle.getMaxOccurs().isUnbounded())) {
                                    
                                    // unbounded compositor with unbounded particle, just wipe repeat from compositor
                                    if (s_logger.isDebugEnabled()) {
                                        s_logger.debug(lead + "clearing unbounded from compositor " +
                                            SchemaUtils.componentPath(childcomp) + " with unbounded only child " +
                                            SchemaUtils.describeComponent(grandchild));
                                    }
                                    compositor.setMaxOccurs(null);
                                    ((ComponentExtension)compositor.getExtension()).setRepeated(false);
                                    
                                }
                            }
                            if (SchemaUtils.isOptional(compositor)) {
                                if (SchemaUtils.isOptional(particle)) {
                                    
                                    // optional compositor with optional particle, just wipe optional from compositor
                                    if (s_logger.isDebugEnabled()) {
                                        s_logger.debug(lead + "clearing optional from compositor " +
                                            SchemaUtils.componentPath(childcomp) + " with optional only child " +
                                            SchemaUtils.describeComponent(grandchild));
                                    }
                                    compositor.setMinOccurs(null);
                                    ((ComponentExtension)compositor.getExtension()).setOptional(false);
                                    
                                } else if (Count.isCountEqual(1, particle.getMinOccurs())) {
                                    
                                    // optional compositor with required particle, so pass optional to particle
                                    if (s_logger.isDebugEnabled()) {
                                        s_logger.debug(lead + "passing optional from compositor " +
                                            SchemaUtils.componentPath(childcomp) + " to required only child " +
                                            SchemaUtils.describeComponent(grandchild));
                                    }
                                    particle.setMinOccurs(Count.COUNT_ZERO);
                                    ((ComponentExtension)grandchild.getExtension()).setOptional(true);
                                    compositor.setMinOccurs(null);
                                    ((ComponentExtension)compositor.getExtension()).setOptional(false);
                                    
                                }
                            }
                            
                            // check if top component is also a compositor
                            if (topcomp instanceof CommonCompositorDefinition) {
                                
                                // nested compositor, check if can be simplified
                                if (SchemaUtils.isSingleton(compositor)) {
                                    
                                    // nested singleton compositor with only child, just replace with the child
                                    topcomp.replaceChild(i, grandchild);
                                    modified = true;
                                    if (s_logger.isDebugEnabled()) {
                                        s_logger.debug(lead + "replacing singleton compositor " +
                                            SchemaUtils.componentPath(childcomp) + " with only child " +
                                            SchemaUtils.describeComponent(grandchild));
                                    }
                                    
                                } else if (compositor instanceof ChoiceElement) {
                                    
                                    // replace choice with sequence to simplify handling, since just one particle
                                    if (s_logger.isDebugEnabled()) {
                                        s_logger.debug(lead + "substituting sequence for choice " +
                                            SchemaUtils.componentPath(childcomp) + " with only one child");
                                    }
                                    SequenceElement sequence = new SequenceElement();
                                    sequence.setAnnotation(compositor.getAnnotation());
                                    sequence.setExtension(compositor.getExtension());
                                    sequence.setMaxOccurs(compositor.getMaxOccurs());
                                    sequence.setMinOccurs(compositor.getMinOccurs());
                                    sequence.getParticleList().add(grandchild);
                                    topcomp.replaceChild(i, sequence);
                                    modified = true;
                                    
                                }
                            }
                        }
                        break;
                    }
                    
                    case SchemaBase.EXTENSION_TYPE:
                    {
                        if (childcomp instanceof SimpleExtensionElement) {
                            
                            // replace empty simple type extension with base type
                            SimpleExtensionElement extend = (SimpleExtensionElement)childcomp;
                            if (extend.getAttributeList().size() == 0 && extend.getAnyAttribute() == null) {
                                modified = substituteTypeDerivation(lead, topcomp, childcomp, extend);
                            }
                            
                        } else {
                            
                            // replace empty complex type extension with base type
                            ComplexExtensionElement extend = (ComplexExtensionElement)child;
                            if (extend.getContentDefinition() == null && extend.getAttributeList().size() == 0 &&
                                extend.getAnyAttribute() == null) {
                                modified = substituteTypeDerivation(lead, topcomp, childcomp, extend);
                            }
                            
                        }
                        break;
                    }
                    
                    case SchemaBase.RESTRICTION_TYPE:
                    {
                        if (childcomp instanceof SimpleRestrictionElement) {
                            
                            // replace simple type restriction with base type unless it has facets
                            SimpleRestrictionElement restrict = (SimpleRestrictionElement)childcomp;
                            if (restrict.getFacetsList().size() == 0 && restrict.getBase() != null) {
                                QName base = restrict.getBase();
                                if (base == null) {
                                    
                                    // derivation using inline base type, just eliminate the restriction
                                    
                                } else {
                                    modified = substituteTypeDerivation(lead, topcomp, childcomp, restrict);
                                }
                            }
                            
                        } else {
                            
                            // always replace complex type restriction with base type
                            ComplexRestrictionElement restrict = (ComplexRestrictionElement)child;
                            modified = substituteTypeDerivation(lead, topcomp, childcomp, restrict);
                            
                        }
                        break;
                    }
                }
                
                // delete child component if flagged for removal
                if (childext.isRemoved()) {
                    removeChild(i);
                    compact = true;
                }
            }
        }
        if (compact) {
            topcomp.compactChildren();
            modified = true;
        }
        
        // handle union normalization after all children have been normalized
        if (topcomp.type() == SchemaBase.UNION_TYPE) {
            
            // start by checking duplicates in the member types
            compact = false;
            UnionElement union = (UnionElement)topcomp;
            Set typeset = new HashSet();
            ArrayList keeptypes = new ArrayList();
            QName[] membertypes = union.getMemberTypes();
            if (membertypes != null) {
                for (int i = 0; i < membertypes.length; i++) {
                    QName type = membertypes[i];
                    if (typeset.contains(type)) {
                        if (s_logger.isDebugEnabled()) {
                            s_logger.debug(lead + "removed redundant member type " + type + " from " + path);
                        }
                    } else {
                        typeset.add(type);
                        keeptypes.add(type);
                    }
                }
            }
            
            // then check inline types for duplicates, simplifying where possible
            count = union.getChildCount();
            for (int i = 0; i < count; i++) {
                SchemaBase child = topcomp.getChild(i);
                if (child instanceof OpenAttrBase) {
                    
                    // child schema component must be an inline simple type definition
                    SimpleTypeElement simple = (SimpleTypeElement)child;
                    boolean keeper = false;
                    SchemaBase derivation = simple.getDerivation();
                    QName repltype = null;
                    if (derivation != null) {
                        
                        // keep the inline definition by default, but check for replacement cases
                        keeper = true;
                        if (derivation.type() == SchemaBase.RESTRICTION_TYPE) {
                            SimpleRestrictionElement innerrestrict = (SimpleRestrictionElement)derivation;
                            if (innerrestrict.getChildCount() == 0) {
                                
                                // treat empty restriction the same as a member type from list
                                repltype = innerrestrict.getBase();
                                if (typeset.contains(repltype)) {
                                    keeper = false;
                                }
                                
                            }
                        } else if (derivation.type() == SchemaBase.UNION_TYPE) {
                            UnionElement innerunion = (UnionElement)derivation;
                            QName[] innertypes = innerunion.getMemberTypes();
                            FilteredSegmentList innerinlines = innerunion.getInlineBaseList();
                            if (innertypes.length + innerinlines.size() == 1) {
                                
                                // replace child union with single type from union
                                if (innertypes.length == 1) {
                                    
                                    // global type reference, delete the child union and check if new type
                                    repltype = innertypes[0];
                                    if (typeset.contains(repltype)) {
                                        keeper = false;
                                    }
                                    
                                } else {
                                    
                                    // inline type definition, replace the child union with the inline type
                                    union.replaceChild(i, (SchemaBase)innerinlines.get(0));
                                    
                                }
                            }
                        }
                    }
                    if (keeper) {
                        if (repltype != null) {
                            
                            // convert unnecessary inline type to member type
                            typeset.add(repltype);
                            removeChild(i);
                            compact = true;
                            if (s_logger.isDebugEnabled()) {
                                s_logger.debug(lead + "converted inline type " + SchemaUtils.describeComponent(topcomp) +
                                    " to member type in " + path);
                            }
                            
                        }
                    } else {
                        
                        // remove inline definition that matches already-processed type
                        removeChild(i);
                        compact = true;
                        if (s_logger.isDebugEnabled()) {
                            s_logger.debug(lead + "removed redundant inline type from " + path);
                        }
                        
                    }
                }
            }
            
            // set the new list of member types for union
            QName[] newtypes = null;
            if (keeptypes.size() > 0) {
                newtypes = (QName[])keeptypes.toArray(new QName[keeptypes.size()]);
            }
            union.setMemberTypes(newtypes);
            
            // clean up for any deleted child components
            if (compact) {
                topcomp.compactChildren();
                modified = true;
            }
            
        }
        if (s_logger.isDebugEnabled()) {
            s_logger.debug(lead + "exiting normalization for node " + path + " with modified " + modified);
        }
        return modified;
    }
}




© 2015 - 2024 Weber Informatics LLC | Privacy Policy