bind_stderr.cpp 3.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155
  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 <system_error>
  13. #include <boost/asio.hpp>
  14. #include <boost/algorithm/string/predicate.hpp>
  15. #include <boost/process/args.hpp>
  16. #include <boost/process/exe.hpp>
  17. #include <boost/process/error.hpp>
  18. #include <boost/process/io.hpp>
  19. #include <boost/process/child.hpp>
  20. #include <boost/process/async_pipe.hpp>
  21. #include <boost/filesystem.hpp>
  22. #include <string>
  23. #include <istream>
  24. #include <iostream>
  25. #include <cstdlib>
  26. #if defined(BOOST_WINDOWS_API)
  27. # include <windows.h>
  28. typedef boost::asio::windows::stream_handle pipe_end;
  29. #elif defined(BOOST_POSIX_API)
  30. # include <sys/wait.h>
  31. # include <unistd.h>
  32. typedef boost::asio::posix::stream_descriptor pipe_end;
  33. #endif
  34. namespace fs = boost::filesystem;
  35. namespace bp = boost::process;
  36. BOOST_AUTO_TEST_SUITE( bind_stderr );
  37. BOOST_AUTO_TEST_CASE(sync_io, *boost::unit_test::timeout(2))
  38. {
  39. using boost::unit_test::framework::master_test_suite;
  40. bp::ipstream is;
  41. std::error_code ec;
  42. bp::child c(
  43. master_test_suite().argv[1],
  44. bp::args={"test", "--echo-stderr", "hello"},
  45. bp::std_err>is,
  46. ec
  47. );
  48. BOOST_REQUIRE(!ec);
  49. std::string s;
  50. is >> s;
  51. BOOST_CHECK_EQUAL(s, "hello");
  52. }
  53. struct read_handler
  54. {
  55. boost::asio::streambuf &buffer_;
  56. read_handler(boost::asio::streambuf &buffer) : buffer_(buffer) {}
  57. void operator()(const boost::system::error_code &ec, std::size_t size)
  58. {
  59. BOOST_REQUIRE(!ec);
  60. std::istream is(&buffer_);
  61. std::string line;
  62. std::getline(is, line);
  63. BOOST_CHECK(boost::algorithm::starts_with(line, "abc"));
  64. }
  65. };
  66. BOOST_AUTO_TEST_CASE(async_io, *boost::unit_test::timeout(2))
  67. {
  68. using boost::unit_test::framework::master_test_suite;
  69. boost::asio::io_context io_context;
  70. bp::async_pipe p(io_context);
  71. std::error_code ec;
  72. bp::child c(
  73. bp::exe=master_test_suite().argv[1],
  74. bp::args+="test",
  75. bp::args+={"--echo-stderr", "abc"},
  76. bp::std_err>p,
  77. ec
  78. );
  79. BOOST_REQUIRE(!ec);
  80. boost::asio::streambuf buffer;
  81. boost::asio::async_read_until(p, buffer, '\n', read_handler(buffer));
  82. io_context.run();
  83. }
  84. BOOST_AUTO_TEST_CASE(nul, *boost::unit_test::timeout(2))
  85. {
  86. using boost::unit_test::framework::master_test_suite;
  87. std::error_code ec;
  88. bp::child c(
  89. bp::exe(master_test_suite().argv[1]),
  90. bp::args+={"test", "--is-nul-stderr"},
  91. bp::std_err>bp::null,
  92. ec
  93. );
  94. BOOST_REQUIRE(!ec);
  95. c.wait();
  96. int exit_code = c.exit_code();
  97. #if defined(BOOST_WINDOWS_API)
  98. BOOST_CHECK_EQUAL(EXIT_SUCCESS, exit_code);
  99. #elif defined(BOOST_POSIX_API)
  100. BOOST_CHECK_EQUAL(EXIT_SUCCESS, exit_code);
  101. #endif
  102. }
  103. BOOST_AUTO_TEST_CASE(file_io, *boost::unit_test::timeout(2))
  104. {
  105. using boost::unit_test::framework::master_test_suite;
  106. fs::path pth =
  107. fs::path(master_test_suite().argv[1]).parent_path() / "std_err_log_file.txt";
  108. std::error_code ec;
  109. bp::child c(
  110. master_test_suite().argv[1],
  111. bp::args={"test", "--echo-stderr", "hello"},
  112. bp::std_err>pth,
  113. ec
  114. );
  115. BOOST_REQUIRE(!ec);
  116. c.wait();
  117. {
  118. fs::ifstream is{pth};
  119. std::string s;
  120. is >> s;
  121. BOOST_CHECK_EQUAL(s, "hello");
  122. }
  123. boost::filesystem::remove(pth);
  124. }
  125. BOOST_AUTO_TEST_SUITE_END();