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

org.jibx.custom.CustomizationCommandLineBase Maven / Gradle / Ivy

/*
 * 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.custom;

import java.io.File;
import java.io.IOException;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.HashMap;
import java.util.Iterator;
import java.util.List;
import java.util.Map;

import org.jibx.runtime.IUnmarshallingContext;
import org.jibx.runtime.JiBXException;

/**
 * Command line processor for all types of customizable tools. This just provides the basic handling of a customizations
 * file, target directory, and overrides of values in the customizations root object.
 * TODO: should extend SchemaCommandLinebase
 * 
 * @author Dennis M. Sosnoski
 */
public abstract class CustomizationCommandLineBase
{
    /** Array of method parameter classes for single String parameter. */
    public static final Class[] STRING_PARAMETER_ARRAY = new Class[] { String.class };
    
    /** Array of classes for String and unmarshaller parameters. */
    public static final Class[] STRING_UNMARSHALLER_PARAMETER_ARRAY =
        new Class[] { String.class, IUnmarshallingContext.class };
    
    /** Number of leading characters in usage lines checked for uniqueness. */
    private static final int USAGE_UNIQUE_CHARS = 4;
    
    /** Ordered array of usage lines for parameters at this level. */
    private static final String[] COMMON_USAGE_LINES =
        new String[] {
            " -c path  input customizations file",
            " -t path  target directory for generated output (default is current directory)",
            " -w       wipe all existing files from generation directory (ignored if current\n" +
            "          directory)",
            " -v       verbose output flag" };
    
    /** Complete array of usage lines. */
    private final String[] m_usageLines;
    
    /** List of specified classes or files. */
    private List m_extraArgs;
    
    /** Target directory for output. */
    private File m_generateDirectory;
    
    /** Verbose output flag. */
    private boolean m_verbose;
    
    /**
     * Constructor. This just merges the usage line defined by subclasses with those at this level, and checks for any
     * duplication (by comparing the first n characters of the lines).
     * 
     * @param lines
     */
    protected CustomizationCommandLineBase(String[] lines) {
        m_usageLines = mergeUsageLines(lines, COMMON_USAGE_LINES);
        for (int i = 1; i < m_usageLines.length; i++) {
            String line = m_usageLines[i];
            if (m_usageLines[i-1].startsWith(line.substring(0, USAGE_UNIQUE_CHARS))) {
                throw new IllegalArgumentException("Internal error - duplicate parameter flags");
            }
        }
    }
    
    /**
     * Process command line arguments array.
     * 
     * @param args
     * @return true if valid, false if not
     * @throws JiBXException
     * @throws IOException
     */
    public boolean processArgs(String[] args) throws JiBXException, IOException {
        boolean wipe = false;
        String custom = null;
        String genpath = null;
        Map overrides = new HashMap();
        ArgList alist = new ArgList(args);
        m_extraArgs = new ArrayList();
        while (alist.hasNext()) {
            String arg = alist.next();
            if ("-c".equalsIgnoreCase(arg)) {
                custom = alist.next();
            } else if ("-t".equalsIgnoreCase(arg)) {
                genpath = alist.next();
            } else if ("-v".equalsIgnoreCase(arg)) {
                m_verbose = true;
            } else if ("-w".equalsIgnoreCase(arg)) {
                wipe = true;
            } else if (arg.startsWith("--") && arg.length() > 2 && Character.isLetter(arg.charAt(2))) {
                if (!putKeyValue(arg.substring(2), overrides)) {
                    alist.setValid(false);
                }
            } else if (!checkParameter(alist)) {
                if (arg.startsWith("-")) {
                    System.err.println("Unknown option flag '" + arg + '\'');
                    alist.setValid(false);
                } else {
                    m_extraArgs.add(alist.current());
                    break;
                }
            }
        }
        
        // collect the extra arguments at end
        while (alist.hasNext()) {
            String arg = alist.next();
            if (arg.startsWith("-")) {
                System.err.println("Command line options must precede all other arguments: error on '" + arg + '\'');
                alist.setValid(false);
                break;
            } else {
                m_extraArgs.add(arg);
            }
        }
        
        // check for valid command line arguments
        if (alist.isValid()) {
            
            // set output directory
            if (genpath == null) {
                m_generateDirectory = new File(".");
                wipe = false;
            } else {
                m_generateDirectory = new File(genpath);
            }
            if (m_generateDirectory.exists()) {
                if (!m_generateDirectory.isDirectory()) {
                    System.out.println("Target path '" + genpath + "' must be a directory");
                    alist.setValid(false);
                }
            } else {
                m_generateDirectory.mkdirs();
                wipe = false;
            }
            if (!m_generateDirectory.canWrite()) {
                System.err.println("Target directory " + m_generateDirectory.getPath() + " is not writable");
                alist.setValid(false);
            } else {
                
                // finish the command line processing
                finishParameters(alist);
                
                // report on the configuration
                if (m_verbose) {
                    verboseDetails();
                    System.out.println("Output to directory " + m_generateDirectory);
                }
                
                // clean generate directory if requested
                if (wipe) {
                    CustomUtils.clean(m_generateDirectory);
                }
                
                // load customizations and check for errors
                if (!loadCustomizations(custom)) {
                    alist.setValid(false);
                } else {
                    
                    // apply command line overrides to customizations
                    Map unknowns = applyOverrides(overrides);
                    if (!unknowns.isEmpty()) {
                        for (Iterator iter = unknowns.keySet().iterator(); iter.hasNext();) {
                            String key = (String)iter.next();
                            System.err.println("Unknown override key '" + key + '\'');
                        }
                        alist.setValid(false);
                    }
                    
                }
            }
            
        } else {
            printUsage();
        }
        return alist.isValid();
    }
    
    /**
     * Get generate directory.
     * 
     * @return directory
     */
    public File getGeneratePath() {
        return m_generateDirectory;
    }
    
    /**
     * Get extra arguments from command line. These extra arguments must follow all parameter flags.
     * 
     * @return args
     */
    public List getExtraArgs() {
        return m_extraArgs;
    }
    
    /**
     * Check if verbose output requested.
     *
     * @return verbose
     */
    public boolean isVerbose() {
        return m_verbose;
    }
    
    /**
     * Set a key=value definition in a map. This is a command line processing assist method that prints an error message
     * directly if the expected format is not found.
     * 
     * @param def
     * @param map
     * @return true if successful, false if error
     */
    public static boolean putKeyValue(String def, Map map) {
        int split = def.indexOf('=');
        if (split >= 0) {
            String key = def.substring(0, split);
            if (map.containsKey(key)) {
                System.err.println("Repeated key item: '" + def + '\'');
                return false;
            } else {
                map.put(key, def.substring(split + 1));
                return true;
            }
        } else {
            System.err.println("Missing '=' in expected key=value item: '" + def + '\'');
            return false;
        }
    }
    
    /**
     * Get the usage lines describing command line parameters.
     *
     * @return lines
     */
    protected String[] getUsageLines() {
        return m_usageLines;
    }
    
    /**
     * Merge two arrays of strings, returning an ordered array containing all the strings from both provided arrays.
     * 
     * @param base
     * @param adds
     * @return ordered merged
     */
    protected static String[] mergeUsageLines(String[] base, String[] adds) {
        if (adds.length == 0) {
            return base;
        } else {
            String fulls[] = new String[base.length + adds.length];
            System.arraycopy(base, 0, fulls, 0, base.length);
            System.arraycopy(adds, 0, fulls, base.length, adds.length);
            Arrays.sort(fulls);
            return fulls;
        }
    }
    
    /**
     * Check extension parameter. This method may be overridden by subclasses to process parameters beyond those known
     * to this base class.
     * 
     * @param alist argument list
     * @return true if parameter processed, false if unknown
     */
    protected boolean checkParameter(ArgList alist) {
        return false;
    }
    
    /**
     * Finish processing of command line parameters. This method may be overridden by subclasses to implement any added
     * processing after all the command line parameters have been handled.
     * 
     * @param alist 
     */
    protected void finishParameters(ArgList alist) {}
    
    /**
     * Print any extension details. This method may be overridden by subclasses to print extension parameter values for
     * verbose output.
     */
    protected void verboseDetails() {}
    
    /**
     * Load the customizations file. This method must load the specified customizations file, or create a default
     * customizations instance, of the appropriate type.
     *
     * @param path customization file path, null if none
     * @return true if successful, false if an error
     * @throws JiBXException 
     * @throws IOException 
     */
    protected abstract boolean loadCustomizations(String path) throws JiBXException, IOException;
    
    /**
     * Apply map of override values to customizations read from file or created as default.
     * 
     * @param overmap override key-value map
     * @return map for key/values not recognized
     */
    protected abstract Map applyOverrides(Map overmap);
    
    /**
     * Print usage information.
     */
    public abstract void printUsage();
    
    /**
     * Wrapper class for command line argument list.
     */
    protected static class ArgList
    {
        private int m_offset;
        
        private final String[] m_args;
        
        private boolean m_valid;
        
        /**
         * Constructor.
         * 
         * @param args
         */
        protected ArgList(String[] args) {
            m_offset = -1;
            m_args = args;
            m_valid = true;
        }
        
        /**
         * Check if another argument value is present.
         * 
         * @return true if argument present, false if all processed
         */
        public boolean hasNext() {
            return m_args.length - m_offset > 1;
        }
        
        /**
         * Get current argument value.
         * 
         * @return argument, or null if none
         */
        public String current() {
            return (m_offset >= 0 && m_offset < m_args.length) ? m_args[m_offset] : null;
        }
        
        /**
         * Get next argument value. If this is called with no argument value available it sets the argument list
         * invalid.
         * 
         * @return argument, or null if none
         */
        public String next() {
            if (++m_offset < m_args.length) {
                return m_args[m_offset];
            } else {
                m_valid = false;
                return null;
            }
        }
        
        /**
         * Set valid state.
         * 
         * @param valid
         */
        public void setValid(boolean valid) {
            m_valid = valid;
        }
        
        /**
         * Check if argument list valid.
         * 
         * @return true if valid, false if not
         */
        public boolean isValid() {
            return m_valid;
        }
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy