2 This file is part of GNUnet.
3 Copyright (C) 2007-2016 GNUnet e.V.
5 GNUnet is free software: you can redistribute it and/or modify it
6 under the terms of the GNU Affero General Public License as published
7 by the Free Software Foundation, either version 3 of the License,
8 or (at your 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 Affero General Public License for more details.
15 You should have received a copy of the GNU Affero General Public License
16 along with this program. If not, see <http://www.gnu.org/licenses/>.
18 SPDX-License-Identifier: AGPL3.0-or-later
22 * @file util/gnunet-service-resolver.c
23 * @brief code to do DNS resolution
24 * @author Christian Grothoff
27 #include "gnunet_util_lib.h"
28 #include "gnunet_protocols.h"
29 #include "gnunet_statistics_service.h"
34 * How long do we wait for DNS answers?
36 #define DNS_TIMEOUT GNUNET_TIME_relative_multiply (GNUNET_TIME_UNIT_SECONDS, 30)
39 * Maximum number of hostnames we cache results for.
41 #define MAX_CACHE 1024
44 * Entry in list of cached DNS records for a hostname.
46 struct RecordListEntry
49 * This is a doubly linked list.
51 struct RecordListEntry *next;
54 * This is a doubly linked list.
56 struct RecordListEntry *prev;
61 struct GNUNET_DNSPARSER_Record *record;
66 * A cached DNS lookup result.
71 * This is a doubly linked list.
73 struct ResolveCache *next;
76 * This is a doubly linked list.
78 struct ResolveCache *prev;
81 * Which hostname is this cache for?
86 * head of a double linked list containing the lookup results
88 struct RecordListEntry *records_head;
91 * tail of a double linked list containing the lookup results
93 struct RecordListEntry *records_tail;
98 * Information about pending lookups.
105 struct ActiveLookup *next;
110 struct ActiveLookup *prev;
113 * The client that queried the records contained in this cache entry.
115 struct GNUNET_SERVICE_Client *client;
118 * handle for cancelling a request
120 struct GNUNET_DNSSTUB_RequestSocket *resolve_handle;
123 * handle for the resolution timeout task
125 struct GNUNET_SCHEDULER_Task *timeout_task;
128 * Which hostname are we resolving?
133 * If @a record_type is #GNUNET_DNSPARSER_TYPE_ALL, did we go again
134 * for the AAAA records yet?
139 * type of queried DNS record
141 uint16_t record_type;
144 * Unique request ID of a client if a query for this hostname/record_type
145 * is currently pending, undefined otherwise.
147 uint32_t client_request_id;
150 * Unique DNS request ID of a client if a query for this hostname/record_type
151 * is currently pending, undefined otherwise.
158 * Start of the linked list of cached DNS lookup results.
160 static struct ResolveCache *cache_head;
163 * Tail of the linked list of cached DNS lookup results.
165 static struct ResolveCache *cache_tail;
168 * Head of the linked list of DNS lookup results from /etc/hosts.
170 static struct ResolveCache *hosts_head;
173 * Tail of the linked list of DNS lookup results from /etc/hosts.
175 static struct ResolveCache *hosts_tail;
178 * Start of the linked list of active DNS lookups.
180 static struct ActiveLookup *lookup_head;
183 * Tail of the linked list of active DNS lookups.
185 static struct ActiveLookup *lookup_tail;
188 * context of dnsstub library
190 static struct GNUNET_DNSSTUB_Context *dnsstub_ctx;
193 * My domain, to be appended to the hostname to get a FQDN.
195 static char *my_domain;
198 * How many entries do we have in #cache_head DLL?
200 static unsigned int cache_size;
204 * Remove @a entry from cache.
206 * @param rc entry to free
209 free_cache_entry (struct ResolveCache *rc)
211 struct RecordListEntry *pos;
213 while (NULL != (pos = rc->records_head))
215 GNUNET_CONTAINER_DLL_remove (rc->records_head, rc->records_tail, pos);
216 GNUNET_DNSPARSER_free_record (pos->record);
217 GNUNET_free (pos->record);
220 GNUNET_free_non_null (rc->hostname);
221 GNUNET_CONTAINER_DLL_remove (cache_head, cache_tail, rc);
228 * Remove @a entry from cache.
230 * @param rc entry to free
233 free_hosts_entry (struct ResolveCache *rc)
235 struct RecordListEntry *pos;
237 while (NULL != (pos = rc->records_head))
239 GNUNET_CONTAINER_DLL_remove (rc->records_head, rc->records_tail, pos);
240 GNUNET_DNSPARSER_free_record (pos->record);
241 GNUNET_free (pos->record);
244 GNUNET_free_non_null (rc->hostname);
245 GNUNET_CONTAINER_DLL_remove (hosts_head, hosts_tail, rc);
252 * Release resources associated with @a al
254 * @param al an active lookup
257 free_active_lookup (struct ActiveLookup *al)
259 GNUNET_CONTAINER_DLL_remove (lookup_head, lookup_tail, al);
260 if (NULL != al->resolve_handle)
262 GNUNET_DNSSTUB_resolve_cancel (al->resolve_handle);
263 al->resolve_handle = NULL;
265 if (NULL != al->timeout_task)
267 GNUNET_SCHEDULER_cancel (al->timeout_task);
268 al->timeout_task = NULL;
270 GNUNET_free_non_null (al->hostname);
276 * Find out if the configuration file line contains a string
277 * starting with "nameserver ", and if so, return a copy of
278 * the nameserver's IP.
280 * @param line line to parse
281 * @param line_len number of characters in @a line
282 * @return NULL if no nameserver is configured in this @a line
285 extract_dns_server (const char *line, size_t line_len)
287 if (0 == strncmp (line, "nameserver ", strlen ("nameserver ")))
288 return GNUNET_strndup (line + strlen ("nameserver "),
289 line_len - strlen ("nameserver "));
295 * Find out if the configuration file line contains a string
296 * starting with "search ", and if so, return a copy of
297 * the machine's search domain.
299 * @param line line to parse
300 * @param line_len number of characters in @a line
301 * @return NULL if no nameserver is configured in this @a line
304 extract_search_domain (const char *line, size_t line_len)
306 if (0 == strncmp (line, "search ", strlen ("search ")))
307 return GNUNET_strndup (line + strlen ("search "),
308 line_len - strlen ("search "));
314 * Reads the list of nameservers from /etc/resolve.conf
316 * @param server_addrs[out] a list of null-terminated server address strings
317 * @return the number of server addresses in @server_addrs, -1 on error
320 lookup_dns_servers (char ***server_addrs)
322 struct GNUNET_DISK_FileHandle *fh;
323 struct GNUNET_DISK_MapHandle *mh;
327 unsigned int num_dns_servers;
329 fh = GNUNET_DISK_file_open ("/etc/resolv.conf",
330 GNUNET_DISK_OPEN_READ,
331 GNUNET_DISK_PERM_NONE);
334 GNUNET_log (GNUNET_ERROR_TYPE_ERROR,
335 "Could not open /etc/resolv.conf. "
336 "DNS resolution will not be possible.\n");
339 if (GNUNET_OK != GNUNET_DISK_file_handle_size (fh, &bytes_read))
341 GNUNET_log (GNUNET_ERROR_TYPE_ERROR,
342 "Could not determine size of /etc/resolv.conf. "
343 "DNS resolution will not be possible.\n");
344 GNUNET_DISK_file_close (fh);
347 if (((unsigned long long) bytes_read) > (unsigned long long) SIZE_MAX)
349 GNUNET_log (GNUNET_ERROR_TYPE_ERROR,
350 "/etc/resolv.conf file too large to mmap. "
351 "DNS resolution will not be possible.\n");
352 GNUNET_DISK_file_close (fh);
355 buf = GNUNET_DISK_file_map (fh,
357 GNUNET_DISK_MAP_TYPE_READ,
358 (size_t) bytes_read);
359 *server_addrs = NULL;
362 while (read_offset < (size_t) bytes_read)
368 newline = strchr (buf + read_offset, '\n');
371 line_len = newline - buf - read_offset;
372 dns_server = extract_dns_server (buf + read_offset, line_len);
373 if (NULL != dns_server)
375 GNUNET_array_append (*server_addrs, num_dns_servers, dns_server);
377 else if (NULL == my_domain)
379 my_domain = extract_search_domain (buf + read_offset, line_len);
381 read_offset += line_len + 1;
383 GNUNET_DISK_file_unmap (mh);
384 GNUNET_DISK_file_close (fh);
385 return (int) num_dns_servers;
390 * Compute name to use for DNS reverse lookups from @a ip.
392 * @param ip IP address to resolve, in binary format, network byte order
393 * @param af address family of @a ip, AF_INET or AF_INET6
396 make_reverse_hostname (const void *ip, int af)
398 char *buf = GNUNET_new_array (80, char);
403 struct in_addr *addr = (struct in_addr *) ip;
404 uint32_t ip_int = addr->s_addr;
406 for (int i = 3; i >= 0; i--)
409 GNUNET_snprintf (buf + pos, 80 - pos, "%u.", ((uint8_t *) &ip_int)[i]);
417 pos += GNUNET_snprintf (buf + pos, 80 - pos, "in-addr.arpa");
419 else if (AF_INET6 == af)
421 struct in6_addr *addr = (struct in6_addr *) ip;
422 for (int i = 15; i >= 0; i--)
425 GNUNET_snprintf (buf + pos, 80 - pos, "%x.", addr->s6_addr[i] & 0xf);
432 n = GNUNET_snprintf (buf + pos, 80 - pos, "%x.", addr->s6_addr[i] >> 4);
440 pos += GNUNET_snprintf (buf + pos, 80 - pos, "ip6.arpa");
448 * Send DNS @a record back to our @a client.
450 * @param record information to transmit
451 * @param record_type requested record type from client
452 * @param client_request_id to which request are we responding
453 * @param client where to send @a record
454 * @return #GNUNET_YES if we sent a reply,
455 * #GNUNET_NO if the record type is not understood or
456 * does not match @a record_type
459 send_reply (struct GNUNET_DNSPARSER_Record *record,
460 uint16_t record_type,
461 uint32_t client_request_id,
462 struct GNUNET_SERVICE_Client *client)
464 struct GNUNET_RESOLVER_ResponseMessage *msg;
465 struct GNUNET_MQ_Envelope *env;
469 switch (record->type)
471 case GNUNET_DNSPARSER_TYPE_CNAME:
472 if (GNUNET_DNSPARSER_TYPE_CNAME != record_type)
474 payload = record->data.hostname;
475 payload_len = strlen (record->data.hostname) + 1;
478 case GNUNET_DNSPARSER_TYPE_PTR:
479 if (GNUNET_DNSPARSER_TYPE_PTR != record_type)
481 payload = record->data.hostname;
482 payload_len = strlen (record->data.hostname) + 1;
485 case GNUNET_DNSPARSER_TYPE_A:
486 if ((GNUNET_DNSPARSER_TYPE_A != record_type) &&
487 (GNUNET_DNSPARSER_TYPE_ALL != record_type))
489 payload = record->data.raw.data;
490 payload_len = record->data.raw.data_len;
493 case GNUNET_DNSPARSER_TYPE_AAAA:
494 if ((GNUNET_DNSPARSER_TYPE_AAAA != record_type) &&
495 (GNUNET_DNSPARSER_TYPE_ALL != record_type))
497 payload = record->data.raw.data;
498 payload_len = record->data.raw.data_len;
502 GNUNET_log (GNUNET_ERROR_TYPE_DEBUG,
503 "Cannot handle DNS response type %u: not supported here\n",
507 env = GNUNET_MQ_msg_extra (msg,
509 GNUNET_MESSAGE_TYPE_RESOLVER_RESPONSE);
510 msg->client_id = client_request_id;
511 GNUNET_memcpy (&msg[1], payload, payload_len);
512 GNUNET_MQ_send (GNUNET_SERVICE_client_get_mq (client), env);
518 * Send message to @a client that we transmitted all
519 * responses for @a client_request_id
521 * @param client_request_id to which request are we responding
522 * @param client where to send @a record
525 send_end_msg (uint32_t client_request_id, struct GNUNET_SERVICE_Client *client)
527 struct GNUNET_RESOLVER_ResponseMessage *msg;
528 struct GNUNET_MQ_Envelope *env;
530 GNUNET_log (GNUNET_ERROR_TYPE_DEBUG, "Sending END message\n");
531 env = GNUNET_MQ_msg (msg, GNUNET_MESSAGE_TYPE_RESOLVER_RESPONSE);
532 msg->client_id = client_request_id;
533 GNUNET_MQ_send (GNUNET_SERVICE_client_get_mq (client), env);
538 * Remove expired entries from @a rc
540 * @param rc entry in resolver cache
541 * @return #GNUNET_YES if @a rc was completely expired
542 * #GNUNET_NO if some entries are left
545 remove_expired (struct ResolveCache *rc)
547 struct GNUNET_TIME_Absolute now = GNUNET_TIME_absolute_get ();
548 struct RecordListEntry *n;
550 for (struct RecordListEntry *pos = rc->records_head; NULL != pos; pos = n)
553 if (now.abs_value_us > pos->record->expiration_time.abs_value_us)
555 GNUNET_CONTAINER_DLL_remove (rc->records_head, rc->records_tail, pos);
556 GNUNET_DNSPARSER_free_record (pos->record);
557 GNUNET_free (pos->record);
561 if (NULL == rc->records_head)
563 free_cache_entry (rc);
571 * Process DNS request for @a hostname with request ID @a request_id
572 * from @a client demanding records of type @a record_type.
574 * @param hostname DNS name to resolve
575 * @param record_type desired record type
576 * @param client_request_id client's request ID
577 * @param client who should get the result?
580 process_get (const char *hostname,
581 uint16_t record_type,
582 uint32_t client_request_id,
583 struct GNUNET_SERVICE_Client *client);
587 * Get an IP address as a string (works for both IPv4 and IPv6). Note
588 * that the resolution happens asynchronously and that the first call
589 * may not immediately result in the FQN (but instead in a
590 * human-readable IP address).
592 * @param hostname what hostname was to be resolved
593 * @param record_type what type of record was requested
594 * @param client_request_id unique identification of the client's request
595 * @param client handle to the client making the request (for sending the reply)
598 try_cache (const char *hostname,
599 uint16_t record_type,
600 uint32_t client_request_id,
601 struct GNUNET_SERVICE_Client *client)
603 struct ResolveCache *pos;
604 struct ResolveCache *next;
608 in_hosts = GNUNET_NO;
609 for (pos = hosts_head; NULL != pos; pos = pos->next)
610 if (0 == strcmp (pos->hostname, hostname))
612 in_hosts = GNUNET_YES;
618 for (pos = next; NULL != pos; pos = next)
621 if (GNUNET_YES == remove_expired (pos))
623 if (0 == strcmp (pos->hostname, hostname))
629 GNUNET_log (GNUNET_ERROR_TYPE_DEBUG, "No cache entry for '%s'\n", hostname);
632 if ((GNUNET_NO == in_hosts) && (cache_head != pos))
634 /* move result to head to achieve LRU for cache eviction */
635 GNUNET_CONTAINER_DLL_remove (cache_head, cache_tail, pos);
636 GNUNET_CONTAINER_DLL_insert (cache_head, cache_tail, pos);
639 for (struct RecordListEntry *rle = pos->records_head; NULL != rle;
642 const struct GNUNET_DNSPARSER_Record *record = rle->record;
644 GNUNET_log (GNUNET_ERROR_TYPE_DEBUG,
645 "Checking cache entry for '%s', record is for '%s'\n",
648 if ((GNUNET_DNSPARSER_TYPE_CNAME == record->type) &&
649 (GNUNET_DNSPARSER_TYPE_CNAME != record_type) && (GNUNET_NO == found))
651 const char *hostname = record->data.hostname;
653 process_get (hostname, record_type, client_request_id, client);
654 return GNUNET_YES; /* counts as a cache "hit" */
656 if (0 == strcmp (record->name, hostname))
657 found |= send_reply (rle->record, record_type, client_request_id, client);
659 if (GNUNET_NO == found)
660 return GNUNET_NO; /* had records, but none matched! */
661 send_end_msg (client_request_id, client);
667 * Create DNS query for @a hostname of type @a type
668 * with DNS request ID @a dns_id.
670 * @param hostname DNS name to query
671 * @param type requested DNS record type
672 * @param dns_id what should be the DNS request ID
673 * @param packet_buf[out] where to write the request packet
674 * @param packet_size[out] set to size of @a packet_buf on success
675 * @return #GNUNET_OK on success
678 pack (const char *hostname,
684 struct GNUNET_DNSPARSER_Query query;
685 struct GNUNET_DNSPARSER_Packet packet;
687 query.name = (char *) hostname;
689 query.dns_traffic_class = GNUNET_TUN_DNS_CLASS_INTERNET;
690 memset (&packet, 0, sizeof(packet));
691 packet.num_queries = 1;
692 packet.queries = &query;
693 packet.id = htons (dns_id);
694 packet.flags.recursion_desired = 1;
696 GNUNET_DNSPARSER_pack (&packet, UINT16_MAX, packet_buf, packet_size))
698 GNUNET_log (GNUNET_ERROR_TYPE_ERROR,
699 "Failed to pack query for hostname `%s'\n",
702 return GNUNET_SYSERR;
708 cache_answers (const char *name,
709 struct GNUNET_DNSPARSER_Record *records,
710 unsigned int num_records)
712 struct ResolveCache *rc;
713 struct GNUNET_DNSPARSER_Record *record;
714 struct RecordListEntry *rle;
716 for (unsigned int i = 0; i != num_records; i++)
718 record = &records[i];
720 for (rc = cache_head; NULL != rc; rc = rc->next)
721 if (0 == strcasecmp (rc->hostname, name))
725 rc = GNUNET_new (struct ResolveCache);
726 rc->hostname = GNUNET_strdup (name);
727 GNUNET_log (GNUNET_ERROR_TYPE_DEBUG,
728 "Caching record for name %s under %s\n",
730 GNUNET_CONTAINER_DLL_insert (cache_head, cache_tail, rc);
733 /* TODO: ought to check first if we have this exact record
734 already in the cache! */
735 rle = GNUNET_new (struct RecordListEntry);
736 rle->record = GNUNET_DNSPARSER_duplicate_record (record);
737 GNUNET_CONTAINER_DLL_insert (rc->records_head, rc->records_tail, rle);
742 * We got a result from DNS. Add it to the cache and
743 * see if we can make our client happy...
745 * @param cls the `struct ActiveLookup`
746 * @param dns the DNS response
747 * @param dns_len number of bytes in @a dns
750 handle_resolve_result (void *cls,
751 const struct GNUNET_TUN_DnsHeader *dns,
754 struct ActiveLookup *al = cls;
755 struct GNUNET_DNSPARSER_Packet *parsed;
757 parsed = GNUNET_DNSPARSER_parse ((const char *) dns, dns_len);
760 GNUNET_log (GNUNET_ERROR_TYPE_ERROR,
761 "Failed to parse DNS reply (hostname %s, request ID %u)\n",
766 if (al->dns_id != ntohs (parsed->id))
768 GNUNET_log (GNUNET_ERROR_TYPE_DEBUG,
769 "Request ID in DNS reply does not match\n");
770 GNUNET_DNSPARSER_free_packet (parsed);
773 if (0 == parsed->num_answers + parsed->num_authority_records
774 + parsed->num_additional_records)
776 GNUNET_log (GNUNET_ERROR_TYPE_DEBUG,
777 "DNS reply (hostname %s, request ID %u) contains no answers\n",
779 (unsigned int) al->client_request_id);
780 /* resume by trying again from cache */
781 if (GNUNET_NO == try_cache (al->hostname,
783 al->client_request_id,
785 /* cache failed, tell client we could not get an answer */
787 send_end_msg (al->client_request_id, al->client);
789 GNUNET_DNSPARSER_free_packet (parsed);
790 free_active_lookup (al);
793 /* LRU-based cache eviction: we remove from tail */
794 while (cache_size > MAX_CACHE)
795 free_cache_entry (cache_tail);
797 GNUNET_log (GNUNET_ERROR_TYPE_DEBUG,
798 "Got reply for hostname %s and request ID %u\n",
800 (unsigned int) al->client_request_id);
802 cache_answers (al->hostname, parsed->answers, parsed->num_answers);
803 cache_answers (al->hostname,
804 parsed->authority_records,
805 parsed->num_authority_records);
806 cache_answers (al->hostname,
807 parsed->additional_records,
808 parsed->num_additional_records);
810 /* see if we need to do the 2nd request for AAAA records */
811 if ((GNUNET_DNSPARSER_TYPE_ALL == al->record_type) &&
812 (GNUNET_NO == al->did_aaaa))
818 dns_id = (uint16_t) GNUNET_CRYPTO_random_u32 (GNUNET_CRYPTO_QUALITY_NONCE,
820 if (GNUNET_OK == pack (al->hostname,
821 GNUNET_DNSPARSER_TYPE_AAAA,
826 al->did_aaaa = GNUNET_YES;
828 GNUNET_DNSSTUB_resolve_cancel (al->resolve_handle);
829 al->resolve_handle = GNUNET_DNSSTUB_resolve (dnsstub_ctx,
832 &handle_resolve_result,
834 GNUNET_free (packet_buf);
835 GNUNET_DNSPARSER_free_packet (parsed);
840 /* resume by trying again from cache */
841 if (GNUNET_NO == try_cache (al->hostname,
843 al->client_request_id,
845 /* cache failed, tell client we could not get an answer */
847 send_end_msg (al->client_request_id, al->client);
849 free_active_lookup (al);
850 GNUNET_DNSPARSER_free_packet (parsed);
855 * We encountered a timeout trying to perform a
858 * @param cls a `struct ActiveLookup`
861 handle_resolve_timeout (void *cls)
863 struct ActiveLookup *al = cls;
865 al->timeout_task = NULL;
866 GNUNET_log (GNUNET_ERROR_TYPE_DEBUG, "DNS lookup timeout!\n");
867 send_end_msg (al->client_request_id, al->client);
868 free_active_lookup (al);
873 * Initiate an active lookup, then cache the result and
874 * try to then complete the resolution.
876 * @param hostname DNS name to resolve
877 * @param record_type record type to locate
878 * @param client_request_id client request ID
879 * @param client handle to the client
880 * @return #GNUNET_OK if the DNS query is now pending
883 resolve_and_cache (const char *hostname,
884 uint16_t record_type,
885 uint32_t client_request_id,
886 struct GNUNET_SERVICE_Client *client)
890 struct ActiveLookup *al;
894 GNUNET_log (GNUNET_ERROR_TYPE_DEBUG, "resolve_and_cache `%s'\n", hostname);
895 dns_id = (uint16_t) GNUNET_CRYPTO_random_u32 (GNUNET_CRYPTO_QUALITY_NONCE,
898 if (GNUNET_DNSPARSER_TYPE_ALL == record_type)
899 type = GNUNET_DNSPARSER_TYPE_A;
902 if (GNUNET_OK != pack (hostname, type, dns_id, &packet_buf, &packet_size))
904 GNUNET_log (GNUNET_ERROR_TYPE_ERROR,
905 "Failed to pack query for hostname `%s'\n",
907 return GNUNET_SYSERR;
910 al = GNUNET_new (struct ActiveLookup);
911 al->hostname = GNUNET_strdup (hostname);
912 al->record_type = record_type;
913 al->client_request_id = client_request_id;
917 GNUNET_SCHEDULER_add_delayed (DNS_TIMEOUT, &handle_resolve_timeout, al);
918 al->resolve_handle = GNUNET_DNSSTUB_resolve (dnsstub_ctx,
921 &handle_resolve_result,
923 GNUNET_free (packet_buf);
924 GNUNET_CONTAINER_DLL_insert (lookup_head, lookup_tail, al);
925 GNUNET_log (GNUNET_ERROR_TYPE_DEBUG,
926 "Resolving %s, client_request_id = %u, dns_id = %u\n",
928 (unsigned int) client_request_id,
929 (unsigned int) dns_id);
935 * Process DNS request for @a hostname with request ID @a client_request_id
936 * from @a client demanding records of type @a record_type.
938 * @param hostname DNS name to resolve
939 * @param record_type desired record type
940 * @param client_request_id client's request ID
941 * @param client who should get the result?
944 process_get (const char *hostname,
945 uint16_t record_type,
946 uint32_t client_request_id,
947 struct GNUNET_SERVICE_Client *client)
951 if (GNUNET_NO != try_cache (hostname, record_type, client_request_id, client))
953 if ((NULL != my_domain) && (NULL == strchr (hostname, (unsigned char) '.')) &&
954 (strlen (hostname) + strlen (my_domain) <= 253))
956 GNUNET_snprintf (fqdn, sizeof(fqdn), "%s.%s", hostname, my_domain);
958 else if (strlen (hostname) < 255)
960 GNUNET_snprintf (fqdn, sizeof(fqdn), "%s", hostname);
965 GNUNET_SERVICE_client_drop (client);
968 if (GNUNET_NO == try_cache (fqdn, record_type, client_request_id, client))
971 resolve_and_cache (fqdn, record_type, client_request_id, client))
973 send_end_msg (client_request_id, client);
980 * Verify well-formedness of GET-message.
982 * @param cls closure, unused
983 * @param get the actual message
984 * @return #GNUNET_OK if @a get is well-formed
987 check_get (void *cls, const struct GNUNET_RESOLVER_GetMessage *get)
994 size = ntohs (get->header.size) - sizeof(*get);
995 direction = ntohl (get->direction);
996 if (GNUNET_NO == direction)
998 GNUNET_MQ_check_zero_termination (get);
1001 af = ntohl (get->af);
1005 if (size != sizeof(struct in_addr))
1008 return GNUNET_SYSERR;
1013 if (size != sizeof(struct in6_addr))
1016 return GNUNET_SYSERR;
1022 return GNUNET_SYSERR;
1029 * Handle GET-message.
1031 * @param cls identification of the client
1032 * @param msg the actual message
1035 handle_get (void *cls, const struct GNUNET_RESOLVER_GetMessage *msg)
1037 struct GNUNET_SERVICE_Client *client = cls;
1040 uint32_t client_request_id;
1043 direction = ntohl (msg->direction);
1044 af = ntohl (msg->af);
1045 client_request_id = msg->client_id;
1046 GNUNET_SERVICE_client_continue (client);
1047 if (GNUNET_NO == direction)
1049 /* IP from hostname */
1050 hostname = GNUNET_strdup ((const char *) &msg[1]);
1051 GNUNET_log (GNUNET_ERROR_TYPE_DEBUG,
1052 "Client asks to resolve `%s'\n",
1057 process_get (hostname,
1058 GNUNET_DNSPARSER_TYPE_ALL,
1065 process_get (hostname,
1066 GNUNET_DNSPARSER_TYPE_A,
1073 process_get (hostname,
1074 GNUNET_DNSPARSER_TYPE_AAAA,
1081 GNUNET_log (GNUNET_ERROR_TYPE_DEBUG, "got invalid af: %d\n", af);
1088 /* hostname from IP */
1089 hostname = make_reverse_hostname (&msg[1], af);
1090 process_get (hostname,
1091 GNUNET_DNSPARSER_TYPE_PTR,
1095 GNUNET_free_non_null (hostname);
1100 * Service is shutting down, clean up.
1102 * @param cls NULL, unused
1105 shutdown_task (void *cls)
1109 while (NULL != lookup_head)
1110 free_active_lookup (lookup_head);
1111 while (NULL != cache_head)
1112 free_cache_entry (cache_head);
1113 while (NULL != hosts_head)
1114 free_hosts_entry (hosts_head);
1115 GNUNET_DNSSTUB_stop (dnsstub_ctx);
1116 GNUNET_free_non_null (my_domain);
1121 * Add information about a host from /etc/hosts
1124 * @param hostname the name of the host
1125 * @param rec_type DNS record type to use
1126 * @param data payload
1127 * @param data_size number of bytes in @a data
1130 add_host (const char *hostname,
1135 struct ResolveCache *rc;
1136 struct RecordListEntry *rle;
1137 struct GNUNET_DNSPARSER_Record *rec;
1139 rec = GNUNET_malloc (sizeof(struct GNUNET_DNSPARSER_Record));
1140 rec->expiration_time = GNUNET_TIME_UNIT_FOREVER_ABS;
1141 rec->type = rec_type;
1142 rec->dns_traffic_class = GNUNET_TUN_DNS_CLASS_INTERNET;
1143 rec->name = GNUNET_strdup (hostname);
1144 rec->data.raw.data = GNUNET_memdup (data, data_size);
1145 rec->data.raw.data_len = data_size;
1146 rle = GNUNET_new (struct RecordListEntry);
1148 rc = GNUNET_new (struct ResolveCache);
1149 rc->hostname = GNUNET_strdup (hostname);
1150 GNUNET_CONTAINER_DLL_insert (rc->records_head, rc->records_tail, rle);
1151 GNUNET_CONTAINER_DLL_insert (hosts_head, hosts_tail, rc);
1156 * Extract host information from a line in /etc/hosts
1158 * @param line the line to parse
1159 * @param line_len number of bytes in @a line
1162 extract_hosts (const char *line, size_t line_len)
1170 /* ignore everything after '#' */
1171 c = memrchr (line, (unsigned char) '#', line_len);
1173 line_len = c - line;
1174 /* ignore leading whitespace */
1175 while ((0 < line_len) && isspace ((unsigned char) *line))
1180 tbuf = GNUNET_strndup (line, line_len);
1181 tok = strtok (tbuf, " \t");
1187 if (1 == inet_pton (AF_INET, tok, &v4))
1189 while (NULL != (tok = strtok (NULL, " \t")))
1190 add_host (tok, GNUNET_DNSPARSER_TYPE_A, &v4, sizeof(struct in_addr));
1192 else if (1 == inet_pton (AF_INET6, tok, &v6))
1194 while (NULL != (tok = strtok (NULL, " \t")))
1195 add_host (tok, GNUNET_DNSPARSER_TYPE_AAAA, &v6, sizeof(struct in6_addr));
1202 * Reads the list of hosts from /etc/hosts.
1205 load_etc_hosts (void)
1207 struct GNUNET_DISK_FileHandle *fh;
1208 struct GNUNET_DISK_MapHandle *mh;
1213 fh = GNUNET_DISK_file_open ("/etc/hosts",
1214 GNUNET_DISK_OPEN_READ,
1215 GNUNET_DISK_PERM_NONE);
1218 GNUNET_log (GNUNET_ERROR_TYPE_INFO, "Failed to open /etc/hosts");
1221 if (GNUNET_OK != GNUNET_DISK_file_handle_size (fh, &bytes_read))
1223 GNUNET_log (GNUNET_ERROR_TYPE_ERROR,
1224 "Could not determin size of /etc/hosts. "
1225 "DNS resolution will not be possible.\n");
1226 GNUNET_DISK_file_close (fh);
1229 if (((unsigned long long) bytes_read) > (unsigned long long) SIZE_MAX)
1231 GNUNET_log (GNUNET_ERROR_TYPE_ERROR,
1232 "/etc/hosts file too large to mmap. "
1233 "DNS resolution will not be possible.\n");
1234 GNUNET_DISK_file_close (fh);
1237 buf = GNUNET_DISK_file_map (fh,
1239 GNUNET_DISK_MAP_TYPE_READ,
1240 (size_t) bytes_read);
1242 while (read_offset < (size_t) bytes_read)
1244 const char *newline;
1247 newline = strchr (buf + read_offset, '\n');
1248 if (NULL == newline)
1250 line_len = newline - buf - read_offset;
1251 extract_hosts (buf + read_offset, line_len);
1252 read_offset += line_len + 1;
1254 GNUNET_DISK_file_unmap (mh);
1255 GNUNET_DISK_file_close (fh);
1260 * Service is starting, initialize everything.
1262 * @param cls NULL, unused
1263 * @param cfg our configuration
1264 * @param sh service handle
1268 const struct GNUNET_CONFIGURATION_Handle *cfg,
1269 struct GNUNET_SERVICE_Handle *sh)
1272 int num_dns_servers;
1277 GNUNET_SCHEDULER_add_shutdown (&shutdown_task, cls);
1278 dnsstub_ctx = GNUNET_DNSSTUB_start (128);
1280 num_dns_servers = lookup_dns_servers (&dns_servers);
1281 if (0 >= num_dns_servers)
1284 GNUNET_ERROR_TYPE_ERROR,
1285 _ ("No DNS server available. DNS resolution will not be possible.\n"));
1288 for (int i = 0; i < num_dns_servers; i++)
1290 int result = GNUNET_DNSSTUB_add_dns_ip (dnsstub_ctx, dns_servers[i]);
1291 GNUNET_log (GNUNET_ERROR_TYPE_DEBUG,
1292 "Adding DNS server '%s': %s\n",
1294 GNUNET_OK == result ? "success" : "failure");
1295 GNUNET_free (dns_servers[i]);
1297 GNUNET_free_non_null (dns_servers);
1302 * Callback called when a client connects to the service.
1304 * @param cls closure for the service, unused
1305 * @param c the new client that connected to the service
1306 * @param mq the message queue used to send messages to the client
1310 connect_cb (void *cls,
1311 struct GNUNET_SERVICE_Client *c,
1312 struct GNUNET_MQ_Handle *mq)
1322 * Callback called when a client disconnected from the service
1324 * @param cls closure for the service
1325 * @param c the client that disconnected
1326 * @param internal_cls should be equal to @a c
1329 disconnect_cb (void *cls, struct GNUNET_SERVICE_Client *c, void *internal_cls)
1331 struct ActiveLookup *n;
1335 GNUNET_assert (c == internal_cls);
1337 for (struct ActiveLookup *al = n; NULL != al; al = n)
1340 if (al->client == c)
1341 free_active_lookup (al);
1347 * Define "main" method using service macro.
1349 GNUNET_SERVICE_MAIN (
1351 GNUNET_SERVICE_OPTION_NONE,
1356 GNUNET_MQ_hd_var_size (get,
1357 GNUNET_MESSAGE_TYPE_RESOLVER_REQUEST,
1358 struct GNUNET_RESOLVER_GetMessage,
1360 GNUNET_MQ_handler_end ());
1363 #if defined(LINUX) && defined(__GLIBC__)
1367 * MINIMIZE heap size (way below 128k) since this process doesn't need much.
1369 void __attribute__ ((constructor)) GNUNET_RESOLVER_memory_init ()
1371 mallopt (M_TRIM_THRESHOLD, 4 * 1024);
1372 mallopt (M_TOP_PAD, 1 * 1024);
1378 /* end of gnunet-service-resolver.c */