Mercurial > masqmail
view src/alias.c @ 234:4b40be6f1cbd
renamed the default route for the local net
it was called: local_net (default)
now it is called: default local_net_route
author | markus schnalke <meillo@marmaro.de> |
---|---|
date | Fri, 22 Oct 2010 11:06:09 -0300 (2010-10-22) |
parents | 3f33a0feeeb0 |
children | 31ee44f45787 |
line wrap: on
line source
/* 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); DEBUG(6) debugf("not_local_addresses: addr_node->data=%s a->address=%s\n", addr_node->data, a->address); 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); DEBUG(6) debugf("local_addresses: addr_node->data=%s a->address=%s\n", addr_node->data, a->address); 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; DEBUG(5) debugf("alias: '%s' is local\n", addr->local_part); if (strcasecmp(addr->local_part, "postmaster") == 0) /* postmaster needs always to be matched caseless see RFC 822 and RFC 5321 */ 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); 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; }