diff options
Diffstat (limited to 'usr.sbin/smtpd/smtpctl.c')
-rw-r--r-- | usr.sbin/smtpd/smtpctl.c | 100 |
1 files changed, 99 insertions, 1 deletions
diff --git a/usr.sbin/smtpd/smtpctl.c b/usr.sbin/smtpd/smtpctl.c index 7d41c10a8e1..e8f0c77b1b0 100644 --- a/usr.sbin/smtpd/smtpctl.c +++ b/usr.sbin/smtpd/smtpctl.c @@ -1,4 +1,4 @@ -/* $OpenBSD: smtpctl.c,v 1.72 2011/11/23 13:48:03 chl Exp $ */ +/* $OpenBSD: smtpctl.c,v 1.73 2011/12/14 18:42:27 eric Exp $ */ /* * Copyright (c) 2006 Pierre-Yves Ritschard <pyr@openbsd.org> @@ -39,12 +39,16 @@ #include "smtpd.h" #include "parser.h" +#include "log.h" void usage(void); static void setup_env(struct smtpd *); static void show_sizes(void); static int show_command_output(struct imsg *); static int show_stats_output(struct imsg *); +static void show_queue(enum queue_kind, int); +static void show_envelope(struct envelope *, int); +static void getflag(u_int *, int, char *, char *, size_t); int proctype; struct imsgbuf *ibuf; @@ -472,3 +476,97 @@ show_stats_output(struct imsg *imsg) return (1); } + +static void +show_queue(enum queue_kind kind, int flags) +{ + struct qwalk *q; + struct envelope envelope; + u_int64_t evpid; + + log_init(1); + + if (chroot(PATH_SPOOL) == -1 || chdir(".") == -1) + err(1, "%s", PATH_SPOOL); + + q = qwalk_new(kind, 0); + + while (qwalk(q, &evpid)) { + if (! queue_envelope_load(kind, evpid, &envelope)) + continue; + show_envelope(&envelope, flags); + } + + qwalk_close(q); +} + +static void +show_envelope(struct envelope *e, int flags) +{ + char status[128]; + + status[0] = '\0'; + + getflag(&e->status, DS_TEMPFAILURE, "TEMPFAIL", + status, sizeof(status)); + + if (e->status) + errx(1, "%016" PRIx64 ": unexpected status 0x%04x", e->id, + e->status); + + getflag(&e->flags, DF_BOUNCE, "BOUNCE", + status, sizeof(status)); + getflag(&e->flags, DF_AUTHENTICATED, "AUTH", + status, sizeof(status)); + getflag(&e->flags, DF_ENQUEUED, "ENQUEUED", + status, sizeof(status)); + getflag(&e->flags, DF_INTERNAL, "INTERNAL", + status, sizeof(status)); + + if (e->flags) + errx(1, "%016" PRIx64 ": unexpected flags 0x%04x", e->id, + e->flags); + + if (status[0]) + status[strlen(status) - 1] = '\0'; + else + strlcpy(status, "-", sizeof(status)); + + switch (e->type) { + case D_MDA: + printf("MDA"); + break; + case D_MTA: + printf("MTA"); + break; + case D_BOUNCE: + printf("BOUNCE"); + break; + default: + printf("UNKNOWN"); + } + + printf("|%016" PRIx64 "|%s|%s@%s|%s@%s|%" PRId64 "|%" PRId64 "|%u", + e->id, + status, + e->sender.user, e->sender.domain, + e->dest.user, e->dest.domain, + (int64_t) e->lasttry, + (int64_t) e->expire, + e->retry); + + if (e->errorline[0] != '\0') + printf("|%s", e->errorline); + + printf("\n"); +} + +static void +getflag(u_int *bitmap, int bit, char *bitstr, char *buf, size_t len) +{ + if (*bitmap & bit) { + *bitmap &= ~bit; + strlcat(buf, bitstr, len); + strlcat(buf, ",", len); + } +} |