19#ifndef REDI_PSTREAM_H_SEEN
20#define REDI_PSTREAM_H_SEEN
36# include <sys/filio.h>
41#if REDI_EVISCERATE_PSTREAMS
47#define PSTREAMS_VERSION 0x0104
68 typedef std::ios_base::openmode
pmode;
89#if __cplusplus >= 201103L
91 using stringable =
decltype((void)std::string(std::declval<const T&>()));
96 template <
typename CharT,
typename Traits = std::
char_traits<CharT> >
98 :
public std::basic_streambuf<CharT, Traits>
103 typedef CharT char_type;
104 typedef Traits traits_type;
105 typedef typename traits_type::int_type int_type;
106 typedef typename traits_type::off_type off_type;
107 typedef typename traits_type::pos_type pos_type;
122#if __cplusplus >= 201103L
167#if REDI_EVISCERATE_PSTREAMS
170 fopen(FILE*& in, FILE*& out, FILE*& err);
200 xsputn(
const char_type* s, std::streamsize n);
204 write(
const char_type* s, std::streamsize n);
208 read(char_type* s, std::streamsize n);
239 create_buffers(
pmode mode);
242 destroy_buffers(
pmode mode);
259#if __cplusplus >= 201103L
260 using basic_streambuf = std::basic_streambuf<char_type, traits_type>;
273 char_type* rbuffer_[2];
274 char_type* rbufstate_[3];
282 template <
typename CharT,
typename Traits = std::
char_traits<CharT> >
284 :
virtual public std::basic_ios<CharT, Traits>
289 typedef std::basic_ios<CharT, Traits> ios_type;
307#if __cplusplus >= 201103L
310 ,
buf_(std::move(rhs.buf_))
319 buf_ = std::move(rhs.buf_);
338 do_open(
const std::string& file,
const argv_type& argv, pmode mode);
358#if REDI_EVISCERATE_PSTREAMS
361 fopen(FILE*& in, FILE*& out, FILE*& err);
380 template <
typename CharT,
typename Traits = std::
char_traits<CharT> >
382 :
public std::basic_istream<CharT, Traits>
386 typedef std::basic_istream<CharT, Traits> istream_type;
401 typedef typename pbase_type::pmode
pmode;
423 : istream_type(NULL),
pbase_type(cmd, readable(mode))
440 : istream_type(NULL),
pbase_type(file, argv, readable(mode))
455 : istream_type(NULL),
pbase_type(argv.at(0), argv, readable(mode))
458#if __cplusplus >= 201103L
459 template<
typename T,
typename =
stringable<T>>
466 : istream_type(std::move(rhs))
467 , pbase_type(std::move(rhs))
473 istream_type::operator=(std::move(rhs));
474 pbase_type::operator=(std::move(rhs));
481 istream_type::swap(rhs);
482 pbase_type::swap(rhs);
506 this->
do_open(cmd, readable(mode));
524 this->
do_open(file, argv, readable(mode));
560 template <
typename CharT,
typename Traits = std::
char_traits<CharT> >
562 :
public std::basic_ostream<CharT, Traits>
566 typedef std::basic_ostream<CharT, Traits> ostream_type;
573 typedef typename pbase_type::pmode
pmode;
630#if __cplusplus >= 201103L
638 template<
typename T,
typename =
stringable<T>>
645 : ostream_type(std::move(rhs))
646 , pbase_type(std::move(rhs))
652 ostream_type::operator=(std::move(rhs));
653 pbase_type::operator=(std::move(rhs));
660 ostream_type::swap(rhs);
661 pbase_type::swap(rhs);
720 template <
typename CharT,
typename Traits = std::
char_traits<CharT> >
722 :
public std::basic_iostream<CharT, Traits>
726 typedef std::basic_iostream<CharT, Traits> iostream_type;
733 typedef typename pbase_type::pmode
pmode;
772 : iostream_type(NULL),
pbase_type(file, argv, mode)
787 : iostream_type(NULL),
pbase_type(argv.at(0), argv, mode)
790#if __cplusplus >= 201103L
798 template<
typename T,
typename =
stringable<T>>
805 : iostream_type(std::move(rhs))
806 , pbase_type(std::move(rhs))
812 iostream_type::operator=(std::move(rhs));
813 pbase_type::operator=(std::move(rhs));
820 iostream_type::swap(rhs);
821 pbase_type::swap(rhs);
862 this->
do_open(file, argv, mode);
910 template <
typename CharT,
typename Traits = std::
char_traits<CharT> >
912 :
public std::basic_ostream<CharT, Traits>
913 ,
private std::basic_istream<CharT, Traits>
917 typedef std::basic_ostream<CharT, Traits> ostream_type;
918 typedef std::basic_istream<CharT, Traits> istream_type;
925 typedef typename pbase_type::pmode
pmode;
932 : ostream_type(NULL), istream_type(NULL),
pbase_type()
947 : ostream_type(NULL) , istream_type(NULL) ,
pbase_type(cmd, mode)
964 : ostream_type(NULL), istream_type(NULL),
pbase_type(file, argv, mode)
979 : ostream_type(NULL), istream_type(NULL)
983#if __cplusplus >= 201103L
991 template<
typename T,
typename =
stringable<T>>
998 : ostream_type(NULL), istream_type(std::move(rhs))
999 , pbase_type(std::move(rhs))
1012 istream_type::swap(rhs);
1013 pbase_type::swap(rhs);
1050 this->
do_open(file, argv, mode);
1103 template <
typename C,
typename T>
1104 inline std::basic_ostream<C,T>&
1108 if (pstreambuf_type* p =
dynamic_cast<pstreambuf_type*
>(s.rdbuf()))
1125 template <
typename C,
typename T>
1137 rpipe_[rsrc_out] = rpipe_[rsrc_err] = -1;
1148 template <
typename C,
typename T>
1160 rpipe_[rsrc_out] = rpipe_[rsrc_err] = -1;
1173 template <
typename C,
typename T>
1187 rpipe_[rsrc_out] = rpipe_[rsrc_err] = -1;
1188 open(file, argv, mode);
1195 template <
typename C,
typename T>
1202#if __cplusplus >= 201103L
1206 template <
typename C,
typename T>
1209 : basic_streambuf(
static_cast<const basic_streambuf&
>(rhs))
1211 , wpipe_(rhs.wpipe_)
1212 , rpipe_{rhs.rpipe_[0], rhs.rpipe_[1]}
1213 , wbuffer_(rhs.wbuffer_)
1214 , rbuffer_{rhs.rbuffer_[0], rhs.rbuffer_[1]}
1215 , rbufstate_{rhs.rbufstate_[0], rhs.rbufstate_[1], rhs.rbufstate_[2]}
1217 , status_(rhs.status_)
1218 , error_(rhs.error_)
1222 rhs.rpipe_[0] = rhs.rpipe_[1] = -1;
1223 rhs.wbuffer_ =
nullptr;
1224 rhs.rbuffer_[0] = rhs.rbuffer_[1] =
nullptr;
1225 rhs.rbufstate_[0] = rhs.rbufstate_[1] = rhs.rbufstate_[2] =
nullptr;
1226 rhs.rsrc_ = rsrc_out;
1229 rhs.setg(
nullptr,
nullptr,
nullptr);
1230 rhs.setp(
nullptr,
nullptr);
1233 template <
typename C,
typename T>
1234 inline basic_pstreambuf<C,T>&
1235 basic_pstreambuf<C,T>::operator=( basic_pstreambuf&& rhs )
noexcept
1238 basic_streambuf::operator=(
static_cast<const basic_streambuf&
>(rhs));
1243 template <
typename C,
typename T>
1245 basic_pstreambuf<C,T>::swap( basic_pstreambuf& rhs )
noexcept
1247 basic_streambuf::swap(
static_cast<basic_streambuf&
>(rhs));
1248 std::swap(ppid_, rhs.ppid_);
1249 std::swap(wpipe_, rhs.wpipe_);
1250 std::swap(rpipe_, rhs.rpipe_);
1251 std::swap(wbuffer_, rhs.wbuffer_);
1252 std::swap(rbuffer_, rhs.rbuffer_);
1253 std::swap(rbufstate_, rhs.rbufstate_);
1254 std::swap(rsrc_, rhs.rsrc_);
1255 std::swap(status_, rhs.status_);
1256 std::swap(error_, rhs.error_);
1287 template <
typename C,
typename T>
1288 basic_pstreambuf<C,T>*
1291 const char * shell_path =
"/bin/sh";
1293 const std::string argv[] = {
"sh",
"-c", command };
1294 return this->open(shell_path,
argv_type(argv, argv+3), mode);
1304 ::execl(shell_path,
"sh",
"-c", command.c_str(), (
char*)NULL);
1319 create_buffers(mode);
1338 if (fd >= 0 && ::close(fd) == 0)
1356 for (std::size_t i = 0; i < N; ++i)
1389 template <
typename C,
typename T>
1403 fd_type ck_exec[] = { -1, -1 };
1404 if (-1 == ::pipe(ck_exec)
1405 || -1 == ::fcntl(ck_exec[RD], F_SETFD, FD_CLOEXEC)
1406 || -1 == ::fcntl(ck_exec[WR], F_SETFD, FD_CLOEXEC))
1409 close_fd_array(ck_exec);
1418 char** arg_v =
new char*[argv.size()+1];
1419 for (std::size_t i = 0; i < argv.size(); ++i)
1421 const std::string& src = argv[i];
1422 char*& dest = arg_v[i];
1423 dest =
new char[src.size()+1];
1424 dest[ src.copy(dest, src.size()) ] =
'\0';
1426 arg_v[argv.size()] = NULL;
1428 ::execvp(file.c_str(), arg_v);
1435 while (::write(ck_exec[WR], &error_,
sizeof(error_)) == -1
1439 ::close(ck_exec[WR]);
1440 ::close(ck_exec[RD]);
1448 close_fd_array(ck_exec);
1455 ::close(ck_exec[WR]);
1456 switch (::read(ck_exec[RD], &error_,
sizeof(error_)))
1460 create_buffers(mode);
1472 ::close(ck_exec[RD]);
1495 template <
typename C,
typename T>
1504 fd_type fd[] = { -1, -1, -1, -1, -1, -1 };
1516 if (!error_ && mode&pstdin && ::pipe(pin))
1519 if (!error_ && mode&pstdout && ::pipe(pout))
1522 if (!error_ && mode&pstderr && ::pipe(perr))
1540 ::dup2(pin[RD], STDIN_FILENO);
1546 ::dup2(pout[WR], STDOUT_FILENO);
1552 ::dup2(perr[WR], STDERR_FILENO);
1556#ifdef _POSIX_JOB_CONTROL
1584 rpipe_[rsrc_out] = pout[RD];
1589 rpipe_[rsrc_err] = perr[RD];
1612 template <
typename C,
typename T>
1616 const bool running = is_open();
1623 destroy_buffers(pstdin|pstdout|pstderr);
1627 close_fd_array(rpipe_);
1632 }
while (wait() == -1 && error() == EINTR);
1634 return running ? this : NULL;
1641 template <
typename C,
typename T>
1642#if __cplusplus >= 201402L && __has_cpp_attribute(deprecated)
1645 __attribute__((deprecated))
1650 rpipe_[rsrc_out] = rpipe_[rsrc_err] = -1;
1651 rbuffer_[rsrc_out] = rbuffer_[rsrc_err] = NULL;
1652 rbufstate_[0] = rbufstate_[1] = rbufstate_[2] = NULL;
1655 template <
typename C,
typename T>
1657 basic_pstreambuf<C,T>::create_buffers(pmode mode)
1662 wbuffer_ =
new char_type[bufsz];
1663 this->setp(wbuffer_, wbuffer_ + bufsz);
1667 delete[] rbuffer_[rsrc_out];
1668 rbuffer_[rsrc_out] =
new char_type[bufsz];
1670 this->setg(rbuffer_[rsrc_out] + pbsz, rbuffer_[rsrc_out] + pbsz,
1671 rbuffer_[rsrc_out] + pbsz);
1675 delete[] rbuffer_[rsrc_err];
1676 rbuffer_[rsrc_err] =
new char_type[bufsz];
1677 if (!(mode & pstdout))
1680 this->setg(rbuffer_[rsrc_err] + pbsz, rbuffer_[rsrc_err] + pbsz,
1681 rbuffer_[rsrc_err] + pbsz);
1686 template <
typename C,
typename T>
1688 basic_pstreambuf<C,T>::destroy_buffers(pmode mode)
1692 this->setp(NULL, NULL);
1698 if (rsrc_ == rsrc_out)
1699 this->setg(NULL, NULL, NULL);
1700 delete[] rbuffer_[rsrc_out];
1701 rbuffer_[rsrc_out] = NULL;
1705 if (rsrc_ == rsrc_err)
1706 this->setg(NULL, NULL, NULL);
1707 delete[] rbuffer_[rsrc_err];
1708 rbuffer_[rsrc_err] = NULL;
1712 template <
typename C,
typename T>
1714 basic_pstreambuf<C,T>::switch_read_buffer(buf_read_src src)
1718 char_type* tmpbufstate[] = {this->eback(), this->gptr(), this->egptr()};
1719 this->setg(rbufstate_[0], rbufstate_[1], rbufstate_[2]);
1720 for (std::size_t i = 0; i < 3; ++i)
1721 rbufstate_[i] = tmpbufstate[i];
1743 template <
typename C,
typename T>
1747 int child_exited = -1;
1751 switch(::waitpid(ppid_, &exit_status, nohang ? WNOHANG : 0))
1763 status_ = exit_status;
1766 destroy_buffers(pstdin);
1773 return child_exited;
1786 template <
typename C,
typename T>
1793 if (::kill(ppid_, signal))
1799 if (signal==SIGTERM || signal==SIGKILL)
1821 template <
typename C,
typename T>
1826#ifdef _POSIX_JOB_CONTROL
1829 pid_t pgid = ::getpgid(ppid_);
1832 else if (pgid == ::getpgrp())
1834 else if (::killpg(pgid, signal))
1852 template <
typename C,
typename T>
1856 return ppid_ == 0 || wait(
true) == 1;
1865 template <
typename C,
typename T>
1875 template <
typename C,
typename T>
1886 template <
typename C,
typename T>
1891 destroy_buffers(pstdin);
1905 template <
typename C,
typename T>
1920 template <
typename C,
typename T>
1927 switch_read_buffer(src);
1943 template <
typename C,
typename T>
1944 typename basic_pstreambuf<C,T>::int_type
1947 if (!empty_buffer())
1948 return traits_type::eof();
1949 else if (!traits_type::eq_int_type(c, traits_type::eof()))
1950 return this->sputc(c);
1952 return traits_type::not_eof(c);
1956 template <
typename C,
typename T>
1960 return !exited() && empty_buffer() ? 0 : -1;
1968 template <
typename C,
typename T>
1972 std::streamsize done = 0;
1975 if (std::streamsize nbuf = this->epptr() - this->pptr())
1977 nbuf = std::min(nbuf, n - done);
1978 traits_type::copy(this->pptr(), s + done, nbuf);
1982 else if (!empty_buffer())
1991 template <
typename C,
typename T>
1995 const std::streamsize count = this->pptr() - this->pbase();
1998 const std::streamsize written = this->write(this->wbuffer_, count);
2001 if (
const std::streamsize unwritten = count - written)
2002 traits_type::move(this->pbase(), this->pbase()+written, unwritten);
2003 this->pbump(-written);
2017 template <
typename C,
typename T>
2018 typename basic_pstreambuf<C,T>::int_type
2021 if (this->gptr() < this->egptr() || fill_buffer())
2022 return traits_type::to_int_type(*this->gptr());
2024 return traits_type::eof();
2035 template <
typename C,
typename T>
2036 typename basic_pstreambuf<C,T>::int_type
2039 if (this->gptr() != this->eback())
2042 if (!traits_type::eq_int_type(c, traits_type::eof()))
2043 *this->gptr() = traits_type::to_char_type(c);
2044 return traits_type::not_eof(c);
2047 return traits_type::eof();
2050 template <
typename C,
typename T>
2055 if (
sizeof(char_type) == 1)
2056 avail = fill_buffer(
true) ? this->egptr() - this->gptr() : -1;
2060 if (::ioctl(rpipe(), FIONREAD, &avail) == -1)
2063 avail /=
sizeof(char_type);
2066 return std::streamsize(avail);
2072 template <
typename C,
typename T>
2076 const std::streamsize pb1 = this->gptr() - this->eback();
2077 const std::streamsize pb2 = pbsz;
2078 const std::streamsize npb = std::min(pb1, pb2);
2080 char_type*
const rbuf = rbuffer();
2083 traits_type::move(rbuf + pbsz - npb, this->gptr() - npb, npb);
2085 std::streamsize rc = -1;
2089 const int flags = ::fcntl(rpipe(), F_GETFL);
2092 const bool blocking = !(flags & O_NONBLOCK);
2094 ::fcntl(rpipe(), F_SETFL, flags | O_NONBLOCK);
2097 rc = read(rbuf + pbsz, bufsz - pbsz);
2099 if (rc == -1 && error_ == EAGAIN)
2105 ::fcntl(rpipe(), F_SETFL, flags);
2109 rc = read(rbuf + pbsz, bufsz - pbsz);
2111 if (rc > 0 || (rc == 0 && non_blocking))
2113 this->setg( rbuf + pbsz - npb,
2120 this->setg(NULL, NULL, NULL);
2132 template <
typename C,
typename T>
2133 inline std::streamsize
2136 std::streamsize nwritten = 0;
2139 nwritten = ::write(wpipe(), s, n *
sizeof(char_type));
2143 nwritten /=
sizeof(char_type);
2155 template <
typename C,
typename T>
2156 inline std::streamsize
2159 std::streamsize nread = 0;
2162 nread = ::read(rpipe(), s, n *
sizeof(char_type));
2166 nread /=
sizeof(char_type);
2172 template <
typename C,
typename T>
2180 template <
typename C,
typename T>
2184 return rpipe_[rsrc_];
2188 template <
typename C,
typename T>
2192 return rpipe_[which];
2196 template <
typename C,
typename T>
2197 inline typename basic_pstreambuf<C,T>::char_type*
2200 return rbuffer_[rsrc_];
2217 template <
typename C,
typename T>
2220 : std::basic_ios<C,T>(NULL)
2224 this->std::basic_ios<C,T>::rdbuf(&
buf_);
2235 template <
typename C,
typename T>
2238 : std::basic_ios<C,T>(NULL)
2242 this->std::basic_ios<C,T>::rdbuf(&
buf_);
2255 template <
typename C,
typename T>
2258 const argv_type& argv,
2260 : std::basic_ios<C,T>(NULL)
2264 this->std::basic_ios<C,T>::rdbuf(&
buf_);
2277 template <
typename C,
typename T>
2291 template <
typename C,
typename T>
2295 if (!buf_.open((command_=cmd), mode))
2296 this->setstate(std::ios_base::failbit);
2308 template <
typename C,
typename T>
2311 const argv_type& argv,
2314 if (!buf_.open((command_=file), argv, mode))
2315 this->setstate(std::ios_base::failbit);
2320 template <
typename C,
typename T>
2325 this->setstate(std::ios_base::failbit);
2326 return buf_.status();
2333 template <
typename C,
typename T>
2337 return buf_.is_open();
2341 template <
typename C,
typename T>
2342 inline const std::string&
2350 template <
typename C,
typename T>
2358#if REDI_EVISCERATE_PSTREAMS
2391 template <
typename C,
typename T>
2395 in = out = err = NULL;
2396 std::size_t open_files = 0;
2399 if ((in = ::fdopen(wpipe(),
"w")))
2401 open_files |= pstdin;
2404 if (rpipe(rsrc_out) > -1)
2406 if ((out = ::fdopen(rpipe(rsrc_out),
"r")))
2408 open_files |= pstdout;
2411 if (rpipe(rsrc_err) > -1)
2413 if ((err = ::fdopen(rpipe(rsrc_err),
"r")))
2415 open_files |= pstderr;
2431 template <
typename C,
typename T>
2433 pstream_common<C,T>::fopen(FILE*& fin, FILE*& fout, FILE*& ferr)
2435 return buf_.fopen(fin, fout, ferr);
Class template for Input PStreams.
Definition pstream.h:385
basic_ipstream(const std::string &file, const argv_type &argv, pmode mode=pstdout)
Constructor that initialises the stream by starting a process.
Definition pstream.h:437
basic_ipstream(const argv_type &argv, pmode mode=pstdout)
Constructor that initialises the stream by starting a process.
Definition pstream.h:454
basic_ipstream()
Default constructor, creates an uninitialised stream.
Definition pstream.h:407
~basic_ipstream()
Destructor.
Definition pstream.h:491
void open(const std::string &file, const argv_type &argv, pmode mode=pstdout)
Start a process.
Definition pstream.h:520
pbase_type::argv_type argv_type
Type used to hold the arguments for a command.
Definition pstream.h:404
basic_ipstream & err()
Set streambuf to read from process' stderr.
Definition pstream.h:543
void open(const std::string &cmd, pmode mode=pstdout)
Start a process.
Definition pstream.h:504
basic_ipstream & out()
Set streambuf to read from process' stdout.
Definition pstream.h:532
basic_ipstream(const std::string &cmd, pmode mode=pstdout)
Constructor that initialises the stream by starting a process.
Definition pstream.h:422
pbase_type::pmode pmode
Type used to specify how to connect to the process.
Definition pstream.h:401
Class template for Output PStreams.
Definition pstream.h:565
void open(const std::string &file, const argv_type &argv, pmode mode=pstdin)
Start a process.
Definition pstream.h:698
basic_opstream(const argv_type &argv, pmode mode=pstdin)
Constructor that initialises the stream by starting a process.
Definition pstream.h:626
basic_opstream(const std::string &file, const argv_type &argv, pmode mode=pstdin)
Constructor that initialises the stream by starting a process.
Definition pstream.h:609
basic_opstream()
Default constructor, creates an uninitialised stream.
Definition pstream.h:579
pbase_type::pmode pmode
Type used to specify how to connect to the process.
Definition pstream.h:573
pbase_type::argv_type argv_type
Type used to hold the arguments for a command.
Definition pstream.h:576
~basic_opstream()
Destructor.
Definition pstream.h:670
basic_opstream(const std::string &cmd, pmode mode=pstdin)
Constructor that initialises the stream by starting a process.
Definition pstream.h:594
void open(const std::string &cmd, pmode mode=pstdin)
Start a process.
Definition pstream.h:682
std::basic_ostream< C, T > & peof(std::basic_ostream< C, T > &s)
Manipulator to close the pipe connected to the process' stdin.
Definition pstream.h:1105
Class template for Bidirectional PStreams.
Definition pstream.h:725
basic_pstream(const std::string &cmd, pmode mode=pstdout|pstdin)
Constructor that initialises the stream by starting a process.
Definition pstream.h:754
void open(const std::string &cmd, pmode mode=pstdout|pstdin)
Start a process.
Definition pstream.h:842
basic_pstream()
Default constructor, creates an uninitialised stream.
Definition pstream.h:739
~basic_pstream()
Destructor.
Definition pstream.h:830
basic_pstream(const argv_type &argv, pmode mode=pstdout|pstdin)
Constructor that initialises the stream by starting a process.
Definition pstream.h:786
void open(const std::string &file, const argv_type &argv, pmode mode=pstdout|pstdin)
Start a process.
Definition pstream.h:858
basic_pstream & out()
Set streambuf to read from process' stdout.
Definition pstream.h:870
basic_pstream(const std::string &file, const argv_type &argv, pmode mode=pstdout|pstdin)
Constructor that initialises the stream by starting a process.
Definition pstream.h:769
pbase_type::argv_type argv_type
Type used to hold the arguments for a command.
Definition pstream.h:736
basic_pstream & err()
Set streambuf to read from process' stderr.
Definition pstream.h:881
pbase_type::pmode pmode
Type used to specify how to connect to the process.
Definition pstream.h:733
Class template for stream buffer.
Definition pstream.h:100
fd_type & rpipe(buf_read_src which)
Return the file descriptor for the specified input pipe.
Definition pstream.h:2190
basic_pstreambuf * killpg(int signal=SIGTERM)
Send a signal to the process' process group.
Definition pstream.h:1823
void close_fd(pstreams::fd_type &fd)
Helper function to close a file descriptor.
Definition pstream.h:1336
int status() const
Return the exit status of the process.
Definition pstream.h:1867
basic_pstreambuf * open(const std::string &file, const argv_type &argv, pmode mode)
Initialise the stream buffer with file and argv.
Definition pstream.h:1391
std::streamsize xsputn(const char_type *s, std::streamsize n)
Insert multiple characters into the pipe.
Definition pstream.h:1970
std::streamsize read(char_type *s, std::streamsize n)
Extract a sequence of characters from the pipe.
Definition pstream.h:2157
std::streamsize write(const char_type *s, std::streamsize n)
Insert a sequence of characters into the pipe.
Definition pstream.h:2134
~basic_pstreambuf()
Destructor.
Definition pstream.h:1197
int_type overflow(int_type c)
Transfer characters to the pipe when character buffer overflows.
Definition pstream.h:1945
int sync()
Write any buffered characters to the stream.
Definition pstream.h:1958
basic_pstreambuf * kill(int signal=SIGTERM)
Send a signal to the process.
Definition pstream.h:1788
void peof()
Close the pipe connected to the process' stdin.
Definition pstream.h:1888
pid_t fork(pmode mode)
Initialise pipes and fork process.
Definition pstream.h:1497
basic_pstreambuf * close()
Close the stream buffer and wait for the process to exit.
Definition pstream.h:1614
basic_pstreambuf(const std::string &file, const argv_type &argv, pmode mode)
Constructor that initialises the buffer with file and argv.
Definition pstream.h:1175
int error() const
Return the error number (errno) for the most recent failed operation.
Definition pstream.h:1877
buf_read_src
Enumerated type to indicate whether stdout or stderr is to be read.
Definition pstream.h:216
void close_fd_array(pstreams::fd_type(&fds)[N])
Helper function to close an array of file descriptors.
Definition pstream.h:1354
fd_type & wpipe()
Return the file descriptor for the output pipe.
Definition pstream.h:2174
bool exited()
Report whether the process has exited.
Definition pstream.h:1854
bool empty_buffer()
Writes buffered characters to the process' stdin pipe.
Definition pstream.h:1993
fd_type fd_t
Definition pstream.h:109
char_type * rbuffer()
Return the active input buffer.
Definition pstream.h:2198
std::streamsize showmanyc()
Report how many characters can be read from active input without blocking.
Definition pstream.h:2052
bool read_err(bool readerr=true)
Change active input source.
Definition pstream.h:1922
bool fill_buffer(bool non_blocking=false)
Definition pstream.h:2074
basic_pstreambuf * open(const std::string &cmd, pmode mode)
Initialise the stream buffer with cmd.
Definition pstream.h:1289
int_type pbackfail(int_type c=traits_type::eof())
Make a character available to be returned by the next extraction.
Definition pstream.h:2037
fd_type & rpipe()
Return the file descriptor for the active input pipe.
Definition pstream.h:2182
basic_pstreambuf()
Default constructor.
Definition pstream.h:1127
bool is_open() const
Report whether the stream buffer has been initialised.
Definition pstream.h:1907
int_type underflow()
Transfer characters from the pipe when the character buffer is empty.
Definition pstream.h:2019
int wait(bool nohang=false)
Wait for the child process to exit.
Definition pstream.h:1745
basic_pstreambuf(const std::string &cmd, pmode mode)
Constructor that initialises the buffer with cmd.
Definition pstream.h:1150
Class template for Restricted PStreams.
Definition pstream.h:916
istream_type & err()
Obtain a reference to the istream that reads the process' stderr.
Definition pstream.h:1071
~basic_rpstream()
Destructor.
Definition pstream.h:1018
basic_rpstream()
Default constructor, creates an uninitialised stream.
Definition pstream.h:931
basic_rpstream(const std::string &cmd, pmode mode=pstdout|pstdin)
Constructor that initialises the stream by starting a process.
Definition pstream.h:946
void open(const std::string &file, const argv_type &argv, pmode mode=pstdout|pstdin)
Start a process.
Definition pstream.h:1046
void open(const std::string &cmd, pmode mode=pstdout|pstdin)
Start a process.
Definition pstream.h:1030
pbase_type::argv_type argv_type
Type used to hold the arguments for a command.
Definition pstream.h:928
pbase_type::pmode pmode
Type used to specify how to connect to the process.
Definition pstream.h:925
basic_rpstream(const argv_type &argv, pmode mode=pstdout|pstdin)
Constructor that initialises the stream by starting a process.
Definition pstream.h:978
basic_rpstream(const std::string &file, const argv_type &argv, pmode mode=pstdout|pstdin)
Constructor that initialises the stream by starting a process.
Definition pstream.h:961
istream_type & out()
Obtain a reference to the istream that reads the process' stdout.
Definition pstream.h:1059
Class template for common base class.
Definition pstream.h:286
pstream_common()
Default constructor.
Definition pstream.h:2219
void do_open(const std::string &cmd, pmode mode)
Start a process.
Definition pstream.h:2293
void do_open(const std::string &file, const argv_type &argv, pmode mode)
Start a process.
Definition pstream.h:2310
pstream_common(const std::string &file, const argv_type &argv, pmode mode)
Constructor that initialises the stream by starting a process.
Definition pstream.h:2257
pstream_common(const std::string &cmd, pmode mode)
Constructor that initialises the stream by starting a process.
Definition pstream.h:2237
streambuf_type * rdbuf() const
Return a pointer to the stream buffer.
Definition pstream.h:2352
streambuf_type buf_
The stream buffer.
Definition pstream.h:366
int close()
Definition pstream.h:2322
virtual ~pstream_common()=0
Pure virtual destructor.
Definition pstream.h:2279
bool is_open() const
Report whether the stream's buffer has been initialised.
Definition pstream.h:2335
std::string command_
The command used to start the process.
Definition pstream.h:365
const std::string & command() const
Return the command used to initialise the stream.
Definition pstream.h:2343
All PStreams classes are declared in namespace redi.
basic_pstreambuf< char > pstreambuf
Type definition for common template specialisation.
Definition pstream.h:1080
basic_ipstream< char > ipstream
Type definition for common template specialisation.
Definition pstream.h:1082
basic_rpstream< char > rpstream
Type definition for common template specialisation.
Definition pstream.h:1088
basic_opstream< char > opstream
Type definition for common template specialisation.
Definition pstream.h:1084
basic_pstream< char > pstream
Type definition for common template specialisation.
Definition pstream.h:1086
Common base class providing constants and typenames.
Definition pstream.h:66
std::ios_base::openmode pmode
Type used to specify how to connect to the process.
Definition pstream.h:68
static const pmode pstderr
Read from stderr.
Definition pstream.h:78
static const pmode pstdin
Write to stdin.
Definition pstream.h:76
static const pmode newpg
Create a new process group for the child process.
Definition pstream.h:81
@ bufsz
Size of pstreambuf buffers.
Definition pstream.h:85
@ pbsz
Number of putback characters kept.
Definition pstream.h:86
static const pmode pstdout
Read from stdout.
Definition pstream.h:77
int fd_type
Type used for file descriptors.
Definition pstream.h:74
std::vector< std::string > argv_type
Type used to hold the arguments for a command.
Definition pstream.h:71