2 This file is part of GNUnet.
3 Copyright (C) 2012-2013 GNUnet e.V.
5 GNUnet is free software; you can redistribute it and/or modify
6 it under the terms of the GNU General Public License as published
7 by the Free Software Foundation; either version 3, or (at your
8 option) any later version.
10 GNUnet is distributed in the hope that it will be useful, but
11 WITHOUT ANY WARRANTY; without even the implied warranty of
12 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
13 General Public License for more details.
15 You should have received a copy of the GNU General Public License
16 along with GNUnet; see the file COPYING. If not, write to the
17 Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor,
18 Boston, MA 02110-1301, USA.
21 * @file gnunet-dns2gns.c
22 * @brief DNS server that translates DNS requests to GNS
23 * @author Christian Grothoff
26 #include <gnunet_util_lib.h>
27 #include <gnunet_dnsparser_lib.h>
28 #include <gnunet_gns_service.h>
29 #include <gnunet_dnsstub_lib.h>
33 * Timeout for DNS requests.
35 #define TIMEOUT GNUNET_TIME_UNIT_MINUTES
38 * Data kept per request.
43 * Socket to use for sending the reply.
45 struct GNUNET_NETWORK_Handle *lsock;
48 * Destination address to use.
53 * Initially, this is the DNS request, it will then be
54 * converted to the DNS response.
56 struct GNUNET_DNSPARSER_Packet *packet;
59 * Our GNS request handle.
61 struct GNUNET_GNS_LookupWithTldRequest *lookup;
64 * Our DNS request handle
66 struct GNUNET_DNSSTUB_RequestSocket *dns_lookup;
69 * Task run on timeout or shutdown to clean up without
72 struct GNUNET_SCHEDULER_Task *timeout_task;
75 * Original UDP request message.
80 * Number of bytes in @e addr.
85 * Number of bytes in @e udp_msg.
90 * ID of the original request.
92 uint16_t original_request_id;
97 * Handle to GNS resolver.
99 struct GNUNET_GNS_Handle *gns;
104 struct GNUNET_DNSSTUB_Context *dns_stub;
107 * Listen socket for IPv4.
109 static struct GNUNET_NETWORK_Handle *listen_socket4;
112 * Listen socket for IPv6.
114 static struct GNUNET_NETWORK_Handle *listen_socket6;
117 * Task for IPv4 socket.
119 static struct GNUNET_SCHEDULER_Task *t4;
122 * Task for IPv6 socket.
124 static struct GNUNET_SCHEDULER_Task *t6;
132 * UDP Port we listen on for inbound DNS requests.
134 static unsigned int listen_port = 53;
137 * Configuration to use.
139 static const struct GNUNET_CONFIGURATION_Handle *cfg;
143 * Task run on shutdown. Cleans up everything.
148 do_shutdown (void *cls)
153 GNUNET_SCHEDULER_cancel (t4);
158 GNUNET_SCHEDULER_cancel (t6);
161 if (NULL != listen_socket4)
163 GNUNET_NETWORK_socket_close (listen_socket4);
164 listen_socket4 = NULL;
166 if (NULL != listen_socket6)
168 GNUNET_NETWORK_socket_close (listen_socket6);
169 listen_socket6 = NULL;
173 GNUNET_GNS_disconnect (gns);
176 if (NULL != dns_stub)
178 GNUNET_DNSSTUB_stop (dns_stub);
185 * Send the response for the given request and clean up.
187 * @param request context for the request.
190 send_response (struct Request *request)
197 GNUNET_DNSPARSER_pack (request->packet,
198 UINT16_MAX /* is this not too much? */,
202 GNUNET_log (GNUNET_ERROR_TYPE_WARNING,
203 _("Failed to pack DNS response into UDP packet!\n"));
207 sret = GNUNET_NETWORK_socket_sendto (request->lsock,
213 (size != (size_t) sret) )
214 GNUNET_log_strerror (GNUNET_ERROR_TYPE_WARNING,
218 GNUNET_SCHEDULER_cancel (request->timeout_task);
219 GNUNET_DNSPARSER_free_packet (request->packet);
220 GNUNET_free (request->udp_msg);
221 GNUNET_free (request);
226 * Task run on timeout. Cleans up request.
228 * @param cls `struct Request *` of the request to clean up
231 do_timeout (void *cls)
233 struct Request *request = cls;
235 if (NULL != request->packet)
236 GNUNET_DNSPARSER_free_packet (request->packet);
237 if (NULL != request->lookup)
238 GNUNET_GNS_lookup_with_tld_cancel (request->lookup);
239 if (NULL != request->dns_lookup)
240 GNUNET_DNSSTUB_resolve_cancel (request->dns_lookup);
241 GNUNET_free (request->udp_msg);
242 GNUNET_free (request);
247 * Iterator called on obtained result for a DNS lookup
250 * @param dns the DNS udp payload
251 * @param r size of the DNS payload
254 dns_result_processor (void *cls,
255 const struct GNUNET_TUN_DnsHeader *dns,
258 struct Request *request = cls;
262 /* DNSSTUB gave up, so we trigger timeout early */
263 GNUNET_SCHEDULER_cancel (request->timeout_task);
264 do_timeout (request);
267 if (request->original_request_id != dns->id)
269 /* for a another query, ignore */
272 request->packet = GNUNET_DNSPARSER_parse ((char*)dns,
274 send_response (request);
279 * Iterator called on obtained result for a GNS lookup.
282 * @param was_gns #GNUNET_NO if the TLD is not configured for GNS
283 * @param rd_count number of records in @a rd
284 * @param rd the records in reply
287 result_processor (void *cls,
290 const struct GNUNET_GNSRECORD_Data *rd)
292 struct Request *request = cls;
293 struct GNUNET_DNSPARSER_Packet *packet;
294 struct GNUNET_DNSPARSER_Record rec;
296 request->lookup = NULL;
297 if (GNUNET_NO == was_gns)
299 /* TLD not configured for GNS, fall back to DNS */
300 GNUNET_log (GNUNET_ERROR_TYPE_DEBUG,
301 "Using DNS resolver IP `%s' to resolve `%s'\n",
303 request->packet->queries[0].name);
304 request->original_request_id = request->packet->id;
305 GNUNET_DNSPARSER_free_packet (request->packet);
306 request->packet = NULL;
307 request->dns_lookup = GNUNET_DNSSTUB_resolve (dns_stub,
309 request->udp_msg_size,
310 &dns_result_processor,
314 packet = request->packet;
315 packet->flags.query_or_response = 1;
316 packet->flags.return_code = GNUNET_TUN_DNS_RETURN_CODE_NO_ERROR;
317 packet->flags.checking_disabled = 0;
318 packet->flags.authenticated_data = 1;
319 packet->flags.zero = 0;
320 packet->flags.recursion_available = 1;
321 packet->flags.message_truncated = 0;
322 packet->flags.authoritative_answer = 0;
323 //packet->flags.opcode = GNUNET_TUN_DNS_OPCODE_STATUS; // ???
324 for (uint32_t i=0;i<rd_count;i++)
326 // FIXME: do we need to hanlde #GNUNET_GNSRECORD_RF_SHADOW_RECORD
327 // here? Or should we do this in libgnunetgns?
328 rec.expiration_time.abs_value_us = rd[i].expiration_time;
329 switch (rd[i].record_type)
331 case GNUNET_DNSPARSER_TYPE_A:
332 GNUNET_assert (sizeof (struct in_addr) == rd[i].data_size);
333 rec.name = GNUNET_strdup (packet->queries[0].name);
334 rec.dns_traffic_class = GNUNET_TUN_DNS_CLASS_INTERNET;
335 rec.type = GNUNET_DNSPARSER_TYPE_A;
336 rec.data.raw.data = GNUNET_new (struct in_addr);
337 GNUNET_memcpy (rec.data.raw.data,
340 rec.data.raw.data_len = sizeof (struct in_addr);
341 GNUNET_array_append (packet->answers,
345 case GNUNET_DNSPARSER_TYPE_AAAA:
346 GNUNET_assert (sizeof (struct in6_addr) == rd[i].data_size);
347 rec.name = GNUNET_strdup (packet->queries[0].name);
348 rec.data.raw.data = GNUNET_new (struct in6_addr);
349 rec.dns_traffic_class = GNUNET_TUN_DNS_CLASS_INTERNET;
350 rec.type = GNUNET_DNSPARSER_TYPE_AAAA;
351 GNUNET_memcpy (rec.data.raw.data,
354 rec.data.raw.data_len = sizeof (struct in6_addr);
355 GNUNET_array_append (packet->answers,
359 case GNUNET_DNSPARSER_TYPE_CNAME:
360 rec.name = GNUNET_strdup (packet->queries[0].name);
361 rec.data.hostname = GNUNET_strdup (rd[i].data);
362 rec.dns_traffic_class = GNUNET_TUN_DNS_CLASS_INTERNET;
363 rec.type = GNUNET_DNSPARSER_TYPE_CNAME;
364 GNUNET_memcpy (rec.data.hostname,
367 GNUNET_array_append (packet->answers,
376 send_response (request);
381 * Handle DNS request.
383 * @param lsock socket to use for sending the reply
384 * @param addr address to use for sending the reply
385 * @param addr_len number of bytes in @a addr
386 * @param udp_msg DNS request payload
387 * @param udp_msg_size number of bytes in @a udp_msg
390 handle_request (struct GNUNET_NETWORK_Handle *lsock,
396 struct Request *request;
397 struct GNUNET_DNSPARSER_Packet *packet;
399 packet = GNUNET_DNSPARSER_parse (udp_msg,
403 GNUNET_log (GNUNET_ERROR_TYPE_WARNING,
404 _("Cannot parse DNS request from %s\n"),
405 GNUNET_a2s (addr, addr_len));
408 GNUNET_log (GNUNET_ERROR_TYPE_DEBUG,
409 "Received request for `%s' with flags %u, #answers %d, #auth %d, #additional %d\n",
410 packet->queries[0].name,
411 (unsigned int) packet->flags.query_or_response,
412 (int) packet->num_answers,
413 (int) packet->num_authority_records,
414 (int) packet->num_additional_records);
415 if ( (0 != packet->flags.query_or_response) ||
416 (0 != packet->num_answers) ||
417 (0 != packet->num_authority_records))
419 GNUNET_log (GNUNET_ERROR_TYPE_WARNING,
420 _("Received malformed DNS request from %s\n"),
421 GNUNET_a2s (addr, addr_len));
422 GNUNET_DNSPARSER_free_packet (packet);
425 if ( (1 != packet->num_queries) )
427 GNUNET_log (GNUNET_ERROR_TYPE_WARNING,
428 _("Received unsupported DNS request from %s\n"),
431 GNUNET_DNSPARSER_free_packet (packet);
434 request = GNUNET_malloc (sizeof (struct Request) + addr_len);
435 request->lsock = lsock;
436 request->packet = packet;
437 request->addr = &request[1];
438 request->addr_len = addr_len;
439 GNUNET_memcpy (&request[1],
442 request->udp_msg_size = udp_msg_size;
443 request->udp_msg = GNUNET_memdup (udp_msg,
445 request->timeout_task = GNUNET_SCHEDULER_add_delayed (TIMEOUT,
448 GNUNET_log (GNUNET_ERROR_TYPE_DEBUG,
449 "Calling GNS on `%s'\n",
450 packet->queries[0].name);
451 request->lookup = GNUNET_GNS_lookup_with_tld (gns,
452 packet->queries[0].name,
453 packet->queries[0].type,
461 * Task to read IPv4 DNS packets.
463 * @param cls the 'listen_socket4'
466 read_dns4 (void *cls)
468 struct sockaddr_in v4;
471 const struct GNUNET_SCHEDULER_TaskContext *tc;
473 GNUNET_assert (listen_socket4 == cls);
474 t4 = GNUNET_SCHEDULER_add_read_net (GNUNET_TIME_UNIT_FOREVER_REL,
478 tc = GNUNET_SCHEDULER_get_task_context ();
479 if (0 == (GNUNET_SCHEDULER_REASON_READ_READY & tc->reason))
480 return; /* shutdown? */
481 size = GNUNET_NETWORK_socket_recvfrom_amount (listen_socket4);
485 return; /* read error!? */
491 addrlen = sizeof (v4);
492 sret = GNUNET_NETWORK_socket_recvfrom (listen_socket4,
495 (struct sockaddr *) &v4,
499 GNUNET_log_strerror (GNUNET_ERROR_TYPE_WARNING,
503 GNUNET_break (size == sret);
504 handle_request (listen_socket4,
514 * Task to read IPv6 DNS packets.
516 * @param cls the 'listen_socket6'
519 read_dns6 (void *cls)
521 struct sockaddr_in6 v6;
524 const struct GNUNET_SCHEDULER_TaskContext *tc;
526 GNUNET_assert (listen_socket6 == cls);
527 t6 = GNUNET_SCHEDULER_add_read_net (GNUNET_TIME_UNIT_FOREVER_REL,
531 tc = GNUNET_SCHEDULER_get_task_context ();
532 if (0 == (GNUNET_SCHEDULER_REASON_READ_READY & tc->reason))
533 return; /* shutdown? */
534 size = GNUNET_NETWORK_socket_recvfrom_amount (listen_socket6);
538 return; /* read error!? */
544 addrlen = sizeof (v6);
545 sret = GNUNET_NETWORK_socket_recvfrom (listen_socket6,
548 (struct sockaddr *) &v6,
552 GNUNET_log_strerror (GNUNET_ERROR_TYPE_WARNING,
556 GNUNET_break (size == sret);
557 handle_request (listen_socket6,
567 * Main function that will be run.
570 * @param args remaining command-line arguments
571 * @param cfgfile name of the configuration file used (for saving, can be NULL!)
572 * @param c configuration
578 const struct GNUNET_CONFIGURATION_Handle *c)
586 GNUNET_log (GNUNET_ERROR_TYPE_ERROR,
587 _("No DNS server specified!\n"));
590 GNUNET_SCHEDULER_add_shutdown (&do_shutdown,
592 if (NULL == (gns = GNUNET_GNS_connect (cfg)))
594 GNUNET_assert (NULL != (dns_stub = GNUNET_DNSSTUB_start (128)));
596 GNUNET_DNSSTUB_add_dns_ip (dns_stub,
599 GNUNET_DNSSTUB_stop (dns_stub);
600 GNUNET_GNS_disconnect (gns);
604 listen_socket4 = GNUNET_NETWORK_socket_create (PF_INET,
607 if (NULL != listen_socket4)
609 struct sockaddr_in v4;
611 memset (&v4, 0, sizeof (v4));
612 v4.sin_family = AF_INET;
613 #if HAVE_SOCKADDR_IN_SIN_LEN
614 v4.sin_len = sizeof (v4);
616 v4.sin_port = htons (listen_port);
618 GNUNET_NETWORK_socket_bind (listen_socket4,
619 (struct sockaddr *) &v4,
622 GNUNET_log_strerror (GNUNET_ERROR_TYPE_ERROR, "bind");
623 GNUNET_NETWORK_socket_close (listen_socket4);
624 listen_socket4 = NULL;
627 listen_socket6 = GNUNET_NETWORK_socket_create (PF_INET6,
630 if (NULL != listen_socket6)
632 struct sockaddr_in6 v6;
634 memset (&v6, 0, sizeof (v6));
635 v6.sin6_family = AF_INET6;
636 #if HAVE_SOCKADDR_IN_SIN_LEN
637 v6.sin6_len = sizeof (v6);
639 v6.sin6_port = htons (listen_port);
641 GNUNET_NETWORK_socket_bind (listen_socket6,
642 (struct sockaddr *) &v6,
645 GNUNET_log_strerror (GNUNET_ERROR_TYPE_ERROR, "bind");
646 GNUNET_NETWORK_socket_close (listen_socket6);
647 listen_socket6 = NULL;
650 if ( (NULL == listen_socket4) &&
651 (NULL == listen_socket6) )
653 GNUNET_GNS_disconnect (gns);
655 GNUNET_DNSSTUB_stop (dns_stub);
659 if (NULL != listen_socket4)
660 t4 = GNUNET_SCHEDULER_add_read_net (GNUNET_TIME_UNIT_FOREVER_REL,
664 if (NULL != listen_socket6)
665 t6 = GNUNET_SCHEDULER_add_read_net (GNUNET_TIME_UNIT_FOREVER_REL,
673 * The main function for the dns2gns daemon.
675 * @param argc number of arguments from the command line
676 * @param argv command line arguments
677 * @return 0 ok, 1 on error
683 struct GNUNET_GETOPT_CommandLineOption options[] = {
684 GNUNET_GETOPT_option_string ('d',
687 gettext_noop ("IP of recursive DNS resolver to use (required)"),
689 GNUNET_GETOPT_option_uint ('p',
692 gettext_noop ("UDP port to listen on for inbound DNS requests; default: 2853"),
694 GNUNET_GETOPT_OPTION_END
699 GNUNET_STRINGS_get_utf8_args (argc, argv,
702 GNUNET_log_setup ("gnunet-dns2gns",
707 GNUNET_PROGRAM_run (argc, argv,
709 _("GNUnet DNS-to-GNS proxy (a DNS server)"),
711 &run, NULL)) ? 0 : 1;
712 GNUNET_free ((void*) argv);
716 /* end of gnunet-dns2gns.c */