diff src/alias.c @ 0:08114f7dcc23 0.2.21

this is masqmail-0.2.21 from oliver kurth
author meillo@marmaro.de
date Fri, 26 Sep 2008 17:05:23 +0200
parents
children 26e34ae9a3e3
line wrap: on
line diff
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/src/alias.c	Fri Sep 26 17:05:23 2008 +0200
@@ -0,0 +1,195 @@
+/*  MasqMail
+    Copyright (C) 2000-2001 Oliver Kurth
+
+    This program is free software; you can redistribute it and/or modify
+    it under the terms of the GNU General Public License as published by
+    the Free Software Foundation; either version 2 of the License, or
+    (at your option) any later version.
+
+    This program is distributed in the hope that it will be useful,
+    but WITHOUT ANY WARRANTY; without even the implied warranty of
+    MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+    GNU General Public License for more details.
+
+    You should have received a copy of the GNU General Public License
+    along with this program; if not, write to the Free Software
+    Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
+*/
+
+#include "masqmail.h"
+#include <fnmatch.h>
+
+gboolean addr_is_local(address *addr)
+{
+  GList *dom_node;
+  GList *addr_node;
+  address *a;
+
+  foreach(conf.local_hosts, dom_node){
+    if(addr->domain == NULL)
+      return TRUE;
+    if(fnmatch(dom_node->data, addr->domain, FNM_CASEFOLD) == 0){
+      foreach(conf.not_local_addresses,addr_node){
+	a = create_address_qualified(addr_node->data, TRUE, conf.host_name);
+	if(addr_isequal(a,addr)){
+	  destroy_address(a);
+	  return FALSE;
+	}
+	destroy_address(a);
+      }
+      return TRUE;
+    }
+  }
+  foreach(conf.local_addresses,addr_node){
+    a = create_address_qualified(addr_node->data, TRUE, conf.host_name);
+    if(addr_isequal(a,addr)){
+      destroy_address(a);
+      return TRUE;
+    }
+    destroy_address(a);
+  }
+  return FALSE;
+}
+
+static
+gboolean addr_isequal_alias(address *addr1, address *addr2)
+{
+  return
+    (conf.alias_local_cmp(addr1->local_part, addr2->local_part) == 0) &&
+    (strcasecmp(addr1->domain, addr2->domain) == 0);
+}
+
+static
+GList *parse_list(gchar *line)
+{
+  GList *list = NULL;
+  gchar buf[256];
+  gchar *p, *q;
+
+  p = line;
+  while(*p != 0){
+    q = buf;
+    while(isspace(*p)) p++;
+    if(*p != '\"'){
+      while(*p && (*p != ',') && (q < buf+255))
+	*(q++) = *(p++);
+      *q = 0;
+    }else{
+      gboolean escape = FALSE;
+      p++;
+      while(*p && (*p != '\"' || escape) && (q < buf+255)){
+	if((*p == '\\') && !escape)
+	  escape = TRUE;
+	else{
+	  escape = FALSE;
+	  *(q++) = *p;
+	}
+	p++;
+      }
+      *q = 0;
+      while(*p && (*p != ',')) p++;
+    }
+    list = g_list_append(list, g_strdup(g_strchomp(buf)));
+    if(*p) p++;
+  }
+  return list;
+}
+
+GList *alias_expand(GList *alias_table, GList *rcpt_list, GList *non_rcpt_list)
+{
+  GList *done_list = NULL;
+  GList *rcpt_node = g_list_copy(rcpt_list);
+
+  while(rcpt_node != NULL){
+    address *addr = (address *)(rcpt_node->data);
+    DEBUG(5) debugf("alias_expand begin: '%s@%s'\n", addr->local_part, addr->domain);
+//    if(addr_is_local(addr) && (addr->local_part[0] != '|') &&
+    if(addr_is_local(addr) && 
+        !(addr->flags & ADDR_FLAG_NOEXPAND)){
+      gchar *val;
+      
+      /* special handling for postmaster */
+      if(strcasecmp(addr->local_part, "postmaster") == 0)
+	val = (gchar *)table_find_func(alias_table, addr->local_part, strcasecmp);
+      else
+	val = (gchar *)table_find_func(alias_table, addr->local_part, conf.alias_local_cmp);
+
+      DEBUG(5) debugf("alias: '%s' is local\n", addr->local_part);
+      if(val != NULL){
+	GList *val_list = parse_list(val);
+	GList *val_node;
+	GList *alias_list = NULL;
+
+	DEBUG(5) debugf("alias: '%s' -> '%s'\n", addr->local_part, val);
+	foreach(val_list, val_node){
+	  gchar *val = (gchar *)(val_node->data);
+	  address *alias_addr;
+	  address *addr_parent = NULL;
+
+	  if(val[0] == '|'){
+	    DEBUG(5) debugf("alias: %s is a pipe address\n", val);
+	    alias_addr = create_address_pipe(val);
+	    DEBUG(5) debugf("alias_pipe: %s is a pipe address\n", alias_addr->local_part);
+	  }else if(val[0] == '\\'){
+	    DEBUG(5) debugf("alias: shall not be expanded: '%s'\n", val);
+	    alias_addr = create_address_qualified(&(val[1]), TRUE, conf.host_name);
+	    alias_addr->flags |= ADDR_FLAG_NOEXPAND;
+	    DEBUG(5) debugf("alias: not expanded: '%s'\n",alias_addr->local_part);
+	  }else{
+	    alias_addr = create_address_qualified(val, TRUE, conf.host_name);
+
+	    /* search in parents for loops: */
+	    for(addr_parent = addr; addr_parent; addr_parent = addr_parent->parent){
+	      if(addr_isequal_alias(alias_addr, addr_parent)){
+		logwrite(LOG_ALERT, "detected alias loop, (ignoring): %s@%s -> %s@%s\n",
+			 addr_parent->local_part, addr_parent->domain,
+			 addr->local_part, addr->domain);
+		break;
+	      }
+	    }
+	  }
+	  if(!addr_parent){
+	    alias_list = g_list_append(alias_list, alias_addr);
+	    alias_addr->parent = addr;
+	  }
+	  g_free(val);
+	}
+	g_list_free(val_list);
+	addr->children = g_list_copy(alias_list);
+	rcpt_node = g_list_concat(rcpt_node, alias_list);
+      }else{
+	DEBUG(5) debugf("alias: '%s' is completed\n", addr->local_part);
+	done_list = g_list_append(done_list, addr);
+      }
+    }else{
+      DEBUG(5) debugf("alias: '%s@%s' is not local\n", addr->local_part, addr->domain);
+      done_list = g_list_append(done_list, addr);
+    }
+    rcpt_node = g_list_next(rcpt_node);
+  }
+
+  /* delete addresses from done_list if they are in the non_rcpt_list */
+  if(non_rcpt_list){
+    GList *rcpt_node_next;
+    for(rcpt_node = g_list_first(done_list);
+	 rcpt_node;
+	 rcpt_node = rcpt_node_next){
+      address *addr = (address *)(rcpt_node->data);
+      GList *non_node;
+
+      rcpt_node_next = g_list_next(rcpt_node);
+
+      foreach(non_rcpt_list, non_node){
+	address *non_addr = (address *)(non_node->data);
+	if(addr_isequal(addr, non_addr)){
+	  done_list = g_list_remove_link(done_list, rcpt_node);
+	  g_list_free_1(rcpt_node);
+	  addr_mark_delivered(addr); /* this address is still in the children lists
+				      of the original address */
+	  break;
+	}
+      }
+    }
+  }
+  return done_list;
+}