summaryrefslogtreecommitdiff
path: root/src/condredirect.c
blob: 534dac75d86d4e1b8d38a14f6437fdc870d5ba16 (plain)
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
69
70
71
72
73
74
75
76
#include <unistd.h>

#include "buffer.h"
#include "env.h"
#include "exit.h"
#include "fmt.h"
#include "logmsg.h"
#include "seek.h"
#include "sig.h"
#include "wait.h"

#include "qmail.h"

#define WHO "condredirect"

struct qmail qqt;

ssize_t mywrite(int fd, char *buf, int len)
{
  qmail_put(&qqt, buf, len);
  return len;
}

char inbuf[BUFFER_INSIZE];
buffer bi = BUFFER_INIT(read, 0, inbuf, sizeof(inbuf));
char outbuf[1];
buffer bo = BUFFER_INIT(mywrite, -1, outbuf, sizeof(outbuf));

char num[FMT_ULONG];

int main(int argc, char **argv)
{
  char *sender;
  char *dtline;
  int pid;
  int wstat;
  char *qqx;

  if (!argv[1] || !argv[2]) logmsg(WHO, 100, USAGE, "condredirect newaddress program [ arg ... ]");

  pid = fork();
  if (pid == -1) logmsg(WHO, 111, FATAL, "unable to fork: ");
  if (pid == 0) {
    execvp(argv[2], argv + 2);
    if (errno) _exit(111);
    _exit(100);
  }
  if (wait_pid(&wstat, pid) == -1) logmsg(WHO, 111, FATAL, "wait failed");
  if (wait_crashed(wstat)) logmsg(WHO, 111, FATAL, "child crashed");
  switch (wait_exitcode(wstat)) {
    case 0:   break;
    case 111: logmsg(WHO, 111, FATAL, "temporary child error");
    default:  _exit(0);
  }

  if (seek_begin(0) == -1) logmsg(WHO, 111, FATAL, "unable to rewind: ");
  sig_pipeignore();

  sender = env_get("SENDER");
  if (!sender) logmsg(WHO, 100, ERROR, "SENDER not set");
  dtline = env_get("DTLINE");
  if (!dtline) logmsg(WHO, 100, ERROR, "DTLINE not set");

  if (qmail_open(&qqt) == -1) logmsg(WHO, 111, FATAL, "unable to fork: ");
  qmail_puts(&qqt, dtline);
  if (buffer_copy(&bo, &bi) != 0) logmsg(WHO, 111, FATAL, "unable to read message: ");
  buffer_flush(&bo);

  num[fmt_ulong(num, qmail_qp(&qqt))] = 0;

  qmail_from(&qqt, sender);
  qmail_to(&qqt, argv[1]);
  qqx = qmail_close(&qqt);
  if (*qqx) logmsg(WHO, *qqx == 'D' ? 100 : 111, FATAL, qqx + 1);
  logmsg(WHO, 0, LOG, B("qp ", num));
}