// file : tests/process/driver.cxx -*- C++ -*- // license : MIT; see accompanying LICENSE file #include <ios> #include <string> #include <vector> #include <chrono> #include <sstream> #include <iterator> // istreambuf_iterator, ostream_iterator #include <algorithm> // copy() #include <iostream> #include <libbutl/path.hxx> #include <libbutl/utility.hxx> // setenv(), getenv() #include <libbutl/process.hxx> #include <libbutl/process-io.hxx> #include <libbutl/optional.hxx> #include <libbutl/fdstream.hxx> #include <libbutl/timestamp.hxx> #undef NDEBUG #include <cassert> using namespace std; using namespace butl; using cstrings = vector<const char*>; bool roundtrip_arg (const path& p, const string& a) { string r; try { cstrings args {p.string ().c_str (), "-a", a.c_str (), nullptr}; process pr (args.data (), 0 /* in */, -1 /* out */, 2 /* err */); ifdstream is (move (pr.in_ofd)); assert (getline (is, r)); assert (pr.wait ()); } catch (const process_error& e) { //cerr << args[0] << ": " << e << endl; if (e.child) exit (1); assert (false); } return r == a; } static bool exec (const path& p, vector<char> in = vector<char> (), bool out = false, bool err = false, bool pipeline = false, bool bin = true, // Set binary mode for file descriptors. dir_path wd = dir_path (), // Set the working dir for the child process. bool env = false) // Set environment variables for the child. { using butl::optional; assert (!in.empty () || (!out && !err)); // Nothing to output if no input. assert (!pipeline || out); // To pipeline need to output something. const char* cwd (!wd.empty () ? wd.string ().c_str () : nullptr); cstrings args {p.string ().c_str (), "-c"}; if (bin) args.push_back ("-b"); const char* evars[] = { "PAR1", "PAR2=2P", "PAR6=66", "PAR7", // Override the process variables. "THR1", "THR2=2T", // Override the thread variables. "CHD1", // Unset a non-existing variable. "CHD2=C2", // Add the new variable. nullptr}; if (env) args.push_back ("-e"); if (cwd != nullptr) args.push_back (cwd); args.push_back (nullptr); try { bool r (true); // If both o and e are true, then redirect STDERR to STDOUT, so both can be // read from the same stream. // process pr (args.data (), !in.empty () ? -1 : -2, out ? -1 : -2, err ? (out ? 1 : -1) : -2, cwd, env ? evars : nullptr); try { if (!in.empty ()) { r = !pr.try_wait (); // Couldn't exit as waiting for the input. auto bin_mode = [bin](auto_fd fd) -> auto_fd { if (bin) fdmode (fd.get (), fdstream_mode::binary); return fd; }; ofdstream os (bin_mode (move (pr.out_fd))); copy (in.begin (), in.end (), ostream_iterator<char> (os)); os.close (); if (out) { vector<char> o; if (pipeline) { // Here we test both passing process output fd as an input for // another process (pr2.in = pr.out), as well as passing process // input fd as an output for another one (pr2.out = pr3.in). The // overall pipeline looks like 'os -> pr -> pr2 -> pr3 -> is'. // process pr3 (args.data (), -1, -1, -2, cwd, env ? evars : nullptr); process pr2 (args.data (), pr, bin_mode (move (pr3.out_fd)).get (), -2, cwd, env ? evars : nullptr); ifdstream is (bin_mode (move (pr3.in_ofd))); o = is.read_binary (); // While at it, make sure that the process::timed_wait() template // function overloads can be properly instantiated/linked. // r = pr2.timed_wait (duration::max ()) && r; r = pr3.timed_wait (chrono::milliseconds::max ()) && r; } else { ifdstream is (bin_mode (move (pr.in_ofd))); o = is.read_binary (); } if (err) { // If STDERR is redirected to STDOUT then output will be // duplicated. // vector<char> v (in); in.reserve (in.size () * 2); in.insert (in.end (), v.begin (), v.end ()); } r = in == o && r; } if (err && !out) { ifdstream is (bin_mode (move (pr.in_efd))); vector<char> e (is.read_binary ()); r = in == e && r; } } } catch (const ios_base::failure&) { r = false; } optional<bool> s; return pr.wait () && (s = pr.try_wait ()) && *s && r; } catch (const process_error& e) { //cerr << args[0] << ": " << e << endl; if (e.child) exit (1); return false; } } static bool exec (const path& p, const string& i, bool o = false, bool e = false, bool pipeline = false, dir_path wd = dir_path (), bool env = false) { return exec ( p, vector<char> (i.begin (), i.end ()), o, e, pipeline, false, wd, env); } // Usages: // // argv[0] // argv[0] -a <args> // argv[0] -c [-b] [-e] [<cwd>] // // In the first form run some basic process execution/communication tests. // // In the second form print the arguments to STDOUT one per line. // // In the third form read the data from STDIN and print it to STDOUT and // STDERR. Also check if the working directory argument matches the current // directory, if specified. // // -b // Set binary mode for the standard streams. // // -e // Check some environment variable values. // int main (int argc, const char* argv[]) { using butl::getenv; using butl::optional; bool child (false); bool bin (false); dir_path wd; // Working directory. bool env (false); // Check the environment variables. assert (argc > 0); if (argc > 1 && string (argv[1]) == "-a") { for (int i (2); i != argc; ++i) cout << argv[i] << endl; return 0; } int i (1); for (; i != argc; ++i) { string v (argv[i]); if (v == "-c") child = true; else if (v == "-b") bin = true; else if (v == "-e") env = true; else { if (!wd.empty ()) break; try { wd = dir_path (v); } catch (const invalid_path&) { break; } } } assert (i == argc); path p; try { p = path (argv[0]); } catch (const invalid_path&) { if (child) return 1; assert (false); } if (child) { // Child process. Check if the working directory argument matches the // current directory if specified. Read input data if requested, optionally // write it to cout and/or cerr. // if (!wd.empty () && wd.realize () != dir_path::current_directory ()) return 1; if (env) { // Check that the variables are (un)set as expected. // if (getenv ("PAR1") || getenv ("PAR2") != optional<string> ("2P") || getenv ("PAR3") != optional<string> ("P3") || getenv ("PAR4") || getenv ("PAR5") != optional<string> ("5P") || getenv ("PAR6") != optional<string> ("66") || getenv ("PAR7") || getenv ("THR1") || getenv ("THR2") != optional<string> ("2T") || getenv ("THR3") != optional<string> ("T3") || getenv ("THR4") || getenv ("CHD1") || getenv ("CHD2") != optional<string> ("C2")) return 1; } try { if (bin) { stdin_fdmode (fdstream_mode::binary); stdout_fdmode (fdstream_mode::binary); stderr_fdmode (fdstream_mode::binary); } vector<char> data ((istreambuf_iterator<char> (cin)), istreambuf_iterator<char> ()); cout.exceptions (istream::badbit); copy (data.begin (), data.end (), ostream_iterator<char> (cout)); cerr.exceptions (istream::badbit); copy (data.begin (), data.end (), ostream_iterator<char> (cerr)); } catch (const ios_base::failure&) { return 1; } return 0; } // Here we set the process and thread environment variables to make sure // that the child process will not see the variables that are requested to // be unset, will see change for the variables that are requested to be set, // and will see the other ones unaffected. // setenv ("PAR1", "P1"); setenv ("PAR2", "P2"); setenv ("PAR3", "P3"); setenv ("PAR4", "P4"); setenv ("PAR5", "P5"); setenv ("PAR6", "P6"); setenv ("PAR7", "P7"); const char* tevars[] = { "THR1=T1", "THR2=T2", "THR3=T3", "THR4", "PAR4", "PAR5=5P", "PAR6", "PAR7=7P", // Override the process variables. nullptr}; auto_thread_env ate (tevars); dir_path owd (dir_path::current_directory ()); // Test processes created as "already terminated". // { process p; assert (!p.wait ()); // "Terminated" abnormally. } { // Note that if to create as just process(0) then the // process(const char* args[], int=0, int=1, int=2) ctor is being called. // process p (process_exit (0)); assert (p.wait ()); // "Exited" successfully. } { process p (process_exit (1)); assert (!p.wait ()); // "Exited" with an error. } assert (roundtrip_arg (p, "-DPATH=\"C:\\\\foo\\\\\"")); // -DPATH="C:\\foo\\" assert (roundtrip_arg (p, "C:\\\\f oo\\\\")); assert (roundtrip_arg (p, "C:\\\"f oo\\\\")); assert (roundtrip_arg (p, "C:\\f oo\\")); const char* s ("ABC\nXYZ"); assert (exec (p)); assert (exec (p, s)); assert (exec (p, s, true)); assert (exec (p, s, true, false, true)); // Same but with piping. assert (exec (p, s, false, true)); assert (exec (p, s, true, true)); assert (exec (p, s, true, true, true)); // Same but with piping. // Passing environment variables to the child process. // assert (exec (p, string (), false, false, false, dir_path (), true)); // Transmit large binary data through the child. // vector<char> v; v.reserve (5000 * 256); for (size_t i (0); i < 5000; ++i) { char c (numeric_limits<char>::min ()); do { v.push_back (c); } while (c++ != numeric_limits<char>::max ()); } assert (exec (p, v, true, true)); assert (exec (p, v, true, true, true)); // Same as above but with piping. // Execute the child using the full path. // path fp (p); fp.complete (); assert (exec (fp)); // Execute the child using the relative path. // dir_path::current_directory (fp.directory ()); assert (exec (dir_path (".") / fp.leaf ())); // Fail for non-existent file path. // assert (!exec (dir_path (".") / path ("dr"))); // Execute the child using file name having PATH variable being properly set. // string paths (fp.directory ().string ()); if (optional<string> p = getenv ("PATH")) paths += string (1, path::traits_type::path_separator) + *p; setenv ("PATH", paths); dir_path::current_directory (fp.directory () / dir_path ("..")); assert (exec (fp.leaf ())); // Same as above but also with changing the child current directory. // assert (exec ( fp.leaf (), vector<char> (), false, false, false, true, fp.directory ())); #ifndef _WIN32 // Check that wait() works properly if the underlying low-level wait // operation fails. // process pr; pr.handle = reinterpret_cast<process::handle_type> (-1); assert (!pr.wait (true) && !pr.wait (false)); #endif // Test execution of Windows batch files. The test file is in the original // working directory. // #ifdef _WIN32 { assert (exec (owd / "test.bat")); assert (exec (owd / "test")); paths = owd.string () + path::traits_type::path_separator + paths; setenv ("PATH", paths); assert (exec (path ("test.bat"))); assert (exec (path ("test"))); assert (!exec (path ("testX.bat"))); } #endif // Test printing process_env to stream. // { auto str = [] (const process_env& env) { ostringstream os; os << env; return os.str (); }; process_path p; assert (str (process_env (p)) == ""); { dir_path d ("dir"); dir_path ds ("d ir"); assert (str (process_env (p, d)) == "PWD=dir"); assert (str (process_env (p, ds)) == "PWD=\"d ir\""); } { dir_path ed; // Empty. const char* vars[] = {nullptr}; assert (str (process_env (p, ed, vars)) == ""); } { const char* vars[] = {"A=B", "A=B C", "A B=C", "A", "A B", nullptr}; assert (str (process_env (p, vars)) == "A=B A=\"B C\" \"A B=C\" A= \"A B=\""); } } }