2 * @file purple-dnsquery.c
6 * Copyright (C) 2010-2016 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(3,0,0)
28 #include "protocols.h"
33 /* wrappers for write() & friends for socket handling */
34 #include "win32/win32dep.h"
37 #include <sys/types.h>
38 #include <sys/socket.h>
39 #include <netinet/in.h>
40 #include <arpa/inet.h>
48 #include "sipe-common.h"
49 #include "sipe-backend.h"
50 #include "sipe-core.h"
52 #include "purple-private.h"
54 struct sipe_dns_query
{
55 struct sipe_backend_private
*purple_private
;
56 sipe_dns_resolved_cb callback
;
58 gpointer purple_query_data
;
60 #if PURPLE_VERSION_CHECK(3,0,0)
70 static void sipe_dns_query_free(struct sipe_dns_query
*query
)
72 #if PURPLE_VERSION_CHECK(3,0,0)
73 g_object_unref(query
->purple_query_data
);
78 #if PURPLE_VERSION_CHECK(3,0,0)
79 static void dns_a_response(GObject
*source
,
83 struct sipe_dns_query
*query
= user_data
;
86 gchar
*address_str
= NULL
;
88 if (!query
->is_valid
) {
89 /* Ignore spurious responses after disconnect */
93 query
->purple_private
->dns_queries
=
94 g_slist_remove(query
->purple_private
->dns_queries
,
97 hosts
= g_resolver_lookup_by_name_finish(G_RESOLVER(source
), res
,
100 if (!error
&& g_list_length(hosts
) > 0) {
101 address_str
= g_inet_address_to_string(hosts
->data
);
104 query
->callback(query
->extradata
, address_str
,
105 address_str
? query
->port
: 0);
109 g_resolver_free_addresses(hosts
);
110 sipe_dns_query_free(query
);
113 static void dns_a_response(GSList
*hosts
,
114 struct sipe_dns_query
*query
,
115 const char *error_message
)
117 char ipstr
[INET6_ADDRSTRLEN
];
118 struct sockaddr
*addr
;
119 const void *addrdata
;
122 /* Ignore spurious responses after disconnect */
123 if (query
->is_valid
) {
124 struct sipe_backend_private
*purple_private
= query
->purple_private
;
126 purple_private
->dns_queries
= g_slist_remove(purple_private
->dns_queries
,
129 if (error_message
|| !g_slist_next(hosts
)) {
130 query
->callback(query
->extradata
, NULL
, 0);
135 addr
= g_slist_next(hosts
)->data
;
136 if (addr
->sa_family
== AF_INET6
) {
137 /* OS provides addr so it must be properly aligned */
138 struct sockaddr_in6
*sin6
= (void *) addr
;
139 addrdata
= &sin6
->sin6_addr
;
140 port
= sin6
->sin6_port
;
142 /* OS provides addr so it must be properly aligned */
143 struct sockaddr_in
*sin
= (void *) addr
;
144 addrdata
= &sin
->sin_addr
;
145 port
= sin
->sin_port
;
148 inet_ntop(addr
->sa_family
, addrdata
, ipstr
, sizeof (ipstr
));
150 query
->callback(query
->extradata
, ipstr
, port
);
155 for (; hosts
; hosts
= g_slist_delete_link(hosts
, hosts
)) {
156 // Free the addrlen, no data in this link
157 hosts
= g_slist_delete_link(hosts
, hosts
);
164 struct sipe_dns_query
*sipe_backend_dns_query_a(struct sipe_core_public
*sipe_public
,
165 const gchar
*hostname
,
167 sipe_dns_resolved_cb callback
,
170 struct sipe_dns_query
*query
= g_new(struct sipe_dns_query
, 1);
171 struct sipe_backend_private
*purple_private
= sipe_public
->backend_private
;
172 #if PURPLE_VERSION_CHECK(3,0,0)
173 GResolver
*resolver
= g_resolver_get_default();
176 query
->purple_private
= purple_private
;
177 query
->callback
= callback
;
178 query
->extradata
= data
;
179 query
->is_valid
= TRUE
;
181 purple_private
->dns_queries
= g_slist_prepend(purple_private
->dns_queries
,
184 #if PURPLE_VERSION_CHECK(3,0,0)
186 query
->purple_query_data
= g_cancellable_new();
188 g_resolver_lookup_by_name_async(resolver
,
190 query
->purple_query_data
,
193 g_object_unref(resolver
);
196 query
->purple_query_data
=
197 #if PURPLE_VERSION_CHECK(2,8,0)
198 purple_dnsquery_a_account(
199 purple_private
->account
,
205 (PurpleDnsQueryConnectFunction
) dns_a_response
,
212 #if PURPLE_VERSION_CHECK(3,0,0)
213 static void dns_srv_response(GObject
*source
,
217 struct sipe_dns_query
*query
= user_data
;
218 GError
*error
= NULL
;
221 if (!query
->is_valid
) {
222 /* Ignore spurious responses after disconnect */
226 query
->purple_private
->dns_queries
=
227 g_slist_remove(query
->purple_private
->dns_queries
,
230 targets
= g_resolver_lookup_service_finish(G_RESOLVER(source
), res
,
233 if (error
|| g_list_length(targets
) == 0) {
234 query
->callback(query
->extradata
, NULL
, 0);
236 query
->callback(query
->extradata
,
237 g_srv_target_get_hostname(targets
->data
),
238 g_srv_target_get_port(targets
->data
));
242 g_resolver_free_targets(targets
);
243 sipe_dns_query_free(query
);
246 static void dns_srv_response(PurpleSrvResponse
*resp
,
248 struct sipe_dns_query
*query
)
250 /* Ignore spurious responses after disconnect */
251 if (query
->is_valid
) {
252 struct sipe_backend_private
*purple_private
= query
->purple_private
;
254 purple_private
->dns_queries
= g_slist_remove(purple_private
->dns_queries
,
258 query
->callback(query
->extradata
, resp
->hostname
, resp
->port
);
260 query
->callback(query
->extradata
, NULL
, 0);
269 struct sipe_dns_query
*sipe_backend_dns_query_srv(struct sipe_core_public
*sipe_public
,
270 const gchar
*protocol
,
271 const gchar
*transport
,
273 sipe_dns_resolved_cb callback
,
276 struct sipe_dns_query
*query
= g_new(struct sipe_dns_query
, 1);
277 struct sipe_backend_private
*purple_private
= sipe_public
->backend_private
;
278 #if PURPLE_VERSION_CHECK(3,0,0)
279 GResolver
*resolver
= g_resolver_get_default();
282 query
->purple_private
= purple_private
;
283 query
->callback
= callback
;
284 query
->extradata
= data
;
285 query
->is_valid
= TRUE
;
287 purple_private
->dns_queries
= g_slist_prepend(purple_private
->dns_queries
,
290 #if PURPLE_VERSION_CHECK(3,0,0)
291 query
->purple_query_data
= g_cancellable_new();
293 g_resolver_lookup_service_async(resolver
,
297 query
->purple_query_data
,
300 g_object_unref(resolver
);
303 query
->purple_query_data
=
304 #if PURPLE_VERSION_CHECK(2,8,0)
305 purple_srv_resolve_account(
306 purple_private
->account
,
313 (PurpleSrvCallback
) dns_srv_response
,
320 static gboolean
dns_query_deferred_destroy(gpointer user_data
)
323 * All pending events on query have been processed.
324 * Now it is safe to destroy the data structure.
326 SIPE_DEBUG_INFO("dns_query_deferred_destroy: %p", user_data
);
327 sipe_dns_query_free(user_data
);
331 void sipe_backend_dns_query_cancel(struct sipe_dns_query
*query
)
333 SIPE_DEBUG_INFO("sipe_backend_dns_query_cancel: %p", query
);
335 if (query
->is_valid
) {
336 struct sipe_backend_private
*purple_private
= query
->purple_private
;
337 purple_private
->dns_queries
= g_slist_remove(purple_private
->dns_queries
,
340 #if PURPLE_VERSION_CHECK(3,0,0)
341 g_cancellable_cancel(query
->purple_query_data
);
343 switch (query
->type
) {
345 purple_dnsquery_destroy(query
->purple_query_data
);
348 #if PURPLE_VERSION_CHECK(2,8,0)
349 purple_srv_txt_query_destroy(query
->purple_query_data
);
351 purple_srv_cancel(query
->purple_query_data
);
357 /* defer deletion of query data structure to idle callback */
358 query
->is_valid
= FALSE
;
359 g_idle_add(dns_query_deferred_destroy
, query
);
363 void sipe_purple_dns_query_cancel_all(struct sipe_backend_private
*purple_private
)
366 SIPE_DEBUG_INFO_NOFORMAT("sipe_purple_dns_query_cancel_all: entered");
367 while ((entry
= purple_private
->dns_queries
) != NULL
)
368 sipe_backend_dns_query_cancel(entry
->data
);