blob: a39653013a16341f1960c0c113e77dd8d93a0cc5 [file] [log] [blame]
Eric Newberrya98bf932015-09-21 00:58:47 -07001/* -*- Mode:C++; c-file-style:"gnu"; indent-tabs-mode:nil; -*- */
Junxiao Shi84d62cb2017-07-12 16:15:18 +00002/*
Davide Pesaventoe422f9e2022-06-03 01:30:23 -04003 * Copyright (c) 2014-2022, Regents of the University of California,
Eric Newberrya98bf932015-09-21 00:58:47 -07004 * 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.
10 *
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/>.
24 */
25
Davide Pesavento096f86a2018-11-10 19:51:45 -050026#include "websocket-channel-fixture.hpp"
Weiwei Liu280d7dd2016-03-02 23:19:26 -070027#include "face/websocket-transport.hpp"
Junxiao Shicde37ad2015-12-24 01:02:05 -070028
Davide Pesavento22fba352017-10-17 15:53:51 -040029#include "test-ip.hpp"
Davide Pesaventob3a23ca2019-05-04 20:40:21 -040030
Davide Pesavento096f86a2018-11-10 19:51:45 -050031#include <boost/mpl/vector.hpp>
Eric Newberrya98bf932015-09-21 00:58:47 -070032
Davide Pesaventoe422f9e2022-06-03 01:30:23 -040033namespace nfd::tests {
Eric Newberrya98bf932015-09-21 00:58:47 -070034
Davide Pesavento9a00fab2016-09-27 11:22:46 +020035BOOST_AUTO_TEST_SUITE(Face)
Weiwei Liu280d7dd2016-03-02 23:19:26 -070036BOOST_FIXTURE_TEST_SUITE(TestWebSocketChannel, WebSocketChannelFixture)
37
Davide Pesavento096f86a2018-11-10 19:51:45 -050038using AddressFamilies = boost::mpl::vector<
39 std::integral_constant<AddressFamily, AddressFamily::V4>,
40 std::integral_constant<AddressFamily, AddressFamily::V6>>;
41
42BOOST_AUTO_TEST_CASE_TEMPLATE(Uri, F, AddressFamilies)
Weiwei Liu280d7dd2016-03-02 23:19:26 -070043{
Davide Pesavento096f86a2018-11-10 19:51:45 -050044 using Address = typename IpAddressFromFamily<F::value>::type;
45 websocket::Endpoint ep(Address::loopback(), 20070);
46 auto channel = this->makeChannel(ep.address(), ep.port());
Davide Pesaventoeee53aa2016-04-11 17:20:21 +020047 BOOST_CHECK_EQUAL(channel->getUri(), FaceUri(ep, "ws"));
Weiwei Liu280d7dd2016-03-02 23:19:26 -070048}
49
Davide Pesavento096f86a2018-11-10 19:51:45 -050050BOOST_AUTO_TEST_CASE_TEMPLATE(Listen, F, AddressFamilies)
Weiwei Liu280d7dd2016-03-02 23:19:26 -070051{
Davide Pesavento096f86a2018-11-10 19:51:45 -050052 using Address = typename IpAddressFromFamily<F::value>::type;
53 auto channel = this->makeChannel(Address());
Weiwei Liu280d7dd2016-03-02 23:19:26 -070054 BOOST_CHECK_EQUAL(channel->isListening(), false);
55
56 channel->listen(nullptr);
57 BOOST_CHECK_EQUAL(channel->isListening(), true);
58
59 // listen() is idempotent
60 BOOST_CHECK_NO_THROW(channel->listen(nullptr));
61 BOOST_CHECK_EQUAL(channel->isListening(), true);
62}
63
Davide Pesavento096f86a2018-11-10 19:51:45 -050064BOOST_AUTO_TEST_CASE_TEMPLATE(MultipleAccepts, F, AddressFamilies)
Weiwei Liu280d7dd2016-03-02 23:19:26 -070065{
Davide Pesavento096f86a2018-11-10 19:51:45 -050066 auto address = getTestIp(F::value, AddressScope::Loopback);
Davide Pesaventoeee53aa2016-04-11 17:20:21 +020067 SKIP_IF_IP_UNAVAILABLE(address);
68 this->listen(address);
Weiwei Liu280d7dd2016-03-02 23:19:26 -070069
70 BOOST_CHECK_EQUAL(listenerChannel->isListening(), true);
71 BOOST_CHECK_EQUAL(listenerChannel->size(), 0);
72
73 websocket::Client client1;
Davide Pesaventoeee53aa2016-04-11 17:20:21 +020074 this->clientConnect(client1);
75
Weiwei Liu280d7dd2016-03-02 23:19:26 -070076 BOOST_CHECK_EQUAL(limitedIo.run(2, // listenerOnFaceCreated, clientHandleOpen
Davide Pesavento00335782018-02-10 22:31:33 -050077 1_s), LimitedIo::EXCEED_OPS);
Weiwei Liu280d7dd2016-03-02 23:19:26 -070078 BOOST_CHECK_EQUAL(listenerChannel->size(), 1);
79
80 websocket::Client client2;
81 websocket::Client client3;
Davide Pesaventoeee53aa2016-04-11 17:20:21 +020082 this->clientConnect(client2);
83 this->clientConnect(client3);
84
Weiwei Liu280d7dd2016-03-02 23:19:26 -070085 BOOST_CHECK_EQUAL(limitedIo.run(4, // 2 listenerOnFaceCreated, 2 clientHandleOpen
Davide Pesavento00335782018-02-10 22:31:33 -050086 2_s), LimitedIo::EXCEED_OPS);
Weiwei Liu280d7dd2016-03-02 23:19:26 -070087 BOOST_CHECK_EQUAL(listenerChannel->size(), 3);
88
Alexander Afanasyev3a2339a2020-05-27 23:05:06 -040089 // check face persistency and channel association
Weiwei Liu280d7dd2016-03-02 23:19:26 -070090 for (const auto& face : listenerFaces) {
91 BOOST_CHECK_EQUAL(face->getPersistency(), ndn::nfd::FACE_PERSISTENCY_ON_DEMAND);
Alexander Afanasyev3a2339a2020-05-27 23:05:06 -040092 BOOST_CHECK_EQUAL(face->getChannel().lock(), listenerChannel);
Weiwei Liu280d7dd2016-03-02 23:19:26 -070093 }
94}
95
Davide Pesavento096f86a2018-11-10 19:51:45 -050096BOOST_AUTO_TEST_CASE_TEMPLATE(Send, F, AddressFamilies)
Weiwei Liu280d7dd2016-03-02 23:19:26 -070097{
Davide Pesavento096f86a2018-11-10 19:51:45 -050098 auto address = getTestIp(F::value, AddressScope::Loopback);
Davide Pesaventoeee53aa2016-04-11 17:20:21 +020099 SKIP_IF_IP_UNAVAILABLE(address);
100 this->initialize(address);
Davide Pesavento096f86a2018-11-10 19:51:45 -0500101 auto transport = listenerFaces.at(0)->getTransport();
Weiwei Liu280d7dd2016-03-02 23:19:26 -0700102
103 Block pkt1 = ndn::encoding::makeStringBlock(300, "hello");
Davide Pesaventob3a23ca2019-05-04 20:40:21 -0400104 transport->send(pkt1);
Weiwei Liu280d7dd2016-03-02 23:19:26 -0700105 BOOST_CHECK_EQUAL(limitedIo.run(1, // clientHandleMessage
Davide Pesavento00335782018-02-10 22:31:33 -0500106 1_s), LimitedIo::EXCEED_OPS);
Weiwei Liu280d7dd2016-03-02 23:19:26 -0700107
108 Block pkt2 = ndn::encoding::makeStringBlock(301, "world!");
Davide Pesaventob3a23ca2019-05-04 20:40:21 -0400109 transport->send(pkt2);
Weiwei Liu280d7dd2016-03-02 23:19:26 -0700110 BOOST_CHECK_EQUAL(limitedIo.run(1, // clientHandleMessage
Davide Pesavento00335782018-02-10 22:31:33 -0500111 1_s), LimitedIo::EXCEED_OPS);
Weiwei Liu280d7dd2016-03-02 23:19:26 -0700112
113 BOOST_REQUIRE_EQUAL(clientReceivedMessages.size(), 2);
114 BOOST_CHECK_EQUAL_COLLECTIONS(
115 reinterpret_cast<const uint8_t*>(clientReceivedMessages[0].data()),
116 reinterpret_cast<const uint8_t*>(clientReceivedMessages[0].data()) + clientReceivedMessages[0].size(),
117 pkt1.begin(), pkt1.end());
118 BOOST_CHECK_EQUAL_COLLECTIONS(
119 reinterpret_cast<const uint8_t*>(clientReceivedMessages[1].data()),
120 reinterpret_cast<const uint8_t*>(clientReceivedMessages[1].data()) + clientReceivedMessages[1].size(),
121 pkt2.begin(), pkt2.end());
122}
123
Davide Pesavento096f86a2018-11-10 19:51:45 -0500124BOOST_AUTO_TEST_CASE_TEMPLATE(Receive, F, AddressFamilies)
Weiwei Liu280d7dd2016-03-02 23:19:26 -0700125{
Davide Pesavento096f86a2018-11-10 19:51:45 -0500126 auto address = getTestIp(F::value, AddressScope::Loopback);
Davide Pesaventoeee53aa2016-04-11 17:20:21 +0200127 SKIP_IF_IP_UNAVAILABLE(address);
128 this->initialize(address);
Weiwei Liu280d7dd2016-03-02 23:19:26 -0700129
130 // use network-layer packets here, otherwise GenericLinkService
131 // won't recognize the packet type and will discard it
132 auto interest1 = makeInterest("ndn:/TpnzGvW9R");
133 auto interest2 = makeInterest("ndn:/QWiIMfj5sL");
134
Davide Pesavento096f86a2018-11-10 19:51:45 -0500135 this->clientSendInterest(*interest1);
Weiwei Liu280d7dd2016-03-02 23:19:26 -0700136 BOOST_CHECK_EQUAL(limitedIo.run(1, // faceAfterReceiveInterest
Davide Pesavento00335782018-02-10 22:31:33 -0500137 1_s), LimitedIo::EXCEED_OPS);
Weiwei Liu280d7dd2016-03-02 23:19:26 -0700138
Davide Pesavento096f86a2018-11-10 19:51:45 -0500139 this->clientSendInterest(*interest2);
Weiwei Liu280d7dd2016-03-02 23:19:26 -0700140 BOOST_CHECK_EQUAL(limitedIo.run(1, // faceAfterReceiveInterest
Davide Pesavento00335782018-02-10 22:31:33 -0500141 1_s), LimitedIo::EXCEED_OPS);
Weiwei Liu280d7dd2016-03-02 23:19:26 -0700142
143 BOOST_REQUIRE_EQUAL(faceReceivedInterests.size(), 2);
144 BOOST_CHECK_EQUAL(faceReceivedInterests[0].getName(), interest1->getName());
145 BOOST_CHECK_EQUAL(faceReceivedInterests[1].getName(), interest2->getName());
146}
147
Davide Pesavento096f86a2018-11-10 19:51:45 -0500148BOOST_AUTO_TEST_CASE_TEMPLATE(FaceClosure, F, AddressFamilies)
Weiwei Liu280d7dd2016-03-02 23:19:26 -0700149{
Davide Pesavento096f86a2018-11-10 19:51:45 -0500150 auto address = getTestIp(F::value, AddressScope::Loopback);
Davide Pesaventoeee53aa2016-04-11 17:20:21 +0200151 SKIP_IF_IP_UNAVAILABLE(address);
152 this->initialize(address);
Weiwei Liu280d7dd2016-03-02 23:19:26 -0700153
Alexander Afanasyev3a2339a2020-05-27 23:05:06 -0400154 BOOST_CHECK_EQUAL(listenerFaces.at(0)->getChannel().lock(), listenerChannel);
155
Davide Pesavento096f86a2018-11-10 19:51:45 -0500156 listenerFaces.at(0)->close();
Weiwei Liu280d7dd2016-03-02 23:19:26 -0700157 BOOST_CHECK_EQUAL(listenerChannel->size(), 0);
158}
159
Davide Pesavento096f86a2018-11-10 19:51:45 -0500160BOOST_AUTO_TEST_CASE_TEMPLATE(RemoteClose, F, AddressFamilies)
Weiwei Liu280d7dd2016-03-02 23:19:26 -0700161{
Davide Pesavento096f86a2018-11-10 19:51:45 -0500162 auto address = getTestIp(F::value, AddressScope::Loopback);
Davide Pesaventoeee53aa2016-04-11 17:20:21 +0200163 SKIP_IF_IP_UNAVAILABLE(address);
164 this->initialize(address);
Weiwei Liu280d7dd2016-03-02 23:19:26 -0700165
166 client.close(clientHandle, websocketpp::close::status::going_away, "");
167 BOOST_CHECK_EQUAL(limitedIo.run(1, // faceClosedSignal
Davide Pesavento00335782018-02-10 22:31:33 -0500168 1_s), LimitedIo::EXCEED_OPS);
Weiwei Liu280d7dd2016-03-02 23:19:26 -0700169 BOOST_CHECK_EQUAL(listenerChannel->size(), 0);
170}
171
Davide Pesavento096f86a2018-11-10 19:51:45 -0500172BOOST_AUTO_TEST_CASE_TEMPLATE(SetPingInterval, F, AddressFamilies)
Weiwei Liu280d7dd2016-03-02 23:19:26 -0700173{
Davide Pesavento096f86a2018-11-10 19:51:45 -0500174 auto address = getTestIp(F::value, AddressScope::Loopback);
Davide Pesaventoeee53aa2016-04-11 17:20:21 +0200175 SKIP_IF_IP_UNAVAILABLE(address);
Davide Pesavento412c9822021-07-02 00:21:05 -0400176 const auto pingInterval = 1500_ms;
Davide Pesavento00335782018-02-10 22:31:33 -0500177 this->initialize(address, pingInterval);
Weiwei Liu280d7dd2016-03-02 23:19:26 -0700178
Davide Pesavento412c9822021-07-02 00:21:05 -0400179 BOOST_CHECK_EQUAL(limitedIo.run(5, // clientHandlePing
180 5 * pingInterval + 1_s), LimitedIo::EXCEED_OPS);
181 BOOST_CHECK_EQUAL(measuredPingIntervals.size(), 4);
182
183 auto avgPingInterval = std::accumulate(measuredPingIntervals.begin(), measuredPingIntervals.end(), 0_ns);
184 avgPingInterval /= measuredPingIntervals.size();
185 BOOST_CHECK_LE(avgPingInterval, pingInterval * 1.1);
186 BOOST_CHECK_GE(avgPingInterval, pingInterval * 0.9);
Weiwei Liu280d7dd2016-03-02 23:19:26 -0700187}
188
Davide Pesavento096f86a2018-11-10 19:51:45 -0500189BOOST_AUTO_TEST_CASE_TEMPLATE(SetPongTimeOut, F, AddressFamilies)
Weiwei Liu280d7dd2016-03-02 23:19:26 -0700190{
Davide Pesavento096f86a2018-11-10 19:51:45 -0500191 auto address = getTestIp(F::value, AddressScope::Loopback);
Davide Pesaventoeee53aa2016-04-11 17:20:21 +0200192 SKIP_IF_IP_UNAVAILABLE(address);
Davide Pesavento00335782018-02-10 22:31:33 -0500193 this->initialize(address, 600_ms, 300_ms);
Davide Pesavento096f86a2018-11-10 19:51:45 -0500194 this->clientShouldPong = false;
Davide Pesaventoeee53aa2016-04-11 17:20:21 +0200195
Weiwei Liu280d7dd2016-03-02 23:19:26 -0700196 BOOST_CHECK_EQUAL(limitedIo.run(2, // clientHandlePing, faceClosedSignal
Davide Pesavento00335782018-02-10 22:31:33 -0500197 2_s), LimitedIo::EXCEED_OPS);
Weiwei Liu280d7dd2016-03-02 23:19:26 -0700198 BOOST_CHECK_EQUAL(listenerChannel->size(), 0);
199
Davide Pesaventoe422f9e2022-06-03 01:30:23 -0400200 auto transport = static_cast<face::WebSocketTransport*>(listenerFaces.at(0)->getTransport());
201 BOOST_CHECK(transport->getState() == face::TransportState::FAILED ||
202 transport->getState() == face::TransportState::CLOSED);
Davide Pesavento00335782018-02-10 22:31:33 -0500203 BOOST_CHECK_GE(transport->getCounters().nOutPings, 1);
204 BOOST_CHECK_LE(transport->getCounters().nOutPings, 2);
Weiwei Liu280d7dd2016-03-02 23:19:26 -0700205 BOOST_CHECK_EQUAL(transport->getCounters().nInPongs, 0);
206}
Junxiao Shicde37ad2015-12-24 01:02:05 -0700207
208BOOST_AUTO_TEST_SUITE_END() // TestWebSocketChannel
209BOOST_AUTO_TEST_SUITE_END() // Face
210
Davide Pesaventoe422f9e2022-06-03 01:30:23 -0400211} // namespace nfd::tests