FileDocCategorySizeDatePackage
Configuration.javaAPI DocExample12495Wed Apr 19 11:17:16 BST 2000com.sun.tools.doclets

Configuration.java

/*
 * @(#)Configuration.java	1.18 00/02/02
 *
 * Copyright 1997-2000 Sun Microsystems, Inc. All Rights Reserved.
 * 
 * This software is the proprietary information of Sun Microsystems, Inc.  
 * Use is subject to license terms.
 * 
 */

package com.sun.tools.doclets;

import com.sun.javadoc.*;
import java.util.*;
import java.io.*;

/**
 * Configure the output based on the options. Doclets should sub-class 
 * Configuration, to configure and add their own options. This class contains
 * all user options which are supported by the 1.1 doclet and the standard
 * doclet.
 *
 * @author Robert Field.
 * @author Atul Dambalkar.
 */
public abstract class Configuration {
    /**
     * The Root of the generated Program Structure from the Doclet API.
     */
    public static RootDoc root;

    /**
     * Destination directory name, in which doclet will generate the entire
     * documentation. Default is current directory.
     */
    public String destdirname = "";

    /**
     * Encoding for this document. Default is default encoding for this 
     * platform.
     */
    public String docencoding = null;

    /**
     * Encoding for this document. Default is default encoding for this 
     * platform.
     */ 
    public String encoding = null;

    /**
     * Generate author specific information for all the classes if @author 
     * tag is used in the doc comment and if -author option is used.
     * <code>showauthor</code> is set to true if -author option is used.
     * Default is don't show author information. 
     */
    public boolean showauthor = false;

    /**
     * Generate version specific information for the all the classes 
     * if @version tag is used in the doc comment and if -version option is 
     * used. <code>showversion</code> is set to true if -version option is 
     * used.Default is don't show version information. 
     */
    public boolean showversion = false;

    /**
     * Don't generate the date in the generated documentation, if -nodate 
     * option is used. <code>nodate</code> is set to true if -nodate 
     * option is used. Default is don't show date. 
     */
    public boolean nodate = false;

    /**
     * Sourcepath from where to read the source files. Default is classpath.
     * 
     */
    public String sourcepath = "";

    /**
     * Don't generate deprecated API information at all, if -nodeprecated 
     * option is used. <code>nodepracted</code> is set to true if 
     * -nodeprecated option is used. Default is generate deprected API 
     * information.
     */
    public boolean nodeprecated = false;

    /**
     * All the packages to be documented.
     */
    public PackageDoc[] packages;

    /**
     * Message Retriever for the doclet, to retrieve message from the resource
     * file for this Configuration, which is common for 1.1 and standard 
     * doclets.
     */
    public static MessageRetriever message = null;

    /**
     * This method should be defined in all those doclets(configurations), 
     * which want to derive themselves from this Configuration. This method
     * can be used to set its own command line options.
     * 
     * @param root Root of the Program Structure generated by Javadoc.
     */
    public abstract void setSpecificDocletOptions(RootDoc root) 
                                           throws DocletAbortException;

    /**
     * This method should be defined in all those doclets 
     * which want to inherit from this Configuration. This method 
     * should return the number of arguments to the command line option. 
     * This method is called by the method {@link #optionLength(String)}.
     * 
     * @param option Command line option under consideration.
     * @return number of arguments to option. Zero return means
     * option not known.  Negative value means error occurred.
     * @see #optionLength(String)  
     */
    public abstract int specificDocletOptionLength(String option);

    /**
     * This method should be defined in all those doclets 
     * which want to inherit from this Configuration. This method
     * can be used to check the validity of its own command line options. 
     * This method is called by {@link #validOptions(String[][], 
     * DocErrorReporter)}.
     * 
     * @param option   Options-array from the Javadoc.
     * @param reporter Error reporter.
     * @return True if all the options are valid else false.
     */
    public abstract boolean specificDocletValidOptions(String option[][],
                                                   DocErrorReporter reporter);

    /**
     * Constructor. Constructs the message retriever with resource file.
     */
    public Configuration() {
        if (message == null) {
            message =
                new MessageRetriever("com.sun.tools.javadoc.resources.doclets");
        }
    }

    /**
     * Set the command line options supported by this configuration.
     *
     * @param root Root of the Program Structure generated by this Javadoc run.
     */
    public void setOptions(RootDoc root) throws DocletAbortException {
        String[][] options = root.options();
        Configuration.root = root;
        packages = root.specifiedPackages();
        Arrays.sort(packages);
        for (int oi = 0; oi < options.length; ++oi) {
            String[] os = options[oi];
            String opt = os[0].toLowerCase();
            if (opt.equals("-d")) {
                destdirname = addTrailingFileSep(os[1]);
            } else  if (opt.equals("-docencoding")) {
                docencoding = os[1];
            } else  if (opt.equals("-encoding")) {
                encoding = os[1];
            } else  if (opt.equals("-author")) {
                showauthor = true;
            } else  if (opt.equals("-version")) {
                showversion = true;
            } else  if (opt.equals("-nodeprecated")) {
                nodeprecated = true;
            } else  if (opt.equals("-xnodate")) {
                nodate = true;
            } else  if (opt.equals("-sourcepath")) {
                sourcepath = os[1];
            } else if (opt.equals("-classpath") && 
                       sourcepath.length() == 0) {
                sourcepath = os[1];
            }
        }
        if (sourcepath.length() == 0) {
            sourcepath = System.getProperty("env.class.path");
        }
        if (docencoding == null) {
            docencoding = encoding;
        }

        HtmlDocWriter.configuration = this;
        setSpecificDocletOptions(root);
    }

    /**
     * Add a traliling file separator, if not found or strip off extra trailing
     * file separators if any.
     *
     * @param path Path under consideration.
     * @return String Properly constructed path string.
     */
    String addTrailingFileSep(String path) {
	String fs = System.getProperty("file.separator");
        String dblfs = fs + fs;
        int indexDblfs;
        while ((indexDblfs = path.indexOf(dblfs)) >= 0) {
            path = path.substring(0, indexDblfs) + 
                        path.substring(indexDblfs + fs.length());
        }
	if (!path.endsWith(fs))
	    path += fs;
	return path;
    }

   /**
    * Returns the "length" of a given option. If an option takes no
    * arguments, its length is one. If it takes one argument, it's
    * length is two, and so on. This method is called by JavaDoc to
    * parse the options it does not recognize. It then calls
    * {@link #validOptions(String[][], DocErrorReporter)} to validate them.
    * <b>Note:</b><br>
    * The options arrive as case-sensitive strings. For options that
    * are not case-sensitive, use toLowerCase() on the option string
    * before comparing it.
    * </blockquote>
    *
    * @return number of arguments + 1 for a option. Zero return means
    * option not known.  Negative value means error occurred.
    */
    public int optionLength(String option) {
        option = option.toLowerCase();
        if (option.equals("-version") ||
            option.equals("-nodeprecated") ||
            option.equals("-author") ||
            option.equals("-xnodate")) {
            return 1;
        } else if (option.equals("-docencoding") ||
                   option.equals("-encoding") ||
                   option.equals("-sourcepath") ||
                   option.equals("-d")) {
            return 2;
        } else {
            return specificDocletOptionLength(option);
        }
    }

    /**
     * After parsing the available options using {@link #optionLength(String)},
     * JavaDoc invokes this method with an array of options-arrays, where
     * the first item in any array is the option, and subsequent items in
     * that array are its arguments. So, if -print is an option that takes
     * no arguments, and -copies is an option that takes 1 argument, then
     * <pre>
     *     -print -copies 3
     * </pre>
     * produces an array of arrays that looks like:
     * <pre>
     *      option[0][0] = -print
     *      option[1][0] = -copies
     *      option[1][1] = 3
     * </pre>
     * (By convention, command line switches start with a "-", but
     * they don't have to.)
     * This method is not required to be written by sub-classes and will
     * default gracefully (to true) if absent.
     * <P>
     * Printing option related error messages (using the provided
     * DocErrorReporter) is the responsibility of this method.
     *
     * @param options  Options used on the command line.
     * @param reporter Error reporter to be used.
     * @return true if all the options are valid.
     */
    public boolean validOptions(String options[][], 
                                DocErrorReporter reporter) {
        boolean docencodingfound = false;
        String encoding = "";
        for (int oi = 0; oi < options.length; oi++) {
            String[] os = options[oi];
            String opt = os[0].toLowerCase();
            if (opt.equals("-d")) {
                destdirname = addTrailingFileSep(os[1]);
                File destDir = new File(destdirname);
                if (!destDir.exists()) {
                    reporter.printError(message.getText(
                        "doclet.destination_directory_not_found_0", 
                                                        destDir.getPath()));
                    return false;
                } else if (!destDir.isDirectory()) {
                    reporter.printError(message.getText(
                        "doclet.destination_directory_not_directory_0", 
                                                        destDir.getPath()));
                    return false;
                } else if (!destDir.canWrite()) {
                    reporter.printError(message.getText(
                        "doclet.destination_directory_not_writable_0", 
                                                        destDir.getPath()));
                    return false;
                }
            } else if (opt.equals("-docencoding")) {
                docencodingfound = true;
                if (!checkOutputFileEncoding(os[1], reporter)) {
                    return false;
                }
            } else if (opt.equals("-encoding")) {
                encoding = os[1];
            }
        }
        if (!docencodingfound && encoding.length() > 0) {
  	    if (!checkOutputFileEncoding(encoding, reporter)) {
                    return false;
            }
        }
        return specificDocletValidOptions(options, reporter);
    }

    /**
     * Check the validity of the given Source or Output File encoding on this 
     * platform. 
     *
     * @param docencoding Output file encoding.
     * @param reporter    Error reporter object.
     */
    protected boolean checkOutputFileEncoding(String docencoding,
                                              DocErrorReporter reporter) {
        OutputStream ost= new ByteArrayOutputStream();
        OutputStreamWriter osw = null;
   	try {
            osw = new OutputStreamWriter(ost, docencoding);
        } catch (UnsupportedEncodingException exc) {
            reporter.printError(message.getText(
                                "doclet.Encoding_not_supported", docencoding));
            return false;                  
        } finally {
            try {
                if (osw != null) {
                    osw.close();
                }
            } catch (IOException exc) {
            }
        } 
        return true;
    }       

}