| /* -*- Mode:C++; c-file-style:"gnu"; indent-tabs-mode:nil; -*- */ |
| /** |
| * Copyright (c) 2014-2017, 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 "face/generic-link-service.hpp" |
| #include "face/face.hpp" |
| #include "dummy-transport.hpp" |
| #include <ndn-cxx/lp/tags.hpp> |
| |
| #include "tests/test-common.hpp" |
| |
| namespace nfd { |
| namespace face { |
| namespace tests { |
| |
| using namespace nfd::tests; |
| |
| BOOST_AUTO_TEST_SUITE(Face) |
| |
| using nfd::Face; |
| |
| class GenericLinkServiceFixture : public BaseFixture |
| { |
| protected: |
| GenericLinkServiceFixture() |
| : service(nullptr) |
| , transport(nullptr) |
| { |
| this->initialize(GenericLinkService::Options()); |
| // By default, GenericLinkService is created with default options. |
| // Test cases may invoke .initialize with alternate options. |
| } |
| |
| void |
| initialize(const GenericLinkService::Options& options, ssize_t mtu = MTU_UNLIMITED) |
| { |
| face.reset(new Face(make_unique<GenericLinkService>(options), |
| make_unique<DummyTransport>("dummy://", |
| "dummy://", |
| ndn::nfd::FACE_SCOPE_NON_LOCAL, |
| ndn::nfd::FACE_PERSISTENCY_PERSISTENT, |
| ndn::nfd::LINK_TYPE_POINT_TO_POINT, |
| mtu))); |
| service = static_cast<GenericLinkService*>(face->getLinkService()); |
| transport = static_cast<DummyTransport*>(face->getTransport()); |
| |
| face->afterReceiveInterest.connect( |
| [this] (const Interest& interest) { receivedInterests.push_back(interest); }); |
| face->afterReceiveData.connect( |
| [this] (const Data& data) { receivedData.push_back(data); }); |
| face->afterReceiveNack.connect( |
| [this] (const lp::Nack& nack) { receivedNacks.push_back(nack); }); |
| } |
| |
| protected: |
| unique_ptr<Face> face; |
| GenericLinkService* service; |
| DummyTransport* transport; |
| std::vector<Interest> receivedInterests; |
| std::vector<Data> receivedData; |
| std::vector<lp::Nack> receivedNacks; |
| }; |
| |
| BOOST_FIXTURE_TEST_SUITE(TestGenericLinkService, GenericLinkServiceFixture) |
| |
| |
| BOOST_AUTO_TEST_SUITE(SimpleSendReceive) // send and receive without other fields |
| |
| BOOST_AUTO_TEST_CASE(SendInterest) |
| { |
| // Initialize with Options that disables all services |
| GenericLinkService::Options options; |
| options.allowLocalFields = false; |
| initialize(options); |
| |
| shared_ptr<Interest> interest1 = makeInterest("/localhost/test"); |
| |
| face->sendInterest(*interest1); |
| |
| BOOST_CHECK_EQUAL(service->getCounters().nOutInterests, 1); |
| BOOST_REQUIRE_EQUAL(transport->sentPackets.size(), 1); |
| lp::Packet interest1pkt; |
| BOOST_REQUIRE_NO_THROW(interest1pkt.wireDecode(transport->sentPackets.back().packet)); |
| BOOST_CHECK(interest1pkt.has<lp::FragmentField>()); |
| BOOST_CHECK(!interest1pkt.has<lp::SequenceField>()); |
| } |
| |
| BOOST_AUTO_TEST_CASE(SendData) |
| { |
| // Initialize with Options that disables all services |
| GenericLinkService::Options options; |
| options.allowLocalFields = false; |
| initialize(options); |
| |
| shared_ptr<Data> data1 = makeData("/localhost/test"); |
| |
| face->sendData(*data1); |
| |
| BOOST_CHECK_EQUAL(service->getCounters().nOutData, 1); |
| BOOST_REQUIRE_EQUAL(transport->sentPackets.size(), 1); |
| lp::Packet data1pkt; |
| BOOST_REQUIRE_NO_THROW(data1pkt.wireDecode(transport->sentPackets.back().packet)); |
| BOOST_CHECK(data1pkt.has<lp::FragmentField>()); |
| BOOST_CHECK(!data1pkt.has<lp::SequenceField>()); |
| } |
| |
| BOOST_AUTO_TEST_CASE(SendNack) |
| { |
| // Initialize with Options that disables all services |
| GenericLinkService::Options options; |
| options.allowLocalFields = false; |
| initialize(options); |
| |
| lp::Nack nack1 = makeNack("/localhost/test", 323, lp::NackReason::NO_ROUTE); |
| |
| face->sendNack(nack1); |
| |
| BOOST_CHECK_EQUAL(service->getCounters().nOutNacks, 1); |
| BOOST_REQUIRE_EQUAL(transport->sentPackets.size(), 1); |
| lp::Packet nack1pkt; |
| BOOST_REQUIRE_NO_THROW(nack1pkt.wireDecode(transport->sentPackets.back().packet)); |
| BOOST_CHECK(nack1pkt.has<lp::NackField>()); |
| BOOST_CHECK(nack1pkt.has<lp::FragmentField>()); |
| BOOST_CHECK(!nack1pkt.has<lp::SequenceField>()); |
| } |
| |
| BOOST_AUTO_TEST_CASE(ReceiveBareInterest) |
| { |
| // Initialize with Options that disables all services |
| GenericLinkService::Options options; |
| options.allowLocalFields = false; |
| initialize(options); |
| |
| shared_ptr<Interest> interest1 = makeInterest("/23Rd9hEiR"); |
| |
| transport->receivePacket(interest1->wireEncode()); |
| |
| BOOST_CHECK_EQUAL(service->getCounters().nInInterests, 1); |
| BOOST_REQUIRE_EQUAL(receivedInterests.size(), 1); |
| BOOST_CHECK_EQUAL(receivedInterests.back(), *interest1); |
| } |
| |
| BOOST_AUTO_TEST_CASE(ReceiveInterest) |
| { |
| // Initialize with Options that disables all services |
| GenericLinkService::Options options; |
| options.allowLocalFields = false; |
| initialize(options); |
| |
| shared_ptr<Interest> interest1 = makeInterest("/23Rd9hEiR"); |
| lp::Packet lpPacket; |
| lpPacket.set<lp::FragmentField>(std::make_pair( |
| interest1->wireEncode().begin(), interest1->wireEncode().end())); |
| lpPacket.set<lp::SequenceField>(0); // force LpPacket encoding |
| |
| transport->receivePacket(lpPacket.wireEncode()); |
| |
| BOOST_CHECK_EQUAL(service->getCounters().nInInterests, 1); |
| BOOST_REQUIRE_EQUAL(receivedInterests.size(), 1); |
| BOOST_CHECK_EQUAL(receivedInterests.back(), *interest1); |
| } |
| |
| BOOST_AUTO_TEST_CASE(ReceiveBareData) |
| { |
| // Initialize with Options that disables all services |
| GenericLinkService::Options options; |
| options.allowLocalFields = false; |
| initialize(options); |
| |
| shared_ptr<Data> data1 = makeData("/12345678"); |
| |
| transport->receivePacket(data1->wireEncode()); |
| |
| BOOST_CHECK_EQUAL(service->getCounters().nInData, 1); |
| BOOST_REQUIRE_EQUAL(receivedData.size(), 1); |
| BOOST_CHECK_EQUAL(receivedData.back(), *data1); |
| } |
| |
| BOOST_AUTO_TEST_CASE(ReceiveData) |
| { |
| // Initialize with Options that disables all services |
| GenericLinkService::Options options; |
| options.allowLocalFields = false; |
| initialize(options); |
| |
| shared_ptr<Data> data1 = makeData("/12345689"); |
| lp::Packet lpPacket; |
| lpPacket.set<lp::FragmentField>(std::make_pair( |
| data1->wireEncode().begin(), data1->wireEncode().end())); |
| lpPacket.set<lp::SequenceField>(0); // force LpPacket encoding |
| |
| transport->receivePacket(lpPacket.wireEncode()); |
| |
| BOOST_CHECK_EQUAL(service->getCounters().nInData, 1); |
| BOOST_REQUIRE_EQUAL(receivedData.size(), 1); |
| BOOST_CHECK_EQUAL(receivedData.back(), *data1); |
| } |
| |
| BOOST_AUTO_TEST_CASE(ReceiveNack) |
| { |
| // Initialize with Options that disables all services |
| GenericLinkService::Options options; |
| options.allowLocalFields = false; |
| initialize(options); |
| |
| lp::Nack nack1 = makeNack("/localhost/test", 323, lp::NackReason::NO_ROUTE); |
| lp::Packet lpPacket; |
| lpPacket.set<lp::FragmentField>(std::make_pair( |
| nack1.getInterest().wireEncode().begin(), nack1.getInterest().wireEncode().end())); |
| lpPacket.set<lp::NackField>(nack1.getHeader()); |
| |
| transport->receivePacket(lpPacket.wireEncode()); |
| |
| BOOST_CHECK_EQUAL(service->getCounters().nInNacks, 1); |
| BOOST_REQUIRE_EQUAL(receivedNacks.size(), 1); |
| BOOST_CHECK(receivedNacks.back().getReason() == nack1.getReason()); |
| BOOST_CHECK(receivedNacks.back().getInterest() == nack1.getInterest()); |
| } |
| |
| BOOST_AUTO_TEST_CASE(ReceiveIdlePacket) |
| { |
| // Initialize with Options that disables all services |
| GenericLinkService::Options options; |
| options.allowLocalFields = false; |
| initialize(options); |
| |
| lp::Packet lpPacket; |
| lpPacket.set<lp::SequenceField>(0); |
| |
| BOOST_CHECK_NO_THROW(transport->receivePacket(lpPacket.wireEncode())); |
| |
| // IDLE packet should be ignored, but is not an error |
| BOOST_CHECK_EQUAL(service->getCounters().nInLpInvalid, 0); |
| BOOST_CHECK_EQUAL(receivedInterests.size(), 0); |
| BOOST_CHECK_EQUAL(receivedData.size(), 0); |
| BOOST_CHECK_EQUAL(receivedNacks.size(), 0); |
| } |
| |
| BOOST_AUTO_TEST_SUITE_END() // SimpleSendReceive |
| |
| BOOST_AUTO_TEST_SUITE(Fragmentation) |
| |
| BOOST_AUTO_TEST_CASE(FragmentationDisabledExceedMtuDrop) |
| { |
| // Initialize with Options that disable fragmentation |
| GenericLinkService::Options options; |
| options.allowFragmentation = false; |
| initialize(options); |
| |
| transport->setMtu(55); |
| |
| shared_ptr<Data> data = makeData("/test/data/123456789/987654321/123456789"); |
| face->sendData(*data); |
| |
| BOOST_CHECK_EQUAL(transport->sentPackets.size(), 0); |
| BOOST_CHECK_EQUAL(service->getCounters().nOutOverMtu, 1); |
| } |
| |
| BOOST_AUTO_TEST_CASE(FragmentationUnlimitedMtu) |
| { |
| // Initialize with Options that enable fragmentation |
| GenericLinkService::Options options; |
| options.allowFragmentation = true; |
| initialize(options); |
| |
| transport->setMtu(MTU_UNLIMITED); |
| |
| shared_ptr<Data> data = makeData("/test/data/123456789/987654321/123456789"); |
| face->sendData(*data); |
| |
| BOOST_CHECK_EQUAL(transport->sentPackets.size(), 1); |
| } |
| |
| BOOST_AUTO_TEST_CASE(FragmentationUnderMtu) |
| { |
| // Initialize with Options that enable fragmentation |
| GenericLinkService::Options options; |
| options.allowFragmentation = true; |
| initialize(options); |
| |
| transport->setMtu(105); |
| |
| shared_ptr<Data> data = makeData("/test/data/123456789/987654321/123456789"); |
| face->sendData(*data); |
| |
| BOOST_CHECK_EQUAL(transport->sentPackets.size(), 1); |
| } |
| |
| BOOST_AUTO_TEST_CASE(FragmentationOverMtu) |
| { |
| // Initialize with Options that enable fragmentation |
| GenericLinkService::Options options; |
| options.allowFragmentation = true; |
| initialize(options); |
| |
| transport->setMtu(60); |
| |
| shared_ptr<Data> data = makeData("/test/data/123456789/987654321/123456789"); |
| face->sendData(*data); |
| |
| BOOST_CHECK_GT(transport->sentPackets.size(), 1); |
| } |
| |
| BOOST_AUTO_TEST_CASE(ReassembleFragments) |
| { |
| // Initialize with Options that enables reassembly |
| GenericLinkService::Options options; |
| options.allowReassembly = true; |
| initialize(options); |
| |
| shared_ptr<Interest> interest = makeInterest( |
| "/mt7P130BHXmtLm5dwaY5dpUM6SWYNN2B05g7y3UhsQuLvDdnTWdNnTeEiLuW3FAbJRSG3tzQ0UfaSEgG9rvYHmsKtgPMag1Hj4Tr"); |
| lp::Packet packet(interest->wireEncode()); |
| |
| // fragment the packet |
| LpFragmenter fragmenter; |
| size_t mtu = 100; |
| bool isOk = false; |
| std::vector<lp::Packet> frags; |
| std::tie(isOk, frags) = fragmenter.fragmentPacket(packet, mtu); |
| BOOST_REQUIRE(isOk); |
| BOOST_CHECK_GT(frags.size(), 1); |
| |
| // receive the fragments |
| for (ssize_t fragIndex = frags.size() - 1; fragIndex >= 0; --fragIndex) { |
| size_t sequence = 1000 + fragIndex; |
| frags[fragIndex].add<lp::SequenceField>(sequence); |
| |
| transport->receivePacket(frags[fragIndex].wireEncode()); |
| |
| if (fragIndex > 0) { |
| BOOST_CHECK(receivedInterests.empty()); |
| BOOST_CHECK_EQUAL(service->getCounters().nReassembling, 1); |
| } |
| else { |
| BOOST_CHECK_EQUAL(receivedInterests.size(), 1); |
| BOOST_CHECK_EQUAL(receivedInterests.back(), *interest); |
| BOOST_CHECK_EQUAL(service->getCounters().nReassembling, 0); |
| } |
| } |
| } |
| |
| BOOST_AUTO_TEST_CASE(ReassemblyDisabledDropFragIndex) |
| { |
| // Initialize with Options that disables reassembly |
| GenericLinkService::Options options; |
| options.allowReassembly = false; |
| initialize(options); |
| |
| shared_ptr<Interest> interest = makeInterest("/IgFe6NvH"); |
| lp::Packet packet(interest->wireEncode()); |
| packet.set<lp::FragIndexField>(140); |
| |
| transport->receivePacket(packet.wireEncode()); |
| |
| BOOST_CHECK_EQUAL(service->getCounters().nInLpInvalid, 0); // not an error |
| BOOST_CHECK(receivedInterests.empty()); |
| } |
| |
| BOOST_AUTO_TEST_CASE(ReassemblyDisabledDropFragCount) |
| { |
| // Initialize with Options that disables reassembly |
| GenericLinkService::Options options; |
| options.allowReassembly = false; |
| initialize(options); |
| |
| shared_ptr<Interest> interest = makeInterest("/SeGmEjvIVX"); |
| lp::Packet packet(interest->wireEncode()); |
| packet.set<lp::FragCountField>(276); |
| |
| transport->receivePacket(packet.wireEncode()); |
| |
| BOOST_CHECK_EQUAL(service->getCounters().nInLpInvalid, 0); // not an error |
| BOOST_CHECK(receivedInterests.empty()); |
| } |
| |
| BOOST_AUTO_TEST_SUITE_END() // Fragmentation |
| |
| BOOST_AUTO_TEST_SUITE(Reliability) |
| |
| BOOST_AUTO_TEST_CASE(SendInterestWithSequence) |
| { |
| // Initialize with Options that enables reliability |
| GenericLinkService::Options options; |
| options.allowLocalFields = false; |
| options.reliabilityOptions.isEnabled = true; |
| initialize(options); |
| |
| shared_ptr<Interest> interest1 = makeInterest("/localhost/test"); |
| |
| face->sendInterest(*interest1); |
| |
| BOOST_CHECK_EQUAL(service->getCounters().nOutInterests, 1); |
| BOOST_REQUIRE_EQUAL(transport->sentPackets.size(), 1); |
| lp::Packet interest1pkt; |
| BOOST_REQUIRE_NO_THROW(interest1pkt.wireDecode(transport->sentPackets.back().packet)); |
| BOOST_CHECK(interest1pkt.has<lp::FragmentField>()); |
| BOOST_CHECK(interest1pkt.has<lp::SequenceField>()); |
| } |
| |
| BOOST_AUTO_TEST_CASE(SendDataWithSequence) |
| { |
| // Initialize with Options that enables reliability |
| GenericLinkService::Options options; |
| options.allowLocalFields = false; |
| options.reliabilityOptions.isEnabled = true; |
| initialize(options); |
| |
| shared_ptr<Data> data1 = makeData("/localhost/test"); |
| |
| face->sendData(*data1); |
| |
| BOOST_CHECK_EQUAL(service->getCounters().nOutData, 1); |
| BOOST_REQUIRE_EQUAL(transport->sentPackets.size(), 1); |
| lp::Packet data1pkt; |
| BOOST_REQUIRE_NO_THROW(data1pkt.wireDecode(transport->sentPackets.back().packet)); |
| BOOST_CHECK(data1pkt.has<lp::FragmentField>()); |
| BOOST_CHECK(data1pkt.has<lp::SequenceField>()); |
| } |
| |
| BOOST_AUTO_TEST_CASE(SendNackWithSequence) |
| { |
| // Initialize with Options that enables reliability |
| GenericLinkService::Options options; |
| options.allowLocalFields = false; |
| options.reliabilityOptions.isEnabled = true; |
| initialize(options); |
| |
| lp::Nack nack1 = makeNack("/localhost/test", 323, lp::NackReason::NO_ROUTE); |
| |
| face->sendNack(nack1); |
| |
| BOOST_CHECK_EQUAL(service->getCounters().nOutNacks, 1); |
| BOOST_REQUIRE_EQUAL(transport->sentPackets.size(), 1); |
| lp::Packet nack1pkt; |
| BOOST_REQUIRE_NO_THROW(nack1pkt.wireDecode(transport->sentPackets.back().packet)); |
| BOOST_CHECK(nack1pkt.has<lp::NackField>()); |
| BOOST_CHECK(nack1pkt.has<lp::FragmentField>()); |
| BOOST_CHECK(nack1pkt.has<lp::SequenceField>()); |
| } |
| |
| BOOST_AUTO_TEST_SUITE_END() // Reliability |
| |
| BOOST_AUTO_TEST_SUITE(LpFields) |
| |
| BOOST_AUTO_TEST_CASE(ReceiveNextHopFaceId) |
| { |
| // Initialize with Options that enables local fields |
| GenericLinkService::Options options; |
| options.allowLocalFields = true; |
| initialize(options); |
| |
| shared_ptr<Interest> interest = makeInterest("/12345678"); |
| lp::Packet packet(interest->wireEncode()); |
| packet.set<lp::NextHopFaceIdField>(1000); |
| |
| transport->receivePacket(packet.wireEncode()); |
| |
| BOOST_REQUIRE_EQUAL(receivedInterests.size(), 1); |
| shared_ptr<lp::NextHopFaceIdTag> tag = receivedInterests.back().getTag<lp::NextHopFaceIdTag>(); |
| BOOST_REQUIRE(tag != nullptr); |
| BOOST_CHECK_EQUAL(*tag, 1000); |
| } |
| |
| BOOST_AUTO_TEST_CASE(ReceiveNextHopFaceIdDisabled) |
| { |
| // Initialize with Options that disables local fields |
| GenericLinkService::Options options; |
| options.allowLocalFields = false; |
| initialize(options); |
| |
| shared_ptr<Interest> interest = makeInterest("/12345678"); |
| lp::Packet packet(interest->wireEncode()); |
| packet.set<lp::NextHopFaceIdField>(1000); |
| |
| transport->receivePacket(packet.wireEncode()); |
| |
| BOOST_CHECK_EQUAL(service->getCounters().nInNetInvalid, 0); // not an error |
| BOOST_CHECK(receivedInterests.empty()); |
| } |
| |
| BOOST_AUTO_TEST_CASE(ReceiveNextHopFaceIdDropData) |
| { |
| // Initialize with Options that enables local fields |
| GenericLinkService::Options options; |
| options.allowLocalFields = true; |
| initialize(options); |
| |
| shared_ptr<Data> data = makeData("/12345678"); |
| lp::Packet packet(data->wireEncode()); |
| packet.set<lp::NextHopFaceIdField>(1000); |
| |
| transport->receivePacket(packet.wireEncode()); |
| |
| BOOST_CHECK_EQUAL(service->getCounters().nInNetInvalid, 1); |
| BOOST_CHECK(receivedData.empty()); |
| } |
| |
| BOOST_AUTO_TEST_CASE(ReceiveNextHopFaceIdDropNack) |
| { |
| // Initialize with Options that enables local fields |
| GenericLinkService::Options options; |
| options.allowLocalFields = true; |
| initialize(options); |
| |
| lp::Nack nack = makeNack("/localhost/test", 123, lp::NackReason::NO_ROUTE); |
| lp::Packet packet; |
| packet.set<lp::FragmentField>(std::make_pair( |
| nack.getInterest().wireEncode().begin(), nack.getInterest().wireEncode().end())); |
| packet.set<lp::NackField>(nack.getHeader()); |
| packet.set<lp::NextHopFaceIdField>(1000); |
| |
| transport->receivePacket(packet.wireEncode()); |
| |
| BOOST_CHECK_EQUAL(service->getCounters().nInNetInvalid, 1); |
| BOOST_CHECK(receivedNacks.empty()); |
| } |
| |
| BOOST_AUTO_TEST_CASE(ReceiveCachePolicy) |
| { |
| // Initialize with Options that disables local fields |
| GenericLinkService::Options options; |
| options.allowLocalFields = false; |
| initialize(options); |
| // CachePolicy is unprivileged and does not require allowLocalFields option. |
| |
| shared_ptr<Data> data = makeData("/12345678"); |
| lp::Packet packet(data->wireEncode()); |
| packet.set<lp::CachePolicyField>(lp::CachePolicy().setPolicy(lp::CachePolicyType::NO_CACHE)); |
| |
| transport->receivePacket(packet.wireEncode()); |
| |
| BOOST_REQUIRE_EQUAL(receivedData.size(), 1); |
| shared_ptr<lp::CachePolicyTag> tag = receivedData.back().getTag<lp::CachePolicyTag>(); |
| BOOST_REQUIRE(tag != nullptr); |
| BOOST_CHECK_EQUAL(tag->get().getPolicy(), lp::CachePolicyType::NO_CACHE); |
| } |
| |
| BOOST_AUTO_TEST_CASE(ReceiveCachePolicyDropInterest) |
| { |
| // Initialize with Options that enables local fields |
| GenericLinkService::Options options; |
| options.allowLocalFields = true; |
| initialize(options); |
| |
| shared_ptr<Interest> interest = makeInterest("/12345678"); |
| lp::Packet packet(interest->wireEncode()); |
| lp::CachePolicy policy; |
| policy.setPolicy(lp::CachePolicyType::NO_CACHE); |
| packet.set<lp::CachePolicyField>(policy); |
| |
| transport->receivePacket(packet.wireEncode()); |
| |
| BOOST_CHECK_EQUAL(service->getCounters().nInNetInvalid, 1); |
| BOOST_CHECK(receivedInterests.empty()); |
| } |
| |
| BOOST_AUTO_TEST_CASE(ReceiveCachePolicyDropNack) |
| { |
| // Initialize with Options that enables local fields |
| GenericLinkService::Options options; |
| options.allowLocalFields = true; |
| initialize(options); |
| |
| lp::Nack nack = makeNack("/localhost/test", 123, lp::NackReason::NO_ROUTE); |
| lp::Packet packet(nack.getInterest().wireEncode()); |
| packet.set<lp::NackField>(nack.getHeader()); |
| lp::CachePolicy policy; |
| policy.setPolicy(lp::CachePolicyType::NO_CACHE); |
| packet.set<lp::CachePolicyField>(policy); |
| |
| transport->receivePacket(packet.wireEncode()); |
| |
| BOOST_CHECK_EQUAL(service->getCounters().nInNetInvalid, 1); |
| BOOST_CHECK(receivedNacks.empty()); |
| } |
| |
| BOOST_AUTO_TEST_CASE(SendIncomingFaceId) |
| { |
| // Initialize with Options that enables local fields |
| GenericLinkService::Options options; |
| options.allowLocalFields = true; |
| initialize(options); |
| |
| shared_ptr<Interest> interest = makeInterest("/12345678"); |
| interest->setTag(make_shared<lp::IncomingFaceIdTag>(1000)); |
| |
| face->sendInterest(*interest); |
| |
| BOOST_REQUIRE_EQUAL(transport->sentPackets.size(), 1); |
| lp::Packet sent(transport->sentPackets.back().packet); |
| BOOST_REQUIRE(sent.has<lp::IncomingFaceIdField>()); |
| BOOST_CHECK_EQUAL(sent.get<lp::IncomingFaceIdField>(), 1000); |
| } |
| |
| BOOST_AUTO_TEST_CASE(SendIncomingFaceIdDisabled) |
| { |
| // Initialize with Options that disables local fields |
| GenericLinkService::Options options; |
| options.allowLocalFields = false; |
| initialize(options); |
| |
| shared_ptr<Interest> interest = makeInterest("/12345678"); |
| interest->setTag(make_shared<lp::IncomingFaceIdTag>(1000)); |
| |
| face->sendInterest(*interest); |
| |
| BOOST_REQUIRE_EQUAL(transport->sentPackets.size(), 1); |
| lp::Packet sent(transport->sentPackets.back().packet); |
| BOOST_CHECK(!sent.has<lp::IncomingFaceIdField>()); |
| } |
| |
| BOOST_AUTO_TEST_CASE(ReceiveIncomingFaceIdIgnoreInterest) |
| { |
| // Initialize with Options that enables local fields |
| GenericLinkService::Options options; |
| options.allowLocalFields = true; |
| initialize(options); |
| |
| shared_ptr<Interest> interest = makeInterest("/12345678"); |
| lp::Packet packet(interest->wireEncode()); |
| packet.set<lp::IncomingFaceIdField>(1000); |
| |
| transport->receivePacket(packet.wireEncode()); |
| |
| BOOST_CHECK_EQUAL(service->getCounters().nInNetInvalid, 0); // not an error |
| BOOST_REQUIRE_EQUAL(receivedInterests.size(), 1); |
| BOOST_CHECK(receivedInterests.back().getTag<lp::IncomingFaceIdTag>() == nullptr); |
| } |
| |
| BOOST_AUTO_TEST_CASE(ReceiveIncomingFaceIdIgnoreData) |
| { |
| // Initialize with Options that enables local fields |
| GenericLinkService::Options options; |
| options.allowLocalFields = true; |
| initialize(options); |
| |
| shared_ptr<Data> data = makeData("/z1megUh9Bj"); |
| lp::Packet packet(data->wireEncode()); |
| packet.set<lp::IncomingFaceIdField>(1000); |
| |
| transport->receivePacket(packet.wireEncode()); |
| |
| BOOST_CHECK_EQUAL(service->getCounters().nInNetInvalid, 0); // not an error |
| BOOST_REQUIRE_EQUAL(receivedData.size(), 1); |
| BOOST_CHECK(receivedData.back().getTag<lp::IncomingFaceIdTag>() == nullptr); |
| } |
| |
| BOOST_AUTO_TEST_CASE(ReceiveIncomingFaceIdIgnoreNack) |
| { |
| // Initialize with Options that enables local fields |
| GenericLinkService::Options options; |
| options.allowLocalFields = true; |
| initialize(options); |
| |
| lp::Nack nack = makeNack("/TPAhdiHz", 278, lp::NackReason::CONGESTION); |
| lp::Packet packet(nack.getInterest().wireEncode()); |
| packet.set<lp::NackField>(nack.getHeader()); |
| packet.set<lp::IncomingFaceIdField>(1000); |
| |
| transport->receivePacket(packet.wireEncode()); |
| |
| BOOST_CHECK_EQUAL(service->getCounters().nInNetInvalid, 0); // not an error |
| BOOST_REQUIRE_EQUAL(receivedNacks.size(), 1); |
| BOOST_CHECK(receivedNacks.back().getTag<lp::IncomingFaceIdTag>() == nullptr); |
| } |
| |
| BOOST_AUTO_TEST_CASE(SendCongestionMarkInterest) |
| { |
| shared_ptr<Interest> interest = makeInterest("/12345678"); |
| interest->setTag(make_shared<lp::CongestionMarkTag>(1)); |
| |
| face->sendInterest(*interest); |
| |
| BOOST_REQUIRE_EQUAL(transport->sentPackets.size(), 1); |
| lp::Packet sent(transport->sentPackets.back().packet); |
| BOOST_REQUIRE(sent.has<lp::CongestionMarkField>()); |
| BOOST_CHECK_EQUAL(sent.get<lp::CongestionMarkField>(), 1); |
| } |
| |
| BOOST_AUTO_TEST_CASE(SendCongestionMarkData) |
| { |
| shared_ptr<Data> data = makeData("/12345678"); |
| data->setTag(make_shared<lp::CongestionMarkTag>(0)); |
| |
| face->sendData(*data); |
| |
| BOOST_REQUIRE_EQUAL(transport->sentPackets.size(), 1); |
| lp::Packet sent(transport->sentPackets.back().packet); |
| BOOST_REQUIRE(sent.has<lp::CongestionMarkField>()); |
| BOOST_CHECK_EQUAL(sent.get<lp::CongestionMarkField>(), 0); |
| } |
| |
| BOOST_AUTO_TEST_CASE(SendCongestionMarkNack) |
| { |
| lp::Nack nack = makeNack("/localhost/test", 123, lp::NackReason::NO_ROUTE); |
| nack.setTag(make_shared<lp::CongestionMarkTag>(std::numeric_limits<uint64_t>::max())); |
| |
| face->sendNack(nack); |
| |
| BOOST_REQUIRE_EQUAL(transport->sentPackets.size(), 1); |
| lp::Packet sent(transport->sentPackets.back().packet); |
| BOOST_REQUIRE(sent.has<lp::CongestionMarkField>()); |
| BOOST_CHECK_EQUAL(sent.get<lp::CongestionMarkField>(), std::numeric_limits<uint64_t>::max()); |
| } |
| |
| BOOST_AUTO_TEST_CASE(ReceiveCongestionMarkInterest) |
| { |
| shared_ptr<Interest> interest = makeInterest("/12345678"); |
| lp::Packet packet(interest->wireEncode()); |
| packet.set<lp::CongestionMarkField>(1); |
| |
| transport->receivePacket(packet.wireEncode()); |
| |
| BOOST_REQUIRE_EQUAL(receivedInterests.size(), 1); |
| shared_ptr<lp::CongestionMarkTag> tag = receivedInterests.back().getTag<lp::CongestionMarkTag>(); |
| BOOST_REQUIRE(tag != nullptr); |
| BOOST_CHECK_EQUAL(*tag, 1); |
| } |
| |
| BOOST_AUTO_TEST_CASE(ReceiveCongestionMarkData) |
| { |
| shared_ptr<Data> data = makeData("/12345678"); |
| lp::Packet packet(data->wireEncode()); |
| packet.set<lp::CongestionMarkField>(1); |
| |
| transport->receivePacket(packet.wireEncode()); |
| |
| BOOST_REQUIRE_EQUAL(receivedData.size(), 1); |
| shared_ptr<lp::CongestionMarkTag> tag = receivedData.back().getTag<lp::CongestionMarkTag>(); |
| BOOST_REQUIRE(tag != nullptr); |
| BOOST_CHECK_EQUAL(*tag, 1); |
| } |
| |
| BOOST_AUTO_TEST_CASE(ReceiveCongestionMarkNack) |
| { |
| lp::Nack nack = makeNack("/localhost/test", 123, lp::NackReason::NO_ROUTE); |
| lp::Packet packet; |
| packet.set<lp::FragmentField>(std::make_pair( |
| nack.getInterest().wireEncode().begin(), nack.getInterest().wireEncode().end())); |
| packet.set<lp::NackField>(nack.getHeader()); |
| packet.set<lp::CongestionMarkField>(1); |
| |
| transport->receivePacket(packet.wireEncode()); |
| |
| BOOST_REQUIRE_EQUAL(receivedNacks.size(), 1); |
| shared_ptr<lp::CongestionMarkTag> tag = receivedNacks.back().getTag<lp::CongestionMarkTag>(); |
| BOOST_REQUIRE(tag != nullptr); |
| BOOST_CHECK_EQUAL(*tag, 1); |
| } |
| |
| BOOST_AUTO_TEST_SUITE_END() // LpFields |
| |
| |
| BOOST_AUTO_TEST_SUITE(Malformed) // receive malformed packets |
| |
| BOOST_AUTO_TEST_CASE(WrongTlvType) |
| { |
| // Initialize with Options that disables all services |
| GenericLinkService::Options options; |
| options.allowLocalFields = false; |
| initialize(options); |
| |
| Block packet = ndn::encoding::makeEmptyBlock(tlv::Name); |
| |
| BOOST_CHECK_NO_THROW(transport->receivePacket(packet)); |
| |
| BOOST_CHECK_EQUAL(service->getCounters().nInLpInvalid, 1); |
| BOOST_CHECK_EQUAL(receivedInterests.size(), 0); |
| BOOST_CHECK_EQUAL(receivedData.size(), 0); |
| BOOST_CHECK_EQUAL(receivedNacks.size(), 0); |
| } |
| |
| BOOST_AUTO_TEST_CASE(Unparsable) |
| { |
| // Initialize with Options that disables all services |
| GenericLinkService::Options options; |
| options.allowLocalFields = false; |
| initialize(options); |
| |
| Block packet = ndn::encoding::makeStringBlock(lp::tlv::LpPacket, "x"); |
| BOOST_CHECK_THROW(packet.parse(), tlv::Error); |
| |
| BOOST_CHECK_NO_THROW(transport->receivePacket(packet)); |
| |
| BOOST_CHECK_EQUAL(service->getCounters().nInLpInvalid, 1); |
| BOOST_CHECK_EQUAL(receivedInterests.size(), 0); |
| BOOST_CHECK_EQUAL(receivedData.size(), 0); |
| BOOST_CHECK_EQUAL(receivedNacks.size(), 0); |
| } |
| |
| BOOST_AUTO_TEST_SUITE_END() // Malformed |
| |
| |
| BOOST_AUTO_TEST_SUITE_END() // TestGenericLinkService |
| BOOST_AUTO_TEST_SUITE_END() // Face |
| |
| } // namespace tests |
| } // namespace face |
| } // namespace nfd |