s/qmail 4.3.23
Next generation secure email transport
Loading...
Searching...
No Matches
qmail-mrtg.c
Go to the documentation of this file.
1#include <unistd.h>
2#include "stralloc.h"
3#include "buffer.h"
4#include "getln.h"
5#include "exit.h"
6#include "open.h"
7#include "scan.h"
8#include "fmt.h"
9#include "case.h"
10#include "now.h"
11#include "str.h"
12#include "datetime.h"
13#include "logmsg.h"
14#include "qmail.h"
15
16#define WHO "qmail-mrtg"
17#define TAI64NLEN 24
18
24
25/* qmail-send */
26
27int local = 0;
28int remote = 0;
29int success = 0;
30int failure = 0;
31int bytes = 0;
32int tlstrans = 0;
33int deferral = 0;
34int bounces = 0;
35int triples = 0;
36int qmtp = 0;
37int qmtps = 0;
38
39/* qmail-smtpd */
40
41int asessions = 0;
42int rsessions = 0;
43int aorig = 0;
44int arcpt = 0;
45int rsend = 0;
46int rhelo = 0;
47int rorigbad = 0;
48int rorigdns = 0;
49int rrcptbad = 0;
50int rrcptfail = 0;
51int rsize = 0;
52int rmime = 0;
53int rloader = 0;
54int rvirus = 0;
55int rspam = 0;
56int rdkim = 0;
57int aauth = 0;
58int rauth = 0;
59int atls = 0;
60int rtls = 0;
61int spfpass = 0;
62int spfail = 0;
63
64/* qmail-pop3d */
65
66int apop = 0;
67int rpop = 0;
68int pok = 0;
69int pdeny = 0;
70
71/* *server + rblsmtpd */
72
73int sok = 0;
74int sdeny = 0;
75int greet = 0;
76int grey = 0;
77int rbl = 0;
78
79char bufsmall[BUFFER_SMALL];
80buffer bo = BUFFER_INIT(buffer_unixwrite,1,bufsmall,sizeof(bufsmall));
81
82static void outs(char *s)
83{
84 if (buffer_puts(&bo,s) == -1) _exit(1);
85 if (buffer_puts(&bo,"\n") == -1) _exit(1);
86 if (buffer_flush(&bo) == -1) _exit(1);
87}
88
89static void out(int i)
90{
91 char num[FMT_ULONG];
92
93 if (buffer_put(&bo,num,fmt_ulong(num,(unsigned long) i)) == -1) _exit(1);
94 if (buffer_puts(&bo,"\n") == -1) _exit(1);
95 if (buffer_flush(&bo) == -1) _exit(1);
96}
97
99buffer bi = BUFFER_INIT(buffer_unixread,0,bufspace,sizeof(bufspace));
100
101static void mrtg_results(char flag)
102{
103 switch (flag) {
104 case '1': out(success); out(tlstrans); break;
105 case '2': bytes = bytes/1024; out(bytes); out(bytes); break;
106 case '3': out(local); out(remote); break;
107 case '4': out(failure); out(deferral); break;
108 case '5': out(bounces); out(triples); break;
109 case '6': qmtps += qmtp; out(qmtp); out(qmtps); break; /* QMTP */
110
111 case 'a': out(asessions); out(rsessions); break; /* total */
112 case 'b': out(aorig); out(arcpt); break; /* accepted */
113 case 'c': out(rsend); out(rhelo); break; /* rejected MTA */
114 case 'd': out(rorigbad); out(rorigdns); break; /* Orig */
115 case 'e': out(rrcptbad); out(rrcptfail); break; /* Recipient */
116 case 'f': out(rmime); out(rloader); break; /* Warlord */
117 case 'g': out(rvirus); out(rspam); break; /* Infected/Spam */
118 case 'h': out(aauth); out(rauth); break; /* Auth */
119 case 'i': out(atls); out(rtls); break; /* TLS */
120 case 'j': out(spfpass); out(spfail); break; /* SPF */
121 case 'k': out(grey); break; /* Greylisted */
122 case 'm': out(rdkim); break; /* DKIM rejected */
123 case 'z': sdeny +=rbl; out(sok); out(sdeny); break; /* reject session */
124
125 case 'A': out(apop); out(rpop); break;
126 case 'B': out(pok); out(pdeny); break;
127
128 default: break;
129 }
130}
131
132static void mrtg_sendlog(char *in, char flag)
133{
134 int i, j, k = 0;
135
136 switch (flag) {
137 case '1': if (case_starts(in,"delivery")) {
138 i = str_chr(in,':') + 2;
139 if (case_starts(in + i,"success:")) success++;
140 i = str_chr(in,'T');
141 if (case_starts(in + i,"TLS_")) tlstrans++;
142 }; break;
143 case '2': if (case_starts(in,"info msg")) {
144 i = str_chr(in,':') + 8;
145 if ((j = str_chr(in + i,' '))) in[i + j] = '\0';
146 bytes += atoi(in + i);
147 }; break;
148 case '3': if (case_starts(in,"status:")) {
149 i = str_rchr(in,'c') + 4;
150 k = str_rchr(in,'r') + 7;
151 if ((j = str_chr(in + i,'/'))) in[i + j] = '\0';
152 if (atoi(in + i) > local) local = atoi(in + i);
153 if ((j = str_chr(in + k,'/'))) in[k + j] = '\0';
154 if (atoi(in + k) > remote) remote = atoi(in + k);
155 }; break;
156 case '4': if (case_starts(in,"delivery")) {
157 i = str_chr(in,':') + 2;
158 if (case_starts(in + i,"failure:")) failure++;
159 if (case_starts(in + i,"deferral:")) deferral++;
160 }; break;
161 case '5': if (case_starts(in,"bounce msg")) bounces++;
162 if (case_starts(in,"triple bounce:")) triples++;
163 break;
164 case '6': if (case_starts(in,"delivery")) {
165 i = str_chr(in,'q');
166 if (case_starts(in + i,"qmtp:_ok")) qmtp++;
167 if (case_starts(in + i,"qmtps:_ok")) qmtps++;
168 }; break;
169 default: break;
170 }
171}
172
173static void mrtg_smtplog(char *in, char flag)
174{
175 int a, d, p, r = 0;
176
177 a = str_chr(in,'A');
178 d = str_chr(in,'D');
179 r = str_chr(in,'R');
180 p = str_chr(in,'P');
181
182 switch (flag) {
183 case 'a': if (case_starts(in + a,"Accept")) asessions++;
184 if (case_starts(in + r,"Reject")) rsessions++;
185 break;
186 case 'b': if (case_starts(in + a,"Accept::ORIG:")) aorig++;
187 if (case_starts(in + a,"Accept::RCPT:")) arcpt++;
188 break;
189 case 'c': if (case_starts(in + r,"Reject::SNDR::Invalid_Relay")) rsend++;
190 if (case_starts(in + r,"Reject::SNDR::Bad_Helo")) rhelo++;
191 if (case_starts(in + r,"Reject::SNDR::DNS_Helo")) rhelo++;
192 break;
193 case 'd': if (case_starts(in + r,"Reject::ORIG::Bad_Mailfrom")) rorigbad++;
194 if (case_starts(in + r,"Reject::ORIG::DNS_MF")) rorigdns++;
195 break;
196 case 'e': if (case_starts(in + r,"Reject::RCPT::Bad_Rcptto")) rrcptbad++;
197 if (case_starts(in + r,"Reject::RCPT::Failed_Rcptto")) rrcptfail++;
198 break;
199 case 'f': if (case_starts(in + r,"Reject::DATA::Invalid_Size")) rsize++;
200 if (case_starts(in + r,"Reject::DATA::Bad_MIME")) rmime++;
201 if (case_starts(in + r,"Reject::DATA::MIME_Attach")) rmime++;
202 if (case_starts(in + r,"Reject::DATA::Bad_Loader")) rloader++;
203 break;
204 case 'g': if (case_starts(in + r,"Reject::DATA::Spam_Message")) rspam++;
205 if (case_starts(in + r,"Reject::DATA::Virus_Infected")) rvirus++;
206 break;
207 case 'h': if (case_starts(in + a,"Accept::AUTH:")) aauth++;
208 if (case_starts(in + r,"Reject::AUTH:")) rauth++;
209 break;
210 case 'i': if (case_starts(in + p,"P:ESMTPS")) atls++;
211 if (case_starts(in + r,"Reject::TLS:")) rtls++;
212 break;
213 case 'j': if (case_starts(in + a,"Accept::SPF:")) spfpass++;
214 if (case_starts(in + r,"Reject::SPF:")) spfail++;
215 break;
216 case 'k': if (case_starts(in + d,"Deferred::SNDR::Grey_Listed")) grey++;
217 break;
218 case 'm': if (case_starts(in + r,"Rejected::SNDR::DKIM_Signature")) rdkim++;
219 break;
220 case 'z': if (case_starts(in,"tcpserver") || case_starts(in,"sslserver") || case_starts(in,"rblsmtpd")) {
221 a = str_chr(in,':') + 2;
222 if (case_starts(in + a,"ok")) sok++;
223 if (case_starts(in + a,"deny")) sdeny++;
224 r = str_chr(in + a,':') + 2;
225 if (case_starts(in + a + r,"451")) rbl++;
226 if (case_starts(in + a + r,"553")) rbl++;
227 if (case_starts(in + a + r,"greetdelay:")) greet++;
228 } break;
229 default: break;
230 }
231}
232
233static void mrtg_pop3log(char *in, char flag)
234{
235 int a, r = 0;
236
237 switch (flag) {
238 case 'A': a = str_chr(in,'A'); r = str_chr(in,'R');
239 if (case_starts(in + a,"Accept::AUTH:")) apop++;
240 if (case_starts(in + r,"Reject::AUTH:")) rpop++;
241 break;
242 case 'B': if (case_starts(in,"tcpserver:") || case_starts(in,"sslserver:")) {
243 a = str_chr(in,':') + 2;
244 if (case_starts(in + a,"ok")) pok++;
245 if (case_starts(in + a,"deny")) pdeny++;
246 }; break;
247 default: break;
248 }
249}
250
251int main(int argc, char * const *argv)
252{
253 int i;
254 int c;
255 int match;
256 int enoughtime = 0;
257 unsigned long u;
258 unsigned long calltime;
259 unsigned long seconds;
260 unsigned long nanoseconds;
261 unsigned int history = 305;
262 char flag;
263
264 stralloc line = {0};
265 calltime = now();
266
267 if (argc < 2)
268 logmsg(WHO,100,USAGE,"qmail-mrtg [ -1 | -2 | -3 | -4 | -5 | -6 |\
269 -a | -b | -c | -d | -e | -f | -g | -h | -i | -j | -k | -m | -z | -A | -B ] [time (min)] \n\
270 qmail-mrtg needs to be called every [time] minutes (i.e. by crontab) - default 305 secs");
271
272 flag = *(argv[1] + 1);
273 if (argc == 3) { scan_ulong(argv[2],&u); history = 60 * u + 5; }
274
275/* Read input lines sequentially */
276
277 buffer_init(&bi,buffer_unixread,0,bufspace,sizeof(bufspace));
278
279 for (;;) {
280 if (getln(&bi,&line,&match,'\n') != 0) _exit(1);
281 if (!match) break;
282 if (!stralloc_0(&line)) _exit(1);
283
284 seconds = 0;
285 nanoseconds = 0;
286
287 if (line.s[0] == '@') { /* tai64 timestamp */
288 for (i = 1; i <= TAI64NLEN; i++) {
289 c = (int)line.s[i];
290 u = c - '0';
291 if (u >= 10) {
292 u = c - 'a';
293 if (u >= 6) break;
294 u += 10;
295 }
296 seconds <<= 4;
297 seconds += nanoseconds >> 28;
298 nanoseconds &= 0xfffffff;
299 nanoseconds <<= 4;
300 nanoseconds += u;
301 }
302 seconds -= 4611686018427387914ULL;
303 seconds = seconds > 0 ? seconds : 0;
304 } else {
305 outs("Error: No TAI64N timestamp available.");
306 _exit(1);
307 }
308
309 if (enoughtime) { /* default history = 305 sec => evaluate logs every ~5 mins */
310 if (seconds <= calltime && seconds >= (calltime - history)) {
311 if (flag >= '1' && flag <= '9') mrtg_sendlog(line.s + TAI64NLEN + 2,flag);
312 else if (flag >= 'a' && flag <= 'z') mrtg_smtplog(line.s + TAI64NLEN + 2,flag);
313 else if (flag >= 'A' && flag <= 'Z') mrtg_pop3log(line.s + TAI64NLEN + 2,flag);
314 }
315 } else {
316 if (seconds) {
317 enoughtime++;
318 } else {
319 outs("Warning: Not enough time left between calls");
320 _exit(1);
321 }
322 }
323 }
324
325 mrtg_results(flag);
326
327 _exit(0);
328}
void outs(char *s)
Definition auto-gid.c:12
#define WHO
Definition bouncesaying.c:8
int main()
Definition chkshsgr.c:6
char num[FMT_ULONG]
Definition chkspawn.c:8
buffer bo
Definition columnt.c:13
buffer bi
stralloc out
Definition dnscname.c:12
void _exit(int)
void p(char *, char *, int, int, int)
Definition install.c:49
unsigned int
Definition ipalloc.h:19
ulongalloc bytes
Definition matchup.c:57
int match
Definition matchup.c:196
datetime_sec now()
Definition now.c:5
char * local
Definition qmail-getpw.c:18
int rhelo
Definition qmail-mrtg.c:46
int tlstrans
Definition qmail-mrtg.c:32
int rauth
Definition qmail-mrtg.c:58
int triples
Definition qmail-mrtg.c:35
int rsessions
Definition qmail-mrtg.c:42
int qmtp
Definition qmail-mrtg.c:36
int bounces
Definition qmail-mrtg.c:34
int rrcptfail
Definition qmail-mrtg.c:50
int rloader
Definition qmail-mrtg.c:53
int arcpt
Definition qmail-mrtg.c:44
int pok
Definition qmail-mrtg.c:68
int failure
Definition qmail-mrtg.c:30
int rbl
Definition qmail-mrtg.c:77
int rpop
Definition qmail-mrtg.c:67
#define TAI64NLEN
Definition qmail-mrtg.c:17
int rspam
Definition qmail-mrtg.c:55
int success
Definition qmail-mrtg.c:29
int sdeny
Definition qmail-mrtg.c:74
int rdkim
Definition qmail-mrtg.c:56
int asessions
Definition qmail-mrtg.c:41
char bufspace[BUFSIZE_LINE]
Definition qmail-mrtg.c:98
int rvirus
Definition qmail-mrtg.c:54
int grey
Definition qmail-mrtg.c:76
int remote
Definition qmail-mrtg.c:28
int spfpass
Definition qmail-mrtg.c:61
int qmtps
Definition qmail-mrtg.c:37
int rsize
Definition qmail-mrtg.c:51
int rorigbad
Definition qmail-mrtg.c:47
char bufsmall[BUFFER_SMALL]
Definition qmail-mrtg.c:79
int rtls
Definition qmail-mrtg.c:60
int greet
Definition qmail-mrtg.c:75
int rrcptbad
Definition qmail-mrtg.c:49
int apop
Definition qmail-mrtg.c:66
int pdeny
Definition qmail-mrtg.c:69
int aauth
Definition qmail-mrtg.c:57
int rorigdns
Definition qmail-mrtg.c:48
int sok
Definition qmail-mrtg.c:73
int atls
Definition qmail-mrtg.c:59
int spfail
Definition qmail-mrtg.c:62
int deferral
Definition qmail-mrtg.c:33
int aorig
Definition qmail-mrtg.c:43
int rsend
Definition qmail-mrtg.c:45
int rmime
Definition qmail-mrtg.c:52
buffer in
Definition qmail-pw2u.c:240
int j
Definition qmail-send.c:926
struct del * d[CHANNELS]
Definition qmail-send.c:726
#define BUFSIZE_LINE
Definition qmail.h:8
uint32_t k[64]
Definition sha256.c:26