blob: cc40905c6fa07a1f3e9d51d64d91f0b4aeced81d [file] [log] [blame]
Junxiao Shi727ed292014-02-19 23:26:45 -07001/* -*- Mode:C++; c-file-style:"gnu"; indent-tabs-mode:nil; -*- */
ashiqopud3ae85d2019-02-17 02:29:55 +00002/*
Teng Liangebc20f62020-06-23 16:55:20 -07003 * Copyright (c) 2014-2020, Regents of the University of California,
Spyridon Mastorakisd0381c02015-02-19 10:29:41 -08004 * Arizona Board of Regents,
5 * Colorado State University,
6 * University Pierre & Marie Curie, Sorbonne University,
7 * Washington University in St. Louis,
8 * Beijing Institute of Technology,
9 * The University of Memphis.
Alexander Afanasyev9bcbc7c2014-04-06 19:37:37 -070010 *
11 * This file is part of NFD (Named Data Networking Forwarding Daemon).
12 * See AUTHORS.md for complete list of NFD authors and contributors.
13 *
14 * NFD is free software: you can redistribute it and/or modify it under the terms
15 * of the GNU General Public License as published by the Free Software Foundation,
16 * either version 3 of the License, or (at your option) any later version.
17 *
18 * NFD is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY;
19 * without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR
20 * PURPOSE. See the GNU General Public License for more details.
21 *
22 * You should have received a copy of the GNU General Public License along with
23 * NFD, e.g., in COPYING.md file. If not, see <http://www.gnu.org/licenses/>.
Junxiao Shi82e7f582014-09-07 15:15:40 -070024 */
Junxiao Shi727ed292014-02-19 23:26:45 -070025
Davide Pesavento97210d52016-10-14 15:45:48 +020026#ifndef NFD_TESTS_DAEMON_FW_STRATEGY_TESTER_HPP
27#define NFD_TESTS_DAEMON_FW_STRATEGY_TESTER_HPP
Junxiao Shi727ed292014-02-19 23:26:45 -070028
Junxiao Shi727ed292014-02-19 23:26:45 -070029#include "fw/strategy.hpp"
Davide Pesavento3dade002019-03-19 11:29:56 -060030
31#include "tests/daemon/limited-io.hpp"
Junxiao Shi727ed292014-02-19 23:26:45 -070032
33namespace nfd {
Spyridon Mastorakisd0381c02015-02-19 10:29:41 -080034namespace fw {
Junxiao Shid9ee45c2014-02-27 15:38:11 -070035namespace tests {
Junxiao Shi727ed292014-02-19 23:26:45 -070036
Davide Pesavento14e71f02019-03-28 17:35:25 -040037using namespace nfd::tests;
38
39/** \brief Extends strategy S for unit testing.
Junxiao Shi727ed292014-02-19 23:26:45 -070040 *
Davide Pesavento14e71f02019-03-28 17:35:25 -040041 * Actions invoked by S are recorded but not passed to forwarder.
Junxiao Shi890afe92016-12-15 14:34:34 +000042 *
43 * StrategyTester should be registered into the strategy registry prior to use.
44 * \code
45 * // appears in or included by every .cpp MyStrategyTester is used
Davide Pesavento14e71f02019-03-28 17:35:25 -040046 * using MyStrategyTester = StrategyTester<MyStrategy>;
Junxiao Shi890afe92016-12-15 14:34:34 +000047 *
48 * // appears in only one .cpp
49 * NFD_REGISTER_STRATEGY(MyStrategyTester);
50 * \endcode
Junxiao Shi727ed292014-02-19 23:26:45 -070051 */
52template<typename S>
53class StrategyTester : public S
54{
55public:
56 explicit
Junxiao Shi890afe92016-12-15 14:34:34 +000057 StrategyTester(Forwarder& forwarder, const Name& name = getStrategyName())
58 : S(forwarder, name)
Junxiao Shi727ed292014-02-19 23:26:45 -070059 {
60 }
Alexander Afanasyev9bcbc7c2014-04-06 19:37:37 -070061
Junxiao Shi890afe92016-12-15 14:34:34 +000062 static Name
63 getStrategyName()
64 {
65 Name name = S::getStrategyName();
66 if (!name.empty() && name[-1].isVersion()) {
67 // insert "tester" before version component
68 name::Component versionComp = name[-1];
69 name = name.getPrefix(-1);
70 name.append("tester");
71 name.append(versionComp);
72 }
73 else {
74 name.append("tester");
75 }
76 return name;
77 }
78
79 /** \brief signal emitted after each Action
80 */
Junxiao Shi5e5e4452015-09-24 16:56:52 -070081 signal::Signal<StrategyTester<S>> afterAction;
Junxiao Shi727ed292014-02-19 23:26:45 -070082
Junxiao Shi99540072017-01-27 19:57:33 +000083 /** \brief execute f and wait for a number of strategy actions
84 * \note The actions may occur either during f() invocation or afterwards.
85 * \return whether expected number of actions have occurred
86 */
Davide Pesavento14e71f02019-03-28 17:35:25 -040087 template<typename F>
Junxiao Shi99540072017-01-27 19:57:33 +000088 bool
Davide Pesavento14e71f02019-03-28 17:35:25 -040089 waitForAction(F&& f, LimitedIo& limitedIo, int nExpectedActions = 1)
Junxiao Shi99540072017-01-27 19:57:33 +000090 {
91 int nActions = 0;
92
93 signal::ScopedConnection conn = afterAction.connect([&] {
94 limitedIo.afterOp();
95 ++nActions;
96 });
97
98 f();
99
100 if (nActions < nExpectedActions) {
101 // A correctly implemented strategy is required to invoke reject pending Interest action if it
102 // decides to not forward an Interest. If a test case is stuck in the call below, check that
103 // rejectPendingInterest is invoked under proper condition.
Davide Pesavento14e71f02019-03-28 17:35:25 -0400104 return limitedIo.run(nExpectedActions - nActions, LimitedIo::UNLIMITED_TIME) == LimitedIo::EXCEED_OPS;
Junxiao Shi99540072017-01-27 19:57:33 +0000105 }
106 return nActions == nExpectedActions;
107 }
108
Junxiao Shi727ed292014-02-19 23:26:45 -0700109protected:
Eric Newberry2377ada2020-09-28 22:40:14 -0700110 pit::OutRecord*
Teng Liangebc20f62020-06-23 16:55:20 -0700111 sendInterest(const shared_ptr<pit::Entry>& pitEntry, Face& egress,
Junxiao Shic5f651f2016-11-17 22:58:12 +0000112 const Interest& interest) override
Junxiao Shib9420cf2016-08-13 04:38:52 +0000113 {
Teng Liangebc20f62020-06-23 16:55:20 -0700114 sendInterestHistory.push_back({pitEntry->getInterest(), egress.getId(), interest});
Eric Newberry2377ada2020-09-28 22:40:14 -0700115 auto it = pitEntry->insertOrUpdateOutRecord(egress, interest);
116 BOOST_ASSERT(it != pitEntry->out_end());
Junxiao Shib9420cf2016-08-13 04:38:52 +0000117 afterAction();
Eric Newberry2377ada2020-09-28 22:40:14 -0700118 return &*it;
Junxiao Shib9420cf2016-08-13 04:38:52 +0000119 }
Junxiao Shi727ed292014-02-19 23:26:45 -0700120
Junxiao Shi890afe92016-12-15 14:34:34 +0000121 void
Junxiao Shib9420cf2016-08-13 04:38:52 +0000122 rejectPendingInterest(const shared_ptr<pit::Entry>& pitEntry) override
123 {
Junxiao Shi8ff0a862016-08-13 04:50:50 +0000124 rejectPendingInterestHistory.push_back({pitEntry->getInterest()});
Junxiao Shib9420cf2016-08-13 04:38:52 +0000125 afterAction();
126 }
Junxiao Shi727ed292014-02-19 23:26:45 -0700127
Eric Newberry2377ada2020-09-28 22:40:14 -0700128 bool
Teng Liangebc20f62020-06-23 16:55:20 -0700129 sendNack(const shared_ptr<pit::Entry>& pitEntry, Face& egress,
Junxiao Shib9420cf2016-08-13 04:38:52 +0000130 const lp::NackHeader& header) override
131 {
Teng Liangebc20f62020-06-23 16:55:20 -0700132 sendNackHistory.push_back({pitEntry->getInterest(), egress.getId(), header});
133 pitEntry->deleteInRecord(egress);
Junxiao Shib9420cf2016-08-13 04:38:52 +0000134 afterAction();
Eric Newberry2377ada2020-09-28 22:40:14 -0700135 return true;
Junxiao Shib9420cf2016-08-13 04:38:52 +0000136 }
Junxiao Shi727ed292014-02-19 23:26:45 -0700137
Junxiao Shi5e5e4452015-09-24 16:56:52 -0700138public:
139 struct SendInterestArgs
140 {
Junxiao Shi8ff0a862016-08-13 04:50:50 +0000141 Interest pitInterest;
Junxiao Shi5e5e4452015-09-24 16:56:52 -0700142 FaceId outFaceId;
Junxiao Shic5f651f2016-11-17 22:58:12 +0000143 Interest interest;
Junxiao Shi5e5e4452015-09-24 16:56:52 -0700144 };
145 std::vector<SendInterestArgs> sendInterestHistory;
146
147 struct RejectPendingInterestArgs
148 {
Junxiao Shi8ff0a862016-08-13 04:50:50 +0000149 Interest pitInterest;
Junxiao Shi5e5e4452015-09-24 16:56:52 -0700150 };
151 std::vector<RejectPendingInterestArgs> rejectPendingInterestHistory;
152
153 struct SendNackArgs
154 {
Junxiao Shi8ff0a862016-08-13 04:50:50 +0000155 Interest pitInterest;
Junxiao Shi5e5e4452015-09-24 16:56:52 -0700156 FaceId outFaceId;
157 lp::NackHeader header;
158 };
159 std::vector<SendNackArgs> sendNackHistory;
Junxiao Shi727ed292014-02-19 23:26:45 -0700160};
161
Junxiao Shid9ee45c2014-02-27 15:38:11 -0700162} // namespace tests
Spyridon Mastorakisd0381c02015-02-19 10:29:41 -0800163} // namespace fw
Junxiao Shi727ed292014-02-19 23:26:45 -0700164} // namespace nfd
165
Davide Pesavento97210d52016-10-14 15:45:48 +0200166#endif // NFD_TESTS_DAEMON_FW_STRATEGY_TESTER_HPP