Home | History | Annotate | Line # | Download | only in cli
common_test.cpp revision 1.1
      1 // Copyright 2011 Google Inc.
      2 // All rights reserved.
      3 //
      4 // Redistribution and use in source and binary forms, with or without
      5 // modification, are permitted provided that the following conditions are
      6 // met:
      7 //
      8 // * Redistributions of source code must retain the above copyright
      9 //   notice, this list of conditions and the following disclaimer.
     10 // * Redistributions in binary form must reproduce the above copyright
     11 //   notice, this list of conditions and the following disclaimer in the
     12 //   documentation and/or other materials provided with the distribution.
     13 // * Neither the name of Google Inc. nor the names of its contributors
     14 //   may be used to endorse or promote products derived from this software
     15 //   without specific prior written permission.
     16 //
     17 // THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
     18 // "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
     19 // LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
     20 // A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
     21 // OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
     22 // SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
     23 // LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
     24 // DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
     25 // THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
     26 // (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
     27 // OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
     28 
     29 #include "cli/common.hpp"
     30 
     31 #include <fstream>
     32 
     33 #include <atf-c++.hpp>
     34 
     35 #include "engine/exceptions.hpp"
     36 #include "engine/filters.hpp"
     37 #include "engine/test_case.hpp"
     38 #include "engine/test_program.hpp"
     39 #include "engine/test_result.hpp"
     40 #include "utils/cmdline/exceptions.hpp"
     41 #include "utils/cmdline/globals.hpp"
     42 #include "utils/cmdline/parser.ipp"
     43 #include "utils/cmdline/ui_mock.hpp"
     44 #include "utils/datetime.hpp"
     45 #include "utils/env.hpp"
     46 #include "utils/fs/exceptions.hpp"
     47 #include "utils/fs/operations.hpp"
     48 #include "utils/fs/path.hpp"
     49 #include "utils/optional.ipp"
     50 #include "utils/sanity.hpp"
     51 
     52 namespace cmdline = utils::cmdline;
     53 namespace config = utils::config;
     54 namespace datetime = utils::datetime;
     55 namespace fs = utils::fs;
     56 
     57 using utils::optional;
     58 
     59 
     60 namespace {
     61 
     62 
     63 /// Syntactic sugar to instantiate engine::test_filter objects.
     64 inline engine::test_filter
     65 mkfilter(const char* test_program, const char* test_case)
     66 {
     67     return engine::test_filter(fs::path(test_program), test_case);
     68 }
     69 
     70 
     71 }  // anonymous namespace
     72 
     73 
     74 ATF_TEST_CASE_WITHOUT_HEAD(build_root_path__default);
     75 ATF_TEST_CASE_BODY(build_root_path__default)
     76 {
     77     std::map< std::string, std::vector< std::string > > options;
     78     const cmdline::parsed_cmdline mock_cmdline(options, cmdline::args_vector());
     79 
     80     ATF_REQUIRE(!cli::build_root_path(mock_cmdline));
     81 }
     82 
     83 
     84 ATF_TEST_CASE_WITHOUT_HEAD(build_root_path__explicit);
     85 ATF_TEST_CASE_BODY(build_root_path__explicit)
     86 {
     87     std::map< std::string, std::vector< std::string > > options;
     88     options["build-root"].push_back("/my//path");
     89     const cmdline::parsed_cmdline mock_cmdline(options, cmdline::args_vector());
     90 
     91     ATF_REQUIRE(cli::build_root_path(mock_cmdline));
     92     ATF_REQUIRE_EQ("/my/path", cli::build_root_path(mock_cmdline).get().str());
     93 }
     94 
     95 
     96 ATF_TEST_CASE_WITHOUT_HEAD(get_home__ok);
     97 ATF_TEST_CASE_BODY(get_home__ok)
     98 {
     99     const fs::path home("/foo/bar");
    100     utils::setenv("HOME", home.str());
    101     const optional< fs::path > computed = cli::get_home();
    102     ATF_REQUIRE(computed);
    103     ATF_REQUIRE_EQ(home, computed.get());
    104 }
    105 
    106 
    107 ATF_TEST_CASE_WITHOUT_HEAD(get_home__missing);
    108 ATF_TEST_CASE_BODY(get_home__missing)
    109 {
    110     utils::unsetenv("HOME");
    111     ATF_REQUIRE(!cli::get_home());
    112 }
    113 
    114 
    115 ATF_TEST_CASE_WITHOUT_HEAD(get_home__invalid);
    116 ATF_TEST_CASE_BODY(get_home__invalid)
    117 {
    118     utils::setenv("HOME", "");
    119     ATF_REQUIRE(!cli::get_home());
    120 }
    121 
    122 
    123 ATF_TEST_CASE_WITHOUT_HEAD(kyuafile_path__default);
    124 ATF_TEST_CASE_BODY(kyuafile_path__default)
    125 {
    126     std::map< std::string, std::vector< std::string > > options;
    127     options["kyuafile"].push_back(cli::kyuafile_option.default_value());
    128     const cmdline::parsed_cmdline mock_cmdline(options, cmdline::args_vector());
    129 
    130     ATF_REQUIRE_EQ(cli::kyuafile_option.default_value(),
    131                    cli::kyuafile_path(mock_cmdline).str());
    132 }
    133 
    134 
    135 ATF_TEST_CASE_WITHOUT_HEAD(kyuafile_path__explicit);
    136 ATF_TEST_CASE_BODY(kyuafile_path__explicit)
    137 {
    138     std::map< std::string, std::vector< std::string > > options;
    139     options["kyuafile"].push_back("/my//path");
    140     const cmdline::parsed_cmdline mock_cmdline(options, cmdline::args_vector());
    141 
    142     ATF_REQUIRE_EQ("/my/path", cli::kyuafile_path(mock_cmdline).str());
    143 }
    144 
    145 
    146 ATF_TEST_CASE_WITHOUT_HEAD(store_path__default__create_directory__ok);
    147 ATF_TEST_CASE_BODY(store_path__default__create_directory__ok)
    148 {
    149     std::map< std::string, std::vector< std::string > > options;
    150     options["store"].push_back(cli::store_option.default_value());
    151     const cmdline::parsed_cmdline mock_cmdline(options, cmdline::args_vector());
    152 
    153     const fs::path home("homedir");
    154     utils::setenv("HOME", home.str());
    155 
    156     ATF_REQUIRE(!fs::exists(home / ".kyua"));
    157     ATF_REQUIRE_EQ(home / ".kyua/store.db", cli::store_path(mock_cmdline));
    158     ATF_REQUIRE(fs::exists(home / ".kyua"));
    159 }
    160 
    161 
    162 ATF_TEST_CASE(store_path__default__create_directory__fail);
    163 ATF_TEST_CASE_HEAD(store_path__default__create_directory__fail)
    164 {
    165     set_md_var("require.user", "unprivileged");
    166 }
    167 ATF_TEST_CASE_BODY(store_path__default__create_directory__fail)
    168 {
    169     std::map< std::string, std::vector< std::string > > options;
    170     options["store"].push_back(cli::store_option.default_value());
    171     const cmdline::parsed_cmdline mock_cmdline(options, cmdline::args_vector());
    172 
    173     const fs::path home("homedir");
    174     utils::setenv("HOME", home.str());
    175     fs::mkdir(home, 0555);
    176 
    177     ATF_REQUIRE_THROW(fs::error, cli::store_path(mock_cmdline).str());
    178     ATF_REQUIRE(!fs::exists(home / ".kyua"));
    179 }
    180 
    181 
    182 ATF_TEST_CASE_WITHOUT_HEAD(store_path__default__no_home);
    183 ATF_TEST_CASE_BODY(store_path__default__no_home)
    184 {
    185     std::map< std::string, std::vector< std::string > > options;
    186     options["store"].push_back(cli::store_option.default_value());
    187     const cmdline::parsed_cmdline mock_cmdline(options, cmdline::args_vector());
    188 
    189     utils::unsetenv("HOME");
    190 
    191     ATF_REQUIRE_EQ("kyua-store.db", cli::store_path(mock_cmdline).str());
    192 }
    193 
    194 
    195 ATF_TEST_CASE_WITHOUT_HEAD(store_path__explicit);
    196 ATF_TEST_CASE_BODY(store_path__explicit)
    197 {
    198     std::map< std::string, std::vector< std::string > > options;
    199     options["store"].push_back("/my//path");
    200     const cmdline::parsed_cmdline mock_cmdline(options, cmdline::args_vector());
    201 
    202     const fs::path home("homedir");
    203     ATF_REQUIRE_EQ("/my/path", cli::store_path(mock_cmdline).str());
    204     ATF_REQUIRE(!fs::exists(home / ".kyua"));
    205 }
    206 
    207 
    208 ATF_TEST_CASE_WITHOUT_HEAD(parse_filters__none);
    209 ATF_TEST_CASE_BODY(parse_filters__none)
    210 {
    211     const cmdline::args_vector args;
    212     const std::set< engine::test_filter > filters = cli::parse_filters(args);
    213     ATF_REQUIRE(filters.empty());
    214 }
    215 
    216 
    217 ATF_TEST_CASE_WITHOUT_HEAD(parse_filters__ok);
    218 ATF_TEST_CASE_BODY(parse_filters__ok)
    219 {
    220     cmdline::args_vector args;
    221     args.push_back("foo");
    222     args.push_back("bar/baz");
    223     args.push_back("other:abc");
    224     args.push_back("other:bcd");
    225     const std::set< engine::test_filter > filters = cli::parse_filters(args);
    226 
    227     std::set< engine::test_filter > exp_filters;
    228     exp_filters.insert(mkfilter("foo", ""));
    229     exp_filters.insert(mkfilter("bar/baz", ""));
    230     exp_filters.insert(mkfilter("other", "abc"));
    231     exp_filters.insert(mkfilter("other", "bcd"));
    232 
    233     ATF_REQUIRE(exp_filters == filters);
    234 }
    235 
    236 
    237 ATF_TEST_CASE_WITHOUT_HEAD(parse_filters__duplicate);
    238 ATF_TEST_CASE_BODY(parse_filters__duplicate)
    239 {
    240     cmdline::args_vector args;
    241     args.push_back("foo/bar//baz");
    242     args.push_back("hello/world:yes");
    243     args.push_back("foo//bar/baz");
    244     ATF_REQUIRE_THROW_RE(cmdline::error, "Duplicate.*'foo/bar/baz'",
    245                          cli::parse_filters(args));
    246 }
    247 
    248 
    249 ATF_TEST_CASE_WITHOUT_HEAD(parse_filters__nondisjoint);
    250 ATF_TEST_CASE_BODY(parse_filters__nondisjoint)
    251 {
    252     cmdline::args_vector args;
    253     args.push_back("foo/bar");
    254     args.push_back("hello/world:yes");
    255     args.push_back("foo/bar:baz");
    256     ATF_REQUIRE_THROW_RE(cmdline::error, "'foo/bar'.*'foo/bar:baz'.*disjoint",
    257                          cli::parse_filters(args));
    258 }
    259 
    260 
    261 ATF_TEST_CASE_WITHOUT_HEAD(report_unused_filters__none);
    262 ATF_TEST_CASE_BODY(report_unused_filters__none)
    263 {
    264     std::set< engine::test_filter > unused;
    265 
    266     cmdline::ui_mock ui;
    267     ATF_REQUIRE(!cli::report_unused_filters(unused, &ui));
    268     ATF_REQUIRE(ui.out_log().empty());
    269     ATF_REQUIRE(ui.err_log().empty());
    270 }
    271 
    272 
    273 ATF_TEST_CASE_WITHOUT_HEAD(report_unused_filters__some);
    274 ATF_TEST_CASE_BODY(report_unused_filters__some)
    275 {
    276     std::set< engine::test_filter > unused;
    277     unused.insert(mkfilter("a/b", ""));
    278     unused.insert(mkfilter("hey/d", "yes"));
    279 
    280     cmdline::ui_mock ui;
    281     cmdline::init("progname");
    282     ATF_REQUIRE(cli::report_unused_filters(unused, &ui));
    283     ATF_REQUIRE(ui.out_log().empty());
    284     ATF_REQUIRE_EQ(2, ui.err_log().size());
    285     ATF_REQUIRE( atf::utils::grep_collection("No.*matched.*'a/b'",
    286                                              ui.err_log()));
    287     ATF_REQUIRE( atf::utils::grep_collection("No.*matched.*'hey/d:yes'",
    288                                              ui.err_log()));
    289 }
    290 
    291 
    292 ATF_TEST_CASE_WITHOUT_HEAD(format_delta);
    293 ATF_TEST_CASE_BODY(format_delta)
    294 {
    295     ATF_REQUIRE_EQ("0.000s", cli::format_delta(datetime::delta()));
    296     ATF_REQUIRE_EQ("0.012s", cli::format_delta(datetime::delta(0, 12300)));
    297     ATF_REQUIRE_EQ("0.999s", cli::format_delta(datetime::delta(0, 999000)));
    298     ATF_REQUIRE_EQ("51.321s", cli::format_delta(datetime::delta(51, 321000)));
    299 }
    300 
    301 
    302 ATF_TEST_CASE_WITHOUT_HEAD(format_result__no_reason);
    303 ATF_TEST_CASE_BODY(format_result__no_reason)
    304 {
    305     ATF_REQUIRE_EQ("passed", cli::format_result(
    306         engine::test_result(engine::test_result::passed)));
    307     ATF_REQUIRE_EQ("failed", cli::format_result(
    308         engine::test_result(engine::test_result::failed)));
    309 }
    310 
    311 
    312 ATF_TEST_CASE_WITHOUT_HEAD(format_result__with_reason);
    313 ATF_TEST_CASE_BODY(format_result__with_reason)
    314 {
    315     ATF_REQUIRE_EQ("broken: Something", cli::format_result(
    316         engine::test_result(engine::test_result::broken, "Something")));
    317     ATF_REQUIRE_EQ("expected_failure: A B C", cli::format_result(
    318         engine::test_result(engine::test_result::expected_failure, "A B C")));
    319     ATF_REQUIRE_EQ("failed: More text", cli::format_result(
    320         engine::test_result(engine::test_result::failed, "More text")));
    321     ATF_REQUIRE_EQ("skipped: Bye", cli::format_result(
    322         engine::test_result(engine::test_result::skipped, "Bye")));
    323 }
    324 
    325 
    326 ATF_TEST_CASE_WITHOUT_HEAD(format_test_case_id__test_case);
    327 ATF_TEST_CASE_BODY(format_test_case_id__test_case)
    328 {
    329     const engine::test_program test_program(
    330         "mock", fs::path("foo/bar/baz"), fs::path("unused-root"),
    331         "unused-suite-name", engine::metadata_builder().build());
    332     const engine::test_case test_case("mock", test_program, "abc",
    333                                       engine::metadata_builder().build());
    334     ATF_REQUIRE_EQ("foo/bar/baz:abc", cli::format_test_case_id(test_case));
    335 }
    336 
    337 
    338 ATF_TEST_CASE_WITHOUT_HEAD(format_test_case_id__test_filter);
    339 ATF_TEST_CASE_BODY(format_test_case_id__test_filter)
    340 {
    341     const engine::test_filter filter(fs::path("foo/bar"), "baz");
    342     ATF_REQUIRE_EQ("foo/bar:baz", cli::format_test_case_id(filter));
    343 }
    344 
    345 
    346 ATF_INIT_TEST_CASES(tcs)
    347 {
    348     ATF_ADD_TEST_CASE(tcs, build_root_path__default);
    349     ATF_ADD_TEST_CASE(tcs, build_root_path__explicit);
    350 
    351     ATF_ADD_TEST_CASE(tcs, get_home__ok);
    352     ATF_ADD_TEST_CASE(tcs, get_home__missing);
    353     ATF_ADD_TEST_CASE(tcs, get_home__invalid);
    354 
    355     ATF_ADD_TEST_CASE(tcs, kyuafile_path__default);
    356     ATF_ADD_TEST_CASE(tcs, kyuafile_path__explicit);
    357 
    358     ATF_ADD_TEST_CASE(tcs, store_path__default__create_directory__ok);
    359     ATF_ADD_TEST_CASE(tcs, store_path__default__create_directory__fail);
    360     ATF_ADD_TEST_CASE(tcs, store_path__default__no_home);
    361     ATF_ADD_TEST_CASE(tcs, store_path__explicit);
    362 
    363     ATF_ADD_TEST_CASE(tcs, parse_filters__none);
    364     ATF_ADD_TEST_CASE(tcs, parse_filters__ok);
    365     ATF_ADD_TEST_CASE(tcs, parse_filters__duplicate);
    366     ATF_ADD_TEST_CASE(tcs, parse_filters__nondisjoint);
    367 
    368     ATF_ADD_TEST_CASE(tcs, report_unused_filters__none);
    369     ATF_ADD_TEST_CASE(tcs, report_unused_filters__some);
    370 
    371     ATF_ADD_TEST_CASE(tcs, format_delta);
    372 
    373     ATF_ADD_TEST_CASE(tcs, format_result__no_reason);
    374     ATF_ADD_TEST_CASE(tcs, format_result__with_reason);
    375 
    376     ATF_ADD_TEST_CASE(tcs, format_test_case_id__test_case);
    377     ATF_ADD_TEST_CASE(tcs, format_test_case_id__test_filter);
    378 }
    379