2 * @file purple-dnsquery.c
6 * Copyright (C) 2010-2017 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-backend.h"
49 #include "sipe-core.h"
51 #include "purple-private.h"
53 struct sipe_dns_query
{
54 struct sipe_backend_private
*purple_private
;
55 sipe_dns_resolved_cb callback
;
57 gpointer purple_query_data
;
59 #if PURPLE_VERSION_CHECK(3,0,0)
69 static void sipe_dns_query_free(struct sipe_dns_query
*query
)
71 #if PURPLE_VERSION_CHECK(3,0,0)
72 g_object_unref(query
->purple_query_data
);
77 #if PURPLE_VERSION_CHECK(3,0,0)
78 static void dns_a_response(GObject
*source
,
82 struct sipe_dns_query
*query
= user_data
;
85 gchar
*address_str
= NULL
;
87 if (!query
->is_valid
) {
88 /* Ignore spurious responses after disconnect */
92 query
->purple_private
->dns_queries
=
93 g_slist_remove(query
->purple_private
->dns_queries
,
96 hosts
= g_resolver_lookup_by_name_finish(G_RESOLVER(source
), res
,
99 if (!error
&& g_list_length(hosts
) > 0) {
100 address_str
= g_inet_address_to_string(hosts
->data
);
103 query
->callback(query
->extradata
, address_str
,
104 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
));
243 g_resolver_free_targets(targets
);
244 sipe_dns_query_free(query
);
247 static void dns_srv_response(PurpleSrvResponse
*resp
,
249 struct sipe_dns_query
*query
)
251 /* Ignore spurious responses after disconnect */
252 if (query
->is_valid
) {
253 struct sipe_backend_private
*purple_private
= query
->purple_private
;
255 purple_private
->dns_queries
= g_slist_remove(purple_private
->dns_queries
,
259 query
->callback(query
->extradata
, resp
->hostname
, resp
->port
);
261 query
->callback(query
->extradata
, NULL
, 0);
270 struct sipe_dns_query
*sipe_backend_dns_query_srv(struct sipe_core_public
*sipe_public
,
271 const gchar
*protocol
,
272 const gchar
*transport
,
274 sipe_dns_resolved_cb callback
,
277 struct sipe_dns_query
*query
= g_new(struct sipe_dns_query
, 1);
278 struct sipe_backend_private
*purple_private
= sipe_public
->backend_private
;
279 #if PURPLE_VERSION_CHECK(3,0,0)
280 GResolver
*resolver
= g_resolver_get_default();
283 query
->purple_private
= purple_private
;
284 query
->callback
= callback
;
285 query
->extradata
= data
;
286 query
->is_valid
= TRUE
;
288 purple_private
->dns_queries
= g_slist_prepend(purple_private
->dns_queries
,
291 #if PURPLE_VERSION_CHECK(3,0,0)
292 query
->purple_query_data
= g_cancellable_new();
294 g_resolver_lookup_service_async(resolver
,
298 query
->purple_query_data
,
301 g_object_unref(resolver
);
304 query
->purple_query_data
=
305 #if PURPLE_VERSION_CHECK(2,8,0)
306 purple_srv_resolve_account(
307 purple_private
->account
,
314 (PurpleSrvCallback
) dns_srv_response
,
321 static gboolean
dns_query_deferred_destroy(gpointer user_data
)
324 * All pending events on query have been processed.
325 * Now it is safe to destroy the data structure.
327 SIPE_DEBUG_INFO("dns_query_deferred_destroy: %p", user_data
);
328 sipe_dns_query_free(user_data
);
332 void sipe_backend_dns_query_cancel(struct sipe_dns_query
*query
)
334 SIPE_DEBUG_INFO("sipe_backend_dns_query_cancel: %p", query
);
336 if (query
->is_valid
) {
337 struct sipe_backend_private
*purple_private
= query
->purple_private
;
338 purple_private
->dns_queries
= g_slist_remove(purple_private
->dns_queries
,
341 #if PURPLE_VERSION_CHECK(3,0,0)
342 g_cancellable_cancel(query
->purple_query_data
);
344 switch (query
->type
) {
346 purple_dnsquery_destroy(query
->purple_query_data
);
349 #if PURPLE_VERSION_CHECK(2,8,0)
350 purple_srv_txt_query_destroy(query
->purple_query_data
);
352 purple_srv_cancel(query
->purple_query_data
);
358 /* defer deletion of query data structure to idle callback */
359 query
->is_valid
= FALSE
;
360 g_idle_add(dns_query_deferred_destroy
, query
);
364 void sipe_purple_dns_query_cancel_all(struct sipe_backend_private
*purple_private
)
367 SIPE_DEBUG_INFO_NOFORMAT("sipe_purple_dns_query_cancel_all: entered");
368 while ((entry
= purple_private
->dns_queries
) != NULL
)
369 sipe_backend_dns_query_cancel(entry
->data
);