aboutsummaryrefslogtreecommitdiffstats
path: root/src
diff options
context:
space:
mode:
Diffstat (limited to 'src')
-rw-r--r--src/client/CMakeLists.txt8
-rw-r--r--src/client/rspamc.c361
2 files changed, 369 insertions, 0 deletions
diff --git a/src/client/CMakeLists.txt b/src/client/CMakeLists.txt
new file mode 100644
index 000000000..186b458f3
--- /dev/null
+++ b/src/client/CMakeLists.txt
@@ -0,0 +1,8 @@
+# rspamc
+SET(RSPAMCSRC rspamc.c)
+
+ADD_EXECUTABLE(rspamc ${RSPAMCSRC})
+SET_TARGET_PROPERTIES(rspamc PROPERTIES COMPILE_FLAGS "-I.. -I../../lib")
+TARGET_LINK_LIBRARIES(rspamc rspamdclient)
+TARGET_LINK_LIBRARIES(rspamc ${CMAKE_REQUIRED_LIBRARIES})
+TARGET_LINK_LIBRARIES(rspamc ${GLIB2_LIBRARIES})
diff --git a/src/client/rspamc.c b/src/client/rspamc.c
new file mode 100644
index 000000000..1bae2970c
--- /dev/null
+++ b/src/client/rspamc.c
@@ -0,0 +1,361 @@
+/*
+ * Copyright (c) 2011, Vsevolod Stakhov
+ * All rights reserved.
+ *
+ * Redistribution and use in source and binary forms, with or without
+ * modification, are permitted provided that the following conditions are met:
+ * * Redistributions of source code must retain the above copyright
+ * notice, this list of conditions and the following disclaimer.
+ * * Redistributions in binary form must reproduce the above copyright
+ * notice, this list of conditions and the following disclaimer in the
+ * documentation and/or other materials provided with the distribution.
+ *
+ * THIS SOFTWARE IS PROVIDED ''AS IS'' AND ANY
+ * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
+ * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
+ * DISCLAIMED. IN NO EVENT SHALL BE LIABLE FOR ANY
+ * DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
+ * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
+ * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
+ * ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
+ * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
+ * SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
+ */
+
+#include "config.h"
+#include "librspamdclient.h"
+
+#define PRINT_FUNC printf
+
+#define DEFAULT_PORT 11333
+#define DEFAULT_CONTROL_PORT 11334
+
+static gchar *connect_str = "localhost";
+static gchar *password;
+static gchar *statfile;
+static gchar *ip;
+static gdouble weight;
+static gboolean pass_all;
+static gboolean tty = FALSE;
+
+static GOptionEntry entries[] =
+{
+ { "connect", 'h', 0, G_OPTION_ARG_STRING, &connect_str, "Specify host and port", NULL },
+ { "password", 'P', 0, G_OPTION_ARG_STRING, &password, "Specify control password", NULL },
+ { "statfile", 's', 0, G_OPTION_ARG_STRING, &statfile, "Statfile to learn (symbol name)", NULL },
+ { "weight", 'w', 0, G_OPTION_ARG_DOUBLE, &weight, "Weight for fuzzy operations", NULL },
+ { "pass", 'p', 0, G_OPTION_ARG_NONE, &pass_all, "Pass all filters", NULL },
+ { "ip", 'i', 0, G_OPTION_ARG_STRING, &ip, "Emulate that message was received from specified ip address", NULL },
+ { NULL, 0, 0, G_OPTION_ARG_NONE, NULL, NULL, NULL }
+};
+
+enum rspamc_command {
+ RSPAMC_COMMAND_UNKNOWN = 0,
+ RSPAMC_COMMAND_SYMBOLS,
+ RSPAMC_COMMAND_LEARN
+};
+
+/*
+ * Parse command line
+ */
+static void
+read_cmd_line (gint *argc, gchar ***argv)
+{
+ GError *error = NULL;
+ GOptionContext *context;
+
+ /* Prepare parser */
+ context = g_option_context_new ("- run rspamc client");
+ g_option_context_set_summary (context, "Summary:\n Rspamd client version " RVERSION "\n Release id: " RID);
+ g_option_context_add_main_entries (context, entries, NULL);
+
+ /* Parse options */
+ if (!g_option_context_parse (context, argc, argv, &error)) {
+ fprintf (stderr, "option parsing failed: %s\n", error->message);
+ exit (EXIT_FAILURE);
+ }
+
+ /* Argc and argv are shifted after this function */
+}
+
+/*
+ * Check rspamc command from string (used for arguments parsing)
+ */
+static enum rspamc_command
+check_rspamc_command (const gchar *cmd)
+{
+ if (g_ascii_strcasecmp (cmd, "SYMBOLS") == 0 ||
+ g_ascii_strcasecmp (cmd, "CHECK") == 0 ||
+ g_ascii_strcasecmp (cmd, "REPORT") == 0) {
+ /* These all are symbols, don't use other commands */
+ return RSPAMC_COMMAND_SYMBOLS;
+ }
+ else if (g_ascii_strcasecmp (cmd, "LEARN") == 0) {
+ return RSPAMC_COMMAND_LEARN;
+ }
+
+ return RSPAMC_COMMAND_UNKNOWN;
+}
+
+/*
+ * Parse connect_str and add server to librspamdclient
+ */
+static void
+add_rspamd_server (gboolean is_control)
+{
+ gchar **vec, *err_str;
+ guint16 port;
+ GError *err = NULL;
+
+ if (connect_str == NULL) {
+ fprintf (stderr, "cannot connect to rspamd server - empty string\n");
+ exit (EXIT_FAILURE);
+ }
+ vec = g_strsplit_set (connect_str, ":", 2);
+ if (vec == NULL || *vec == NULL) {
+ fprintf (stderr, "cannot connect to rspamd server: %s\n", connect_str);
+ exit (EXIT_FAILURE);
+ }
+
+ if (vec[1] == NULL) {
+ port = is_control ? DEFAULT_CONTROL_PORT : DEFAULT_PORT;
+ }
+ else {
+ port = strtoul (vec[1], &err_str, 10);
+ if (*err_str != '\0') {
+ fprintf (stderr, "cannot connect to rspamd server: %s, at pos %s\n", connect_str, err_str);
+ exit (EXIT_FAILURE);
+ }
+ }
+
+ if (! rspamd_add_server (vec[0], port, port, &err)) {
+ fprintf (stderr, "cannot connect to rspamd server: %s, error: %s\n", connect_str, err->message);
+ exit (EXIT_FAILURE);
+ }
+}
+
+static void
+show_metric_result (gpointer key, gpointer value, gpointer ud)
+{
+ struct rspamd_metric *metric = value;
+ struct rspamd_symbol *s;
+ GList *cur;
+ GHashTableIter it;
+ gpointer k, v;
+ gboolean first;
+
+ if (metric->is_skipped) {
+ PRINT_FUNC ("%s: Skipped\n", key);
+ }
+ else {
+ if (tty) {
+ PRINT_FUNC ("\033[1m%s:\033[0m %s [ %.2f / %.2f ]\n", key,
+ metric->score > metric->required_score ? "True" : "False",
+ metric->score, metric->required_score);
+ }
+ else {
+ PRINT_FUNC ("%s: %s [ %.2f / %.2f ]\n", key,
+ metric->score > metric->required_score ? "True" : "False",
+ metric->score, metric->required_score);
+ }
+ if (tty) {
+ if (metric->action) {
+ PRINT_FUNC ("\033[1mAction:\033[0m %s\n", metric->action);
+ }
+ PRINT_FUNC ("\033[1mSymbols: \033[0m");
+ }
+ else {
+ if (metric->action) {
+ PRINT_FUNC ("Action: %s\n", metric->action);
+ }
+ PRINT_FUNC ("Symbols: ");
+ }
+ if (metric->symbols) {
+ first = TRUE;
+ g_hash_table_iter_init (&it, metric->symbols);
+ while (g_hash_table_iter_next (&it, &k, &v)) {
+ s = v;
+ if (! first) {
+ PRINT_FUNC (", ");
+ }
+ else {
+ first = FALSE;
+ }
+ PRINT_FUNC ("%s(%.2f)", s->name, s->weight);
+ if (s->options) {
+ PRINT_FUNC ("(");
+ cur = g_list_first (s->options);
+ while (cur) {
+ if (cur->next) {
+ PRINT_FUNC ("%s,", cur->data);
+ }
+ else {
+ PRINT_FUNC ("%s)", cur->data);
+ }
+ cur = g_list_next (cur);
+ }
+ }
+
+ }
+ }
+ PRINT_FUNC ("\n");
+ }
+}
+
+static void
+print_rspamd_result (struct rspamd_result *res)
+{
+ GHashTableIter it;
+ gpointer k, v;
+
+ g_assert (res != 0);
+
+ if (tty) {
+ printf ("\033[1m");
+ }
+ PRINT_FUNC ("Results for host: %s\n\n", connect_str);
+ if (tty) {
+ printf ("\033[0m");
+ }
+ g_hash_table_foreach (res->metrics, show_metric_result, NULL);
+ /* Show other headers */
+ g_hash_table_iter_init (&it, res->headers);
+ PRINT_FUNC ("\n");
+ while (g_hash_table_iter_next (&it, &k, &v)) {
+ if (tty) {
+ PRINT_FUNC ("\033[1m%s:\033[0m %s\n", k, v);
+ }
+ else {
+ PRINT_FUNC ("%s: %s\n", k, v);
+ }
+ }
+ PRINT_FUNC ("\n");
+}
+
+static void
+add_options (GHashTable *opts)
+{
+ if (ip != NULL) {
+ g_hash_table_insert (opts, "Ip", ip);
+ }
+ if (pass_all) {
+ g_hash_table_insert (opts, "Pass", "all");
+ }
+}
+
+/*
+ * Scan STDIN
+ */
+static void
+scan_rspamd_stdin ()
+{
+ gchar *in_buf;
+
+ gint r = 0, len;
+ GError *err = NULL;
+ struct rspamd_result *res;
+ GHashTable *opts;
+
+ /* Init options hash */
+ opts = g_hash_table_new (g_str_hash, g_str_equal);
+ add_options (opts);
+ /* Add server */
+ add_rspamd_server (FALSE);
+
+ /* Allocate input buffer */
+ len = BUFSIZ;
+ in_buf = g_malloc (len);
+
+ /* Read stdin */
+ while (!feof (stdin)) {
+ r += fread (in_buf + r, 1, len - r, stdin);
+ if (len - r < len / 2) {
+ /* Grow buffer */
+ len *= 2;
+ in_buf = g_realloc (in_buf, len);
+ }
+ }
+ res = rspamd_scan_memory (in_buf, r, opts, &err);
+ g_hash_table_destroy (opts);
+ if (err != NULL) {
+ fprintf (stderr, "cannot scan message: %s\n", err->message);
+ exit (EXIT_FAILURE);
+ }
+ print_rspamd_result (res);
+}
+
+static void
+scan_rspamd_file (const gchar *file)
+{
+ GError *err = NULL;
+ struct rspamd_result *res;
+ GHashTable *opts;
+
+ /* Init options hash */
+ opts = g_hash_table_new (g_str_hash, g_str_equal);
+ add_options (opts);
+
+ /* Add server */
+ add_rspamd_server (FALSE);
+
+ res = rspamd_scan_file (file, opts, &err);
+ g_hash_table_destroy (opts);
+ if (err != NULL) {
+ fprintf (stderr, "cannot scan message: %s\n", err->message);
+ exit (EXIT_FAILURE);
+ }
+ print_rspamd_result (res);
+}
+
+gint
+main (gint argc, gchar **argv, gchar **env)
+{
+ enum rspamc_command cmd;
+ gint i;
+
+ rspamd_client_init ();
+
+ read_cmd_line (&argc, &argv);
+ tty = isatty (STDOUT_FILENO);
+ /* Now read other args from argc and argv */
+ if (argc == 1) {
+ /* No args, just read stdin */
+ scan_rspamd_stdin ();
+ }
+ else if (argc == 2) {
+ /* One argument is whether command or filename */
+ if ((cmd = check_rspamc_command (argv[1])) != RSPAMC_COMMAND_UNKNOWN) {
+ /* In case of command read stdin */
+ if (cmd == RSPAMC_COMMAND_SYMBOLS) {
+ scan_rspamd_stdin ();
+ }
+ else {
+ /* XXX: implement this */
+ }
+ }
+ else {
+ scan_rspamd_file (argv[1]);
+ }
+ }
+ else {
+ if ((cmd = check_rspamc_command (argv[1])) != RSPAMC_COMMAND_UNKNOWN) {
+ /* In case of command read arguments starting from 2 */
+ for (i = 2; i < argc; i ++) {
+ if (cmd == RSPAMC_COMMAND_SYMBOLS) {
+ scan_rspamd_file (argv[i]);
+ }
+ else {
+ /* XXX: implement this */
+ }
+ }
+ }
+ else {
+ for (i = 1; i < argc; i ++) {
+ scan_rspamd_file (argv[i]);
+ }
+ }
+ }
+
+
+ return 0;
+}