view src/protocols/simple/srvresolve.c @ 11249:b4b1be482b4e

[gaim-migrate @ 13418] sf patch #1235519, from Sadrul Habib Chowdhury This is a pretty big patch that makes Gaim correctly save and restore the current status (away/available, away message, available message, invisible, etc). The GaimGtkStatusBoxWidget thing I think defaults to "Available" every time its created, which overrides the setting that was saved to the XML file. So that still needs to be fixed before this will really work. Anyway, mad props to Sadrul for putting up with my requests on this patch committer: Tailor Script <tailor@pidgin.im>
author Mark Doliner <mark@kingant.net>
date Sat, 13 Aug 2005 05:22:09 +0000
parents 5f79dfde334c
children b073da869a55
line wrap: on
line source

/**
 * @file simple.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 "srvresolve.h"

typedef union {
	HEADER hdr;
	u_char buf[1024];
} queryans;

struct getserver_return *getserver(const char *domain, const char *srv) {
	static struct getserver_return ret;
	queryans answer;
	int size;
	int qdcount;
	int ancount;
	gchar *end;
	gchar *cp;
	gchar name[256];
	gchar *bestname = NULL;
	int bestport = 5060;
	int bestpri=99999;
	int type, dlen, pref, weight, port;
	gchar *query = g_strdup_printf("%s.%s",srv,domain);

	
	size = res_query( query, C_IN, T_SRV, (u_char*)&answer, sizeof( answer));

	g_free(query);

	qdcount = ntohs(answer.hdr.qdcount);
	ancount = ntohs(answer.hdr.ancount);

	
	cp = (char*)&answer + sizeof(HEADER);
	end = (char*)&answer + size;

	// skip over unwanted stuff
	while (qdcount-- > 0 && cp < end) {
		size = dn_expand( (char*)&answer, end, cp, name, 256);
		if(size < 0) return NULL;
		cp += size + QFIXEDSZ;
	}

	while (ancount-- > 0 && cp < end) {
		size = dn_expand((char*)&answer, end, cp, name, 256);
		if(size < 0)
			return NULL;

		cp += size;
	
		NS_GET16(type,cp);
		cp += 6; // skip ttl and class

		NS_GET16(dlen,cp);

		if (type == T_SRV) {
			NS_GET16(pref,cp);

			NS_GET16(weight, cp);

			NS_GET16(port, cp);

			size = dn_expand( (char*)&answer, end, cp, name, 256);
			if(size < 0 )
				return NULL;

			cp += size;

			if(pref<bestpri) {
				if( bestname) g_free(bestname);
				bestname = g_strdup(name);
				bestpri = pref;
				bestport = port;
			}
		} else {
			cp += dlen;
		}
	}
	if(bestpri < 99999) {
		ret.name = bestname;
		ret.port = bestport;
	} else {
		ret.name = g_strdup(domain);
		ret.port = 5060;
	}
	return &ret;
}