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

org.jibx.schema.codegen.custom.SchemaCustom 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.custom;

import java.util.Collections;
import java.util.HashSet;
import java.util.Iterator;
import java.util.List;
import java.util.Set;

import org.apache.log4j.Logger;
import org.jibx.binding.util.ObjectStack;
import org.jibx.custom.CustomUtils;
import org.jibx.runtime.IUnmarshallingContext;
import org.jibx.schema.INamed;
import org.jibx.schema.SchemaContextTracker;
import org.jibx.schema.SchemaUtils;
import org.jibx.schema.SchemaVisitor;
import org.jibx.schema.TreeWalker;
import org.jibx.schema.codegen.PackageHolder;
import org.jibx.schema.elements.AnnotatedBase;
import org.jibx.schema.elements.FacetElement;
import org.jibx.schema.elements.OpenAttrBase;
import org.jibx.schema.elements.SchemaBase;
import org.jibx.schema.elements.SchemaElement;
import org.jibx.schema.elements.SchemaPath;
import org.jibx.schema.validation.ValidationContext;
import org.jibx.util.StringArray;

/**
 * Individual schema customization information.
 * 
 * @author Dennis M. Sosnoski
 */
public class SchemaCustom extends SchemaRootBase
{
    /** Logger for class. */
    private static final Logger s_logger = Logger.getLogger(SchemaCustom.class.getName());
    
    /** Enumeration of allowed attribute names */
    public static final StringArray s_allowedAttributes =
        new StringArray(new String[] { "excludes", "force-types", "includes", "name", "namespace" },
        SchemaRootBase.s_allowedAttributes);
    
    //
    // Bound instance data
    
    /** Schema name. */
    private String m_name;
    
    /** Schema namespace. */
    private String m_namespace;
    
    /** Always specify property types flag. */
    private boolean m_forceTypes;
    
    /** Global names included in code generation. */
    private String[] m_includes;
    
    /** Global names excluded from code generation. */
    private String[] m_excludes;
    
    //
    // Internal instance data
    
    /** Schema definition. */
    private SchemaElement m_schema;
    
    /** Extension attached to actual schema element (only used for children). */
    private SchemaExtension m_extension;
    
    /**
     * Constructor.
     * 
     * @param parent
     */
    public SchemaCustom(SchemasetCustom parent) {
        super(parent);
    }
    
    /**
     * Constructor for constructing instance directly.
     * 
     * @param parent
     * @param name schema name
     * @param namespace target namespace URI
     * @param includes definition names to be included in generation
     * @param excludes definition names to be excluded from generation
     */
    public SchemaCustom(SchemasetCustom parent, String name, String namespace, String[] includes, String[] excludes) {
        this(parent);
        m_name = name;
        m_namespace = namespace;
        m_includes = includes;
        m_excludes = excludes;
    }
    
    /**
     * Make sure all attributes are defined.
     * 
     * @param uctx unmarshalling context
     */
    private void preSet(IUnmarshallingContext uctx) {
        validateAttributes(uctx, s_allowedAttributes);
    }
    
    /**
     * Get the names of definitions to be included in generation.
     *
     * @return names
     */
    public String[] getIncludes() {
        return m_includes;
    }
    
    /**
     * Set the names of definitions to be included in generation. This only has any effect on the code generation if
     * called before {@link #extend(PackageHolder, ValidationContext)} is called.
     *
     * @param includes
     */
    public void setIncludes(String[] includes) {
        m_includes = includes;
    }
    
    /**
     * Get the names of definitions to be excluded from generation.
     *
     * @return names
     */
    public String[] getExcludes() {
        return m_includes;
    }
    
    /**
     * Set the names of definitions to be excluded from generation. This only has any effect on the code generation if
     * called before {@link #extend(PackageHolder, ValidationContext)} is called.
     *
     * @param excludes
     */
    public void setExcludes(String[] excludes) {
        m_excludes = excludes;
    }
    
    /**
     * Get schema name.
     * 
     * @return name
     */
    public String getName() {
        return m_name;
    }
    
    /**
     * Set schema name.
     * 
     * @param name
     */
    public void setName(String name) {
        m_name = name;
    }
    
    /**
     * Get schema namespace.
     * 
     * @return namespace
     */
    public String getNamespace() {
        return m_namespace;
    }
    
    /**
     * Set schema namespace.
     * 
     * @param namespace
     */
    public void setNamespace(String namespace) {
        m_namespace = namespace;
    }
    
    /**
     * Check if type specifications forced for schema.
     * 
     * @return true if types forced, false if not
     */
    public boolean isForceTypes() {
        return m_forceTypes;
    }
    
    /**
     * Set type specifications forced for schema.
     * 
     * @param force true if types forced, false if not
     */
    public void setForceTypes(Boolean force) {
        m_forceTypes = force.booleanValue();
    }
    
    /**
     * Get schema definition.
     * 
     * @return schema
     */
    public SchemaElement getSchema() {
        return m_schema;
    }
    
    /**
     * Set schema definition.
     * 
     * @param name
     * @param schema
     */
    public void setSchema(String name, SchemaElement schema) {
        if (m_name == null) {
            m_name = name;
        }
        if (m_namespace == null) {
            m_namespace = schema.getEffectiveNamespace();
        }
        m_schema = schema;
    }
    
    /**
     * Check if this customization matches a particular schema.
     * 
     * @param name
     * @param schema
     * @return true if a match, false if not
     */
    public boolean checkMatch(String name, SchemaElement schema) {
        return (m_name == null || (m_name.equals(name)))
            && (m_namespace == null || (m_namespace.equals(schema.getEffectiveNamespace())));
    }
    
    /**
     * Build the extensions tree for a global definition.
     * 
     * @param visitor
     * @param wlkr
     * @param anno
     */
    private void extendGlobal(ExtensionBuilderVisitor visitor, TreeWalker wlkr, GlobalExtension anno) {
        visitor.setRoot(anno);
        wlkr.walkChildren(anno.getComponent(), visitor);
    }
    
    /**
     * Strip the annotation components (at any level) from a schema definitions.
     */
    public void stripAnnotations() {
            
        // first delete all annotation children of the schema element itself
        TreeWalker wlkr = new TreeWalker(null, new SchemaContextTracker());
        for (int i = 0; i < m_schema.getChildCount(); i++) {
            if (m_schema.getChild(i).type() == SchemaBase.ANNOTATION_TYPE) {
                m_schema.detachChild(i);
            }
        }
        m_schema.compactChildren();
        
        // now remove all embedded annotations within any definitions
        wlkr.walkSchema(m_schema, new AnnotationDeletionVisitor());
    }
    
    /**
     * Evaluate the remaining path for a customization after the first step, and apply it to the extension for each
     * matching schema component. If no matches are found or multiple matches are found this generates a warning.
     *
     * @param path customization path
     * @param match starting point for path
     * @param custom customization information
     * @param vctx validation context
     */
    private static void applyRemainingCustomizationPath(SchemaPath path, OpenAttrBase match, ComponentCustom custom,
        ValidationContext vctx) {
        List matches = path.partialMatchMultiple(1, path.getPathLength()-1, match);
        if (matches.size() == 0) {
            vctx.addWarning("No matches found for customization expression", custom);
        } else {
            if (matches.size() > 1) {
                vctx.addWarning("Found " + matches.size() + " matches for customization expression", custom);
            }
            for (Iterator iter = matches.iterator(); iter.hasNext();) {
                OpenAttrBase target = (OpenAttrBase)iter.next();
                custom.apply((ComponentExtension)target.getExtension(), vctx);
            }
        }
    }

    /**
     * Build the schema extension structure. This first builds extensions for all the global definitions in the schema,
     * marking the ones specified to be included or excluded from the schema, and for all the child components of the
     * non-excluded globals. It then applies the customizations to the extensions.
     * 
     * @param pack package for generated classes (null if no code generation)
     * @param vctx validation context
     */
    public void extend(PackageHolder pack, ValidationContext vctx) {
        
        // build the basic include and exclude sets
        Set inclset = Collections.EMPTY_SET;
        Set exclset = Collections.EMPTY_SET;
        if (m_includes != null) {
            inclset = CustomUtils.nameSet(m_includes);
        }
        if (m_excludes != null) {
            exclset = CustomUtils.nameSet(m_excludes);
        }
        
        // check for any conflicts between the two
        if (!inclset.isEmpty() && !exclset.isEmpty()) {
            for (Iterator iter = inclset.iterator(); iter.hasNext();) {
                Object next = iter.next();
                if (exclset.contains(next)) {
                    vctx.addError("Name '" + next.toString() + "' cannot be on both include and exclude list", this);
                }
            }
        }
        
        // build the extensions for each global definition component (and descendant components)
        m_extension = new SchemaExtension(m_schema, this, pack);
        List globals = m_schema.getTopLevelChildren();
        ExtensionBuilderVisitor builder = new ExtensionBuilderVisitor();
        // Level level = TreeWalker.setLogging(s_logger.getLevel());
        Set foundset = Collections.EMPTY_SET;
        if (m_includes != null || m_excludes != null) {
            foundset = new HashSet();
        }
        TreeWalker wlkr = new TreeWalker(null, new SchemaContextTracker());
        for (Iterator iter = globals.iterator(); iter.hasNext();) {
            OpenAttrBase global = (OpenAttrBase)iter.next();
            if (global instanceof INamed) {
                
                // set up the basic global definition extension
                GlobalExtension exten = new GlobalExtension(m_extension, global);
                String name = ((INamed)global).getName();
                boolean exclude = exclset.contains(name);
                exten.setRemoved(exclude);
                if (exclude) {
                    foundset.add(name);
                }
                boolean include = inclset.contains(name);
                exten.setIncluded(include);
                if (include) {
                    foundset.add(name);
                }
                
                // initialize extensions for full tree of non-excluded global definition components
                if (!exten.isRemoved()) {
                    extendGlobal(builder, wlkr, exten);
                } else if (global.type() == SchemaBase.SIMPLETYPE_TYPE ||
                    global.type() == SchemaBase.COMPLEXTYPE_TYPE) {
                    setReplacement(((INamed)global).getQName(), null);
                }
            }
        }
        // TreeWalker.setLogging(level);
        
        // report any names not found
        if (foundset.size() < exclset.size() + inclset.size()) {
            Set mergeset = new HashSet();
            mergeset.addAll(exclset);
            mergeset.addAll(inclset);
            for (Iterator iterator = mergeset.iterator(); iterator.hasNext();) {
                String name = (String)iterator.next();
                if (!foundset.contains(name)) {
                    vctx.addWarning("Name '" + name + "' not found in schema", this);
                }
            }
        }
        
        // use child customizations to amend the generated extensions
        int size = getChildren().size();
        for (int i = 0; i < size; i++) {
            ComponentCustom custom = (ComponentCustom)getChildren().get(i);
            SchemaPath path = custom.buildPath(vctx);
            if (path != null) {
                if (path.isWildStart()) {
                    vctx.addError("Top level customizations cannot use wildcard as first step", custom);
                } else {
                    
                    // match only the first path step
                    OpenAttrBase match = path.partialMatchUnique(0, 0, m_schema);
                    if (s_logger.isDebugEnabled()) {
                        if (match == null) {
                            s_logger.debug("No global schema component found for customization " + custom);
                        } else {
                            s_logger.debug("Matched customization " + custom + " to global schema component "
                                + SchemaUtils.describeComponent(match));
                        }
                    }
                    if (match != null) {
                        String name = ((INamed)match).getName();
                        GlobalExtension exten = (GlobalExtension)match.getExtension();
                        if (custom.isExcluded()) {
                            
                            // check if customization applies to descendant of global definition component
                            if (path.getPathLength() == 1) {
                                
                                // force exclude if generation skipped by customization
                                if (exten.isIncluded()) {
                                    vctx.addWarning("Name '" + name
                                        + "' is on include list for schema, but excluded by customization", custom);
                                }
                                exten.setIncluded(false);
                                exten.setRemoved(true);
                                
                            } else {
                                
                                // apply customization to target component extension(s)
                                applyRemainingCustomizationPath(path, match, custom, vctx);
                                
                            }
                            
                        } else {
                            
                            // check for customization for global excluded at schema level
                            if (exten.isRemoved()) {
                                vctx.addWarning("Name '" + name
                                    + "' is on excludes list for schema, but has a customization", custom);
                                exten.setRemoved(false);
                                extendGlobal(builder, wlkr, exten);
                            }
                            
                            // check if customization applies to descendant of global definition component
                            if (path.getPathLength() > 1) {
                                applyRemainingCustomizationPath(path, match, custom, vctx);
                            } else {
                                custom.apply((ComponentExtension)match.getExtension(), vctx);
                            }
                            
                        }
                    }
                }
            }
        }
        
        // flag extensions for facets to be removed from schema
        SchemaVisitor visitor = new FacetRemoverVisitor(this);
        for (Iterator iter = globals.iterator(); iter.hasNext();) {
            OpenAttrBase global = (OpenAttrBase)iter.next();
            ComponentExtension exten = (ComponentExtension)global.getExtension();
            if (exten != null && !exten.isRemoved()) {
                wlkr.walkElement(global, visitor);
            }
        }
    }

    /**
     * Factory used during unmarshalling.
     * 
     * @param ictx
     * @return instance
     */
    private static SchemaCustom factory(IUnmarshallingContext ictx) {
        return new SchemaCustom((SchemasetCustom)getContainingObject(ictx));
    }

    /**
     * Visitor to delete annotations from schema components.
     */
    private static class AnnotationDeletionVisitor extends SchemaVisitor
    {
        /**
         * Visit any component of schema definition. This just deletes any annotation present.
         * 
         * @param node
         * @return true to continue expansion
         */
        public boolean visit(AnnotatedBase node) {
            node.setAnnotation(null);
            return true;
        }
    }
    
    /**
     * Visitor to build basic extensions for schema components. This also sets class and base names for the extensions,
     * if the component has a name.
     */
    private static class ExtensionBuilderVisitor extends SchemaVisitor
    {
        /** Extension for root component being expanded. */
        private GlobalExtension m_root;
        
        /**
         * Set the extension for the root of the schema definition component to be expanded.
         * 
         * @param root
         */
        public void setRoot(GlobalExtension root) {
            m_root = root;
        }
        
        /**
         * Visit any component of schema definition. This just creates the extension for the component.
         * 
         * @param node
         * @return true to continue expansion
         */
        public boolean visit(AnnotatedBase node) {
            node.setExtension(new ComponentExtension(node, m_root));
            return true;
        }
    }
    
    /**
     * Visitor to flag extensions to remove unused facets. This relies on each customization being set as the type
     * substitution handler for the corresponding extension.
     */
    private static class FacetRemoverVisitor extends SchemaVisitor
    {
        /** Stack of active customizations. */
        private ObjectStack m_customStack;
        
        /** Currently active customization. */
        private NestingCustomBase m_currentCustom;
        
        /**
         * Constructor.
         * 
         * @param root customization for root element being processed
         */
        public FacetRemoverVisitor(SchemaCustom root) {
            m_customStack = new ObjectStack();
            m_currentCustom = root;
        }
        
        /**
         * Exit the generic precursor class of all elements which can have customizations. This just pops the saved
         * customization for the higher level off the stack.
         * 
         * @param node
         */
        public void exit(AnnotatedBase node) {
            super.exit(node);
            m_currentCustom = (NestingCustomBase)m_customStack.pop();
        }
        
        /**
         * Visit a facet element. This first calls the handling for the supertype, in order to activate a customization
         * that applies to this particular element, then checks if the facet element subtype is to be included in the
         * code generation.
         * 
         * @param node
         * @return true if continuing expansion, false if not
         */
        public boolean visit(FacetElement node) {
            boolean ret = super.visit(node);
            if ((m_currentCustom.getActiveFacetsMask() & node.bit()) == 0) {
                ((ComponentExtension)node.getExtension()).setRemoved(true);
            }
            return ret;
        }
        
        /**
         * Visit the generic precursor class of all elements which can have customizations. This saves the current
         * customization on the stack, then checks for one associated with the current element and makes that active if
         * found.
         * 
         * @param node
         * @return true if continuing expansion, false if not
         */
        public boolean visit(AnnotatedBase node) {
            m_customStack.push(m_currentCustom);
            NestingCustomBase custom = (NestingCustomBase)(((ComponentExtension)node.getExtension())).getCustom();
            if (custom != null) {
                m_currentCustom = custom;
            }
            return super.visit(node);
        }
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy