Alexander Afanasyev | 3136792 | 2015-02-09 20:51:10 -0800 | [diff] [blame] | 1 | /* -*- Mode:C++; c-file-style:"gnu"; indent-tabs-mode:nil; -*- */ |
Davide Pesavento | a314808 | 2018-04-12 18:21:54 -0400 | [diff] [blame] | 2 | /* |
Davide Pesavento | 19779d8 | 2019-02-14 13:40:04 -0500 | [diff] [blame] | 3 | * Copyright (c) 2014-2019, Regents of the University of California, |
Alexander Afanasyev | 3136792 | 2015-02-09 20:51:10 -0800 | [diff] [blame] | 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 | |
Junxiao Shi | b260017 | 2016-07-11 08:53:53 +0000 | [diff] [blame] | 26 | #include "service.hpp" |
Davide Pesavento | 9f8b10e | 2018-08-22 08:45:37 +0000 | [diff] [blame] | 27 | |
Davide Pesavento | 9f8b10e | 2018-08-22 08:45:37 +0000 | [diff] [blame] | 28 | #include "fib-updater.hpp" |
Davide Pesavento | 9f8b10e | 2018-08-22 08:45:37 +0000 | [diff] [blame] | 29 | #include "readvertise/client-to-nlsr-readvertise-policy.hpp" |
Yanbiao Li | f48d080 | 2018-06-01 03:00:02 -0700 | [diff] [blame] | 30 | #include "readvertise/host-to-gateway-readvertise-policy.hpp" |
Davide Pesavento | 9f8b10e | 2018-08-22 08:45:37 +0000 | [diff] [blame] | 31 | #include "readvertise/nfd-rib-readvertise-destination.hpp" |
| 32 | #include "readvertise/readvertise.hpp" |
| 33 | |
Davide Pesavento | 2cae8ca | 2019-04-18 20:48:05 -0400 | [diff] [blame] | 34 | #include "common/global.hpp" |
| 35 | #include "common/logger.hpp" |
Alexander Afanasyev | 3136792 | 2015-02-09 20:51:10 -0800 | [diff] [blame] | 36 | |
| 37 | #include <boost/property_tree/info_parser.hpp> |
Alexander Afanasyev | 3136792 | 2015-02-09 20:51:10 -0800 | [diff] [blame] | 38 | #include <ndn-cxx/transport/tcp-transport.hpp> |
Davide Pesavento | 9f8b10e | 2018-08-22 08:45:37 +0000 | [diff] [blame] | 39 | #include <ndn-cxx/transport/unix-transport.hpp> |
Alexander Afanasyev | 3136792 | 2015-02-09 20:51:10 -0800 | [diff] [blame] | 40 | |
| 41 | namespace nfd { |
| 42 | namespace rib { |
| 43 | |
Junxiao Shi | f4cfed1 | 2018-08-22 23:26:29 +0000 | [diff] [blame] | 44 | NFD_LOG_INIT(RibService); |
Alexander Afanasyev | 3136792 | 2015-02-09 20:51:10 -0800 | [diff] [blame] | 45 | |
Teng Liang | 04d5ce6 | 2018-08-06 10:20:24 +0800 | [diff] [blame] | 46 | Service* Service::s_instance = nullptr; |
| 47 | |
Wenkai Zheng | 6598fb0 | 2019-09-08 18:03:46 -0700 | [diff] [blame] | 48 | const std::string CFG_SECTION = "rib"; |
| 49 | const std::string CFG_LOCALHOST_SECURITY = "localhost_security"; |
| 50 | const std::string CFG_LOCALHOP_SECURITY = "localhop_security"; |
Teng Liang | 18c2b29 | 2019-10-18 14:31:04 -0700 | [diff] [blame] | 51 | const std::string CFG_PA_VALIDATION = "prefix_announcement_validation"; |
Wenkai Zheng | 6598fb0 | 2019-09-08 18:03:46 -0700 | [diff] [blame] | 52 | const std::string CFG_PREFIX_PROPAGATE = "auto_prefix_propagate"; |
| 53 | const std::string CFG_READVERTISE_NLSR = "readvertise_nlsr"; |
| 54 | const Name READVERTISE_NLSR_PREFIX = "/localhost/nlsr"; |
| 55 | const uint64_t PROPAGATE_DEFAULT_COST = 15; |
| 56 | const time::milliseconds PROPAGATE_DEFAULT_TIMEOUT = 10_s; |
Junxiao Shi | f4cfed1 | 2018-08-22 23:26:29 +0000 | [diff] [blame] | 57 | |
| 58 | static ConfigSection |
| 59 | loadConfigSectionFromFile(const std::string& filename) |
Alexander Afanasyev | 3136792 | 2015-02-09 20:51:10 -0800 | [diff] [blame] | 60 | { |
Junxiao Shi | f4cfed1 | 2018-08-22 23:26:29 +0000 | [diff] [blame] | 61 | 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 | */ |
| 71 | static shared_ptr<ndn::Transport> |
| 72 | makeLocalNfdTransport(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 |
| 76 | auto path = config.get<std::string>("face_system.unix.path", "/var/run/nfd.sock"); |
| 77 | return make_shared<ndn::UnixTransport>(path); |
Teng Liang | 04d5ce6 | 2018-08-06 10:20:24 +0800 | [diff] [blame] | 78 | } |
Junxiao Shi | f4cfed1 | 2018-08-22 23:26:29 +0000 | [diff] [blame] | 79 | else if (config.get_child_optional("face_system.tcp") && |
| 80 | config.get<std::string>("face_system.tcp.listen", "yes") == "yes") { |
| 81 | // default port should be the same as in TcpFactory::processConfig |
| 82 | auto port = config.get<std::string>("face_system.tcp.port", "6363"); |
| 83 | return make_shared<ndn::TcpTransport>("localhost", port); |
Teng Liang | 04d5ce6 | 2018-08-06 10:20:24 +0800 | [diff] [blame] | 84 | } |
Junxiao Shi | f4cfed1 | 2018-08-22 23:26:29 +0000 | [diff] [blame] | 85 | else { |
Davide Pesavento | 19779d8 | 2019-02-14 13:40:04 -0500 | [diff] [blame] | 86 | NDN_THROW(ConfigFile::Error("No transport is available to communicate with NFD")); |
Junxiao Shi | f4cfed1 | 2018-08-22 23:26:29 +0000 | [diff] [blame] | 87 | } |
| 88 | } |
| 89 | |
| 90 | Service::Service(const std::string& configFile, ndn::KeyChain& keyChain) |
Ashlesh Gawande | f84bac5 | 2018-08-27 07:39:31 +0000 | [diff] [blame] | 91 | : Service(keyChain, makeLocalNfdTransport(loadConfigSectionFromFile(configFile)), |
| 92 | [&configFile] (ConfigFile& config, bool isDryRun) { |
| 93 | config.parse(configFile, isDryRun); |
| 94 | }) |
Junxiao Shi | f4cfed1 | 2018-08-22 23:26:29 +0000 | [diff] [blame] | 95 | { |
Alexander Afanasyev | 3136792 | 2015-02-09 20:51:10 -0800 | [diff] [blame] | 96 | } |
| 97 | |
Davide Pesavento | 9f8b10e | 2018-08-22 08:45:37 +0000 | [diff] [blame] | 98 | Service::Service(const ConfigSection& configSection, ndn::KeyChain& keyChain) |
Ashlesh Gawande | f84bac5 | 2018-08-27 07:39:31 +0000 | [diff] [blame] | 99 | : Service(keyChain, makeLocalNfdTransport(configSection), |
| 100 | [&configSection] (ConfigFile& config, bool isDryRun) { |
| 101 | config.parse(configSection, isDryRun, "internal://nfd.conf"); |
| 102 | }) |
Junxiao Shi | f4cfed1 | 2018-08-22 23:26:29 +0000 | [diff] [blame] | 103 | { |
Junxiao Shi | f4cfed1 | 2018-08-22 23:26:29 +0000 | [diff] [blame] | 104 | } |
| 105 | |
Ashlesh Gawande | f84bac5 | 2018-08-27 07:39:31 +0000 | [diff] [blame] | 106 | template<typename ConfigParseFunc> |
| 107 | Service::Service(ndn::KeyChain& keyChain, shared_ptr<ndn::Transport> localNfdTransport, |
| 108 | const ConfigParseFunc& configParse) |
Junxiao Shi | f4cfed1 | 2018-08-22 23:26:29 +0000 | [diff] [blame] | 109 | : m_keyChain(keyChain) |
| 110 | , m_face(std::move(localNfdTransport), getGlobalIoService(), m_keyChain) |
| 111 | , m_nfdController(m_face, m_keyChain) |
| 112 | , m_fibUpdater(m_rib, m_nfdController) |
| 113 | , m_dispatcher(m_face, m_keyChain) |
Davide Pesavento | 0a71dd3 | 2019-03-17 20:36:18 -0400 | [diff] [blame] | 114 | , m_ribManager(m_rib, m_face, m_keyChain, m_nfdController, m_dispatcher) |
Alexander Afanasyev | 3136792 | 2015-02-09 20:51:10 -0800 | [diff] [blame] | 115 | { |
Teng Liang | 04d5ce6 | 2018-08-06 10:20:24 +0800 | [diff] [blame] | 116 | if (s_instance != nullptr) { |
Davide Pesavento | 19779d8 | 2019-02-14 13:40:04 -0500 | [diff] [blame] | 117 | NDN_THROW(std::logic_error("RIB service cannot be instantiated more than once")); |
Teng Liang | 04d5ce6 | 2018-08-06 10:20:24 +0800 | [diff] [blame] | 118 | } |
| 119 | if (&getGlobalIoService() != &getRibIoService()) { |
Davide Pesavento | 19779d8 | 2019-02-14 13:40:04 -0500 | [diff] [blame] | 120 | NDN_THROW(std::logic_error("RIB service must run on RIB thread")); |
Teng Liang | 04d5ce6 | 2018-08-06 10:20:24 +0800 | [diff] [blame] | 121 | } |
| 122 | s_instance = this; |
Ashlesh Gawande | f84bac5 | 2018-08-27 07:39:31 +0000 | [diff] [blame] | 123 | |
| 124 | ConfigFile config(ConfigFile::ignoreUnknownSection); |
| 125 | config.addSectionHandler(CFG_SECTION, bind(&Service::processConfig, this, _1, _2, _3)); |
| 126 | configParse(config, true); |
| 127 | configParse(config, false); |
| 128 | |
| 129 | m_ribManager.registerWithNfd(); |
| 130 | m_ribManager.enableLocalFields(); |
Alexander Afanasyev | 3136792 | 2015-02-09 20:51:10 -0800 | [diff] [blame] | 131 | } |
| 132 | |
Teng Liang | 04d5ce6 | 2018-08-06 10:20:24 +0800 | [diff] [blame] | 133 | Service::~Service() |
| 134 | { |
| 135 | s_instance = nullptr; |
| 136 | } |
| 137 | |
| 138 | Service& |
| 139 | Service::get() |
| 140 | { |
| 141 | if (s_instance == nullptr) { |
Davide Pesavento | 19779d8 | 2019-02-14 13:40:04 -0500 | [diff] [blame] | 142 | NDN_THROW(std::logic_error("RIB service is not instantiated")); |
Teng Liang | 04d5ce6 | 2018-08-06 10:20:24 +0800 | [diff] [blame] | 143 | } |
| 144 | if (&getGlobalIoService() != &getRibIoService()) { |
Davide Pesavento | 19779d8 | 2019-02-14 13:40:04 -0500 | [diff] [blame] | 145 | NDN_THROW(std::logic_error("Must get RIB service on RIB thread")); |
Teng Liang | 04d5ce6 | 2018-08-06 10:20:24 +0800 | [diff] [blame] | 146 | } |
| 147 | return *s_instance; |
| 148 | } |
Alexander Afanasyev | c3ea5a7 | 2015-02-12 20:14:16 -0800 | [diff] [blame] | 149 | |
Alexander Afanasyev | 3136792 | 2015-02-09 20:51:10 -0800 | [diff] [blame] | 150 | void |
Junxiao Shi | f4cfed1 | 2018-08-22 23:26:29 +0000 | [diff] [blame] | 151 | Service::processConfig(const ConfigSection& section, bool isDryRun, const std::string& filename) |
Alexander Afanasyev | 3136792 | 2015-02-09 20:51:10 -0800 | [diff] [blame] | 152 | { |
Junxiao Shi | f4cfed1 | 2018-08-22 23:26:29 +0000 | [diff] [blame] | 153 | if (isDryRun) { |
| 154 | checkConfig(section, filename); |
| 155 | } |
| 156 | else { |
| 157 | applyConfig(section, filename); |
| 158 | } |
| 159 | } |
Alexander Afanasyev | 3136792 | 2015-02-09 20:51:10 -0800 | [diff] [blame] | 160 | |
Junxiao Shi | f4cfed1 | 2018-08-22 23:26:29 +0000 | [diff] [blame] | 161 | void |
| 162 | Service::checkConfig(const ConfigSection& section, const std::string& filename) |
| 163 | { |
Wenkai Zheng | 6598fb0 | 2019-09-08 18:03:46 -0700 | [diff] [blame] | 164 | bool hasLocalhop = false; |
| 165 | bool hasPropagate = false; |
| 166 | |
Junxiao Shi | f4cfed1 | 2018-08-22 23:26:29 +0000 | [diff] [blame] | 167 | for (const auto& item : section) { |
| 168 | const std::string& key = item.first; |
Ashlesh Gawande | f84bac5 | 2018-08-27 07:39:31 +0000 | [diff] [blame] | 169 | const ConfigSection& value = item.second; |
Teng Liang | 18c2b29 | 2019-10-18 14:31:04 -0700 | [diff] [blame] | 170 | if (key == CFG_LOCALHOST_SECURITY || key == CFG_LOCALHOP_SECURITY || key == CFG_PA_VALIDATION) { |
Wenkai Zheng | 6598fb0 | 2019-09-08 18:03:46 -0700 | [diff] [blame] | 171 | hasLocalhop = key == CFG_LOCALHOP_SECURITY; |
Alexander Afanasyev | 3c0c035 | 2018-10-17 11:59:23 -0400 | [diff] [blame] | 172 | ndn::ValidatorConfig testValidator(m_face); |
| 173 | testValidator.load(value, filename); |
Junxiao Shi | f4cfed1 | 2018-08-22 23:26:29 +0000 | [diff] [blame] | 174 | } |
| 175 | else if (key == CFG_PREFIX_PROPAGATE) { |
Wenkai Zheng | 6598fb0 | 2019-09-08 18:03:46 -0700 | [diff] [blame] | 176 | hasPropagate = true; |
Junxiao Shi | f4cfed1 | 2018-08-22 23:26:29 +0000 | [diff] [blame] | 177 | // AutoPrefixPropagator does not support config dry-run |
| 178 | } |
| 179 | else if (key == CFG_READVERTISE_NLSR) { |
| 180 | ConfigFile::parseYesNo(item, CFG_SECTION + "." + CFG_READVERTISE_NLSR); |
| 181 | } |
| 182 | else { |
Davide Pesavento | 19779d8 | 2019-02-14 13:40:04 -0500 | [diff] [blame] | 183 | NDN_THROW(ConfigFile::Error("Unrecognized option " + CFG_SECTION + "." + key)); |
Junxiao Shi | f4cfed1 | 2018-08-22 23:26:29 +0000 | [diff] [blame] | 184 | } |
| 185 | } |
Wenkai Zheng | 6598fb0 | 2019-09-08 18:03:46 -0700 | [diff] [blame] | 186 | |
| 187 | if (hasLocalhop && hasPropagate) { |
| 188 | NDN_THROW(ConfigFile::Error(CFG_LOCALHOP_SECURITY + " and " + CFG_PREFIX_PROPAGATE + |
| 189 | " cannot be enabled at the same time")); |
| 190 | } |
Junxiao Shi | f4cfed1 | 2018-08-22 23:26:29 +0000 | [diff] [blame] | 191 | } |
| 192 | |
| 193 | void |
| 194 | Service::applyConfig(const ConfigSection& section, const std::string& filename) |
| 195 | { |
| 196 | bool wantPrefixPropagate = false; |
| 197 | bool wantReadvertiseNlsr = false; |
| 198 | |
| 199 | for (const auto& item : section) { |
| 200 | const std::string& key = item.first; |
| 201 | const ConfigSection& value = item.second; |
| 202 | if (key == CFG_LOCALHOST_SECURITY) { |
| 203 | m_ribManager.applyLocalhostConfig(value, filename); |
| 204 | } |
| 205 | else if (key == CFG_LOCALHOP_SECURITY) { |
| 206 | m_ribManager.enableLocalhop(value, filename); |
| 207 | } |
Teng Liang | 18c2b29 | 2019-10-18 14:31:04 -0700 | [diff] [blame] | 208 | else if (key == CFG_PA_VALIDATION) { |
| 209 | m_ribManager.applyPaConfig(value, filename); |
| 210 | } |
Junxiao Shi | f4cfed1 | 2018-08-22 23:26:29 +0000 | [diff] [blame] | 211 | else if (key == CFG_PREFIX_PROPAGATE) { |
Junxiao Shi | f4cfed1 | 2018-08-22 23:26:29 +0000 | [diff] [blame] | 212 | wantPrefixPropagate = true; |
Yanbiao Li | f48d080 | 2018-06-01 03:00:02 -0700 | [diff] [blame] | 213 | |
| 214 | if (!m_readvertisePropagation) { |
| 215 | NFD_LOG_DEBUG("Enabling automatic prefix propagation"); |
| 216 | |
Yanbiao Li | f48d080 | 2018-06-01 03:00:02 -0700 | [diff] [blame] | 217 | auto cost = item.second.get_optional<uint64_t>("cost"); |
Wenkai Zheng | 6598fb0 | 2019-09-08 18:03:46 -0700 | [diff] [blame] | 218 | auto parameters = ndn::nfd::ControlParameters() |
| 219 | .setCost(cost.value_or(PROPAGATE_DEFAULT_COST)) |
| 220 | .setOrigin(ndn::nfd::ROUTE_ORIGIN_CLIENT); |
Yanbiao Li | f48d080 | 2018-06-01 03:00:02 -0700 | [diff] [blame] | 221 | |
Yanbiao Li | f48d080 | 2018-06-01 03:00:02 -0700 | [diff] [blame] | 222 | auto timeout = item.second.get_optional<uint64_t>("timeout"); |
Wenkai Zheng | 6598fb0 | 2019-09-08 18:03:46 -0700 | [diff] [blame] | 223 | auto options = ndn::nfd::CommandOptions() |
| 224 | .setPrefix(RibManager::LOCALHOP_TOP_PREFIX) |
| 225 | .setTimeout(timeout ? time::milliseconds(*timeout) : PROPAGATE_DEFAULT_TIMEOUT); |
Yanbiao Li | f48d080 | 2018-06-01 03:00:02 -0700 | [diff] [blame] | 226 | |
| 227 | m_readvertisePropagation = make_unique<Readvertise>( |
| 228 | m_rib, |
Yanbiao Li | f48d080 | 2018-06-01 03:00:02 -0700 | [diff] [blame] | 229 | make_unique<HostToGatewayReadvertisePolicy>(m_keyChain, item.second), |
| 230 | make_unique<NfdRibReadvertiseDestination>(m_nfdController, m_rib, options, parameters)); |
| 231 | } |
Junxiao Shi | f4cfed1 | 2018-08-22 23:26:29 +0000 | [diff] [blame] | 232 | } |
| 233 | else if (key == CFG_READVERTISE_NLSR) { |
| 234 | wantReadvertiseNlsr = ConfigFile::parseYesNo(item, CFG_SECTION + "." + CFG_READVERTISE_NLSR); |
| 235 | } |
| 236 | else { |
Davide Pesavento | 19779d8 | 2019-02-14 13:40:04 -0500 | [diff] [blame] | 237 | NDN_THROW(ConfigFile::Error("Unrecognized option " + CFG_SECTION + "." + key)); |
Junxiao Shi | f4cfed1 | 2018-08-22 23:26:29 +0000 | [diff] [blame] | 238 | } |
Alexander Afanasyev | 3136792 | 2015-02-09 20:51:10 -0800 | [diff] [blame] | 239 | } |
| 240 | |
Yanbiao Li | f48d080 | 2018-06-01 03:00:02 -0700 | [diff] [blame] | 241 | if (!wantPrefixPropagate && m_readvertisePropagation != nullptr) { |
| 242 | NFD_LOG_DEBUG("Disabling automatic prefix propagation"); |
| 243 | m_readvertisePropagation.reset(); |
Davide Pesavento | 9f8b10e | 2018-08-22 08:45:37 +0000 | [diff] [blame] | 244 | } |
| 245 | |
Junxiao Shi | f4cfed1 | 2018-08-22 23:26:29 +0000 | [diff] [blame] | 246 | if (wantReadvertiseNlsr && m_readvertiseNlsr == nullptr) { |
| 247 | NFD_LOG_DEBUG("Enabling readvertise-to-nlsr"); |
Yanbiao Li | f48d080 | 2018-06-01 03:00:02 -0700 | [diff] [blame] | 248 | auto options = ndn::nfd::CommandOptions().setPrefix(READVERTISE_NLSR_PREFIX); |
Davide Pesavento | 9f8b10e | 2018-08-22 08:45:37 +0000 | [diff] [blame] | 249 | m_readvertiseNlsr = make_unique<Readvertise>( |
| 250 | m_rib, |
| 251 | make_unique<ClientToNlsrReadvertisePolicy>(), |
Yanbiao Li | f48d080 | 2018-06-01 03:00:02 -0700 | [diff] [blame] | 252 | make_unique<NfdRibReadvertiseDestination>(m_nfdController, m_rib, options)); |
Davide Pesavento | 9f8b10e | 2018-08-22 08:45:37 +0000 | [diff] [blame] | 253 | } |
Junxiao Shi | f4cfed1 | 2018-08-22 23:26:29 +0000 | [diff] [blame] | 254 | else if (!wantReadvertiseNlsr && m_readvertiseNlsr != nullptr) { |
| 255 | NFD_LOG_DEBUG("Disabling readvertise-to-nlsr"); |
Davide Pesavento | 9f8b10e | 2018-08-22 08:45:37 +0000 | [diff] [blame] | 256 | m_readvertiseNlsr.reset(); |
| 257 | } |
Alexander Afanasyev | 3136792 | 2015-02-09 20:51:10 -0800 | [diff] [blame] | 258 | } |
| 259 | |
Alexander Afanasyev | 3136792 | 2015-02-09 20:51:10 -0800 | [diff] [blame] | 260 | } // namespace rib |
| 261 | } // namespace nfd |