aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorVsevolod Stakhov <vsevolod@highsecure.ru>2017-08-12 21:57:01 +0100
committerVsevolod Stakhov <vsevolod@highsecure.ru>2017-08-12 21:57:01 +0100
commiteaddd226401393ff96b1f9505e1b530c59bc5504 (patch)
tree0f1070e67e7544c1f464fca4ec5bf487ff24a8bd
parente1445e75f907d3a7bf8f80a7e1039044b05794be (diff)
downloadrspamd-eaddd226401393ff96b1f9505e1b530c59bc5504.tar.gz
rspamd-eaddd226401393ff96b1f9505e1b530c59bc5504.zip
[Feature] Add gzip compression support for rspamd controller
-rw-r--r--CMakeLists.txt5
-rw-r--r--src/libserver/worker_util.c80
-rw-r--r--src/libutil/http.c8
-rw-r--r--src/libutil/http.h1
4 files changed, 85 insertions, 9 deletions
diff --git a/CMakeLists.txt b/CMakeLists.txt
index e6a89c7cf..2deefa15f 100644
--- a/CMakeLists.txt
+++ b/CMakeLists.txt
@@ -581,9 +581,6 @@ IF(CMAKE_SYSTEM_NAME STREQUAL "Linux")
LIST(APPEND CMAKE_REQUIRED_LIBRARIES resolv)
LIST(APPEND CMAKE_REQUIRED_LIBRARIES nsl)
MESSAGE(STATUS "Configuring for Linux")
- IF(ENABLE_STATIC MATCHES "ON")
- LIST(APPEND CMAKE_REQUIRED_LIBRARIES z)
- ENDIF()
IF(EXISTS "/etc/debian_version")
SET(LINUX_START_SCRIPT "rspamd_debian.in")
ELSE(EXISTS "/etc/debian_version")
@@ -702,6 +699,8 @@ ProcessPackage(LIBSSL LIBRARY ssl INCLUDE ssl.h INCLUDE_SUFFIXES include/openssl
ROOT ${OPENSSL_ROOT_DIR} MODULES openssl libssl)
ProcessPackage(MAGIC LIBRARY magic INCLUDE magic.h INCLUDE_SUFFIXES include/libmagic
ROOT ${LIBMAGIC_ROOT_DIR} MODULES magic)
+ProcessPackage(LIBZ LIBRARY z INCLUDE zlib.h INCLUDE_SUFFIXES include/zlib
+ ROOT ${LIBZ_ROOT_DIR} MODULES z)
IF(ENABLE_HYPERSCAN MATCHES "ON")
ProcessPackage(HYPERSCAN LIBRARY hs INCLUDE hs.h INCLUDE_SUFFIXES
diff --git a/src/libserver/worker_util.c b/src/libserver/worker_util.c
index fa510d335..77770ae3f 100644
--- a/src/libserver/worker_util.c
+++ b/src/libserver/worker_util.c
@@ -42,6 +42,7 @@
#ifdef HAVE_LIBUTIL_H
#include <libutil.h>
#endif
+#include "zlib.h"
static void rspamd_worker_ignore_signal (int signo);
/**
@@ -313,9 +314,6 @@ rspamd_worker_stop_accept (struct rspamd_worker *worker)
{
GList *cur;
struct event *events;
- GHashTableIter it;
- struct rspamd_worker_signal_handler *sigh;
- gpointer k, v;
struct rspamd_map *map;
/* Remove all events */
@@ -368,6 +366,73 @@ rspamd_worker_stop_accept (struct rspamd_worker *worker)
}
}
+static rspamd_fstring_t *
+rspamd_controller_maybe_compress (struct rspamd_http_connection_entry *entry,
+ rspamd_fstring_t *buf, struct rspamd_http_message *msg)
+{
+ z_stream strm;
+ gint rc;
+ rspamd_fstring_t *comp;
+ gchar *p;
+ gsize remain;
+
+ if (entry->support_gzip) {
+ memset (&strm, 0, sizeof (strm));
+ rc = deflateInit2 (&strm, Z_DEFAULT_COMPRESSION, Z_DEFLATED,
+ MAX_WBITS + 16, MAX_MEM_LEVEL - 1, Z_DEFAULT_STRATEGY);
+
+ if (rc != Z_OK) {
+ return buf;
+ }
+
+ comp = rspamd_fstring_sized_new (MIN (buf->len, 32768));
+
+ strm.avail_in = buf->len;
+ strm.next_in = (guchar *)buf->str;
+ p = comp->str;
+ remain = comp->allocated;
+
+ while (strm.avail_in != 0) {
+ strm.avail_out = remain;
+ strm.next_out = p;
+
+ rc = deflate (&strm, Z_FINISH);
+
+ if (rc != Z_OK) {
+ if (rc == Z_STREAM_END) {
+ break;
+ }
+ else {
+ rspamd_fstring_free (comp);
+ deflateEnd (&strm);
+
+ return buf;
+ }
+ }
+
+ comp->len = strm.total_out;
+
+ if (strm.avail_out == 0 && strm.avail_in != 0) {
+ /* Need to allocate more */
+ remain = comp->len;
+ comp = rspamd_fstring_grow (comp, comp->allocated +
+ strm.avail_in + 10);
+ p = comp->str + remain;
+ remain = comp->allocated - remain;
+ }
+ }
+
+ deflateEnd (&strm);
+ comp->len = strm.total_out;
+ rspamd_fstring_free (buf); /* We replace buf with its compressed version */
+ rspamd_http_message_add_header (msg, "Content-Encoding", "gzip");
+
+ return comp;
+ }
+
+ return buf;
+}
+
void
rspamd_controller_send_error (struct rspamd_http_connection_entry *entry,
gint code, const gchar *error_msg, ...)
@@ -387,7 +452,8 @@ rspamd_controller_send_error (struct rspamd_http_connection_entry *entry,
msg->code = code;
reply = rspamd_fstring_sized_new (msg->status->len + 16);
rspamd_printf_fstring (&reply, "{\"error\":\"%V\"}", msg->status);
- rspamd_http_message_set_body_from_fstring_steal (msg, reply);
+ rspamd_http_message_set_body_from_fstring_steal (msg,
+ rspamd_controller_maybe_compress (entry, reply, msg));
rspamd_http_connection_reset (entry->conn);
rspamd_http_router_insert_headers (entry->rt, msg);
rspamd_http_connection_write_message (entry->conn,
@@ -413,7 +479,8 @@ rspamd_controller_send_string (struct rspamd_http_connection_entry *entry,
msg->code = 200;
msg->status = rspamd_fstring_new_init ("OK", 2);
reply = rspamd_fstring_new_init (str, strlen (str));
- rspamd_http_message_set_body_from_fstring_steal (msg, reply);
+ rspamd_http_message_set_body_from_fstring_steal (msg,
+ rspamd_controller_maybe_compress (entry, reply, msg));
rspamd_http_connection_reset (entry->conn);
rspamd_http_router_insert_headers (entry->rt, msg);
rspamd_http_connection_write_message (entry->conn,
@@ -440,7 +507,8 @@ rspamd_controller_send_ucl (struct rspamd_http_connection_entry *entry,
msg->status = rspamd_fstring_new_init ("OK", 2);
reply = rspamd_fstring_sized_new (BUFSIZ);
rspamd_ucl_emit_fstring (obj, UCL_EMIT_JSON_COMPACT, &reply);
- rspamd_http_message_set_body_from_fstring_steal (msg, reply);
+ rspamd_http_message_set_body_from_fstring_steal (msg,
+ rspamd_controller_maybe_compress (entry, reply, msg));
rspamd_http_connection_reset (entry->conn);
rspamd_http_router_insert_headers (entry->rt, msg);
rspamd_http_connection_write_message (entry->conn,
diff --git a/src/libutil/http.c b/src/libutil/http.c
index 2ef897da8..7ff981e4e 100644
--- a/src/libutil/http.c
+++ b/src/libutil/http.c
@@ -3209,6 +3209,7 @@ rspamd_http_router_finish_handler (struct rspamd_http_connection *conn,
GError *err;
rspamd_ftok_t lookup;
+ const rspamd_ftok_t *encoding;
struct http_parser_url u;
guint i;
rspamd_regexp_t *re;
@@ -3280,6 +3281,13 @@ rspamd_http_router_finish_handler (struct rspamd_http_connection *conn,
entry->is_reply = TRUE;
+ encoding = rspamd_http_message_find_header (msg, "Accept-Encoding");
+
+ if (encoding && rspamd_substring_search (encoding->begin, encoding->len,
+ "gzip", 4) != -1) {
+ entry->support_gzip = TRUE;
+ }
+
if (handler != NULL) {
return handler (entry, msg);
}
diff --git a/src/libutil/http.h b/src/libutil/http.h
index e2e332d6f..3696c4e18 100644
--- a/src/libutil/http.h
+++ b/src/libutil/http.h
@@ -124,6 +124,7 @@ struct rspamd_http_connection_entry {
struct rspamd_http_connection *conn;
gpointer ud;
gboolean is_reply;
+ gboolean support_gzip;
struct rspamd_http_connection_entry *prev, *next;
};