blob: 0c674b3985a6e46c6962aa58e575c6f906865bd7 [file] [log] [blame]
Alexander Afanasyev31367922015-02-09 20:51:10 -08001/* -*- Mode:C++; c-file-style:"gnu"; indent-tabs-mode:nil; -*- */
Davide Pesaventoa3148082018-04-12 18:21:54 -04002/*
Davide Pesavento412c9822021-07-02 00:21:05 -04003 * Copyright (c) 2014-2021, Regents of the University of California,
Alexander Afanasyev31367922015-02-09 20:51:10 -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 Shib2600172016-07-11 08:53:53 +000026#include "service.hpp"
Davide Pesavento9f8b10e2018-08-22 08:45:37 +000027
Davide Pesavento9f8b10e2018-08-22 08:45:37 +000028#include "fib-updater.hpp"
Davide Pesavento9f8b10e2018-08-22 08:45:37 +000029#include "readvertise/client-to-nlsr-readvertise-policy.hpp"
Yanbiao Lif48d0802018-06-01 03:00:02 -070030#include "readvertise/host-to-gateway-readvertise-policy.hpp"
Davide Pesavento9f8b10e2018-08-22 08:45:37 +000031#include "readvertise/nfd-rib-readvertise-destination.hpp"
32#include "readvertise/readvertise.hpp"
33
Davide Pesavento2cae8ca2019-04-18 20:48:05 -040034#include "common/global.hpp"
35#include "common/logger.hpp"
Alexander Afanasyev31367922015-02-09 20:51:10 -080036
37#include <boost/property_tree/info_parser.hpp>
Alexander Afanasyev31367922015-02-09 20:51:10 -080038#include <ndn-cxx/transport/tcp-transport.hpp>
Davide Pesavento9f8b10e2018-08-22 08:45:37 +000039#include <ndn-cxx/transport/unix-transport.hpp>
Alexander Afanasyev31367922015-02-09 20:51:10 -080040
41namespace nfd {
42namespace rib {
43
Junxiao Shif4cfed12018-08-22 23:26:29 +000044NFD_LOG_INIT(RibService);
Alexander Afanasyev31367922015-02-09 20:51:10 -080045
Teng Liang04d5ce62018-08-06 10:20:24 +080046Service* Service::s_instance = nullptr;
47
Davide Pesavento412c9822021-07-02 00:21:05 -040048const std::string CFG_RIB = "rib";
Wenkai Zheng6598fb02019-09-08 18:03:46 -070049const std::string CFG_LOCALHOST_SECURITY = "localhost_security";
50const std::string CFG_LOCALHOP_SECURITY = "localhop_security";
Teng Liang18c2b292019-10-18 14:31:04 -070051const std::string CFG_PA_VALIDATION = "prefix_announcement_validation";
Wenkai Zheng6598fb02019-09-08 18:03:46 -070052const std::string CFG_PREFIX_PROPAGATE = "auto_prefix_propagate";
53const std::string CFG_READVERTISE_NLSR = "readvertise_nlsr";
54const Name READVERTISE_NLSR_PREFIX = "/localhost/nlsr";
55const uint64_t PROPAGATE_DEFAULT_COST = 15;
56const time::milliseconds PROPAGATE_DEFAULT_TIMEOUT = 10_s;
Junxiao Shif4cfed12018-08-22 23:26:29 +000057
58static ConfigSection
59loadConfigSectionFromFile(const std::string& filename)
Alexander Afanasyev31367922015-02-09 20:51:10 -080060{
Junxiao Shif4cfed12018-08-22 23:26:29 +000061 ConfigSection config;
62 // Any format errors should have been caught already
63 boost::property_tree::read_info(filename, config);
64 return config;
65}
66
67/**
68 * \brief Look into the config file and construct appropriate transport to communicate with NFD
69 * If NFD-RIB instance was initialized with config file, INFO format is assumed
70 */
71static shared_ptr<ndn::Transport>
72makeLocalNfdTransport(const ConfigSection& config)
73{
74 if (config.get_child_optional("face_system.unix")) {
75 // default socket path should be the same as in UnixStreamFactory::processConfig
Eric Newberry22974902020-04-06 23:41:00 -070076#ifdef __linux__
77 auto path = config.get<std::string>("face_system.unix.path", "/run/nfd.sock");
78#else
Junxiao Shif4cfed12018-08-22 23:26:29 +000079 auto path = config.get<std::string>("face_system.unix.path", "/var/run/nfd.sock");
Eric Newberry22974902020-04-06 23:41:00 -070080#endif // __linux__
Junxiao Shif4cfed12018-08-22 23:26:29 +000081 return make_shared<ndn::UnixTransport>(path);
Teng Liang04d5ce62018-08-06 10:20:24 +080082 }
Junxiao Shif4cfed12018-08-22 23:26:29 +000083 else if (config.get_child_optional("face_system.tcp") &&
84 config.get<std::string>("face_system.tcp.listen", "yes") == "yes") {
85 // default port should be the same as in TcpFactory::processConfig
86 auto port = config.get<std::string>("face_system.tcp.port", "6363");
87 return make_shared<ndn::TcpTransport>("localhost", port);
Teng Liang04d5ce62018-08-06 10:20:24 +080088 }
Junxiao Shif4cfed12018-08-22 23:26:29 +000089 else {
Davide Pesavento19779d82019-02-14 13:40:04 -050090 NDN_THROW(ConfigFile::Error("No transport is available to communicate with NFD"));
Junxiao Shif4cfed12018-08-22 23:26:29 +000091 }
92}
93
94Service::Service(const std::string& configFile, ndn::KeyChain& keyChain)
Ashlesh Gawandef84bac52018-08-27 07:39:31 +000095 : Service(keyChain, makeLocalNfdTransport(loadConfigSectionFromFile(configFile)),
96 [&configFile] (ConfigFile& config, bool isDryRun) {
97 config.parse(configFile, isDryRun);
98 })
Junxiao Shif4cfed12018-08-22 23:26:29 +000099{
Alexander Afanasyev31367922015-02-09 20:51:10 -0800100}
101
Davide Pesavento9f8b10e2018-08-22 08:45:37 +0000102Service::Service(const ConfigSection& configSection, ndn::KeyChain& keyChain)
Ashlesh Gawandef84bac52018-08-27 07:39:31 +0000103 : Service(keyChain, makeLocalNfdTransport(configSection),
104 [&configSection] (ConfigFile& config, bool isDryRun) {
105 config.parse(configSection, isDryRun, "internal://nfd.conf");
106 })
Junxiao Shif4cfed12018-08-22 23:26:29 +0000107{
Junxiao Shif4cfed12018-08-22 23:26:29 +0000108}
109
Ashlesh Gawandef84bac52018-08-27 07:39:31 +0000110template<typename ConfigParseFunc>
111Service::Service(ndn::KeyChain& keyChain, shared_ptr<ndn::Transport> localNfdTransport,
112 const ConfigParseFunc& configParse)
Junxiao Shif4cfed12018-08-22 23:26:29 +0000113 : m_keyChain(keyChain)
114 , m_face(std::move(localNfdTransport), getGlobalIoService(), m_keyChain)
115 , m_nfdController(m_face, m_keyChain)
116 , m_fibUpdater(m_rib, m_nfdController)
117 , m_dispatcher(m_face, m_keyChain)
Davide Pesavento0a71dd32019-03-17 20:36:18 -0400118 , m_ribManager(m_rib, m_face, m_keyChain, m_nfdController, m_dispatcher)
Alexander Afanasyev31367922015-02-09 20:51:10 -0800119{
Teng Liang04d5ce62018-08-06 10:20:24 +0800120 if (s_instance != nullptr) {
Davide Pesavento19779d82019-02-14 13:40:04 -0500121 NDN_THROW(std::logic_error("RIB service cannot be instantiated more than once"));
Teng Liang04d5ce62018-08-06 10:20:24 +0800122 }
123 if (&getGlobalIoService() != &getRibIoService()) {
Davide Pesavento19779d82019-02-14 13:40:04 -0500124 NDN_THROW(std::logic_error("RIB service must run on RIB thread"));
Teng Liang04d5ce62018-08-06 10:20:24 +0800125 }
126 s_instance = this;
Ashlesh Gawandef84bac52018-08-27 07:39:31 +0000127
128 ConfigFile config(ConfigFile::ignoreUnknownSection);
Davide Pesavento412c9822021-07-02 00:21:05 -0400129 config.addSectionHandler(CFG_RIB, [this] (auto&&... args) {
130 processConfig(std::forward<decltype(args)>(args)...);
131 });
Ashlesh Gawandef84bac52018-08-27 07:39:31 +0000132 configParse(config, true);
133 configParse(config, false);
134
135 m_ribManager.registerWithNfd();
136 m_ribManager.enableLocalFields();
Alexander Afanasyev31367922015-02-09 20:51:10 -0800137}
138
Teng Liang04d5ce62018-08-06 10:20:24 +0800139Service::~Service()
140{
141 s_instance = nullptr;
142}
143
144Service&
145Service::get()
146{
147 if (s_instance == nullptr) {
Davide Pesavento19779d82019-02-14 13:40:04 -0500148 NDN_THROW(std::logic_error("RIB service is not instantiated"));
Teng Liang04d5ce62018-08-06 10:20:24 +0800149 }
150 if (&getGlobalIoService() != &getRibIoService()) {
Davide Pesavento19779d82019-02-14 13:40:04 -0500151 NDN_THROW(std::logic_error("Must get RIB service on RIB thread"));
Teng Liang04d5ce62018-08-06 10:20:24 +0800152 }
153 return *s_instance;
154}
Alexander Afanasyevc3ea5a72015-02-12 20:14:16 -0800155
Alexander Afanasyev31367922015-02-09 20:51:10 -0800156void
Junxiao Shif4cfed12018-08-22 23:26:29 +0000157Service::processConfig(const ConfigSection& section, bool isDryRun, const std::string& filename)
Alexander Afanasyev31367922015-02-09 20:51:10 -0800158{
Junxiao Shif4cfed12018-08-22 23:26:29 +0000159 if (isDryRun) {
160 checkConfig(section, filename);
161 }
162 else {
163 applyConfig(section, filename);
164 }
165}
Alexander Afanasyev31367922015-02-09 20:51:10 -0800166
Junxiao Shif4cfed12018-08-22 23:26:29 +0000167void
168Service::checkConfig(const ConfigSection& section, const std::string& filename)
169{
Wenkai Zheng6598fb02019-09-08 18:03:46 -0700170 bool hasLocalhop = false;
171 bool hasPropagate = false;
172
Junxiao Shif4cfed12018-08-22 23:26:29 +0000173 for (const auto& item : section) {
174 const std::string& key = item.first;
Ashlesh Gawandef84bac52018-08-27 07:39:31 +0000175 const ConfigSection& value = item.second;
Jongseok Lee854866f2020-03-10 19:16:25 -0400176 if (key == CFG_LOCALHOST_SECURITY || key == CFG_PA_VALIDATION) {
177 ndn::ValidatorConfig testValidator(m_face);
178 testValidator.load(value, filename);
179 }
180 else if (key == CFG_LOCALHOP_SECURITY) {
181 hasLocalhop = true;
Alexander Afanasyev3c0c0352018-10-17 11:59:23 -0400182 ndn::ValidatorConfig testValidator(m_face);
183 testValidator.load(value, filename);
Junxiao Shif4cfed12018-08-22 23:26:29 +0000184 }
185 else if (key == CFG_PREFIX_PROPAGATE) {
Wenkai Zheng6598fb02019-09-08 18:03:46 -0700186 hasPropagate = true;
Junxiao Shif4cfed12018-08-22 23:26:29 +0000187 // AutoPrefixPropagator does not support config dry-run
188 }
189 else if (key == CFG_READVERTISE_NLSR) {
Davide Pesavento412c9822021-07-02 00:21:05 -0400190 ConfigFile::parseYesNo(item, CFG_RIB + "." + CFG_READVERTISE_NLSR);
Junxiao Shif4cfed12018-08-22 23:26:29 +0000191 }
192 else {
Davide Pesavento412c9822021-07-02 00:21:05 -0400193 NDN_THROW(ConfigFile::Error("Unrecognized option " + CFG_RIB + "." + key));
Junxiao Shif4cfed12018-08-22 23:26:29 +0000194 }
195 }
Wenkai Zheng6598fb02019-09-08 18:03:46 -0700196
197 if (hasLocalhop && hasPropagate) {
198 NDN_THROW(ConfigFile::Error(CFG_LOCALHOP_SECURITY + " and " + CFG_PREFIX_PROPAGATE +
199 " cannot be enabled at the same time"));
200 }
Junxiao Shif4cfed12018-08-22 23:26:29 +0000201}
202
203void
204Service::applyConfig(const ConfigSection& section, const std::string& filename)
205{
206 bool wantPrefixPropagate = false;
207 bool wantReadvertiseNlsr = false;
208
209 for (const auto& item : section) {
210 const std::string& key = item.first;
211 const ConfigSection& value = item.second;
212 if (key == CFG_LOCALHOST_SECURITY) {
213 m_ribManager.applyLocalhostConfig(value, filename);
214 }
215 else if (key == CFG_LOCALHOP_SECURITY) {
216 m_ribManager.enableLocalhop(value, filename);
217 }
Teng Liang18c2b292019-10-18 14:31:04 -0700218 else if (key == CFG_PA_VALIDATION) {
219 m_ribManager.applyPaConfig(value, filename);
220 }
Junxiao Shif4cfed12018-08-22 23:26:29 +0000221 else if (key == CFG_PREFIX_PROPAGATE) {
Junxiao Shif4cfed12018-08-22 23:26:29 +0000222 wantPrefixPropagate = true;
Yanbiao Lif48d0802018-06-01 03:00:02 -0700223
224 if (!m_readvertisePropagation) {
225 NFD_LOG_DEBUG("Enabling automatic prefix propagation");
226
Yanbiao Lif48d0802018-06-01 03:00:02 -0700227 auto cost = item.second.get_optional<uint64_t>("cost");
Wenkai Zheng6598fb02019-09-08 18:03:46 -0700228 auto parameters = ndn::nfd::ControlParameters()
229 .setCost(cost.value_or(PROPAGATE_DEFAULT_COST))
230 .setOrigin(ndn::nfd::ROUTE_ORIGIN_CLIENT);
Yanbiao Lif48d0802018-06-01 03:00:02 -0700231
Yanbiao Lif48d0802018-06-01 03:00:02 -0700232 auto timeout = item.second.get_optional<uint64_t>("timeout");
Wenkai Zheng6598fb02019-09-08 18:03:46 -0700233 auto options = ndn::nfd::CommandOptions()
234 .setPrefix(RibManager::LOCALHOP_TOP_PREFIX)
235 .setTimeout(timeout ? time::milliseconds(*timeout) : PROPAGATE_DEFAULT_TIMEOUT);
Yanbiao Lif48d0802018-06-01 03:00:02 -0700236
237 m_readvertisePropagation = make_unique<Readvertise>(
238 m_rib,
Yanbiao Lif48d0802018-06-01 03:00:02 -0700239 make_unique<HostToGatewayReadvertisePolicy>(m_keyChain, item.second),
240 make_unique<NfdRibReadvertiseDestination>(m_nfdController, m_rib, options, parameters));
241 }
Junxiao Shif4cfed12018-08-22 23:26:29 +0000242 }
243 else if (key == CFG_READVERTISE_NLSR) {
Davide Pesavento412c9822021-07-02 00:21:05 -0400244 wantReadvertiseNlsr = ConfigFile::parseYesNo(item, CFG_RIB + "." + CFG_READVERTISE_NLSR);
Junxiao Shif4cfed12018-08-22 23:26:29 +0000245 }
246 else {
Davide Pesavento412c9822021-07-02 00:21:05 -0400247 NDN_THROW(ConfigFile::Error("Unrecognized option " + CFG_RIB + "." + key));
Junxiao Shif4cfed12018-08-22 23:26:29 +0000248 }
Alexander Afanasyev31367922015-02-09 20:51:10 -0800249 }
250
Yanbiao Lif48d0802018-06-01 03:00:02 -0700251 if (!wantPrefixPropagate && m_readvertisePropagation != nullptr) {
252 NFD_LOG_DEBUG("Disabling automatic prefix propagation");
253 m_readvertisePropagation.reset();
Davide Pesavento9f8b10e2018-08-22 08:45:37 +0000254 }
255
Junxiao Shif4cfed12018-08-22 23:26:29 +0000256 if (wantReadvertiseNlsr && m_readvertiseNlsr == nullptr) {
257 NFD_LOG_DEBUG("Enabling readvertise-to-nlsr");
Yanbiao Lif48d0802018-06-01 03:00:02 -0700258 auto options = ndn::nfd::CommandOptions().setPrefix(READVERTISE_NLSR_PREFIX);
Davide Pesavento9f8b10e2018-08-22 08:45:37 +0000259 m_readvertiseNlsr = make_unique<Readvertise>(
260 m_rib,
261 make_unique<ClientToNlsrReadvertisePolicy>(),
Yanbiao Lif48d0802018-06-01 03:00:02 -0700262 make_unique<NfdRibReadvertiseDestination>(m_nfdController, m_rib, options));
Davide Pesavento9f8b10e2018-08-22 08:45:37 +0000263 }
Junxiao Shif4cfed12018-08-22 23:26:29 +0000264 else if (!wantReadvertiseNlsr && m_readvertiseNlsr != nullptr) {
265 NFD_LOG_DEBUG("Disabling readvertise-to-nlsr");
Davide Pesavento9f8b10e2018-08-22 08:45:37 +0000266 m_readvertiseNlsr.reset();
267 }
Alexander Afanasyev31367922015-02-09 20:51:10 -0800268}
269
Alexander Afanasyev31367922015-02-09 20:51:10 -0800270} // namespace rib
271} // namespace nfd