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
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
|
#include <sys/stat.h>
#include <unistd.h>
#include "buffer.h"
#include "error.h"
#include "exit.h"
#include "fd.h"
#include "open.h"
#include "pathexec.h"
#include "wait.h"
#include "auto_qmail.h"
#include "ipalloc.h"
#include "tcpto.h"
void initialize(int argc, char **argv)
{
tcpto_clean();
}
int truncreport = 0;
void report(buffer *log, int wstat, char *s, int len)
{
int j;
int k;
int result;
int orr;
if (wait_crashed(wstat)) {
buffer_putsflush(log, "Zqmail-spawn: qmail-remote crashed.\n");
return;
}
switch (wait_exitcode(wstat)) {
case 0: break;
case 111: buffer_putsflush(log, "Zqmail-rspawn: Unable to run qmail-remote.\n"); break;
default: buffer_putsflush(log, "Dqmail-rspawn: Unable to run qmail-remote. \n"); return;
}
if (!len) {
buffer_putsflush(log, "Zqmail-rspawn: qmail-remote produced no output.\n");
return;
}
result = -1;
j = 0;
for (k = 0; k < len; ++k) {
if (!s[k]) {
if (s[j] == 'K') {
result = 1;
break;
}
if (s[j] == 'Z') {
result = 0;
break;
}
if (s[j] == 'D') break;
j = k + 1;
}
}
orr = result;
switch (s[0]) {
case 's': orr = 0; break;
case 'h': orr = -1;
}
switch (orr) {
case 1: buffer_put(log, "K", 1); break;
case 0: buffer_put(log, "Z", 1); break;
case -1: buffer_put(log, "D", 1); break;
}
for (k = 1; k < len;) {
if (!s[k++]) {
buffer_puts(log, s + 1);
if (result <= orr && k < len) {
switch (s[k]) {
case 'Z':
case 'D':
case 'K': buffer_puts(log, s + k + 1);
}
}
break;
}
}
}
int spawn(int fdmess, int fdout, const char *s, char *r, const int at)
{
int f;
char *(args[5]);
struct stat st;
if (chdir(auto_qmail) == -1) _exit(110);
if (!stat("control/dkimdomains", &st))
args[0] = "qmail-dksign";
else
args[0] = "qmail-remote";
args[1] = r + at + 1;
args[2] = s;
args[3] = r;
args[4] = 0;
if (chdir("queue/mess") == -1) _exit(110);
if (!(f = vfork())) {
if (fd_move(0, fdmess) == -1) _exit(111);
if (fd_move(1, fdout) == -1) _exit(111);
if (fd_copy(2, 1) == -1) _exit(111);
pathexec(args);
if (errno) _exit(111);
_exit(100);
}
return f;
}
|