diff options
author | Vsevolod Stakhov <vsevolod@highsecure.ru> | 2014-09-22 17:27:05 +0100 |
---|---|---|
committer | Vsevolod Stakhov <vsevolod@highsecure.ru> | 2014-09-22 17:27:05 +0100 |
commit | 17e0a0778389ab4741747beb8cae5eedab0327a3 (patch) | |
tree | 8b1eecb667ce58959e750fddea81e5d9a712ef31 | |
parent | 05a46b8f2181daf06170613ce382c0c7ba58c7ad (diff) | |
download | rspamd-17e0a0778389ab4741747beb8cae5eedab0327a3.tar.gz rspamd-17e0a0778389ab4741747beb8cae5eedab0327a3.zip |
Implement the proper reset of HTTP messages.
-rw-r--r-- | src/libutil/http.c | 73 | ||||
-rw-r--r-- | src/libutil/ref.h | 16 | ||||
-rw-r--r-- | src/worker.c | 13 |
3 files changed, 75 insertions, 27 deletions
diff --git a/src/libutil/http.c b/src/libutil/http.c index 5cd5e2e12..debbad7bc 100644 --- a/src/libutil/http.c +++ b/src/libutil/http.c @@ -27,11 +27,15 @@ #include "util.h" #include "printf.h" #include "logger.h" +#include "ref.h" #include <limits.h> struct rspamd_http_connection_private { - GString *buf; + struct _rspamd_http_privbuf { + GString *data; + ref_entry_t ref; + } *buf; gboolean new_header; struct rspamd_http_header *header; struct http_parser parser; @@ -79,6 +83,17 @@ http_error_quark (void) return g_quark_from_static_string ("http-error-quark"); } +static void +rspamd_http_privbuf_dtor (gpointer ud) +{ + struct _rspamd_http_privbuf *p = (struct _rspamd_http_privbuf *)ud; + + if (p->data) { + g_string_free (p->data, TRUE); + } + g_slice_free1 (sizeof (struct _rspamd_http_privbuf), p); +} + static const gchar * rspamd_http_code_to_str (gint code) { @@ -642,12 +657,16 @@ rspamd_http_event_handler (int fd, short what, gpointer ud) { struct rspamd_http_connection *conn = (struct rspamd_http_connection *)ud; struct rspamd_http_connection_private *priv; + struct _rspamd_http_privbuf *pbuf; GString *buf; gssize r; GError *err; priv = conn->priv; - buf = priv->buf; + pbuf = priv->buf; + REF_RETAIN (pbuf); + rspamd_http_connection_ref (conn); + buf = priv->buf->data; if (what == EV_READ) { r = read (fd, buf->str, buf->allocated_len); @@ -658,10 +677,15 @@ rspamd_http_event_handler (int fd, short what, gpointer ud) strerror (errno)); conn->error_handler (conn, err); g_error_free (err); + + REF_RELEASE (pbuf); + rspamd_http_connection_unref (conn); + return; } else if (r == 0) { if (conn->finished) { + REF_RELEASE (pbuf); rspamd_http_connection_unref (conn); return; } @@ -671,12 +695,16 @@ rspamd_http_event_handler (int fd, short what, gpointer ud) "IO read error: unexpected EOF"); conn->error_handler (conn, err); g_error_free (err); + + REF_RELEASE (pbuf); + rspamd_http_connection_unref (conn); + return; } } else { buf->len = r; - rspamd_http_connection_ref (conn); + if (http_parser_execute (&priv->parser, &priv->parser_cb, buf->str, r) != (size_t)r) { err = g_error_new (HTTP_ERROR, priv->parser.http_errno, @@ -684,10 +712,12 @@ rspamd_http_event_handler (int fd, short what, gpointer ud) http_errno_description (priv->parser.http_errno)); conn->error_handler (conn, err); g_error_free (err); + + REF_RELEASE (pbuf); rspamd_http_connection_unref (conn); + return; } - rspamd_http_connection_unref (conn); } } else if (what == EV_TIMEOUT) { @@ -695,13 +725,19 @@ rspamd_http_event_handler (int fd, short what, gpointer ud) "IO timeout"); rspamd_http_connection_ref (conn); conn->error_handler (conn, err); - rspamd_http_connection_unref (conn); g_error_free (err); + + REF_RELEASE (pbuf); + rspamd_http_connection_unref (conn); + return; } else if (what == EV_WRITE) { rspamd_http_write_helper (conn); } + + REF_RELEASE (pbuf); + rspamd_http_connection_unref (conn); } static void @@ -773,8 +809,7 @@ rspamd_http_connection_reset (struct rspamd_http_connection *conn) /* Clear priv */ event_del (&priv->ev); if (priv->buf != NULL) { - g_string_free (priv->buf, TRUE); - priv->buf = NULL; + REF_RELEASE (priv->buf); } rspamd_http_parser_reset (conn); @@ -817,7 +852,9 @@ rspamd_http_connection_read_message (struct rspamd_http_connection *conn, priv->ptv = &priv->tv; } priv->header = NULL; - priv->buf = g_string_sized_new (BUFSIZ); + priv->buf = g_slice_alloc0 (sizeof (*priv->buf)); + REF_INIT_RETAIN (priv->buf, rspamd_http_privbuf_dtor); + priv->buf->data = g_string_sized_new (BUFSIZ); priv->new_header = TRUE; event_set (&priv->ev, @@ -842,6 +879,7 @@ rspamd_http_connection_write_message (struct rspamd_http_connection *conn, gchar datebuf[64], *pbody; gint i; gsize bodylen; + GString *buf; conn->fd = fd; conn->ud = ud; @@ -855,7 +893,10 @@ rspamd_http_connection_write_message (struct rspamd_http_connection *conn, priv->ptv = &priv->tv; } priv->header = NULL; - priv->buf = g_string_sized_new (128); + priv->buf = g_slice_alloc0 (sizeof (*priv->buf)); + REF_INIT_RETAIN (priv->buf, rspamd_http_privbuf_dtor); + priv->buf->data = g_string_sized_new (128); + buf = priv->buf->data; if (msg->method < HTTP_SYMBOLS) { if (msg->body == NULL || msg->body->len == 0) { @@ -899,7 +940,7 @@ rspamd_http_connection_write_message (struct rspamd_http_connection *conn, if (mime_type == NULL) { mime_type = "text/plain"; } - rspamd_printf_gstring (priv->buf, "HTTP/1.1 %d %s\r\n" + rspamd_printf_gstring (buf, "HTTP/1.1 %d %s\r\n" "Connection: close\r\n" "Server: %s\r\n" "Date: %s\r\n" @@ -915,19 +956,19 @@ rspamd_http_connection_write_message (struct rspamd_http_connection *conn, } else { /* Legacy spamd reply */ - rspamd_printf_gstring (priv->buf, "RSPAMD/1.3 0 EX_OK\r\n"); + rspamd_printf_gstring (buf, "RSPAMD/1.3 0 EX_OK\r\n"); } } else { /* Format request */ if (host == NULL && msg->host == NULL) { /* Fallback to HTTP/1.0 */ - rspamd_printf_gstring (priv->buf, "%s %v HTTP/1.0\r\n" + rspamd_printf_gstring (buf, "%s %v HTTP/1.0\r\n" "Content-Length: %z\r\n", http_method_str (msg->method), msg->url, bodylen); } else { - rspamd_printf_gstring (priv->buf, "%s %v HTTP/1.1\r\n" + rspamd_printf_gstring (buf, "%s %v HTTP/1.1\r\n" "Connection: close\r\n" "Host: %s\r\n" "Content-Length: %z\r\n", @@ -937,7 +978,7 @@ rspamd_http_connection_write_message (struct rspamd_http_connection *conn, } } /* Allocate iov */ - priv->wr_total = bodylen + priv->buf->len + 2; + priv->wr_total = bodylen + buf->len + 2; DL_FOREACH (msg->headers, hdr) { /* <name><: ><value><\r\n> */ @@ -948,8 +989,8 @@ rspamd_http_connection_write_message (struct rspamd_http_connection *conn, priv->wr_pos = 0; /* Now set up all iov */ - priv->out[0].iov_base = priv->buf->str; - priv->out[0].iov_len = priv->buf->len; + priv->out[0].iov_base = buf->str; + priv->out[0].iov_len = buf->len; i = 1; LL_FOREACH (msg->headers, hdr) { diff --git a/src/libutil/ref.h b/src/libutil/ref.h index a8016b16d..8caab5c8f 100644 --- a/src/libutil/ref.h +++ b/src/libutil/ref.h @@ -36,35 +36,47 @@ typedef struct ref_entry_s { } ref_entry_t; #define REF_INIT(obj, dtor_cb) do { \ + if ((obj) != NULL) { \ (obj)->ref.refcount = 0; \ (obj)->ref.dtor = (ref_dtor_cb_t)(dtor_cb); \ + } \ } while (0) #define REF_INIT_RETAIN(obj, dtor_cb) do { \ + if ((obj) != NULL) { \ (obj)->ref.refcount = 1; \ (obj)->ref.dtor = (ref_dtor_cb_t)(dtor_cb); \ + } \ } while (0) #ifdef HAVE_ATOMIC_BUILTINS #define REF_RETAIN(obj) do { \ + if ((obj) != NULL) { \ __sync_add_and_fetch (&(obj)->ref.refcount, 1); \ + } \ } while (0) #define REF_RELEASE(obj) do { \ - unsigned int rc = __sync_sub_and_fetch (&(obj)->ref.refcount, 1); \ - if (rc == 0 && (obj)->ref.dtor) { \ + if ((obj) != NULL) { \ + unsigned int _rc_priv = __sync_sub_and_fetch (&(obj)->ref.refcount, 1); \ + if (_rc_priv == 0 && (obj)->ref.dtor) { \ (obj)->ref.dtor (obj); \ } \ + } \ } while (0) #else #define REF_RETAIN(obj) do { \ + if ((obj) != NULL) { \ (obj)->ref.refcount ++; \ + } \ } while (0) #define REF_RELEASE(obj) do { \ + if ((obj) != NULL) { \ if (--(obj)->ref.refcount == 0 && (obj)->ref.dtor) { \ (obj)->ref.dtor (obj); \ } \ + } \ } while (0) #endif diff --git a/src/worker.c b/src/worker.c index 26e59ace2..30da517fe 100644 --- a/src/worker.c +++ b/src/worker.c @@ -141,7 +141,7 @@ rspamd_worker_error_handler (struct rspamd_http_connection *conn, GError *err) msg_info ("abnormally closing connection from: %s, error: %s", rspamd_inet_address_to_string (&task->client_addr), err->message); - if (task->state != CLOSING_CONNECTION && task->state != WRITING_REPLY) { + if (task->state != WRITING_REPLY && task->state != CLOSING_CONNECTION) { /* We still need to write a reply */ task->error_code = err->code; task->last_error = @@ -161,12 +161,13 @@ rspamd_worker_finish_handler (struct rspamd_http_connection *conn, { struct rspamd_task *task = (struct rspamd_task *) conn->ud; - if (task->state == CLOSING_CONNECTION) { + if (msg->type == HTTP_RESPONSE) { + /* We are done here */ msg_debug ("normally closing connection from: %s", rspamd_inet_address_to_string (&task->client_addr)); destroy_session (task->s); } - else if (task->state != WRITING_REPLY) { + else { /* * If all filters have finished their tasks, this function will trigger * writing a reply. @@ -174,12 +175,6 @@ rspamd_worker_finish_handler (struct rspamd_http_connection *conn, task->s->wanna_die = TRUE; check_session_pending (task->s); } - else { - /* - * We are going to write a reply to a client - */ - task->state = CLOSING_CONNECTION; - } return 0; } |