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

org.jibx.custom.classes.CustomBase Maven / Gradle / Ivy

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

package org.jibx.custom.classes;

import java.util.Collection;

import org.jibx.runtime.EnumSet;
import org.jibx.runtime.IUnmarshallingContext;
import org.jibx.runtime.impl.UnmarshallingContext;
import org.jibx.schema.validation.ValidationContext;
import org.jibx.util.NameUtilities;
import org.jibx.util.StringArray;

/**
 * Base class for all customizations. This defines a way to navigate up the tree of nested components without making
 * assumptions about the specific type of the containing components. This allows for other types of customizations,
 * beyond the binding customizations included directly in this package. This also includes enumeration definitions which
 * are used with both base and extension customizations.
 * 
 * @author Dennis M. Sosnoski
 */
public class CustomBase
{
    // name style value set information
    public static final int CAMEL_CASE_NAMES = 0;
    
    public static final int UPPER_CAMEL_CASE_NAMES = 1;
    
    public static final int HYPHENATED_NAMES = 2;
    
    public static final int DOTTED_NAMES = 3;
    
    public static final int UNDERSCORED_NAMES = 4;
    
    public static final EnumSet s_nameStyleEnum =
        new EnumSet(CAMEL_CASE_NAMES, new String[] { "camel-case", "upper-camel-case", "hyphenated", "dotted",
            "underscored" });
    
    // require value set information
    public static final int REQUIRE_NONE = 0;
    
    public static final int REQUIRE_PRIMITIVES = 1;
    
    public static final int REQUIRE_OBJECTS = 2;
    
    public static final int REQUIRE_ALL = 3;
    
    public static final EnumSet s_requireEnum =
        new EnumSet(REQUIRE_NONE, new String[] { "none", "primitives", "objects", "all" });
    
    // derive-namespace value set information
    public static final int DERIVE_NONE = 0;
    
    public static final int DERIVE_BY_PACKAGE = 1;
    
    public static final int DERIVE_FIXED = 2;
    
    public static final EnumSet s_namespaceStyleEnum =
        new EnumSet(DERIVE_NONE, new String[] { "none", "package", "fixed" });
    
    // parent element (null if none) - would be final, except for unmarshalling
    private SharedNestingBase m_parent;
    
    /**
     * Constructor.
     * 
     * @param parent
     */
    public CustomBase(SharedNestingBase parent) {
        m_parent = parent;
    }
    
    /**
     * Get container.
     * 
     * @return container
     */
    public SharedNestingBase getParent() {
        return m_parent;
    }
    
    /**
     * Get global customizations root.
     * 
     * @return global customization
     */
    public GlobalCustom getGlobal() {
        CustomBase parent = m_parent;
        while (!(parent instanceof GlobalCustom)) {
            parent = parent.getParent();
        }
        return (GlobalCustom)parent;
    }
    
    /**
     * Convert class, method, or parameter name to XML name.
     * 
     * @param base class or simple field name to be converted
     * @param code conversion format style code
     * @return XML name
     */
    public static String convertName(String base, int code) {
        
        // strip off trailing array indication
        while (base.endsWith("[]")) {
            base = base.substring(0, base.length() - 2);
        }
        
        // skip any leading special characters in name
        int length = base.length();
        int offset = -1;
        char chr = 0;
        while (++offset < length && ((chr = base.charAt(offset)) == '_' || (chr == '$')));
        if (offset >= length) {
            return "_";
        } else {
            
            // make sure valid first character of name
            StringBuffer buff = new StringBuffer();
            if (Character.isDigit(chr)) {
                buff.append('_');
            }
            
            // scan for word splits in supplied name
            boolean split = true;
            boolean caps = false;
            while (offset < length) {
                
                // split if underscore or change to upper case, or upper ending
                chr = base.charAt(offset++);
                boolean nextlower = offset < length && !Character.isUpperCase(base.charAt(offset));
                if (chr == '_') {
                    split = true;
                    continue;
                } else if (Character.isUpperCase(chr)) {
                    if (!caps || nextlower) {
                        split = true;
                    }
                    if (code != CAMEL_CASE_NAMES && code != UPPER_CAMEL_CASE_NAMES) {
                        chr = Character.toLowerCase(chr);
                    }
                }
                if (split) {
                    
                    // convert word break
                    caps = !nextlower;
                    boolean tolower = false;
                    char separator = 0;
                    switch (code) {
                        
                        case CAMEL_CASE_NAMES: {
                            if (buff.length() != 0) {
                                chr = Character.toUpperCase(chr);
                                tolower = false;
                            } else {
                                tolower = nextlower || offset == length;
                            }
                            break;
                        }
                            
                        case UPPER_CAMEL_CASE_NAMES: {
                            tolower = false;
                            chr = Character.toUpperCase(chr);
                            break;
                        }
                            
                        case HYPHENATED_NAMES: {
                            separator = '-';
                            break;
                        }
                            
                        case DOTTED_NAMES: {
                            separator = '.';
                            break;
                        }
                            
                        case UNDERSCORED_NAMES: {
                            separator = '_';
                            break;
                        }
                            
                    }
                    if (separator > 0 && buff.length() > 0) {
                        buff.append(separator);
                    }
                    if (tolower) {
                        chr = Character.toLowerCase(chr);
                    }
                    split = false;
                    
                }
                if (chr != '$') {
                    
                    // no split, just append the character
                    buff.append(chr);
                    
                }
            }
            return buff.toString();
        }
    }
    
    /**
     * Derive name for item in a collection. If the supplied collection name ends in a recognized plural form the
     * derived item name is the singular version of the collection name. Otherwise, it is the converted name of the
     * collection item class, or just "item" if the class is unknown. TODO: internationalization?
     * 
     * @param cname collection name (null if none)
     * @param type item type (null if unknown)
     * @param code conversion format style code
     * @return item name
     */
    public static String deriveItemName(String cname, String type, int code) {
        if (cname != null) {
            String name = NameUtilities.depluralize(cname);
            if (!name.equals(cname)) {
                return name;
            }
        }
        if (type != null && !"java.lang.Object".equals(type)) {
            return convertName(type.substring(type.lastIndexOf('.') + 1), code);
        } else {
            return "item";
        }
    }
    
    /**
     * Get the package from a fully-qualified type name.
     * 
     * @param type fully-qualified type name
     * @return package of the type (empty string if in default package)
     */
    public static String packageOfType(String type) {
        int split = type.lastIndexOf('.');
        if (split >= 0) {
            return type.substring(0, split);
        } else {
            return "";
        }
    }
    
    /**
     * Create a namespace URL from a package path.
     * 
     * @param pkgpth fully-qualified package name
     * @return namespace based on package (null if none)
     */
    public static String packageToNamespace(String pkgpth) {
        int mark = pkgpth.indexOf('.');
        if (mark >= 0) {
            StringBuffer buff = new StringBuffer();
            buff.append("http://");
            String comp = pkgpth.substring(0, mark);
            int base = mark + 1;
            char delim = '.';
            if ("com".equals(comp) || "net".equals(comp) || "org".equals(comp)) {
                mark = pkgpth.indexOf('.', base);
                if (mark > 0) {
                    buff.append(pkgpth.substring(base, mark));
                } else {
                    buff.append(pkgpth.substring(base));
                }
                buff.append('.');
                base = mark + 1;
                delim = '/';
            }
            buff.append(comp);
            while (mark > 0) {
                buff.append(delim);
                base = mark + 1;
                mark = pkgpth.indexOf('.', base);
                if (mark > 0) {
                    buff.append(pkgpth.substring(base, mark));
                } else {
                    buff.append(pkgpth.substring(base));
                }
            }
            return buff.toString();
        } else {
            return null;
        }
    }
    
    /**
     * Derive namespace using specified technique.
     * 
     * @param uri base namespace URI (null if none)
     * @param pkgpth fully qualified package name
     * @param style namespace style code
     * @return derived namespace
     */
    public static String deriveNamespace(String uri, String pkgpth, int style) {
        switch (style) {
            case DERIVE_NONE:
                return null;
                
            case DERIVE_FIXED:
                return uri;
                
            case DERIVE_BY_PACKAGE: {
                if (uri == null) {
                    return packageToNamespace(pkgpth);
                } else if (pkgpth == null) {
                    return uri;
                } else {
                    
                    // append the last package to passed URI
                    String pack;
                    int start = pkgpth.lastIndexOf('.');
                    if (start >= 0) {
                        pack = pkgpth.substring(start + 1);
                    } else {
                        pack = pkgpth;
                    }
                    if (uri.endsWith("/")) {
                        return uri + pack;
                    } else {
                        return uri + '/' + pack;
                    }
                }
            }
                
            default:
                throw new IllegalStateException("Invalid style code");
        }
    }
    
    /**
     * Validate attributes of element. This is designed to be called during unmarshalling as part of the pre-set method
     * processing when a subclass instance is being created.
     * 
     * @param ictx unmarshalling context
     * @param attrs attributes array
     */
    protected void validateAttributes(IUnmarshallingContext ictx, StringArray attrs) {
        
        // setup for attribute access
        ValidationContext vctx = (ValidationContext)ictx.getUserContext();
        UnmarshallingContext uctx = (UnmarshallingContext)ictx;
        
        // loop through all attributes of current element
        for (int i = 0; i < uctx.getAttributeCount(); i++) {
            
            // check if nonamespace attribute is in the allowed set
            String name = uctx.getAttributeName(i);
            if (uctx.getAttributeNamespace(i).length() == 0) {
                if (attrs.indexOf(name) < 0) {
                    vctx.addWarning("Undefined attribute " + name, this);
                }
            }
        }
    }
    
    /**
     * Gets the parent element link from the unmarshalling stack. This method is for use by factories during
     * unmarshalling.
     * 
     * @param ictx unmarshalling context
     * @return containing class
     */
    protected static Object getContainingObject(IUnmarshallingContext ictx) {
        Object parent = ictx.getStackTop();
        if (parent instanceof Collection) {
            parent = ictx.getStackObject(1);
        }
        return parent;
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy