masqmail

changeset 187:bd7c52a36b0c

improved mservdetect in various ways errors are handled better (no segfaults anymore) copied the relevant part of interface.c into mservdetect.c described how I think the mserver protocol works
author meillo@marmaro.de
date Thu, 15 Jul 2010 00:14:26 +0200 (2010-07-14)
parents 3dff59a4e764
children bfa7a8b566da
files src/Makefile.am src/Makefile.in src/mservdetect.c
diffstat 3 files changed, 141 insertions(+), 63 deletions(-) [+]
line diff
     1.1 --- a/src/Makefile.am	Wed Jul 14 23:26:20 2010 +0200
     1.2 +++ b/src/Makefile.am	Thu Jul 15 00:14:26 2010 +0200
     1.3 @@ -71,7 +71,6 @@
     1.4  	masqmail.h\
     1.5  	readsock.h\
     1.6  	mservdetect.c\
     1.7 -	interface.c\
     1.8  	readsock.c\
     1.9  	peopen.c
    1.10  
     2.1 --- a/src/Makefile.in	Wed Jul 14 23:26:20 2010 +0200
     2.2 +++ b/src/Makefile.in	Thu Jul 15 00:14:26 2010 +0200
     2.3 @@ -61,8 +61,8 @@
     2.4  	timeival.$(OBJEXT)
     2.5  masqmail_OBJECTS = $(am_masqmail_OBJECTS)
     2.6  masqmail_DEPENDENCIES =
     2.7 -am_mservdetect_OBJECTS = mservdetect.$(OBJEXT) interface.$(OBJEXT) \
     2.8 -	readsock.$(OBJEXT) peopen.$(OBJEXT)
     2.9 +am_mservdetect_OBJECTS = mservdetect.$(OBJEXT) readsock.$(OBJEXT) \
    2.10 +	peopen.$(OBJEXT)
    2.11  mservdetect_OBJECTS = $(am_mservdetect_OBJECTS)
    2.12  mservdetect_DEPENDENCIES =
    2.13  am_readtest_OBJECTS = readsock.$(OBJEXT) readtest.$(OBJEXT)
    2.14 @@ -299,7 +299,6 @@
    2.15  	masqmail.h\
    2.16  	readsock.h\
    2.17  	mservdetect.c\
    2.18 -	interface.c\
    2.19  	readsock.c\
    2.20  	peopen.c
    2.21  
     3.1 --- a/src/mservdetect.c	Wed Jul 14 23:26:20 2010 +0200
     3.2 +++ b/src/mservdetect.c	Thu Jul 15 00:14:26 2010 +0200
     3.3 @@ -1,5 +1,6 @@
     3.4  /*  MasqMail
     3.5      Copyright (C) 1999-2001 Oliver Kurth
     3.6 +    Copyright (C) 2010 markus schnalke <meillo@marmaro.de>
     3.7  
     3.8      This program is free software; you can redistribute it and/or modify
     3.9      it under the terms of the GNU General Public License as published by
    3.10 @@ -22,56 +23,143 @@
    3.11  #include "readsock.h"
    3.12  
    3.13  
    3.14 +
    3.15 +gboolean
    3.16 +init_sockaddr(struct sockaddr_in * name, interface * iface)
    3.17 +{
    3.18 +	struct hostent *he;
    3.19 +	struct in_addr ia;
    3.20 +
    3.21 +	if (inet_aton(iface->address, &ia) != 0) {
    3.22 +		/* IP address */
    3.23 +		memcpy(&(name->sin_addr), &ia, sizeof(name->sin_addr));
    3.24 +	} else {
    3.25 +		if ((he = gethostbyname(iface->address)) == NULL) {
    3.26 +			logwrite(LOG_ALERT, "local address '%s' unknown. (deleting)\n", iface->address);
    3.27 +			return FALSE;
    3.28 +		}
    3.29 +		memcpy(&(name->sin_addr), he->h_addr, sizeof(name->sin_addr));
    3.30 +	}
    3.31 +	name->sin_family = AF_INET;
    3.32 +	name->sin_port = htons(iface->port);
    3.33 +
    3.34 +	return TRUE;
    3.35 +}
    3.36 +
    3.37 +
    3.38 +int
    3.39 +make_server_socket(interface * iface)
    3.40 +{
    3.41 +	int sock = -1;
    3.42 +	struct sockaddr_in server;
    3.43 +
    3.44 +	memset(&server, 0, sizeof(struct sockaddr_in));
    3.45 +
    3.46 +	/* Create the socket. */
    3.47 +	sock = socket(PF_INET, SOCK_STREAM, 0);
    3.48 +	if (sock < 0) {
    3.49 +		logwrite(LOG_ALERT, "socket: %s\n", strerror(errno));
    3.50 +		return -1;
    3.51 +	}
    3.52 +
    3.53 +	if (init_sockaddr(&server, iface)) {
    3.54 +		/* bind the socket */
    3.55 +		if (bind(sock, (struct sockaddr *) &server, sizeof(server)) < 0) {
    3.56 +			logwrite(LOG_ALERT, "bind: %s\n", strerror(errno));
    3.57 +			return -1;
    3.58 +		}
    3.59 +	} else {
    3.60 +		close(sock);
    3.61 +		return -1;
    3.62 +	}
    3.63 +
    3.64 +	return sock;
    3.65 +}
    3.66 +
    3.67 +
    3.68 +
    3.69 +
    3.70  gchar*
    3.71  mserver_detect_online(interface * iface)
    3.72  {
    3.73  	struct sockaddr_in saddr;
    3.74  	gchar *ret = NULL;
    3.75  
    3.76 -	if (init_sockaddr(&saddr, iface)) {
    3.77 -		int sock = socket(PF_INET, SOCK_STREAM, 0);
    3.78 -		int dup_sock;
    3.79 -		if (connect(sock, (struct sockaddr *) (&saddr), sizeof(saddr)) == 0) {
    3.80 -			FILE *in, *out;
    3.81 -			char buf[256];
    3.82 +	if (!init_sockaddr(&saddr, iface)) {
    3.83 +		return NULL;
    3.84 +	}
    3.85  
    3.86 -			dup_sock = dup(sock);
    3.87 -			out = fdopen(sock, "w");
    3.88 -			in = fdopen(dup_sock, "r");
    3.89 +	int sock = socket(PF_INET, SOCK_STREAM, 0);
    3.90 +	int dup_sock;
    3.91 +	if (connect(sock, (struct sockaddr *) (&saddr), sizeof(saddr)) != 0) {
    3.92 +		return NULL;
    3.93 +	}
    3.94  
    3.95 -			if (read_sockline(in, buf, 256, 15, READSOCKL_CHUG)) {
    3.96 -				if (strncmp(buf, "READY", 5) == 0) {
    3.97 -					fprintf(out, "STAT\n");
    3.98 -					fflush(out);
    3.99 -					if (read_sockline(in, buf, 256, 15, READSOCKL_CHUG)) {
   3.100 -						if (strncmp(buf, "DOWN", 4) == 0) {
   3.101 -							ret = NULL;
   3.102 -						} else if (strncmp(buf, "UP", 2) == 0) {
   3.103 -							gchar *p = buf + 3;
   3.104 -							while ((*p != ':') && *p)
   3.105 -								p++;
   3.106 -							if (*p) {
   3.107 -								*p = 0;
   3.108 -								p++;
   3.109 -								if ((atoi(p) >= 0) && *p)
   3.110 -									ret = g_strdup(buf + 3);
   3.111 -							} else
   3.112 -								logwrite(LOG_ALERT, "unexpected response from mserver after STAT cmd: %s", buf);
   3.113 -						} else {
   3.114 -							logwrite(LOG_ALERT, "unexpected response from mserver after STAT cmd: %s", buf);
   3.115 -						}
   3.116 +	FILE *in, *out;
   3.117 +	char buf[256];
   3.118 +
   3.119 +	dup_sock = dup(sock);
   3.120 +	out = fdopen(sock, "w");
   3.121 +	in = fdopen(dup_sock, "r");
   3.122 +
   3.123 +	if (!read_sockline(in, buf, 256, 15, READSOCKL_CHUG)) {
   3.124 +		return NULL;
   3.125 +	}
   3.126 +
   3.127 +	/* this is the protocol (reverse engineered):
   3.128 +	   S: READY
   3.129 +	   C: STAT
   3.130 +	   S: DOWN
   3.131 +	   C: QUIT
   3.132 +	   -> offline
   3.133 +	   
   3.134 +	   S: READY
   3.135 +	   C: STAT
   3.136 +	   S: UP foo:-1
   3.137 +	   C: QUIT
   3.138 +	   -> offline
   3.139 +	   
   3.140 +	   S: READY
   3.141 +	   C: STAT
   3.142 +	   S: UP foo:1
   3.143 +	   C: QUIT
   3.144 +	   -> online, `foo' gets printed
   3.145 +	*/
   3.146 +
   3.147 +	if (strncmp(buf, "READY", 5) == 0) {
   3.148 +		fprintf(out, "STAT\n");
   3.149 +		fflush(out);
   3.150 +		if (read_sockline(in, buf, 256, 15, READSOCKL_CHUG)) {
   3.151 +			if (strncmp(buf, "DOWN", 4) == 0) {
   3.152 +				ret = NULL;
   3.153 +			} else if (strncmp(buf, "UP", 2) == 0) {
   3.154 +				gchar *p = buf + 3;
   3.155 +				while ((*p != ':') && *p) {
   3.156 +					p++;
   3.157 +				}
   3.158 +				if (*p) {
   3.159 +					*p = '\0';
   3.160 +					p++;
   3.161 +					if ((atoi(p) >= 0) && *p) {
   3.162 +						/* `UP foo:N', where `N' is a non-negative number */
   3.163 +						ret = g_strdup(buf + 3);
   3.164  					}
   3.165 +				} else {
   3.166 +					fprintf(stderr, "unexpected response from mserver after STAT cmd: %s", buf);
   3.167  				}
   3.168 -				fprintf(out, "QUIT");
   3.169 -				fflush(out);
   3.170 -
   3.171 -				close(sock);
   3.172 -				close(dup_sock);
   3.173 -				fclose(in);
   3.174 -				fclose(out);
   3.175 +			} else {
   3.176 +				fprintf(stderr, "unexpected response from mserver after STAT cmd: %s", buf);
   3.177  			}
   3.178  		}
   3.179  	}
   3.180 +	fprintf(out, "QUIT");
   3.181 +	fflush(out);
   3.182 +
   3.183 +	close(sock);
   3.184 +	close(dup_sock);
   3.185 +	fclose(in);
   3.186 +	fclose(out);
   3.187 +
   3.188  	return ret;
   3.189  }
   3.190  
   3.191 @@ -82,39 +170,31 @@
   3.192  	va_list args;
   3.193  	va_start(args, fmt);
   3.194  
   3.195 -	vfprintf(stdout, fmt, args);
   3.196 +	vfprintf(stderr, fmt, args);
   3.197  
   3.198  	va_end(args);
   3.199  }
   3.200  
   3.201 -void
   3.202 -debugf(const char *fmt, ...)
   3.203 -{
   3.204 -	va_list args;
   3.205 -	va_start(args, fmt);
   3.206 -
   3.207 -	vfprintf(stdout, fmt, args);
   3.208 -
   3.209 -	va_end(args);
   3.210 -}
   3.211  
   3.212  int
   3.213  main(int argc, char *argv[])
   3.214  {
   3.215 -	if (argc == 3) {
   3.216 -		interface iface;
   3.217 -		gchar *name;
   3.218 +	interface iface;
   3.219 +	gchar *name;
   3.220  
   3.221 -		iface.address = g_strdup(argv[1]);
   3.222 -		iface.port = atoi(argv[2]);
   3.223 +	if (argc != 3) {
   3.224 +		fprintf(stderr, "usage: %s HOST PORT\n", argv[0]);
   3.225 +		return 1;
   3.226 +	}
   3.227  
   3.228 -		name = mserver_detect_online(&iface);
   3.229 +	iface.address = g_strdup(argv[1]);
   3.230 +	iface.port = atoi(argv[2]);
   3.231  
   3.232 +	name = mserver_detect_online(&iface);
   3.233 +
   3.234 +	if (name) {
   3.235  		printf("%s\n", name);
   3.236 -
   3.237 -		exit(EXIT_SUCCESS);
   3.238 -	} else {
   3.239 -		fprintf(stderr, "usage %s <host> <port>\n", argv[0]);
   3.240 -		exit(EXIT_FAILURE);
   3.241 +		return 0;
   3.242  	}
   3.243 +	return 1;
   3.244  }