flaggy 0.2.1

A library for parsing command-line flags.
Documentation
// Copyright 2015 Axel Rasmussen
//
// 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.

use crate::command::{parse_command, Command, CommandResult};
use crate::error::*;
use crate::help;
use crate::value::build_values;
use std::io::Write;

fn parse_and_execute_impl<E, W: Write>(
    program: &str,
    args: &[String],
    mut commands: Vec<Command<E>>,
    mut output_writer: Option<W>,
    print_program_help: bool,
    print_command_name: bool,
) -> Result<Option<CommandResult<E>>> {
    let mut args_iterator = args.iter().peekable();

    let command_idx = match parse_command(&mut args_iterator, &commands) {
        Ok(c) => c,
        Err(e) => {
            if print_program_help {
                help::print_program_help(output_writer.as_mut(), program, &commands, e)?;
            }
            return Ok(None);
        }
    };
    let mut command = commands.remove(command_idx);

    let values = match build_values(&command.flags, args_iterator) {
        Ok(vs) => vs,
        Err(e) => {
            help::print_command_help(
                output_writer.as_mut(),
                program,
                &command,
                print_command_name,
                e,
            )?;
            return Ok(None);
        }
    };

    Ok(Some(command.execute(values)))
}

/// This function parses the given program parameters, and calls the appropriate
/// command callback. There are essentially three return values:
///
/// - An error, if the process of trying to execute the command resulted in an
///   unexpected error.
///
/// - Ok(None), if the command was not executed, but no error should be surfaced
///   to the caller (e.g., if the process' input flags were invalid in some way,
///   but the error was handled internally by printing program help).
///
/// - The result returned by the actual command, which was executed.
///
/// This is the function which should be used for typical multi-command
/// programs.
pub(crate) fn parse_and_execute<E, W: Write>(
    program: &str,
    args: &[String],
    commands: Vec<Command<E>>,
    output_writer: Option<W>,
) -> Result<Option<CommandResult<E>>> {
    parse_and_execute_impl(program, args, commands, output_writer, true, true)
}

/// This function parses the given program parameters and calls the given
/// command's callback. It prints out usage information if the parameters are
/// invalid, and returns a reasonable exit code for the process.
///
/// This is the function which should be used for typical single-command
/// programs.
pub(crate) fn parse_and_execute_single_command<E, W: Write>(
    program: &str,
    args: &[String],
    command: Command<E>,
    output_writer: Option<W>,
) -> Result<Option<CommandResult<E>>> {
    let args: Vec<String> = Some(command.name.clone())
        .into_iter()
        .chain(args.iter().cloned())
        .collect();
    parse_and_execute_impl(
        program,
        args.as_slice(),
        vec![command],
        output_writer,
        false,
        false,
    )
}