123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334 |
- //===- unittest/Support/OptionParsingTest.cpp - OptTable tests ------------===//
- //
- // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
- // See https://llvm.org/LICENSE.txt for license information.
- // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
- //
- //===----------------------------------------------------------------------===//
- #include "llvm/ADT/STLExtras.h"
- #include "llvm/Option/Arg.h"
- #include "llvm/Option/ArgList.h"
- #include "llvm/Option/Option.h"
- #include "gtest/gtest.h"
- using namespace llvm;
- using namespace llvm::opt;
- enum ID {
- OPT_INVALID = 0, // This is not an option ID.
- #define OPTION(PREFIX, NAME, ID, KIND, GROUP, ALIAS, ALIASARGS, FLAGS, PARAM, \
- HELPTEXT, METAVAR, VALUES) \
- OPT_##ID,
- #include "Opts.inc"
- LastOption
- #undef OPTION
- };
- #define PREFIX(NAME, VALUE) const char *const NAME[] = VALUE;
- #include "Opts.inc"
- #undef PREFIX
- enum OptionFlags {
- OptFlag1 = (1 << 4),
- OptFlag2 = (1 << 5),
- OptFlag3 = (1 << 6)
- };
- static const OptTable::Info InfoTable[] = {
- #define OPTION(PREFIX, NAME, ID, KIND, GROUP, ALIAS, ALIASARGS, FLAGS, PARAM, \
- HELPTEXT, METAVAR, VALUES) \
- {PREFIX, NAME, HELPTEXT, METAVAR, OPT_##ID, Option::KIND##Class, \
- PARAM, FLAGS, OPT_##GROUP, OPT_##ALIAS, ALIASARGS, VALUES},
- #include "Opts.inc"
- #undef OPTION
- };
- namespace {
- class TestOptTable : public OptTable {
- public:
- TestOptTable(bool IgnoreCase = false)
- : OptTable(InfoTable, IgnoreCase) {}
- };
- }
- const char *Args[] = {
- "-A",
- "-Bhi",
- "--C=desu",
- "-C", "bye",
- "-D,adena",
- "-E", "apple", "bloom",
- "-Fblarg",
- "-F", "42",
- "-Gchuu", "2"
- };
- TEST(Option, OptionParsing) {
- TestOptTable T;
- unsigned MAI, MAC;
- InputArgList AL = T.ParseArgs(Args, MAI, MAC);
- // Check they all exist.
- EXPECT_TRUE(AL.hasArg(OPT_A));
- EXPECT_TRUE(AL.hasArg(OPT_B));
- EXPECT_TRUE(AL.hasArg(OPT_C));
- EXPECT_TRUE(AL.hasArg(OPT_D));
- EXPECT_TRUE(AL.hasArg(OPT_E));
- EXPECT_TRUE(AL.hasArg(OPT_F));
- EXPECT_TRUE(AL.hasArg(OPT_G));
- // Check the values.
- EXPECT_EQ("hi", AL.getLastArgValue(OPT_B));
- EXPECT_EQ("bye", AL.getLastArgValue(OPT_C));
- EXPECT_EQ("adena", AL.getLastArgValue(OPT_D));
- std::vector<std::string> Es = AL.getAllArgValues(OPT_E);
- EXPECT_EQ("apple", Es[0]);
- EXPECT_EQ("bloom", Es[1]);
- EXPECT_EQ("42", AL.getLastArgValue(OPT_F));
- std::vector<std::string> Gs = AL.getAllArgValues(OPT_G);
- EXPECT_EQ("chuu", Gs[0]);
- EXPECT_EQ("2", Gs[1]);
- // Check the help text.
- std::string Help;
- raw_string_ostream RSO(Help);
- T.PrintHelp(RSO, "test", "title!");
- EXPECT_NE(std::string::npos, Help.find("-A"));
- // Check usage line.
- T.PrintHelp(RSO, "name [options] file...", "title!");
- EXPECT_NE(std::string::npos, Help.find("USAGE: name [options] file...\n"));
- // Test aliases.
- auto Cs = AL.filtered(OPT_C);
- ASSERT_NE(Cs.begin(), Cs.end());
- EXPECT_EQ("desu", StringRef((*Cs.begin())->getValue()));
- ArgStringList ASL;
- (*Cs.begin())->render(AL, ASL);
- ASSERT_EQ(2u, ASL.size());
- EXPECT_EQ("-C", StringRef(ASL[0]));
- EXPECT_EQ("desu", StringRef(ASL[1]));
- }
- TEST(Option, ParseWithFlagExclusions) {
- TestOptTable T;
- unsigned MAI, MAC;
- // Exclude flag3 to avoid parsing as OPT_SLASH_C.
- InputArgList AL = T.ParseArgs(Args, MAI, MAC,
- /*FlagsToInclude=*/0,
- /*FlagsToExclude=*/OptFlag3);
- EXPECT_TRUE(AL.hasArg(OPT_A));
- EXPECT_TRUE(AL.hasArg(OPT_C));
- EXPECT_FALSE(AL.hasArg(OPT_SLASH_C));
- // Exclude flag1 to avoid parsing as OPT_C.
- AL = T.ParseArgs(Args, MAI, MAC,
- /*FlagsToInclude=*/0,
- /*FlagsToExclude=*/OptFlag1);
- EXPECT_TRUE(AL.hasArg(OPT_B));
- EXPECT_FALSE(AL.hasArg(OPT_C));
- EXPECT_TRUE(AL.hasArg(OPT_SLASH_C));
- const char *NewArgs[] = { "/C", "foo", "--C=bar" };
- AL = T.ParseArgs(NewArgs, MAI, MAC);
- EXPECT_TRUE(AL.hasArg(OPT_SLASH_C));
- EXPECT_TRUE(AL.hasArg(OPT_C));
- EXPECT_EQ("foo", AL.getLastArgValue(OPT_SLASH_C));
- EXPECT_EQ("bar", AL.getLastArgValue(OPT_C));
- }
- TEST(Option, ParseAliasInGroup) {
- TestOptTable T;
- unsigned MAI, MAC;
- const char *MyArgs[] = { "-I" };
- InputArgList AL = T.ParseArgs(MyArgs, MAI, MAC);
- EXPECT_TRUE(AL.hasArg(OPT_H));
- }
- TEST(Option, AliasArgs) {
- TestOptTable T;
- unsigned MAI, MAC;
- const char *MyArgs[] = { "-J", "-Joo" };
- InputArgList AL = T.ParseArgs(MyArgs, MAI, MAC);
- EXPECT_TRUE(AL.hasArg(OPT_B));
- EXPECT_EQ("foo", AL.getAllArgValues(OPT_B)[0]);
- EXPECT_EQ("bar", AL.getAllArgValues(OPT_B)[1]);
- }
- TEST(Option, IgnoreCase) {
- TestOptTable T(true);
- unsigned MAI, MAC;
- const char *MyArgs[] = { "-a", "-joo" };
- InputArgList AL = T.ParseArgs(MyArgs, MAI, MAC);
- EXPECT_TRUE(AL.hasArg(OPT_A));
- EXPECT_TRUE(AL.hasArg(OPT_B));
- }
- TEST(Option, DoNotIgnoreCase) {
- TestOptTable T;
- unsigned MAI, MAC;
- const char *MyArgs[] = { "-a", "-joo" };
- InputArgList AL = T.ParseArgs(MyArgs, MAI, MAC);
- EXPECT_FALSE(AL.hasArg(OPT_A));
- EXPECT_FALSE(AL.hasArg(OPT_B));
- }
- TEST(Option, SlurpEmpty) {
- TestOptTable T;
- unsigned MAI, MAC;
- const char *MyArgs[] = { "-A", "-slurp" };
- InputArgList AL = T.ParseArgs(MyArgs, MAI, MAC);
- EXPECT_TRUE(AL.hasArg(OPT_A));
- EXPECT_TRUE(AL.hasArg(OPT_Slurp));
- EXPECT_EQ(0U, AL.getAllArgValues(OPT_Slurp).size());
- }
- TEST(Option, Slurp) {
- TestOptTable T;
- unsigned MAI, MAC;
- const char *MyArgs[] = { "-A", "-slurp", "-B", "--", "foo" };
- InputArgList AL = T.ParseArgs(MyArgs, MAI, MAC);
- EXPECT_EQ(AL.size(), 2U);
- EXPECT_TRUE(AL.hasArg(OPT_A));
- EXPECT_FALSE(AL.hasArg(OPT_B));
- EXPECT_TRUE(AL.hasArg(OPT_Slurp));
- EXPECT_EQ(3U, AL.getAllArgValues(OPT_Slurp).size());
- EXPECT_EQ("-B", AL.getAllArgValues(OPT_Slurp)[0]);
- EXPECT_EQ("--", AL.getAllArgValues(OPT_Slurp)[1]);
- EXPECT_EQ("foo", AL.getAllArgValues(OPT_Slurp)[2]);
- }
- TEST(Option, SlurpJoinedEmpty) {
- TestOptTable T;
- unsigned MAI, MAC;
- const char *MyArgs[] = { "-A", "-slurpjoined" };
- InputArgList AL = T.ParseArgs(MyArgs, MAI, MAC);
- EXPECT_TRUE(AL.hasArg(OPT_A));
- EXPECT_TRUE(AL.hasArg(OPT_SlurpJoined));
- EXPECT_EQ(AL.getAllArgValues(OPT_SlurpJoined).size(), 0U);
- }
- TEST(Option, SlurpJoinedOneJoined) {
- TestOptTable T;
- unsigned MAI, MAC;
- const char *MyArgs[] = { "-A", "-slurpjoinedfoo" };
- InputArgList AL = T.ParseArgs(MyArgs, MAI, MAC);
- EXPECT_TRUE(AL.hasArg(OPT_A));
- EXPECT_TRUE(AL.hasArg(OPT_SlurpJoined));
- EXPECT_EQ(AL.getAllArgValues(OPT_SlurpJoined).size(), 1U);
- EXPECT_EQ(AL.getAllArgValues(OPT_SlurpJoined)[0], "foo");
- }
- TEST(Option, SlurpJoinedAndSeparate) {
- TestOptTable T;
- unsigned MAI, MAC;
- const char *MyArgs[] = { "-A", "-slurpjoinedfoo", "bar", "baz" };
- InputArgList AL = T.ParseArgs(MyArgs, MAI, MAC);
- EXPECT_TRUE(AL.hasArg(OPT_A));
- EXPECT_TRUE(AL.hasArg(OPT_SlurpJoined));
- EXPECT_EQ(3U, AL.getAllArgValues(OPT_SlurpJoined).size());
- EXPECT_EQ("foo", AL.getAllArgValues(OPT_SlurpJoined)[0]);
- EXPECT_EQ("bar", AL.getAllArgValues(OPT_SlurpJoined)[1]);
- EXPECT_EQ("baz", AL.getAllArgValues(OPT_SlurpJoined)[2]);
- }
- TEST(Option, SlurpJoinedButSeparate) {
- TestOptTable T;
- unsigned MAI, MAC;
- const char *MyArgs[] = { "-A", "-slurpjoined", "foo", "bar", "baz" };
- InputArgList AL = T.ParseArgs(MyArgs, MAI, MAC);
- EXPECT_TRUE(AL.hasArg(OPT_A));
- EXPECT_TRUE(AL.hasArg(OPT_SlurpJoined));
- EXPECT_EQ(3U, AL.getAllArgValues(OPT_SlurpJoined).size());
- EXPECT_EQ("foo", AL.getAllArgValues(OPT_SlurpJoined)[0]);
- EXPECT_EQ("bar", AL.getAllArgValues(OPT_SlurpJoined)[1]);
- EXPECT_EQ("baz", AL.getAllArgValues(OPT_SlurpJoined)[2]);
- }
- TEST(Option, FlagAliasToJoined) {
- TestOptTable T;
- unsigned MAI, MAC;
- // Check that a flag alias provides an empty argument to a joined option.
- const char *MyArgs[] = { "-K" };
- InputArgList AL = T.ParseArgs(MyArgs, MAI, MAC);
- EXPECT_EQ(AL.size(), 1U);
- EXPECT_TRUE(AL.hasArg(OPT_B));
- EXPECT_EQ(1U, AL.getAllArgValues(OPT_B).size());
- EXPECT_EQ("", AL.getAllArgValues(OPT_B)[0]);
- }
- TEST(Option, FindNearest) {
- TestOptTable T;
- std::string Nearest;
- // Options that are too short should not be considered
- // "near" other short options.
- EXPECT_GT(T.findNearest("-A", Nearest), 4U);
- EXPECT_GT(T.findNearest("/C", Nearest), 4U);
- EXPECT_GT(T.findNearest("--C=foo", Nearest), 4U);
- // The nearest candidate should mirror the amount of prefix
- // characters used in the original string.
- EXPECT_EQ(1U, T.findNearest("-blorb", Nearest));
- EXPECT_EQ(Nearest, "-blorp");
- EXPECT_EQ(1U, T.findNearest("--blorm", Nearest));
- EXPECT_EQ(Nearest, "--blorp");
- EXPECT_EQ(1U, T.findNearest("-blarg", Nearest));
- EXPECT_EQ(Nearest, "-blarn");
- EXPECT_EQ(1U, T.findNearest("--blarm", Nearest));
- EXPECT_EQ(Nearest, "--blarn");
- EXPECT_EQ(1U, T.findNearest("-fjormp", Nearest));
- EXPECT_EQ(Nearest, "--fjormp");
- // The nearest candidate respects the prefix and value delimiter
- // of the original string.
- EXPECT_EQ(1U, T.findNearest("/framb:foo", Nearest));
- EXPECT_EQ(Nearest, "/cramb:foo");
- // `--glormp` should have an editing distance > 0 from `--glormp=`.
- EXPECT_GT(T.findNearest("--glorrmp", Nearest), 0U);
- EXPECT_EQ(Nearest, "--glorrmp=");
- EXPECT_EQ(0U, T.findNearest("--glorrmp=foo", Nearest));
- // `--blurmps` should correct to `--blurmp`, not `--blurmp=`, even though
- // both naively have an editing distance of 1.
- EXPECT_EQ(1U, T.findNearest("--blurmps", Nearest));
- EXPECT_EQ(Nearest, "--blurmp");
- // ...but `--blurmps=foo` should correct to `--blurmp=foo`.
- EXPECT_EQ(1U, T.findNearest("--blurmps=foo", Nearest));
- EXPECT_EQ(Nearest, "--blurmp=foo");
- // Flags should be included and excluded as specified.
- EXPECT_EQ(1U, T.findNearest("-doopf", Nearest, /*FlagsToInclude=*/OptFlag2));
- EXPECT_EQ(Nearest, "-doopf2");
- EXPECT_EQ(1U, T.findNearest("-doopf", Nearest,
- /*FlagsToInclude=*/0,
- /*FlagsToExclude=*/OptFlag2));
- EXPECT_EQ(Nearest, "-doopf1");
- }
- TEST(DISABLED_Option, FindNearestFIXME) {
- TestOptTable T;
- std::string Nearest;
- // FIXME: Options with joined values should not have those values considered
- // when calculating distance. The test below would fail if run, but it should
- // succeed.
- EXPECT_EQ(1U, T.findNearest("--erbghFoo", Nearest));
- EXPECT_EQ(Nearest, "--ermghFoo");
- }
|