#include "fd.h"
#include "stralloc.h"
#include "readwrite.h"
#include "sgetopt.h"
#include "wait.h"
#include "env.h"
#include "byte.h"
#include "str.h"
#include "alloc.h"
#include "exit.h"
#include "fork.h"
#include "case.h"
#include "subfd.h"
#include "error.h"
#include "substdio.h"
#include "sig.h"
void die(e,s) int e; char *s; { substdio_putsflush(subfderr,s); _exit(e); }
void die_usage() { die(100,"qsmhook: fatal: incorrect usage\n"); }
void die_temp() { die(111,"qsmhook: fatal: temporary problem\n"); }
void die_read() { die(111,"qsmhook: fatal: unable to read message\n"); }
void die_badcmd() { die(100,"qsmhook: fatal: command not found\n"); }
int flagrpline = 0; char *rpline;
int flagufline = 1; char *ufline;
int flagdtline = 0; char *dtline;
char *host;
char *sender;
char *recip;
stralloc newarg = {0};
substdio ssout;
char outbuf[SUBSTDIO_OUTSIZE];
substdio ssin;
char inbuf[SUBSTDIO_INSIZE];
void main(argc,argv)
int argc;
char **argv;
{
int pid;
int wstat;
int pi[2];
int opt;
char **arg;
char *x;
int i;
int flagesc;
sig_pipeignore();
if (!(dtline = env_get("DTLINE"))) die_usage();
if (!(rpline = env_get("RPLINE"))) die_usage();
if (!(ufline = env_get("UFLINE"))) die_usage();
if (!(recip = env_get("LOCAL"))) die_usage();
if (!(host = env_get("HOST"))) die_usage();
if (!(sender = env_get("SENDER"))) die_usage();
while ((opt = getopt(argc,argv,"DFlMmnPsx:")) != opteof)
switch(opt)
{
case 'D': case 'F': case 'M': break; /* be serious */
case 'l': flagdtline = 1; break; /* also return-receipt-to? blech */
case 'm': break; /* we only handle one recipient anyway */
case 'n': flagufline = 0; break;
case 's': break; /* could call quote() otherwise, i suppose... */
case 'P': flagrpline = 1; break;
case 'x':
if (case_starts(recip,optarg))
recip += str_len(optarg);
break;
default:
_exit(100);
}
argc -= optind;
argv += optind;
if (!*argv) die_usage();
for (arg = argv;x = *arg;++arg)
{
if (!stralloc_copys(&newarg,"")) die_temp();
flagesc = 0;
for (i = 0;x[i];++i)
if (flagesc)
{
switch(x[i])
{
case '%': if (!stralloc_cats(&newarg,"%")) die_temp(); break;
case 'g': if (!stralloc_cats(&newarg,sender)) die_temp(); break;
case 'h': if (!stralloc_cats(&newarg,host)) die_temp(); break;
case 'u': if (!stralloc_cats(&newarg,recip)) die_temp(); break;
}
flagesc = 0;
}
else
if (x[i] == '%')
flagesc = 1;
else
if (!stralloc_append(&newarg,&x[i])) die_temp();
if (!stralloc_0(&newarg)) die_temp();
i = str_len(newarg.s) + 1;
if (!(x = alloc(i))) die_temp();
byte_copy(x,i,newarg.s);
*arg = x;
}
if (pipe(pi) == -1) die_temp();
switch(pid = fork())
{
case -1:
die_temp();
case 0:
close(pi[1]);
if (fd_move(0,pi[0]) == -1) die_temp();
sig_pipedefault();
execvp(*argv,argv);
if (error_temp(errno)) die_temp();
die_badcmd();
}
close(pi[0]);
substdio_fdbuf(&ssout,write,pi[1],outbuf,sizeof(outbuf));
substdio_fdbuf(&ssin,read,0,inbuf,sizeof(inbuf));
if (flagufline) substdio_bputs(&ssout,ufline);
if (flagrpline) substdio_bputs(&ssout,rpline);
if (flagdtline) substdio_bputs(&ssout,dtline);
if (substdio_copy(&ssout,&ssin) == -2) die_read();
substdio_flush(&ssout);
close(pi[1]);
if (wait_pid(&wstat,pid) == -1) die_temp();
if (wait_crashed(wstat)) die_temp();
_exit(wait_exitcode(wstat));
}
syntax highlighted by Code2HTML, v. 0.9.1