mirror of
https://github.com/CLIUtils/CLI11.git
synced 2025-04-29 12:13:52 +00:00
1043 lines
31 KiB
C++
1043 lines
31 KiB
C++
// Copyright (c) 2017-2025, University of Cincinnati, developed by Henry Schreiner
|
|
// under NSF AWARD 1414736 and by the respective contributors.
|
|
// All rights reserved.
|
|
//
|
|
// SPDX-License-Identifier: BSD-3-Clause
|
|
|
|
#include "app_helper.hpp"
|
|
|
|
#include <cmath>
|
|
|
|
#include <array>
|
|
#include <chrono>
|
|
#include <cstdint>
|
|
#include <map>
|
|
#include <memory>
|
|
#include <string>
|
|
#include <unordered_map>
|
|
#include <utility>
|
|
#include <vector>
|
|
|
|
#if defined(CLI11_CPP17)
|
|
#if defined(__has_include)
|
|
#if __has_include(<string_view>)
|
|
#include <string_view>
|
|
#define CLI11_HAS_STRING_VIEW
|
|
#endif
|
|
#endif
|
|
#endif
|
|
|
|
TEST_CASE_METHOD(TApp, "SimpleTransform", "[transform]") {
|
|
int value{0};
|
|
auto *opt = app.add_option("-s", value)->transform(CLI::Transformer({{"one", std::string("1")}}));
|
|
args = {"-s", "one"};
|
|
run();
|
|
CHECK(app.count("-s") == 1u);
|
|
CHECK(opt->count() == 1u);
|
|
CHECK(1 == value);
|
|
}
|
|
|
|
TEST_CASE_METHOD(TApp, "SimpleTransformInitList", "[transform]") {
|
|
int value{0};
|
|
auto *opt = app.add_option("-s", value)->transform(CLI::Transformer({{"one", "1"}}));
|
|
args = {"-s", "one"};
|
|
run();
|
|
CHECK(app.count("-s") == 1u);
|
|
CHECK(opt->count() == 1u);
|
|
CHECK(1 == value);
|
|
}
|
|
|
|
TEST_CASE_METHOD(TApp, "SimpleNumericalTransform", "[transform]") {
|
|
int value{0};
|
|
auto *opt = app.add_option("-s", value)->transform(CLI::Transformer(CLI::TransformPairs<int>{{"one", 1}}));
|
|
args = {"-s", "one"};
|
|
run();
|
|
CHECK(app.count("-s") == 1u);
|
|
CHECK(opt->count() == 1u);
|
|
CHECK(1 == value);
|
|
}
|
|
|
|
TEST_CASE_METHOD(TApp, "EnumTransform", "[transform]") {
|
|
enum class test_cli : std::int16_t { val1 = 3, val2 = 4, val3 = 17 };
|
|
test_cli value{test_cli::val2};
|
|
auto *opt = app.add_option("-s", value)
|
|
->transform(CLI::Transformer(CLI::TransformPairs<test_cli>{
|
|
{"val1", test_cli::val1}, {"val2", test_cli::val2}, {"val3", test_cli::val3}}));
|
|
args = {"-s", "val1"};
|
|
run();
|
|
CHECK(app.count("-s") == 1u);
|
|
CHECK(opt->count() == 1u);
|
|
CHECK(test_cli::val1 == value);
|
|
|
|
args = {"-s", "val2"};
|
|
run();
|
|
CHECK(test_cli::val2 == value);
|
|
|
|
args = {"-s", "val3"};
|
|
run();
|
|
CHECK(test_cli::val3 == value);
|
|
|
|
args = {"-s", "val4"};
|
|
CHECK_THROWS_AS(run(), CLI::ConversionError);
|
|
|
|
// transformer doesn't do any checking so this still works
|
|
args = {"-s", "5"};
|
|
run();
|
|
CHECK(static_cast<std::int16_t>(5) == static_cast<std::int16_t>(value));
|
|
}
|
|
|
|
TEST_CASE_METHOD(TApp, "EnumCheckedTransform", "[transform]") {
|
|
enum class test_cli : std::int16_t { val1 = 3, val2 = 4, val3 = 17 };
|
|
test_cli value{test_cli::val1};
|
|
auto *opt = app.add_option("-s", value)
|
|
->transform(CLI::CheckedTransformer(CLI::TransformPairs<test_cli>{
|
|
{"val1", test_cli::val1}, {"val2", test_cli::val2}, {"val3", test_cli::val3}}));
|
|
args = {"-s", "val1"};
|
|
run();
|
|
CHECK(app.count("-s") == 1u);
|
|
CHECK(opt->count() == 1u);
|
|
CHECK(test_cli::val1 == value);
|
|
|
|
args = {"-s", "val2"};
|
|
run();
|
|
CHECK(test_cli::val2 == value);
|
|
|
|
args = {"-s", "val3"};
|
|
run();
|
|
CHECK(test_cli::val3 == value);
|
|
|
|
args = {"-s", "17"};
|
|
run();
|
|
CHECK(test_cli::val3 == value);
|
|
|
|
args = {"-s", "val4"};
|
|
CHECK_THROWS_AS(run(), CLI::ValidationError);
|
|
|
|
args = {"-s", "5"};
|
|
CHECK_THROWS_AS(run(), CLI::ValidationError);
|
|
}
|
|
|
|
// from to-mas-kral Issue #1086
|
|
TEST_CASE_METHOD(TApp, "EnumCheckedTransformUint8", "[transform]") {
|
|
enum class FooType : std::uint8_t { A, B };
|
|
auto type = FooType::B;
|
|
|
|
const std::map<std::string, FooType> foo_map{
|
|
{"a", FooType::A},
|
|
{"b", FooType::B},
|
|
};
|
|
|
|
app.add_option("-f,--foo", type, "FooType")
|
|
->transform(CLI::CheckedTransformer(foo_map, CLI::ignore_case))
|
|
->default_val(FooType::A)
|
|
->force_callback();
|
|
|
|
run();
|
|
CHECK(type == FooType::A);
|
|
}
|
|
|
|
// from jzakrzewski Issue #330
|
|
TEST_CASE_METHOD(TApp, "EnumCheckedDefaultTransform", "[transform]") {
|
|
enum class existing : std::int16_t { abort, overwrite, remove };
|
|
app.add_option("--existing", "What to do if file already exists in the destination")
|
|
->transform(
|
|
CLI::CheckedTransformer(std::unordered_map<std::string, existing>{{"abort", existing::abort},
|
|
{"overwrite", existing ::overwrite},
|
|
{"delete", existing::remove},
|
|
{"remove", existing::remove}}))
|
|
->default_val("abort");
|
|
args = {"--existing", "overwrite"};
|
|
run();
|
|
CHECK(existing::overwrite == app.get_option("--existing")->as<existing>());
|
|
args.clear();
|
|
run();
|
|
CHECK(existing::abort == app.get_option("--existing")->as<existing>());
|
|
}
|
|
|
|
// test from https://github.com/CLIUtils/CLI11/issues/369 [Jakub Zakrzewski](https://github.com/jzakrzewski)
|
|
TEST_CASE_METHOD(TApp, "EnumCheckedDefaultTransformCallback", "[transform]") {
|
|
enum class existing : std::int16_t { abort, overwrite, remove };
|
|
auto cmd = std::make_shared<CLI::App>("deploys the repository somewhere", "deploy");
|
|
cmd->add_option("--existing", "What to do if file already exists in the destination")
|
|
->transform(
|
|
CLI::CheckedTransformer(std::unordered_map<std::string, existing>{{"abort", existing::abort},
|
|
{"overwrite", existing::overwrite},
|
|
{"delete", existing::remove},
|
|
{"remove", existing::remove}}))
|
|
->default_val("abort");
|
|
|
|
cmd->callback([cmd]() { CHECK(cmd->get_option("--existing")->as<existing>() == existing::abort); });
|
|
app.add_subcommand(cmd);
|
|
|
|
args = {"deploy"};
|
|
run();
|
|
}
|
|
|
|
TEST_CASE_METHOD(TApp, "SimpleTransformFn", "[transform]") {
|
|
int value{0};
|
|
auto *opt = app.add_option("-s", value)->transform(CLI::Transformer({{"one", "1"}}, CLI::ignore_case));
|
|
args = {"-s", "ONE"};
|
|
run();
|
|
CHECK(app.count("-s") == 1u);
|
|
CHECK(opt->count() == 1u);
|
|
CHECK(1 == value);
|
|
}
|
|
|
|
#if defined(CLI11_HAS_STRING_VIEW)
|
|
TEST_CASE_METHOD(TApp, "StringViewTransformFn", "[transform]") {
|
|
std::string value;
|
|
std::map<std::string_view, std::string_view> map = {// key length > std::string().capacity() [SSO length]
|
|
{"a-rather-long-argument", "mapped"}};
|
|
app.add_option("-s", value)->transform(CLI::CheckedTransformer(map));
|
|
args = {"-s", "a-rather-long-argument"};
|
|
run();
|
|
CHECK("mapped" == value);
|
|
}
|
|
|
|
#endif
|
|
|
|
TEST_CASE_METHOD(TApp, "SimpleNumericalTransformFn", "[transform]") {
|
|
int value{0};
|
|
auto *opt =
|
|
app.add_option("-s", value)
|
|
->transform(CLI::Transformer(std::vector<std::pair<std::string, int>>{{"one", 1}}, CLI::ignore_case));
|
|
args = {"-s", "ONe"};
|
|
run();
|
|
CHECK(app.count("-s") == 1u);
|
|
CHECK(opt->count() == 1u);
|
|
CHECK(1 == value);
|
|
}
|
|
|
|
TEST_CASE_METHOD(TApp, "SimpleNumericalTransformFnVector", "[transform]") {
|
|
std::vector<std::pair<std::string, int>> conversions{{"one", 1}, {"two", 2}};
|
|
int value{0};
|
|
auto *opt = app.add_option("-s", value)->transform(CLI::Transformer(conversions, CLI::ignore_case));
|
|
args = {"-s", "ONe"};
|
|
run();
|
|
CHECK(app.count("-s") == 1u);
|
|
CHECK(opt->count() == 1u);
|
|
CHECK(1 == value);
|
|
}
|
|
|
|
TEST_CASE_METHOD(TApp, "SimpleNumericalTransformFnArray", "[transform]") {
|
|
std::array<std::pair<std::string, int>, 2> conversions;
|
|
conversions[0] = std::make_pair(std::string("one"), 1);
|
|
conversions[1] = std::make_pair(std::string("two"), 2);
|
|
|
|
int value{0};
|
|
auto *opt = app.add_option("-s", value)->transform(CLI::Transformer(conversions, CLI::ignore_case));
|
|
args = {"-s", "ONe"};
|
|
run();
|
|
CHECK(app.count("-s") == 1u);
|
|
CHECK(opt->count() == 1u);
|
|
CHECK(1 == value);
|
|
}
|
|
|
|
#ifdef CLI11_CPP14
|
|
// zero copy constexpr array operation with transformer example and test
|
|
TEST_CASE_METHOD(TApp, "SimpleNumericalTransformFnconstexprArray", "[transform]") {
|
|
constexpr std::pair<const char *, int> p1{"one", 1};
|
|
constexpr std::pair<const char *, int> p2{"two", 2};
|
|
constexpr std::array<std::pair<const char *, int>, 2> conversions_c{{p1, p2}};
|
|
|
|
int value{0};
|
|
auto *opt = app.add_option("-s", value)->transform(CLI::Transformer(&conversions_c, CLI::ignore_case));
|
|
args = {"-s", "ONe"};
|
|
run();
|
|
CHECK(app.count("-s") == 1u);
|
|
CHECK(opt->count() == 1u);
|
|
CHECK(1 == value);
|
|
|
|
args = {"-s", "twO"};
|
|
run();
|
|
CHECK(app.count("-s") == 1u);
|
|
CHECK(opt->count() == 1u);
|
|
CHECK(2 == value);
|
|
}
|
|
#endif
|
|
|
|
TEST_CASE_METHOD(TApp, "EnumTransformFn", "[transform]") {
|
|
enum class test_cli : std::int16_t { val1 = 3, val2 = 4, val3 = 17 };
|
|
test_cli value{test_cli::val2};
|
|
auto *opt = app.add_option("-s", value)
|
|
->transform(CLI::Transformer(CLI::TransformPairs<test_cli>{{"val1", test_cli::val1},
|
|
{"val2", test_cli::val2},
|
|
{"val3", test_cli::val3}},
|
|
CLI::ignore_case,
|
|
CLI::ignore_underscore));
|
|
args = {"-s", "val_1"};
|
|
run();
|
|
CHECK(app.count("-s") == 1u);
|
|
CHECK(opt->count() == 1u);
|
|
CHECK(test_cli::val1 == value);
|
|
|
|
args = {"-s", "VAL_2"};
|
|
run();
|
|
CHECK(test_cli::val2 == value);
|
|
|
|
args = {"-s", "VAL3"};
|
|
run();
|
|
CHECK(test_cli::val3 == value);
|
|
|
|
args = {"-s", "val_4"};
|
|
CHECK_THROWS_AS(run(), CLI::ConversionError);
|
|
}
|
|
|
|
TEST_CASE_METHOD(TApp, "EnumTransformFnMap", "[transform]") {
|
|
enum class test_cli : std::int16_t { val1 = 3, val2 = 4, val3 = 17 };
|
|
std::map<std::string, test_cli> map{{"val1", test_cli::val1}, {"val2", test_cli::val2}, {"val3", test_cli::val3}};
|
|
test_cli value{test_cli::val3};
|
|
auto *opt = app.add_option("-s", value)->transform(CLI::Transformer(map, CLI::ignore_case, CLI::ignore_underscore));
|
|
args = {"-s", "val_1"};
|
|
run();
|
|
CHECK(app.count("-s") == 1u);
|
|
CHECK(opt->count() == 1u);
|
|
CHECK(test_cli::val1 == value);
|
|
|
|
args = {"-s", "VAL_2"};
|
|
run();
|
|
CHECK(test_cli::val2 == value);
|
|
|
|
args = {"-s", "VAL3"};
|
|
run();
|
|
CHECK(test_cli::val3 == value);
|
|
|
|
args = {"-s", "val_4"};
|
|
CHECK_THROWS_AS(run(), CLI::ConversionError);
|
|
}
|
|
|
|
TEST_CASE_METHOD(TApp, "EnumTransformFnPtrMap", "[transform]") {
|
|
enum class test_cli : std::int16_t { val1 = 3, val2 = 4, val3 = 17, val4 = 37 };
|
|
std::map<std::string, test_cli> map{{"val1", test_cli::val1}, {"val2", test_cli::val2}, {"val3", test_cli::val3}};
|
|
test_cli value{test_cli::val2};
|
|
auto *opt =
|
|
app.add_option("-s", value)->transform(CLI::Transformer(&map, CLI::ignore_case, CLI::ignore_underscore));
|
|
args = {"-s", "val_1"};
|
|
run();
|
|
CHECK(app.count("-s") == 1u);
|
|
CHECK(opt->count() == 1u);
|
|
CHECK(test_cli::val1 == value);
|
|
|
|
args = {"-s", "VAL_2"};
|
|
run();
|
|
CHECK(test_cli::val2 == value);
|
|
|
|
args = {"-s", "VAL3"};
|
|
run();
|
|
CHECK(test_cli::val3 == value);
|
|
|
|
args = {"-s", "val_4"};
|
|
CHECK_THROWS_AS(run(), CLI::ConversionError);
|
|
|
|
map["val4"] = test_cli::val4;
|
|
run();
|
|
CHECK(test_cli::val4 == value);
|
|
}
|
|
|
|
TEST_CASE_METHOD(TApp, "EnumTransformFnSharedPtrMap", "[transform]") {
|
|
enum class test_cli : std::int16_t { val1 = 3, val2 = 4, val3 = 17, val4 = 37 };
|
|
auto map = std::make_shared<std::unordered_map<std::string, test_cli>>();
|
|
auto &mp = *map;
|
|
mp["val1"] = test_cli::val1;
|
|
mp["val2"] = test_cli::val2;
|
|
mp["val3"] = test_cli::val3;
|
|
|
|
test_cli value{test_cli::val2};
|
|
auto *opt = app.add_option("-s", value)->transform(CLI::Transformer(map, CLI::ignore_case, CLI::ignore_underscore));
|
|
args = {"-s", "val_1"};
|
|
run();
|
|
CHECK(app.count("-s") == 1u);
|
|
CHECK(opt->count() == 1u);
|
|
CHECK(test_cli::val1 == value);
|
|
|
|
args = {"-s", "VAL_2"};
|
|
run();
|
|
CHECK(test_cli::val2 == value);
|
|
|
|
args = {"-s", "VAL3"};
|
|
run();
|
|
CHECK(test_cli::val3 == value);
|
|
|
|
args = {"-s", "val_4"};
|
|
CHECK_THROWS_AS(run(), CLI::ConversionError);
|
|
|
|
mp["val4"] = test_cli::val4;
|
|
run();
|
|
CHECK(test_cli::val4 == value);
|
|
}
|
|
|
|
// Test a cascade of transform functions
|
|
TEST_CASE_METHOD(TApp, "TransformCascade", "[transform]") {
|
|
|
|
std::string output;
|
|
auto *opt = app.add_option("-s", output);
|
|
opt->transform(CLI::Transformer({{"abc", "abcd"}, {"bbc", "bbcd"}, {"cbc", "cbcd"}}, CLI::ignore_case));
|
|
opt->transform(
|
|
CLI::Transformer({{"ab", "abc"}, {"bc", "bbc"}, {"cb", "cbc"}}, CLI::ignore_case, CLI::ignore_underscore));
|
|
opt->transform(CLI::Transformer({{"a", "ab"}, {"b", "bb"}, {"c", "cb"}}, CLI::ignore_case));
|
|
opt->check(CLI::IsMember({"abcd", "bbcd", "cbcd"}));
|
|
args = {"-s", "abcd"};
|
|
run();
|
|
CHECK("abcd" == output);
|
|
|
|
args = {"-s", "Bbc"};
|
|
run();
|
|
CHECK("bbcd" == output);
|
|
|
|
args = {"-s", "C_B"};
|
|
run();
|
|
CHECK("cbcd" == output);
|
|
|
|
args = {"-s", "A"};
|
|
run();
|
|
CHECK("abcd" == output);
|
|
}
|
|
|
|
// Test a cascade of transform functions
|
|
TEST_CASE_METHOD(TApp, "TransformCascadeDeactivate", "[transform]") {
|
|
|
|
std::string output;
|
|
auto *opt = app.add_option("-s", output);
|
|
opt->transform(
|
|
CLI::Transformer({{"abc", "abcd"}, {"bbc", "bbcd"}, {"cbc", "cbcd"}}, CLI::ignore_case).name("tform1"));
|
|
opt->transform(
|
|
CLI::Transformer({{"ab", "abc"}, {"bc", "bbc"}, {"cb", "cbc"}}, CLI::ignore_case, CLI::ignore_underscore)
|
|
.name("tform2")
|
|
.active(false));
|
|
opt->transform(CLI::Transformer({{"a", "ab"}, {"b", "bb"}, {"c", "cb"}}, CLI::ignore_case).name("tform3"));
|
|
opt->check(CLI::IsMember({"abcd", "bbcd", "cbcd"}).name("check"));
|
|
args = {"-s", "abcd"};
|
|
run();
|
|
CHECK("abcd" == output);
|
|
|
|
args = {"-s", "Bbc"};
|
|
run();
|
|
CHECK("bbcd" == output);
|
|
|
|
args = {"-s", "C_B"};
|
|
CHECK_THROWS_AS(run(), CLI::ValidationError);
|
|
|
|
auto *validator = opt->get_validator("tform2");
|
|
CHECK(!validator->get_active());
|
|
CHECK("tform2" == validator->get_name());
|
|
validator->active();
|
|
CHECK(validator->get_active());
|
|
args = {"-s", "C_B"};
|
|
run();
|
|
CHECK("cbcd" == output);
|
|
|
|
opt->get_validator("check")->active(false);
|
|
args = {"-s", "gsdgsgs"};
|
|
run();
|
|
CHECK("gsdgsgs" == output);
|
|
|
|
CHECK_THROWS_AS(opt->get_validator("sdfsdf"), CLI::OptionNotFound);
|
|
}
|
|
|
|
TEST_CASE_METHOD(TApp, "IntTransformFn", "[transform]") {
|
|
std::string value;
|
|
app.add_option("-s", value)
|
|
->transform(
|
|
CLI::CheckedTransformer(std::map<int, int>{{15, 5}, {18, 6}, {21, 7}}, [](int in) { return in - 10; }));
|
|
args = {"-s", "25"};
|
|
run();
|
|
CHECK("5" == value);
|
|
|
|
args = {"-s", "6"};
|
|
run();
|
|
CHECK("6" == value);
|
|
|
|
args = {"-s", "45"};
|
|
CHECK_THROWS_AS(run(), CLI::ValidationError);
|
|
|
|
args = {"-s", "val_4"};
|
|
CHECK_THROWS_AS(run(), CLI::ValidationError);
|
|
}
|
|
|
|
TEST_CASE_METHOD(TApp, "IntTransformNonConvertible", "[transform]") {
|
|
std::string value;
|
|
app.add_option("-s", value)->transform(CLI::Transformer(std::map<int, int>{{15, 5}, {18, 6}, {21, 7}}));
|
|
args = {"-s", "15"};
|
|
run();
|
|
CHECK("5" == value);
|
|
|
|
args = {"-s", "18"};
|
|
run();
|
|
CHECK("6" == value);
|
|
|
|
// value can't be converted to int so it is just ignored
|
|
args = {"-s", "abcd"};
|
|
run();
|
|
CHECK("abcd" == value);
|
|
}
|
|
|
|
TEST_CASE_METHOD(TApp, "IntTransformNonMerge", "[transform]") {
|
|
std::string value;
|
|
app.add_option("-s", value)
|
|
->transform(CLI::Transformer(std::map<int, int>{{15, 5}, {18, 6}, {21, 7}}) &
|
|
CLI::Transformer(std::map<int, int>{{25, 5}, {28, 6}, {31, 7}}),
|
|
"merge");
|
|
args = {"-s", "15"};
|
|
run();
|
|
CHECK("5" == value);
|
|
|
|
args = {"-s", "18"};
|
|
run();
|
|
CHECK("6" == value);
|
|
|
|
// value can't be converted to int so it is just ignored
|
|
args = {"-s", "abcd"};
|
|
run();
|
|
CHECK("abcd" == value);
|
|
|
|
args = {"-s", "25"};
|
|
run();
|
|
CHECK("5" == value);
|
|
|
|
args = {"-s", "31"};
|
|
run();
|
|
CHECK("7" == value);
|
|
|
|
auto help = app.help();
|
|
CHECK(help.find("15->5") != std::string::npos);
|
|
CHECK(help.find("25->5") != std::string::npos);
|
|
|
|
auto *validator = app.get_option("-s")->get_validator();
|
|
help = validator->get_description();
|
|
CHECK(help.find("15->5") != std::string::npos);
|
|
CHECK(help.find("25->5") != std::string::npos);
|
|
|
|
auto *validator2 = app.get_option("-s")->get_validator("merge");
|
|
CHECK(validator == validator2);
|
|
}
|
|
|
|
TEST_CASE_METHOD(TApp, "IntTransformMergeWithCustomValidator", "[transform]") {
|
|
std::string value;
|
|
auto *opt = app.add_option("-s", value)
|
|
->transform(CLI::Transformer(std::map<int, int>{{15, 5}, {18, 6}, {21, 7}}) |
|
|
CLI::Validator(
|
|
[](std::string &element) {
|
|
if(element == "frog") {
|
|
element = "hops";
|
|
}
|
|
return std::string{};
|
|
},
|
|
std::string{}),
|
|
"check");
|
|
args = {"-s", "15"};
|
|
run();
|
|
CHECK("5" == value);
|
|
|
|
args = {"-s", "18"};
|
|
run();
|
|
CHECK("6" == value);
|
|
|
|
// value can't be converted to int so it is just ignored
|
|
args = {"-s", "frog"};
|
|
run();
|
|
CHECK("hops" == value);
|
|
|
|
args = {"-s", "25"};
|
|
run();
|
|
CHECK("25" == value);
|
|
|
|
auto help = app.help();
|
|
CHECK(help.find("15->5") != std::string::npos);
|
|
CHECK(help.find("OR") == std::string::npos);
|
|
|
|
auto *validator = opt->get_validator("check");
|
|
CHECK("check" == validator->get_name());
|
|
validator->active(false);
|
|
help = app.help();
|
|
CHECK(help.find("15->5") == std::string::npos);
|
|
}
|
|
|
|
TEST_CASE_METHOD(TApp, "BoundTests", "[transform]") {
|
|
double value = NAN;
|
|
app.add_option("-s", value)->transform(CLI::Bound(3.4, 5.9));
|
|
args = {"-s", "15"};
|
|
run();
|
|
CHECK(5.9 == value);
|
|
|
|
args = {"-s", "3.689"};
|
|
run();
|
|
CHECK(std::stod("3.689") == value);
|
|
|
|
// value can't be converted to int so it is just ignored
|
|
args = {"-s", "abcd"};
|
|
CHECK_THROWS_AS(run(), CLI::ValidationError);
|
|
|
|
args = {"-s", "2.5"};
|
|
run();
|
|
CHECK(3.4 == value);
|
|
|
|
auto help = app.help();
|
|
CHECK(help.find("bounded to") != std::string::npos);
|
|
CHECK(help.find("[3.4 - 5.9]") != std::string::npos);
|
|
}
|
|
|
|
TEST_CASE_METHOD(TApp, "NumberWithUnitCorrectlySplitNumber", "[transform]") {
|
|
std::map<std::string, int> mapping{{"a", 10}, {"b", 100}, {"cc", 1000}};
|
|
|
|
int value = 0;
|
|
app.add_option("-n", value)->transform(CLI::AsNumberWithUnit(mapping));
|
|
|
|
args = {"-n", "42"};
|
|
run();
|
|
CHECK(42 == value);
|
|
|
|
args = {"-n", "42a"};
|
|
run();
|
|
CHECK(420 == value);
|
|
|
|
args = {"-n", " 42 cc "};
|
|
run();
|
|
CHECK(42000 == value);
|
|
args = {"-n", " -42 cc "};
|
|
run();
|
|
CHECK(-42000 == value);
|
|
}
|
|
|
|
TEST_CASE_METHOD(TApp, "NumberWithUnitFloatTest", "[transform]") {
|
|
std::map<std::string, double> mapping{{"a", 10}, {"b", 100}, {"cc", 1000}};
|
|
double value{0.0};
|
|
app.add_option("-n", value)->transform(CLI::AsNumberWithUnit(mapping));
|
|
|
|
args = {"-n", "42"};
|
|
run();
|
|
CHECK(42 == Approx(value));
|
|
|
|
args = {"-n", ".5"};
|
|
run();
|
|
CHECK(.5 == Approx(value));
|
|
|
|
args = {"-n", "42.5 a"};
|
|
run();
|
|
CHECK(425 == Approx(value));
|
|
|
|
args = {"-n", "42.cc"};
|
|
run();
|
|
CHECK(42000 == Approx(value));
|
|
}
|
|
|
|
TEST_CASE_METHOD(TApp, "NumberWithUnitCaseSensitive", "[transform]") {
|
|
std::map<std::string, int> mapping{{"a", 10}, {"A", 100}};
|
|
|
|
int value{0};
|
|
app.add_option("-n", value)->transform(CLI::AsNumberWithUnit(mapping, CLI::AsNumberWithUnit::CASE_SENSITIVE));
|
|
|
|
args = {"-n", "42a"};
|
|
run();
|
|
CHECK(420 == value);
|
|
|
|
args = {"-n", "42A"};
|
|
run();
|
|
CHECK(4200 == value);
|
|
}
|
|
|
|
TEST_CASE_METHOD(TApp, "NumberWithUnitCaseInsensitive", "[transform]") {
|
|
std::map<std::string, int> mapping{{"a", 10}, {"B", 100}};
|
|
|
|
int value{0};
|
|
app.add_option("-n", value)->transform(CLI::AsNumberWithUnit(mapping, CLI::AsNumberWithUnit::CASE_INSENSITIVE));
|
|
|
|
args = {"-n", "42a"};
|
|
run();
|
|
CHECK(420 == value);
|
|
|
|
args = {"-n", "42A"};
|
|
run();
|
|
CHECK(420 == value);
|
|
|
|
args = {"-n", "42b"};
|
|
run();
|
|
CHECK(4200 == value);
|
|
|
|
args = {"-n", "42B"};
|
|
run();
|
|
CHECK(4200 == value);
|
|
}
|
|
|
|
TEST_CASE_METHOD(TApp, "NumberWithUnitMandatoryUnit", "[transform]") {
|
|
std::map<std::string, int> mapping{{"a", 10}, {"A", 100}};
|
|
|
|
int value{0};
|
|
app.add_option("-n", value)
|
|
->transform(CLI::AsNumberWithUnit(
|
|
mapping, CLI::AsNumberWithUnit::UNIT_REQUIRED | CLI::AsNumberWithUnit::CASE_SENSITIVE));
|
|
|
|
args = {"-n", "42a"};
|
|
run();
|
|
CHECK(420 == value);
|
|
|
|
args = {"-n", "42A"};
|
|
run();
|
|
CHECK(4200 == value);
|
|
|
|
args = {"-n", "42"};
|
|
CHECK_THROWS_AS(run(), CLI::ValidationError);
|
|
}
|
|
|
|
TEST_CASE_METHOD(TApp, "NumberWithUnitMandatoryUnit2", "[transform]") {
|
|
std::map<std::string, int> mapping{{"a", 10}, {"B", 100}};
|
|
|
|
int value{0};
|
|
app.add_option("-n", value)
|
|
->transform(CLI::AsNumberWithUnit(
|
|
mapping, CLI::AsNumberWithUnit::UNIT_REQUIRED | CLI::AsNumberWithUnit::CASE_INSENSITIVE));
|
|
|
|
args = {"-n", "42A"};
|
|
run();
|
|
CHECK(420 == value);
|
|
|
|
args = {"-n", "42b"};
|
|
run();
|
|
CHECK(4200 == value);
|
|
|
|
args = {"-n", "42"};
|
|
CHECK_THROWS_AS(run(), CLI::ValidationError);
|
|
}
|
|
|
|
TEST_CASE_METHOD(TApp, "NumberWithUnitBadMapping", "[transform]") {
|
|
CHECK_THROWS_AS(CLI::AsNumberWithUnit(std::map<std::string, int>{{"a", 10}, {"A", 100}},
|
|
CLI::AsNumberWithUnit::CASE_INSENSITIVE),
|
|
CLI::ValidationError);
|
|
CHECK_THROWS_AS(CLI::AsNumberWithUnit(std::map<std::string, int>{{"a", 10}, {"9", 100}}), CLI::ValidationError);
|
|
CHECK_THROWS_AS(CLI::AsNumberWithUnit(std::map<std::string, int>{{"a", 10}, {"AA A", 100}}), CLI::ValidationError);
|
|
CHECK_THROWS_AS(CLI::AsNumberWithUnit(std::map<std::string, int>{{"a", 10}, {"", 100}}), CLI::ValidationError);
|
|
}
|
|
|
|
TEST_CASE_METHOD(TApp, "NumberWithUnitBadInput", "[transform]") {
|
|
std::map<std::string, int> mapping{{"a", 10}, {"b", 100}};
|
|
|
|
int value{0};
|
|
app.add_option("-n", value)->transform(CLI::AsNumberWithUnit(mapping));
|
|
|
|
args = {"-n", "13 a b"};
|
|
CHECK_THROWS_AS(run(), CLI::ValidationError);
|
|
args = {"-n", "13 c"};
|
|
CHECK_THROWS_AS(run(), CLI::ValidationError);
|
|
args = {"-n", "a"};
|
|
// Assume 1.0 unit
|
|
CHECK_NOTHROW(run());
|
|
args = {"-n", "12.0a"};
|
|
CHECK_THROWS_AS(run(), CLI::ValidationError);
|
|
args = {"-n", "a5"};
|
|
CHECK_THROWS_AS(run(), CLI::ValidationError);
|
|
args = {"-n", ""};
|
|
CHECK_THROWS_AS(run(), CLI::ValidationError);
|
|
args = {"-n", "13 a-"};
|
|
CHECK_THROWS_AS(run(), CLI::ValidationError);
|
|
}
|
|
|
|
static const std::map<std::string, std::string> validValues = {
|
|
{"test\\u03C0\\u00e9", from_u8string(u8"test\u03C0\u00E9")},
|
|
{"test\\u03C0\\u00e9", from_u8string(u8"test\u73C0\u0057")},
|
|
{"test\\U0001F600\\u00E9", from_u8string(u8"test\U0001F600\u00E9")},
|
|
{R"("this\nis\na\nfour\tline test")", "this\nis\na\nfour\tline test"},
|
|
{"'B\"(\\x35\\xa7\\x46)\"'", std::string{0x35, static_cast<char>(0xa7), 0x46}},
|
|
{"B\"(\\x35\\xa7\\x46)\"", std::string{0x35, static_cast<char>(0xa7), 0x46}},
|
|
{"test\\ntest", "test\ntest"},
|
|
{"\"test\\ntest", "\"test\ntest"},
|
|
{R"('this\nis\na\nfour\tline test')", R"(this\nis\na\nfour\tline test)"},
|
|
{R"("this\nis\na\nfour\tline test")", "this\nis\na\nfour\tline test"},
|
|
{R"(`this\nis\na\nfour\tline test`)", R"(this\nis\na\nfour\tline test)"}};
|
|
|
|
TEST_CASE_METHOD(TApp, "StringEscapeValid", "[transform]") {
|
|
|
|
auto test_data = GENERATE(from_range(validValues));
|
|
|
|
std::string value{};
|
|
|
|
app.add_option("-n", value)->transform(CLI::EscapedString);
|
|
|
|
args = {"-n", test_data.first};
|
|
|
|
run();
|
|
CHECK(test_data.second == value);
|
|
}
|
|
|
|
static const std::vector<std::string> invalidValues = {"test\\U0001M600\\u00E9",
|
|
"test\\U0001E600\\u00M9",
|
|
"test\\U0001E600\\uD8E9",
|
|
"test\\U0001E600\\uD8",
|
|
"test\\U0001E60",
|
|
"test\\qbad"};
|
|
|
|
TEST_CASE_METHOD(TApp, "StringEscapeInvalid", "[transform]") {
|
|
|
|
auto test_data = GENERATE(from_range(invalidValues));
|
|
|
|
std::string value{};
|
|
|
|
app.add_option("-n", value)->transform(CLI::EscapedString);
|
|
|
|
args = {"-n", test_data};
|
|
|
|
CHECK_THROWS_AS(run(), CLI::ValidationError);
|
|
}
|
|
|
|
TEST_CASE_METHOD(TApp, "NumberWithUnitIntOverflow", "[transform]") {
|
|
std::map<std::string, int> mapping{{"a", 1000000}, {"b", 100}, {"c", 101}};
|
|
|
|
std::int32_t value = 0;
|
|
app.add_option("-n", value)->transform(CLI::AsNumberWithUnit(mapping));
|
|
|
|
args = {"-n", "1000 a"};
|
|
run();
|
|
CHECK(1000000000 == value);
|
|
|
|
args = {"-n", "1000000 a"};
|
|
CHECK_THROWS_AS(run(), CLI::ValidationError);
|
|
|
|
args = {"-n", "-1000000 a"};
|
|
CHECK_THROWS_AS(run(), CLI::ValidationError);
|
|
|
|
args = {"-n", "21474836 b"};
|
|
run();
|
|
CHECK(2147483600 == value);
|
|
|
|
args = {"-n", "21474836 c"};
|
|
CHECK_THROWS_AS(run(), CLI::ValidationError);
|
|
}
|
|
|
|
TEST_CASE_METHOD(TApp, "NumberWithUnitFloatOverflow", "[transform]") {
|
|
std::map<std::string, float> mapping{{"a", 2.f}, {"b", 1.f}, {"c", 0.f}};
|
|
|
|
float value{0.0F};
|
|
app.add_option("-n", value)->transform(CLI::AsNumberWithUnit(mapping));
|
|
|
|
args = {"-n", "3e+38 a"};
|
|
CHECK_THROWS_AS(run(), CLI::ValidationError);
|
|
|
|
args = {"-n", "3e+38 b"};
|
|
run();
|
|
CHECK(3e+38f == Approx(value));
|
|
|
|
args = {"-n", "3e+38 c"};
|
|
run();
|
|
CHECK(0.f == Approx(value));
|
|
}
|
|
|
|
TEST_CASE_METHOD(TApp, "AsSizeValue1000_1024", "[transform]") {
|
|
std::uint64_t value{0};
|
|
app.add_option("-s", value)->transform(CLI::AsSizeValue(true));
|
|
|
|
args = {"-s", "10240"};
|
|
run();
|
|
CHECK(10240u == value);
|
|
|
|
args = {"-s", "1b"};
|
|
run();
|
|
CHECK(1u == value);
|
|
|
|
std::uint64_t k_value{1000u};
|
|
std::uint64_t ki_value{1024u};
|
|
args = {"-s", "1k"};
|
|
run();
|
|
CHECK(k_value == value);
|
|
args = {"-s", "1kb"};
|
|
run();
|
|
CHECK(k_value == value);
|
|
args = {"-s", "1 Kb"};
|
|
run();
|
|
CHECK(k_value == value);
|
|
args = {"-s", "1ki"};
|
|
run();
|
|
CHECK(ki_value == value);
|
|
args = {"-s", "1kib"};
|
|
run();
|
|
CHECK(ki_value == value);
|
|
|
|
k_value = 1000ull * 1000u;
|
|
ki_value = 1024ull * 1024u;
|
|
args = {"-s", "1m"};
|
|
run();
|
|
CHECK(k_value == value);
|
|
args = {"-s", "1mb"};
|
|
run();
|
|
CHECK(k_value == value);
|
|
args = {"-s", "1mi"};
|
|
run();
|
|
CHECK(ki_value == value);
|
|
args = {"-s", "1mib"};
|
|
run();
|
|
CHECK(ki_value == value);
|
|
|
|
k_value = 1000ull * 1000u * 1000u;
|
|
ki_value = 1024ull * 1024u * 1024u;
|
|
args = {"-s", "1g"};
|
|
run();
|
|
CHECK(k_value == value);
|
|
args = {"-s", "1gb"};
|
|
run();
|
|
CHECK(k_value == value);
|
|
args = {"-s", "1gi"};
|
|
run();
|
|
CHECK(ki_value == value);
|
|
args = {"-s", "1gib"};
|
|
run();
|
|
CHECK(ki_value == value);
|
|
|
|
k_value = 1000ull * 1000u * 1000u * 1000u;
|
|
ki_value = 1024ull * 1024u * 1024u * 1024u;
|
|
args = {"-s", "1t"};
|
|
run();
|
|
CHECK(k_value == value);
|
|
args = {"-s", "1tb"};
|
|
run();
|
|
CHECK(k_value == value);
|
|
args = {"-s", "1ti"};
|
|
run();
|
|
CHECK(ki_value == value);
|
|
args = {"-s", "1tib"};
|
|
run();
|
|
CHECK(ki_value == value);
|
|
|
|
k_value = 1000ull * 1000u * 1000u * 1000u * 1000u;
|
|
ki_value = 1024ull * 1024u * 1024u * 1024u * 1024u;
|
|
args = {"-s", "1p"};
|
|
run();
|
|
CHECK(k_value == value);
|
|
args = {"-s", "1pb"};
|
|
run();
|
|
CHECK(k_value == value);
|
|
args = {"-s", "1pi"};
|
|
run();
|
|
CHECK(ki_value == value);
|
|
args = {"-s", "1pib"};
|
|
run();
|
|
CHECK(ki_value == value);
|
|
|
|
k_value = 1000ull * 1000u * 1000u * 1000u * 1000u * 1000u;
|
|
ki_value = 1024ull * 1024u * 1024u * 1024u * 1024u * 1024u;
|
|
args = {"-s", "1e"};
|
|
run();
|
|
CHECK(k_value == value);
|
|
args = {"-s", "1eb"};
|
|
run();
|
|
CHECK(k_value == value);
|
|
args = {"-s", "1ei"};
|
|
run();
|
|
CHECK(ki_value == value);
|
|
args = {"-s", "1eib"};
|
|
run();
|
|
CHECK(ki_value == value);
|
|
}
|
|
|
|
TEST_CASE_METHOD(TApp, "duration_test", "[transform]") {
|
|
std::chrono::seconds duration{1};
|
|
|
|
app.option_defaults()->ignore_case();
|
|
app.add_option_function<std::size_t>(
|
|
"--duration",
|
|
[&](size_t a_value) { duration = std::chrono::seconds{a_value}; },
|
|
"valid units: sec, min, h, day.")
|
|
->capture_default_str()
|
|
->transform(CLI::AsNumberWithUnit(
|
|
std::map<std::string, std::size_t>{{"sec", 1}, {"min", 60}, {"h", 3600}, {"day", 24 * 3600}}));
|
|
CHECK_NOTHROW(app.parse(std::vector<std::string>{"1 day", "--duration"}));
|
|
|
|
CHECK(std::chrono::seconds(86400) == duration);
|
|
}
|
|
|
|
TEST_CASE_METHOD(TApp, "AsSizeValue1024", "[transform]") {
|
|
std::uint64_t value{0};
|
|
app.add_option("-s", value)->transform(CLI::AsSizeValue(false));
|
|
|
|
args = {"-s", "10240"};
|
|
run();
|
|
CHECK(10240u == value);
|
|
|
|
args = {"-s", "1b"};
|
|
run();
|
|
CHECK(1u == value);
|
|
|
|
std::uint64_t ki_value{1024u};
|
|
args = {"-s", "1k"};
|
|
run();
|
|
CHECK(ki_value == value);
|
|
args = {"-s", "1kb"};
|
|
run();
|
|
CHECK(ki_value == value);
|
|
args = {"-s", "1 Kb"};
|
|
run();
|
|
CHECK(ki_value == value);
|
|
args = {"-s", "1ki"};
|
|
run();
|
|
CHECK(ki_value == value);
|
|
args = {"-s", "1kib"};
|
|
run();
|
|
CHECK(ki_value == value);
|
|
|
|
ki_value = 1024ull * 1024u;
|
|
args = {"-s", "1m"};
|
|
run();
|
|
CHECK(ki_value == value);
|
|
args = {"-s", "1mb"};
|
|
run();
|
|
CHECK(ki_value == value);
|
|
args = {"-s", "1mi"};
|
|
run();
|
|
CHECK(ki_value == value);
|
|
args = {"-s", "1mib"};
|
|
run();
|
|
CHECK(ki_value == value);
|
|
|
|
ki_value = 1024ull * 1024u * 1024u;
|
|
args = {"-s", "1g"};
|
|
run();
|
|
CHECK(ki_value == value);
|
|
args = {"-s", "1gb"};
|
|
run();
|
|
CHECK(ki_value == value);
|
|
args = {"-s", "1gi"};
|
|
run();
|
|
CHECK(ki_value == value);
|
|
args = {"-s", "1gib"};
|
|
run();
|
|
CHECK(ki_value == value);
|
|
|
|
ki_value = 1024ull * 1024u * 1024u * 1024u;
|
|
args = {"-s", "1t"};
|
|
run();
|
|
CHECK(ki_value == value);
|
|
args = {"-s", "1tb"};
|
|
run();
|
|
CHECK(ki_value == value);
|
|
args = {"-s", "1ti"};
|
|
run();
|
|
CHECK(ki_value == value);
|
|
args = {"-s", "1tib"};
|
|
run();
|
|
CHECK(ki_value == value);
|
|
|
|
ki_value = 1024ull * 1024u * 1024u * 1024u * 1024u;
|
|
args = {"-s", "1p"};
|
|
run();
|
|
CHECK(ki_value == value);
|
|
args = {"-s", "1pb"};
|
|
run();
|
|
CHECK(ki_value == value);
|
|
args = {"-s", "1pi"};
|
|
run();
|
|
CHECK(ki_value == value);
|
|
args = {"-s", "1pib"};
|
|
run();
|
|
CHECK(ki_value == value);
|
|
|
|
ki_value = 1024ull * 1024u * 1024u * 1024u * 1024u * 1024u;
|
|
args = {"-s", "1e"};
|
|
run();
|
|
CHECK(ki_value == value);
|
|
args = {"-s", "1eb"};
|
|
run();
|
|
CHECK(ki_value == value);
|
|
args = {"-s", "1ei"};
|
|
run();
|
|
CHECK(ki_value == value);
|
|
args = {"-s", "1eib"};
|
|
run();
|
|
CHECK(ki_value == value);
|
|
}
|