blob: 93e1ad9f360504a2fb4032d2146136e49c728262 [file] [log] [blame]
/* -*- Mode:C++; c-file-style:"gnu"; indent-tabs-mode:nil; -*- */
/*
* Copyright (c) 2014-2024, The University of Memphis,
* Regents of the University of California,
* Arizona Board of Regents.
*
* This file is part of NLSR (Named-data Link State Routing).
* See AUTHORS.md for complete list of NLSR authors and contributors.
*
* NLSR 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.
*
* NLSR 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
* NLSR, e.g., in COPYING.md file. If not, see <http://www.gnu.org/licenses/>.
*/
#ifndef NLSR_LSDB_HPP
#define NLSR_LSDB_HPP
#include "communication/sync-logic-handler.hpp"
#include "conf-parameter.hpp"
#include "lsa/lsa.hpp"
#include "lsa/name-lsa.hpp"
#include "lsa/coordinate-lsa.hpp"
#include "lsa/adj-lsa.hpp"
#include "sequencing-manager.hpp"
#include "statistics.hpp"
#include "test-access-control.hpp"
#include <ndn-cxx/ims/in-memory-storage-fifo.hpp>
#include <ndn-cxx/ims/in-memory-storage-persistent.hpp>
#include <ndn-cxx/security/key-chain.hpp>
#include <ndn-cxx/util/segmenter.hpp>
#include <ndn-cxx/util/segment-fetcher.hpp>
#include <ndn-cxx/util/signal.hpp>
#include <ndn-cxx/util/time.hpp>
#include <boost/multi_index_container.hpp>
#include <boost/multi_index/composite_key.hpp>
#include <boost/multi_index/hashed_index.hpp>
namespace nlsr {
namespace bmi = boost::multi_index;
inline constexpr ndn::time::seconds GRACE_PERIOD = 10_s;
enum class LsdbUpdate {
INSTALLED,
UPDATED,
REMOVED
};
class Lsdb
{
public:
Lsdb(ndn::Face& face, ndn::KeyChain& keyChain, ConfParameter& confParam);
~Lsdb();
/*! \brief Returns whether the LSDB contains some LSA.
*/
bool
doesLsaExist(const ndn::Name& router, Lsa::Type lsaType)
{
return m_lsdb.get<byName>().find(std::make_tuple(router, lsaType)) != m_lsdb.end();
}
/*! \brief Builds a name LSA for this router and then installs it
into the LSDB.
*/
void
buildAndInstallOwnNameLsa();
PUBLIC_WITH_TESTS_ELSE_PRIVATE:
/*! \brief Builds a cor. LSA for this router and installs it into the LSDB. */
void
buildAndInstallOwnCoordinateLsa();
public:
/*! \brief Schedules a build of this router's LSA. */
void
scheduleAdjLsaBuild();
void
writeLog() const;
/* \brief Process interest which can be either:
* 1) Discovery interest from segment fetcher:
* /localhop/<network>/nlsr/LSA/<site>/<router>/<lsaType>/<seqNo>
* 2) Interest containing segment number:
* /localhop/<network>/nlsr/LSA/<site>/<router>/<lsaType>/<seqNo>/<version>/<segmentNo>
*/
void
processInterest(const ndn::Name& name, const ndn::Interest& interest);
bool
getIsBuildAdjLsaScheduled() const
{
return m_isBuildAdjLsaScheduled;
}
SyncLogicHandler&
getSync()
{
return m_sync;
}
template<typename T>
std::shared_ptr<T>
findLsa(const ndn::Name& router) const
{
return std::static_pointer_cast<T>(findLsa(router, T::type()));
}
struct name_hash {
int
operator()(const ndn::Name& name) const {
return std::hash<ndn::Name>{}(name);
}
};
struct enum_class_hash {
template<typename T>
int
operator()(T t) const {
return static_cast<int>(t);
}
};
struct byName{};
struct byType{};
using LsaContainer = boost::multi_index_container<
std::shared_ptr<Lsa>,
bmi::indexed_by<
bmi::hashed_unique<
bmi::tag<byName>,
bmi::composite_key<
Lsa,
bmi::const_mem_fun<Lsa, ndn::Name, &Lsa::getOriginRouterCopy>,
bmi::const_mem_fun<Lsa, Lsa::Type, &Lsa::getType>
>,
bmi::composite_key_hash<name_hash, enum_class_hash>
>,
bmi::hashed_non_unique<
bmi::tag<byType>,
bmi::const_mem_fun<Lsa, Lsa::Type, &Lsa::getType>,
enum_class_hash
>
>
>;
template<typename T>
std::pair<LsaContainer::index<Lsdb::byType>::type::iterator,
LsaContainer::index<Lsdb::byType>::type::iterator>
getLsdbIterator() const
{
return m_lsdb.get<byType>().equal_range(T::type());
}
PUBLIC_WITH_TESTS_ELSE_PRIVATE:
std::shared_ptr<Lsa>
findLsa(const ndn::Name& router, Lsa::Type lsaType) const
{
auto it = m_lsdb.get<byName>().find(std::make_tuple(router, lsaType));
return it != m_lsdb.end() ? *it : nullptr;
}
void
incrementDataSentStats(Lsa::Type lsaType)
{
if (lsaType == Lsa::Type::NAME) {
lsaIncrementSignal(Statistics::PacketType::SENT_NAME_LSA_DATA);
}
else if (lsaType == Lsa::Type::ADJACENCY) {
lsaIncrementSignal(Statistics::PacketType::SENT_ADJ_LSA_DATA);
}
else if (lsaType == Lsa::Type::COORDINATE) {
lsaIncrementSignal(Statistics::PacketType::SENT_COORD_LSA_DATA);
}
}
void
incrementInterestRcvdStats(Lsa::Type lsaType)
{
if (lsaType == Lsa::Type::NAME) {
lsaIncrementSignal(Statistics::PacketType::RCV_NAME_LSA_INTEREST);
}
else if (lsaType == Lsa::Type::ADJACENCY) {
lsaIncrementSignal(Statistics::PacketType::RCV_ADJ_LSA_INTEREST);
}
else if (lsaType == Lsa::Type::COORDINATE) {
lsaIncrementSignal(Statistics::PacketType::RCV_COORD_LSA_INTEREST);
}
}
void
incrementInterestSentStats(Lsa::Type lsaType)
{
if (lsaType == Lsa::Type::NAME) {
lsaIncrementSignal(Statistics::PacketType::SENT_NAME_LSA_INTEREST);
}
else if (lsaType == Lsa::Type::ADJACENCY) {
lsaIncrementSignal(Statistics::PacketType::SENT_ADJ_LSA_INTEREST);
}
else if (lsaType == Lsa::Type::COORDINATE) {
lsaIncrementSignal(Statistics::PacketType::SENT_COORD_LSA_INTEREST);
}
}
/*! Returns whether a seq. no. from a certain router signals a new LSA.
\param originRouter The name of the originating router.
\param lsaType The type of the LSA.
\param seqNo The sequence number to check.
*/
bool
isLsaNew(const ndn::Name& originRouter, Lsa::Type lsaType, uint64_t seqNo) const
{
// Is the name in the LSDB and the supplied seq no is the highest so far
auto lsaPtr = findLsa(originRouter, lsaType);
return lsaPtr ? lsaPtr->getSeqNo() < seqNo : true;
}
void
installLsa(std::shared_ptr<Lsa> lsa);
/*! \brief Remove a name LSA from the LSDB.
\param router The name of the router that published the LSA to remove.
\param lsaType The type of the LSA.
This function will remove a name LSA from the LSDB by finding an
LSA whose name matches key. This removal also causes the NPT to
remove those name prefixes if no more LSAs advertise them.
*/
void
removeLsa(const ndn::Name& router, Lsa::Type lsaType);
void
removeLsa(const LsaContainer::index<Lsdb::byName>::type::iterator& lsaIt);
/*! \brief Attempts to construct an adj. LSA.
This function will attempt to construct an adjacency LSA. An LSA
can only be built when the status of all of the router's neighbors
is known. That is, when we are not currently trying to contact any
neighbor.
*/
void
buildAdjLsa();
/*! \brief Wrapper event to build and install an adj. LSA for this router. */
void
buildAndInstallOwnAdjLsa();
/*! \brief Schedules a refresh/expire event in the scheduler.
\param lsa The LSA.
\param expTime How many seconds to wait before triggering the event.
*/
ndn::scheduler::EventId
scheduleLsaExpiration(std::shared_ptr<Lsa> lsa, ndn::time::seconds expTime);
/*! \brief Either allow to expire, or refresh a name LSA.
\param lsa The LSA.
*/
void
expireOrRefreshLsa(std::shared_ptr<Lsa> lsa);
bool
processInterestForLsa(const ndn::Interest& interest, const ndn::Name& originRouter,
Lsa::Type lsaType, uint64_t seqNo);
void
expressInterest(const ndn::Name& interestName, uint32_t timeoutCount, uint64_t incomingFaceId,
ndn::time::steady_clock::time_point deadline = DEFAULT_LSA_RETRIEVAL_DEADLINE);
/*!
\brief Error callback when SegmentFetcher fails to return an LSA
In all error cases, a reattempt to fetch the LSA will be made.
Segment validation can fail either because the packet does not have a
valid signature (fatal) or because some of the certificates in the trust chain
could not be fetched (non-fatal).
Currently, the library does not provide clear indication (besides a plain-text message
in the error callback) of the reason for the failure nor the segment that failed
to be validated, thus we will continue to try to fetch the LSA until the deadline
is reached.
*/
void
onFetchLsaError(uint32_t errorCode, const std::string& msg,
const ndn::Name& interestName, uint32_t retransmitNo,
const ndn::time::steady_clock::time_point& deadline,
ndn::Name lsaName, uint64_t seqNo);
/*!
\brief Success callback when SegmentFetcher returns a valid LSA
\param interestName The base Interest used to fetch the LSA in the format:
/<network>/NLSR/LSA/<site>/%C1.Router/<router>/<lsa-type>/<seqNo>
*/
void
afterFetchLsa(const ndn::ConstBufferPtr& bufferPtr, const ndn::Name& interestName);
void
emitSegmentValidatedSignal(const ndn::Data& data)
{
afterSegmentValidatedSignal(data);
}
ndn::time::system_clock::time_point
getLsaExpirationTimePoint() const
{
return ndn::time::system_clock::now() + ndn::time::seconds(m_confParam.getRouterDeadInterval());
}
public:
ndn::signal::Signal<Lsdb, Statistics::PacketType> lsaIncrementSignal;
ndn::signal::Signal<Lsdb, ndn::Data> afterSegmentValidatedSignal;
using AfterLsdbModified = ndn::signal::Signal<Lsdb, std::shared_ptr<Lsa>, LsdbUpdate,
std::list<ndn::Name>, std::list<ndn::Name>>;
AfterLsdbModified onLsdbModified;
PUBLIC_WITH_TESTS_ELSE_PRIVATE:
ndn::Face& m_face;
ndn::Scheduler m_scheduler;
ConfParameter& m_confParam;
SyncLogicHandler m_sync;
LsaContainer m_lsdb;
ndn::time::seconds m_lsaRefreshTime;
ndn::time::seconds m_adjLsaBuildInterval;
const ndn::Name& m_thisRouterPrefix;
// Maps the name of an LSA to its highest known sequence number from sync;
// Used to stop NLSR from trying to fetch outdated LSAs
std::map<ndn::Name, uint64_t> m_highestSeqNo;
SequencingManager m_sequencingManager;
ndn::signal::ScopedConnection m_onNewLsaConnection;
std::set<std::shared_ptr<ndn::SegmentFetcher>> m_fetchers;
ndn::Segmenter m_segmenter;
ndn::InMemoryStorageFifo m_segmentFifo;
bool m_isBuildAdjLsaScheduled;
int64_t m_adjBuildCount;
ndn::scheduler::ScopedEventId m_scheduledAdjLsaBuild;
ndn::InMemoryStoragePersistent m_lsaStorage;
static inline const ndn::time::steady_clock::time_point DEFAULT_LSA_RETRIEVAL_DEADLINE =
ndn::time::steady_clock::time_point::min();
};
} // namespace nlsr
#endif // NLSR_LSDB_HPP