// Copyright 2015 The Kyua Authors. // All rights reserved. // // Redistribution and use in source and binary forms, with or without // modification, are permitted provided that the following conditions are // met: // // * Redistributions of source code must retain the above copyright // notice, this list of conditions and the following disclaimer. // * Redistributions in binary form must reproduce the above copyright // notice, this list of conditions and the following disclaimer in the // documentation and/or other materials provided with the distribution. // * Neither the name of Google Inc. nor the names of its contributors // may be used to endorse or promote products derived from this software // without specific prior written permission. // // THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS // "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT // LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR // A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT // OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, // SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT // LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, // DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY // THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT // (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE // OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. #include "engine/tap_parser.hpp" #include #include #include "engine/exceptions.hpp" #include "utils/format/containers.ipp" #include "utils/format/macros.hpp" #include "utils/fs/path.hpp" namespace fs = utils::fs; namespace { /// Helper to execute parse_tap_output() on inline text contents. /// /// \param contents The TAP output to parse. /// /// \return The tap_summary object resultingafter the parse. /// /// \throw engine::load_error If parse_tap_output() fails. static engine::tap_summary do_parse(const std::string& contents) { std::ofstream output("tap.txt"); ATF_REQUIRE(output); output << contents; output.close(); return engine::parse_tap_output(fs::path("tap.txt")); } } // anonymous namespace ATF_TEST_CASE_WITHOUT_HEAD(tap_summary__bailed_out); ATF_TEST_CASE_BODY(tap_summary__bailed_out) { const engine::tap_summary summary = engine::tap_summary::new_bailed_out(); ATF_REQUIRE(summary.bailed_out()); } ATF_TEST_CASE_WITHOUT_HEAD(tap_summary__some_results); ATF_TEST_CASE_BODY(tap_summary__some_results) { const engine::tap_summary summary = engine::tap_summary::new_results( engine::tap_plan(1, 5), 3, 2); ATF_REQUIRE(!summary.bailed_out()); ATF_REQUIRE_EQ(engine::tap_plan(1, 5), summary.plan()); ATF_REQUIRE_EQ(3, summary.ok_count()); ATF_REQUIRE_EQ(2, summary.not_ok_count()); } ATF_TEST_CASE_WITHOUT_HEAD(tap_summary__all_skipped); ATF_TEST_CASE_BODY(tap_summary__all_skipped) { const engine::tap_summary summary = engine::tap_summary::new_all_skipped( "Skipped"); ATF_REQUIRE(!summary.bailed_out()); ATF_REQUIRE_EQ(engine::tap_plan(1, 0), summary.plan()); ATF_REQUIRE_EQ("Skipped", summary.all_skipped_reason()); } ATF_TEST_CASE_WITHOUT_HEAD(tap_summary__equality_operators); ATF_TEST_CASE_BODY(tap_summary__equality_operators) { const engine::tap_summary bailed_out = engine::tap_summary::new_bailed_out(); const engine::tap_summary all_skipped_1 = engine::tap_summary::new_all_skipped("Reason 1"); const engine::tap_summary results_1 = engine::tap_summary::new_results(engine::tap_plan(1, 5), 3, 2); // Self-equality checks. ATF_REQUIRE( bailed_out == bailed_out); ATF_REQUIRE(!(bailed_out != bailed_out)); ATF_REQUIRE( all_skipped_1 == all_skipped_1); ATF_REQUIRE(!(all_skipped_1 != all_skipped_1)); ATF_REQUIRE( results_1 == results_1); ATF_REQUIRE(!(results_1 != results_1)); // Cross-equality checks. ATF_REQUIRE(!(bailed_out == all_skipped_1)); ATF_REQUIRE( bailed_out != all_skipped_1); ATF_REQUIRE(!(bailed_out == results_1)); ATF_REQUIRE( bailed_out != results_1); ATF_REQUIRE(!(all_skipped_1 == results_1)); ATF_REQUIRE( all_skipped_1 != results_1); // Checks for the all_skipped "type". const engine::tap_summary all_skipped_2 = engine::tap_summary::new_all_skipped("Reason 2"); ATF_REQUIRE(!(all_skipped_1 == all_skipped_2)); ATF_REQUIRE( all_skipped_1 != all_skipped_2); // Checks for the results "type", different plan. const engine::tap_summary results_2 = engine::tap_summary::new_results(engine::tap_plan(2, 6), results_1.ok_count(), results_1.not_ok_count()); ATF_REQUIRE(!(results_1 == results_2)); ATF_REQUIRE( results_1 != results_2); // Checks for the results "type", different counts. const engine::tap_summary results_3 = engine::tap_summary::new_results(results_1.plan(), results_1.not_ok_count(), results_1.ok_count()); ATF_REQUIRE(!(results_1 == results_3)); ATF_REQUIRE( results_1 != results_3); } ATF_TEST_CASE_WITHOUT_HEAD(tap_summary__output); ATF_TEST_CASE_BODY(tap_summary__output) { { const engine::tap_summary summary = engine::tap_summary::new_bailed_out(); ATF_REQUIRE_EQ( "tap_summary{bailed_out=true}", (F("%s") % summary).str()); } { const engine::tap_summary summary = engine::tap_summary::new_results(engine::tap_plan(5, 10), 2, 4); ATF_REQUIRE_EQ( "tap_summary{bailed_out=false, plan=5..10, ok_count=2, " "not_ok_count=4}", (F("%s") % summary).str()); } { const engine::tap_summary summary = engine::tap_summary::new_all_skipped("Who knows"); ATF_REQUIRE_EQ( "tap_summary{bailed_out=false, plan=1..0, " "all_skipped_reason=Who knows}", (F("%s") % summary).str()); } } ATF_TEST_CASE_WITHOUT_HEAD(parse_tap_output__only_one_result); ATF_TEST_CASE_BODY(parse_tap_output__only_one_result) { const engine::tap_summary summary = do_parse( "1..1\n" "ok - 1\n"); const engine::tap_summary exp_summary = engine::tap_summary::new_results(engine::tap_plan(1, 1), 1, 0); ATF_REQUIRE_EQ(exp_summary, summary); } ATF_TEST_CASE_WITHOUT_HEAD(parse_tap_output__all_pass); ATF_TEST_CASE_BODY(parse_tap_output__all_pass) { const engine::tap_summary summary = do_parse( "1..8\n" "ok - 1\n" " Some diagnostic message\n" "ok - 2 This test also passed\n" "garbage line\n" "ok - 3 This test passed\n" "not ok 4 # SKIP Some reason\n" "not ok 5 # TODO Another reason\n" "ok - 6 Doesn't make a difference SKIP\n" "ok - 7 Doesn't make a difference either TODO\n" "ok # Also works without a number\n"); const engine::tap_summary exp_summary = engine::tap_summary::new_results(engine::tap_plan(1, 8), 8, 0); ATF_REQUIRE_EQ(exp_summary, summary); } ATF_TEST_CASE_WITHOUT_HEAD(parse_tap_output__some_fail); ATF_TEST_CASE_BODY(parse_tap_output__some_fail) { const engine::tap_summary summary = do_parse( "garbage line\n" "not ok - 1 This test failed\n" "ok - 2 This test passed\n" "not ok - 3 This test failed\n" "1..6\n" "not ok - 4 This test failed\n" "ok - 5 This test passed\n" "not ok # Fails as well without a number\n"); const engine::tap_summary exp_summary = engine::tap_summary::new_results(engine::tap_plan(1, 6), 2, 4); ATF_REQUIRE_EQ(exp_summary, summary); } ATF_TEST_CASE_WITHOUT_HEAD(parse_tap_output__skip_and_todo_variants); ATF_TEST_CASE_BODY(parse_tap_output__skip_and_todo_variants) { const engine::tap_summary summary = do_parse( "1..8\n" "not ok - 1 # SKIP Some reason\n" "not ok - 2 # skip Some reason\n" "not ok - 3 # Skipped Some reason\n" "not ok - 4 # skipped Some reason\n" "not ok - 5 # Skipped: Some reason\n" "not ok - 6 # skipped: Some reason\n" "not ok - 7 # TODO Some reason\n" "not ok - 8 # todo Some reason\n"); const engine::tap_summary exp_summary = engine::tap_summary::new_results(engine::tap_plan(1, 8), 8, 0); ATF_REQUIRE_EQ(exp_summary, summary); } ATF_TEST_CASE_WITHOUT_HEAD(parse_tap_output__skip_all_with_reason); ATF_TEST_CASE_BODY(parse_tap_output__skip_all_with_reason) { const engine::tap_summary summary = do_parse( "1..0 SKIP Some reason for skipping\n" "ok - 1\n" " Some diagnostic message\n" "ok - 6 Doesn't make a difference SKIP\n" "ok - 7 Doesn't make a difference either TODO\n"); const engine::tap_summary exp_summary = engine::tap_summary::new_all_skipped("Some reason for skipping"); ATF_REQUIRE_EQ(exp_summary, summary); } ATF_TEST_CASE_WITHOUT_HEAD(parse_tap_output__skip_all_without_reason); ATF_TEST_CASE_BODY(parse_tap_output__skip_all_without_reason) { const engine::tap_summary summary = do_parse( "1..0 unrecognized # garbage skip\n"); const engine::tap_summary exp_summary = engine::tap_summary::new_all_skipped("No reason specified"); ATF_REQUIRE_EQ(exp_summary, summary); } ATF_TEST_CASE_WITHOUT_HEAD(parse_tap_output__skip_all_invalid); ATF_TEST_CASE_BODY(parse_tap_output__skip_all_invalid) { ATF_REQUIRE_THROW_RE(engine::load_error, "Skipped plan must be 1\\.\\.0", do_parse("1..3 # skip\n")); } ATF_TEST_CASE_WITHOUT_HEAD(parse_tap_output__plan_at_end); ATF_TEST_CASE_BODY(parse_tap_output__plan_at_end) { const engine::tap_summary summary = do_parse( "ok - 1\n" " Some diagnostic message\n" "ok - 2 This test also passed\n" "garbage line\n" "ok - 3 This test passed\n" "not ok 4 # SKIP Some reason\n" "not ok 5 # TODO Another reason\n" "ok - 6 Doesn't make a difference SKIP\n" "ok - 7 Doesn't make a difference either TODO\n" "1..7\n"); const engine::tap_summary exp_summary = engine::tap_summary::new_results(engine::tap_plan(1, 7), 7, 0); ATF_REQUIRE_EQ(exp_summary, summary); } ATF_TEST_CASE_WITHOUT_HEAD(parse_tap_output__stray_oks); ATF_TEST_CASE_BODY(parse_tap_output__stray_oks) { const engine::tap_summary summary = do_parse( "1..3\n" "ok - 1\n" "ok\n" "ok - 2 This test also passed\n" "not ok\n" "ok - 3 This test passed\n"); const engine::tap_summary exp_summary = engine::tap_summary::new_results(engine::tap_plan(1, 3), 3, 0); ATF_REQUIRE_EQ(exp_summary, summary); } ATF_TEST_CASE_WITHOUT_HEAD(parse_tap_output__no_plan); ATF_TEST_CASE_BODY(parse_tap_output__no_plan) { ATF_REQUIRE_THROW_RE( engine::load_error, "Output did not contain any TAP plan", do_parse( "not ok - 1 This test failed\n" "ok - 2 This test passed\n")); } ATF_TEST_CASE_WITHOUT_HEAD(parse_tap_output__double_plan); ATF_TEST_CASE_BODY(parse_tap_output__double_plan) { ATF_REQUIRE_THROW_RE( engine::load_error, "Found duplicate plan", do_parse( "garbage line\n" "1..5\n" "not ok - 1 This test failed\n" "ok - 2 This test passed\n" "1..8\n" "ok\n")); } ATF_TEST_CASE_WITHOUT_HEAD(parse_tap_output__inconsistent_plan); ATF_TEST_CASE_BODY(parse_tap_output__inconsistent_plan) { ATF_REQUIRE_THROW_RE( engine::load_error, "Reported plan differs from actual executed tests", do_parse( "1..3\n" "not ok - 1 This test failed\n" "ok - 2 This test passed\n")); } ATF_TEST_CASE_WITHOUT_HEAD(parse_tap_output__inconsistent_trailing_plan); ATF_TEST_CASE_BODY(parse_tap_output__inconsistent_trailing_plan) { ATF_REQUIRE_THROW_RE( engine::load_error, "Reported plan differs from actual executed tests", do_parse( "not ok - 1 This test failed\n" "ok - 2 This test passed\n" "1..3\n")); } ATF_TEST_CASE_WITHOUT_HEAD(parse_tap_output__insane_plan); ATF_TEST_CASE_BODY(parse_tap_output__insane_plan) { ATF_REQUIRE_THROW_RE( engine::load_error, "Invalid value", do_parse("120830981209831..234891793874080981092803981092312\n")); } ATF_TEST_CASE_WITHOUT_HEAD(parse_tap_output__reversed_plan); ATF_TEST_CASE_BODY(parse_tap_output__reversed_plan) { ATF_REQUIRE_THROW_RE(engine::load_error, "Found reversed plan 8\\.\\.5", do_parse("8..5\n")); } ATF_TEST_CASE_WITHOUT_HEAD(parse_tap_output__bail_out); ATF_TEST_CASE_BODY(parse_tap_output__bail_out) { const engine::tap_summary summary = do_parse( "1..3\n" "not ok - 1 This test failed\n" "Bail out! There is some unknown problem\n" "ok - 2 This test passed\n"); const engine::tap_summary exp_summary = engine::tap_summary::new_bailed_out(); ATF_REQUIRE_EQ(exp_summary, summary); } ATF_TEST_CASE_WITHOUT_HEAD(parse_tap_output__bail_out_wins_over_no_plan); ATF_TEST_CASE_BODY(parse_tap_output__bail_out_wins_over_no_plan) { const engine::tap_summary summary = do_parse( "not ok - 1 This test failed\n" "Bail out! There is some unknown problem\n" "ok - 2 This test passed\n"); const engine::tap_summary exp_summary = engine::tap_summary::new_bailed_out(); ATF_REQUIRE_EQ(exp_summary, summary); } ATF_TEST_CASE_WITHOUT_HEAD(parse_tap_output__open_failure); ATF_TEST_CASE_BODY(parse_tap_output__open_failure) { ATF_REQUIRE_THROW_RE(engine::load_error, "hello.txt.*Failed to open", engine::parse_tap_output(fs::path("hello.txt"))); } ATF_INIT_TEST_CASES(tcs) { ATF_ADD_TEST_CASE(tcs, tap_summary__bailed_out); ATF_ADD_TEST_CASE(tcs, tap_summary__some_results); ATF_ADD_TEST_CASE(tcs, tap_summary__all_skipped); ATF_ADD_TEST_CASE(tcs, tap_summary__equality_operators); ATF_ADD_TEST_CASE(tcs, tap_summary__output); ATF_ADD_TEST_CASE(tcs, parse_tap_output__only_one_result); ATF_ADD_TEST_CASE(tcs, parse_tap_output__all_pass); ATF_ADD_TEST_CASE(tcs, parse_tap_output__some_fail); ATF_ADD_TEST_CASE(tcs, parse_tap_output__skip_and_todo_variants); ATF_ADD_TEST_CASE(tcs, parse_tap_output__skip_all_without_reason); ATF_ADD_TEST_CASE(tcs, parse_tap_output__skip_all_with_reason); ATF_ADD_TEST_CASE(tcs, parse_tap_output__skip_all_invalid); ATF_ADD_TEST_CASE(tcs, parse_tap_output__plan_at_end); ATF_ADD_TEST_CASE(tcs, parse_tap_output__stray_oks); ATF_ADD_TEST_CASE(tcs, parse_tap_output__no_plan); ATF_ADD_TEST_CASE(tcs, parse_tap_output__double_plan); ATF_ADD_TEST_CASE(tcs, parse_tap_output__inconsistent_plan); ATF_ADD_TEST_CASE(tcs, parse_tap_output__inconsistent_trailing_plan); ATF_ADD_TEST_CASE(tcs, parse_tap_output__insane_plan); ATF_ADD_TEST_CASE(tcs, parse_tap_output__reversed_plan); ATF_ADD_TEST_CASE(tcs, parse_tap_output__bail_out); ATF_ADD_TEST_CASE(tcs, parse_tap_output__bail_out_wins_over_no_plan); ATF_ADD_TEST_CASE(tcs, parse_tap_output__open_failure); }