// file : libbuild2/build/script/parser.test.cxx -*- C++ -*- // license : MIT; see accompanying LICENSE file #include <iostream> #include <libbuild2/types.hxx> #include <libbuild2/utility.hxx> #include <libbuild2/target.hxx> #include <libbuild2/context.hxx> #include <libbuild2/scheduler.hxx> #include <libbuild2/file-cache.hxx> #include <libbuild2/build/script/script.hxx> // line #include <libbuild2/build/script/parser.hxx> #include <libbuild2/build/script/runner.hxx> #undef NDEBUG #include <cassert> using namespace std; namespace build2 { namespace build { namespace script { class print_runner: public runner { public: print_runner (bool line, bool iterations): line_ (line), iterations_ (iterations) {} virtual void enter (environment&, const location&) override {} virtual void run (environment& env, const command_expr& e, const iteration_index* ii, size_t i, const function<command_function>& cf, const location& ll) override { // If the functions is specified, then just execute it with an empty // stdin so it can perform the housekeeping (stop replaying tokens, // increment line index, etc). // if (cf != nullptr) { assert (e.size () == 1 && !e[0].pipe.empty ()); const command& c (e[0].pipe.back ()); // Must be enforced by the caller. // assert (!c.out && !c.err && !c.exit); cf (env, c.arguments, fdopen_null (), nullptr /* pipe */, nullopt /* deadline */, ll); } cout << e; if (line_ || iterations_) print_line_info (ii, i); cout << endl; } virtual bool run_cond (environment&, const command_expr& e, const iteration_index* ii, size_t i, const location&) override { cout << "? " << e; if (line_ || iterations_) print_line_info (ii, i); cout << endl; return e.back ().pipe.back ().program.recall.string () == "true"; } virtual void leave (environment&, const location&) override {} private: void print_line_info (const iteration_index* ii, size_t i) const { cout << " #"; if (line_) cout << ' ' << i; if (iterations_ && ii != nullptr) { string s; for (const iteration_index* i (ii); i != nullptr; i = i->prev) s.insert (0, " i" + to_string (i->index)); cout << s; } } private: bool line_; bool iterations_; }; // Usages: // // argv[0] [-l] [-r] // argv[0] -b [-t] // argv[0] -d [-t] // argv[0] -g [-t] [<diag-name>] // argv[0] -q // // In the first form read the script from stdin and trace the script // body execution to stdout using the custom print runner. // // In the second form read the script from stdin, parse it and dump the // script body lines to stdout. // // In the third form read the script from stdin, parse it and dump the // depdb preamble lines to stdout. // // In the forth form read the script from stdin, parse it and print the // low-verbosity script diagnostics name or custom low-verbosity // diagnostics to stdout. If the script doesn't deduce any of them, then // print the diagnostics and exit with non-zero code. // // In the fifth form read the script from stdin, parse it and print // line tokens quoting information to stdout. // // -l // Print the script line number for each executed expression. // // -r // Print the loop iteration numbers for each executed expression. // // -b // Dump the parsed script body to stdout. // // -d // Dump the parsed script depdb preamble to stdout. // // -g // Dump the low-verbosity script diagnostics name or custom // low-verbosity diagnostics to stdout. // // -t // Print true if the body (-b), depdb preamble (-d), or diag preamble // (-g) references the temporary directory and false otherwise. // // -q // Print the parsed script tokens quoting information to sdout. If a // token is quoted follow its representation with its quoting // information in the [<quoting>/<completeness>] form, where: // // <quoting> := 'S' | 'D' | 'M' // <completeness> := 'C' | 'P' // int main (int argc, char* argv[]) { tracer trace ("main"); enum class mode { run, body, depdb_preamble, diag, quoting } m (mode::run); bool print_line (false); bool print_iterations (false); optional<string> diag_name; bool temp_dir (false); for (int i (1); i != argc; ++i) { string a (argv[i]); if (a == "-l") print_line = true; else if (a == "-r") print_iterations = true; else if (a == "-b") m = mode::body; else if (a == "-d") m = mode::depdb_preamble; else if (a == "-g") m = mode::diag; else if (a == "-t") { assert (m == mode::body || m == mode::depdb_preamble || m == mode::diag); temp_dir = true; } else if (a == "-q") m = mode::quoting; else { if (m == mode::diag) { diag_name = move (a); break; } assert (false); } } assert (!print_line || m == mode::run || m == mode::diag); assert (!print_iterations || m == mode::run || m == mode::diag); assert (!diag_name || m == mode::diag); // Fake build system driver, default verbosity. // init_diag (1); init (nullptr, argv[0], true); // Serial execution. // scheduler sched (1); global_mutexes mutexes (1); file_cache fcache (true); context ctx (sched, mutexes, fcache); try { cin.exceptions (istream::failbit | istream::badbit); // Enter mock target. Use fixed name and path so that we can use // them in expected results. Strictly speaking target path should // be absolute. However, the buildscript implementation doesn't // really care. // file& tt ( ctx.targets.insert<file> (work, dir_path (), "driver", string (), trace)); tt.path (path ("driver")); const scope& bs (tt.base_scope ()); small_vector<action, 1> acts {perform_update_id}; // Parse and run. // parser p (ctx); path_name nm ("buildfile"); script s (p.pre_parse (bs, tt.type (), acts, cin, nm, 11 /* line */, (m != mode::diag ? optional<string> ("test") : move (diag_name)), location (nm, 10))); switch (m) { case mode::run: { environment e (perform_update_id, tt, bs, false /* temp_dir */); print_runner r (print_line, print_iterations); bool exec_diag (!s.diag_preamble.empty ()); if (exec_diag) { if (s.diag_preamble_temp_dir) e.set_temp_dir_variable (); p.execute_diag_preamble (ctx.global_scope, ctx.global_scope, e, s, r, false /* diag */, true /* enter */, false /* leave */); } if (s.body_temp_dir && !s.diag_preamble_temp_dir) e.set_temp_dir_variable (); p.execute_body (ctx.global_scope, ctx.global_scope, e, s, r, !exec_diag /* enter */); break; } case mode::diag: { if (s.diag_name) { cout << "name: " << *s.diag_name << endl; } else { if (!temp_dir) { environment e (perform_update_id, tt, bs, s.diag_preamble_temp_dir); print_runner r (print_line, print_iterations); names diag (p.execute_diag_preamble (ctx.global_scope, ctx.global_scope, e, s, r, true /* diag */, true /* enter */, true /* leave */).first); cout << "diag: " << diag << endl; } else cout << (s.diag_preamble_temp_dir ? "true" : "false") << endl; } break; } case mode::body: { if (!temp_dir) dump (cout, "", s.body); else cout << (s.body_temp_dir ? "true" : "false") << endl; break; } case mode::depdb_preamble: { if (!temp_dir) dump (cout, "", s.depdb_preamble); else cout << (s.depdb_preamble_temp_dir ? "true" : "false") << endl; break; } case mode::quoting: { for (const line& l: s.body) { for (const replay_token& rt: l.tokens) { if (&rt != &l.tokens[0]) cout << ' '; const token& t (rt.token); cout << t; char q ('\0'); switch (t.qtype) { case quote_type::single: q = 'S'; break; case quote_type::double_: q = 'D'; break; case quote_type::mixed: q = 'M'; break; case quote_type::unquoted: break; } if (q != '\0') cout << " [" << q << (t.qcomp ? "/C" : "/P") << ']'; } } cout << endl; } } } catch (const failed&) { return 1; } return 0; } } } } int main (int argc, char* argv[]) { return build2::build::script::main (argc, argv); }