blob: 2e294876f295a1fb5ad189343ab7c40f2e68bdc9 [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/*
Davide Pesavento11fc3eb2024-01-26 01:46:56 -05003 * Copyright (c) 2016-2024, 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"
Davide Pesavento97a33b22019-10-17 22:10:47 -040032#include "data-fetcher.hpp"
Andrea Tosatto672b9a72016-01-05 16:18:20 +010033
Davide Pesavento2ab04a22023-09-15 22:08:22 -040034#include <boost/asio/io_context.hpp>
35#include <boost/asio/post.hpp>
Davide Pesaventoa0e6b602021-01-21 19:47:04 -050036
Davide Pesavento5748e822024-01-26 18:40:22 -050037#include <iostream>
38
Davide Pesaventob3570c62022-02-19 19:19:00 -050039namespace ndn::chunks {
Andrea Tosatto672b9a72016-01-05 16:18:20 +010040
Davide Pesavento97a33b22019-10-17 22:10:47 -040041PipelineInterests::PipelineInterests(Face& face, const Options& opts)
42 : m_options(opts)
43 , m_face(face)
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{
Davide Pesavento296b3852019-10-20 16:00:16 -040052 BOOST_ASSERT(m_options.disableVersionDiscovery ||
53 (!versionedName.empty() && versionedName[-1].isVersion()));
Davide Pesaventoe9c69852017-11-04 18:08:37 -040054 BOOST_ASSERT(dataCb != nullptr);
Davide Pesavento296b3852019-10-20 16:00:16 -040055
Chavoosh Ghasemi5cb67012019-02-15 09:56:57 -080056 m_prefix = versionedName;
Davide Pesaventoe9c69852017-11-04 18:08:37 -040057 m_onData = std::move(dataCb);
58 m_onFailure = std::move(failureCb);
Davide Pesaventoe9c69852017-11-04 18:08:37 -040059
Davide Pesaventobf1c0692017-01-15 19:15:09 -050060 // record the start time of the pipeline
61 m_startTime = time::steady_clock::now();
62
Weiwei Liue4765012016-06-01 00:10:29 -070063 doRun();
Andrea Tosatto672b9a72016-01-05 16:18:20 +010064}
65
66void
67PipelineInterests::cancel()
68{
Weiwei Liue4765012016-06-01 00:10:29 -070069 if (m_isStopping)
70 return;
Andrea Tosatto672b9a72016-01-05 16:18:20 +010071
Weiwei Liue4765012016-06-01 00:10:29 -070072 m_isStopping = true;
73 doCancel();
Andrea Tosatto672b9a72016-01-05 16:18:20 +010074}
75
Ryan Wickman2c9933c2018-06-12 11:51:51 -050076bool
77PipelineInterests::allSegmentsReceived() const
78{
Chavoosh Ghasemi5cb67012019-02-15 09:56:57 -080079 return m_nReceived > 0 &&
80 m_hasFinalBlockId &&
81 static_cast<uint64_t>(m_nReceived - 1) >= m_lastSegmentNo;
Ryan Wickman2c9933c2018-06-12 11:51:51 -050082}
83
Davide Pesaventob587cfd2017-11-04 16:29:50 -040084uint64_t
85PipelineInterests::getNextSegmentNo()
86{
Davide Pesaventob587cfd2017-11-04 16:29:50 -040087 return m_nextSegmentNo++;
88}
89
Andrea Tosatto672b9a72016-01-05 16:18:20 +010090void
Davide Pesaventoe9c69852017-11-04 18:08:37 -040091PipelineInterests::onData(const Data& data)
92{
93 m_nReceived++;
94 m_receivedSize += data.getContent().value_size();
95
96 m_onData(data);
97}
98
99void
Weiwei Liue4765012016-06-01 00:10:29 -0700100PipelineInterests::onFailure(const std::string& reason)
Andrea Tosatto672b9a72016-01-05 16:18:20 +0100101{
Weiwei Liue4765012016-06-01 00:10:29 -0700102 if (m_isStopping)
Andrea Tosatto672b9a72016-01-05 16:18:20 +0100103 return;
104
Weiwei Liue4765012016-06-01 00:10:29 -0700105 cancel();
Andrea Tosatto672b9a72016-01-05 16:18:20 +0100106
Davide Pesavento2ab04a22023-09-15 22:08:22 -0400107 if (m_onFailure) {
Davide Pesavento7e9d7e42023-11-11 15:00:03 -0500108 boost::asio::post(m_face.getIoContext(), [this, reason] { m_onFailure(reason); });
Davide Pesavento2ab04a22023-09-15 22:08:22 -0400109 }
Andrea Tosatto672b9a72016-01-05 16:18:20 +0100110}
111
Davide Pesavento97a33b22019-10-17 22:10:47 -0400112void
113PipelineInterests::printOptions() const
114{
115 std::cerr << "Pipeline parameters:\n"
116 << "\tRequest fresh content = " << (m_options.mustBeFresh ? "yes" : "no") << "\n"
117 << "\tInterest lifetime = " << m_options.interestLifetime << "\n"
118 << "\tMax retries on timeout or Nack = " <<
119 (m_options.maxRetriesOnTimeoutOrNack == DataFetcher::MAX_RETRIES_INFINITE ?
Davide Pesavento11fc3eb2024-01-26 01:46:56 -0500120 "infinite" : std::to_string(m_options.maxRetriesOnTimeoutOrNack)) << "\n";
Davide Pesavento97a33b22019-10-17 22:10:47 -0400121}
122
123void
124PipelineInterests::printSummary() const
125{
126 using namespace ndn::time;
127 duration<double, seconds::period> timeElapsed = steady_clock::now() - getStartTime();
128 double throughput = 8 * m_receivedSize / timeElapsed.count();
129
130 std::cerr << "\n\nAll segments have been received.\n"
131 << "Time elapsed: " << timeElapsed << "\n"
132 << "Segments received: " << m_nReceived << "\n"
133 << "Transferred size: " << m_receivedSize / 1e3 << " kB" << "\n"
134 << "Goodput: " << formatThroughput(throughput) << "\n";
135}
136
Chavoosh Ghasemi4d36ed52017-10-31 22:26:25 +0000137std::string
138PipelineInterests::formatThroughput(double throughput)
139{
140 int pow = 0;
141 while (throughput >= 1000.0 && pow < 4) {
142 throughput /= 1000.0;
143 pow++;
144 }
145 switch (pow) {
146 case 0:
Davide Pesavento11fc3eb2024-01-26 01:46:56 -0500147 return std::to_string(throughput) + " bit/s";
Chavoosh Ghasemi4d36ed52017-10-31 22:26:25 +0000148 case 1:
Davide Pesavento11fc3eb2024-01-26 01:46:56 -0500149 return std::to_string(throughput) + " kbit/s";
Chavoosh Ghasemi4d36ed52017-10-31 22:26:25 +0000150 case 2:
Davide Pesavento11fc3eb2024-01-26 01:46:56 -0500151 return std::to_string(throughput) + " Mbit/s";
Chavoosh Ghasemi4d36ed52017-10-31 22:26:25 +0000152 case 3:
Davide Pesavento11fc3eb2024-01-26 01:46:56 -0500153 return std::to_string(throughput) + " Gbit/s";
Chavoosh Ghasemi4d36ed52017-10-31 22:26:25 +0000154 case 4:
Davide Pesavento11fc3eb2024-01-26 01:46:56 -0500155 return std::to_string(throughput) + " Tbit/s";
Chavoosh Ghasemi4d36ed52017-10-31 22:26:25 +0000156 }
157 return "";
158}
159
Davide Pesaventob3570c62022-02-19 19:19:00 -0500160} // namespace ndn::chunks