1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
|
// file : butl/sendmail.ixx -*- C++ -*-
// copyright : Copyright (c) 2014-2017 Code Synthesis Ltd
// license : MIT; see accompanying LICENSE file
#include <utility> // move(), forward()
namespace butl
{
template <typename E, typename... O>
inline sendmail::
sendmail (E&& err,
const std::string& from,
const std::string& subj,
const recipients_type& to,
const recipients_type& cc,
const recipients_type& bcc,
O&&... options)
: sendmail ([] (const char* [], std::size_t) {},
std::forward<E> (err),
from,
subj,
to,
cc,
bcc,
std::forward<O> (options)...)
{
}
template <typename C, typename E, typename... O>
inline sendmail::
sendmail (const C& cmdc,
E&& err,
const std::string& from,
const std::string& subj,
const recipients_type& to,
const recipients_type& cc,
const recipients_type& bcc,
O&&... options)
{
fdpipe pipe (fdopen_pipe ()); // Text mode seems appropriate.
process& p (*this);
p = process_start (cmdc,
pipe.in,
2, // No output expected so redirect to stderr.
std::forward<E> (err),
dir_path (),
"sendmail",
"-i", // Don't treat '.' as the end of input.
"-t", // Read recipients from headers.
std::forward<O> (options)...);
// Close the reading end of the pipe not to block on writing if sendmail
// terminates before that.
//
pipe.in.close ();
out.open (std::move (pipe.out));
// Write headers.
//
// Note that if this throws, then the ofdstream will be closed first
// (which should signal to the process we are done). Then the process's
// destructor will wait for its termination ignoring any errors.
//
headers (from, subj, to, cc, bcc);
}
}
|