Mercurial > masqmail-0.2
diff src/connect.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/connect.c Fri Sep 26 17:05:23 2008 +0200 @@ -0,0 +1,151 @@ +/* MasqMail + Copyright (C) 1999 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" + +static +GList *resolve_ip(GList *list, gchar *ip) +{ + struct in_addr ia; + if(inet_aton(ip, &ia)){ + mxip_addr mxip; + + mxip.name = g_strdup(ip); + mxip.pref = 0; + mxip.ip = (guint32) *(guint32 *)(&ia); + list = g_list_append(list, g_memdup(&mxip, sizeof(mxip))); + } + /* logwrite(LOG_ALERT, "invalid address '%s': inet_aton() failed\n", ip);*/ + return NULL; +} + +mxip_addr *connect_hostlist(int *psockfd, gchar *host, guint port, + GList *addr_list) +{ + GList *addr_node; + struct sockaddr_in saddr; + + DEBUG(5) debugf("connect_hostlist entered\n"); + + for(addr_node = g_list_first(addr_list); + addr_node; + addr_node = g_list_next(addr_node)){ + mxip_addr *addr = (mxip_addr *)(addr_node->data); + + *psockfd = socket(PF_INET, SOCK_STREAM, 0); + + memset(&saddr, 0, sizeof(saddr)); + + saddr.sin_family = AF_INET; + saddr.sin_port = htons(port); + + /* clumsy, but makes compiler happy: */ + saddr.sin_addr = *(struct in_addr*)(&(addr->ip)); + DEBUG(5) debugf("trying ip %s port %d\n", inet_ntoa(saddr.sin_addr), port); + if(connect(*psockfd, (struct sockaddr *)(&saddr), sizeof(saddr)) == 0){ + DEBUG(5) debugf("connected to %s\n", inet_ntoa(saddr.sin_addr)); + return addr; + }else{ + int saved_errno = errno; + + close(*psockfd); + + logwrite(LOG_WARNING, "connection to %s failed: %s\n", + inet_ntoa(saddr.sin_addr), strerror(errno)); + + errno = saved_errno; + + if((saved_errno != ECONNREFUSED) && + (saved_errno != ETIMEDOUT) && + (saved_errno != ENETUNREACH) && + (saved_errno != EHOSTUNREACH)) + + return NULL; + } + } + return NULL; +} + +/* Given a list of resolver functions, this function + resolve the host and tries to connect to the addresses + returned. If a connection attemp is timed out or refused, + the next address is tried. + + TODO: the resolver functions might return duplicate addresses, + if attempt failed for one it should not be tried again. +*/ + +mxip_addr *connect_resolvelist(int *psockfd, gchar *host, guint port, + GList *res_func_list) +{ + GList *res_node; + GList *addr_list; + + DEBUG(5) debugf("connect_resolvelist entered\n"); + + h_errno = 0; + + if(isdigit(host[0])){ + mxip_addr *addr; + + addr_list = resolve_ip(NULL, host); + if(addr_list){ + addr = connect_hostlist(psockfd, host, port, addr_list); + g_list_free(addr_list); + return addr; + } + /* previous versions complained, until someone tried to use a hostname + out there that begins with a digit. eg. '3dwars.de'. */ + } + + if(res_func_list == NULL){ + logwrite(LOG_ALERT, "res_funcs == NULL !!!\n"); + exit(EXIT_FAILURE); + } + + foreach(res_func_list, res_node){ + resolve_func res_func; + DEBUG(6) debugf("connect_resolvelist 1a\n"); + res_func = (resolve_func)(res_node->data); + + if(res_func == NULL){ + logwrite(LOG_ALERT, "res_func == NULL !!!\n"); + exit(EXIT_FAILURE); + } + + errno = 0; + if((addr_list = res_func(NULL, host))){ + + mxip_addr *addr; + if((addr = connect_hostlist(psockfd, host, port, addr_list))) + return addr; + + DEBUG(5){ + debugf("connect_hostlist failed: %s\n", strerror(errno)); + } + + g_list_free(addr_list); + }else{ + if(!g_list_next(res_node)){ + logwrite(LOG_ALERT, "could not resolve %s: %s\n", host, hstrerror(h_errno)); + } + } + } + return NULL; + +} +