2 * @file purple-dnsquery.c
6 * Copyright (C) 2010-2018 SIPE Project <http://sipe.sourceforge.net/>
8 * This program is free software; you can redistribute it and/or modify
9 * it under the terms of the GNU General Public License as published by
10 * the Free Software Foundation; either version 2 of the License, or
11 * (at your option) any later version.
13 * This program is distributed in the hope that it will be useful,
14 * but WITHOUT ANY WARRANTY; without even the implied warranty of
15 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
16 * GNU General Public License for more details.
18 * You should have received a copy of the GNU General Public License
19 * along with this program; if not, write to the Free Software
20 * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
27 #if PURPLE_VERSION_CHECK(2,8,0)
30 #if PURPLE_VERSION_CHECK(3,0,0)
31 #include "protocols.h"
36 /* wrappers for write() & friends for socket handling */
37 #include "win32/win32dep.h"
40 #include <sys/types.h>
41 #include <sys/socket.h>
42 #include <netinet/in.h>
43 #include <arpa/inet.h>
51 #include "sipe-backend.h"
52 #include "sipe-core.h"
54 #include "purple-private.h"
56 struct sipe_dns_query
{
57 struct sipe_backend_private
*purple_private
;
58 sipe_dns_resolved_cb callback
;
60 gpointer purple_query_data
;
62 #if PURPLE_VERSION_CHECK(3,0,0)
72 static void sipe_dns_query_free(struct sipe_dns_query
*query
)
74 #if PURPLE_VERSION_CHECK(3,0,0)
75 g_object_unref(query
->purple_query_data
);
80 #if PURPLE_VERSION_CHECK(3,0,0)
81 static void dns_a_response(GObject
*source
,
85 struct sipe_dns_query
*query
= user_data
;
88 gchar
*address_str
= NULL
;
90 if (!query
->is_valid
) {
91 /* Ignore spurious responses after disconnect */
95 query
->purple_private
->dns_queries
=
96 g_slist_remove(query
->purple_private
->dns_queries
,
99 hosts
= g_resolver_lookup_by_name_finish(G_RESOLVER(source
), res
,
102 if (!error
&& g_list_length(hosts
) > 0) {
103 address_str
= g_inet_address_to_string(hosts
->data
);
106 query
->callback(query
->extradata
, address_str
,
107 address_str
? query
->port
: 0);
112 g_resolver_free_addresses(hosts
);
113 sipe_dns_query_free(query
);
116 static void dns_a_response(GSList
*hosts
,
117 struct sipe_dns_query
*query
,
118 const char *error_message
)
120 char ipstr
[INET6_ADDRSTRLEN
];
121 struct sockaddr
*addr
;
122 const void *addrdata
;
125 /* Ignore spurious responses after disconnect */
126 if (query
->is_valid
) {
127 struct sipe_backend_private
*purple_private
= query
->purple_private
;
129 purple_private
->dns_queries
= g_slist_remove(purple_private
->dns_queries
,
132 if (error_message
|| !g_slist_next(hosts
)) {
133 query
->callback(query
->extradata
, NULL
, 0);
138 addr
= g_slist_next(hosts
)->data
;
139 if (addr
->sa_family
== AF_INET6
) {
140 /* OS provides addr so it must be properly aligned */
141 struct sockaddr_in6
*sin6
= (void *) addr
;
142 addrdata
= &sin6
->sin6_addr
;
143 port
= sin6
->sin6_port
;
145 /* OS provides addr so it must be properly aligned */
146 struct sockaddr_in
*sin
= (void *) addr
;
147 addrdata
= &sin
->sin_addr
;
148 port
= sin
->sin_port
;
151 inet_ntop(addr
->sa_family
, addrdata
, ipstr
, sizeof (ipstr
));
153 query
->callback(query
->extradata
, ipstr
, port
);
158 for (; hosts
; hosts
= g_slist_delete_link(hosts
, hosts
)) {
159 // Free the addrlen, no data in this link
160 hosts
= g_slist_delete_link(hosts
, hosts
);
167 struct sipe_dns_query
*sipe_backend_dns_query_a(struct sipe_core_public
*sipe_public
,
168 const gchar
*hostname
,
170 sipe_dns_resolved_cb callback
,
173 struct sipe_dns_query
*query
= g_new(struct sipe_dns_query
, 1);
174 struct sipe_backend_private
*purple_private
= sipe_public
->backend_private
;
175 #if PURPLE_VERSION_CHECK(3,0,0)
176 GResolver
*resolver
= g_resolver_get_default();
179 query
->purple_private
= purple_private
;
180 query
->callback
= callback
;
181 query
->extradata
= data
;
182 query
->is_valid
= TRUE
;
184 purple_private
->dns_queries
= g_slist_prepend(purple_private
->dns_queries
,
187 #if PURPLE_VERSION_CHECK(3,0,0)
189 query
->purple_query_data
= g_cancellable_new();
191 g_resolver_lookup_by_name_async(resolver
,
193 query
->purple_query_data
,
196 g_object_unref(resolver
);
199 query
->purple_query_data
=
200 #if PURPLE_VERSION_CHECK(2,8,0)
201 purple_dnsquery_a_account(
202 purple_private
->account
,
208 (PurpleDnsQueryConnectFunction
) dns_a_response
,
215 #if PURPLE_VERSION_CHECK(3,0,0)
216 static void dns_srv_response(GObject
*source
,
220 struct sipe_dns_query
*query
= user_data
;
221 GError
*error
= NULL
;
224 if (!query
->is_valid
) {
225 /* Ignore spurious responses after disconnect */
229 query
->purple_private
->dns_queries
=
230 g_slist_remove(query
->purple_private
->dns_queries
,
233 targets
= g_resolver_lookup_service_finish(G_RESOLVER(source
), res
,
236 if (error
|| g_list_length(targets
) == 0) {
237 query
->callback(query
->extradata
, NULL
, 0);
239 query
->callback(query
->extradata
,
240 g_srv_target_get_hostname(targets
->data
),
241 g_srv_target_get_port(targets
->data
));
246 g_resolver_free_targets(targets
);
247 sipe_dns_query_free(query
);
250 static void dns_srv_response(PurpleSrvResponse
*resp
,
252 struct sipe_dns_query
*query
)
254 /* Ignore spurious responses after disconnect */
255 if (query
->is_valid
) {
256 struct sipe_backend_private
*purple_private
= query
->purple_private
;
258 purple_private
->dns_queries
= g_slist_remove(purple_private
->dns_queries
,
262 query
->callback(query
->extradata
, resp
->hostname
, resp
->port
);
264 query
->callback(query
->extradata
, NULL
, 0);
273 struct sipe_dns_query
*sipe_backend_dns_query_srv(struct sipe_core_public
*sipe_public
,
274 const gchar
*protocol
,
275 const gchar
*transport
,
277 sipe_dns_resolved_cb callback
,
280 struct sipe_dns_query
*query
= g_new(struct sipe_dns_query
, 1);
281 struct sipe_backend_private
*purple_private
= sipe_public
->backend_private
;
282 #if PURPLE_VERSION_CHECK(3,0,0)
283 GResolver
*resolver
= g_resolver_get_default();
286 query
->purple_private
= purple_private
;
287 query
->callback
= callback
;
288 query
->extradata
= data
;
289 query
->is_valid
= TRUE
;
291 purple_private
->dns_queries
= g_slist_prepend(purple_private
->dns_queries
,
294 #if PURPLE_VERSION_CHECK(3,0,0)
295 query
->purple_query_data
= g_cancellable_new();
297 g_resolver_lookup_service_async(resolver
,
301 query
->purple_query_data
,
304 g_object_unref(resolver
);
307 query
->purple_query_data
=
308 #if PURPLE_VERSION_CHECK(2,8,0)
309 purple_srv_resolve_account(
310 purple_private
->account
,
317 (PurpleSrvCallback
) dns_srv_response
,
324 static gboolean
dns_query_deferred_destroy(gpointer user_data
)
327 * All pending events on query have been processed.
328 * Now it is safe to destroy the data structure.
330 SIPE_DEBUG_INFO("dns_query_deferred_destroy: %p", user_data
);
331 sipe_dns_query_free(user_data
);
335 void sipe_backend_dns_query_cancel(struct sipe_dns_query
*query
)
337 SIPE_DEBUG_INFO("sipe_backend_dns_query_cancel: %p", query
);
339 if (query
->is_valid
) {
340 struct sipe_backend_private
*purple_private
= query
->purple_private
;
341 purple_private
->dns_queries
= g_slist_remove(purple_private
->dns_queries
,
344 #if PURPLE_VERSION_CHECK(3,0,0)
345 g_cancellable_cancel(query
->purple_query_data
);
347 switch (query
->type
) {
349 purple_dnsquery_destroy(query
->purple_query_data
);
352 #if PURPLE_VERSION_CHECK(2,8,0)
353 purple_srv_txt_query_destroy(query
->purple_query_data
);
355 purple_srv_cancel(query
->purple_query_data
);
361 /* defer deletion of query data structure to idle callback */
362 query
->is_valid
= FALSE
;
363 g_idle_add(dns_query_deferred_destroy
, query
);
367 void sipe_purple_dns_query_cancel_all(struct sipe_backend_private
*purple_private
)
370 SIPE_DEBUG_INFO_NOFORMAT("sipe_purple_dns_query_cancel_all: entered");
371 while ((entry
= purple_private
->dns_queries
) != NULL
)
372 sipe_backend_dns_query_cancel(entry
->data
);