diff options
-rw-r--r-- | src/libserver/milter.c | 116 | ||||
-rw-r--r-- | src/libserver/milter_internal.h | 22 |
2 files changed, 125 insertions, 13 deletions
diff --git a/src/libserver/milter.c b/src/libserver/milter.c index 9bb6ead39..79bcd80e5 100644 --- a/src/libserver/milter.c +++ b/src/libserver/milter.c @@ -54,25 +54,71 @@ rspamd_milter_plan_io (struct rspamd_milter_session *session, event_add (&priv->ev, priv->ptv); } +static void +rspamd_milter_on_protocol_error (struct rspamd_milter_session *session, + struct rspamd_milter_private *priv, GError *err) +{ + REF_RETAIN (session); + priv->err_cb (priv->fd, session, priv->ud, err); + REF_RELEASE (session); + g_error_free (err); +} + +static gboolean +rspamd_milter_process_command (struct rspamd_milter_session *session, + struct rspamd_milter_private *priv) +{ + switch (priv->parser.cur_cmd) { + case RSPAMD_MILTER_CMD_ABORT: + break; + case RSPAMD_MILTER_CMD_BODY: + break; + case RSPAMD_MILTER_CMD_CONNECT: + break; + case RSPAMD_MILTER_CMD_MACRO: + break; + case RSPAMD_MILTER_CMD_BODYEOB: + break; + case RSPAMD_MILTER_CMD_HELO: + break; + case RSPAMD_MILTER_CMD_QUIT_NC: + break; + case RSPAMD_MILTER_CMD_HEADER: + break; + case RSPAMD_MILTER_CMD_MAIL: + break; + case RSPAMD_MILTER_CMD_EOH: + break; + case RSPAMD_MILTER_CMD_OPTNEG: + break; + case RSPAMD_MILTER_CMD_QUIT: + break; + case RSPAMD_MILTER_CMD_RCPT: + break; + case RSPAMD_MILTER_CMD_DATA: + break; + default: + break; + } + + return TRUE; +} + static gboolean rspamd_milter_consume_input (struct rspamd_milter_session *session, struct rspamd_milter_private *priv) { const guchar *p, *end; + GError *err; p = priv->parser.buf->str + priv->parser.pos; end = priv->parser.buf->str + priv->parser.buf->len; while (p < end) { switch (priv->parser.state) { - case st_read_cmd: - priv->parser.cur_cmd = *p; - priv->parser.state = st_len_1; - priv->parser.datalen = 0; - p++; - break; case st_len_1: /* The first length byte in big endian order */ + priv->parser.datalen = 0; priv->parser.datalen |= *p << 24; priv->parser.state = st_len_2; p++; @@ -92,14 +138,62 @@ rspamd_milter_consume_input (struct rspamd_milter_session *session, case st_len_4: /* The fourth length byte in big endian order */ priv->parser.datalen |= *p; + priv->parser.state = st_read_cmd; + p++; + break; + case st_read_cmd: + priv->parser.cur_cmd = *p; priv->parser.state = st_read_data; + + if (priv->parser.datalen < 1) { + err = g_error_new (rspamd_milter_quark (), EINVAL, + "Command length is too short"); + rspamd_milter_on_protocol_error (session, priv, err); + + return FALSE; + } + else { + /* Eat command itself */ + priv->parser.datalen --; + } + p++; break; case st_read_data: /* We might need some more data in buffer for further steps */ + if (priv->parser.buf->allocated < priv->parser.datalen) { + priv->parser.buf = rspamd_fstring_grow (priv->parser.buf, + priv->parser.pos + priv->parser.datalen); + /* This can realloc buffer */ + p = priv->parser.buf->str + priv->parser.pos; + rspamd_milter_plan_io (session, priv, EV_READ); + goto end; + } + else { + /* We may have the full command available */ + if (p + priv->parser.datalen <= end) { + /* We need to process command */ + if (!rspamd_milter_process_command (session, priv)) { + return FALSE; + } + + p += priv->parser.datalen; + priv->parser.state = st_len_1; + priv->parser.cur_cmd = '\0'; + } + else { + /* Need to read more */ + rspamd_milter_plan_io (session, priv, EV_READ); + goto end; + } + } break; } } +end: + + priv->parser.pos = p - (const guchar *)priv->parser.buf->str; + return TRUE; } static gboolean @@ -113,13 +207,13 @@ rspamd_milter_handle_session (struct rspamd_milter_session *session, switch (priv->state) { case RSPAMD_MILTER_READ_MORE: - if (priv->parser.pos >= priv->parser.buf->allocated) { + if (priv->parser.buf->len >= priv->parser.buf->allocated) { priv->parser.buf = rspamd_fstring_grow (priv->parser.buf, - priv->parser.pos * 2); + priv->parser.buf->len * 2); } - r = read (priv->fd, priv->parser.buf->str + priv->parser.pos, - priv->parser.buf->allocated - priv->parser.pos); + r = read (priv->fd, priv->parser.buf->str + priv->parser.buf->len, + priv->parser.buf->allocated - priv->parser.buf->len); if (r == -1) { if (errno == EAGAIN || errno == EINTR) { @@ -169,7 +263,7 @@ rspamd_milter_handle_socket (gint fd, const struct timeval *tv, priv->ud = ud; priv->fin_cb = finish_cb; priv->err_cb = error_cb; - priv->parser.state = st_read_cmd; + priv->parser.state = st_len_1; priv->parser.buf = rspamd_fstring_sized_new (100); priv->ev_base = ev_base; priv->state = RSPAMD_MILTER_READ_MORE; diff --git a/src/libserver/milter_internal.h b/src/libserver/milter_internal.h index fed2e6f11..933044772 100644 --- a/src/libserver/milter_internal.h +++ b/src/libserver/milter_internal.h @@ -21,11 +21,11 @@ #include <event.h> enum rspamd_milter_state { - st_read_cmd, - st_len_1, + st_len_1 = 0, st_len_2, st_len_3, st_len_4, + st_read_cmd, st_read_data }; @@ -64,4 +64,22 @@ struct rspamd_milter_private { int fd; }; +enum rspamd_milter_io_cmd { + RSPAMD_MILTER_CMD_ABORT = 'A', /* Abort */ + RSPAMD_MILTER_CMD_BODY = 'B', /* Body chunk */ + RSPAMD_MILTER_CMD_CONNECT = 'C', /* Connection information */ + RSPAMD_MILTER_CMD_MACRO = 'D', /* Define macro */ + RSPAMD_MILTER_CMD_BODYEOB = 'E', /* final body chunk (end of message) */ + RSPAMD_MILTER_CMD_HELO = 'H', /* HELO/EHLO */ + RSPAMD_MILTER_CMD_QUIT_NC = 'K', /* QUIT but new connection follows */ + RSPAMD_MILTER_CMD_HEADER = 'L', /* Header */ + RSPAMD_MILTER_CMD_MAIL = 'M', /* MAIL from */ + RSPAMD_MILTER_CMD_EOH = 'N', /* EOH */ + RSPAMD_MILTER_CMD_OPTNEG = 'O', /* Option negotiation */ + RSPAMD_MILTER_CMD_QUIT = 'Q', /* QUIT */ + RSPAMD_MILTER_CMD_RCPT = 'R', /* RCPT to */ + RSPAMD_MILTER_CMD_DATA = 'T', /* DATA */ + RSPAMD_MILTER_CMD_UNKNOWN = 'U' /* Any unknown command */ +}; + #endif |