blob: dfb8768c98f534933259c0ae4d6bf7ce1e5b08ab [file] [log] [blame]
Zhuo Lib3558892016-08-12 15:51:12 -07001/* -*- Mode:C++; c-file-style:"gnu"; indent-tabs-mode:nil; -*- */
Junxiao Shib54aabd2018-04-16 19:36:24 +00002/*
Junxiao Shifd2d1012022-01-11 18:20:38 +00003 * Copyright (c) 2014-2022, Arizona Board of Regents.
Zhuo Lib3558892016-08-12 15:51:12 -07004 *
5 * This file is part of ndn-tools (Named Data Networking Essential Tools).
6 * See AUTHORS.md for complete list of ndn-tools authors and contributors.
7 *
8 * ndn-tools is free software: you can redistribute it and/or modify it under the terms
9 * of the GNU General Public License as published by the Free Software Foundation,
10 * either version 3 of the License, or (at your option) any later version.
11 *
12 * ndn-tools is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY;
13 * without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR
14 * PURPOSE. See the GNU General Public License for more details.
15 *
16 * You should have received a copy of the GNU General Public License along with
17 * ndn-tools, e.g., in COPYING.md file. If not, see <http://www.gnu.org/licenses/>.
18 */
19
20#include "tools/peek/ndnpeek/ndnpeek.hpp"
21
22#include "tests/test-common.hpp"
Davide Pesavento66777622020-10-09 18:46:03 -040023#include "tests/io-fixture.hpp"
Davide Pesaventobf28cd72017-08-13 17:22:47 -040024
Zhuo Lib3558892016-08-12 15:51:12 -070025#include <ndn-cxx/util/dummy-client-face.hpp>
26
27#include <boost/mpl/vector.hpp>
Davide Pesaventoa8600b02019-12-22 18:52:36 -050028#if BOOST_VERSION >= 105900
29#include <boost/test/tools/output_test_stream.hpp>
30#else
31#include <boost/test/output_test_stream.hpp>
32#endif
Zhuo Lib3558892016-08-12 15:51:12 -070033
34namespace ndn {
35namespace peek {
36namespace tests {
37
38using namespace ndn::tests;
39using boost::test_tools::output_test_stream;
40
41class CoutRedirector : noncopyable
42{
43public:
44 explicit
45 CoutRedirector(std::ostream& destination)
46 {
47 m_originalBuf = std::cout.rdbuf(destination.rdbuf());
48 }
49
50 ~CoutRedirector()
51 {
52 std::cout.rdbuf(m_originalBuf);
53 }
54
55private:
56 std::streambuf* m_originalBuf;
57};
58
Davide Pesavento96028232019-08-17 01:26:13 -040059static PeekOptions
60makeDefaultOptions()
61{
62 PeekOptions opt;
63 opt.name = "/peek/test";
64 return opt;
65}
66
Davide Pesavento66777622020-10-09 18:46:03 -040067class NdnPeekFixture : public IoFixture
Zhuo Lib3558892016-08-12 15:51:12 -070068{
69protected:
Zhuo Lib3558892016-08-12 15:51:12 -070070 void
Davide Pesavento96028232019-08-17 01:26:13 -040071 initialize(const PeekOptions& opts = makeDefaultOptions())
Zhuo Lib3558892016-08-12 15:51:12 -070072 {
73 peek = make_unique<NdnPeek>(face, opts);
74 }
75
76protected:
Davide Pesavento66777622020-10-09 18:46:03 -040077 ndn::util::DummyClientFace face{m_io};
Zhuo Lib3558892016-08-12 15:51:12 -070078 output_test_stream output;
79 unique_ptr<NdnPeek> peek;
80};
81
Zhuo Lib3558892016-08-12 15:51:12 -070082class OutputFull
83{
84public:
85 static PeekOptions
86 makeOptions()
87 {
88 return makeDefaultOptions();
89 }
90
91 static void
92 checkOutput(output_test_stream& output, const Data& data)
93 {
94 const Block& block = data.wireEncode();
95 std::string expected(reinterpret_cast<const char*>(block.wire()), block.size());
96 BOOST_CHECK(output.is_equal(expected));
97 }
98
99 static void
100 checkOutput(output_test_stream& output, const lp::Nack& nack)
101 {
102 const Block& block = nack.getHeader().wireEncode();
103 std::string expected(reinterpret_cast<const char*>(block.wire()), block.size());
104 BOOST_CHECK(output.is_equal(expected));
105 }
106};
107
108class OutputPayloadOnly
109{
110public:
111 static PeekOptions
112 makeOptions()
113 {
114 PeekOptions opt = makeDefaultOptions();
115 opt.wantPayloadOnly = true;
116 return opt;
117 }
118
119 static void
120 checkOutput(output_test_stream& output, const Data& data)
121 {
122 const Block& block = data.getContent();
123 std::string expected(reinterpret_cast<const char*>(block.value()), block.value_size());
124 BOOST_CHECK(output.is_equal(expected));
125 }
126
127 static void
128 checkOutput(output_test_stream& output, const lp::Nack& nack)
129 {
130 std::string expected = boost::lexical_cast<std::string>(nack.getReason()) + '\n';
131 BOOST_CHECK(output.is_equal(expected));
132 }
133};
134
135BOOST_AUTO_TEST_SUITE(Peek)
136BOOST_FIXTURE_TEST_SUITE(TestNdnPeek, NdnPeekFixture)
137
138using OutputChecks = boost::mpl::vector<OutputFull, OutputPayloadOnly>;
139
140BOOST_AUTO_TEST_CASE_TEMPLATE(Default, OutputCheck, OutputChecks)
141{
142 auto options = OutputCheck::makeOptions();
143 initialize(options);
144
Junxiao Shib54aabd2018-04-16 19:36:24 +0000145 auto data = makeData(options.name);
Davide Pesavento59984282022-02-16 22:41:03 -0500146 data->setContent({'n', 'd', 'n', 'p', 'e', 'e', 'k'});
Zhuo Lib3558892016-08-12 15:51:12 -0700147
148 {
149 CoutRedirector redir(output);
150 peek->start();
Davide Pesavento66777622020-10-09 18:46:03 -0400151 this->advanceClocks(25_ms, 4);
Zhuo Lib3558892016-08-12 15:51:12 -0700152 face.receive(*data);
153 }
154
155 OutputCheck::checkOutput(output, *data);
156 BOOST_REQUIRE_EQUAL(face.sentInterests.size(), 1);
Junxiao Shifd2d1012022-01-11 18:20:38 +0000157 const auto& interest = face.sentInterests.back();
158 BOOST_CHECK_EQUAL(interest.getCanBePrefix(), false);
159 BOOST_CHECK_EQUAL(interest.getMustBeFresh(), false);
160 BOOST_CHECK_EQUAL(interest.getForwardingHint().empty(), true);
161 BOOST_CHECK_EQUAL(interest.getInterestLifetime(), DEFAULT_INTEREST_LIFETIME);
162 BOOST_CHECK(interest.getHopLimit() == nullopt);
163 BOOST_CHECK(!interest.hasApplicationParameters());
Davide Pesavento87434be2019-07-25 19:04:23 -0400164 BOOST_CHECK(peek->getResult() == NdnPeek::Result::DATA);
Zhuo Lib3558892016-08-12 15:51:12 -0700165}
166
Junxiao Shib54aabd2018-04-16 19:36:24 +0000167BOOST_AUTO_TEST_CASE_TEMPLATE(NonDefault, OutputCheck, OutputChecks)
Zhuo Lib3558892016-08-12 15:51:12 -0700168{
169 auto options = OutputCheck::makeOptions();
Junxiao Shib54aabd2018-04-16 19:36:24 +0000170 options.canBePrefix = true;
Zhuo Lib3558892016-08-12 15:51:12 -0700171 options.mustBeFresh = true;
Junxiao Shifd2d1012022-01-11 18:20:38 +0000172 options.forwardingHint.emplace_back("/fh");
Junxiao Shib54aabd2018-04-16 19:36:24 +0000173 options.interestLifetime = 200_ms;
Davide Pesaventoc214e072019-08-17 01:33:28 -0400174 options.hopLimit = 64;
Zhuo Lib3558892016-08-12 15:51:12 -0700175 initialize(options);
176
Junxiao Shib54aabd2018-04-16 19:36:24 +0000177 auto data = makeData(Name(options.name).append("suffix"));
Junxiao Shi96192952019-05-22 15:45:12 +0000178 data->setFreshnessPeriod(1_s);
Davide Pesavento59984282022-02-16 22:41:03 -0500179 data->setContent({'n', 'd', 'n', 'p', 'e', 'e', 'k'});
Zhuo Lib3558892016-08-12 15:51:12 -0700180
181 {
182 CoutRedirector redir(output);
183 peek->start();
Davide Pesavento66777622020-10-09 18:46:03 -0400184 this->advanceClocks(25_ms, 4);
Zhuo Lib3558892016-08-12 15:51:12 -0700185 face.receive(*data);
186 }
187
188 OutputCheck::checkOutput(output, *data);
189 BOOST_REQUIRE_EQUAL(face.sentInterests.size(), 1);
Junxiao Shifd2d1012022-01-11 18:20:38 +0000190 const auto& interest = face.sentInterests.back();
191 BOOST_CHECK_EQUAL(interest.getCanBePrefix(), true);
192 BOOST_CHECK_EQUAL(interest.getMustBeFresh(), true);
193 BOOST_TEST(interest.getForwardingHint() == std::vector<Name>({"/fh"}),
194 boost::test_tools::per_element());
195 BOOST_CHECK_EQUAL(interest.getInterestLifetime(), 200_ms);
196 BOOST_CHECK(interest.getHopLimit() == 64);
197 BOOST_CHECK(!interest.hasApplicationParameters());
Davide Pesavento87434be2019-07-25 19:04:23 -0400198 BOOST_CHECK(peek->getResult() == NdnPeek::Result::DATA);
Zhuo Lib3558892016-08-12 15:51:12 -0700199}
200
201BOOST_AUTO_TEST_CASE_TEMPLATE(ReceiveNackWithReason, OutputCheck, OutputChecks)
202{
203 auto options = OutputCheck::makeOptions();
204 initialize(options);
205 lp::Nack nack;
206
207 {
208 CoutRedirector redir(output);
209 peek->start();
Davide Pesavento66777622020-10-09 18:46:03 -0400210 this->advanceClocks(25_ms, 4);
Zhuo Lib3558892016-08-12 15:51:12 -0700211 nack = makeNack(face.sentInterests.at(0), lp::NackReason::NO_ROUTE);
212 face.receive(nack);
213 }
214
215 OutputCheck::checkOutput(output, nack);
216 BOOST_CHECK_EQUAL(face.sentInterests.size(), 1);
Davide Pesavento87434be2019-07-25 19:04:23 -0400217 BOOST_CHECK(peek->getResult() == NdnPeek::Result::NACK);
Zhuo Lib3558892016-08-12 15:51:12 -0700218}
219
220BOOST_AUTO_TEST_CASE_TEMPLATE(ReceiveNackWithoutReason, OutputCheck, OutputChecks)
221{
222 auto options = OutputCheck::makeOptions();
223 initialize(options);
224 lp::Nack nack;
225
226 {
227 CoutRedirector redir(output);
228 peek->start();
Davide Pesavento66777622020-10-09 18:46:03 -0400229 this->advanceClocks(25_ms, 4);
Zhuo Lib3558892016-08-12 15:51:12 -0700230 nack = makeNack(face.sentInterests.at(0), lp::NackReason::NONE);
231 face.receive(nack);
232 }
233
234 OutputCheck::checkOutput(output, nack);
235 BOOST_CHECK_EQUAL(face.sentInterests.size(), 1);
Davide Pesavento87434be2019-07-25 19:04:23 -0400236 BOOST_CHECK(peek->getResult() == NdnPeek::Result::NACK);
Zhuo Lib3558892016-08-12 15:51:12 -0700237}
238
Davide Pesavento96028232019-08-17 01:26:13 -0400239BOOST_AUTO_TEST_CASE(ApplicationParameters)
240{
241 auto options = makeDefaultOptions();
242 options.applicationParameters = make_shared<Buffer>("hello", 5);
243 initialize(options);
244
245 peek->start();
Davide Pesavento66777622020-10-09 18:46:03 -0400246 this->advanceClocks(25_ms, 4);
Davide Pesavento96028232019-08-17 01:26:13 -0400247
248 BOOST_REQUIRE_EQUAL(face.sentInterests.size(), 1);
249 BOOST_CHECK_EQUAL(face.sentInterests.back().getCanBePrefix(), false);
250 BOOST_CHECK_EQUAL(face.sentInterests.back().getMustBeFresh(), false);
251 BOOST_CHECK_EQUAL(face.sentInterests.back().getForwardingHint().empty(), true);
252 BOOST_CHECK_EQUAL(face.sentInterests.back().getInterestLifetime(), DEFAULT_INTEREST_LIFETIME);
253 BOOST_CHECK_EQUAL(face.sentInterests.back().hasApplicationParameters(), true);
254 BOOST_CHECK_EQUAL(face.sentInterests.back().getApplicationParameters(), "2405 68656C6C6F"_block);
255}
256
Davide Pesavento65d11552019-06-09 19:15:50 -0400257BOOST_AUTO_TEST_CASE(NoTimeout)
Zhuo Lib3558892016-08-12 15:51:12 -0700258{
259 auto options = makeDefaultOptions();
Davide Pesavento65d11552019-06-09 19:15:50 -0400260 options.interestLifetime = 1_s;
261 options.timeout = nullopt;
Zhuo Lib3558892016-08-12 15:51:12 -0700262 initialize(options);
263
Davide Pesavento65d11552019-06-09 19:15:50 -0400264 BOOST_CHECK_EQUAL(face.sentInterests.size(), 0);
Zhuo Lib3558892016-08-12 15:51:12 -0700265
266 peek->start();
Davide Pesavento66777622020-10-09 18:46:03 -0400267 this->advanceClocks(100_ms, 9);
Zhuo Lib3558892016-08-12 15:51:12 -0700268 BOOST_CHECK_EQUAL(face.sentInterests.size(), 1);
Davide Pesavento65d11552019-06-09 19:15:50 -0400269 BOOST_CHECK_EQUAL(face.getNPendingInterests(), 1);
Davide Pesavento87434be2019-07-25 19:04:23 -0400270 BOOST_CHECK(peek->getResult() == NdnPeek::Result::UNKNOWN);
Davide Pesavento65d11552019-06-09 19:15:50 -0400271
Davide Pesavento66777622020-10-09 18:46:03 -0400272 this->advanceClocks(100_ms, 2);
Davide Pesavento65d11552019-06-09 19:15:50 -0400273 BOOST_CHECK_EQUAL(face.sentInterests.size(), 1);
274 BOOST_CHECK_EQUAL(face.getNPendingInterests(), 0);
Davide Pesavento87434be2019-07-25 19:04:23 -0400275 BOOST_CHECK(peek->getResult() == NdnPeek::Result::TIMEOUT);
Zhuo Lib3558892016-08-12 15:51:12 -0700276}
277
278BOOST_AUTO_TEST_CASE(TimeoutLessThanLifetime)
279{
280 auto options = makeDefaultOptions();
Junxiao Shib54aabd2018-04-16 19:36:24 +0000281 options.interestLifetime = 200_ms;
282 options.timeout = 100_ms;
Zhuo Lib3558892016-08-12 15:51:12 -0700283 initialize(options);
284
Davide Pesavento65d11552019-06-09 19:15:50 -0400285 BOOST_CHECK_EQUAL(face.sentInterests.size(), 0);
Zhuo Lib3558892016-08-12 15:51:12 -0700286
287 peek->start();
Davide Pesavento66777622020-10-09 18:46:03 -0400288 this->advanceClocks(25_ms, 6);
Zhuo Lib3558892016-08-12 15:51:12 -0700289
290 BOOST_CHECK_EQUAL(face.sentInterests.size(), 1);
Davide Pesavento65d11552019-06-09 19:15:50 -0400291 BOOST_CHECK_EQUAL(face.getNPendingInterests(), 0);
Davide Pesavento87434be2019-07-25 19:04:23 -0400292 BOOST_CHECK(peek->getResult() == NdnPeek::Result::TIMEOUT);
Zhuo Lib3558892016-08-12 15:51:12 -0700293}
294
295BOOST_AUTO_TEST_CASE(TimeoutGreaterThanLifetime)
296{
297 auto options = makeDefaultOptions();
Junxiao Shib54aabd2018-04-16 19:36:24 +0000298 options.interestLifetime = 50_ms;
299 options.timeout = 200_ms;
Zhuo Lib3558892016-08-12 15:51:12 -0700300 initialize(options);
301
Davide Pesavento65d11552019-06-09 19:15:50 -0400302 BOOST_CHECK_EQUAL(face.sentInterests.size(), 0);
Zhuo Lib3558892016-08-12 15:51:12 -0700303
304 peek->start();
Davide Pesavento66777622020-10-09 18:46:03 -0400305 this->advanceClocks(25_ms, 4);
Zhuo Lib3558892016-08-12 15:51:12 -0700306
307 BOOST_CHECK_EQUAL(face.sentInterests.size(), 1);
Davide Pesavento65d11552019-06-09 19:15:50 -0400308 BOOST_CHECK_EQUAL(face.getNPendingInterests(), 0);
Davide Pesavento87434be2019-07-25 19:04:23 -0400309 BOOST_CHECK(peek->getResult() == NdnPeek::Result::TIMEOUT);
Zhuo Lib3558892016-08-12 15:51:12 -0700310}
311
Davide Pesavento96028232019-08-17 01:26:13 -0400312BOOST_AUTO_TEST_CASE(OversizedPacket)
313{
314 auto options = makeDefaultOptions();
315 options.applicationParameters = make_shared<Buffer>(MAX_NDN_PACKET_SIZE);
316 initialize(options);
317
318 peek->start();
Davide Pesavento66777622020-10-09 18:46:03 -0400319 BOOST_CHECK_THROW(this->advanceClocks(1_ms, 10), Face::OversizedPacketError);
Davide Pesavento96028232019-08-17 01:26:13 -0400320
321 BOOST_CHECK_EQUAL(face.sentInterests.size(), 0);
322}
323
Zhuo Lib3558892016-08-12 15:51:12 -0700324BOOST_AUTO_TEST_SUITE_END() // TestNdnPeek
325BOOST_AUTO_TEST_SUITE_END() // Peek
326
327} // namespace tests
328} // namespace peek
329} // namespace ndn