blob: f099295410a65d5f59f5962fbab62d8e7249a126 [file] [log] [blame]
Alexander Afanasyev2a655f72015-01-26 18:38:33 -08001/* -*- Mode:C++; c-file-style:"gnu"; indent-tabs-mode:nil; -*- */
Junxiao Shif748a4e2017-07-05 23:41:48 +00002/*
Davide Pesavento457f1352018-07-14 17:17:38 -04003 * Copyright (c) 2014-2018, Regents of the University of California,
Alexander Afanasyev2a655f72015-01-26 18:38:33 -08004 * 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
Junxiao Shif748a4e2017-07-05 23:41:48 +000026#include "dns-srv.hpp"
Alexander Afanasyev2a655f72015-01-26 18:38:33 -080027
28#include <sys/types.h>
Davide Pesavento457f1352018-07-14 17:17:38 -040029#include <arpa/nameser.h>
Alexander Afanasyev2a655f72015-01-26 18:38:33 -080030#include <netinet/in.h>
31#include <resolv.h>
Alexander Afanasyev2a655f72015-01-26 18:38:33 -080032
33#ifdef __APPLE__
34#include <arpa/nameser_compat.h>
35#endif
36
Davide Pesaventoa997d292017-08-24 20:16:59 -040037#include <iostream>
38
Davide Pesavento457f1352018-07-14 17:17:38 -040039#include <boost/endian/conversion.hpp>
40
Alexander Afanasyev2a655f72015-01-26 18:38:33 -080041namespace ndn {
42namespace tools {
43namespace autoconfig {
44
Davide Pesavento457f1352018-07-14 17:17:38 -040045using namespace std::string_literals;
46
Junxiao Shif748a4e2017-07-05 23:41:48 +000047union QueryAnswer
Alexander Afanasyev2a655f72015-01-26 18:38:33 -080048{
49 HEADER header;
50 uint8_t buf[NS_PACKETSZ];
51};
52
Junxiao Shif748a4e2017-07-05 23:41:48 +000053/** \brief Parse SRV record
54 * \return FaceUri of the hub from the SRV record
55 * \throw DnsSrvError if SRV record cannot be parsed
Alexander Afanasyev2a655f72015-01-26 18:38:33 -080056 */
Junxiao Shif748a4e2017-07-05 23:41:48 +000057static std::string
58parseSrvRr(const QueryAnswer& queryAnswer, int answerSize)
Alexander Afanasyev2a655f72015-01-26 18:38:33 -080059{
60 // The references of the next classes are:
61 // http://www.diablotin.com/librairie/networking/dnsbind/ch14_02.htm
62
63 struct rechdr
64 {
65 uint16_t type;
66 uint16_t iclass;
67 uint32_t ttl;
68 uint16_t length;
69 };
70
71 struct srv_t
72 {
73 uint16_t priority;
74 uint16_t weight;
75 uint16_t port;
76 uint8_t* target;
77 };
78
Davide Pesavento457f1352018-07-14 17:17:38 -040079 uint16_t ancount = queryAnswer.header.ancount;
80 if (boost::endian::big_to_native(ancount) == 0) {
Junxiao Shif748a4e2017-07-05 23:41:48 +000081 BOOST_THROW_EXCEPTION(DnsSrvError("SRV record cannot be parsed"));
Alexander Afanasyev2a655f72015-01-26 18:38:33 -080082 }
83
84 const uint8_t* blob = queryAnswer.buf + NS_HFIXEDSZ;
Alexander Afanasyev2a655f72015-01-26 18:38:33 -080085 blob += dn_skipname(blob, queryAnswer.buf + answerSize) + NS_QFIXEDSZ;
86
87 char srvName[NS_MAXDNAME];
88 int serverNameSize = dn_expand(queryAnswer.buf, // message pointer
89 queryAnswer.buf + answerSize, // end of message
90 blob, // compressed server name
91 srvName, // expanded server name
92 NS_MAXDNAME);
93 if (serverNameSize <= 0) {
Junxiao Shif748a4e2017-07-05 23:41:48 +000094 BOOST_THROW_EXCEPTION(DnsSrvError("SRV record cannot be parsed (error decoding domain name)"));
Alexander Afanasyev2a655f72015-01-26 18:38:33 -080095 }
96
97 const srv_t* server = reinterpret_cast<const srv_t*>(&blob[sizeof(rechdr)]);
Davide Pesavento457f1352018-07-14 17:17:38 -040098 uint16_t port = boost::endian::big_to_native(server->port);
Alexander Afanasyev2a655f72015-01-26 18:38:33 -080099
100 blob += serverNameSize + NS_HFIXEDSZ + NS_QFIXEDSZ;
101
102 char hostName[NS_MAXDNAME];
103 int hostNameSize = dn_expand(queryAnswer.buf, // message pointer
104 queryAnswer.buf + answerSize, // end of message
105 blob, // compressed host name
106 hostName, // expanded host name
107 NS_MAXDNAME);
108 if (hostNameSize <= 0) {
Junxiao Shif748a4e2017-07-05 23:41:48 +0000109 BOOST_THROW_EXCEPTION(DnsSrvError("SRV record cannot be parsed (error decoding host name)"));
Alexander Afanasyev2a655f72015-01-26 18:38:33 -0800110 }
111
Davide Pesavento457f1352018-07-14 17:17:38 -0400112 return "udp://"s + hostName + ":" + to_string(port);
Alexander Afanasyev2a655f72015-01-26 18:38:33 -0800113}
114
Junxiao Shif748a4e2017-07-05 23:41:48 +0000115std::string
116querySrvRr(const std::string& fqdn)
117{
118 std::string srvDomain = "_ndn._udp." + fqdn;
119 std::cerr << "Sending DNS query for SRV record for " << srvDomain << std::endl;
120
121 res_init();
122
123 _res.retrans = 1;
124 _res.retry = 2;
125 _res.ndots = 10;
126
127 QueryAnswer queryAnswer;
128 int answerSize = res_query(srvDomain.data(),
129 ns_c_in,
130 ns_t_srv,
131 queryAnswer.buf,
132 sizeof(queryAnswer));
133 if (answerSize == 0) {
134 BOOST_THROW_EXCEPTION(DnsSrvError("No DNS SRV records found for " + srvDomain));
135 }
136 return parseSrvRr(queryAnswer, answerSize);
137}
138
139/**
140 * @brief Send DNS SRV request using search domain list
141 */
142std::string
143querySrvRrSearch()
144{
145 std::cerr << "Sending DNS query for SRV record for _ndn._udp" << std::endl;
146
147 QueryAnswer queryAnswer;
148
149 res_init();
150
151 _res.retrans = 1;
152 _res.retry = 2;
153 _res.ndots = 10;
154
155 int answerSize = res_search("_ndn._udp",
156 ns_c_in,
157 ns_t_srv,
158 queryAnswer.buf,
159 sizeof(queryAnswer));
160
161 if (answerSize == 0) {
162 BOOST_THROW_EXCEPTION(DnsSrvError("No DNS SRV records found for _ndn._udp"));
163 }
164
165 return parseSrvRr(queryAnswer, answerSize);
166}
167
Alexander Afanasyev2a655f72015-01-26 18:38:33 -0800168} // namespace autoconfig
169} // namespace tools
170} // namespace ndn