NeoMutt  2025-12-11-911-gd8d604
Teaching an old dog new tricks
DOXYGEN
Loading...
Searching...
No Matches
sendmail.c
Go to the documentation of this file.
1
25
31
32#include "config.h"
33#include <errno.h>
34#include <fcntl.h>
35#include <limits.h>
36#include <regex.h>
37#include <signal.h>
38#include <stdbool.h>
39#include <string.h>
40#include <sys/stat.h>
41#include <sys/types.h>
42#include <sys/wait.h>
43#include <unistd.h>
44#include "mutt/lib.h"
45#include "address/lib.h"
46#include "config/lib.h"
47#include "core/lib.h"
48#include "gui/lib.h"
49#include "sendmail.h"
50#include "expando/lib.h"
51#include "nntp/lib.h"
52#include "pager/lib.h"
53#include "globals.h"
54#include "module_data.h"
55#ifdef HAVE_SYSEXITS_H
56#include <sysexits.h>
57#else
58#define EX_OK 0
59#endif
60
61/* For execvp environment setting in send_msg() */
62#ifndef __USE_GNU
63extern char **environ;
64#endif
65
70static void alarm_handler(int sig)
71{
73 mod_data->sig_alrm = 1;
74}
75
89static int send_msg(const char *path, struct StringArray *args, const char *msg,
90 char **tempfile, int wait_time)
91{
92 sigset_t set;
93 int st;
95
97
98 sigemptyset(&set);
99 /* we also don't want to be stopped right now */
100 sigaddset(&set, SIGTSTP);
101 sigprocmask(SIG_BLOCK, &set, NULL);
102
103 if ((wait_time >= 0) && tempfile)
104 {
105 struct Buffer *tmp = buf_pool_get();
106 buf_mktemp(tmp);
107 *tempfile = buf_strdup(tmp);
108 buf_pool_release(&tmp);
109 }
110
111 pid_t pid = fork();
112 if (pid == 0)
113 {
114 struct sigaction act = { 0 };
115 struct sigaction oldalrm = { 0 };
116
117 /* save parent's ID before setsid() */
118 pid_t ppid = getppid();
119
120 /* we want the delivery to continue even after the main process dies,
121 * so we put ourselves into another session right away */
122 setsid();
123
124 /* next we close all open files */
125 close(0);
126#ifdef OPEN_MAX
127 for (int fd = tempfile ? 1 : 3; fd < OPEN_MAX; fd++)
128 close(fd);
129#elif defined(_POSIX_OPEN_MAX)
130 for (int fd = tempfile ? 1 : 3; fd < _POSIX_OPEN_MAX; fd++)
131 close(fd);
132#else
133 if (tempfile)
134 {
135 close(1);
136 close(2);
137 }
138#endif
139
140 /* now the second fork() */
141 pid = fork();
142 if (pid == 0)
143 {
144 /* "msg" will be opened as stdin */
145 if (open(msg, O_RDONLY, 0) < 0)
146 {
147 unlink(msg);
148 _exit(S_ERR);
149 }
150 unlink(msg);
151
152 if ((wait_time >= 0) && tempfile && *tempfile)
153 {
154 /* *tempfile will be opened as stdout */
155 if (open(*tempfile, O_WRONLY | O_APPEND | O_CREAT | O_EXCL, 0600) < 0)
156 _exit(S_ERR);
157 /* redirect stderr to *tempfile too */
158 if (dup(1) < 0)
159 _exit(S_ERR);
160 }
161 else if (tempfile)
162 {
163 if (open("/dev/null", O_WRONLY | O_APPEND) < 0) /* stdout */
164 _exit(S_ERR);
165 if (open("/dev/null", O_RDWR | O_APPEND) < 0) /* stderr */
166 _exit(S_ERR);
167 }
168
170
171 /* execvpe is a glibc extension, so just manually set environ */
173 execvp(path, (char **) args->entries);
174 _exit(S_ERR);
175 }
176 else if (pid == -1)
177 {
178 unlink(msg);
179 FREE(tempfile);
180 _exit(S_ERR);
181 }
182
183 /* wait_time > 0: interrupt waitpid() after wait_time seconds
184 * wait_time = 0: wait forever
185 * wait_time < 0: don't wait */
186 if (wait_time > 0)
187 {
188 mod_data->sig_alrm = 0;
189 act.sa_handler = alarm_handler;
190#ifdef SA_INTERRUPT
191 /* need to make sure waitpid() is interrupted on SIGALRM */
192 act.sa_flags = SA_INTERRUPT;
193#else
194 act.sa_flags = 0;
195#endif
196 sigemptyset(&act.sa_mask);
197 sigaction(SIGALRM, &act, &oldalrm);
198 alarm(wait_time);
199 }
200 else if (wait_time < 0)
201 {
202 _exit(0xff & EX_OK);
203 }
204
205 if (waitpid(pid, &st, 0) > 0)
206 {
207 st = WIFEXITED(st) ? WEXITSTATUS(st) : S_ERR;
208 if (wait_time && (st == (0xff & EX_OK)) && tempfile && *tempfile)
209 {
210 unlink(*tempfile); /* no longer needed */
211 FREE(tempfile);
212 }
213 }
214 else
215 {
216 st = ((wait_time > 0) && (errno == EINTR) && mod_data->sig_alrm) ? S_BKG : S_ERR;
217 if ((wait_time > 0) && tempfile && *tempfile)
218 {
219 unlink(*tempfile);
220 FREE(tempfile);
221 }
222 }
223
224 if (wait_time > 0)
225 {
226 /* reset alarm; not really needed, but... */
227 alarm(0);
228 sigaction(SIGALRM, &oldalrm, NULL);
229 }
230
231 if ((kill(ppid, 0) == -1) && (errno == ESRCH) && tempfile && *tempfile)
232 {
233 /* the parent is already dead */
234 unlink(*tempfile);
235 FREE(tempfile);
236 }
237
238 _exit(st);
239 }
240
241 sigprocmask(SIG_UNBLOCK, &set, NULL);
242
243 if ((pid != -1) && (waitpid(pid, &st, 0) > 0))
244 st = WIFEXITED(st) ? WEXITSTATUS(st) : S_ERR; /* return child status */
245 else
246 st = S_ERR; /* error */
247
249
250 return st;
251}
252
258static void add_args_one(struct StringArray *args, const struct Address *addr)
259{
260 /* weed out group mailboxes, since those are for display only */
261 if (addr->mailbox && !addr->group)
262 {
263 ARRAY_ADD(args, buf_string(addr->mailbox));
264 }
265}
266
272static void add_args(struct StringArray *args, struct AddressList *al)
273{
274 if (!al)
275 return;
276
277 struct Address *a = NULL;
278 TAILQ_FOREACH(a, al, entries)
279 {
280 add_args_one(args, a);
281 }
282}
283
299int mutt_invoke_sendmail(struct Mailbox *m, struct AddressList *from,
300 struct AddressList *to, struct AddressList *cc,
301 struct AddressList *bcc, const char *msg,
302 bool eightbit, struct ConfigSubset *sub)
303{
304 char *ps = NULL, *path = NULL, *s = NULL, *childout = NULL;
305 struct StringArray args = ARRAY_HEAD_INITIALIZER;
306 struct StringArray extra_args = ARRAY_HEAD_INITIALIZER;
307 int i;
308
309 if (OptNewsSend)
310 {
311 struct Buffer *cmd = buf_pool_get();
312
313 const struct Expando *c_inews_command = cs_subset_expando(sub, "inews_command");
315 cmd->dsize, NeoMutt->env, cmd);
316 if (buf_is_empty(cmd))
317 {
318 i = nntp_post(m, msg);
319 unlink(msg);
320 buf_pool_release(&cmd);
321 return i;
322 }
323
324 s = buf_strdup(cmd);
325 buf_pool_release(&cmd);
326 }
327 else
328 {
329 const char *const c_sendmail = cs_subset_string(sub, "sendmail");
330 s = mutt_str_dup(c_sendmail);
331 }
332
333 if (!s)
334 {
335 mutt_error(_("$sendmail must be set in order to send mail"));
336 return -1;
337 }
338
339 ps = s;
340 i = 0;
341 while ((ps = strtok(ps, " ")))
342 {
343 if (i)
344 {
345 if (mutt_str_equal(ps, "--"))
346 break;
347 ARRAY_ADD(&args, ps);
348 }
349 else
350 {
351 path = mutt_str_dup(ps);
352 ps = strrchr(ps, '/');
353 if (ps)
354 ps++;
355 else
356 ps = path;
357 ARRAY_ADD(&args, ps);
358 }
359 ps = NULL;
360 i++;
361 }
362
363 if (!OptNewsSend)
364 {
365 /* If $sendmail contained a "--", we save the recipients to append to
366 * args after other possible options added below. */
367 if (ps)
368 {
369 ps = NULL;
370 while ((ps = strtok(ps, " ")))
371 {
372 ARRAY_ADD(&extra_args, ps);
373 ps = NULL;
374 }
375 }
376
377 const bool c_use_8bit_mime = cs_subset_bool(sub, "use_8bit_mime");
378 if (eightbit && c_use_8bit_mime)
379 ARRAY_ADD(&args, "-B8BITMIME");
380
381 const bool c_use_envelope_from = cs_subset_bool(sub, "use_envelope_from");
382 if (c_use_envelope_from)
383 {
384 const struct Address *c_envelope_from_address = cs_subset_address(sub, "envelope_from_address");
385 if (c_envelope_from_address)
386 {
387 ARRAY_ADD(&args, "-f");
388 add_args_one(&args, c_envelope_from_address);
389 }
390 else if (!TAILQ_EMPTY(from) && !TAILQ_NEXT(TAILQ_FIRST(from), entries))
391 {
392 ARRAY_ADD(&args, "-f");
393 add_args(&args, from);
394 }
395 }
396
397 const char *const c_dsn_notify = cs_subset_string(sub, "dsn_notify");
398 if (c_dsn_notify)
399 {
400 ARRAY_ADD(&args, "-N");
401 ARRAY_ADD(&args, c_dsn_notify);
402 }
403
404 const char *const c_dsn_return = cs_subset_string(sub, "dsn_return");
405 if (c_dsn_return)
406 {
407 ARRAY_ADD(&args, "-R");
408 ARRAY_ADD(&args, c_dsn_return);
409 }
410 ARRAY_ADD(&args, "--");
411 const char **e = NULL;
412 ARRAY_FOREACH(e, &extra_args)
413 {
414 ARRAY_ADD(&args, *e);
415 }
416 add_args(&args, to);
417 add_args(&args, cc);
418 add_args(&args, bcc);
419 }
420
421 ARRAY_ADD(&args, NULL);
422
423 const short c_sendmail_wait = cs_subset_number(sub, "sendmail_wait");
424 i = send_msg(path, &args, msg, OptGui ? &childout : NULL, c_sendmail_wait);
425
426 /* Some user's $sendmail command uses gpg for password decryption,
427 * and is set up to prompt using ncurses pinentry. If we
428 * mutt_endwin() it leaves other users staring at a blank screen.
429 * So instead, just force a hard redraw on the next refresh. */
430 if (OptGui)
431 {
433 }
434
435 if (i != (EX_OK & 0xff))
436 {
437 if (i != S_BKG)
438 {
439 const char *e = mutt_str_sysexit(i);
440 mutt_error(_("Error sending message, child exited %d (%s)"), i, NONULL(e));
441 if (childout)
442 {
443 struct stat st = { 0 };
444
445 if ((stat(childout, &st) == 0) && (st.st_size > 0))
446 {
447 struct PagerData pdata = { 0 };
448 struct PagerView pview = { &pdata };
449
450 pdata.fname = childout;
451
452 pview.banner = _("Output of the delivery process");
453 pview.flags = MUTT_PAGER_NONE;
454 pview.mode = PAGER_MODE_OTHER;
455
456 mutt_do_pager(&pview, NULL);
457 }
458 }
459 }
460 }
461 else if (childout)
462 {
463 unlink(childout);
464 }
465
466 FREE(&childout);
467 FREE(&path);
468 FREE(&s);
469 ARRAY_FREE(&args);
470 ARRAY_FREE(&extra_args);
471
472 if (i == (EX_OK & 0xff))
473 i = 0;
474 else if (i == S_BKG)
475 i = 1;
476 else
477 i = -1;
478 return i;
479}
const struct Address * cs_subset_address(const struct ConfigSubset *sub, const char *name)
Get an Address config item by name.
Email Address Handling.
#define ARRAY_ADD(head, elem)
Add an element at the end of the array.
Definition array.h:157
#define ARRAY_FOREACH(elem, head)
Iterate over all elements of the array.
Definition array.h:223
#define ARRAY_FREE(head)
Release all memory.
Definition array.h:209
#define ARRAY_HEAD_INITIALIZER
Static initializer for arrays.
Definition array.h:58
bool buf_is_empty(const struct Buffer *buf)
Is the Buffer empty?
Definition buffer.c:291
char * buf_strdup(const struct Buffer *buf)
Copy a Buffer's string.
Definition buffer.c:571
static const char * buf_string(const struct Buffer *buf)
Convert a buffer to a const char * "string".
Definition buffer.h:96
const char * cs_subset_string(const struct ConfigSubset *sub, const char *name)
Get a string config item by name.
Definition helpers.c:291
short cs_subset_number(const struct ConfigSubset *sub, const char *name)
Get a number config item by name.
Definition helpers.c:143
bool cs_subset_bool(const struct ConfigSubset *sub, const char *name)
Get a boolean config item by name.
Definition helpers.c:47
const struct Expando * cs_subset_expando(const struct ConfigSubset *sub, const char *name)
Get an Expando config item by name.
Convenience wrapper for the config headers.
Convenience wrapper for the core headers.
void mutt_need_hard_redraw(void)
Force a hard refresh.
Definition curs_lib.c:102
int mutt_do_pager(struct PagerView *pview, struct Email *e)
Display some page-able text to the user (help or attachment)
Definition do_pager.c:122
int expando_filter(const struct Expando *exp, const struct ExpandoRenderCallback *erc, void *data, MuttFormatFlags flags, int max_cols, char **env_list, struct Buffer *buf)
Render an Expando and run the result through a filter.
Definition filter.c:139
Parse Expando string.
const struct ExpandoRenderCallback NntpRenderCallbacks[]
Callbacks for Newsrc Expandos.
bool OptGui
(pseudo) when the gui (and curses) are started
Definition globals.c:48
bool OptNewsSend
(pseudo) used to change behavior when posting
Definition globals.c:54
Global variables.
#define mutt_error(...)
Definition logging2.h:94
Convenience wrapper for the gui headers.
#define FREE(x)
Free memory and set the pointer to NULL.
Definition memory.h:68
@ MODULE_ID_SEND
ModuleSend, Send
Definition module_api.h:90
Convenience wrapper for the library headers.
#define _(a)
Definition message.h:28
char * mutt_str_dup(const char *str)
Copy a string, safely.
Definition string.c:257
bool mutt_str_equal(const char *a, const char *b)
Compare two strings.
Definition string.c:665
const char * mutt_str_sysexit(int err_num)
Return a string matching an error code.
Definition string.c:173
void * neomutt_get_module_data(struct NeoMutt *n, enum ModuleId id)
Get the private data for a Module.
Definition neomutt.c:663
Usenet network mailbox type; talk to an NNTP server.
int nntp_post(struct Mailbox *m, const char *msg)
Post article.
Definition nntp.c:1945
GUI display a file/email/help in a viewport with paging.
#define MUTT_PAGER_NONE
No flags are set.
Definition lib.h:63
@ PAGER_MODE_OTHER
Pager is invoked via 3rd path. Non-email content is likely to be shown.
Definition lib.h:143
struct Buffer * buf_pool_get(void)
Get a Buffer from the pool.
Definition pool.c:91
void buf_pool_release(struct Buffer **ptr)
Return a Buffer to the pool.
Definition pool.c:111
#define TAILQ_FOREACH(var, head, field)
Definition queue.h:782
#define TAILQ_FIRST(head)
Definition queue.h:780
#define TAILQ_NEXT(elm, field)
Definition queue.h:889
#define TAILQ_EMPTY(head)
Definition queue.h:778
@ MUTT_FORMAT_NONE
No flags are set.
Definition render.h:37
Send private Module data.
static void alarm_handler(int sig)
Async notification of an alarm signal.
Definition sendmail.c:70
static int send_msg(const char *path, struct StringArray *args, const char *msg, char **tempfile, int wait_time)
Invoke sendmail in a subshell.
Definition sendmail.c:89
static void add_args(struct StringArray *args, struct AddressList *al)
Add a list of Addresses to a dynamic array.
Definition sendmail.c:272
int mutt_invoke_sendmail(struct Mailbox *m, struct AddressList *from, struct AddressList *to, struct AddressList *cc, struct AddressList *bcc, const char *msg, bool eightbit, struct ConfigSubset *sub)
Run sendmail.
Definition sendmail.c:299
static void add_args_one(struct StringArray *args, const struct Address *addr)
Add an Address to a dynamic array.
Definition sendmail.c:258
char ** environ
#define EX_OK
Definition sendmail.c:58
Send email using sendmail.
void mutt_sig_reset_child_signals(void)
Reset ignored signals back to the default.
Definition signal.c:336
void mutt_sig_block_system(void)
Block signals before calling exec()
Definition signal.c:260
void mutt_sig_unblock_system(bool restore)
Restore previously blocked signals.
Definition signal.c:284
#define S_ERR
Definition string2.h:47
#define NONULL(x)
Definition string2.h:44
#define S_BKG
Definition string2.h:48
An email address.
Definition address.h:35
bool group
Group mailbox?
Definition address.h:38
struct Buffer * mailbox
Mailbox and host address.
Definition address.h:37
String manipulation buffer.
Definition buffer.h:36
size_t dsize
Length of data.
Definition buffer.h:39
A set of inherited config items.
Definition subset.h:46
Parsed Expando trees.
Definition expando.h:41
A mailbox.
Definition mailbox.h:81
Container for Accounts, Notifications.
Definition neomutt.h:41
char ** env
Private copy of the environment variables.
Definition neomutt.h:57
Data to be displayed by PagerView.
Definition lib.h:162
const char * fname
Name of the file to read.
Definition lib.h:166
Paged view into some data.
Definition lib.h:173
struct PagerData * pdata
Data that pager displays. NOTNULL.
Definition lib.h:174
enum PagerMode mode
Pager mode.
Definition lib.h:175
PagerFlags flags
Additional settings to tweak pager's function.
Definition lib.h:176
const char * banner
Title to display in status bar.
Definition lib.h:177
Send private Module data.
Definition module_data.h:33
volatile sig_atomic_t sig_alrm
true after SIGALRM is received
Definition module_data.h:36
#define buf_mktemp(buf)
Definition tmp.h:33