xref: /netbsd-src/external/bsd/kyua-cli/dist/cli/cmd_help_test.cpp (revision 6b3a42af15b5e090c339512c790dd68f3d11a9d8)
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