// Copyright (c) 2015 Egor Tensin // This file is part of the "AES tools" project. // For details, see https://github.com/egor-tensin/aes-tools. // Distributed under the MIT License. #pragma once #include "block_input.hpp" #include "data_parsers.hpp" #include #include #include #include #include #include #include #include #include namespace { class CommandLineParser; class Settings { public: aes::Algorithm algorithm = AES_AES128; aes::Mode mode = AES_ECB; bool use_boxes = false; bool verbose = false; private: Settings() = default; friend class CommandLineParser; }; class CommandLineParser { public: explicit CommandLineParser(const std::string& argv0) : prog_name{boost::filesystem::path{argv0}.filename().string()} , options{"Options"} { } Settings parse(int argc, char** argv, std::vector& inputs) { Settings settings; namespace po = boost::program_options; options.add_options() ("help,h", "show this message and exit") ("use-boxes,b", po::bool_switch(&settings.use_boxes)->default_value(false), "use the \"boxes\" interface") ("mode,m", po::value(&settings.mode)->required(), "set mode of operation") ("algorithm,a", po::value(&settings.algorithm)->required(), "set algorithm") ("verbose,v", po::bool_switch(&settings.verbose)->default_value(false), "enable verbose output"); std::vector args; po::options_description hidden_options; hidden_options.add_options() ("positional", po::value>(&args)); po::options_description all_options; all_options.add(options).add(hidden_options); po::positional_options_description positional_options; positional_options.add("positional", -1); po::variables_map vm; po::store(po::command_line_parser(argc, argv) .options(all_options) .positional(positional_options) .run(), vm); if (vm.count("help")) { help_flag = true; return settings; } po::notify(vm); inputs = parse_inputs(settings, std::deque{ std::make_move_iterator(args.begin()), std::make_move_iterator(args.end())}); return settings; } bool exit_with_usage() const { return help_flag; } private: static std::vector parse_inputs( const Settings& settings, std::deque&& args) { std::vector inputs; while (!args.empty()) inputs.emplace_back(parse_input(settings, args)); return inputs; } static Input parse_input( const Settings& settings, std::deque& args) { std::string key{std::move(args.front())}; args.pop_front(); std::string iv; if (aes::mode_requires_init_vector(settings.mode)) { if (args.empty()) throw boost::program_options::error{"an initialization vector is required for the selected mode of operation"}; iv = std::move(args.front()); args.pop_front(); } auto blocks = parse_blocks(args); if (aes::mode_requires_init_vector(settings.mode)) return {key, iv, std::move(blocks)}; else return {key, std::move(blocks)}; } static std::vector parse_blocks(std::deque& args) { std::vector blocks; while (!args.empty()) { std::string block{std::move(args.front())}; args.pop_front(); if (block == "--") break; blocks.emplace_back(std::move(block)); } return blocks; } const std::string prog_name; boost::program_options::options_description options; bool help_flag = false; friend std::ostream& operator<<(std::ostream&, const CommandLineParser&); }; std::ostream& operator<<(std::ostream& os, const CommandLineParser& cmd_parser) { return os << "Usage: " << cmd_parser.prog_name << " [OPTIONS...] [-- KEY [IV] [BLOCK...]...]\n" << cmd_parser.options << "\n"; } }