diff libpurple/dnssrv.c @ 15374:5fe8042783c1

Rename gtk/ and libgaim/ to pidgin/ and libpurple/
author Sean Egan <seanegan@gmail.com>
date Sat, 20 Jan 2007 02:32:10 +0000
parents
children 61b42cf81aa4
line wrap: on
line diff
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/libpurple/dnssrv.c	Sat Jan 20 02:32:10 2007 +0000
@@ -0,0 +1,407 @@
+/**
+ * @file dnssrv.c
+ *
+ * gaim
+ *
+ * Copyright (C) 2005 Thomas Butter <butter@uni-mannheim.de>
+ *
+ * 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 "internal.h"
+
+#ifndef _WIN32
+#include <resolv.h>
+#include <arpa/nameser.h>
+#ifdef HAVE_ARPA_NAMESER_COMPAT_H
+#include <arpa/nameser_compat.h>
+#endif
+#ifndef T_SRV
+#define T_SRV	33
+#endif
+#else
+#include <windns.h>
+/* Missing from the mingw headers */
+#ifndef DNS_TYPE_SRV
+# define DNS_TYPE_SRV 33
+#endif
+#endif
+
+#include "dnssrv.h"
+#include "eventloop.h"
+#include "debug.h"
+
+#ifndef _WIN32
+typedef union {
+	HEADER hdr;
+	u_char buf[1024];
+} queryans;
+#else
+static DNS_STATUS WINAPI (*MyDnsQuery_UTF8) (
+	PCSTR lpstrName, WORD wType, DWORD fOptions,
+	PIP4_ARRAY aipServers, PDNS_RECORD* ppQueryResultsSet,
+	PVOID* pReserved) = NULL;
+static void WINAPI (*MyDnsRecordListFree) (PDNS_RECORD pRecordList,
+	DNS_FREE_TYPE FreeType) = NULL;
+#endif
+
+struct _GaimSrvQueryData {
+	GaimSrvCallback cb;
+	gpointer extradata;
+	guint handle;
+#ifdef _WIN32
+	GThread *resolver;
+	char *query;
+	char *error_message;
+	GSList *results;
+#endif
+};
+
+static gint
+responsecompare(gconstpointer ar, gconstpointer br)
+{
+	GaimSrvResponse *a = (GaimSrvResponse*)ar;
+	GaimSrvResponse *b = (GaimSrvResponse*)br;
+
+	if(a->pref == b->pref) {
+		if(a->weight == b->weight)
+			return 0;
+		if(a->weight < b->weight)
+			return -1;
+		return 1;
+	}
+	if(a->pref < b->pref)
+		return -1;
+	return 1;
+}
+
+#ifndef _WIN32
+
+static void
+resolve(int in, int out)
+{
+	GList *ret = NULL;
+	GaimSrvResponse *srvres;
+	queryans answer;
+	int size;
+	int qdcount;
+	int ancount;
+	guchar *end;
+	guchar *cp;
+	gchar name[256];
+	guint16 type, dlen, pref, weight, port;
+	gchar query[256];
+
+	if (read(in, query, 256) <= 0)
+		_exit(0);
+
+	size = res_query( query, C_IN, T_SRV, (u_char*)&answer, sizeof( answer));
+
+	qdcount = ntohs(answer.hdr.qdcount);
+	ancount = ntohs(answer.hdr.ancount);
+
+	cp = (guchar*)&answer + sizeof(HEADER);
+	end = (guchar*)&answer + size;
+
+	/* skip over unwanted stuff */
+	while (qdcount-- > 0 && cp < end) {
+		size = dn_expand( (unsigned char*)&answer, end, cp, name, 256);
+		if(size < 0) goto end;
+		cp += size + QFIXEDSZ;
+	}
+
+	while (ancount-- > 0 && cp < end) {
+		size = dn_expand((unsigned char*)&answer, end, cp, name, 256);
+		if(size < 0)
+			goto end;
+
+		cp += size;
+
+		GETSHORT(type,cp);
+
+		/* skip ttl and class since we already know it */
+		cp += 6;
+
+		GETSHORT(dlen,cp);
+
+		if (type == T_SRV) {
+			GETSHORT(pref,cp);
+
+			GETSHORT(weight,cp);
+
+			GETSHORT(port,cp);
+
+			size = dn_expand( (unsigned char*)&answer, end, cp, name, 256);
+			if(size < 0 )
+				goto end;
+
+			cp += size;
+
+			srvres = g_new0(GaimSrvResponse, 1);
+			strcpy(srvres->hostname, name);
+			srvres->pref = pref;
+			srvres->port = port;
+			srvres->weight = weight;
+
+			ret = g_list_insert_sorted(ret, srvres, responsecompare);
+		} else {
+			cp += dlen;
+		}
+	}
+
+end:
+	size = g_list_length(ret);
+	write(out, &size, sizeof(int));
+	while (ret != NULL)
+	{
+		write(out, ret->data, sizeof(GaimSrvResponse));
+		g_free(ret->data);
+		ret = g_list_remove(ret, ret->data);
+	}
+
+	_exit(0);
+}
+
+static void
+resolved(gpointer data, gint source, GaimInputCondition cond)
+{
+	int size;
+	GaimSrvQueryData *query_data = (GaimSrvQueryData*)data;
+	GaimSrvResponse *res;
+	GaimSrvResponse *tmp;
+	int i;
+	GaimSrvCallback cb = query_data->cb;
+
+	read(source, &size, sizeof(int));
+	gaim_debug_info("dnssrv","found %d SRV entries\n", size);
+	tmp = res = g_new0(GaimSrvResponse, size);
+	for (i = 0; i < size; i++) {
+		read(source, tmp++, sizeof(GaimSrvResponse));
+	}
+
+	cb(res, size, query_data->extradata);
+
+	gaim_srv_cancel(query_data);
+}
+
+#else /* _WIN32 */
+
+/** The Jabber Server code was inspiration for parts of this. */
+
+static gboolean
+res_main_thread_cb(gpointer data)
+{
+	GaimSrvResponse *srvres = NULL;
+	int size = 0;
+	GaimSrvQueryData *query_data = data;
+
+	if(query_data->error_message != NULL)
+		gaim_debug_error("dnssrv", query_data->error_message);
+	else {
+		GaimSrvResponse *srvres_tmp = NULL;
+		GSList *lst = query_data->results;
+
+		size = g_slist_length(query_data->results);
+
+		if(query_data->cb)
+			srvres_tmp = srvres = g_new0(GaimSrvResponse, size);
+		while (lst) {
+			if(query_data->cb)
+				memcpy(srvres_tmp++, lst->data, sizeof(GaimSrvResponse));
+			g_free(lst->data);
+			lst = g_slist_remove(lst, lst->data);
+		}
+
+		query_data->results = NULL;
+	}
+
+	gaim_debug_info("dnssrv", "found %d SRV entries\n", size);
+
+	if(query_data->cb)
+		query_data->cb(srvres, size, query_data->extradata);
+
+	query_data->resolver = NULL;
+	query_data->handle = 0;
+
+	gaim_srv_cancel(query_data);
+
+	return FALSE;
+}
+
+static gpointer
+res_thread(gpointer data)
+{
+	PDNS_RECORD dr = NULL;
+	int type = DNS_TYPE_SRV;
+	DNS_STATUS ds;
+	GaimSrvQueryData *query_data = data;
+
+	ds = MyDnsQuery_UTF8(query_data->query, type, DNS_QUERY_STANDARD, NULL, &dr, NULL);
+	if (ds != ERROR_SUCCESS) {
+		gchar *msg = g_win32_error_message(ds);
+		query_data->error_message = g_strdup_printf("Couldn't look up SRV record. %s (%lu).\n", msg, ds);
+		g_free(msg);
+	} else {
+		PDNS_RECORD dr_tmp;
+		GSList *lst = NULL;
+		DNS_SRV_DATA *srv_data;
+		GaimSrvResponse *srvres;
+
+		for (dr_tmp = dr; dr_tmp != NULL; dr_tmp = dr_tmp->pNext) {
+			/* Discard any incorrect entries. I'm not sure if this is necessary */
+			if (dr_tmp->wType != type || strcmp(dr_tmp->pName, query_data->query) != 0) {
+				continue;
+			}
+
+			srv_data = &dr_tmp->Data.SRV;
+			srvres = g_new0(GaimSrvResponse, 1);
+			strncpy(srvres->hostname, srv_data->pNameTarget, 255);
+			srvres->hostname[255] = '\0';
+			srvres->pref = srv_data->wPriority;
+			srvres->port = srv_data->wPort;
+			srvres->weight = srv_data->wWeight;
+
+			lst = g_slist_insert_sorted(lst, srvres, responsecompare);
+		}
+
+		MyDnsRecordListFree(dr, DnsFreeRecordList);
+		query_data->results = lst;
+	}
+
+	/* back to main thread */
+	/* Note: this should *not* be attached to query_data->handle - it will cause leakage */
+	g_idle_add(res_main_thread_cb, query_data);
+
+	g_thread_exit(NULL);
+	return NULL;
+}
+
+#endif
+
+GaimSrvQueryData *
+gaim_srv_resolve(const char *protocol, const char *transport, const char *domain, GaimSrvCallback cb, gpointer extradata)
+{
+	char *query;
+	GaimSrvQueryData *query_data;
+#ifndef _WIN32
+	int in[2], out[2];
+	int pid;
+#else
+	GError* err = NULL;
+	static gboolean initialized = FALSE;
+#endif
+
+	query = g_strdup_printf("_%s._%s.%s", protocol, transport, domain);
+	gaim_debug_info("dnssrv","querying SRV record for %s\n", query);
+
+#ifndef _WIN32
+	if(pipe(in) || pipe(out)) {
+		gaim_debug_error("dnssrv", "Could not create pipe\n");
+		g_free(query);
+		cb(NULL, 0, extradata);
+		return NULL;
+	}
+
+	pid = fork();
+	if (pid == -1) {
+		gaim_debug_error("dnssrv", "Could not create process!\n");
+		cb(NULL, 0, extradata);
+		g_free(query);
+		return NULL;
+	}
+
+	/* Child */
+	if (pid == 0)
+	{
+		close(out[0]);
+		close(in[1]);
+		resolve(in[0], out[1]);
+	}
+
+	close(out[1]);
+	close(in[0]);
+
+	if (write(in[1], query, strlen(query)+1) < 0)
+		gaim_debug_error("dnssrv", "Could not write to SRV resolver\n");
+
+	query_data = g_new0(GaimSrvQueryData, 1);
+	query_data->cb = cb;
+	query_data->extradata = extradata;
+	query_data->handle = gaim_input_add(out[0], GAIM_INPUT_READ, resolved, query_data);
+
+	g_free(query);
+
+	return query_data;
+#else
+	if (!initialized) {
+		MyDnsQuery_UTF8 = (void*) wgaim_find_and_loadproc("dnsapi.dll", "DnsQuery_UTF8");
+		MyDnsRecordListFree = (void*) wgaim_find_and_loadproc(
+			"dnsapi.dll", "DnsRecordListFree");
+		initialized = TRUE;
+	}
+
+	query_data = g_new0(GaimSrvQueryData, 1);
+	query_data->cb = cb;
+	query_data->query = query;
+	query_data->extradata = extradata;
+
+	if (!MyDnsQuery_UTF8 || !MyDnsRecordListFree) {
+		query_data->error_message = g_strdup_printf("System missing DNS API (Requires W2K+)\n");
+
+		/* Asynchronously call the callback since stuff may not expect
+		 * the callback to be called before this returns */
+		query_data->handle = g_idle_add(res_main_thread_cb, query_data);
+
+		return query_data;
+	}
+
+	query_data->resolver = g_thread_create(res_thread, query_data, FALSE, &err);
+	if (query_data->resolver == NULL)
+	{
+		query_data->error_message = g_strdup_printf("SRV thread create failure: %s\n", err ? err->message : "");
+		g_error_free(err);
+
+		/* Asynchronously call the callback since stuff may not expect
+		 * the callback to be called before this returns */
+		query_data->handle = g_idle_add(res_main_thread_cb, query_data);
+
+		return query_data;
+	}
+
+	return query_data;
+#endif
+}
+
+void
+gaim_srv_cancel(GaimSrvQueryData *query_data)
+{
+	if (query_data->handle > 0)
+		gaim_input_remove(query_data->handle);
+#ifdef _WIN32
+	if (query_data->resolver != NULL)
+	{
+		/*
+		 * It's not really possible to kill a thread.  So instead we
+		 * just set the callback to NULL and let the DNS lookup
+		 * finish.
+		 */
+		query_data->cb = NULL;
+		return;
+	}
+	g_free(query_data->query);
+	g_free(query_data->error_message);
+#endif
+	g_free(query_data);
+}