blob: c4f27b36f52b7219d9d44b886e743753fb0cfae5 [file] [log] [blame]
Alexander Afanasyev2a655f72015-01-26 18:38:33 -08001/* -*- Mode:C++; c-file-style:"gnu"; indent-tabs-mode:nil; -*- */
2/**
3 * Copyright (c) 2014-2015, Regents of the University of California,
4 * Arizona Board of Regents,
5 * Colorado State University,
6 * University Pierre & Marie Curie, Sorbonne University,
7 * Washington University in St. Louis,
8 * Beijing Institute of Technology,
9 * The University of Memphis.
10 *
11 * This file is part of NFD (Named Data Networking Forwarding Daemon).
12 * See AUTHORS.md for complete list of NFD authors and contributors.
13 *
14 * NFD is free software: you can redistribute it and/or modify it under the terms
15 * of the GNU General Public License as published by the Free Software Foundation,
16 * either version 3 of the License, or (at your option) any later version.
17 *
18 * NFD is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY;
19 * without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR
20 * PURPOSE. See the GNU General Public License for more details.
21 *
22 * You should have received a copy of the GNU General Public License along with
23 * NFD, e.g., in COPYING.md file. If not, see <http://www.gnu.org/licenses/>.
24 */
25
26#include "base-dns.hpp"
27
28#include <sys/types.h>
29#include <netinet/in.h>
30#include <resolv.h>
31#include <arpa/nameser.h>
32
33#ifdef __APPLE__
34#include <arpa/nameser_compat.h>
35#endif
36
37namespace ndn {
38namespace tools {
39namespace autoconfig {
40
41union BaseDns::QueryAnswer
42{
43 HEADER header;
44 uint8_t buf[NS_PACKETSZ];
45};
46
47BaseDns::BaseDns(Face& face, KeyChain& keyChain, const NextStageCallback& nextStageOnFailure)
48 : Base(face, keyChain, nextStageOnFailure)
49{
50}
51
52std::string
53BaseDns::querySrvRr(const std::string& fqdn)
54{
55 std::string srvDomain = "_ndn._udp." + fqdn;
56 std::cerr << "Sending DNS query for SRV record for " << srvDomain << std::endl;
57
Alexander Afanasyeve46279dc2015-01-29 15:39:17 -080058 res_init();
59
60 _res.retrans = 1;
61 _res.retry = 2;
62 _res.ndots = 10;
63
Alexander Afanasyev2a655f72015-01-26 18:38:33 -080064 QueryAnswer queryAnswer;
65 int answerSize = res_query(srvDomain.c_str(),
66 ns_c_in,
67 ns_t_srv,
68 queryAnswer.buf,
69 sizeof(queryAnswer));
70 if (answerSize == 0) {
Spyridon Mastorakis149e02c2015-07-27 13:22:22 -070071 BOOST_THROW_EXCEPTION(Error("No DNS SRV records found for " + srvDomain));
Alexander Afanasyev2a655f72015-01-26 18:38:33 -080072 }
73 return parseSrvRr(queryAnswer, answerSize);
74}
75
76/**
77 * @brief Send DNS SRV request using search domain list
78 */
79std::string
80BaseDns::querySrvRrSearch()
81{
82 std::cerr << "Sending DNS query for SRV record for _ndn._udp" << std::endl;
83
84 QueryAnswer queryAnswer;
85
Alexander Afanasyeve46279dc2015-01-29 15:39:17 -080086 res_init();
87
88 _res.retrans = 1;
Alexander Afanasyev2a655f72015-01-26 18:38:33 -080089 _res.retry = 2;
90 _res.ndots = 10;
91
92 int answerSize = res_search("_ndn._udp",
93 ns_c_in,
94 ns_t_srv,
95 queryAnswer.buf,
96 sizeof(queryAnswer));
97
98 if (answerSize == 0) {
Spyridon Mastorakis149e02c2015-07-27 13:22:22 -070099 BOOST_THROW_EXCEPTION(Error("No DNS SRV records found for _ndn._udp"));
Alexander Afanasyev2a655f72015-01-26 18:38:33 -0800100 }
101
102 return parseSrvRr(queryAnswer, answerSize);
103}
104
105std::string
106BaseDns::parseSrvRr(const QueryAnswer& queryAnswer, int answerSize)
107{
108 // The references of the next classes are:
109 // http://www.diablotin.com/librairie/networking/dnsbind/ch14_02.htm
110
111 struct rechdr
112 {
113 uint16_t type;
114 uint16_t iclass;
115 uint32_t ttl;
116 uint16_t length;
117 };
118
119 struct srv_t
120 {
121 uint16_t priority;
122 uint16_t weight;
123 uint16_t port;
124 uint8_t* target;
125 };
126
127 if (ntohs(queryAnswer.header.ancount) == 0) {
Spyridon Mastorakis149e02c2015-07-27 13:22:22 -0700128 BOOST_THROW_EXCEPTION(Error("SRV record cannot be parsed"));
Alexander Afanasyev2a655f72015-01-26 18:38:33 -0800129 }
130
131 const uint8_t* blob = queryAnswer.buf + NS_HFIXEDSZ;
132
133 blob += dn_skipname(blob, queryAnswer.buf + answerSize) + NS_QFIXEDSZ;
134
135 char srvName[NS_MAXDNAME];
136 int serverNameSize = dn_expand(queryAnswer.buf, // message pointer
137 queryAnswer.buf + answerSize, // end of message
138 blob, // compressed server name
139 srvName, // expanded server name
140 NS_MAXDNAME);
141 if (serverNameSize <= 0) {
Spyridon Mastorakis149e02c2015-07-27 13:22:22 -0700142 BOOST_THROW_EXCEPTION(Error("SRV record cannot be parsed (error decoding domain name)"));
Alexander Afanasyev2a655f72015-01-26 18:38:33 -0800143 }
144
145 const srv_t* server = reinterpret_cast<const srv_t*>(&blob[sizeof(rechdr)]);
146 uint16_t convertedPort = be16toh(server->port);
147
148 blob += serverNameSize + NS_HFIXEDSZ + NS_QFIXEDSZ;
149
150 char hostName[NS_MAXDNAME];
151 int hostNameSize = dn_expand(queryAnswer.buf, // message pointer
152 queryAnswer.buf + answerSize, // end of message
153 blob, // compressed host name
154 hostName, // expanded host name
155 NS_MAXDNAME);
156 if (hostNameSize <= 0) {
Spyridon Mastorakis149e02c2015-07-27 13:22:22 -0700157 BOOST_THROW_EXCEPTION(Error("SRV record cannot be parsed (error decoding host name)"));
Alexander Afanasyev2a655f72015-01-26 18:38:33 -0800158 }
159
160 std::string uri = "udp://";
161 uri.append(hostName);
162 uri.append(":");
Davide Pesaventoec2a6982015-09-18 23:59:17 +0200163 uri.append(to_string(convertedPort));
Alexander Afanasyev2a655f72015-01-26 18:38:33 -0800164
165 return uri;
166}
167
168} // namespace autoconfig
169} // namespace tools
170} // namespace ndn