blob: e32ad7dd374593e199d3cd8c2ab98c015bd6b04b [file] [log] [blame]
Andrea Tosatto672b9a72016-01-05 16:18:20 +01001/* -*- Mode:C++; c-file-style:"gnu"; indent-tabs-mode:nil; -*- */
Chavoosh Ghasemi4d36ed52017-10-31 22:26:25 +00002/*
Chavoosh Ghasemi5cb67012019-02-15 09:56:57 -08003 * Copyright (c) 2016-2019, Regents of the University of California,
Davide Pesaventobf1c0692017-01-15 19:15:09 -05004 * Colorado State University,
5 * University Pierre & Marie Curie, Sorbonne University.
Andrea Tosatto672b9a72016-01-05 16:18:20 +01006 *
7 * This file is part of ndn-tools (Named Data Networking Essential Tools).
8 * See AUTHORS.md for complete list of ndn-tools authors and contributors.
9 *
10 * ndn-tools is free software: you can redistribute it and/or modify it under the terms
11 * of the GNU General Public License as published by the Free Software Foundation,
12 * either version 3 of the License, or (at your option) any later version.
13 *
14 * ndn-tools is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY;
15 * without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR
16 * PURPOSE. See the GNU General Public License for more details.
17 *
18 * You should have received a copy of the GNU General Public License along with
19 * ndn-tools, e.g., in COPYING.md file. If not, see <http://www.gnu.org/licenses/>.
20 *
21 * See AUTHORS.md for complete list of ndn-cxx authors and contributors.
22 *
23 * @author Wentao Shang
24 * @author Steve DiBenedetto
25 * @author Andrea Tosatto
Weiwei Liue4765012016-06-01 00:10:29 -070026 * @author Davide Pesavento
27 * @author Weiwei Liu
Chavoosh Ghasemi4d36ed52017-10-31 22:26:25 +000028 * @author Chavoosh Ghasemi
Andrea Tosatto672b9a72016-01-05 16:18:20 +010029 */
30
31#include "pipeline-interests.hpp"
Andrea Tosatto672b9a72016-01-05 16:18:20 +010032
33namespace ndn {
34namespace chunks {
35
Weiwei Liue4765012016-06-01 00:10:29 -070036PipelineInterests::PipelineInterests(Face& face)
Andrea Tosatto672b9a72016-01-05 16:18:20 +010037 : m_face(face)
Davide Pesaventoe9c69852017-11-04 18:08:37 -040038 , m_hasFinalBlockId(false)
Andrea Tosatto672b9a72016-01-05 16:18:20 +010039 , m_lastSegmentNo(0)
Chavoosh Ghasemi4d36ed52017-10-31 22:26:25 +000040 , m_nReceived(0)
Davide Pesaventob587cfd2017-11-04 16:29:50 -040041 , m_receivedSize(0)
Davide Pesaventob587cfd2017-11-04 16:29:50 -040042 , m_nextSegmentNo(0)
Weiwei Liue4765012016-06-01 00:10:29 -070043 , m_isStopping(false)
Andrea Tosatto672b9a72016-01-05 16:18:20 +010044{
Andrea Tosatto672b9a72016-01-05 16:18:20 +010045}
46
Weiwei Liue4765012016-06-01 00:10:29 -070047PipelineInterests::~PipelineInterests() = default;
Andrea Tosatto672b9a72016-01-05 16:18:20 +010048
49void
Chavoosh Ghasemi5cb67012019-02-15 09:56:57 -080050PipelineInterests::run(const Name& versionedName, DataCallback dataCb, FailureCallback failureCb)
Andrea Tosatto672b9a72016-01-05 16:18:20 +010051{
Chavoosh Ghasemi5cb67012019-02-15 09:56:57 -080052 BOOST_ASSERT(!versionedName.empty() && versionedName[-1].isVersion());
Davide Pesaventoe9c69852017-11-04 18:08:37 -040053 BOOST_ASSERT(dataCb != nullptr);
Chavoosh Ghasemi5cb67012019-02-15 09:56:57 -080054 m_prefix = versionedName;
Davide Pesaventoe9c69852017-11-04 18:08:37 -040055 m_onData = std::move(dataCb);
56 m_onFailure = std::move(failureCb);
Davide Pesaventoe9c69852017-11-04 18:08:37 -040057
Davide Pesaventobf1c0692017-01-15 19:15:09 -050058 // record the start time of the pipeline
59 m_startTime = time::steady_clock::now();
60
Weiwei Liue4765012016-06-01 00:10:29 -070061 doRun();
Andrea Tosatto672b9a72016-01-05 16:18:20 +010062}
63
64void
65PipelineInterests::cancel()
66{
Weiwei Liue4765012016-06-01 00:10:29 -070067 if (m_isStopping)
68 return;
Andrea Tosatto672b9a72016-01-05 16:18:20 +010069
Weiwei Liue4765012016-06-01 00:10:29 -070070 m_isStopping = true;
71 doCancel();
Andrea Tosatto672b9a72016-01-05 16:18:20 +010072}
73
Ryan Wickman2c9933c2018-06-12 11:51:51 -050074bool
75PipelineInterests::allSegmentsReceived() const
76{
Chavoosh Ghasemi5cb67012019-02-15 09:56:57 -080077 return m_nReceived > 0 &&
78 m_hasFinalBlockId &&
79 static_cast<uint64_t>(m_nReceived - 1) >= m_lastSegmentNo;
Ryan Wickman2c9933c2018-06-12 11:51:51 -050080}
81
Davide Pesaventob587cfd2017-11-04 16:29:50 -040082uint64_t
83PipelineInterests::getNextSegmentNo()
84{
Davide Pesaventob587cfd2017-11-04 16:29:50 -040085 return m_nextSegmentNo++;
86}
87
Andrea Tosatto672b9a72016-01-05 16:18:20 +010088void
Davide Pesaventoe9c69852017-11-04 18:08:37 -040089PipelineInterests::onData(const Data& data)
90{
91 m_nReceived++;
92 m_receivedSize += data.getContent().value_size();
93
94 m_onData(data);
95}
96
97void
Weiwei Liue4765012016-06-01 00:10:29 -070098PipelineInterests::onFailure(const std::string& reason)
Andrea Tosatto672b9a72016-01-05 16:18:20 +010099{
Weiwei Liue4765012016-06-01 00:10:29 -0700100 if (m_isStopping)
Andrea Tosatto672b9a72016-01-05 16:18:20 +0100101 return;
102
Weiwei Liue4765012016-06-01 00:10:29 -0700103 cancel();
Andrea Tosatto672b9a72016-01-05 16:18:20 +0100104
Weiwei Liue4765012016-06-01 00:10:29 -0700105 if (m_onFailure)
106 m_face.getIoService().post([this, reason] { m_onFailure(reason); });
Andrea Tosatto672b9a72016-01-05 16:18:20 +0100107}
108
Chavoosh Ghasemi4d36ed52017-10-31 22:26:25 +0000109std::string
110PipelineInterests::formatThroughput(double throughput)
111{
112 int pow = 0;
113 while (throughput >= 1000.0 && pow < 4) {
114 throughput /= 1000.0;
115 pow++;
116 }
117 switch (pow) {
118 case 0:
119 return to_string(throughput) + " bit/s";
120 case 1:
121 return to_string(throughput) + " kbit/s";
122 case 2:
123 return to_string(throughput) + " Mbit/s";
124 case 3:
125 return to_string(throughput) + " Gbit/s";
126 case 4:
127 return to_string(throughput) + " Tbit/s";
128 }
129 return "";
130}
131
132void
133PipelineInterests::printSummary() const
134{
Davide Pesaventob587cfd2017-11-04 16:29:50 -0400135 using namespace ndn::time;
Davide Pesaventoba560662019-06-26 22:45:44 -0400136 duration<double, seconds::period> timeElapsed = steady_clock::now() - getStartTime();
137 double throughput = 8 * m_receivedSize / timeElapsed.count();
Chavoosh Ghasemi4d36ed52017-10-31 22:26:25 +0000138
schneiderklaus8ff3abd2019-03-12 22:15:12 -0700139 std::cerr << "\n\nAll segments have been received.\n"
Chavoosh Ghasemi4d36ed52017-10-31 22:26:25 +0000140 << "Time elapsed: " << timeElapsed << "\n"
schneiderklaus8ff3abd2019-03-12 22:15:12 -0700141 << "Segments received: " << m_nReceived << "\n"
Davide Pesaventoba560662019-06-26 22:45:44 -0400142 << "Transferred size: " << m_receivedSize / 1e3 << " kB" << "\n"
Chavoosh Ghasemi4d36ed52017-10-31 22:26:25 +0000143 << "Goodput: " << formatThroughput(throughput) << "\n";
144}
145
Andrea Tosatto672b9a72016-01-05 16:18:20 +0100146} // namespace chunks
147} // namespace ndn