summaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
-rw-r--r--src/client/rspamc.c142
1 files changed, 132 insertions, 10 deletions
diff --git a/src/client/rspamc.c b/src/client/rspamc.c
index a8c65c2f8..3530fa66c 100644
--- a/src/client/rspamc.c
+++ b/src/client/rspamc.c
@@ -118,6 +118,17 @@ static GOptionEntry entries[] =
{ NULL, 0, 0, G_OPTION_ARG_NONE, NULL, NULL, NULL }
};
+/* Copy to avoid linking with librspamdserver */
+enum rspamd_metric_action {
+ METRIC_ACTION_REJECT = 0,
+ METRIC_ACTION_SOFT_REJECT,
+ METRIC_ACTION_REWRITE_SUBJECT,
+ METRIC_ACTION_ADD_HEADER,
+ METRIC_ACTION_GREYLIST,
+ METRIC_ACTION_NOACTION,
+ METRIC_ACTION_MAX
+};
+
static void rspamc_symbols_output (FILE *out, ucl_object_t *obj);
static void rspamc_uptime_output (FILE *out, ucl_object_t *obj);
static void rspamc_counters_output (FILE *out, ucl_object_t *obj);
@@ -786,6 +797,110 @@ rspamc_output_headers (FILE *out, struct rspamd_http_message *msg)
rspamd_fprintf (out, "\n");
}
+static gboolean
+rspamd_action_from_str (const gchar *data, gint *result)
+{
+ if (g_ascii_strncasecmp (data, "reject", sizeof ("reject") - 1) == 0) {
+ *result = METRIC_ACTION_REJECT;
+ }
+ else if (g_ascii_strncasecmp (data, "greylist",
+ sizeof ("greylist") - 1) == 0) {
+ *result = METRIC_ACTION_GREYLIST;
+ }
+ else if (g_ascii_strncasecmp (data, "add_header", sizeof ("add_header") -
+ 1) == 0) {
+ *result = METRIC_ACTION_ADD_HEADER;
+ }
+ else if (g_ascii_strncasecmp (data, "rewrite_subject",
+ sizeof ("rewrite_subject") - 1) == 0) {
+ *result = METRIC_ACTION_REWRITE_SUBJECT;
+ }
+ else if (g_ascii_strncasecmp (data, "add header", sizeof ("add header") -
+ 1) == 0) {
+ *result = METRIC_ACTION_ADD_HEADER;
+ }
+ else if (g_ascii_strncasecmp (data, "rewrite subject",
+ sizeof ("rewrite subject") - 1) == 0) {
+ *result = METRIC_ACTION_REWRITE_SUBJECT;
+ }
+ else if (g_ascii_strncasecmp (data, "soft_reject",
+ sizeof ("soft_reject") - 1) == 0) {
+ *result = METRIC_ACTION_SOFT_REJECT;
+ }
+ else if (g_ascii_strncasecmp (data, "soft reject",
+ sizeof ("soft reject") - 1) == 0) {
+ *result = METRIC_ACTION_SOFT_REJECT;
+ }
+ else {
+ return FALSE;
+ }
+ return TRUE;
+}
+
+static void
+rspamc_mime_output (FILE *out, ucl_object_t *result, GString *input)
+{
+ GMimeStream *stream;
+ GByteArray ar;
+ GMimeParser *parser;
+ GMimeMessage *message;
+ const ucl_object_t *metric, *res;
+ const gchar *action = "no action";
+ gint act;
+ gboolean is_spam = FALSE;
+ const gchar *hdr_scanned, *hdr_spam;
+
+ ar.data = input->str;
+ ar.len = input->len;
+
+ stream = g_mime_stream_mem_new_with_byte_array (&ar);
+ g_mime_stream_mem_set_owner (GMIME_STREAM_MEM (stream), FALSE);
+ parser = g_mime_parser_new_with_stream (stream);
+ g_object_unref (stream);
+
+ message = g_mime_parser_construct_message (parser);
+
+ if (message == NULL) {
+ rspamd_fprintf (stderr,"cannot construct mime from stream");
+ return;
+ }
+
+ metric = ucl_object_find_key (result, "default");
+
+ if (metric != NULL) {
+ res = ucl_object_find_key (metric, "action");
+
+ if (res) {
+ action = ucl_object_tostring (res);
+ }
+ }
+
+ rspamd_action_from_str (action, &act);
+
+ if (act < METRIC_ACTION_GREYLIST) {
+ is_spam = TRUE;
+ }
+
+ hdr_scanned = "rspamc " RVERSION;
+ g_mime_object_append_header (GMIME_OBJECT (message), "X-Spam-Scanner",
+ hdr_scanned);
+ if (is_spam) {
+ hdr_spam = "yes";
+ g_mime_object_append_header (GMIME_OBJECT (message), "X-Spam", hdr_spam);
+ }
+
+ g_mime_object_append_header (GMIME_OBJECT (message), "X-Spam-Action",
+ action);
+
+ /* Write message */
+ stream = g_mime_stream_file_new (out);
+ g_mime_stream_file_set_owner (GMIME_STREAM_FILE (stream), FALSE);
+ g_mime_object_write_to_stream (GMIME_OBJECT (message), stream);
+ g_object_unref (stream);
+ g_object_unref (message);
+ g_object_unref (parser);
+}
+
static void
rspamc_client_execute_cmd (struct rspamc_command *cmd, ucl_object_t *result,
GString *input)
@@ -815,18 +930,24 @@ rspamc_client_execute_cmd (struct rspamc_command *cmd, ucl_object_t *result,
out = fdopen (infd, "w");
if (result != NULL) {
- if (raw || cmd->command_output_func == NULL) {
- if (json) {
- ucl_out = ucl_object_emit (result, UCL_EMIT_JSON);
+
+ if (cmd->cmd == RSPAMC_COMMAND_SYMBOLS && mime_output && input) {
+ rspamc_mime_output (out, result, input);
+ }
+ else {
+ if (raw || cmd->command_output_func == NULL) {
+ if (json) {
+ ucl_out = ucl_object_emit (result, UCL_EMIT_JSON);
+ }
+ else {
+ ucl_out = ucl_object_emit (result, UCL_EMIT_CONFIG);
+ }
+ rspamd_fprintf (out, "%s", ucl_out);
+ free (ucl_out);
}
else {
- ucl_out = ucl_object_emit (result, UCL_EMIT_CONFIG);
+ cmd->command_output_func (out, result);
}
- rspamd_fprintf (out, "%s", ucl_out);
- free (ucl_out);
- }
- else {
- cmd->command_output_func (out, result);
}
ucl_object_unref (result);
@@ -835,7 +956,6 @@ rspamc_client_execute_cmd (struct rspamc_command *cmd, ucl_object_t *result,
rspamd_fprintf (out, "%s\n", err->message);
}
- rspamd_fprintf (out, "\n");
fflush (out);
fclose (out);
@@ -1028,6 +1148,7 @@ main (gint argc, gchar **argv, gchar **env)
}
ev_base = event_init ();
+ g_mime_init (0);
/* Now read other args from argc and argv */
if (argc == 1) {
@@ -1114,6 +1235,7 @@ main (gint argc, gchar **argv, gchar **env)
event_base_loop (ev_base, 0);
g_hash_table_destroy (kwattrs);
+ g_mime_shutdown ();
return 0;
}