service.cpp
Go to the documentation of this file.
1 /* -*- Mode:C++; c-file-style:"gnu"; indent-tabs-mode:nil; -*- */
2 /*
3  * Copyright (c) 2014-2019, 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 "service.hpp"
27 
28 #include "fib-updater.hpp"
33 
34 #include "common/global.hpp"
35 #include "common/logger.hpp"
36 
37 #include <boost/property_tree/info_parser.hpp>
38 #include <ndn-cxx/transport/tcp-transport.hpp>
39 #include <ndn-cxx/transport/unix-transport.hpp>
40 
41 namespace nfd {
42 namespace rib {
43 
44 NFD_LOG_INIT(RibService);
45 
46 Service* Service::s_instance = nullptr;
47 
48 static const std::string CFG_SECTION = "rib";
49 static const std::string CFG_LOCALHOST_SECURITY = "localhost_security";
50 static const std::string CFG_LOCALHOP_SECURITY = "localhop_security";
51 static const std::string CFG_PREFIX_PROPAGATE = "auto_prefix_propagate";
52 static const std::string CFG_READVERTISE_NLSR = "readvertise_nlsr";
53 static const Name READVERTISE_NLSR_PREFIX = "/localhost/nlsr";
54 static const uint64_t PROPAGATE_DEFAULT_COST = 15;
55 static const time::milliseconds PROPAGATE_DEFAULT_TIMEOUT = 10_s;
56 
57 static ConfigSection
58 loadConfigSectionFromFile(const std::string& filename)
59 {
60  ConfigSection config;
61  // Any format errors should have been caught already
62  boost::property_tree::read_info(filename, config);
63  return config;
64 }
65 
70 static shared_ptr<ndn::Transport>
72 {
73  if (config.get_child_optional("face_system.unix")) {
74  // default socket path should be the same as in UnixStreamFactory::processConfig
75  auto path = config.get<std::string>("face_system.unix.path", "/var/run/nfd.sock");
76  return make_shared<ndn::UnixTransport>(path);
77  }
78  else if (config.get_child_optional("face_system.tcp") &&
79  config.get<std::string>("face_system.tcp.listen", "yes") == "yes") {
80  // default port should be the same as in TcpFactory::processConfig
81  auto port = config.get<std::string>("face_system.tcp.port", "6363");
82  return make_shared<ndn::TcpTransport>("localhost", port);
83  }
84  else {
85  NDN_THROW(ConfigFile::Error("No transport is available to communicate with NFD"));
86  }
87 }
88 
89 Service::Service(const std::string& configFile, ndn::KeyChain& keyChain)
91  [&configFile] (ConfigFile& config, bool isDryRun) {
92  config.parse(configFile, isDryRun);
93  })
94 {
95 }
96 
97 Service::Service(const ConfigSection& configSection, ndn::KeyChain& keyChain)
98  : Service(keyChain, makeLocalNfdTransport(configSection),
99  [&configSection] (ConfigFile& config, bool isDryRun) {
100  config.parse(configSection, isDryRun, "internal://nfd.conf");
101  })
102 {
103 }
104 
105 template<typename ConfigParseFunc>
106 Service::Service(ndn::KeyChain& keyChain, shared_ptr<ndn::Transport> localNfdTransport,
107  const ConfigParseFunc& configParse)
108  : m_keyChain(keyChain)
109  , m_face(std::move(localNfdTransport), getGlobalIoService(), m_keyChain)
110  , m_nfdController(m_face, m_keyChain)
111  , m_fibUpdater(m_rib, m_nfdController)
112  , m_dispatcher(m_face, m_keyChain)
113  , m_ribManager(m_rib, m_face, m_keyChain, m_nfdController, m_dispatcher)
114 {
115  if (s_instance != nullptr) {
116  NDN_THROW(std::logic_error("RIB service cannot be instantiated more than once"));
117  }
118  if (&getGlobalIoService() != &getRibIoService()) {
119  NDN_THROW(std::logic_error("RIB service must run on RIB thread"));
120  }
121  s_instance = this;
122 
124  config.addSectionHandler(CFG_SECTION, bind(&Service::processConfig, this, _1, _2, _3));
125  configParse(config, true);
126  configParse(config, false);
127 
128  m_ribManager.registerWithNfd();
129  m_ribManager.enableLocalFields();
130 }
131 
133 {
134  s_instance = nullptr;
135 }
136 
137 Service&
139 {
140  if (s_instance == nullptr) {
141  NDN_THROW(std::logic_error("RIB service is not instantiated"));
142  }
143  if (&getGlobalIoService() != &getRibIoService()) {
144  NDN_THROW(std::logic_error("Must get RIB service on RIB thread"));
145  }
146  return *s_instance;
147 }
148 
149 void
150 Service::processConfig(const ConfigSection& section, bool isDryRun, const std::string& filename)
151 {
152  if (isDryRun) {
153  checkConfig(section, filename);
154  }
155  else {
156  applyConfig(section, filename);
157  }
158 }
159 
160 void
161 Service::checkConfig(const ConfigSection& section, const std::string& filename)
162 {
163  for (const auto& item : section) {
164  const std::string& key = item.first;
165  const ConfigSection& value = item.second;
166  if (key == CFG_LOCALHOST_SECURITY || key == CFG_LOCALHOP_SECURITY) {
167  ndn::ValidatorConfig testValidator(m_face);
168  testValidator.load(value, filename);
169  }
170  else if (key == CFG_PREFIX_PROPAGATE) {
171  // AutoPrefixPropagator does not support config dry-run
172  }
173  else if (key == CFG_READVERTISE_NLSR) {
174  ConfigFile::parseYesNo(item, CFG_SECTION + "." + CFG_READVERTISE_NLSR);
175  }
176  else {
177  NDN_THROW(ConfigFile::Error("Unrecognized option " + CFG_SECTION + "." + key));
178  }
179  }
180 }
181 
182 void
183 Service::applyConfig(const ConfigSection& section, const std::string& filename)
184 {
185  bool wantPrefixPropagate = false;
186  bool wantReadvertiseNlsr = false;
187 
188  for (const auto& item : section) {
189  const std::string& key = item.first;
190  const ConfigSection& value = item.second;
191  if (key == CFG_LOCALHOST_SECURITY) {
192  m_ribManager.applyLocalhostConfig(value, filename);
193  }
194  else if (key == CFG_LOCALHOP_SECURITY) {
195  m_ribManager.enableLocalhop(value, filename);
196  }
197  else if (key == CFG_PREFIX_PROPAGATE) {
198  wantPrefixPropagate = true;
199 
200  if (!m_readvertisePropagation) {
201  NFD_LOG_DEBUG("Enabling automatic prefix propagation");
202 
203  auto parameters = ndn::nfd::ControlParameters()
204  .setCost(PROPAGATE_DEFAULT_COST)
205  .setOrigin(ndn::nfd::ROUTE_ORIGIN_CLIENT);
206  auto cost = item.second.get_optional<uint64_t>("cost");
207  if (cost) {
208  parameters.setCost(*cost);
209  }
210 
211  auto options = ndn::nfd::CommandOptions()
213  .setTimeout(PROPAGATE_DEFAULT_TIMEOUT);
214  auto timeout = item.second.get_optional<uint64_t>("timeout");
215  if (timeout) {
216  options.setTimeout(time::milliseconds(*timeout));
217  }
218 
219  m_readvertisePropagation = make_unique<Readvertise>(
220  m_rib,
221  make_unique<HostToGatewayReadvertisePolicy>(m_keyChain, item.second),
222  make_unique<NfdRibReadvertiseDestination>(m_nfdController, m_rib, options, parameters));
223  }
224  }
225  else if (key == CFG_READVERTISE_NLSR) {
226  wantReadvertiseNlsr = ConfigFile::parseYesNo(item, CFG_SECTION + "." + CFG_READVERTISE_NLSR);
227  }
228  else {
229  NDN_THROW(ConfigFile::Error("Unrecognized option " + CFG_SECTION + "." + key));
230  }
231  }
232 
233  if (!wantPrefixPropagate && m_readvertisePropagation != nullptr) {
234  NFD_LOG_DEBUG("Disabling automatic prefix propagation");
235  m_readvertisePropagation.reset();
236  }
237 
238  if (wantReadvertiseNlsr && m_readvertiseNlsr == nullptr) {
239  NFD_LOG_DEBUG("Enabling readvertise-to-nlsr");
240  auto options = ndn::nfd::CommandOptions().setPrefix(READVERTISE_NLSR_PREFIX);
241  m_readvertiseNlsr = make_unique<Readvertise>(
242  m_rib,
243  make_unique<ClientToNlsrReadvertisePolicy>(),
244  make_unique<NfdRibReadvertiseDestination>(m_nfdController, m_rib, options));
245  }
246  else if (!wantReadvertiseNlsr && m_readvertiseNlsr != nullptr) {
247  NFD_LOG_DEBUG("Disabling readvertise-to-nlsr");
248  m_readvertiseNlsr.reset();
249  }
250 }
251 
252 } // namespace rib
253 } // namespace nfd
static const Name READVERTISE_NLSR_PREFIX
Definition: service.cpp:53
void addSectionHandler(const std::string &sectionName, ConfigSectionHandler subscriber)
setup notification of configuration file sections
Definition: config-file.cpp:77
static Service & get()
Get a reference to the only instance of this class.
Definition: service.cpp:138
static const std::string CFG_SECTION
Definition: service.cpp:48
static const time::milliseconds PROPAGATE_DEFAULT_TIMEOUT
Definition: service.cpp:55
configuration file parsing utility
Definition: config-file.hpp:57
static ConfigSection loadConfigSectionFromFile(const std::string &filename)
Definition: service.cpp:58
static const std::string CFG_LOCALHOST_SECURITY
Definition: service.cpp:49
void enableLocalhop(const ConfigSection &section, const std::string &filename)
Apply localhop_security configuration and allow accepting commands on /localhop/nfd/rib prefix...
Definition: rib-manager.cpp:77
static bool parseYesNo(const ConfigSection &node, const std::string &key, const std::string &sectionName)
parse a config option that can be either "yes" or "no"
Definition: config-file.cpp:60
static const std::string CFG_READVERTISE_NLSR
Definition: service.cpp:52
void applyLocalhostConfig(const ConfigSection &section, const std::string &filename)
Apply localhost_security configuration.
Definition: rib-manager.cpp:71
static shared_ptr< ndn::Transport > makeLocalNfdTransport(const ConfigSection &config)
Look into the config file and construct appropriate transport to communicate with NFD If NFD-RIB inst...
Definition: service.cpp:71
boost::asio::io_service & getRibIoService()
Definition: global.cpp:70
static const uint64_t PROPAGATE_DEFAULT_COST
Definition: service.cpp:54
void registerWithNfd()
Start accepting commands and dataset requests.
Definition: rib-manager.cpp:90
Service(const std::string &configFile, ndn::KeyChain &keyChain)
create NFD-RIB service
Definition: service.cpp:89
boost::property_tree::ptree ConfigSection
a config file section
Definition: config-file.hpp:37
Copyright (c) 2014-2015, Regents of the University of California, Arizona Board of Regents...
Definition: algorithm.hpp:32
static const Name LOCALHOP_TOP_PREFIX
static const std::string CFG_LOCALHOP_SECURITY
Definition: service.cpp:50
#define NFD_LOG_DEBUG
Definition: logger.hpp:38
#define NFD_LOG_INIT(name)
Definition: logger.hpp:31
static const std::string CFG_PREFIX_PROPAGATE
Definition: service.cpp:51
~Service()
Destructor.
Definition: service.cpp:132
void enableLocalFields()
Enable NDNLP IncomingFaceId field in order to support self-registration commands. ...
initializes and executes NFD-RIB service thread
Definition: service.hpp:52
static void ignoreUnknownSection(const std::string &filename, const std::string &sectionName, const ConfigSection &section, bool isDryRun)
Definition: config-file.cpp:51
boost::asio::io_service & getGlobalIoService()
Returns the global io_service instance for the calling thread.
Definition: global.cpp:36