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

com.github.rvesse.airline.help.markdown.MarkdownGlobalUsageGenerator Maven / Gradle / Ivy

/**
 * Copyright (C) 2010-16 the original author or authors.
 *
 * Licensed under the Apache License, Version 2.0 (the "License");
 * you may not use this file except in compliance with the License.
 * You may obtain a copy of the License at
 *
 *     http://www.apache.org/licenses/LICENSE-2.0
 *
 * Unless required by applicable law or agreed to in writing, software
 * distributed under the License is distributed on an "AS IS" BASIS,
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 * See the License for the specific language governing permissions and
 * limitations under the License.
 */
package com.github.rvesse.airline.help.markdown;

import java.io.IOException;
import java.util.ArrayList;
import java.util.Comparator;
import java.util.List;

import org.apache.commons.lang3.StringUtils;

import com.github.rvesse.airline.help.UsageHelper;
import com.github.rvesse.airline.help.common.AbstractPrintedCommandUsageGenerator;
import com.github.rvesse.airline.help.common.AbstractPrintedGlobalUsageGenerator;
import com.github.rvesse.airline.help.sections.HelpHint;
import com.github.rvesse.airline.io.printers.UsagePrinter;
import com.github.rvesse.airline.model.CommandGroupMetadata;
import com.github.rvesse.airline.model.CommandMetadata;
import com.github.rvesse.airline.model.GlobalMetadata;
import com.github.rvesse.airline.model.OptionMetadata;
import com.github.rvesse.airline.parser.aliases.UserAliasesSource;
import com.github.rvesse.airline.utils.AirlineUtils;

import static com.github.rvesse.airline.help.UsageHelper.DEFAULT_OPTION_COMPARATOR;

public class MarkdownGlobalUsageGenerator extends AbstractPrintedGlobalUsageGenerator {

    private final MarkdownUsageHelper helper;
    protected final AbstractPrintedCommandUsageGenerator commandUsageGenerator;

    public MarkdownGlobalUsageGenerator() {
        this(DEFAULT_COLUMNS, UsageHelper.DEFAULT_HINT_COMPARATOR, UsageHelper.DEFAULT_OPTION_COMPARATOR,
                UsageHelper.DEFAULT_COMMAND_COMPARATOR, UsageHelper.DEFAULT_COMMAND_GROUP_COMPARATOR, false,
                new MarkdownCommandUsageGenerator(false));
    }

    public MarkdownGlobalUsageGenerator(boolean includeHidden) {
        this(DEFAULT_COLUMNS, UsageHelper.DEFAULT_HINT_COMPARATOR, UsageHelper.DEFAULT_OPTION_COMPARATOR,
                UsageHelper.DEFAULT_COMMAND_COMPARATOR, UsageHelper.DEFAULT_COMMAND_GROUP_COMPARATOR, includeHidden,
                new MarkdownCommandUsageGenerator(includeHidden));
    }

    public MarkdownGlobalUsageGenerator(int columns) {
        this(columns, UsageHelper.DEFAULT_HINT_COMPARATOR, DEFAULT_OPTION_COMPARATOR,
                UsageHelper.DEFAULT_COMMAND_COMPARATOR, UsageHelper.DEFAULT_COMMAND_GROUP_COMPARATOR, false,
                new MarkdownCommandUsageGenerator(columns));
    }

    public MarkdownGlobalUsageGenerator(int columns, boolean includeHidden) {
        this(columns, UsageHelper.DEFAULT_HINT_COMPARATOR, DEFAULT_OPTION_COMPARATOR,
                UsageHelper.DEFAULT_COMMAND_COMPARATOR, UsageHelper.DEFAULT_COMMAND_GROUP_COMPARATOR, includeHidden,
                new MarkdownCommandUsageGenerator(columns, includeHidden));
    }

    public MarkdownGlobalUsageGenerator(int columnSize, Comparator hintComparator,
            Comparator optionComparator, Comparator commandComparator,
            Comparator commandGroupComparator, boolean includeHidden,
            AbstractPrintedCommandUsageGenerator commandUsageGenerator) {
        super(columnSize, hintComparator, optionComparator, commandComparator, commandGroupComparator, includeHidden);
        helper = createHelper(optionComparator, includeHidden);
        this.commandUsageGenerator = commandUsageGenerator;
    }

    protected MarkdownUsageHelper createHelper(Comparator optionComparator,
            boolean includeHidden) {
        return new MarkdownUsageHelper(optionComparator, includeHidden);
    }

    @Override
    protected void usage(GlobalMetadata global, UsagePrinter out) throws IOException {
        // Name and description
        outputDescription(out, global);

        // Synopsis
        outputSynopsis(out, global);

        // Options
        List options = sortOptions(global.getOptions());
        if (options.size() > 0) {
            helper.outputOptions(out, options);
        }

        // Commands and Command Groups
        out.append("# COMMANDS").newline().newline();
        outputCommandList(out, global);
        outputCommandUsages(out, global);

        // Aliases
        if (global.getParserConfiguration().getUserAliasesSource() != null) {
            outputUserAliases(out, global, global.getParserConfiguration().getUserAliasesSource());
        }
    }

    /**
     * Outputs a documentation section listing the commands
     * 
     * @param out
     *            Usage printer
     * @param global
     *            Global meta-data
     * @throws IOException Thrown if there is a problem generating usage output
     */
    protected void outputCommandList(UsagePrinter out, GlobalMetadata global) throws IOException {
        for (CommandMetadata command : sortCommands(global.getDefaultGroupCommands())) {
            outputCommandDescription(out, null, command);
        }

        outputGroupCommandsList(out, global, global.getCommandGroups());
    }

    protected void outputGroupCommandsList(UsagePrinter out, GlobalMetadata global,
            List groups) throws IOException {
        if (groups.size() == 0)
            return;

        for (CommandGroupMetadata group : sortCommandGroups(groups)) {
            if (group.isHidden() && !this.includeHidden())
                continue;

            for (CommandMetadata command : sortCommands(group.getCommands())) {
                outputCommandDescription(out, group, command);
            }

            if (group.getSubGroups().size() > 0) {
                UsagePrinter subGroupPrinter = out.newIndentedPrinter(4);
                outputGroupCommandsList(subGroupPrinter, global, group.getSubGroups());
            }
        }
    }

    /**
     * Outputs a documentation section with a synopsis of CLI usage
     * 
     * @param out
     *            Usage printer
     * @param global
     *            Global meta-data
     * 
     * @throws IOException Thrown if there is a problem generating usage output
     */
    protected void outputSynopsis(UsagePrinter out, GlobalMetadata global) throws IOException {
        out.append("# SYNOPSIS").newline().newline();

        out.append(String.format("`%s`", global.getName()));
        helper.outputOptionsSynopsis(out, global.getOptions());

        if (global.getCommandGroups().size() > 0) {
            if (global.getDefaultGroupCommands().size() > 0)
                out.append("[");
            out.append("*group*");
            if (global.getDefaultGroupCommands().size() > 0)
                out.append("]");
        }

        out.append("*command* [ *command-args* ]").newline().newline();
    }

    /**
     * Outputs a documentation section with a description of the CLI
     * 
     * @param out
     *            Usage printer
     * @param global
     *            Global meta-data
     * @throws IOException Thrown if there is a problem generating usage output
     */
    protected void outputDescription(UsagePrinter out, GlobalMetadata global) throws IOException {
        out.append("# NAME").newline().newline();

        out.append(String.format("`%s`", global.getName())).append(" -").append(global.getDescription()).newline()
                .newline();
    }

    /**
     * Outputs the description for a command
     * 
     * @param out
     *            Usage printer
     * @param group
     *            Group meta-data
     * @param command
     *            Command meta-data
     * @throws IOException Thrown if there is a problem generating usage output
     */
    protected void outputCommandDescription(UsagePrinter out, CommandGroupMetadata group, CommandMetadata command)
            throws IOException {
        if (!command.isHidden() || this.includeHidden()) {
            // New bullet point
            out.append(" - ");
            out = out.newIndentedPrinter(2);

            // Build name and wrap in backticks
            StringBuilder name = new StringBuilder();
            name.append('`');
            if (group != null) {
                name.append(group.getName()).append(' ');
            }
            name.append(command.getName());
            name.append('`');
            out.append(name.toString()).newline();

            if (command.getDescription() != null) {
                out.newline().append(command.getDescription()).newline();
            }
            out.newline();
        }
    }

    protected void outputUserAliases(UsagePrinter out, GlobalMetadata global, UserAliasesSource userAliases)
            throws IOException {
        if (userAliases == null)
            return;

        out.append("# USER DEFINED ALIASES").newline().newline();

        out.append(String.format(
                "This CLI supports user defined aliases which may be placed in a %s file located in %s the following location(s):",
                userAliases.getFilename(), userAliases.getSearchLocations().size() > 1 ? "one/more of" : "")).newline()
                .newline();

        for (String location : userAliases.getSearchLocations()) {
            out.appendOnOneLine(String.format("1. `%s`", location)).newline();
        }
        out.newline();

        out.newline();
        if (userAliases.getSearchLocations().size() > 1) {
            out.append(
                    "Where the file exists in multiple locations then the files are merged together with the earlier locations taking precedence.")
                    .newline().newline();
        }
        out.append("This file contains aliases defined in Java properties file style e.g.").newline().newline();

        UsagePrinter examplePrinter = out.newIndentedPrinter(4);
        examplePrinter
                .append(String.format("%sfoo=bar --flag",
                        StringUtils.isNotBlank(userAliases.getPrefix()) ? userAliases.getPrefix() : ""))
                .newline().newline();
        examplePrinter.flush();

        out.append(
                "Here an alias foo is defined which causes the bar command to be invoked with the `--flag` option passed to it.");
        if (StringUtils.isNotBlank(userAliases.getPrefix())) {
            out.append("Aliases are distinguished from other properties in the file by the prefix `"
                    + userAliases.getPrefix() + "` as seen in the example.").newline();
        }
        out.newline();
        out.append("Alias definitions are subject to the following conditions:").newline().newline();

        UsagePrinter restrictionsPrinter = out.newIndentedPrinter(2);
        if (global.getParserConfiguration().aliasesOverrideBuiltIns()) {
            restrictionsPrinter.append("- Aliases may override existing commands");
        } else {
            restrictionsPrinter.append("- Aliases cannot override existing commands");
        }
        restrictionsPrinter.newline();
        if (global.getParserConfiguration().aliasesMayChain()) {
            restrictionsPrinter.append(
                    "- Aliases may be defined in terms of other aliases provided circular references are not created");
        } else {
            restrictionsPrinter.append("- Aliases cannot be defined in terms of other aliases");
        }
        restrictionsPrinter.newline();
        restrictionsPrinter.flush();
    }

    /**
     * Outputs the command usages for all groups
     * 
     * @param printer
     *            Usage printer
     * @param global
     *            Global meta-data
     * 
     * @throws IOException Thrown if there is a problem generating usage output
     */
    protected void outputCommandUsages(UsagePrinter printer, GlobalMetadata global) throws IOException {
        // Default group usages
        outputDefaultGroupCommandUsages(printer, global);

        // Other group usages
        for (CommandGroupMetadata group : sortCommandGroups(global.getCommandGroups())) {
            if (group.isHidden() && !this.includeHidden())
                continue;

            List groupPath = new ArrayList();
            groupPath.add(group);
            outputGroupCommandUsages(printer, global, groupPath);
        }
    }

    /**
     * Outputs the command usages for the commands in the default group
     * 
     * @param printer
     *            Usage printer
     * @param global
     *            Global meta-data
     * 
     * @throws IOException Thrown if there is a problem generating usage output
     */
    protected void outputDefaultGroupCommandUsages(UsagePrinter printer, GlobalMetadata global) throws IOException {
        for (CommandMetadata command : sortCommands(global.getDefaultGroupCommands())) {
            if (command.isHidden() && !this.includeHidden())
                continue;

            // Horizontal rule
            printer.append("---").newline().newline();

            printer.flush();
            commandUsageGenerator.usage(global.getName(), (String[]) null, command.getName(), command,
                    global.getParserConfiguration(), printer);
        }
    }

    /**
     * Outputs the command usages for the commands in the given group
     * 
     * @param printer
     *            Usage printer
     * @param global
     *            Global Meta-data
     * @param groups
     *            Groups Meta-data
     * 
     * @throws IOException Thrown if there is a problem generating usage output
     */
    protected void outputGroupCommandUsages(UsagePrinter printer, GlobalMetadata global,
            List groups) throws IOException {
        CommandGroupMetadata group = groups.get(groups.size() - 1);

        // Commands in the group
        for (CommandMetadata command : sortCommands(group.getCommands())) {
            if (command.isHidden() && !this.includeHidden())
                continue;

            // Horizontal rule
            printer.append("---").newline().newline();

            printer.flush();
            commandUsageGenerator.usage(global.getName(), UsageHelper.toGroupNames(groups), command.getName(), command,
                    global.getParserConfiguration(), printer);
        }

        // Sub-groups
        for (CommandGroupMetadata subGroup : sortCommandGroups(group.getSubGroups())) {
            if (subGroup.isHidden() && !this.includeHidden())
                continue;

            List subGroupPath = AirlineUtils.listCopy(groups);
            subGroupPath.add(subGroup);
            outputGroupCommandUsages(printer, global, subGroupPath);
        }
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy