aboutsummaryrefslogtreecommitdiffstats
path: root/plugin.c
diff options
context:
space:
mode:
authorJohannes Berg <johannes@sipsolutions.net>2007-09-30 15:10:07 +0200
committerJohannes Berg <johannes@sipsolutions.net>2007-09-30 15:10:07 +0200
commit34ad5cf745e1a1c86243844433f9e2e202523a1e (patch)
tree21e57a8c54bacd21b483de0d500364f80f0ef9c4 /plugin.c
parentd30cef3c11f07889c2f8e86340b013174d28fc5c (diff)
downloaddovecot-antispam-34ad5cf745e1a1c86243844433f9e2e202523a1e.tar.gz
dovecot-antispam-34ad5cf745e1a1c86243844433f9e2e202523a1e.tar.xz
dovecot-antispam-34ad5cf745e1a1c86243844433f9e2e202523a1e.zip
seems to work
Diffstat (limited to '')
-rw-r--r--plugin.c185
1 files changed, 152 insertions, 33 deletions
diff --git a/plugin.c b/plugin.c
index 3f7eb97..944c446 100644
--- a/plugin.c
+++ b/plugin.c
@@ -25,43 +25,53 @@
* Install the plugin in the usual dovecot module location.
*/
-#include <unistd.h>
-#include <sys/wait.h>
#include <stdlib.h>
-#include <stdio.h>
-#include <errno.h>
-#include <fcntl.h>
-#include <time.h>
-#ifdef DEBUG
-#include <syslog.h>
-#endif
/* dovecot headers we need */
#include "lib.h"
-#include "mempool.h"
-#include "str.h"
-#include "strfuncs.h"
-#include "commands.h"
-#include "imap-search.h"
-#include "lib-storage/mail-storage.h"
-#include "mail-storage.h"
#include "client.h"
#include "ostream.h"
+#include "imap-search.h"
/* internal stuff we need */
#include "plugin.h"
-static struct signature *list_add(pool_t pool, struct signature *list)
+static pool_t global_pool;
+static char **trash_folders = NULL;
+static char *default_spam_folders[] = {
+ "SPAM",
+ NULL
+};
+static char **spam_folders = default_spam_folders;
+#ifdef BACKEND_WANT_SIGNATURE
+static char *signature_hdr = "X-DSPAM-Signature";
+#endif
+
+static struct strlist *list_add(pool_t pool, struct strlist *list)
{
- struct signature *n;
+ struct strlist *n;
- n = p_malloc(pool, sizeof(struct signature));
+ n = p_malloc(pool, sizeof(struct strlist));
n->next = list;
- n->sig = NULL;
+ n->str = NULL;
return n;
}
+static bool mailbox_in_list(struct mail_storage *storage, struct mailbox *box,
+ char **list)
+{
+ if (!list)
+ return FALSE;
+
+ while (*list) {
+ if (mailbox_equals(box, storage, *list))
+ return TRUE;
+ list++;
+ }
+
+ return FALSE;
+}
static void client_send_sendalive_if_needed(struct client *client)
{
@@ -79,9 +89,16 @@ static void client_send_sendalive_if_needed(struct client *client)
}
}
+#define GENERIC_ERROR -1
+#define SIGNATURE_MISSING -2
+#define BACKEND_FAILURE -3
+
+/* mostly copied from cmd-copy.c (notes added where changed) */
+/* MODIFIED: prototype to include "src_spam" */
static int fetch_and_copy(struct client *client,
struct mailbox_transaction_context *t,
- struct mail_search_arg *search_args)
+ struct mail_search_arg *search_args,
+ bool src_spam)
{
struct mail_search_context *search_ctx;
struct mailbox_transaction_context *src_trans;
@@ -90,6 +107,14 @@ static int fetch_and_copy(struct client *client,
struct mail *mail;
unsigned int copy_count = 0;
int ret;
+ /* MODIFIED: new variables */
+ pool_t pool = pool_alloconly_create("antispam-copy-pool", 1024);
+#ifdef BACKEND_WANT_SIGNATURE
+ const char *signature;
+ struct strlist *siglist = NULL;
+#else
+#error Not implemented
+#endif
src_trans = mailbox_transaction_begin(client->mailbox, 0);
search_ctx = mailbox_search_init(src_trans, NULL, search_args, NULL);
@@ -106,6 +131,19 @@ static int fetch_and_copy(struct client *client,
if ((++copy_count % COPY_CHECK_INTERVAL) == 0)
client_send_sendalive_if_needed(client);
+ /* MODIFIED: keep track of mail as we copy */
+#ifdef BACKEND_WANT_SIGNATURE
+ signature = mail_get_first_header(mail, signature_hdr);
+ if (is_empty_str(signature)) {
+ ret = SIGNATURE_MISSING;
+ break;
+ }
+ siglist = list_add(pool, siglist);
+ siglist->str = p_strdup(pool, signature);
+#else
+#error Not implemented
+#endif
+
keywords_list = mail_get_keywords(mail);
keywords = strarray_length(keywords_list) == 0 ? NULL :
mailbox_keywords_create(t, keywords_list);
@@ -122,10 +160,25 @@ static int fetch_and_copy(struct client *client,
if (mailbox_transaction_commit(&src_trans, 0) < 0)
ret = -1;
+ /* MODIFIED: pass to backend */
+#ifdef BACKEND_WANT_SIGNATURE
+ /* got all signatures now, pass them to backend if no errors */
+ if (ret == 0) {
+ ret = backend(pool, src_spam, siglist);
+ if (ret)
+ ret = BACKEND_FAILURE;
+ }
+#else
+#error Not implemented
+#endif
+ /* MODIFIED: kill pool */
+ pool_unref(pool);
+
return ret;
}
+/* mostly copied from cmd-copy.c (notes added where changed) */
static bool cmd_copy_antispam(struct client_command_context *cmd)
{
struct client *client = cmd->client;
@@ -136,6 +189,8 @@ static bool cmd_copy_antispam(struct client_command_context *cmd)
const char *messageset, *mailbox;
enum mailbox_sync_flags sync_flags = 0;
int ret;
+ /* MODIFIED: added variables */
+ bool dst_spam, src_spam;
/* <message set> <mailbox> */
if (!client_read_string_args(cmd, 2, &messageset, &mailbox))
@@ -156,9 +211,11 @@ static bool cmd_copy_antispam(struct client_command_context *cmd)
if (storage == NULL)
return TRUE;
- if (mailbox_equals(client->mailbox, storage, mailbox))
+ if (mailbox_equals(client->mailbox, storage, mailbox)) {
destbox = client->mailbox;
- else {
+ /* MODIFIED: don't try to reclassify on copy within folder */
+ return cmd_copy(cmd);
+ } else {
destbox = mailbox_open(storage, mailbox, NULL,
MAILBOX_OPEN_SAVEONLY |
MAILBOX_OPEN_FAST |
@@ -169,9 +226,29 @@ static bool cmd_copy_antispam(struct client_command_context *cmd)
}
}
+ /* MODIFIED: Trash detection */
+ if (mailbox_in_list(storage, client->mailbox, trash_folders) ||
+ mailbox_in_list(storage, destbox, trash_folders)) {
+ mailbox_close(&destbox);
+ return cmd_copy(cmd);
+ }
+
+ /* MODIFIED: from/to-SPAM detection */
+ src_spam = mailbox_in_list(storage, client->mailbox, spam_folders);
+ dst_spam = mailbox_in_list(storage, destbox, spam_folders);
+ /*
+ * "both spam" can happen with multiple spam folders,
+ * "none spam" is the common case where spam folders are not involved
+ */
+ if ((src_spam && dst_spam) ||
+ (!src_spam && !dst_spam)) {
+ mailbox_close(&destbox);
+ return cmd_copy(cmd);
+ }
+
t = mailbox_transaction_begin(destbox,
MAILBOX_TRANSACTION_FLAG_EXTERNAL);
- ret = fetch_and_copy(client, t, search_arg);
+ ret = fetch_and_copy(client, t, search_arg, src_spam);
if (ret <= 0)
mailbox_transaction_rollback(&t);
@@ -209,17 +286,16 @@ static bool cmd_append_antispam(struct client_command_context *cmd)
return FALSE;
storage = client_find_storage(cmd, &mailbox);
- if (storage == NULL)
+ if (!storage)
return FALSE;
- box =
- mailbox_open(storage, mailbox, NULL,
- MAILBOX_OPEN_FAST | MAILBOX_OPEN_KEEP_RECENT);
- if (box != NULL) {
- if (mailbox_equals(box, storage, "SPAM")) {
+ box = mailbox_open(storage, mailbox, NULL,
+ MAILBOX_OPEN_FAST | MAILBOX_OPEN_KEEP_RECENT);
+ if (box) {
+ if (mailbox_in_list(storage, box, spam_folders)) {
mailbox_close(&box);
return cmd_sync(cmd, 0, 0,
- "NO Cannot APPEND to SPAM box, sorry.");
+ "NO Cannot APPEND to spam folder.");
}
mailbox_close(&box);
@@ -229,8 +305,49 @@ static bool cmd_append_antispam(struct client_command_context *cmd)
}
-void dspam_init(void)
+void antispam_init(void)
{
+ char *tmp, **iter;
+
+ debug("antispam plugin intialising\n");
+
+ global_pool = pool_alloconly_create("antispam-pool", 1024);
+
+ tmp = getenv("ANTISPAM_TRASH");
+ if (tmp)
+ trash_folders = p_strsplit(global_pool, tmp, ";");
+
+ if (trash_folders) {
+ iter = trash_folders;
+ while (*iter) {
+ debug("antispam: \"%s\" is trash folder\n", *iter);
+ iter++;
+ }
+ } else
+ debug("antispam: no trash folders\n");
+
+ tmp = getenv("ANTISPAM_SPAM");
+ if (tmp)
+ spam_folders = p_strsplit(global_pool, tmp, ";");
+
+ if (spam_folders) {
+ iter = spam_folders;
+ while (*iter) {
+ debug("antispam: \"%s\" is spam folder\n", *iter);
+ iter++;
+ }
+ } else
+ debug("antispam: no spam folders\n");
+
+#ifdef BACKEND_WANT_SIGNATURE
+ tmp = getenv("ANTISPAM_SIGNATURE");
+ if (tmp)
+ signature_hdr = tmp;
+ debug("antispam: signature header line is \"%s\"\n", signature_hdr);
+#endif
+
+ backend_init(global_pool);
+
command_unregister("COPY");
command_unregister("APPEND");
command_unregister("UID COPY");
@@ -244,6 +361,8 @@ void dspam_init(void)
command_register(i_strdup("APPEND"), cmd_append_antispam);
}
-void dspam_deinit(void)
+void antispam_deinit(void)
{
+ backend_exit();
+ pool_unref(global_pool);
}