bind_stdout.cpp 3.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171
  1. // Copyright (c) 2006, 2007 Julio M. Merino Vidal
  2. // Copyright (c) 2008 Ilya Sokolov, Boris Schaeling
  3. // Copyright (c) 2009 Boris Schaeling
  4. // Copyright (c) 2010 Felipe Tanus, Boris Schaeling
  5. // Copyright (c) 2011, 2012 Jeff Flinn, Boris Schaeling
  6. //
  7. // Distributed under the Boost Software License, Version 1.0. (See accompanying
  8. // file LICENSE_1_0.txt or copy at http://www.boost.org/LICENSE_1_0.txt)
  9. #define BOOST_TEST_MAIN
  10. #define BOOST_TEST_IGNORE_SIGCHLD
  11. #include <boost/test/included/unit_test.hpp>
  12. #include <boost/system/error_code.hpp>
  13. #include <boost/asio.hpp>
  14. #include <boost/algorithm/string/predicate.hpp>
  15. #include <boost/process/error.hpp>
  16. #include <boost/process/io.hpp>
  17. #include <boost/process/args.hpp>
  18. #include <boost/process/child.hpp>
  19. #include <boost/process/async_pipe.hpp>
  20. #include <system_error>
  21. #include <boost/filesystem.hpp>
  22. #include <string>
  23. #include <istream>
  24. #include <cstdlib>
  25. #if defined(BOOST_WINDOWS_API)
  26. # include <windows.h>
  27. typedef boost::asio::windows::stream_handle pipe_end;
  28. #elif defined(BOOST_POSIX_API)
  29. # include <sys/wait.h>
  30. # include <unistd.h>
  31. typedef boost::asio::posix::stream_descriptor pipe_end;
  32. #endif
  33. BOOST_AUTO_TEST_SUITE( bind_stdout );
  34. namespace fs = boost::filesystem;
  35. namespace bp = boost::process;
  36. BOOST_AUTO_TEST_CASE(sync_io, *boost::unit_test::timeout(5))
  37. {
  38. using boost::unit_test::framework::master_test_suite;
  39. bp::ipstream is;
  40. std::error_code ec;
  41. bp::child c(
  42. master_test_suite().argv[1],
  43. bp::args+={"test", "--echo-stdout", "hello"},
  44. bp::std_out > is,
  45. ec
  46. );
  47. BOOST_CHECK(!ec);
  48. std::string s;
  49. BOOST_TEST_CHECKPOINT("Starting read");
  50. is >> s;
  51. BOOST_TEST_CHECKPOINT("Finished read");
  52. BOOST_CHECK_EQUAL(s, "hello");
  53. }
  54. struct read_handler
  55. {
  56. boost::asio::streambuf &buffer_;
  57. read_handler(boost::asio::streambuf &buffer) : buffer_(buffer) {}
  58. void operator()(const boost::system::error_code &ec, std::size_t size)
  59. {
  60. BOOST_REQUIRE(!ec);
  61. std::istream is(&buffer_);
  62. std::string line;
  63. std::getline(is, line);
  64. BOOST_CHECK(boost::algorithm::starts_with(line, "abc"));
  65. }
  66. };
  67. BOOST_AUTO_TEST_CASE(async_io, *boost::unit_test::timeout(2))
  68. {
  69. using boost::unit_test::framework::master_test_suite;
  70. boost::asio::io_context io_context;
  71. bp::async_pipe p(io_context);
  72. std::error_code ec;
  73. bp::child c(
  74. master_test_suite().argv[1],
  75. "test", "--echo-stdout", "abc",
  76. bp::std_out > p,
  77. ec
  78. );
  79. BOOST_REQUIRE(!ec);
  80. boost::asio::streambuf buffer;
  81. boost::asio::async_read_until(p, buffer, '\n',
  82. read_handler(buffer));
  83. io_context.run();
  84. }
  85. BOOST_AUTO_TEST_CASE(nul, *boost::unit_test::timeout(2))
  86. {
  87. using boost::unit_test::framework::master_test_suite;
  88. std::error_code ec;
  89. bp::child c(
  90. master_test_suite().argv[1],
  91. bp::args+={"test", "--is-nul-stdout"},
  92. bp::std_out>bp::null,
  93. ec
  94. );
  95. BOOST_REQUIRE(!ec);
  96. c.wait();
  97. int exit_code = c.exit_code();
  98. #if defined(BOOST_WINDOWS_API)
  99. BOOST_CHECK_EQUAL(EXIT_SUCCESS, exit_code);
  100. #elif defined(BOOST_POSIX_API)
  101. BOOST_CHECK_EQUAL(EXIT_SUCCESS, exit_code);
  102. #endif
  103. }
  104. BOOST_AUTO_TEST_CASE(file_io, *boost::unit_test::timeout(2))
  105. {
  106. using boost::unit_test::framework::master_test_suite;
  107. fs::path pth =
  108. fs::path(master_test_suite().argv[1]).parent_path() / "std_out_log_file.txt";
  109. FILE* f = fopen(pth.string().c_str(), "w");
  110. BOOST_REQUIRE(f != nullptr);
  111. std::error_code ec;
  112. bp::child c(
  113. master_test_suite().argv[1],
  114. bp::args={"test", "--echo-stdout", "hello"},
  115. bp::std_out>f,
  116. ec
  117. );
  118. BOOST_REQUIRE(!ec);
  119. fclose(f);
  120. c.wait();
  121. {
  122. fs::ifstream is{pth};
  123. std::string s;
  124. is >> s;
  125. BOOST_CHECK_EQUAL(s, "hello");
  126. }
  127. boost::filesystem::remove(pth);
  128. }
  129. BOOST_AUTO_TEST_SUITE_END();