blob: 3ddf8f6605697d4c161e77a8f75d70a4515a8789 [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/*
3 * Copyright (c) 2014-2019, 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:
Junxiao Shi890afe92016-12-15 14:34:34 +0000110 void
ashiqopuc7079482019-02-20 05:34:37 +0000111 sendInterest(const shared_ptr<pit::Entry>& pitEntry, const FaceEndpoint& egress,
Junxiao Shic5f651f2016-11-17 22:58:12 +0000112 const Interest& interest) override
Junxiao Shib9420cf2016-08-13 04:38:52 +0000113 {
ashiqopuc7079482019-02-20 05:34:37 +0000114 sendInterestHistory.push_back({pitEntry->getInterest(), egress.face.getId(), interest});
115 pitEntry->insertOrUpdateOutRecord(egress.face, egress.endpoint, interest);
Junxiao Shib9420cf2016-08-13 04:38:52 +0000116 afterAction();
117 }
Junxiao Shi727ed292014-02-19 23:26:45 -0700118
Junxiao Shi890afe92016-12-15 14:34:34 +0000119 void
Junxiao Shib9420cf2016-08-13 04:38:52 +0000120 rejectPendingInterest(const shared_ptr<pit::Entry>& pitEntry) override
121 {
Junxiao Shi8ff0a862016-08-13 04:50:50 +0000122 rejectPendingInterestHistory.push_back({pitEntry->getInterest()});
Junxiao Shib9420cf2016-08-13 04:38:52 +0000123 afterAction();
124 }
Junxiao Shi727ed292014-02-19 23:26:45 -0700125
Junxiao Shi890afe92016-12-15 14:34:34 +0000126 void
ashiqopuc7079482019-02-20 05:34:37 +0000127 sendNack(const shared_ptr<pit::Entry>& pitEntry, const FaceEndpoint& egress,
Junxiao Shib9420cf2016-08-13 04:38:52 +0000128 const lp::NackHeader& header) override
129 {
ashiqopuc7079482019-02-20 05:34:37 +0000130 sendNackHistory.push_back({pitEntry->getInterest(), egress.face.getId(), header});
131 pitEntry->deleteInRecord(egress.face, egress.endpoint);
Junxiao Shib9420cf2016-08-13 04:38:52 +0000132 afterAction();
133 }
Junxiao Shi727ed292014-02-19 23:26:45 -0700134
Junxiao Shi5e5e4452015-09-24 16:56:52 -0700135public:
136 struct SendInterestArgs
137 {
Junxiao Shi8ff0a862016-08-13 04:50:50 +0000138 Interest pitInterest;
Junxiao Shi5e5e4452015-09-24 16:56:52 -0700139 FaceId outFaceId;
Junxiao Shic5f651f2016-11-17 22:58:12 +0000140 Interest interest;
Junxiao Shi5e5e4452015-09-24 16:56:52 -0700141 };
142 std::vector<SendInterestArgs> sendInterestHistory;
143
144 struct RejectPendingInterestArgs
145 {
Junxiao Shi8ff0a862016-08-13 04:50:50 +0000146 Interest pitInterest;
Junxiao Shi5e5e4452015-09-24 16:56:52 -0700147 };
148 std::vector<RejectPendingInterestArgs> rejectPendingInterestHistory;
149
150 struct SendNackArgs
151 {
Junxiao Shi8ff0a862016-08-13 04:50:50 +0000152 Interest pitInterest;
Junxiao Shi5e5e4452015-09-24 16:56:52 -0700153 FaceId outFaceId;
154 lp::NackHeader header;
155 };
156 std::vector<SendNackArgs> sendNackHistory;
Junxiao Shi727ed292014-02-19 23:26:45 -0700157};
158
Junxiao Shid9ee45c2014-02-27 15:38:11 -0700159} // namespace tests
Spyridon Mastorakisd0381c02015-02-19 10:29:41 -0800160} // namespace fw
Junxiao Shi727ed292014-02-19 23:26:45 -0700161} // namespace nfd
162
Davide Pesavento97210d52016-10-14 15:45:48 +0200163#endif // NFD_TESTS_DAEMON_FW_STRATEGY_TESTER_HPP