// file : libbuild2/test/rule.cxx -*- C++ -*- // copyright : Copyright (c) 2014-2019 Code Synthesis Ltd // license : MIT; see accompanying LICENSE file #include <libbuild2/test/rule.hxx> #include <libbuild2/scope.hxx> #include <libbuild2/target.hxx> #include <libbuild2/algorithm.hxx> #include <libbuild2/filesystem.hxx> #include <libbuild2/diagnostics.hxx> #include <libbuild2/test/target.hxx> #include <libbuild2/test/script/parser.hxx> #include <libbuild2/test/script/runner.hxx> #include <libbuild2/test/script/script.hxx> using namespace std; using namespace butl; namespace build2 { namespace test { bool rule:: match (action, target&, const string&) const { // We always match, even if this target is not testable (so that we can // ignore it; see apply()). // return true; } recipe rule:: apply (action a, target& t) const { // Note that we are called both as the outer part during the update-for- // test pre-operation and as the inner part during the test operation // itself. // // In both cases we first determine if the target is testable and return // noop if it's not. Otherwise, in the first case (update for test) we // delegate to the normal update and in the second (test) -- perform the // test. // // And to add a bit more complexity, we want to handle aliases slightly // differently: we may not want to ignore their prerequisites if the // alias is not testable since their prerequisites could be. // // Here is the state matrix: // // test'able | pass'able | neither // | | // update-for-test delegate (& pass) | pass | noop // ---------------------------------------+-------------+--------- // test test (& pass) | pass | noop // auto& pts (t.prerequisite_targets[a]); // Resolve group members. // if (!see_through || t.type ().see_through) { // Remember that we are called twice: first during update for test // (pre-operation) and then during test. During the former, we rely on // the normall update rule to resolve the group members. During the // latter, there will be no rule to do this but the group will already // have been resolved by the pre-operation. // // If the rule could not resolve the group, then we ignore it. // group_view gv (a.outer () ? resolve_members (a, t) : t.group_members (a)); if (gv.members != nullptr) { for (size_t i (0); i != gv.count; ++i) { if (const target* m = gv.members[i]) pts.push_back (m); } match_members (a, t, pts); } } // If we are passing-through, then match our prerequisites. // if (t.is_a<alias> () && pass (t)) { // For the test operation we have to implement our own search and // match because we need to ignore prerequisites that are outside of // our project. They can be from projects that don't use the test // module (and thus won't have a suitable rule). Or they can be from // no project at all (e.g., installed). Also, generally, not testing // stuff that's not ours seems right. // match_prerequisites (a, t, t.root_scope ()); } size_t pass_n (pts.size ()); // Number of pass-through prerequisites. // See if it's testable and if so, what kind. // bool test (false); bool script (false); if (this->test (t)) { // We have two very different cases: testscript and simple test (plus // it may not be a testable target at all). So as the first step // determine which case this is. // // If we have any prerequisites of the testscript{} type, then this is // the testscript case. // // If we can, go inside see-through groups. Normally groups won't be // resolvable for this action but then normally they won't contain any // testscripts either. In other words, if there is a group that // contains testscripts as members then it will need to arrange for // the members to be resolvable (e.g., by registering an appropriate // rule for the test operation). // for (prerequisite_member p: group_prerequisite_members (a, t, members_mode::maybe)) { if (include (a, t, p) != include_type::normal) // Excluded/ad hoc. continue; if (p.is_a<testscript> ()) { if (!script) { script = true; // We treat this target as testable unless the test variable is // explicitly set to false. // const name* n (cast_null<name> (t[var_test])); test = (n == nullptr || !n->simple () || n->value != "false"); if (!test) break; } // Collect testscripts after the pass-through prerequisites. // const target& pt (p.search (t)); // Note that for the test operation itself we don't match nor // execute them relying on update to assign their paths. // // Causing update for test inputs/scripts is tricky: we cannot // match for update-for-install because this same rule will match // and since the target is not testable, it will return the noop // recipe. // // So what we are going to do is directly match (and also execute; // see below) a recipe for the inner update (who thought we could // do that... but it seems we can). While at first it might feel // iffy, it does make sense: the outer rule we would have matched // would have simply delegated to the inner so we might as well // take a shortcut. The only potential drawback of this approach // is that we won't be able to provide any for-test customizations // when updating test inputs/scripts. But such a need seems rather // far fetched. // if (a.operation () == update_id) match_inner (a, pt); pts.push_back (&pt); } } // If this is not a script, then determine if it is a simple test. // Ignore testscript files themselves at the outset. // if (!script && !t.is_a<testscript> ()) { // For the simple case whether this is a test is controlled by the // test variable. Also, it feels redundant to specify, say, "test = // true" and "test.stdout = test.out" -- the latter already says this // is a test. // const name* n (cast_null<name> (t[var_test])); // If the test variable is explicitly set to false then we treat // it as not testable regardless of what other test.* variables // or prerequisites we might have. // // Note that the test variable can be set to an "override" target // (which means 'true' for our purposes). // if (n != nullptr && n->simple () && n->value == "false") test = false; else { // Look for test input/stdin/stdout prerequisites. The same group // reasoning as in the testscript case above. // for (prerequisite_member p: group_prerequisite_members (a, t, members_mode::maybe)) { const auto& vars (p.prerequisite.vars); if (vars.empty ()) // Common case. continue; if (include (a, t, p) != include_type::normal) // Excluded/ad hoc. continue; bool rt ( cast_false<bool> (vars[test_roundtrip])); bool si (rt || cast_false<bool> (vars[test_stdin])); bool so (rt || cast_false<bool> (vars[test_stdout])); bool in ( cast_false<bool> (vars[test_input])); if (si || so || in) { // Verify it is file-based. // if (!p.is_a<file> ()) { fail << "test." << (si ? "stdin" : so ? "stdout" : "input") << " prerequisite " << p << " of target " << t << " is not a file"; } if (!test) { test = true; // First matching prerequisite. Establish the structure in // pts: the first element (after pass_n) is stdin (can be // NULL), the second is stdout (can be NULL), and everything // after that (if any) is inputs. // pts.push_back (nullptr); // stdin pts.push_back (nullptr); // stdout } // Collect them after the pass-through prerequisites. // // Note that for the test operation itself we don't match nor // execute them relying on update to assign their paths. // auto match = [a, &p, &t] () -> const target* { const target& pt (p.search (t)); // The same match_inner() rationale as for the testcript // prerequisites above. // if (a.operation () == update_id) match_inner (a, pt); return &pt; }; if (si) { if (pts[pass_n] != nullptr) fail << "multiple test.stdin prerequisites for target " << t; pts[pass_n] = match (); } if (so) { if (pts[pass_n + 1] != nullptr) fail << "multiple test.stdout prerequisites for target " << t; pts[pass_n + 1] = match (); } if (in) pts.push_back (match ()); } } if (!test) test = (n != nullptr); // We have the test variable. if (!test) test = t[test_options] || t[test_arguments]; } } } // Neither testing nor passing-through. // if (!test && pass_n == 0) return noop_recipe; // If we are only passing-through, then use the default recipe (which // will execute all the matched prerequisites). // if (!test) return default_recipe; // Being here means we are definitely testing and maybe passing-through. // if (a.operation () == update_id) { // For the update pre-operation match the inner rule (actual update). // match_inner (a, t); return [pass_n] (action a, const target& t) { return perform_update (a, t, pass_n); }; } else { if (script) { return [pass_n, this] (action a, const target& t) { return perform_script (a, t, pass_n); }; } else { return [pass_n, this] (action a, const target& t) { return perform_test (a, t, pass_n); }; } } } target_state rule:: perform_update (action a, const target& t, size_t pass_n) { // First execute the inner recipe then execute prerequisites. // target_state ts (execute_inner (a, t)); if (pass_n != 0) ts |= straight_execute_prerequisites (a, t, pass_n); ts |= straight_execute_prerequisites_inner (a, t, 0, pass_n); return ts; } static script::scope_state perform_script_impl (const target& t, const testscript& ts, const dir_path& wd, const common& c) { using namespace script; scope_state r; try { build2::test::script::script s (t, ts, wd); { parser p; p.pre_parse (s); default_runner r (c); p.execute (s, r); } r = s.state; } catch (const failed&) { r = scope_state::failed; } return r; } target_state rule:: perform_script (action a, const target& t, size_t pass_n) const { // First pass through. // if (pass_n != 0) straight_execute_prerequisites (a, t, pass_n); // Figure out whether the testscript file is called 'testscript', in // which case it should be the only one. // auto& pts (t.prerequisite_targets[a]); size_t pts_n (pts.size ()); bool one; { optional<bool> o; for (size_t i (pass_n); i != pts_n; ++i) { const testscript& ts (*pts[i]->is_a<testscript> ()); bool r (ts.name == "testscript"); if ((r && o) || (!r && o && *o)) fail << "both 'testscript' and other names specified for " << t; o = r; } assert (o); // We should have a testscript or we wouldn't be here. one = *o; } // Calculate root working directory. It is in the out_base of the target // and is called just test for dir{} targets and test-<target-name> for // other targets. // dir_path wd (t.out_dir ()); if (t.is_a<dir> ()) wd /= "test"; else wd /= "test-" + t.name; // Are we backlinking the test working directory to src? (See // backlink_*() in algorithm.cxx for details.) // const scope& bs (t.base_scope ()); const scope& rs (*bs.root_scope ()); const path& buildignore_file (rs.root_extra->buildignore_file); dir_path bl; if (cast_false<bool> (rs.vars[var_forwarded])) { bl = bs.src_path () / wd.leaf (bs.out_path ()); clean_backlink (bl, verb_never); } // If this is a (potentially) multi-testscript test, then create (and // later cleanup) the root directory. If this is just 'testscript', then // the root directory is used directly as test's working directory and // it's the runner's responsibility to create and clean it up. // // Note that we create the root directory containing the .buildignore // file to make sure that it is ignored by name patterns (see the // buildignore description for details). // // What should we do if the directory already exists? We used to fail // which meant the user had to go and clean things up manually every // time a test failed. This turned out to be really annoying. So now we // issue a warning and clean it up automatically. The drawbacks of this // approach are the potential loss of data from the previous failed test // run and the possibility of deleting user-created files. // if (exists (static_cast<const path&> (wd), false)) fail << "working directory " << wd << " is a file/symlink"; if (exists (wd)) { if (before != output_before::clean) { bool fail (before == output_before::fail); (fail ? error : warn) << "working directory " << wd << " exists " << (empty_buildignore (wd, buildignore_file) ? "" : "and is not empty ") << "at the beginning of the test"; if (fail) throw failed (); } // Remove the directory itself not to confuse the runner which tries // to detect when tests stomp on each others feet. // build2::rmdir_r (wd, true, 2); } // Delay actually creating the directory in case all the tests are // ignored (via config.test). // bool mk (!one); // Start asynchronous execution of the testscripts. // wait_guard wg; if (!dry_run) wg = wait_guard (target::count_busy (), t[a].task_count); // Result vector. // using script::scope_state; vector<scope_state> res; res.reserve (pts_n - pass_n); // Make sure there are no reallocations. for (size_t i (pass_n); i != pts_n; ++i) { const testscript& ts (*pts[i]->is_a<testscript> ()); // If this is just the testscript, then its id path is empty (and it // can only be ignored by ignoring the test target, which makes sense // since it's the only testscript file). // if (one || test (t, path (ts.name))) { // Because the creation of the output directory is shared between us // and the script implementation (plus the fact that we actually // don't clean the existing one), we are going to ignore it for // dry-run. // if (!dry_run) { if (mk) { mkdir_buildignore (wd, buildignore_file, 2); mk = false; } } if (verb) { diag_record dr (text); dr << "test " << ts; if (!t.is_a<alias> ()) dr << ' ' << t; } res.push_back (dry_run ? scope_state::passed : scope_state::unknown); if (!dry_run) { scope_state& r (res.back ()); if (!sched.async (target::count_busy (), t[a].task_count, [this] (const diag_frame* ds, scope_state& r, const target& t, const testscript& ts, const dir_path& wd) { diag_frame::stack_guard dsg (ds); r = perform_script_impl (t, ts, wd, *this); }, diag_frame::stack (), ref (r), cref (t), cref (ts), cref (wd))) { // Executed synchronously. If failed and we were not asked to // keep going, bail out. // if (r == scope_state::failed && !keep_going) break; } } } } if (!dry_run) wg.wait (); // Re-examine. // bool bad (false); for (scope_state r: res) { switch (r) { case scope_state::passed: break; case scope_state::failed: bad = true; break; case scope_state::unknown: assert (false); } if (bad) break; } // Cleanup. // if (!dry_run) { if (!bad && !one && !mk && after == output_after::clean) { if (!empty_buildignore (wd, buildignore_file)) fail << "working directory " << wd << " is not empty at the " << "end of the test"; rmdir_buildignore (wd, buildignore_file, 2); } } // Backlink if the working directory exists. // // If we dry-run then presumably all tests passed and we shouldn't // have anything left unless we are keeping the output. // if (!bl.empty () && (dry_run ? after == output_after::keep : exists (wd))) update_backlink (wd, bl, true /* changed */); if (bad) throw failed (); return target_state::changed; } // The format of args shall be: // // name1 arg arg ... nullptr // name2 arg arg ... nullptr // ... // nameN arg arg ... nullptr nullptr // static bool run_test (const target& t, diag_record& dr, char const** args, process* prev = nullptr) { // Find the next process, if any. // char const** next (args); for (next++; *next != nullptr; next++) ; next++; // Redirect stdout to a pipe unless we are last. // int out (*next != nullptr ? -1 : 1); bool pr; process_exit pe; try { process p (prev == nullptr ? process (args, 0, out) // First process. : process (args, *prev, out)); // Next process. pr = *next == nullptr || run_test (t, dr, next, &p); p.wait (); assert (p.exit); pe = *p.exit; } catch (const process_error& e) { error << "unable to execute " << args[0] << ": " << e; if (e.child) exit (1); throw failed (); } bool wr (pe.normal () && pe.code () == 0); if (!wr) { if (pr) // First failure? dr << fail << "test " << t << " failed"; // Multi test: test 1. dr << error; print_process (dr, args); dr << " " << pe; } return pr && wr; } target_state rule:: perform_test (action a, const target& tt, size_t pass_n) const { // First pass through. // if (pass_n != 0) straight_execute_prerequisites (a, tt, pass_n); // See if we have the test executable override. // path p; { // Note that the test variable's visibility is target. // lookup l (tt[var_test]); // Note that we have similar code for scripted tests. // const target* t (nullptr); if (l.defined ()) { const name* n (cast_null<name> (l)); if (n == nullptr) fail << "invalid test executable override: null value"; else if (n->empty ()) fail << "invalid test executable override: empty value"; else if (n->simple ()) { // Ignore the special 'true' value. // if (n->value != "true") p = path (n->value); else t = &tt; } else if (n->directory ()) fail << "invalid test executable override: '" << *n << "'"; else { // Must be a target name. // // @@ OUT: what if this is a @-qualified pair of names? // t = search_existing (*n, tt.base_scope ()); if (t == nullptr) fail << "invalid test executable override: unknown target: '" << *n << "'"; } } else // By default we set it to the test target's path. // t = &tt; if (t != nullptr) { if (auto* pt = t->is_a<path_target> ()) { // Do some sanity checks: the target better be up-to-date with // an assigned path. // p = pt->path (); if (p.empty ()) fail << "target " << *pt << " specified in the test variable " << "is out of date" << info << "consider specifying it as a prerequisite of " << tt; } else fail << "target " << *t << (t != &tt ? " specified in the test variable " : " requested to be tested ") << "is not path-based"; } } // See apply() for the structure of prerequisite_targets in the presence // of test.{input,stdin,stdout}. // auto& pts (tt.prerequisite_targets[a]); size_t pts_n (pts.size ()); cstrings args; // Do we have stdin? // // We simulate stdin redirect (<file) with a fake (already terminate) // cat pipe (cat file |). // bool sin (pass_n != pts_n && pts[pass_n] != nullptr); process cat; if (sin) { const file& it (pts[pass_n]->as<file> ()); const path& ip (it.path ()); assert (!ip.empty ()); // Should have been assigned by update. cat = process (process_exit (0)); // Successfully exited. if (!dry_run) { try { cat.in_ofd = fdopen (ip, fdopen_mode::in); } catch (const io_error& e) { fail << "unable to open " << ip << ": " << e; } } // Purely for diagnostics. // args.push_back ("cat"); args.push_back (ip.string ().c_str ()); args.push_back (nullptr); } // If dry-run, the target may not exist. // process_path pp (!dry_run ? run_search (p, true /* init */) : try_run_search (p, true)); args.push_back (pp.empty () ? p.string ().c_str () : pp.recall_string ()); // Do we have options and/or arguments? // if (auto l = tt[test_options]) append_options (args, cast<strings> (l)); if (auto l = tt[test_arguments]) append_options (args, cast<strings> (l)); // Do we have inputs? // for (size_t i (pass_n + 2); i < pts_n; ++i) { const file& it (pts[i]->as<file> ()); const path& ip (it.path ()); assert (!ip.empty ()); // Should have been assigned by update. args.push_back (ip.string ().c_str ()); } args.push_back (nullptr); // Do we have stdout? // path dp ("diff"); process_path dpp; if (pass_n != pts_n && pts[pass_n + 1] != nullptr) { const file& ot (pts[pass_n + 1]->as<file> ()); const path& op (ot.path ()); assert (!op.empty ()); // Should have been assigned by update. dpp = run_search (dp, true); args.push_back (dpp.recall_string ()); args.push_back ("-u"); // Note that MinGW-built diff utility (as of 3.3) fails trying to // detect if stdin contains text or binary data. We will help it a bit // to workaround the issue. // #ifdef _WIN32 args.push_back ("--text"); #endif // Ignore Windows newline fluff if that's what we are running on. // if (cast<target_triplet> (tt[test_target]).class_ == "windows") args.push_back ("--strip-trailing-cr"); args.push_back (op.string ().c_str ()); args.push_back ("-"); args.push_back (nullptr); } args.push_back (nullptr); // Second. if (verb >= 2) print_process (args); else if (verb) text << "test " << tt; if (!dry_run) { diag_record dr; if (!run_test (tt, dr, args.data () + (sin ? 3 : 0), // Skip cat. sin ? &cat : nullptr)) { dr << info << "test command line: "; print_process (dr, args); dr << endf; // return } } return target_state::changed; } } }