command_line_test.cpp 30 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934
  1. // Part of the Carbon Language project, under the Apache License v2.0 with LLVM
  2. // Exceptions. See /LICENSE for license information.
  3. // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
  4. #include "common/command_line.h"
  5. #include <gmock/gmock.h>
  6. #include <gtest/gtest.h>
  7. #include "llvm/Support/FormatVariadic.h"
  8. #include "testing/base/test_raw_ostream.h"
  9. namespace Carbon::CommandLine {
  10. namespace {
  11. using ::Carbon::Testing::TestRawOstream;
  12. using ::testing::ElementsAre;
  13. using ::testing::Eq;
  14. using ::testing::HasSubstr;
  15. using ::testing::Not;
  16. using ::testing::StrEq;
  17. constexpr CommandInfo TestCommandInfo = {
  18. .name = "test",
  19. .help = "A test command.",
  20. .help_epilogue = "TODO",
  21. };
  22. enum class TestEnum : int8_t {
  23. Val1,
  24. Val2,
  25. };
  26. enum class TestSubcommand : int8_t {
  27. Sub1,
  28. Sub2,
  29. };
  30. TEST(ArgParserTest, BasicCommand) {
  31. bool flag = false;
  32. int integer_option = -1;
  33. llvm::StringRef string_option = "";
  34. auto parse = [&](llvm::ArrayRef<llvm::StringRef> args) {
  35. return Parse(
  36. args, llvm::errs(), llvm::errs(), TestCommandInfo, [&](auto& b) {
  37. b.AddFlag({.name = "flag"}, [&](auto& arg_b) { arg_b.Set(&flag); });
  38. b.AddIntegerOption({.name = "option1"},
  39. [&](auto& arg_b) { arg_b.Set(&integer_option); });
  40. b.AddStringOption({.name = "option2"},
  41. [&](auto& arg_b) { arg_b.Set(&string_option); });
  42. b.Do([] {});
  43. });
  44. };
  45. EXPECT_THAT(parse({"--flag", "--option2=value", "--option1=42"}),
  46. Eq(ParseResult::Success));
  47. EXPECT_TRUE(flag);
  48. EXPECT_THAT(integer_option, Eq(42));
  49. EXPECT_THAT(string_option, StrEq("value"));
  50. }
  51. TEST(ArgParserTest, BooleanFlags) {
  52. bool flag = false;
  53. auto parse = [&](llvm::ArrayRef<llvm::StringRef> args, llvm::raw_ostream& s) {
  54. return Parse(args, s, s, TestCommandInfo, [&](auto& b) {
  55. b.AddFlag({.name = "flag"}, [&](auto& arg_b) { arg_b.Set(&flag); });
  56. b.Do([] {});
  57. });
  58. };
  59. EXPECT_THAT(parse({"--no-flag"}, llvm::errs()), Eq(ParseResult::Success));
  60. EXPECT_FALSE(flag);
  61. EXPECT_THAT(parse({"--flag", "--no-flag"}, llvm::errs()),
  62. Eq(ParseResult::Success));
  63. EXPECT_FALSE(flag);
  64. EXPECT_THAT(parse({"--no-flag", "--flag"}, llvm::errs()),
  65. Eq(ParseResult::Success));
  66. EXPECT_TRUE(flag);
  67. EXPECT_THAT(parse({"--no-flag", "--flag", "--flag=false"}, llvm::errs()),
  68. Eq(ParseResult::Success));
  69. EXPECT_FALSE(flag);
  70. EXPECT_THAT(parse({"--no-flag", "--flag=true"}, llvm::errs()),
  71. Eq(ParseResult::Success));
  72. EXPECT_TRUE(flag);
  73. EXPECT_THAT(parse({"--no-flag", "--flag=true", "--no-flag"}, llvm::errs()),
  74. Eq(ParseResult::Success));
  75. EXPECT_FALSE(flag);
  76. TestRawOstream os;
  77. EXPECT_THAT(parse({"--no-flag=true"}, os), Eq(ParseResult::Error));
  78. EXPECT_THAT(
  79. os.TakeStr(),
  80. StrEq("ERROR: Cannot specify a value when using a flag name prefixed "
  81. "with 'no-' -- that prefix implies a value of 'false'.\n"));
  82. EXPECT_THAT(parse({"--no-flag=false"}, os), Eq(ParseResult::Error));
  83. EXPECT_THAT(
  84. os.TakeStr(),
  85. StrEq("ERROR: Cannot specify a value when using a flag name prefixed "
  86. "with 'no-' -- that prefix implies a value of 'false'.\n"));
  87. }
  88. TEST(ArgParserTest, ArgDefaults) {
  89. bool flag = false;
  90. int integer_option = -1;
  91. llvm::StringRef string_option = "";
  92. auto parse = [&](llvm::ArrayRef<llvm::StringRef> args, llvm::raw_ostream& s) {
  93. return Parse(args, s, s, TestCommandInfo, [&](auto& b) {
  94. b.AddFlag({.name = "flag"}, [&](auto& arg_b) {
  95. arg_b.Default(true);
  96. arg_b.Set(&flag);
  97. });
  98. b.AddIntegerOption({.name = "option1"}, [&](auto& arg_b) {
  99. arg_b.Default(7);
  100. arg_b.Set(&integer_option);
  101. });
  102. b.AddStringOption({.name = "option2"}, [&](auto& arg_b) {
  103. arg_b.Default("default");
  104. arg_b.Set(&string_option);
  105. });
  106. b.Do([] {});
  107. });
  108. };
  109. EXPECT_THAT(parse({}, llvm::errs()), Eq(ParseResult::Success));
  110. EXPECT_TRUE(flag);
  111. EXPECT_THAT(integer_option, Eq(7));
  112. EXPECT_THAT(string_option, StrEq("default"));
  113. EXPECT_THAT(parse({"--option1", "--option2"}, llvm::errs()),
  114. Eq(ParseResult::Success));
  115. EXPECT_TRUE(flag);
  116. EXPECT_THAT(integer_option, Eq(7));
  117. EXPECT_THAT(string_option, StrEq("default"));
  118. EXPECT_THAT(parse({"--option1=42", "--option2=explicit"}, llvm::errs()),
  119. Eq(ParseResult::Success));
  120. EXPECT_TRUE(flag);
  121. EXPECT_THAT(integer_option, Eq(42));
  122. EXPECT_THAT(string_option, StrEq("explicit"));
  123. EXPECT_THAT(
  124. parse({"--option1=42", "--option2=explicit", "--option1", "--option2"},
  125. llvm::errs()),
  126. Eq(ParseResult::Success));
  127. EXPECT_TRUE(flag);
  128. EXPECT_THAT(integer_option, Eq(7));
  129. EXPECT_THAT(string_option, StrEq("default"));
  130. }
  131. TEST(ArgParserTest, ShortArgs) {
  132. bool flag = false;
  133. bool example = false;
  134. int integer_option = -1;
  135. auto parse = [&](llvm::ArrayRef<llvm::StringRef> args, llvm::raw_ostream& s) {
  136. return Parse(args, s, s, TestCommandInfo, [&](auto& b) {
  137. b.AddFlag({.name = "flag", .short_name = "f"},
  138. [&](auto& arg_b) { arg_b.Set(&flag); });
  139. b.AddFlag({.name = "example", .short_name = "x"},
  140. [&](auto& arg_b) { arg_b.Set(&example); });
  141. b.AddIntegerOption({.name = "option1", .short_name = "o"},
  142. [&](auto& arg_b) {
  143. arg_b.Default(123);
  144. arg_b.Set(&integer_option);
  145. });
  146. b.AddIntegerOption({.name = "option2", .short_name = "z"},
  147. [&](auto& arg_b) { arg_b.Set(&integer_option); });
  148. b.Do([] {});
  149. });
  150. };
  151. EXPECT_THAT(parse({"-f", "-o=42", "-x"}, llvm::errs()),
  152. Eq(ParseResult::Success));
  153. EXPECT_TRUE(flag);
  154. EXPECT_TRUE(example);
  155. EXPECT_THAT(integer_option, Eq(42));
  156. flag = false;
  157. example = false;
  158. EXPECT_THAT(parse({"--option1=13", "-xfo"}, llvm::errs()),
  159. Eq(ParseResult::Success));
  160. EXPECT_TRUE(flag);
  161. EXPECT_TRUE(example);
  162. EXPECT_THAT(integer_option, Eq(123));
  163. TestRawOstream os;
  164. EXPECT_THAT(parse({"-xzf"}, os), Eq(ParseResult::Error));
  165. EXPECT_THAT(os.TakeStr(),
  166. StrEq("ERROR: Option '-z' (short for '--option2') requires a "
  167. "value to be provided and none was.\n"));
  168. EXPECT_THAT(parse({"-xz=123"}, os), Eq(ParseResult::Error));
  169. EXPECT_THAT(
  170. os.TakeStr(),
  171. StrEq("ERROR: Cannot provide a value to the group of multiple short "
  172. "options '-xz=...'; values must be provided to a single option, "
  173. "using either the short or long spelling.\n"));
  174. }
  175. TEST(ArgParserTest, PositionalArgs) {
  176. bool flag = false;
  177. llvm::StringRef string_option = "";
  178. llvm::StringRef source_string;
  179. llvm::StringRef dest_string;
  180. auto parse = [&](llvm::ArrayRef<llvm::StringRef> args, llvm::raw_ostream& s) {
  181. return Parse(args, s, s, TestCommandInfo, [&](auto& b) {
  182. b.AddFlag({.name = "flag"}, [&](auto& arg_b) { arg_b.Set(&flag); });
  183. b.AddStringOption({.name = "option"},
  184. [&](auto& arg_b) { arg_b.Set(&string_option); });
  185. b.AddStringPositionalArg({.name = "source"}, [&](auto& arg_b) {
  186. arg_b.Set(&source_string);
  187. arg_b.Required(true);
  188. });
  189. b.AddStringPositionalArg({.name = "dest"}, [&](auto& arg_b) {
  190. arg_b.Set(&dest_string);
  191. arg_b.Required(true);
  192. });
  193. b.Do([] {});
  194. });
  195. };
  196. TestRawOstream os;
  197. EXPECT_THAT(parse({"--flag", "--option=x"}, os), Eq(ParseResult::Error));
  198. EXPECT_THAT(
  199. os.TakeStr(),
  200. StrEq("ERROR: Not all required positional arguments were provided. First "
  201. "missing and required positional argument: 'source'\n"));
  202. EXPECT_THAT(parse({"src", "--flag", "--option=value", "dst"}, llvm::errs()),
  203. Eq(ParseResult::Success));
  204. EXPECT_TRUE(flag);
  205. EXPECT_THAT(string_option, StrEq("value"));
  206. EXPECT_THAT(source_string, StrEq("src"));
  207. EXPECT_THAT(dest_string, StrEq("dst"));
  208. EXPECT_THAT(parse({"src2", "--", "dst2"}, llvm::errs()),
  209. Eq(ParseResult::Success));
  210. EXPECT_THAT(source_string, StrEq("src2"));
  211. EXPECT_THAT(dest_string, StrEq("dst2"));
  212. EXPECT_THAT(parse({"-", "--", "-"}, llvm::errs()), Eq(ParseResult::Success));
  213. EXPECT_THAT(source_string, StrEq("-"));
  214. EXPECT_THAT(dest_string, StrEq("-"));
  215. }
  216. TEST(ArgParserTest, PositionalAppendArgs) {
  217. bool flag = false;
  218. llvm::StringRef string_option = "";
  219. llvm::SmallVector<llvm::StringRef> source_strings;
  220. llvm::SmallVector<llvm::StringRef> dest_strings;
  221. auto parse = [&](llvm::ArrayRef<llvm::StringRef> args, llvm::raw_ostream& s) {
  222. return Parse(args, s, s, TestCommandInfo, [&](auto& b) {
  223. b.AddFlag({.name = "flag"}, [&](auto& arg_b) { arg_b.Set(&flag); });
  224. b.AddStringOption({.name = "option"},
  225. [&](auto& arg_b) { arg_b.Set(&string_option); });
  226. b.AddStringPositionalArg({.name = "sources"}, [&](auto& arg_b) {
  227. arg_b.Append(&source_strings);
  228. arg_b.Required(true);
  229. });
  230. b.AddStringPositionalArg({.name = "dest"}, [&](auto& arg_b) {
  231. arg_b.Append(&dest_strings);
  232. arg_b.Required(true);
  233. });
  234. b.Do([] {});
  235. });
  236. };
  237. TestRawOstream os;
  238. EXPECT_THAT(parse({"--flag", "--option=x"}, os), Eq(ParseResult::Error));
  239. EXPECT_THAT(
  240. os.TakeStr(),
  241. StrEq("ERROR: Not all required positional arguments were provided. First "
  242. "missing and required positional argument: 'sources'\n"));
  243. EXPECT_THAT(
  244. parse({"src1", "--flag", "src2", "--option=value", "--", "--dst--"},
  245. llvm::errs()),
  246. Eq(ParseResult::Success));
  247. EXPECT_TRUE(flag);
  248. EXPECT_THAT(string_option, StrEq("value"));
  249. EXPECT_THAT(source_strings, ElementsAre(StrEq("src1"), StrEq("src2")));
  250. EXPECT_THAT(dest_strings, ElementsAre(StrEq("--dst--")));
  251. source_strings.clear();
  252. dest_strings.clear();
  253. EXPECT_THAT(
  254. parse({"--", "--src1--", "--src2--", "--", "dst1", "dst2"}, llvm::errs()),
  255. Eq(ParseResult::Success));
  256. EXPECT_THAT(source_strings,
  257. ElementsAre(StrEq("--src1--"), StrEq("--src2--")));
  258. EXPECT_THAT(dest_strings, ElementsAre(StrEq("dst1"), StrEq("dst2")));
  259. source_strings.clear();
  260. dest_strings.clear();
  261. EXPECT_THAT(parse({"--", "--", "dst1", "dst2"}, llvm::errs()),
  262. Eq(ParseResult::Success));
  263. EXPECT_THAT(source_strings, ElementsAre());
  264. EXPECT_THAT(dest_strings, ElementsAre(StrEq("dst1"), StrEq("dst2")));
  265. }
  266. TEST(ArgParserTest, BasicSubcommands) {
  267. bool flag = false;
  268. llvm::StringRef option1 = "";
  269. llvm::StringRef option2 = "";
  270. TestSubcommand subcommand;
  271. bool subsub_command = false;
  272. auto parse = [&](llvm::ArrayRef<llvm::StringRef> args, llvm::raw_ostream& s) {
  273. return Parse(args, s, s, TestCommandInfo, [&](auto& b) {
  274. b.AddSubcommand({.name = "sub1"}, [&](auto& sub_b) {
  275. sub_b.AddFlag({.name = "flag"}, [&](auto& arg_b) { arg_b.Set(&flag); });
  276. sub_b.AddStringOption({.name = "option"},
  277. [&](auto& arg_b) { arg_b.Set(&option1); });
  278. sub_b.Do([&] { subcommand = TestSubcommand::Sub1; });
  279. });
  280. b.AddSubcommand({.name = "sub2"}, [&](auto& sub_b) {
  281. sub_b.AddStringOption({.name = "option"},
  282. [&](auto& arg_b) { arg_b.Set(&option1); });
  283. sub_b.AddSubcommand({.name = "subsub"}, [&](auto& subsub_b) {
  284. subsub_b.AddStringOption({.name = "option"},
  285. [&](auto& arg_b) { arg_b.Set(&option2); });
  286. subsub_b.Do([&] { subsub_command = true; });
  287. });
  288. sub_b.Do([&] { subcommand = TestSubcommand::Sub2; });
  289. });
  290. b.RequiresSubcommand();
  291. });
  292. };
  293. TestRawOstream os;
  294. EXPECT_THAT(parse({}, os), Eq(ParseResult::Error));
  295. EXPECT_THAT(os.TakeStr(), StrEq("ERROR: No subcommand specified. Available "
  296. "subcommands: 'sub1', 'sub2', or 'help'\n"));
  297. EXPECT_THAT(parse({"--flag"}, os), Eq(ParseResult::Error));
  298. EXPECT_THAT(os.TakeStr(), StrEq("ERROR: Unknown option '--flag'\n"));
  299. EXPECT_THAT(parse({"sub3"}, os), Eq(ParseResult::Error));
  300. EXPECT_THAT(os.TakeStr(), StrEq("ERROR: Invalid subcommand 'sub3'. Available "
  301. "subcommands: 'sub1', 'sub2', or 'help'\n"));
  302. EXPECT_THAT(parse({"--flag", "sub1"}, os), Eq(ParseResult::Error));
  303. EXPECT_THAT(os.TakeStr(), StrEq("ERROR: Unknown option '--flag'\n"));
  304. EXPECT_THAT(parse({"sub1", "--flag"}, llvm::errs()),
  305. Eq(ParseResult::Success));
  306. EXPECT_THAT(subcommand, Eq(TestSubcommand::Sub1));
  307. EXPECT_TRUE(flag);
  308. EXPECT_THAT(parse({"sub2", "--option=value"}, llvm::errs()),
  309. Eq(ParseResult::Success));
  310. EXPECT_THAT(subcommand, Eq(TestSubcommand::Sub2));
  311. EXPECT_THAT(option1, StrEq("value"));
  312. EXPECT_THAT(parse({"sub2", "--option=abc", "subsub42", "--option=xyz"}, os),
  313. Eq(ParseResult::Error));
  314. EXPECT_THAT(os.TakeStr(),
  315. StrEq("ERROR: Invalid subcommand 'subsub42'. Available "
  316. "subcommands: 'subsub', or 'help'\n"));
  317. EXPECT_THAT(
  318. parse({"sub2", "--option=abc", "subsub", "--option=xyz"}, llvm::errs()),
  319. Eq(ParseResult::Success));
  320. EXPECT_TRUE(subsub_command);
  321. EXPECT_THAT(option1, StrEq("abc"));
  322. EXPECT_THAT(option2, StrEq("xyz"));
  323. }
  324. TEST(ArgParserTest, Appending) {
  325. llvm::SmallVector<int> integers;
  326. llvm::SmallVector<llvm::StringRef> strings;
  327. auto parse = [&](llvm::ArrayRef<llvm::StringRef> args, llvm::raw_ostream& s) {
  328. return Parse(args, s, s, TestCommandInfo, [&](auto& b) {
  329. b.AddIntegerOption({.name = "int"},
  330. [&](auto& arg_b) { arg_b.Append(&integers); });
  331. b.AddStringOption({.name = "str"},
  332. [&](auto& arg_b) { arg_b.Append(&strings); });
  333. b.Do([] {});
  334. });
  335. };
  336. EXPECT_THAT(parse({}, llvm::errs()), Eq(ParseResult::Success));
  337. EXPECT_THAT(integers, ElementsAre());
  338. EXPECT_THAT(strings, ElementsAre());
  339. EXPECT_THAT(parse({"--int=1", "--int=2", "--int=3"}, llvm::errs()),
  340. Eq(ParseResult::Success));
  341. EXPECT_THAT(integers, ElementsAre(Eq(1), Eq(2), Eq(3)));
  342. EXPECT_THAT(strings, ElementsAre());
  343. EXPECT_THAT(parse({"--str=a", "--str=b", "--str=c"}, llvm::errs()),
  344. Eq(ParseResult::Success));
  345. EXPECT_THAT(integers, ElementsAre(Eq(1), Eq(2), Eq(3)));
  346. EXPECT_THAT(strings, ElementsAre(StrEq("a"), StrEq("b"), StrEq("c")));
  347. EXPECT_THAT(parse({"--str=d", "--int=4", "--str=e"}, llvm::errs()),
  348. Eq(ParseResult::Success));
  349. EXPECT_THAT(integers, ElementsAre(Eq(1), Eq(2), Eq(3), Eq(4)));
  350. EXPECT_THAT(strings, ElementsAre(StrEq("a"), StrEq("b"), StrEq("c"),
  351. StrEq("d"), StrEq("e")));
  352. }
  353. TEST(ArgParserTest, PositionalAppending) {
  354. llvm::SmallVector<llvm::StringRef> option_strings;
  355. llvm::SmallVector<llvm::StringRef> strings1;
  356. llvm::SmallVector<llvm::StringRef> strings2;
  357. llvm::SmallVector<llvm::StringRef> strings3;
  358. auto parse = [&](llvm::ArrayRef<llvm::StringRef> args, llvm::raw_ostream& s) {
  359. return Parse(args, s, s, TestCommandInfo, [&](auto& b) {
  360. b.AddStringOption({.name = "opt"},
  361. [&](auto& arg_b) { arg_b.Append(&option_strings); });
  362. b.AddStringPositionalArg({.name = "s1"},
  363. [&](auto& arg_b) { arg_b.Append(&strings1); });
  364. b.AddStringPositionalArg({.name = "s2"},
  365. [&](auto& arg_b) { arg_b.Append(&strings2); });
  366. b.AddStringPositionalArg({.name = "s3"},
  367. [&](auto& arg_b) { arg_b.Append(&strings3); });
  368. b.Do([] {});
  369. });
  370. };
  371. EXPECT_THAT(parse({"a", "--opt=x", "b", "--opt=y", "--", "c", "--opt=z", "d",
  372. "--", "e", "f"},
  373. llvm::errs()),
  374. Eq(ParseResult::Success));
  375. EXPECT_THAT(option_strings, ElementsAre(StrEq("x"), StrEq("y")));
  376. EXPECT_THAT(strings1, ElementsAre(StrEq("a"), StrEq("b")));
  377. EXPECT_THAT(strings2, ElementsAre(StrEq("c"), StrEq("--opt=z"), StrEq("d")));
  378. EXPECT_THAT(strings3, ElementsAre(StrEq("e"), StrEq("f")));
  379. }
  380. TEST(ArgParserTest, OneOfOption) {
  381. int value = 0;
  382. auto parse = [&](llvm::ArrayRef<llvm::StringRef> args, llvm::raw_ostream& s) {
  383. return Parse(args, s, s, TestCommandInfo, [&](auto& b) {
  384. b.AddOneOfOption({.name = "option"}, [&](auto& arg_b) {
  385. arg_b.SetOneOf(
  386. {
  387. arg_b.OneOfValue("x", 1),
  388. arg_b.OneOfValue("y", 2),
  389. arg_b.OneOfValue("z", 3),
  390. },
  391. &value);
  392. });
  393. b.Do([] {});
  394. });
  395. };
  396. EXPECT_THAT(parse({"--option=x"}, llvm::errs()), Eq(ParseResult::Success));
  397. EXPECT_THAT(value, Eq(1));
  398. EXPECT_THAT(parse({"--option=y"}, llvm::errs()), Eq(ParseResult::Success));
  399. EXPECT_THAT(value, Eq(2));
  400. EXPECT_THAT(parse({"--option=z"}, llvm::errs()), Eq(ParseResult::Success));
  401. EXPECT_THAT(value, Eq(3));
  402. constexpr const char* ErrorStr =
  403. "ERROR: Option '--option={0}' has an invalid value '{0}'; valid values "
  404. "are: 'x', 'y', or 'z'\n";
  405. TestRawOstream os;
  406. EXPECT_THAT(parse({"--option=a"}, os), Eq(ParseResult::Error));
  407. EXPECT_THAT(os.TakeStr(), StrEq(llvm::formatv(ErrorStr, "a")));
  408. EXPECT_THAT(parse({"--option="}, os), Eq(ParseResult::Error));
  409. EXPECT_THAT(os.TakeStr(), StrEq(llvm::formatv(ErrorStr, "")));
  410. EXPECT_THAT(parse({"--option=xx"}, os), Eq(ParseResult::Error));
  411. EXPECT_THAT(os.TakeStr(), StrEq(llvm::formatv(ErrorStr, "xx")));
  412. EXPECT_THAT(parse({"--option=\xFF"}, os), Eq(ParseResult::Error));
  413. EXPECT_THAT(os.TakeStr(), StrEq(llvm::formatv(ErrorStr, "\\FF")));
  414. EXPECT_THAT(parse({llvm::StringRef("--option=\0", 10)}, os),
  415. Eq(ParseResult::Error));
  416. EXPECT_THAT(os.TakeStr(), StrEq(llvm::formatv(ErrorStr, "\\00")));
  417. }
  418. TEST(ArgParserTest, OneOfOptionAppending) {
  419. llvm::SmallVector<int> values;
  420. auto parse = [&](llvm::ArrayRef<llvm::StringRef> args, llvm::raw_ostream& s) {
  421. return Parse(args, s, s, TestCommandInfo, [&](auto& b) {
  422. b.AddOneOfOption({.name = "option"}, [&](auto& arg_b) {
  423. arg_b.AppendOneOf(
  424. {
  425. arg_b.OneOfValue("x", 1),
  426. arg_b.OneOfValue("y", 2),
  427. arg_b.OneOfValue("z", 3),
  428. },
  429. &values);
  430. });
  431. b.Do([] {});
  432. });
  433. };
  434. EXPECT_THAT(parse({}, llvm::errs()), Eq(ParseResult::Success));
  435. EXPECT_THAT(values, ElementsAre());
  436. EXPECT_THAT(parse({"--option=x", "--option=y", "--option=z"}, llvm::errs()),
  437. Eq(ParseResult::Success));
  438. EXPECT_THAT(values, ElementsAre(Eq(1), Eq(2), Eq(3)));
  439. EXPECT_THAT(parse({"--option=y", "--option=y", "--option=x"}, llvm::errs()),
  440. Eq(ParseResult::Success));
  441. EXPECT_THAT(values, ElementsAre(Eq(1), Eq(2), Eq(3), Eq(2), Eq(2), Eq(1)));
  442. }
  443. TEST(ArgParserTest, RequiredArgs) {
  444. int integer_option;
  445. llvm::StringRef string_option;
  446. TestSubcommand subcommand;
  447. int integer_sub_option;
  448. llvm::StringRef string_sub_option;
  449. TestEnum enum_sub_option;
  450. auto parse = [&](llvm::ArrayRef<llvm::StringRef> args, llvm::raw_ostream& s) {
  451. return Parse(args, s, s, TestCommandInfo, [&](auto& b) {
  452. b.AddIntegerOption({.name = "opt1"}, [&](auto& arg_b) {
  453. arg_b.Required(true);
  454. arg_b.Set(&integer_option);
  455. });
  456. b.AddStringOption({.name = "opt2"}, [&](auto& arg_b) {
  457. arg_b.Required(true);
  458. arg_b.Set(&string_option);
  459. });
  460. b.AddSubcommand({.name = "sub1"}, [&](auto& sub_b) {
  461. sub_b.AddIntegerOption({.name = "sub-opt1"}, [&](auto& arg_b) {
  462. arg_b.Required(true);
  463. arg_b.Set(&integer_sub_option);
  464. });
  465. sub_b.AddStringOption({.name = "sub-opt2"}, [&](auto& arg_b) {
  466. arg_b.Required(true);
  467. arg_b.Set(&string_sub_option);
  468. });
  469. sub_b.Do([&] { subcommand = TestSubcommand::Sub1; });
  470. });
  471. b.AddSubcommand({.name = "sub2"}, [&](auto& sub_b) {
  472. sub_b.AddIntegerOption({.name = "sub-opt1"}, [&](auto& arg_b) {
  473. arg_b.Required(true);
  474. arg_b.Set(&integer_sub_option);
  475. });
  476. sub_b.AddStringOption({.name = "sub-opt2"}, [&](auto& arg_b) {
  477. arg_b.Required(true);
  478. arg_b.Set(&string_sub_option);
  479. });
  480. sub_b.AddOneOfOption({.name = "sub-opt3"}, [&](auto& arg_b) {
  481. arg_b.Required(true);
  482. arg_b.SetOneOf(
  483. {
  484. arg_b.OneOfValue("a", TestEnum::Val1),
  485. arg_b.OneOfValue("b", TestEnum::Val2),
  486. },
  487. &enum_sub_option);
  488. });
  489. sub_b.Do([&] { subcommand = TestSubcommand::Sub2; });
  490. });
  491. b.Do([] {});
  492. });
  493. };
  494. constexpr const char* ErrorStr =
  495. "ERROR: Required option '{0}' not provided.\n";
  496. TestRawOstream os;
  497. EXPECT_THAT(parse({}, os), Eq(ParseResult::Error));
  498. auto errors = os.TakeStr();
  499. EXPECT_THAT(errors, HasSubstr(llvm::formatv(ErrorStr, "--opt1")));
  500. EXPECT_THAT(errors, HasSubstr(llvm::formatv(ErrorStr, "--opt2")));
  501. EXPECT_THAT(parse({"--opt2=xyz"}, os), Eq(ParseResult::Error));
  502. EXPECT_THAT(os.TakeStr(), StrEq(llvm::formatv(ErrorStr, "--opt1")));
  503. EXPECT_THAT(parse({"--opt2=xyz", "--opt1=42"}, llvm::errs()),
  504. Eq(ParseResult::Success));
  505. EXPECT_THAT(integer_option, Eq(42));
  506. EXPECT_THAT(string_option, StrEq("xyz"));
  507. EXPECT_THAT(parse({"--opt2=xyz", "--opt1=42", "sub2"}, os),
  508. Eq(ParseResult::Error));
  509. errors = os.TakeStr();
  510. EXPECT_THAT(errors, HasSubstr(llvm::formatv(ErrorStr, "--sub-opt1")));
  511. EXPECT_THAT(errors, HasSubstr(llvm::formatv(ErrorStr, "--sub-opt2")));
  512. EXPECT_THAT(errors, HasSubstr(llvm::formatv(ErrorStr, "--sub-opt3")));
  513. EXPECT_THAT(parse({"--opt2=xyz", "--opt1=42", "sub2", "--sub-opt3=b"}, os),
  514. Eq(ParseResult::Error));
  515. errors = os.TakeStr();
  516. EXPECT_THAT(errors, HasSubstr(llvm::formatv(ErrorStr, "--sub-opt1")));
  517. EXPECT_THAT(errors, HasSubstr(llvm::formatv(ErrorStr, "--sub-opt2")));
  518. EXPECT_THAT(errors, Not(HasSubstr(llvm::formatv(ErrorStr, "--sub-opt3"))));
  519. EXPECT_THAT(parse({"--opt2=xyz", "--opt1=42", "sub2", "--sub-opt3=b",
  520. "--sub-opt1=13"},
  521. os),
  522. Eq(ParseResult::Error));
  523. EXPECT_THAT(os.TakeStr(), StrEq(llvm::formatv(ErrorStr, "--sub-opt2")));
  524. EXPECT_THAT(parse({"--opt2=xyz", "--opt1=42", "sub2", "--sub-opt3=b",
  525. "--sub-opt1=13", "--sub-opt2=abc"},
  526. llvm::errs()),
  527. Eq(ParseResult::Success));
  528. EXPECT_THAT(integer_option, Eq(42));
  529. EXPECT_THAT(string_option, StrEq("xyz"));
  530. EXPECT_THAT(subcommand, Eq(TestSubcommand::Sub2));
  531. EXPECT_THAT(integer_sub_option, Eq(13));
  532. EXPECT_THAT(string_sub_option, StrEq("abc"));
  533. EXPECT_THAT(enum_sub_option, Eq(TestEnum::Val2));
  534. }
  535. TEST(ArgParserTest, HelpAndVersion) {
  536. bool flag = false;
  537. int storage = -1;
  538. llvm::StringRef string = "";
  539. auto parse = [&](llvm::ArrayRef<llvm::StringRef> args, llvm::raw_ostream& s) {
  540. return Parse( // Force line break.
  541. args, s, s,
  542. {
  543. .name = "test",
  544. .version = "Test Command -- version 1.2.3",
  545. .build_info = R"""(
  546. Build timestamp: )""" __TIMESTAMP__ R"""(
  547. Build config: test-config-info
  548. )""",
  549. .help = R"""(
  550. A test command.
  551. Lots more information about the test command.
  552. )""",
  553. .help_epilogue = R"""(
  554. Closing remarks.
  555. )""",
  556. },
  557. [&](auto& b) {
  558. b.AddFlag(
  559. {
  560. .name = "flag",
  561. .short_name = "f",
  562. .help = R"""(
  563. A boolean flag.
  564. )""",
  565. },
  566. [&](auto& arg_b) { arg_b.Set(&flag); });
  567. b.AddFlag(
  568. {
  569. .name = "hidden_flag",
  570. .help = R"""(
  571. A *hidden* boolean flag.
  572. )""",
  573. },
  574. [&](auto& arg_b) {
  575. arg_b.HelpHidden(true);
  576. arg_b.Set(&flag);
  577. });
  578. b.AddSubcommand(
  579. {
  580. .name = "edit",
  581. .help = R"""(
  582. Edit the widget.
  583. This will take the provided widgets and edit them.
  584. )""",
  585. .help_epilogue = R"""(
  586. That's all.
  587. )""",
  588. },
  589. [&](auto& sub_b) {
  590. sub_b.AddIntegerOption(
  591. {
  592. .name = "option",
  593. .short_name = "o",
  594. .help = R"""(
  595. An integer option.
  596. )""",
  597. },
  598. [&](auto& arg_b) { arg_b.Set(&storage); });
  599. sub_b.Do([] {});
  600. });
  601. b.AddSubcommand(
  602. {
  603. .name = "run",
  604. .help = R"""(
  605. Run wombats across the screen.
  606. This will cause several wombats to run across your screen.
  607. )""",
  608. .help_epilogue = R"""(
  609. Or it won't, who knows.
  610. )""",
  611. },
  612. [&](auto& sub_b) {
  613. sub_b.AddStringOption(
  614. {
  615. .name = "option",
  616. .short_name = "o",
  617. .help = R"""(
  618. A string option.
  619. )""",
  620. },
  621. [&](auto& arg_b) { arg_b.Set(&string); });
  622. sub_b.AddOneOfOption(
  623. {
  624. .name = "one-of-option",
  625. .help = R"""(
  626. A one-of option.
  627. )""",
  628. },
  629. [&](auto& arg_b) {
  630. arg_b.SetOneOf(
  631. {
  632. arg_b.OneOfValue("x", 1),
  633. arg_b.OneOfValue("y", 2),
  634. arg_b.OneOfValue("z", 3),
  635. },
  636. &storage);
  637. });
  638. sub_b.Do([] {});
  639. });
  640. b.AddSubcommand(
  641. {
  642. .name = "hidden",
  643. .help = R"""(
  644. A hidden subcommand.
  645. )""",
  646. },
  647. [&](auto& sub_b) {
  648. sub_b.HelpHidden(true);
  649. sub_b.Do([] {});
  650. });
  651. b.RequiresSubcommand();
  652. });
  653. };
  654. TestRawOstream os;
  655. EXPECT_THAT(parse({"--flag", "--help"}, os), Eq(ParseResult::MetaSuccess));
  656. std::string help_flag_output = os.TakeStr();
  657. EXPECT_THAT(help_flag_output, StrEq(llvm::StringRef(R"""(
  658. Test Command -- version 1.2.3
  659. A test command.
  660. Lots more information about the test command.
  661. Build info:
  662. Build timestamp: )""" __TIMESTAMP__ R"""(
  663. Build config: test-config-info
  664. Usage:
  665. test [-f] edit [--option=...]
  666. test [-f] run [OPTIONS]
  667. Subcommands:
  668. edit
  669. Edit the widget.
  670. This will take the provided widgets and edit them.
  671. run
  672. Run wombats across the screen.
  673. This will cause several wombats to run across your screen.
  674. help
  675. Prints help information for the command, including a description, command line usage, and details of each subcommand and option that can be provided.
  676. version
  677. Prints the version of this command.
  678. Command options:
  679. -f, --flag
  680. A boolean flag.
  681. Closing remarks.
  682. )""")
  683. .ltrim('\n')));
  684. EXPECT_THAT(parse({"help"}, os), Eq(ParseResult::MetaSuccess));
  685. EXPECT_THAT(os.TakeStr(), StrEq(help_flag_output));
  686. EXPECT_THAT(parse({"--version"}, os), Eq(ParseResult::MetaSuccess));
  687. std::string version_flag_output = os.TakeStr();
  688. EXPECT_THAT(version_flag_output, StrEq(llvm::StringRef(R"""(
  689. Test Command -- version 1.2.3
  690. Build timestamp: )""" __TIMESTAMP__ R"""(
  691. Build config: test-config-info
  692. )""")
  693. .ltrim('\n')));
  694. EXPECT_THAT(parse({"version"}, os), Eq(ParseResult::MetaSuccess));
  695. EXPECT_THAT(os.TakeStr(), StrEq(version_flag_output));
  696. EXPECT_THAT(parse({"--flag", "edit", "--option=42", "--help"}, os),
  697. Eq(ParseResult::MetaSuccess));
  698. std::string edit_help_output = os.TakeStr();
  699. EXPECT_THAT(edit_help_output, StrEq(llvm::StringRef(R"""(
  700. Edit the widget.
  701. This will take the provided widgets and edit them.
  702. Subcommand 'edit' usage:
  703. test [-f] edit [--option=...]
  704. Subcommand 'edit' options:
  705. -o, --option=...
  706. An integer option.
  707. --help[=(full|short)]
  708. Prints help information for the subcommand, including a description, command line usage, and details of each option that can be provided.
  709. Possible values:
  710. - full (default)
  711. - short
  712. That's all.
  713. )""")
  714. .ltrim('\n')));
  715. EXPECT_THAT(parse({"help", "edit"}, os), Eq(ParseResult::MetaSuccess));
  716. EXPECT_THAT(os.TakeStr(), StrEq(edit_help_output));
  717. EXPECT_THAT(parse({"--flag", "run", "--option=abc", "--help"}, os),
  718. Eq(ParseResult::MetaSuccess));
  719. std::string run_help_output = os.TakeStr();
  720. EXPECT_THAT(run_help_output, StrEq(llvm::StringRef(R"""(
  721. Run wombats across the screen.
  722. This will cause several wombats to run across your screen.
  723. Subcommand 'run' usage:
  724. test [-f] run [OPTIONS]
  725. Subcommand 'run' options:
  726. -o, --option=...
  727. A string option.
  728. --one-of-option=(x|y|z)
  729. A one-of option.
  730. Possible values:
  731. - x
  732. - y
  733. - z
  734. --help[=(full|short)]
  735. Prints help information for the subcommand, including a description, command line usage, and details of each option that can be provided.
  736. Possible values:
  737. - full (default)
  738. - short
  739. Or it won't, who knows.
  740. )""")
  741. .ltrim('\n')));
  742. EXPECT_THAT(parse({"help", "run"}, os), Eq(ParseResult::MetaSuccess));
  743. EXPECT_THAT(os.TakeStr(), StrEq(run_help_output));
  744. }
  745. TEST(ArgParserTest, HelpMarkdownLike) {
  746. bool flag = false;
  747. auto parse = [&](llvm::ArrayRef<llvm::StringRef> args, llvm::raw_ostream& s) {
  748. return Parse( // Force line break.
  749. args, s, s, {.name = "test"}, [&](auto& b) {
  750. b.AddFlag(
  751. {
  752. .name = "flag",
  753. .help = R"""(
  754. A boolean flag.
  755. Preformatted
  756. code....
  757. ........
  758. But
  759. here
  760. lines
  761. are
  762. collapsed.
  763. ```
  764. x
  765. y
  766. z
  767. ```
  768. )""",
  769. },
  770. [&](auto& arg_b) { arg_b.Set(&flag); });
  771. b.Do([] {});
  772. });
  773. };
  774. TestRawOstream os;
  775. EXPECT_THAT(parse({"--help"}, os), Eq(ParseResult::MetaSuccess));
  776. EXPECT_THAT(os.TakeStr(), StrEq(llvm::StringRef(R"""(
  777. Usage:
  778. test [--flag]
  779. Options:
  780. --flag
  781. A boolean flag.
  782. Preformatted
  783. code....
  784. ........
  785. But here lines are collapsed.
  786. ```
  787. x
  788. y
  789. z
  790. ```
  791. --help[=(full|short)]
  792. Prints help information for the command, including a description, command line usage, and details of each option that can be provided.
  793. Possible values:
  794. - full (default)
  795. - short
  796. )""")
  797. .ltrim('\n')));
  798. }
  799. } // namespace
  800. } // namespace Carbon::CommandLine