Hi,

In the interests of load-balancing and redundancy, we found it would be helpful if there was more control over how the list of spamd servers in spamd_addresses is used. The attached patch adds an MX-style priority as an optional third argument to each ip/port combination, so that it is possible to offload spam checking to a server or group of servers. and fall back onto another server should none of these be contactable.

The general idea is that as spam processing becomes a greater drain on system resources, it becomes practical to offload this burden onto a dedicated machine (which can then become a cluster of machines once the load becomes too high). This leaves the MX server free to concentrate on mail delivery and processing ACL checks. However in the event of a failure of the offload server, it would be useful to be able to fall back onto a backup running elsewhere, for example on localhost.

This patch preserves the usual spamd_address behaviour at each level of priority; the default priority for a server is 10 and a random(ish) server will be chosen to begin with, then the list cycles in order.

I have tested it a little to ensure the functionality is correct, and would be interested in any feedback and comments on the idea. The patch is against the 4.63 source tree.

Cheers,
--alex
diff -ur exim-4.63/src/spam.c exim-4.63-spam-mx/src/spam.c
--- exim-4.63/src/spam.c        Mon Jul 31 15:19:48 2006
+++ exim-4.63-spam-mx/src/spam.c        Tue Aug 22 16:46:08 2006
@@ -20,6 +20,18 @@
 uschar prev_user_name[128] = "";
 int spam_ok = 0;
 int spam_rc = 0;
+int rand_server = 0;
+int num_servers = 0;
+
+int sort_by_prio(const void *ia, const void *ib) {
+   spamd_address_container * const *a = ia;
+   const spamd_address_container * const *b = ib;
+   if ((*a)->priority - (*b)->priority)
+     return (*a)->priority - (*b)->priority;
+   /* pseudo-randomisation of equal priorities
+    * will only change once per second, as before */
+   return ((*a)->pos + rand_server)%num_servers - ((*b)->pos + 
rand_server)%num_servers;
+}  
 
 int spam(uschar **listptr) {
   int sep = 0;
@@ -92,14 +104,14 @@
   /* socket does not start with '/' -> network socket */
   if (*spamd_address != '/') {
     time_t now = time(NULL);
-    int num_servers = 0;
     int current_server = 0;
-    int start_server = 0;
     uschar *address = NULL;
     uschar *spamd_address_list_ptr = spamd_address;
     uschar address_buffer[256];
     spamd_address_container * spamd_address_vector[32];
 
+    num_servers = 0;
+    
     /* Check how many spamd servers we have
        and register their addresses */
     while ((address = string_nextinlist(&spamd_address_list_ptr, &sep,
@@ -108,9 +120,17 @@
 
       spamd_address_container *this_spamd =
         (spamd_address_container *)store_get(sizeof(spamd_address_container));
+      
+      /* set a default spamd priority of 10 */
+      this_spamd->priority = 10;
+      /* store current server number for randomisation purposes */
+      this_spamd->pos      = num_servers;
 
       /* grok spamd address and port */
-      if( sscanf(CS address, "%s %u", this_spamd->tcp_addr, 
&(this_spamd->tcp_port)) != 2 ) {
+      if( sscanf(CS address, "%s %u %i", 
+                 this_spamd->tcp_addr,
+                 &(this_spamd->tcp_port), 
+                 &(this_spamd->priority)) < 2 ) {
         log_write(0, LOG_MAIN,
           "spam acl condition: warning - invalid spamd address: '%s'", 
address);
         continue;
@@ -130,13 +150,15 @@
       return DEFER;
     };
 
-    current_server = start_server = (int)now % num_servers;
+    rand_server = (int)now % num_servers;
+    qsort(spamd_address_vector, num_servers, sizeof(*spamd_address_vector), 
sort_by_prio);
 
     while (1) {
 
-      debug_printf("trying server %s, port %u\n",
+      debug_printf("trying server %s, port %u, priority %d\n",
                    spamd_address_vector[current_server]->tcp_addr,
-                   spamd_address_vector[current_server]->tcp_port);
+                   spamd_address_vector[current_server]->tcp_port,
+                   spamd_address_vector[current_server]->priority);
 
       /* contact a spamd */
       if ( (spamd_sock = ip_socket(SOCK_STREAM, AF_INET)) < 0) {
@@ -161,9 +183,7 @@
          spamd_address_vector[current_server]->tcp_port,
          strerror(errno));
       current_server++;
-      if (current_server >= num_servers)
-        current_server = 0;
-      if (current_server == start_server) {
+      if (current_server == num_servers) {
         log_write(0, LOG_MAIN|LOG_PANIC, "spam acl condition: all spamd 
servers failed");
         (void)fclose(mbox_file);
         (void)close(spamd_sock);
diff -ur exim-4.63/src/spam.h exim-4.63-spam-mx/src/spam.h
--- exim-4.63/src/spam.h        Mon Jul 31 15:19:48 2006
+++ exim-4.63-spam-mx/src/spam.h        Tue Aug 22 13:35:20 2006
@@ -25,6 +25,8 @@
 typedef struct spamd_address_container {
   uschar tcp_addr[24];
   unsigned int tcp_port;
+  int priority;
+  int pos;
 } spamd_address_container;
 
 #endif
-- 
## List details at http://www.exim.org/mailman/listinfo/exim-users 
## Exim details at http://www.exim.org/
## Please use the Wiki with this list - http://www.exim.org/eximwiki/

Reply via email to