blob: 2379d5ca9457a67f801e296caa46e9e1c634c58c [file] [log] [blame]
Alexander Afanasyevc169a812014-05-20 20:37:29 -04001/* -*- Mode:C++; c-file-style:"gnu"; indent-tabs-mode:nil; -*- */
Junxiao Shi7b6b79d2014-03-26 20:59:35 -07002/**
Spyridon Mastorakis429634f2015-02-19 17:35:33 -08003 * Copyright (c) 2013-2015 Regents of the University of California.
Alexander Afanasyevdfa52c42014-04-24 21:10:11 -07004 *
5 * This file is part of ndn-cxx library (NDN C++ library with eXperimental eXtensions).
Alexander Afanasyevdfa52c42014-04-24 21:10:11 -07006 *
Alexander Afanasyevc169a812014-05-20 20:37:29 -04007 * ndn-cxx library is free software: you can redistribute it and/or modify it under the
8 * terms of the GNU Lesser General Public License as published by the Free Software
9 * Foundation, either version 3 of the License, or (at your option) any later version.
10 *
11 * ndn-cxx library is distributed in the hope that it will be useful, but WITHOUT ANY
12 * WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A
13 * PARTICULAR PURPOSE. See the GNU Lesser General Public License for more details.
14 *
15 * You should have received copies of the GNU General Public License and GNU Lesser
16 * General Public License along with ndn-cxx, e.g., in COPYING.md file. If not, see
17 * <http://www.gnu.org/licenses/>.
18 *
19 * See AUTHORS.md for complete list of ndn-cxx authors and contributors.
Junxiao Shi7b6b79d2014-03-26 20:59:35 -070020 */
21
22#include "management/nfd-controller.hpp"
Junxiao Shi7b6b79d2014-03-26 20:59:35 -070023#include "management/nfd-control-response.hpp"
24
Alexander Afanasyev258ec2b2014-05-14 16:15:37 -070025#include <boost/tuple/tuple.hpp>
26
Alexander Afanasyevb1db7c62014-04-03 14:57:25 -070027#include "boost-test.hpp"
Junxiao Shia60d9362014-11-12 09:38:21 -070028#include "util/dummy-client-face.hpp"
Junxiao Shib1990df2015-11-05 00:14:44 +000029#include "../make-interest-data.hpp"
Alexander Afanasyevd3a55b22014-11-18 19:23:28 -080030#include "../unit-test-time-fixture.hpp"
Junxiao Shi7b6b79d2014-03-26 20:59:35 -070031
32namespace ndn {
33namespace nfd {
Alexander Afanasyev4abdbf12014-08-11 12:48:54 -070034namespace tests {
35
Junxiao Shia60d9362014-11-12 09:38:21 -070036using ndn::util::DummyClientFace;
Junxiao Shi7b6b79d2014-03-26 20:59:35 -070037
Junxiao Shib1990df2015-11-05 00:14:44 +000038BOOST_AUTO_TEST_SUITE(Management)
Junxiao Shi7b6b79d2014-03-26 20:59:35 -070039
Alexander Afanasyevd3a55b22014-11-18 19:23:28 -080040class CommandFixture : public ndn::tests::UnitTestTimeFixture
Junxiao Shi7b6b79d2014-03-26 20:59:35 -070041{
42protected:
43 CommandFixture()
Alexander Afanasyev9bdbb832015-12-30 12:54:22 -080044 : face(io)
45 , controller(face, keyChain)
Junxiao Shi7b6b79d2014-03-26 20:59:35 -070046 , commandSucceedCallback(bind(&CommandFixture::onCommandSucceed, this, _1))
47 , commandFailCallback(bind(&CommandFixture::onCommandFail, this, _1, _2))
48 {
49 }
50
51private:
52 void
53 onCommandSucceed(const ControlParameters& parameters)
54 {
55 commandSucceedHistory.push_back(boost::make_tuple(parameters));
56 }
57
58 void
59 onCommandFail(uint32_t code, const std::string& reason)
60 {
61 commandFailHistory.push_back(boost::make_tuple(code, reason));
62 }
63
64protected:
Alexander Afanasyev9bdbb832015-12-30 12:54:22 -080065 DummyClientFace face;
Junxiao Shi7b6b79d2014-03-26 20:59:35 -070066 KeyChain keyChain;
Junxiao Shi415b17c2014-11-12 00:43:25 -070067 Controller controller;
Junxiao Shi7b6b79d2014-03-26 20:59:35 -070068
69 Controller::CommandSucceedCallback commandSucceedCallback;
70 typedef boost::tuple<ControlParameters> CommandSucceedArgs;
71 std::vector<CommandSucceedArgs> commandSucceedHistory;
72
73 Controller::CommandFailCallback commandFailCallback;
74 typedef boost::tuple<uint32_t,std::string> CommandFailArgs;
75 std::vector<CommandFailArgs> commandFailHistory;
76};
77
Junxiao Shib1990df2015-11-05 00:14:44 +000078BOOST_FIXTURE_TEST_SUITE(TestNfdController, CommandFixture)
79
80BOOST_AUTO_TEST_CASE(CommandSuccess)
Junxiao Shi7b6b79d2014-03-26 20:59:35 -070081{
82 ControlParameters parameters;
Junxiao Shi5de006b2014-10-26 20:20:52 -070083 parameters.setUri("tcp4://192.0.2.1:6363");
Junxiao Shi7b6b79d2014-03-26 20:59:35 -070084
85 BOOST_CHECK_NO_THROW(controller.start<FaceCreateCommand>(
Junxiao Shi70911652014-08-12 10:14:24 -070086 parameters,
87 commandSucceedCallback,
88 commandFailCallback));
Alexander Afanasyevd3a55b22014-11-18 19:23:28 -080089
90 advanceClocks(time::milliseconds(1));
Junxiao Shi7b6b79d2014-03-26 20:59:35 -070091
Alexander Afanasyev9bdbb832015-12-30 12:54:22 -080092 BOOST_REQUIRE_EQUAL(face.sentInterests.size(), 1);
93 const Interest& requestInterest = face.sentInterests[0];
Junxiao Shi7b6b79d2014-03-26 20:59:35 -070094
95 FaceCreateCommand command;
Junxiao Shi5de006b2014-10-26 20:20:52 -070096 BOOST_CHECK(Name("/localhost/nfd/faces/create").isPrefixOf(requestInterest.getName()));
97 // 9 components: ndn:/localhost/nfd/faces/create/<parameters>/<signed Interest x4>
Junxiao Shi6a90f372014-10-13 20:29:30 -070098 BOOST_REQUIRE_EQUAL(requestInterest.getName().size(), 9);
Junxiao Shi7b6b79d2014-03-26 20:59:35 -070099 ControlParameters request;
100 // 4th component: <parameters>
Junxiao Shi6a90f372014-10-13 20:29:30 -0700101 BOOST_REQUIRE_NO_THROW(request.wireDecode(requestInterest.getName().at(4).blockFromValue()));
Junxiao Shi7b6b79d2014-03-26 20:59:35 -0700102 BOOST_CHECK_NO_THROW(command.validateRequest(request));
103 BOOST_CHECK_EQUAL(request.getUri(), parameters.getUri());
Junxiao Shi415b17c2014-11-12 00:43:25 -0700104 BOOST_CHECK_EQUAL(requestInterest.getInterestLifetime(), CommandOptions::DEFAULT_TIMEOUT);
Junxiao Shi7b6b79d2014-03-26 20:59:35 -0700105
106 ControlParameters responseBody;
107 responseBody.setUri("tcp4://192.0.2.1:6363")
Yukai Tud93c5fc2015-08-25 11:37:16 +0800108 .setFaceId(22)
109 .setFacePersistency(ndn::nfd::FacePersistency::FACE_PERSISTENCY_PERSISTENT);
Junxiao Shi7b6b79d2014-03-26 20:59:35 -0700110 ControlResponse responsePayload(201, "created");
111 responsePayload.setBody(responseBody.wireEncode());
112
Junxiao Shib1990df2015-11-05 00:14:44 +0000113 auto responseData = util::makeData(requestInterest.getName());
114 responseData->setContent(responsePayload.wireEncode());
Alexander Afanasyev9bdbb832015-12-30 12:54:22 -0800115 face.receive(*responseData);
Alexander Afanasyevd3a55b22014-11-18 19:23:28 -0800116
117 advanceClocks(time::milliseconds(1));
Junxiao Shi7b6b79d2014-03-26 20:59:35 -0700118
119 BOOST_CHECK_EQUAL(commandFailHistory.size(), 0);
120 BOOST_REQUIRE_EQUAL(commandSucceedHistory.size(), 1);
121 const ControlParameters& response = commandSucceedHistory[0].get<0>();
122 BOOST_CHECK_EQUAL(response.getUri(), responseBody.getUri());
123 BOOST_CHECK_EQUAL(response.getFaceId(), responseBody.getFaceId());
124}
125
Junxiao Shib1990df2015-11-05 00:14:44 +0000126BOOST_AUTO_TEST_CASE(CommandInvalidRequest)
Junxiao Shi7b6b79d2014-03-26 20:59:35 -0700127{
128 ControlParameters parameters;
129 parameters.setName("ndn:/should-not-have-this-field");
130 // Uri is missing
131
132 BOOST_CHECK_THROW(controller.start<FaceCreateCommand>(
133 parameters,
134 commandSucceedCallback,
135 commandFailCallback),
136 ControlCommand::ArgumentError);
137}
138
Junxiao Shib1990df2015-11-05 00:14:44 +0000139BOOST_AUTO_TEST_CASE(CommandErrorCode)
Junxiao Shi7b6b79d2014-03-26 20:59:35 -0700140{
141 ControlParameters parameters;
Junxiao Shi5de006b2014-10-26 20:20:52 -0700142 parameters.setUri("tcp4://192.0.2.1:6363");
Junxiao Shi7b6b79d2014-03-26 20:59:35 -0700143
144 BOOST_CHECK_NO_THROW(controller.start<FaceCreateCommand>(
145 parameters,
146 commandSucceedCallback,
147 commandFailCallback));
Alexander Afanasyevd3a55b22014-11-18 19:23:28 -0800148 advanceClocks(time::milliseconds(1));
Junxiao Shi7b6b79d2014-03-26 20:59:35 -0700149
Alexander Afanasyev9bdbb832015-12-30 12:54:22 -0800150 BOOST_REQUIRE_EQUAL(face.sentInterests.size(), 1);
151 const Interest& requestInterest = face.sentInterests[0];
Junxiao Shi7b6b79d2014-03-26 20:59:35 -0700152
153 ControlResponse responsePayload(401, "Not Authenticated");
154
Junxiao Shib1990df2015-11-05 00:14:44 +0000155 auto responseData = util::makeData(requestInterest.getName());
156 responseData->setContent(responsePayload.wireEncode());
Alexander Afanasyev9bdbb832015-12-30 12:54:22 -0800157 face.receive(*responseData);
Alexander Afanasyevd3a55b22014-11-18 19:23:28 -0800158 advanceClocks(time::milliseconds(1));
Junxiao Shi7b6b79d2014-03-26 20:59:35 -0700159
160 BOOST_CHECK_EQUAL(commandSucceedHistory.size(), 0);
161 BOOST_REQUIRE_EQUAL(commandFailHistory.size(), 1);
162 BOOST_CHECK_EQUAL(commandFailHistory[0].get<0>(), 401);
163}
164
Junxiao Shib1990df2015-11-05 00:14:44 +0000165BOOST_AUTO_TEST_CASE(CommandInvalidResponse)
Junxiao Shi7b6b79d2014-03-26 20:59:35 -0700166{
167 ControlParameters parameters;
Junxiao Shi5de006b2014-10-26 20:20:52 -0700168 parameters.setUri("tcp4://192.0.2.1:6363");
Junxiao Shi7b6b79d2014-03-26 20:59:35 -0700169
170 BOOST_CHECK_NO_THROW(controller.start<FaceCreateCommand>(
171 parameters,
172 commandSucceedCallback,
173 commandFailCallback));
Alexander Afanasyevd3a55b22014-11-18 19:23:28 -0800174 advanceClocks(time::milliseconds(1));
Junxiao Shi7b6b79d2014-03-26 20:59:35 -0700175
Alexander Afanasyev9bdbb832015-12-30 12:54:22 -0800176 BOOST_REQUIRE_EQUAL(face.sentInterests.size(), 1);
177 const Interest& requestInterest = face.sentInterests[0];
Junxiao Shi7b6b79d2014-03-26 20:59:35 -0700178
179 ControlParameters responseBody;
180 responseBody.setUri("tcp4://192.0.2.1:6363")
181 .setName("ndn:/should-not-have-this-field");
182 // FaceId is missing
183 ControlResponse responsePayload(201, "created");
184 responsePayload.setBody(responseBody.wireEncode());
185
Junxiao Shib1990df2015-11-05 00:14:44 +0000186 auto responseData = util::makeData(requestInterest.getName());
187 responseData->setContent(responsePayload.wireEncode());
Alexander Afanasyev9bdbb832015-12-30 12:54:22 -0800188 face.receive(*responseData);
Alexander Afanasyevd3a55b22014-11-18 19:23:28 -0800189 advanceClocks(time::milliseconds(1));
Junxiao Shi7b6b79d2014-03-26 20:59:35 -0700190
191 BOOST_CHECK_EQUAL(commandSucceedHistory.size(), 0);
192 BOOST_REQUIRE_EQUAL(commandFailHistory.size(), 1);
193}
194
Junxiao Shib1990df2015-11-05 00:14:44 +0000195BOOST_AUTO_TEST_CASE(CommandNack)
196{
197 ControlParameters parameters;
198 parameters.setUri("tcp4://192.0.2.1:6363");
199
200 BOOST_CHECK_NO_THROW(controller.start<FaceCreateCommand>(
201 parameters,
202 commandSucceedCallback,
203 commandFailCallback));
204 advanceClocks(time::milliseconds(1));
205
Alexander Afanasyev9bdbb832015-12-30 12:54:22 -0800206 BOOST_REQUIRE_EQUAL(face.sentInterests.size(), 1);
207 const Interest& requestInterest = face.sentInterests[0];
Junxiao Shib1990df2015-11-05 00:14:44 +0000208
209 auto responseNack = util::makeNack(requestInterest, lp::NackReason::NO_ROUTE);
Alexander Afanasyev9bdbb832015-12-30 12:54:22 -0800210 face.receive(responseNack);
Junxiao Shib1990df2015-11-05 00:14:44 +0000211 advanceClocks(time::milliseconds(1));
212
213 BOOST_REQUIRE_EQUAL(commandFailHistory.size(), 1);
214 BOOST_CHECK_EQUAL(commandFailHistory[0].get<0>(), Controller::ERROR_NACK);
215}
216
217BOOST_AUTO_TEST_CASE(OptionsPrefix)
Junxiao Shi5de006b2014-10-26 20:20:52 -0700218{
219 ControlParameters parameters;
220 parameters.setName("/ndn/com/example");
221 parameters.setFaceId(400);
222
223 CommandOptions options;
224 options.setPrefix("/localhop/net/example/router1/nfd");
225
226 BOOST_CHECK_NO_THROW(controller.start<RibRegisterCommand>(
227 parameters,
228 commandSucceedCallback,
229 commandFailCallback,
230 options));
Alexander Afanasyevd3a55b22014-11-18 19:23:28 -0800231 advanceClocks(time::milliseconds(1));
Junxiao Shi5de006b2014-10-26 20:20:52 -0700232
Alexander Afanasyev9bdbb832015-12-30 12:54:22 -0800233 BOOST_REQUIRE_EQUAL(face.sentInterests.size(), 1);
234 const Interest& requestInterest = face.sentInterests[0];
Junxiao Shi5de006b2014-10-26 20:20:52 -0700235
236 FaceCreateCommand command;
237 BOOST_CHECK(Name("/localhop/net/example/router1/nfd/rib/register").isPrefixOf(
238 requestInterest.getName()));
239}
240
Junxiao Shib1990df2015-11-05 00:14:44 +0000241BOOST_AUTO_TEST_CASE(OptionsTimeout)
Junxiao Shi5de006b2014-10-26 20:20:52 -0700242{
243 ControlParameters parameters;
244 parameters.setUri("tcp4://192.0.2.1:6363");
245
246 CommandOptions options;
247 options.setTimeout(time::milliseconds(50));
248
249 BOOST_CHECK_NO_THROW(controller.start<FaceCreateCommand>(
250 parameters,
251 commandSucceedCallback,
252 commandFailCallback,
253 options));
Alexander Afanasyevd3a55b22014-11-18 19:23:28 -0800254 advanceClocks(time::milliseconds(1), 101); // Face's PIT granularity is 100ms
Junxiao Shi5de006b2014-10-26 20:20:52 -0700255
256 BOOST_REQUIRE_EQUAL(commandFailHistory.size(), 1);
257 BOOST_CHECK_EQUAL(commandFailHistory[0].get<0>(), Controller::ERROR_TIMEOUT);
258}
259
Junxiao Shib1990df2015-11-05 00:14:44 +0000260BOOST_AUTO_TEST_SUITE_END() // TestController
261BOOST_AUTO_TEST_SUITE_END() // Management
Junxiao Shi7b6b79d2014-03-26 20:59:35 -0700262
Alexander Afanasyev4abdbf12014-08-11 12:48:54 -0700263} // namespace tests
Junxiao Shi7b6b79d2014-03-26 20:59:35 -0700264} // namespace nfd
265} // namespace ndn