/* $OpenBSD: mfa_session.c,v 1.10 2012/09/29 11:02:41 eric Exp $ */ /* * Copyright (c) 2011 Gilles Chehade * * Permission to use, copy, modify, and distribute this software for any * purpose with or without fee is hereby granted, provided that the above * copyright notice and this permission notice appear in all copies. * * THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES * WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF * MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR * ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES * WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN * ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF * OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE. */ #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include "smtpd.h" #include "log.h" static int mfa_session_proceed(struct mfa_session *); static void mfa_session_pickup(struct mfa_session *); static void mfa_session_fail(struct mfa_session *); static void mfa_session_destroy(struct mfa_session *); static void mfa_session_done(struct mfa_session *); static void mfa_session_imsg(int, short, void *); struct tree sessions; void mfa_session(struct submit_status *ss, enum session_state state) { struct mfa_session *ms; ms = xcalloc(1, sizeof(*ms), "mfa_session"); ms->id = generate_uid(); ms->ss = *ss; ms->ss.code = 250; ms->state = state; ms->filter = TAILQ_FIRST(env->sc_filters); tree_xset(&sessions, ms->id, ms); if (ms->filter == NULL) mfa_session_done(ms); else if (!mfa_session_proceed(ms)) mfa_session_fail(ms); } static int mfa_session_proceed(struct mfa_session *ms) { struct filter_msg fm; fm.id = ms->id; fm.cl_id = ms->ss.id; fm.version = FILTER_API_VERSION; switch (ms->state) { case S_CONNECTED: fm.type = FILTER_CONNECT; if (strlcpy(fm.u.connect.hostname, ms->ss.envelope.hostname, sizeof(fm.u.connect.hostname)) >= sizeof(fm.u.connect.hostname)) fatalx("mfa_session_proceed: CONNECT: truncation"); fm.u.connect.hostaddr = ms->ss.envelope.ss; break; case S_HELO: fm.type = FILTER_HELO; if (strlcpy(fm.u.helo.helohost, ms->ss.envelope.helo, sizeof(fm.u.helo.helohost)) >= sizeof(fm.u.helo.helohost)) fatalx("mfa_session_proceed: HELO: truncation"); break; case S_MAIL_MFA: fm.type = FILTER_MAIL; if (strlcpy(fm.u.mail.user, ms->ss.u.maddr.user, sizeof(fm.u.mail.user)) >= sizeof(fm.u.mail.user)) fatalx("mfa_session_proceed: MAIL: user truncation"); if (strlcpy(fm.u.mail.domain, ms->ss.u.maddr.domain, sizeof(fm.u.mail.domain)) >= sizeof(fm.u.mail.domain)) fatalx("mfa_session_proceed: MAIL: domain truncation"); break; case S_RCPT_MFA: fm.type = FILTER_RCPT; if (strlcpy(fm.u.mail.user, ms->ss.u.maddr.user, sizeof(fm.u.mail.user)) >= sizeof(fm.u.mail.user)) fatalx("mfa_session_proceed: RCPT: user truncation"); if (strlcpy(fm.u.mail.domain, ms->ss.u.maddr.domain, sizeof(fm.u.mail.domain)) >= sizeof(fm.u.mail.domain)) fatalx("mfa_session_proceed: RCPT: domain truncation"); break; case S_DATACONTENT: fm.type = FILTER_DATALINE; if (strlcpy(fm.u.dataline.line, ms->ss.u.dataline, sizeof(fm.u.dataline.line)) >= sizeof(fm.u.dataline.line)) fatalx("mfa_session_proceed: DATA: line truncation"); break; case S_QUIT: fm.type = FILTER_QUIT; break; case S_CLOSE: fm.type = FILTER_CLOSE; break; case S_RSET: fm.type = FILTER_RSET; break; default: fatalx("mfa_session_proceed: no such state"); } imsg_compose(ms->filter->ibuf, fm.type, 0, 0, -1, &fm, sizeof(fm)); event_set(&ms->filter->ev, ms->filter->ibuf->fd, EV_READ|EV_WRITE, mfa_session_imsg, ms->filter); event_add(&ms->filter->ev, NULL); return 1; } static void mfa_session_pickup(struct mfa_session *ms) { if (ms->fm.code == STATUS_REJECT) { mfa_session_fail(ms); return; } ms->filter = TAILQ_NEXT(ms->filter, f_entry); if (ms->filter == NULL) mfa_session_done(ms); else mfa_session_proceed(ms); } static void mfa_session_done(struct mfa_session *ms) { enum imsg_type imsg_type; switch (ms->state) { case S_CONNECTED: imsg_type = IMSG_MFA_CONNECT; break; case S_HELO: imsg_type = IMSG_MFA_HELO; break; case S_MAIL_MFA: if (ms->ss.code != 530) { imsg_compose_event(env->sc_ievs[PROC_LKA], IMSG_LKA_MAIL, 0, 0, -1, &ms->ss, sizeof(ms->ss)); mfa_session_destroy(ms); return; } imsg_type = IMSG_MFA_MAIL; break; case S_RCPT_MFA: if (ms->ss.code != 530) { imsg_compose_event(env->sc_ievs[PROC_LKA], IMSG_LKA_RULEMATCH, 0, 0, -1, &ms->ss, sizeof(ms->ss)); mfa_session_destroy(ms); return; } imsg_type = IMSG_MFA_RCPT; break; case S_DATACONTENT: if (ms->ss.code != 530 && ms->fm.code != 0) (void)strlcpy(ms->ss.u.dataline, ms->fm.u.dataline.line, sizeof(ms->ss.u.dataline)); imsg_type = IMSG_MFA_DATALINE; break; case S_QUIT: imsg_type = IMSG_MFA_QUIT; break; case S_CLOSE: imsg_type = IMSG_MFA_CLOSE; /* Why answer back to SMTP? The session is closed! */ mfa_session_destroy(ms); return; case S_RSET: imsg_type = IMSG_MFA_RSET; break; default: fatalx("mfa_session_done: unsupported state"); } imsg_compose_event(env->sc_ievs[PROC_SMTP], imsg_type, 0, 0, -1, &ms->ss, sizeof(struct submit_status)); mfa_session_destroy(ms); } static void mfa_session_fail(struct mfa_session *ms) { ms->ss.code = 530; mfa_session_done(ms); } static void mfa_session_destroy(struct mfa_session *ms) { tree_xpop(&sessions, ms->id); free(ms); } static void mfa_session_imsg(int fd, short event, void *p) { struct filter *filter = p; struct mfa_session *ms; struct imsg imsg; ssize_t n; struct filter_msg fm; short evflags = EV_READ; if (event & EV_READ) { n = imsg_read(filter->ibuf); if (n == -1) fatal("mfa_session_imsg: imsg_read"); if (n == 0) { event_del(&filter->ev); event_loopexit(NULL); return; } } if (event & EV_WRITE) { if (msgbuf_write(&filter->ibuf->w) == -1) fatal("mfa_session_imsg: msgbuf_write"); if (filter->ibuf->w.queued) evflags |= EV_WRITE; } for (;;) { n = imsg_get(filter->ibuf, &imsg); if (n == -1) fatalx("mfa_session_imsg: imsg_get"); if (n == 0) break; if ((imsg.hdr.len - IMSG_HEADER_SIZE) != sizeof(fm)) fatalx("mfa_session_imsg: corrupted imsg"); memcpy(&fm, imsg.data, sizeof (fm)); if (fm.version != FILTER_API_VERSION) fatalx("mfa_session_imsg: API version mismatch"); switch (imsg.hdr.type) { case FILTER_CONNECT: case FILTER_HELO: case FILTER_EHLO: case FILTER_MAIL: case FILTER_RCPT: case FILTER_DATALINE: case FILTER_QUIT: case FILTER_CLOSE: case FILTER_RSET: ms = tree_xget(&sessions, fm.id); /* overwrite filter code */ ms->fm.code = fm.code; /* success, overwrite */ if (fm.code == STATUS_ACCEPT) ms->fm = fm; mfa_session_pickup(ms); break; default: fatalx("mfa_session_imsg: unsupported imsg"); } imsg_free(&imsg); } event_set(&filter->ev, filter->ibuf->fd, evflags, mfa_session_imsg, filter); event_add(&filter->ev, NULL); }