1*6b3a42afSjmmv // Copyright 2010 Google Inc.
2*6b3a42afSjmmv // All rights reserved.
3*6b3a42afSjmmv //
4*6b3a42afSjmmv // Redistribution and use in source and binary forms, with or without
5*6b3a42afSjmmv // modification, are permitted provided that the following conditions are
6*6b3a42afSjmmv // met:
7*6b3a42afSjmmv //
8*6b3a42afSjmmv // * Redistributions of source code must retain the above copyright
9*6b3a42afSjmmv // notice, this list of conditions and the following disclaimer.
10*6b3a42afSjmmv // * Redistributions in binary form must reproduce the above copyright
11*6b3a42afSjmmv // notice, this list of conditions and the following disclaimer in the
12*6b3a42afSjmmv // documentation and/or other materials provided with the distribution.
13*6b3a42afSjmmv // * Neither the name of Google Inc. nor the names of its contributors
14*6b3a42afSjmmv // may be used to endorse or promote products derived from this software
15*6b3a42afSjmmv // without specific prior written permission.
16*6b3a42afSjmmv //
17*6b3a42afSjmmv // THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
18*6b3a42afSjmmv // "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
19*6b3a42afSjmmv // LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
20*6b3a42afSjmmv // A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
21*6b3a42afSjmmv // OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
22*6b3a42afSjmmv // SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
23*6b3a42afSjmmv // LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
24*6b3a42afSjmmv // DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
25*6b3a42afSjmmv // THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
26*6b3a42afSjmmv // (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
27*6b3a42afSjmmv // OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
28*6b3a42afSjmmv
29*6b3a42afSjmmv #include "cli/cmd_help.hpp"
30*6b3a42afSjmmv
31*6b3a42afSjmmv #include <algorithm>
32*6b3a42afSjmmv #include <cstdlib>
33*6b3a42afSjmmv #include <iterator>
34*6b3a42afSjmmv
35*6b3a42afSjmmv #include <atf-c++.hpp>
36*6b3a42afSjmmv
37*6b3a42afSjmmv #include "cli/common.ipp"
38*6b3a42afSjmmv #include "engine/config.hpp"
39*6b3a42afSjmmv #include "utils/cmdline/commands_map.ipp"
40*6b3a42afSjmmv #include "utils/cmdline/exceptions.hpp"
41*6b3a42afSjmmv #include "utils/cmdline/globals.hpp"
42*6b3a42afSjmmv #include "utils/cmdline/options.hpp"
43*6b3a42afSjmmv #include "utils/cmdline/parser.hpp"
44*6b3a42afSjmmv #include "utils/cmdline/ui_mock.hpp"
45*6b3a42afSjmmv #include "utils/defs.hpp"
46*6b3a42afSjmmv #include "utils/sanity.hpp"
47*6b3a42afSjmmv
48*6b3a42afSjmmv namespace cmdline = utils::cmdline;
49*6b3a42afSjmmv namespace config = utils::config;
50*6b3a42afSjmmv
51*6b3a42afSjmmv using cli::cmd_help;
52*6b3a42afSjmmv
53*6b3a42afSjmmv
54*6b3a42afSjmmv namespace {
55*6b3a42afSjmmv
56*6b3a42afSjmmv
57*6b3a42afSjmmv /// Mock command with a simple definition (no options, no arguments).
58*6b3a42afSjmmv ///
59*6b3a42afSjmmv /// Attempting to run this command will result in a crash. It is only provided
60*6b3a42afSjmmv /// to validate the generation of interactive help.
61*6b3a42afSjmmv class cmd_mock_simple : public cli::cli_command {
62*6b3a42afSjmmv public:
63*6b3a42afSjmmv /// Constructs a new mock command.
64*6b3a42afSjmmv ///
65*6b3a42afSjmmv /// \param name_ The name of the command to create.
cmd_mock_simple(const char * name_)66*6b3a42afSjmmv cmd_mock_simple(const char* name_) : cli::cli_command(
67*6b3a42afSjmmv name_, "", 0, 0, "Simple command")
68*6b3a42afSjmmv {
69*6b3a42afSjmmv }
70*6b3a42afSjmmv
71*6b3a42afSjmmv /// Runs the mock command.
72*6b3a42afSjmmv ///
73*6b3a42afSjmmv /// \param unused_ui Object to interact with the I/O of the program.
74*6b3a42afSjmmv /// \param unused_cmdline Representation of the command line to the
75*6b3a42afSjmmv /// subcommand.
76*6b3a42afSjmmv /// \param unused_user_config The runtime configuration of the program.
77*6b3a42afSjmmv ///
78*6b3a42afSjmmv /// \return Nothing because this function is never called.
79*6b3a42afSjmmv int
run(cmdline::ui * UTILS_UNUSED_PARAM (ui),const cmdline::parsed_cmdline & UTILS_UNUSED_PARAM (cmdline),const config::tree & UTILS_UNUSED_PARAM (user_config))80*6b3a42afSjmmv run(cmdline::ui* UTILS_UNUSED_PARAM(ui),
81*6b3a42afSjmmv const cmdline::parsed_cmdline& UTILS_UNUSED_PARAM(cmdline),
82*6b3a42afSjmmv const config::tree& UTILS_UNUSED_PARAM(user_config))
83*6b3a42afSjmmv {
84*6b3a42afSjmmv UNREACHABLE;
85*6b3a42afSjmmv }
86*6b3a42afSjmmv };
87*6b3a42afSjmmv
88*6b3a42afSjmmv
89*6b3a42afSjmmv /// Mock command with a complex definition (some options, some arguments).
90*6b3a42afSjmmv ///
91*6b3a42afSjmmv /// Attempting to run this command will result in a crash. It is only provided
92*6b3a42afSjmmv /// to validate the generation of interactive help.
93*6b3a42afSjmmv class cmd_mock_complex : public cli::cli_command {
94*6b3a42afSjmmv public:
95*6b3a42afSjmmv /// Constructs a new mock command.
96*6b3a42afSjmmv ///
97*6b3a42afSjmmv /// \param name_ The name of the command to create.
cmd_mock_complex(const char * name_)98*6b3a42afSjmmv cmd_mock_complex(const char* name_) : cli::cli_command(
99*6b3a42afSjmmv name_, "[arg1 .. argN]", 0, 2, "Complex command")
100*6b3a42afSjmmv {
101*6b3a42afSjmmv add_option(cmdline::bool_option("flag_a", "Flag A"));
102*6b3a42afSjmmv add_option(cmdline::bool_option('b', "flag_b", "Flag B"));
103*6b3a42afSjmmv add_option(cmdline::string_option('c', "flag_c", "Flag C", "c_arg"));
104*6b3a42afSjmmv add_option(cmdline::string_option("flag_d", "Flag D", "d_arg", "foo"));
105*6b3a42afSjmmv }
106*6b3a42afSjmmv
107*6b3a42afSjmmv /// Runs the mock command.
108*6b3a42afSjmmv ///
109*6b3a42afSjmmv /// \param unused_ui Object to interact with the I/O of the program.
110*6b3a42afSjmmv /// \param unused_cmdline Representation of the command line to the
111*6b3a42afSjmmv /// subcommand.
112*6b3a42afSjmmv /// \param unused_user_config The runtime configuration of the program.
113*6b3a42afSjmmv ///
114*6b3a42afSjmmv /// \return Nothing because this function is never called.
115*6b3a42afSjmmv int
run(cmdline::ui * UTILS_UNUSED_PARAM (ui),const cmdline::parsed_cmdline & UTILS_UNUSED_PARAM (cmdline),const config::tree & UTILS_UNUSED_PARAM (user_config))116*6b3a42afSjmmv run(cmdline::ui* UTILS_UNUSED_PARAM(ui),
117*6b3a42afSjmmv const cmdline::parsed_cmdline& UTILS_UNUSED_PARAM(cmdline),
118*6b3a42afSjmmv const config::tree& UTILS_UNUSED_PARAM(user_config))
119*6b3a42afSjmmv {
120*6b3a42afSjmmv UNREACHABLE;
121*6b3a42afSjmmv }
122*6b3a42afSjmmv };
123*6b3a42afSjmmv
124*6b3a42afSjmmv
125*6b3a42afSjmmv /// Initializes the cmdline library and generates the set of test commands.
126*6b3a42afSjmmv ///
127*6b3a42afSjmmv /// \param [out] commands A mapping that is updated to contain the commands to
128*6b3a42afSjmmv /// use for testing.
129*6b3a42afSjmmv static void
setup(cmdline::commands_map<cli::cli_command> & commands)130*6b3a42afSjmmv setup(cmdline::commands_map< cli::cli_command >& commands)
131*6b3a42afSjmmv {
132*6b3a42afSjmmv cmdline::init("progname");
133*6b3a42afSjmmv
134*6b3a42afSjmmv commands.insert(new cmd_mock_simple("mock_simple"));
135*6b3a42afSjmmv commands.insert(new cmd_mock_complex("mock_complex"));
136*6b3a42afSjmmv
137*6b3a42afSjmmv commands.insert(new cmd_mock_simple("mock_simple_2"), "First");
138*6b3a42afSjmmv commands.insert(new cmd_mock_complex("mock_complex_2"), "First");
139*6b3a42afSjmmv
140*6b3a42afSjmmv commands.insert(new cmd_mock_simple("mock_simple_3"), "Second");
141*6b3a42afSjmmv }
142*6b3a42afSjmmv
143*6b3a42afSjmmv
144*6b3a42afSjmmv /// Performs a test on the global help (not that of a subcommand).
145*6b3a42afSjmmv ///
146*6b3a42afSjmmv /// \param general_options The genral options supported by the tool, if any.
147*6b3a42afSjmmv /// \param expected_options Expected lines of help output documenting the
148*6b3a42afSjmmv /// options in general_options.
149*6b3a42afSjmmv /// \param ui The cmdline::mock_ui object to which to write the output.
150*6b3a42afSjmmv static void
global_test(const cmdline::options_vector & general_options,const std::vector<std::string> & expected_options,cmdline::ui_mock & ui)151*6b3a42afSjmmv global_test(const cmdline::options_vector& general_options,
152*6b3a42afSjmmv const std::vector< std::string >& expected_options,
153*6b3a42afSjmmv cmdline::ui_mock& ui)
154*6b3a42afSjmmv {
155*6b3a42afSjmmv cmdline::commands_map< cli::cli_command > mock_commands;
156*6b3a42afSjmmv setup(mock_commands);
157*6b3a42afSjmmv
158*6b3a42afSjmmv cmdline::args_vector args;
159*6b3a42afSjmmv args.push_back("help");
160*6b3a42afSjmmv
161*6b3a42afSjmmv cmd_help cmd(&general_options, &mock_commands);
162*6b3a42afSjmmv ATF_REQUIRE_EQ(EXIT_SUCCESS, cmd.main(&ui, args, engine::default_config()));
163*6b3a42afSjmmv
164*6b3a42afSjmmv std::vector< std::string > expected;
165*6b3a42afSjmmv
166*6b3a42afSjmmv expected.push_back("Usage: progname [general_options] command "
167*6b3a42afSjmmv "[command_options] [args]");
168*6b3a42afSjmmv if (!general_options.empty()) {
169*6b3a42afSjmmv expected.push_back("");
170*6b3a42afSjmmv expected.push_back("Available general options:");
171*6b3a42afSjmmv std::copy(expected_options.begin(), expected_options.end(),
172*6b3a42afSjmmv std::back_inserter(expected));
173*6b3a42afSjmmv }
174*6b3a42afSjmmv expected.push_back("");
175*6b3a42afSjmmv expected.push_back("Generic commands:");
176*6b3a42afSjmmv expected.push_back(" mock_complex Complex command.");
177*6b3a42afSjmmv expected.push_back(" mock_simple Simple command.");
178*6b3a42afSjmmv expected.push_back("");
179*6b3a42afSjmmv expected.push_back("First commands:");
180*6b3a42afSjmmv expected.push_back(" mock_complex_2 Complex command.");
181*6b3a42afSjmmv expected.push_back(" mock_simple_2 Simple command.");
182*6b3a42afSjmmv expected.push_back("");
183*6b3a42afSjmmv expected.push_back("Second commands:");
184*6b3a42afSjmmv expected.push_back(" mock_simple_3 Simple command.");
185*6b3a42afSjmmv expected.push_back("");
186*6b3a42afSjmmv expected.push_back("See kyua(1) for more details.");
187*6b3a42afSjmmv
188*6b3a42afSjmmv ATF_REQUIRE(expected == ui.out_log());
189*6b3a42afSjmmv ATF_REQUIRE(ui.err_log().empty());
190*6b3a42afSjmmv }
191*6b3a42afSjmmv
192*6b3a42afSjmmv
193*6b3a42afSjmmv } // anonymous namespace
194*6b3a42afSjmmv
195*6b3a42afSjmmv
196*6b3a42afSjmmv ATF_TEST_CASE_WITHOUT_HEAD(global__no_options);
ATF_TEST_CASE_BODY(global__no_options)197*6b3a42afSjmmv ATF_TEST_CASE_BODY(global__no_options)
198*6b3a42afSjmmv {
199*6b3a42afSjmmv cmdline::ui_mock ui;
200*6b3a42afSjmmv
201*6b3a42afSjmmv cmdline::options_vector general_options;
202*6b3a42afSjmmv
203*6b3a42afSjmmv global_test(general_options, std::vector< std::string >(), ui);
204*6b3a42afSjmmv }
205*6b3a42afSjmmv
206*6b3a42afSjmmv
207*6b3a42afSjmmv ATF_TEST_CASE_WITHOUT_HEAD(global__some_options);
ATF_TEST_CASE_BODY(global__some_options)208*6b3a42afSjmmv ATF_TEST_CASE_BODY(global__some_options)
209*6b3a42afSjmmv {
210*6b3a42afSjmmv cmdline::ui_mock ui;
211*6b3a42afSjmmv
212*6b3a42afSjmmv cmdline::options_vector general_options;
213*6b3a42afSjmmv const cmdline::bool_option flag_a("flag_a", "Flag A");
214*6b3a42afSjmmv general_options.push_back(&flag_a);
215*6b3a42afSjmmv const cmdline::string_option flag_c('c', "lc", "Flag C", "X");
216*6b3a42afSjmmv general_options.push_back(&flag_c);
217*6b3a42afSjmmv
218*6b3a42afSjmmv std::vector< std::string > expected;
219*6b3a42afSjmmv expected.push_back(" --flag_a Flag A.");
220*6b3a42afSjmmv expected.push_back(" -c X, --lc=X Flag C.");
221*6b3a42afSjmmv
222*6b3a42afSjmmv global_test(general_options, expected, ui);
223*6b3a42afSjmmv }
224*6b3a42afSjmmv
225*6b3a42afSjmmv
226*6b3a42afSjmmv ATF_TEST_CASE_WITHOUT_HEAD(subcommand__simple);
ATF_TEST_CASE_BODY(subcommand__simple)227*6b3a42afSjmmv ATF_TEST_CASE_BODY(subcommand__simple)
228*6b3a42afSjmmv {
229*6b3a42afSjmmv cmdline::options_vector general_options;
230*6b3a42afSjmmv
231*6b3a42afSjmmv cmdline::commands_map< cli::cli_command > mock_commands;
232*6b3a42afSjmmv setup(mock_commands);
233*6b3a42afSjmmv
234*6b3a42afSjmmv cmdline::args_vector args;
235*6b3a42afSjmmv args.push_back("help");
236*6b3a42afSjmmv args.push_back("mock_simple");
237*6b3a42afSjmmv
238*6b3a42afSjmmv cmd_help cmd(&general_options, &mock_commands);
239*6b3a42afSjmmv cmdline::ui_mock ui;
240*6b3a42afSjmmv ATF_REQUIRE_EQ(EXIT_SUCCESS, cmd.main(&ui, args, engine::default_config()));
241*6b3a42afSjmmv ATF_REQUIRE(atf::utils::grep_collection("^Usage: progname \\[general_options\\] "
242*6b3a42afSjmmv "mock_simple$", ui.out_log()));
243*6b3a42afSjmmv ATF_REQUIRE(!atf::utils::grep_collection("Available.*options", ui.out_log()));
244*6b3a42afSjmmv ATF_REQUIRE(atf::utils::grep_collection("^See kyua-mock_simple\\(1\\) for more "
245*6b3a42afSjmmv "details.", ui.out_log()));
246*6b3a42afSjmmv ATF_REQUIRE(ui.err_log().empty());
247*6b3a42afSjmmv }
248*6b3a42afSjmmv
249*6b3a42afSjmmv
250*6b3a42afSjmmv ATF_TEST_CASE_WITHOUT_HEAD(subcommand__complex);
ATF_TEST_CASE_BODY(subcommand__complex)251*6b3a42afSjmmv ATF_TEST_CASE_BODY(subcommand__complex)
252*6b3a42afSjmmv {
253*6b3a42afSjmmv cmdline::options_vector general_options;
254*6b3a42afSjmmv const cmdline::bool_option global_a("global_a", "Global A");
255*6b3a42afSjmmv general_options.push_back(&global_a);
256*6b3a42afSjmmv const cmdline::string_option global_c('c', "global_c", "Global C",
257*6b3a42afSjmmv "c_global");
258*6b3a42afSjmmv general_options.push_back(&global_c);
259*6b3a42afSjmmv
260*6b3a42afSjmmv cmdline::commands_map< cli::cli_command > mock_commands;
261*6b3a42afSjmmv setup(mock_commands);
262*6b3a42afSjmmv
263*6b3a42afSjmmv cmdline::args_vector args;
264*6b3a42afSjmmv args.push_back("help");
265*6b3a42afSjmmv args.push_back("mock_complex");
266*6b3a42afSjmmv
267*6b3a42afSjmmv cmd_help cmd(&general_options, &mock_commands);
268*6b3a42afSjmmv cmdline::ui_mock ui;
269*6b3a42afSjmmv ATF_REQUIRE_EQ(EXIT_SUCCESS, cmd.main(&ui, args, engine::default_config()));
270*6b3a42afSjmmv ATF_REQUIRE(atf::utils::grep_collection(
271*6b3a42afSjmmv "^Usage: progname \\[general_options\\] mock_complex "
272*6b3a42afSjmmv "\\[command_options\\] \\[arg1 .. argN\\]$", ui.out_log()));
273*6b3a42afSjmmv ATF_REQUIRE(atf::utils::grep_collection("Available general options",
274*6b3a42afSjmmv ui.out_log()));
275*6b3a42afSjmmv ATF_REQUIRE(atf::utils::grep_collection("--global_a", ui.out_log()));
276*6b3a42afSjmmv ATF_REQUIRE(atf::utils::grep_collection("--global_c=c_global",
277*6b3a42afSjmmv ui.out_log()));
278*6b3a42afSjmmv ATF_REQUIRE(atf::utils::grep_collection("Available command options",
279*6b3a42afSjmmv ui.out_log()));
280*6b3a42afSjmmv ATF_REQUIRE(atf::utils::grep_collection("--flag_a *Flag A",
281*6b3a42afSjmmv ui.out_log()));
282*6b3a42afSjmmv ATF_REQUIRE(atf::utils::grep_collection("-b.*--flag_b *Flag B",
283*6b3a42afSjmmv ui.out_log()));
284*6b3a42afSjmmv ATF_REQUIRE(atf::utils::grep_collection(
285*6b3a42afSjmmv "-c c_arg.*--flag_c=c_arg *Flag C", ui.out_log()));
286*6b3a42afSjmmv ATF_REQUIRE(atf::utils::grep_collection(
287*6b3a42afSjmmv "--flag_d=d_arg *Flag D.*default.*foo", ui.out_log()));
288*6b3a42afSjmmv ATF_REQUIRE(atf::utils::grep_collection(
289*6b3a42afSjmmv "^See kyua-mock_complex\\(1\\) for more details.", ui.out_log()));
290*6b3a42afSjmmv ATF_REQUIRE(ui.err_log().empty());
291*6b3a42afSjmmv }
292*6b3a42afSjmmv
293*6b3a42afSjmmv
294*6b3a42afSjmmv ATF_TEST_CASE_WITHOUT_HEAD(subcommand__unknown);
ATF_TEST_CASE_BODY(subcommand__unknown)295*6b3a42afSjmmv ATF_TEST_CASE_BODY(subcommand__unknown)
296*6b3a42afSjmmv {
297*6b3a42afSjmmv cmdline::options_vector general_options;
298*6b3a42afSjmmv
299*6b3a42afSjmmv cmdline::commands_map< cli::cli_command > mock_commands;
300*6b3a42afSjmmv setup(mock_commands);
301*6b3a42afSjmmv
302*6b3a42afSjmmv cmdline::args_vector args;
303*6b3a42afSjmmv args.push_back("help");
304*6b3a42afSjmmv args.push_back("foobar");
305*6b3a42afSjmmv
306*6b3a42afSjmmv cmd_help cmd(&general_options, &mock_commands);
307*6b3a42afSjmmv cmdline::ui_mock ui;
308*6b3a42afSjmmv ATF_REQUIRE_THROW_RE(cmdline::usage_error, "command foobar.*not exist",
309*6b3a42afSjmmv cmd.main(&ui, args, engine::default_config()));
310*6b3a42afSjmmv ATF_REQUIRE(ui.out_log().empty());
311*6b3a42afSjmmv ATF_REQUIRE(ui.err_log().empty());
312*6b3a42afSjmmv }
313*6b3a42afSjmmv
314*6b3a42afSjmmv
315*6b3a42afSjmmv ATF_TEST_CASE_WITHOUT_HEAD(invalid_args);
ATF_TEST_CASE_BODY(invalid_args)316*6b3a42afSjmmv ATF_TEST_CASE_BODY(invalid_args)
317*6b3a42afSjmmv {
318*6b3a42afSjmmv cmdline::options_vector general_options;
319*6b3a42afSjmmv
320*6b3a42afSjmmv cmdline::commands_map< cli::cli_command > mock_commands;
321*6b3a42afSjmmv setup(mock_commands);
322*6b3a42afSjmmv
323*6b3a42afSjmmv cmdline::args_vector args;
324*6b3a42afSjmmv args.push_back("help");
325*6b3a42afSjmmv args.push_back("mock_simple");
326*6b3a42afSjmmv args.push_back("mock_complex");
327*6b3a42afSjmmv
328*6b3a42afSjmmv cmd_help cmd(&general_options, &mock_commands);
329*6b3a42afSjmmv cmdline::ui_mock ui;
330*6b3a42afSjmmv ATF_REQUIRE_THROW_RE(cmdline::usage_error, "Too many arguments",
331*6b3a42afSjmmv cmd.main(&ui, args, engine::default_config()));
332*6b3a42afSjmmv ATF_REQUIRE(ui.out_log().empty());
333*6b3a42afSjmmv ATF_REQUIRE(ui.err_log().empty());
334*6b3a42afSjmmv }
335*6b3a42afSjmmv
336*6b3a42afSjmmv
ATF_INIT_TEST_CASES(tcs)337*6b3a42afSjmmv ATF_INIT_TEST_CASES(tcs)
338*6b3a42afSjmmv {
339*6b3a42afSjmmv ATF_ADD_TEST_CASE(tcs, global__no_options);
340*6b3a42afSjmmv ATF_ADD_TEST_CASE(tcs, global__some_options);
341*6b3a42afSjmmv ATF_ADD_TEST_CASE(tcs, subcommand__simple);
342*6b3a42afSjmmv ATF_ADD_TEST_CASE(tcs, subcommand__complex);
343*6b3a42afSjmmv ATF_ADD_TEST_CASE(tcs, subcommand__unknown);
344*6b3a42afSjmmv ATF_ADD_TEST_CASE(tcs, invalid_args);
345*6b3a42afSjmmv }
346