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

org.jibx.schema.TreeWalker Maven / Gradle / Ivy

/*
 * Copyright (c) 2006-2007, 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;

import org.apache.log4j.Level;
import org.apache.log4j.Logger;
import org.jibx.schema.elements.AllElement;
import org.jibx.schema.elements.AnnotatedBase;
import org.jibx.schema.elements.AnnotationElement;
import org.jibx.schema.elements.AnyAttributeElement;
import org.jibx.schema.elements.AnyElement;
import org.jibx.schema.elements.AppInfoElement;
import org.jibx.schema.elements.AttributeElement;
import org.jibx.schema.elements.AttributeGroupElement;
import org.jibx.schema.elements.AttributeGroupRefElement;
import org.jibx.schema.elements.ChoiceElement;
import org.jibx.schema.elements.ComplexContentElement;
import org.jibx.schema.elements.ComplexExtensionElement;
import org.jibx.schema.elements.ComplexRestrictionElement;
import org.jibx.schema.elements.ComplexTypeElement;
import org.jibx.schema.elements.DocumentationElement;
import org.jibx.schema.elements.ElementElement;
import org.jibx.schema.elements.FacetElement;
import org.jibx.schema.elements.GroupElement;
import org.jibx.schema.elements.GroupRefElement;
import org.jibx.schema.elements.ImportElement;
import org.jibx.schema.elements.IncludeElement;
import org.jibx.schema.elements.KeyBase;
import org.jibx.schema.elements.ListElement;
import org.jibx.schema.elements.NotationElement;
import org.jibx.schema.elements.OpenAttrBase;
import org.jibx.schema.elements.RedefineElement;
import org.jibx.schema.elements.SchemaBase;
import org.jibx.schema.elements.SchemaElement;
import org.jibx.schema.elements.SchemaLocationBase;
import org.jibx.schema.elements.SelectionBase;
import org.jibx.schema.elements.SequenceElement;
import org.jibx.schema.elements.SimpleContentElement;
import org.jibx.schema.elements.SimpleExtensionElement;
import org.jibx.schema.elements.SimpleRestrictionElement;
import org.jibx.schema.elements.SimpleTypeElement;
import org.jibx.schema.elements.UnionElement;

/**
 * Handles walking the tree structure of schema model. This traverses the structure defined by the nesting of elements
 * and schema references in the XML representation.
 * 
 * @author Dennis M. Sosnoski
 */
public class TreeWalker
{
    /** Logger for class. */
    private static final Logger s_logger = Logger.getLogger(TreeWalker.class.getName());
    
    /** Selector for elements to be skipped when walking tree (null if unused). */
    private final ISkipElements m_skipSet;
    
    /** Listener for entering and exiting referenced schemas. (null if unused). */
    private final ISchemaListener m_schemaListener;
    
    /**
     * Constructor.
     * 
     * @param skip selector for elements to be skipped (null if none skipped)
     * @param listen schema reference listener (null if none)
     */
    public TreeWalker(ISkipElements skip, ISchemaListener listen) {
        m_skipSet = skip;
        m_schemaListener = listen;
    }
    
    /**
     * Control the logging level for this class. Since the generated logs at debug level can become huge, this gives a
     * way for external code to provide granular control over the logging.
     *
     * @param level
     * @return prior level
     */
    public static Level setLogging(Level level) {
        Level prior = s_logger.getLevel();
        s_logger.setLevel(level);
        return prior;
    }
    
    /**
     * Walk entire schema model.
     * 
     * @param schema root element of schema to be traversed
     * @param visitor target visitor for element notifications
     */
    public void walkSchema(SchemaElement schema, SchemaVisitor visitor) {
        if (schema != null) {
            if (m_schemaListener == null || m_schemaListener.enterSchema(schema)) {
                walkElement(schema, visitor);
                if (m_schemaListener != null) {
                    m_schemaListener.exitSchema();
                }
            }
        }
    }
    
    /**
     * Walk schema model element tree. This recursively traverses the schema model tree rooted in the supplied element,
     * including the element itself, notifying the visitor of each element visited during the traversal.
     * 
     * @param root node of tree to be toured
     * @param visitor target visitor for element notifications
     */
    public void walkElement(SchemaBase root, SchemaVisitor visitor) {
        
        // check for fatal error on element
        if (m_skipSet != null && m_skipSet.isSkipped(root)) {
            if (s_logger.isDebugEnabled() && root instanceof OpenAttrBase) {
                s_logger.debug("Skipping node " + SchemaUtils.componentPath((OpenAttrBase)root));
            }
            return;
        }
        
        // visit the actual root of tree
        boolean expand = false;
        if (s_logger.isDebugEnabled() && root instanceof OpenAttrBase) {
            s_logger.debug("Entering node " + SchemaUtils.componentPath((OpenAttrBase)root));
        }
        switch (root.type()) {
            
            case SchemaBase.ALL_TYPE:
                expand = visitor.visit((AllElement)root);
                break;
            
            case SchemaBase.ANNOTATION_TYPE:
                expand = visitor.visit((AnnotationElement)root);
                break;
            
            case SchemaBase.ANY_TYPE:
                expand = visitor.visit((AnyElement)root);
                break;
            
            case SchemaBase.ANYATTRIBUTE_TYPE:
                expand = visitor.visit((AnyAttributeElement)root);
                break;
            
            case SchemaBase.APPINFO_TYPE:
                expand = visitor.visit((AppInfoElement)root);
                break;
            
            case SchemaBase.ATTRIBUTE_TYPE:
                expand = visitor.visit((AttributeElement)root);
                break;
            
            case SchemaBase.ATTRIBUTEGROUP_TYPE:
                if (root instanceof AttributeGroupElement) {
                    expand = visitor.visit((AttributeGroupElement)root);
                } else {
                    expand = visitor.visit((AttributeGroupRefElement)root);
                }
                break;
            
            case SchemaBase.CHOICE_TYPE:
                expand = visitor.visit((ChoiceElement)root);
                break;
            
            case SchemaBase.COMPLEXCONTENT_TYPE:
                expand = visitor.visit((ComplexContentElement)root);
                break;
            
            case SchemaBase.COMPLEXTYPE_TYPE:
                expand = visitor.visit((ComplexTypeElement)root);
                break;
            
            case SchemaBase.DOCUMENTATION_TYPE:
                expand = visitor.visit((DocumentationElement)root);
                break;
            
            case SchemaBase.ELEMENT_TYPE:
                expand = visitor.visit((ElementElement)root);
                break;
            
            case SchemaBase.EXTENSION_TYPE:
                if (root instanceof ComplexExtensionElement) {
                    expand = visitor.visit((ComplexExtensionElement)root);
                } else {
                    expand = visitor.visit((SimpleExtensionElement)root);
                }
                break;
            
            case SchemaBase.FIELD_TYPE:
                expand = visitor.visit((SelectionBase.FieldElement)root);
                break;
            
            case SchemaBase.GROUP_TYPE:
                if (root instanceof GroupElement) {
                    expand = visitor.visit((GroupElement)root);
                } else {
                    expand = visitor.visit((GroupRefElement)root);
                }
                break;
            
            case SchemaBase.IMPORT_TYPE:
                expand = visitor.visit((ImportElement)root);
                break;
            
            case SchemaBase.INCLUDE_TYPE:
                expand = visitor.visit((IncludeElement)root);
                break;
            
            case SchemaBase.KEY_TYPE:
                expand = visitor.visit((KeyBase.KeyElement)root);
                break;
            
            case SchemaBase.KEYREF_TYPE:
                expand = visitor.visit((KeyBase.KeyrefElement)root);
                break;
            
            case SchemaBase.LIST_TYPE:
                expand = visitor.visit((ListElement)root);
                break;
            
            case SchemaBase.NOTATION_TYPE:
                expand = visitor.visit((NotationElement)root);
                break;
            
            case SchemaBase.REDEFINE_TYPE:
                expand = visitor.visit((RedefineElement)root);
                break;
            
            case SchemaBase.RESTRICTION_TYPE:
                if (root instanceof SimpleRestrictionElement) {
                    expand = visitor.visit((SimpleRestrictionElement)root);
                } else {
                    expand = visitor.visit((ComplexRestrictionElement)root);
                }
                break;
            
            case SchemaBase.SCHEMA_TYPE:
                expand = visitor.visit((SchemaElement)root);
                break;
            
            case SchemaBase.SELECTOR_TYPE:
                expand = visitor.visit((SelectionBase.SelectorElement)root);
                break;
            
            case SchemaBase.SEQUENCE_TYPE:
                expand = visitor.visit((SequenceElement)root);
                break;
            
            case SchemaBase.SIMPLECONTENT_TYPE:
                expand = visitor.visit((SimpleContentElement)root);
                break;
            
            case SchemaBase.SIMPLETYPE_TYPE:
                expand = visitor.visit((SimpleTypeElement)root);
                break;
            
            case SchemaBase.UNION_TYPE:
                expand = visitor.visit((UnionElement)root);
                break;
            
            case SchemaBase.UNIQUE_TYPE:
                expand = visitor.visit((KeyBase.UniqueElement)root);
                break;
            
            case SchemaBase.ENUMERATION_TYPE:
            case SchemaBase.FRACTIONDIGITS_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.TOTALDIGITS_TYPE:
            case SchemaBase.WHITESPACE_TYPE:
                expand = visitor.visit((FacetElement)root);
                break;
            
            default:
                throw new IllegalStateException("Internal error: unknown element type");
                
        }
        
        // check for expansion needed
        if (expand && (m_skipSet == null || !m_skipSet.isSkipped(root))) {
            walkChildren(root, visitor);
        }
        
        // exit the actual root of tree
        switch (root.type()) {
            
            case SchemaBase.ALL_TYPE:
                visitor.exit((AllElement)root);
                break;
            
            case SchemaBase.ANNOTATION_TYPE:
                visitor.exit((AnnotationElement)root);
                break;
            
            case SchemaBase.ANY_TYPE:
                visitor.exit((AnyElement)root);
                break;
            
            case SchemaBase.ANYATTRIBUTE_TYPE:
                visitor.exit((AnyAttributeElement)root);
                break;
            
            case SchemaBase.APPINFO_TYPE:
                visitor.exit((AppInfoElement)root);
                break;
            
            case SchemaBase.ATTRIBUTE_TYPE:
                visitor.exit((AttributeElement)root);
                break;
            
            case SchemaBase.ATTRIBUTEGROUP_TYPE:
                if (root instanceof AttributeGroupElement) {
                    visitor.exit((AttributeGroupElement)root);
                } else {
                    visitor.exit((AttributeGroupRefElement)root);
                }
                break;
            
            case SchemaBase.CHOICE_TYPE:
                visitor.exit((ChoiceElement)root);
                break;
            
            case SchemaBase.COMPLEXCONTENT_TYPE:
                visitor.exit((ComplexContentElement)root);
                break;
            
            case SchemaBase.COMPLEXTYPE_TYPE:
                visitor.exit((ComplexTypeElement)root);
                break;
            
            case SchemaBase.DOCUMENTATION_TYPE:
                visitor.exit((DocumentationElement)root);
                break;
            
            case SchemaBase.ELEMENT_TYPE:
                visitor.exit((ElementElement)root);
                break;
            
            case SchemaBase.EXTENSION_TYPE:
                if (root instanceof ComplexExtensionElement) {
                    visitor.exit((ComplexExtensionElement)root);
                } else {
                    visitor.exit((SimpleExtensionElement)root);
                }
                break;
            
            case SchemaBase.FIELD_TYPE:
                visitor.exit((SelectionBase.FieldElement)root);
                break;
            
            case SchemaBase.GROUP_TYPE:
                if (root instanceof GroupElement) {
                    visitor.exit((GroupElement)root);
                } else {
                    visitor.exit((GroupRefElement)root);
                }
                break;
            
            case SchemaBase.IMPORT_TYPE:
                visitor.exit((ImportElement)root);
                break;
            
            case SchemaBase.INCLUDE_TYPE:
                visitor.exit((IncludeElement)root);
                break;
            
            case SchemaBase.KEY_TYPE:
                visitor.exit((KeyBase.KeyElement)root);
                break;
            
            case SchemaBase.KEYREF_TYPE:
                visitor.exit((KeyBase.KeyrefElement)root);
                break;
            
            case SchemaBase.LIST_TYPE:
                visitor.exit((ListElement)root);
                break;
            
            case SchemaBase.NOTATION_TYPE:
                visitor.exit((NotationElement)root);
                break;
            
            case SchemaBase.REDEFINE_TYPE:
                visitor.exit((RedefineElement)root);
                break;
            
            case SchemaBase.RESTRICTION_TYPE:
                if (root instanceof SimpleRestrictionElement) {
                    visitor.exit((SimpleRestrictionElement)root);
                } else {
                    visitor.exit((ComplexRestrictionElement)root);
                }
                break;
            
            case SchemaBase.SCHEMA_TYPE:
                visitor.exit((SchemaElement)root);
                break;
            
            case SchemaBase.SELECTOR_TYPE:
                visitor.exit((SelectionBase.SelectorElement)root);
                break;
            
            case SchemaBase.SEQUENCE_TYPE:
                visitor.exit((SequenceElement)root);
                break;
            
            case SchemaBase.SIMPLECONTENT_TYPE:
                visitor.exit((SimpleContentElement)root);
                break;
            
            case SchemaBase.SIMPLETYPE_TYPE:
                visitor.exit((SimpleTypeElement)root);
                break;
            
            case SchemaBase.UNION_TYPE:
                visitor.exit((UnionElement)root);
                break;
            
            case SchemaBase.UNIQUE_TYPE:
                visitor.exit((KeyBase.UniqueElement)root);
                break;
            
            case SchemaBase.ENUMERATION_TYPE:
            case SchemaBase.FRACTIONDIGITS_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.TOTALDIGITS_TYPE:
            case SchemaBase.WHITESPACE_TYPE:
                visitor.exit((FacetElement)root);
                break;
            
            default:
                break;
            
        }
        if (s_logger.isDebugEnabled() && root instanceof OpenAttrBase) {
            s_logger.debug("Left node " + SchemaUtils.componentPath((OpenAttrBase)root));
        }
    }
    
    /**
     * Walk the descendants of a root element. This recursively traverses the schema model tree rooted in the supplied
     * element, excluding the element itself, notifying the visitor of each element visited during the traversal.
     * 
     * @param root node of tree to be toured
     * @param visitor target visitor for element notifications
     */
    public void walkChildren(SchemaBase root, SchemaVisitor visitor) {
        
        // first expand the annotation, if present
        if (root instanceof AnnotatedBase) {
            AnnotationElement anno = ((AnnotatedBase)root).getAnnotation();
            if (anno != null) {
                walkElement(anno, visitor);
            }
        }
        
        // perform special handling for element expansion
        if (root instanceof SchemaLocationBase) {
            
            // references just delegate to the included schema element
            SchemaElement schema = ((SchemaLocationBase)root).getReferencedSchema();
            s_logger.debug(" checking referenced schema " + schema.getResolver().getName());
            if (schema != null && (m_schemaListener == null || m_schemaListener.enterSchema(schema))) {
                walkElement(schema, visitor);
                if (m_schemaListener != null) {
                    m_schemaListener.exitSchema();
                }
            }
        }
        
        // handle normal element expansion
        if (root instanceof OpenAttrBase) {
            OpenAttrBase parent = (OpenAttrBase)root;
            int count = parent.getChildCount();
            for (int i = 0; i < count; i++) {
                walkElement(parent.getChild(i), visitor);
            }
        }
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy