blob: 814feb076872af81da3ec5789ac6f8dd23cc7311 [file] [log] [blame]
/* -*- Mode:C++; c-file-style:"gnu"; indent-tabs-mode:nil; -*- */
/*
* Copyright (c) 2014-2024, Regents of the University of California,
* Arizona Board of Regents,
* Colorado State University,
* University Pierre & Marie Curie, Sorbonne University,
* Washington University in St. Louis,
* Beijing Institute of Technology,
* The University of Memphis.
*
* This file is part of NFD (Named Data Networking Forwarding Daemon).
* See AUTHORS.md for complete list of NFD authors and contributors.
*
* NFD is free software: you can redistribute it and/or modify it under the terms
* of the GNU General Public License as published by the Free Software Foundation,
* either version 3 of the License, or (at your option) any later version.
*
* NFD is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY;
* without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR
* PURPOSE. See the GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License along with
* NFD, e.g., in COPYING.md file. If not, see <http://www.gnu.org/licenses/>.
*/
#include "service.hpp"
#include "fib-updater.hpp"
#include "readvertise/client-to-nlsr-readvertise-policy.hpp"
#include "readvertise/host-to-gateway-readvertise-policy.hpp"
#include "readvertise/nfd-rib-readvertise-destination.hpp"
#include "readvertise/readvertise.hpp"
#include "common/global.hpp"
#include "common/logger.hpp"
#include <boost/property_tree/info_parser.hpp>
#include <ndn-cxx/transport/tcp-transport.hpp>
#include <ndn-cxx/transport/unix-transport.hpp>
namespace nfd::rib {
NFD_LOG_INIT(RibService);
const std::string CFG_RIB = "rib";
const std::string CFG_LOCALHOST_SECURITY = "localhost_security";
const std::string CFG_LOCALHOP_SECURITY = "localhop_security";
const std::string CFG_PA_VALIDATION = "prefix_announcement_validation";
const std::string CFG_PREFIX_PROPAGATE = "auto_prefix_propagate";
const std::string CFG_READVERTISE_NLSR = "readvertise_nlsr";
const Name READVERTISE_NLSR_PREFIX = "/localhost/nlsr";
constexpr uint64_t PROPAGATE_DEFAULT_COST = 15;
constexpr time::milliseconds PROPAGATE_DEFAULT_TIMEOUT = 10_s;
static ConfigSection
loadConfigSectionFromFile(const std::string& filename)
{
ConfigSection config;
// Any format errors should have been caught already
boost::property_tree::read_info(filename, config);
return config;
}
// Look into NFD's config file and construct an appropriate transport to communicate with NFD.
static shared_ptr<ndn::Transport>
makeLocalNfdTransport(const ConfigSection& config)
{
if (config.get_child_optional("face_system.unix")) {
// default socket path should be the same as in UnixStreamFactory::processConfig
#ifdef __linux__
auto path = config.get<std::string>("face_system.unix.path", "/run/nfd/nfd.sock");
#else
auto path = config.get<std::string>("face_system.unix.path", "/var/run/nfd/nfd.sock");
#endif // __linux__
return make_shared<ndn::UnixTransport>(path);
}
else if (config.get_child_optional("face_system.tcp") &&
config.get<std::string>("face_system.tcp.listen", "yes") == "yes") {
// default port should be the same as in TcpFactory::processConfig
auto port = config.get<std::string>("face_system.tcp.port", "6363");
return make_shared<ndn::TcpTransport>("localhost", port);
}
else {
NDN_THROW(ConfigFile::Error("No transport is available to communicate with NFD"));
}
}
Service::Service(const std::string& configFile, ndn::KeyChain& keyChain)
: Service(keyChain, makeLocalNfdTransport(loadConfigSectionFromFile(configFile)),
[&configFile] (ConfigFile& config, bool isDryRun) {
config.parse(configFile, isDryRun);
})
{
}
Service::Service(const ConfigSection& configSection, ndn::KeyChain& keyChain)
: Service(keyChain, makeLocalNfdTransport(configSection),
[&configSection] (ConfigFile& config, bool isDryRun) {
config.parse(configSection, isDryRun, "internal://nfd.conf");
})
{
}
template<typename ConfigParseFunc>
Service::Service(ndn::KeyChain& keyChain, shared_ptr<ndn::Transport> localNfdTransport,
const ConfigParseFunc& configParse)
: m_keyChain(keyChain)
, m_face(std::move(localNfdTransport), getGlobalIoService(), m_keyChain)
, m_nfdController(m_face, m_keyChain)
, m_fibUpdater(m_rib, m_nfdController)
, m_dispatcher(m_face, m_keyChain)
, m_ribManager(m_rib, m_face, m_keyChain, m_nfdController, m_dispatcher)
{
if (s_instance != nullptr) {
NDN_THROW(std::logic_error("RIB service cannot be instantiated more than once"));
}
if (&getGlobalIoService() != &getRibIoService()) {
NDN_THROW(std::logic_error("RIB service must run on RIB thread"));
}
s_instance = this;
ConfigFile config(ConfigFile::ignoreUnknownSection);
config.addSectionHandler(CFG_RIB, [this] (auto&&... args) {
processConfig(std::forward<decltype(args)>(args)...);
});
configParse(config, true);
configParse(config, false);
m_ribManager.registerWithNfd();
m_ribManager.enableLocalFields();
}
Service::~Service()
{
s_instance = nullptr;
}
Service&
Service::get()
{
if (s_instance == nullptr) {
NDN_THROW(std::logic_error("RIB service is not instantiated"));
}
if (&getGlobalIoService() != &getRibIoService()) {
NDN_THROW(std::logic_error("Must get RIB service on RIB thread"));
}
return *s_instance;
}
void
Service::processConfig(const ConfigSection& section, bool isDryRun, const std::string& filename)
{
if (isDryRun) {
checkConfig(section, filename);
}
else {
applyConfig(section, filename);
}
}
void
Service::checkConfig(const ConfigSection& section, const std::string& filename)
{
bool hasLocalhop = false;
bool hasPropagate = false;
for (const auto& item : section) {
const std::string& key = item.first;
const ConfigSection& value = item.second;
if (key == CFG_LOCALHOST_SECURITY || key == CFG_PA_VALIDATION) {
ndn::ValidatorConfig testValidator(m_face);
testValidator.load(value, filename);
}
else if (key == CFG_LOCALHOP_SECURITY) {
hasLocalhop = true;
ndn::ValidatorConfig testValidator(m_face);
testValidator.load(value, filename);
}
else if (key == CFG_PREFIX_PROPAGATE) {
hasPropagate = true;
// AutoPrefixPropagator does not support config dry-run
}
else if (key == CFG_READVERTISE_NLSR) {
ConfigFile::parseYesNo(item, CFG_RIB + "." + CFG_READVERTISE_NLSR);
}
else {
NDN_THROW(ConfigFile::Error("Unrecognized option " + CFG_RIB + "." + key));
}
}
if (hasLocalhop && hasPropagate) {
NDN_THROW(ConfigFile::Error(CFG_LOCALHOP_SECURITY + " and " + CFG_PREFIX_PROPAGATE +
" cannot be enabled at the same time"));
}
}
void
Service::applyConfig(const ConfigSection& section, const std::string& filename)
{
bool wantPrefixPropagate = false;
bool wantReadvertiseNlsr = false;
for (const auto& item : section) {
const std::string& key = item.first;
const ConfigSection& value = item.second;
if (key == CFG_LOCALHOST_SECURITY) {
m_ribManager.applyLocalhostConfig(value, filename);
}
else if (key == CFG_LOCALHOP_SECURITY) {
m_ribManager.enableLocalhop(value, filename);
}
else if (key == CFG_PA_VALIDATION) {
m_ribManager.applyPaConfig(value, filename);
}
else if (key == CFG_PREFIX_PROPAGATE) {
wantPrefixPropagate = true;
if (!m_readvertisePropagation) {
NFD_LOG_DEBUG("Enabling automatic prefix propagation");
auto cost = item.second.get_optional<uint64_t>("cost");
auto parameters = ndn::nfd::ControlParameters()
.setCost(cost.value_or(PROPAGATE_DEFAULT_COST))
.setOrigin(ndn::nfd::ROUTE_ORIGIN_CLIENT);
auto timeout = item.second.get_optional<uint64_t>("timeout");
auto options = ndn::nfd::CommandOptions()
.setPrefix(RibManager::LOCALHOP_TOP_PREFIX)
.setTimeout(timeout ? time::milliseconds(*timeout) : PROPAGATE_DEFAULT_TIMEOUT);
m_readvertisePropagation = make_unique<Readvertise>(
m_rib,
make_unique<HostToGatewayReadvertisePolicy>(m_keyChain, item.second),
make_unique<NfdRibReadvertiseDestination>(m_nfdController, m_rib, options, parameters));
}
}
else if (key == CFG_READVERTISE_NLSR) {
wantReadvertiseNlsr = ConfigFile::parseYesNo(item, CFG_RIB + "." + CFG_READVERTISE_NLSR);
}
else {
NDN_THROW(ConfigFile::Error("Unrecognized option " + CFG_RIB + "." + key));
}
}
if (!wantPrefixPropagate && m_readvertisePropagation != nullptr) {
NFD_LOG_DEBUG("Disabling automatic prefix propagation");
m_readvertisePropagation.reset();
}
if (wantReadvertiseNlsr && m_readvertiseNlsr == nullptr) {
NFD_LOG_DEBUG("Enabling readvertise-to-nlsr");
auto options = ndn::nfd::CommandOptions().setPrefix(READVERTISE_NLSR_PREFIX);
m_readvertiseNlsr = make_unique<Readvertise>(
m_rib,
make_unique<ClientToNlsrReadvertisePolicy>(),
make_unique<NfdRibReadvertiseDestination>(m_nfdController, m_rib, options));
}
else if (!wantReadvertiseNlsr && m_readvertiseNlsr != nullptr) {
NFD_LOG_DEBUG("Disabling readvertise-to-nlsr");
m_readvertiseNlsr.reset();
}
}
} // namespace nfd::rib