blob: 073d88c3fc642faab2eaec48572fe2f631c4cd3d [file] [log] [blame]
/* -*- Mode:C++; c-file-style:"gnu"; indent-tabs-mode:nil; -*- */
/*
* Copyright (c) 2014-2023, 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 "table/fib.hpp"
#include "table/pit.hpp"
#include "table/measurements.hpp"
#include "tests/test-common.hpp"
#include "tests/daemon/global-io-fixture.hpp"
#include "tests/daemon/face/dummy-face.hpp"
#include <ndn-cxx/util/concepts.hpp>
namespace nfd::tests {
using namespace nfd::fib;
NDN_CXX_ASSERT_FORWARD_ITERATOR(Fib::const_iterator);
BOOST_AUTO_TEST_SUITE(Table)
BOOST_FIXTURE_TEST_SUITE(TestFib, GlobalIoFixture)
BOOST_AUTO_TEST_CASE(FibEntry)
{
NameTree nameTree;
Fib fib(nameTree);
Name prefix("/pxWhfFza");
shared_ptr<Face> face1 = make_shared<DummyFace>();
shared_ptr<Face> face2 = make_shared<DummyFace>();
Face* expectedFace = nullptr;
uint64_t expectedCost = 0;
size_t nNewNextHopSignals = 0;
fib.afterNewNextHop.connect(
[&] (const Name& prefix1, const NextHop& nextHop) {
++nNewNextHopSignals;
BOOST_CHECK_EQUAL(prefix1, prefix);
BOOST_CHECK_EQUAL(&nextHop.getFace(), expectedFace);
BOOST_CHECK_EQUAL(nextHop.getCost(), expectedCost);
});
Entry& entry = *fib.insert(prefix).first;
BOOST_CHECK_EQUAL(entry.getPrefix(), prefix);
// []
BOOST_CHECK_EQUAL(entry.getNextHops().size(), 0);
expectedFace = face1.get();
expectedCost = 20;
fib.addOrUpdateNextHop(entry, *face1, 20);
// [(face1,20)]
BOOST_CHECK_EQUAL(entry.getNextHops().size(), 1);
BOOST_CHECK_EQUAL(&entry.getNextHops().begin()->getFace(), face1.get());
BOOST_CHECK_EQUAL(entry.getNextHops().begin()->getCost(), 20);
BOOST_CHECK_EQUAL(nNewNextHopSignals, 1);
fib.addOrUpdateNextHop(entry, *face1, 30);
// [(face1,30)]
BOOST_CHECK_EQUAL(entry.getNextHops().size(), 1);
BOOST_CHECK_EQUAL(&entry.getNextHops().begin()->getFace(), face1.get());
BOOST_CHECK_EQUAL(entry.getNextHops().begin()->getCost(), 30);
BOOST_CHECK_EQUAL(nNewNextHopSignals, 1);
expectedFace = face2.get();
expectedCost = 40;
fib.addOrUpdateNextHop(entry, *face2, 40);
// [(face1,30), (face2,40)]
BOOST_CHECK_EQUAL(entry.getNextHops().size(), 2);
{
auto it = entry.getNextHops().begin();
BOOST_REQUIRE(it != entry.getNextHops().end());
BOOST_CHECK_EQUAL(&it->getFace(), face1.get());
BOOST_CHECK_EQUAL(it->getCost(), 30);
++it;
BOOST_REQUIRE(it != entry.getNextHops().end());
BOOST_CHECK_EQUAL(&it->getFace(), face2.get());
BOOST_CHECK_EQUAL(it->getCost(), 40);
++it;
BOOST_CHECK(it == entry.getNextHops().end());
BOOST_CHECK_EQUAL(nNewNextHopSignals, 2);
}
fib.addOrUpdateNextHop(entry, *face2, 10);
// [(face2,10), (face1,30)]
BOOST_CHECK_EQUAL(entry.getNextHops().size(), 2);
{
auto it = entry.getNextHops().begin();
BOOST_REQUIRE(it != entry.getNextHops().end());
BOOST_CHECK_EQUAL(&it->getFace(), face2.get());
BOOST_CHECK_EQUAL(it->getCost(), 10);
++it;
BOOST_REQUIRE(it != entry.getNextHops().end());
BOOST_CHECK_EQUAL(&it->getFace(), face1.get());
BOOST_CHECK_EQUAL(it->getCost(), 30);
++it;
BOOST_CHECK(it == entry.getNextHops().end());
BOOST_CHECK_EQUAL(nNewNextHopSignals, 2);
}
Fib::RemoveNextHopResult status = fib.removeNextHop(entry, *face1);
// [(face2,10)]
BOOST_CHECK(status == Fib::RemoveNextHopResult::NEXTHOP_REMOVED);
BOOST_CHECK_EQUAL(entry.getNextHops().size(), 1);
BOOST_CHECK_EQUAL(entry.getNextHops().begin()->getFace().getId(), face2->getId());
BOOST_CHECK_EQUAL(entry.getNextHops().begin()->getCost(), 10);
status = fib.removeNextHop(entry, *face1);
// [(face2,10)]
BOOST_CHECK(status == Fib::RemoveNextHopResult::NO_SUCH_NEXTHOP);
BOOST_CHECK_EQUAL(entry.getNextHops().size(), 1);
BOOST_CHECK_EQUAL(entry.getNextHops().begin()->getFace().getId(), face2->getId());
BOOST_CHECK_EQUAL(entry.getNextHops().begin()->getCost(), 10);
status = fib.removeNextHop(entry, *face2);
// []
BOOST_CHECK(status == Fib::RemoveNextHopResult::FIB_ENTRY_REMOVED);
BOOST_CHECK(fib.findExactMatch(prefix) == nullptr);
}
BOOST_AUTO_TEST_CASE(Insert_LongestPrefixMatch)
{
NameTree nameTree;
Fib fib(nameTree);
// []
BOOST_CHECK_EQUAL(fib.size(), 0);
BOOST_CHECK_EQUAL(fib.findLongestPrefixMatch("/A").getPrefix(), "/"); // the empty entry
std::pair<Entry*, bool> insertRes = fib.insert("/");
BOOST_CHECK_EQUAL(insertRes.second, true);
BOOST_REQUIRE(insertRes.first != nullptr);
BOOST_CHECK_EQUAL(insertRes.first->getPrefix(), "/");
// ['/']
BOOST_CHECK_EQUAL(fib.size(), 1);
BOOST_CHECK_EQUAL(fib.findLongestPrefixMatch("/A").getPrefix(), "/");
insertRes = fib.insert("/A");
BOOST_CHECK_EQUAL(insertRes.second, true);
BOOST_REQUIRE(insertRes.first != nullptr);
BOOST_CHECK_EQUAL(insertRes.first->getPrefix(), "/A");
// ['/', '/A']
BOOST_CHECK_EQUAL(fib.size(), 2);
insertRes = fib.insert("/A");
BOOST_CHECK_EQUAL(insertRes.second, false);
BOOST_REQUIRE(insertRes.first != nullptr);
BOOST_CHECK_EQUAL(insertRes.first->getPrefix(), "/A");
// ['/', '/A']
BOOST_CHECK_EQUAL(fib.size(), 2);
BOOST_CHECK_EQUAL(fib.findLongestPrefixMatch("/A").getPrefix(), "/A");
BOOST_CHECK_EQUAL(fib.findLongestPrefixMatch("/A/B/C/D").getPrefix(), "/A");
insertRes = fib.insert("/A/B/C");
BOOST_CHECK_EQUAL(insertRes.second, true);
BOOST_REQUIRE(insertRes.first != nullptr);
BOOST_CHECK_EQUAL(insertRes.first->getPrefix(), "/A/B/C");
// ['/', '/A', '/A/B/C']
BOOST_CHECK_EQUAL(fib.size(), 3);
BOOST_CHECK_EQUAL(fib.findLongestPrefixMatch("/A").getPrefix(), "/A");
BOOST_CHECK_EQUAL(fib.findLongestPrefixMatch("/A/B").getPrefix(), "/A");
BOOST_CHECK_EQUAL(fib.findLongestPrefixMatch("/A/B/C/D").getPrefix(), "/A/B/C");
BOOST_CHECK_EQUAL(fib.findLongestPrefixMatch("/E").getPrefix(), "/");
}
BOOST_AUTO_TEST_CASE(LongestPrefixMatchWithPitEntry)
{
NameTree nameTree;
Fib fib(nameTree);
shared_ptr<Data> dataABC = makeData("/A/B/C");
Name fullNameABC = dataABC->getFullName();
shared_ptr<Data> dataADE = makeData("/A/D/E");
Name fullNameADE = dataADE->getFullName();
fib.insert("/A");
fib.insert(fullNameABC);
Pit pit(nameTree);
shared_ptr<Interest> interestAB = makeInterest("/A/B");
shared_ptr<pit::Entry> pitAB = pit.insert(*interestAB).first;
shared_ptr<Interest> interestABC = makeInterest(fullNameABC);
shared_ptr<pit::Entry> pitABC = pit.insert(*interestABC).first;
shared_ptr<Interest> interestADE = makeInterest(fullNameADE);
shared_ptr<pit::Entry> pitADE = pit.insert(*interestADE).first;
size_t nNameTreeEntriesBefore = nameTree.size();
BOOST_CHECK_EQUAL(fib.findLongestPrefixMatch(*pitAB).getPrefix(), "/A");
BOOST_CHECK_EQUAL(fib.findLongestPrefixMatch(*pitABC).getPrefix(), fullNameABC);
BOOST_CHECK_EQUAL(fib.findLongestPrefixMatch(*pitADE).getPrefix(), "/A");
BOOST_CHECK_EQUAL(nameTree.size(), nNameTreeEntriesBefore);
}
BOOST_AUTO_TEST_CASE(LongestPrefixMatchWithMeasurementsEntry)
{
NameTree nameTree;
Fib fib(nameTree);
fib.insert("/A");
fib.insert("/A/B/C");
Measurements measurements(nameTree);
measurements::Entry& mAB = measurements.get("/A/B");
measurements::Entry& mABCD = measurements.get("/A/B/C/D");
BOOST_CHECK_EQUAL(fib.findLongestPrefixMatch(mAB).getPrefix(), "/A");
BOOST_CHECK_EQUAL(fib.findLongestPrefixMatch(mABCD).getPrefix(), "/A/B/C");
}
static void
validateFindExactMatch(Fib& fib, const Name& target)
{
BOOST_TEST_INFO_SCOPE(target);
const Entry* entry = fib.findExactMatch(target);
BOOST_REQUIRE(entry != nullptr);
BOOST_CHECK_EQUAL(entry->getPrefix(), target);
}
static void
validateNoExactMatch(Fib& fib, const Name& target)
{
BOOST_TEST_INFO_SCOPE(target);
const Entry* entry = fib.findExactMatch(target);
BOOST_CHECK(entry == nullptr);
}
BOOST_AUTO_TEST_CASE(ExactMatch)
{
NameTree nameTree;
Fib fib(nameTree);
fib.insert("/A");
fib.insert("/A/B");
fib.insert("/A/B/C");
validateFindExactMatch(fib, "/A");
validateFindExactMatch(fib, "/A/B");
validateFindExactMatch(fib, "/A/B/C");
validateNoExactMatch(fib, "/");
validateNoExactMatch(fib, "/does/not/exist");
}
BOOST_AUTO_TEST_CASE(ExactMatchGap)
{
NameTree nameTree;
Fib fib(nameTree);
fib.insert("/X");
fib.insert("/X/Y/Z");
validateNoExactMatch(fib, "/X/Y");
}
BOOST_AUTO_TEST_CASE(ExactMatchEmpty)
{
NameTree nameTree;
Fib fib(nameTree);
validateNoExactMatch(fib, "/");
validateNoExactMatch(fib, "/nothing/here");
}
static void
validateErase(Fib& fib, const Name& target)
{
BOOST_TEST_INFO_SCOPE(target);
fib.erase(target);
BOOST_CHECK(fib.findExactMatch(target) == nullptr);
}
BOOST_AUTO_TEST_CASE(Erase)
{
NameTree nameTree;
Fib fib(nameTree);
fib.insert("/");
fib.insert("/A");
fib.insert("/A/B");
fib.insert("/A/B/C");
// check if we remove the right thing and leave
// everything else alone
validateErase(fib, "/A/B");
validateFindExactMatch(fib, "/A");
validateFindExactMatch(fib, "/A/B/C");
validateFindExactMatch(fib, "/");
validateErase(fib, "/A/B/C");
validateFindExactMatch(fib, "/A");
validateFindExactMatch(fib, "/");
validateErase(fib, "/A");
validateFindExactMatch(fib, "/");
validateErase(fib, "/");
validateNoExactMatch(fib, "/");
}
BOOST_AUTO_TEST_CASE(EraseGap)
{
NameTree nameTree;
Fib fib(nameTree);
fib.insert("/X");
fib.insert("/X/Y/Z");
fib.erase("/X/Y"); //should do nothing
validateFindExactMatch(fib, "/X");
validateFindExactMatch(fib, "/X/Y/Z");
}
BOOST_AUTO_TEST_CASE(EraseEmpty)
{
NameTree nameTree;
Fib fib(nameTree);
BOOST_CHECK_NO_THROW(fib.erase("/does/not/exist"));
validateErase(fib, "/");
BOOST_CHECK_NO_THROW(fib.erase("/still/does/not/exist"));
}
BOOST_AUTO_TEST_CASE(EraseNameTreeEntry)
{
NameTree nameTree;
Fib fib(nameTree);
size_t nNameTreeEntriesBefore = nameTree.size();
fib.insert("/A/B/C");
fib.erase("/A/B/C");
BOOST_CHECK_EQUAL(nameTree.size(), nNameTreeEntriesBefore);
}
BOOST_AUTO_TEST_CASE(Iterator)
{
NameTree nameTree;
Fib fib(nameTree);
Name nameA("/A");
Name nameAB("/A/B");
Name nameABC("/A/B/C");
Name nameRoot("/");
fib.insert(nameA);
fib.insert(nameAB);
fib.insert(nameABC);
fib.insert(nameRoot);
std::set<Name> expected{nameA, nameAB, nameABC, nameRoot};
for (Fib::const_iterator it = fib.begin(); it != fib.end(); it++) {
bool isInSet = expected.find(it->getPrefix()) != expected.end();
BOOST_CHECK(isInSet);
expected.erase(it->getPrefix());
}
BOOST_CHECK_EQUAL(expected.size(), 0);
}
BOOST_AUTO_TEST_SUITE_END() // TestFib
BOOST_AUTO_TEST_SUITE_END() // Table
} // namespace nfd::tests