| /* -*- Mode:C++; c-file-style:"gnu"; indent-tabs-mode:nil; -*- */ |
| /** |
| * Copyright (c) 2014-2016, 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 "rib/rib-manager.hpp" |
| #include "manager-common-fixture.hpp" |
| |
| #include <ndn-cxx/management/nfd-rib-entry.hpp> |
| #include <ndn-cxx/management/nfd-face-status.hpp> |
| #include <ndn-cxx/management/nfd-face-event-notification.hpp> |
| #include <ndn-cxx/util/random.hpp> |
| |
| namespace nfd { |
| namespace rib { |
| namespace tests { |
| |
| using namespace nfd::tests; |
| |
| struct ConfigurationStatus |
| { |
| bool isLocalhostConfigured; |
| bool isLocalhopConfigured; |
| }; |
| |
| class RibManagerFixture : public ManagerCommonFixture |
| { |
| public: |
| explicit |
| RibManagerFixture(const ConfigurationStatus& status, |
| bool shouldClearRib) |
| : m_commands(m_face->sentInterests) |
| , m_status(status) |
| , m_manager(m_dispatcher, *m_face, m_keyChain) |
| , m_rib(m_manager.m_rib) |
| { |
| m_rib.m_onSendBatchFromQueue = bind(&RibManagerFixture::onSendBatchFromQueue, this, _1); |
| |
| const std::string prefix = "rib\n{\n"; |
| const std::string suffix = "}"; |
| const std::string localhostSection = " localhost_security\n" |
| " {\n" |
| " trust-anchor\n" |
| " {\n" |
| " type any\n" |
| " }\n" |
| " }\n"; |
| const std::string localhopSection = " localhop_security\n" |
| " {\n" |
| " trust-anchor\n" |
| " {\n" |
| " type any\n" |
| " }\n" |
| " }\n"; |
| |
| std::string ribSection = ""; |
| if (m_status.isLocalhostConfigured) { |
| ribSection += localhostSection; |
| } |
| if (m_status.isLocalhopConfigured) { |
| ribSection += localhopSection; |
| } |
| const std::string CONFIG_STR = prefix + ribSection + suffix; |
| |
| ConfigFile config; |
| m_manager.setConfigFile(config); |
| config.parse(CONFIG_STR, true, "test-rib"); |
| |
| registerWithNfd(); |
| |
| if (shouldClearRib) { |
| clearRib(); |
| } |
| } |
| |
| private: |
| void |
| registerWithNfd() |
| { |
| m_manager.registerWithNfd(); |
| advanceClocks(time::milliseconds(1)); |
| |
| auto replyFibAddCommand = [this] (const Interest& interest) { |
| nfd::ControlParameters params(interest.getName().get(-5).blockFromValue()); |
| params.setFaceId(1).setCost(0); |
| nfd::ControlResponse resp; |
| |
| resp.setCode(200).setBody(params.wireEncode()); |
| shared_ptr<Data> data = make_shared<Data>(interest.getName()); |
| data->setContent(resp.wireEncode()); |
| |
| m_keyChain.sign(*data, ndn::security::SigningInfo(ndn::security::SigningInfo::SIGNER_TYPE_SHA256)); |
| |
| m_face->getIoService().post([this, data] { m_face->receive(*data); }); |
| }; |
| |
| Name commandPrefix("/localhost/nfd/fib/add-nexthop"); |
| for (auto&& command : m_commands) { |
| if (commandPrefix.isPrefixOf(command.getName())) { |
| replyFibAddCommand(command); |
| advanceClocks(time::milliseconds(1)); |
| } |
| } |
| |
| // clear commands and responses |
| m_responses.clear(); |
| m_commands.clear(); |
| } |
| |
| void clearRib() |
| { |
| while (!m_rib.empty()) { |
| auto& name = m_rib.begin()->first; |
| auto& routes = m_rib.begin()->second->getRoutes(); |
| while (!routes.empty()) { |
| m_rib.erase(name, *routes.begin()); |
| } |
| } |
| } |
| |
| public: |
| ControlParameters |
| makeRegisterParameters(const Name& name, uint64_t id = 0, |
| const time::milliseconds expiry = time::milliseconds::max()) |
| { |
| return ControlParameters() |
| .setName(name) |
| .setFaceId(id) |
| .setOrigin(128) |
| .setCost(10) |
| .setFlags(0) |
| .setExpirationPeriod(expiry); |
| } |
| |
| ControlParameters |
| makeUnregisterParameters(const Name& name, uint64_t id = 0) |
| { |
| return ControlParameters() |
| .setName(name) |
| .setFaceId(id) |
| .setOrigin(128); |
| } |
| |
| void |
| onSendBatchFromQueue(const RibUpdateBatch& batch) |
| { |
| BOOST_ASSERT(batch.begin() != batch.end()); |
| RibUpdate update = *(batch.begin()); |
| |
| // Simulate a successful response from NFD |
| FibUpdater& updater = m_manager.m_fibUpdater; |
| m_manager.m_rib.onFibUpdateSuccess(batch, updater.m_inheritedRoutes, |
| bind(&RibManager::onRibUpdateSuccess, &m_manager, update)); |
| } |
| |
| public: |
| enum class CheckCommandResult { |
| OK, |
| OUT_OF_BOUNDARY, |
| WRONG_FORMAT, |
| WRONG_VERB, |
| WRONG_PARAMS_FORMAT, |
| WRONG_PARAMS_NAME, |
| WRONG_PARAMS_FACE |
| }; |
| |
| CheckCommandResult |
| checkCommand(size_t idx, const char* verbStr, ControlParameters expectedParams) |
| { |
| if (idx > m_commands.size()) { |
| return CheckCommandResult::OUT_OF_BOUNDARY; |
| } |
| const auto& name = m_commands[idx].getName(); |
| |
| if (!FIB_COMMAND_PREFIX.isPrefixOf(name) || FIB_COMMAND_PREFIX.size() >= name.size()) { |
| return CheckCommandResult::WRONG_FORMAT; |
| } |
| const auto& verb = name[FIB_COMMAND_PREFIX.size()]; |
| |
| Name::Component expectedVerb(verbStr); |
| if (verb != expectedVerb) { |
| return CheckCommandResult::WRONG_VERB; |
| } |
| |
| ControlParameters parameters; |
| try { |
| Block rawParameters = name[FIB_COMMAND_PREFIX.size() + 1].blockFromValue(); |
| parameters.wireDecode(rawParameters); |
| } |
| catch (...) { |
| return CheckCommandResult::WRONG_PARAMS_FORMAT; |
| } |
| |
| if (parameters.getName() != expectedParams.getName()) { |
| return CheckCommandResult::WRONG_PARAMS_NAME; |
| } |
| |
| if (parameters.getFaceId() != expectedParams.getFaceId()) { |
| return CheckCommandResult::WRONG_PARAMS_FACE; |
| } |
| |
| return CheckCommandResult::OK; |
| } |
| |
| public: |
| std::vector<Interest>& m_commands; |
| ConfigurationStatus m_status; |
| static const Name FIB_COMMAND_PREFIX; |
| |
| protected: |
| RibManager m_manager; |
| Rib& m_rib; |
| }; |
| |
| const Name RibManagerFixture::FIB_COMMAND_PREFIX("/localhost/nfd/fib"); |
| |
| std::ostream& |
| operator<<(std::ostream& os, const RibManagerFixture::CheckCommandResult& result) |
| { |
| switch (result) { |
| case RibManagerFixture::CheckCommandResult::OK: |
| os << "OK"; |
| break; |
| case RibManagerFixture::CheckCommandResult::OUT_OF_BOUNDARY: |
| os << "OUT_OF_BOUNDARY"; |
| break; |
| case RibManagerFixture::CheckCommandResult::WRONG_FORMAT: |
| os << "WRONG_FORMAT"; |
| break; |
| case RibManagerFixture::CheckCommandResult::WRONG_VERB: |
| os << "WRONG_VERB"; |
| break; |
| case RibManagerFixture::CheckCommandResult::WRONG_PARAMS_FORMAT: |
| os << "WRONG_COST"; |
| break; |
| case RibManagerFixture::CheckCommandResult::WRONG_PARAMS_NAME: |
| os << "WRONG_PARAMS_NAME"; |
| break; |
| case RibManagerFixture::CheckCommandResult::WRONG_PARAMS_FACE: |
| os << "WRONG_PARAMS_FACE"; |
| break; |
| default: |
| break; |
| }; |
| |
| return os; |
| } |
| |
| BOOST_AUTO_TEST_SUITE(Rib) |
| BOOST_AUTO_TEST_SUITE(TestRibManager) |
| |
| class AddTopPrefixFixture : public RibManagerFixture |
| { |
| public: |
| AddTopPrefixFixture() |
| : RibManagerFixture({true, true}, false) |
| { |
| } |
| }; |
| |
| BOOST_FIXTURE_TEST_CASE(AddTopPrefix, AddTopPrefixFixture) |
| { |
| BOOST_CHECK_EQUAL(m_rib.size(), 2); |
| |
| std::vector<Name> ribEntryNames; |
| for (auto&& entry : m_rib) { |
| ribEntryNames.push_back(entry.first); |
| } |
| BOOST_CHECK_EQUAL(ribEntryNames[0], "/localhop/nfd"); |
| BOOST_CHECK_EQUAL(ribEntryNames[1], "/localhost/nfd"); |
| } |
| |
| class UnauthorizedRibManagerFixture : public RibManagerFixture |
| { |
| public: |
| UnauthorizedRibManagerFixture() |
| : RibManagerFixture({false, false}, true) |
| { |
| } |
| }; |
| |
| class LocalhostAuthorizedRibManagerFixture : public RibManagerFixture |
| { |
| public: |
| LocalhostAuthorizedRibManagerFixture() |
| : RibManagerFixture({true, false}, true) |
| { |
| } |
| }; |
| |
| class LocalhopAuthorizedRibManagerFixture : public RibManagerFixture |
| { |
| public: |
| LocalhopAuthorizedRibManagerFixture() |
| : RibManagerFixture({false, true}, true) |
| { |
| } |
| }; |
| |
| class AuthorizedRibManagerFixture : public RibManagerFixture |
| { |
| public: |
| AuthorizedRibManagerFixture() |
| : RibManagerFixture({true, true}, true) |
| { |
| } |
| }; |
| |
| typedef boost::mpl::vector< |
| UnauthorizedRibManagerFixture, |
| LocalhostAuthorizedRibManagerFixture, |
| LocalhopAuthorizedRibManagerFixture, |
| AuthorizedRibManagerFixture |
| > AllFixtures; |
| |
| BOOST_FIXTURE_TEST_CASE_TEMPLATE(CommandAuthorization, T, AllFixtures, T) |
| { |
| auto parameters = this->makeRegisterParameters("/test-authorization", 9527); |
| auto commandHost = this->makeControlCommandRequest("/localhost/nfd/rib/register", parameters); |
| auto commandHop = this->makeControlCommandRequest("/localhop/nfd/rib/register", parameters); |
| auto successResp = this->makeResponse(200, "Success", parameters); |
| auto failureResp = ControlResponse(403, "authorization rejected"); |
| |
| BOOST_CHECK_EQUAL(this->m_responses.size(), 0); |
| this->receiveInterest(commandHost); |
| this->receiveInterest(commandHop); |
| |
| auto nExpectedResponses = this->m_status.isLocalhopConfigured ? 2 : 1; |
| auto expectedLocalhostResponse = this->m_status.isLocalhostConfigured ? successResp : failureResp; |
| auto expectedLocalhopResponse = this->m_status.isLocalhopConfigured ? successResp : failureResp; |
| |
| BOOST_REQUIRE_EQUAL(this->m_responses.size(), nExpectedResponses); |
| BOOST_CHECK_EQUAL(this->checkResponse(0, commandHost->getName(), expectedLocalhostResponse), |
| ManagerCommonFixture::CheckResponseResult::OK); |
| if (nExpectedResponses == 2) { |
| BOOST_CHECK_EQUAL(this->checkResponse(1, commandHop->getName(), expectedLocalhopResponse), |
| ManagerCommonFixture::CheckResponseResult::OK); |
| } |
| } |
| |
| BOOST_FIXTURE_TEST_SUITE(RegisterUnregister, LocalhostAuthorizedRibManagerFixture) |
| |
| BOOST_AUTO_TEST_CASE(Basic) |
| { |
| auto paramsRegister = makeRegisterParameters("/test-register-unregister", 9527); |
| auto paramsUnregister = makeUnregisterParameters("/test-register-unregister", 9527); |
| |
| auto setInFaceId = [] (shared_ptr<Interest> commandInterest) { |
| commandInterest->setTag(make_shared<lp::IncomingFaceIdTag>(1234)); |
| }; |
| |
| auto commandRegister = makeControlCommandRequest("/localhost/nfd/rib/register", paramsRegister, setInFaceId); |
| auto commandUnregister = makeControlCommandRequest("/localhost/nfd/rib/unregister", paramsUnregister, setInFaceId); |
| |
| receiveInterest(commandRegister); |
| receiveInterest(commandUnregister); |
| |
| BOOST_REQUIRE_EQUAL(m_responses.size(), 2); |
| BOOST_CHECK_EQUAL(checkResponse(0, commandRegister->getName(), makeResponse(200, "Success", paramsRegister)), |
| CheckResponseResult::OK); |
| BOOST_CHECK_EQUAL(checkResponse(1, commandUnregister->getName(), makeResponse(200, "Success", paramsUnregister)), |
| CheckResponseResult::OK); |
| |
| BOOST_REQUIRE_EQUAL(m_commands.size(), 2); |
| BOOST_CHECK_EQUAL(checkCommand(0, "add-nexthop", paramsRegister), CheckCommandResult::OK); |
| BOOST_CHECK_EQUAL(checkCommand(1, "remove-nexthop", paramsUnregister), CheckCommandResult::OK); |
| } |
| |
| BOOST_AUTO_TEST_CASE(SelfOperation) |
| { |
| auto paramsRegister = makeRegisterParameters("/test-self-register-unregister"); |
| auto paramsUnregister = makeUnregisterParameters("/test-self-register-unregister"); |
| BOOST_CHECK_EQUAL(paramsRegister.getFaceId(), 0); |
| BOOST_CHECK_EQUAL(paramsUnregister.getFaceId(), 0); |
| |
| const uint64_t inFaceId = 9527; |
| auto setInFaceId = [&inFaceId] (shared_ptr<Interest> commandInterest) { |
| commandInterest->setTag(make_shared<lp::IncomingFaceIdTag>(inFaceId)); |
| }; |
| auto commandRegister = makeControlCommandRequest("/localhost/nfd/rib/register", paramsRegister, setInFaceId); |
| auto commandUnregister = makeControlCommandRequest("/localhost/nfd/rib/unregister", paramsUnregister, setInFaceId); |
| |
| receiveInterest(commandRegister); |
| receiveInterest(commandUnregister); |
| |
| paramsRegister.setFaceId(inFaceId); |
| paramsUnregister.setFaceId(inFaceId); |
| |
| BOOST_REQUIRE_EQUAL(m_responses.size(), 2); |
| BOOST_CHECK_EQUAL(checkResponse(0, commandRegister->getName(), makeResponse(200, "Success", paramsRegister)), |
| CheckResponseResult::OK); |
| BOOST_CHECK_EQUAL(checkResponse(1, commandUnregister->getName(), makeResponse(200, "Success", paramsUnregister)), |
| CheckResponseResult::OK); |
| |
| BOOST_REQUIRE_EQUAL(m_commands.size(), 2); |
| BOOST_CHECK_EQUAL(checkCommand(0, "add-nexthop", paramsRegister), CheckCommandResult::OK); |
| BOOST_CHECK_EQUAL(checkCommand(1, "remove-nexthop", paramsUnregister), CheckCommandResult::OK); |
| } |
| |
| BOOST_AUTO_TEST_CASE(Expiration) |
| { |
| auto paramsRegister = makeRegisterParameters("/test-expiry", 9527, time::milliseconds(50)); |
| auto paramsUnregister = makeRegisterParameters("/test-expiry", 9527); |
| receiveInterest(makeControlCommandRequest("/localhost/nfd/rib/register", paramsRegister)); |
| |
| advanceClocks(time::milliseconds(55)); |
| BOOST_REQUIRE_EQUAL(m_commands.size(), 2); // the registered route has expired |
| BOOST_CHECK_EQUAL(checkCommand(0, "add-nexthop", paramsRegister), CheckCommandResult::OK); |
| BOOST_CHECK_EQUAL(checkCommand(1, "remove-nexthop", paramsUnregister), CheckCommandResult::OK); |
| |
| m_commands.clear(); |
| paramsRegister.setExpirationPeriod(time::milliseconds(100)); |
| receiveInterest(makeControlCommandRequest("/localhost/nfd/rib/register", paramsRegister)); |
| |
| advanceClocks(time::milliseconds(55)); |
| BOOST_REQUIRE_EQUAL(m_commands.size(), 1); // the registered route is still active |
| BOOST_CHECK_EQUAL(checkCommand(0, "add-nexthop", paramsRegister), CheckCommandResult::OK); |
| } |
| |
| BOOST_AUTO_TEST_SUITE_END() // RegisterUnregister |
| |
| // @todo Remove when ndn::nfd::RibEntry implements operator!= |
| class RibEntry : public ndn::nfd::RibEntry |
| { |
| public: |
| RibEntry() = default; |
| |
| RibEntry(const ndn::nfd::RibEntry& entry) |
| : ndn::nfd::RibEntry(entry) |
| { |
| } |
| }; |
| |
| bool |
| operator!=(const RibEntry& left, const RibEntry& right) |
| { |
| if (left.getName() != right.getName()) { |
| return true; |
| } |
| |
| auto leftRoutes = left.getRoutes(); |
| auto rightRoutes = right.getRoutes(); |
| if (leftRoutes.size() != rightRoutes.size()) { |
| return true; |
| } |
| |
| for (auto&& route : leftRoutes) { |
| auto hitEntry = |
| std::find_if(rightRoutes.begin(), rightRoutes.end(), [&] (const ndn::nfd::Route& record) { |
| return route.getFaceId() == record.getFaceId() && |
| route.getCost() == record.getCost() && |
| route.getOrigin() == record.getOrigin() && |
| route.getFlags() == record.getFlags() && |
| route.getExpirationPeriod() == record.getExpirationPeriod(); |
| }); |
| |
| if (hitEntry == rightRoutes.end()) { |
| return true; |
| } |
| } |
| |
| return false; |
| } |
| |
| BOOST_FIXTURE_TEST_CASE(RibDataset, UnauthorizedRibManagerFixture) |
| { |
| uint64_t faceId = 0; |
| auto generateRoute = [&faceId] () -> Route { |
| Route route; |
| route.faceId = ++faceId; |
| route.cost = ndn::random::generateWord64(); |
| route.expires = time::steady_clock::TimePoint::max(); |
| return route; |
| }; |
| |
| const size_t nEntries = 108; |
| std::set<Name> actualPrefixes; |
| for (size_t i = 0; i < nEntries; ++i) { |
| Name prefix = Name("/test-dataset").appendNumber(i); |
| actualPrefixes.insert(prefix); |
| m_rib.insert(prefix, generateRoute()); |
| if (i & 0x1) { |
| m_rib.insert(prefix, generateRoute()); |
| m_rib.insert(prefix, generateRoute()); |
| } |
| } |
| |
| receiveInterest(makeInterest("/localhost/nfd/rib/list")); |
| |
| Block content; |
| BOOST_CHECK_NO_THROW(content = concatenateResponses()); |
| BOOST_CHECK_NO_THROW(content.parse()); |
| BOOST_REQUIRE_EQUAL(content.elements().size(), nEntries); |
| |
| std::vector<RibEntry> receivedRecords, expectedRecords; |
| for (size_t idx = 0; idx < nEntries; ++idx) { |
| BOOST_TEST_MESSAGE("processing element: " << idx); |
| |
| RibEntry decodedEntry; |
| BOOST_REQUIRE_NO_THROW(decodedEntry.wireDecode(content.elements()[idx])); |
| receivedRecords.push_back(decodedEntry); |
| |
| actualPrefixes.erase(decodedEntry.getName()); |
| |
| auto matchedEntryIt = m_rib.find(decodedEntry.getName()); |
| BOOST_REQUIRE(matchedEntryIt != m_rib.end()); |
| |
| auto matchedEntry = matchedEntryIt->second; |
| BOOST_REQUIRE(matchedEntry != nullptr); |
| |
| RibEntry record; |
| record.setName(matchedEntry->getName()); |
| const auto& routes = matchedEntry->getRoutes(); |
| for (auto&& route : routes) { |
| ndn::nfd::Route routeRecord; |
| routeRecord.setFaceId(route.faceId); |
| routeRecord.setOrigin(route.origin); |
| routeRecord.setFlags(route.flags); |
| routeRecord.setCost(route.cost); |
| record.addRoute(routeRecord); |
| } |
| expectedRecords.push_back(record); |
| } |
| |
| BOOST_CHECK_EQUAL(actualPrefixes.size(), 0); |
| |
| BOOST_CHECK_EQUAL_COLLECTIONS(receivedRecords.begin(), receivedRecords.end(), |
| expectedRecords.begin(), expectedRecords.end()); |
| } |
| |
| BOOST_FIXTURE_TEST_SUITE(FaceMonitor, LocalhostAuthorizedRibManagerFixture) |
| |
| BOOST_AUTO_TEST_CASE(FetchActiveFacesEvent) |
| { |
| BOOST_CHECK_EQUAL(m_commands.size(), 0); |
| |
| advanceClocks(time::seconds(301)); // RibManager::ACTIVE_FACE_FETCH_INTERVAL = 300s |
| BOOST_REQUIRE_EQUAL(m_commands.size(), 2); |
| BOOST_CHECK_EQUAL(m_commands[0].getName(), "/localhost/nfd/faces/events"); |
| BOOST_CHECK_EQUAL(m_commands[1].getName(), "/localhost/nfd/faces/list"); |
| } |
| |
| BOOST_AUTO_TEST_CASE(RemoveInvalidFaces) |
| { |
| auto parameters1 = makeRegisterParameters("/test-remove-invalid-faces-1"); |
| auto parameters2 = makeRegisterParameters("/test-remove-invalid-faces-2"); |
| receiveInterest(makeControlCommandRequest("/localhost/nfd/rib/register", parameters1.setFaceId(1))); |
| receiveInterest(makeControlCommandRequest("/localhost/nfd/rib/register", parameters1.setFaceId(2))); |
| receiveInterest(makeControlCommandRequest("/localhost/nfd/rib/register", parameters2.setFaceId(2))); |
| BOOST_REQUIRE_EQUAL(m_rib.size(), 3); |
| |
| ndn::nfd::FaceStatus status; |
| status.setFaceId(1); |
| |
| auto data = makeData("/localhost/nfd/faces/list"); |
| data->setContent(status.wireEncode()); |
| |
| auto buffer = make_shared<ndn::OBufferStream>(); |
| buffer->write(reinterpret_cast<const char*>(data->getContent().value()), |
| data->getContent().value_size()); |
| |
| m_manager.removeInvalidFaces(buffer); |
| advanceClocks(time::milliseconds(100)); |
| BOOST_REQUIRE_EQUAL(m_rib.size(), 1); |
| |
| auto it1 = m_rib.find("/test-remove-invalid-faces-1"); |
| auto it2 = m_rib.find("/test-remove-invalid-faces-2"); |
| BOOST_CHECK(it2 == m_rib.end()); |
| BOOST_REQUIRE(it1 != m_rib.end()); |
| BOOST_CHECK(it1->second->hasFaceId(1)); |
| BOOST_CHECK(!it1->second->hasFaceId(2)); |
| } |
| |
| BOOST_AUTO_TEST_CASE(OnNotification) |
| { |
| auto parameters1 = makeRegisterParameters("/test-face-event-notification-1", 1); |
| auto parameters2 = makeRegisterParameters("/test-face-event-notification-2", 1); |
| receiveInterest(makeControlCommandRequest("/localhost/nfd/rib/register", parameters1)); |
| receiveInterest(makeControlCommandRequest("/localhost/nfd/rib/register", parameters2)); |
| BOOST_REQUIRE_EQUAL(m_rib.size(), 2); |
| |
| auto makeNotification = [] (ndn::nfd::FaceEventKind eventKind, uint64_t faceId) -> ndn::nfd::FaceEventNotification { |
| ndn::nfd::FaceEventNotification notification; |
| notification.setKind(eventKind).setFaceId(faceId); |
| return notification; |
| }; |
| |
| m_manager.onNotification(makeNotification(ndn::nfd::FaceEventKind::FACE_EVENT_DESTROYED, 1)); |
| advanceClocks(time::milliseconds(100)); |
| BOOST_CHECK_EQUAL(m_rib.size(), 0); |
| |
| m_manager.onNotification(makeNotification(ndn::nfd::FaceEventKind::FACE_EVENT_CREATED, 2)); |
| advanceClocks(time::milliseconds(100)); |
| BOOST_CHECK_EQUAL(m_rib.size(), 0); |
| } |
| |
| BOOST_AUTO_TEST_SUITE_END() // FaceMonitor |
| |
| BOOST_AUTO_TEST_SUITE_END() // TestRibManager |
| BOOST_AUTO_TEST_SUITE_END() // Rib |
| |
| } // namespace tests |
| } // namespace rib |
| } // namespace nfd |