blob: 09a1277d6e99fa7d37aac04312cbb3b7d04adff9 [file] [log] [blame]
Junxiao Shi18244e82016-12-03 15:42:01 +00001/* -*- Mode:C++; c-file-style:"gnu"; indent-tabs-mode:nil; -*- */
2/**
3 * Copyright (c) 2013-2016 Regents of the University of California.
4 *
5 * This file is part of ndn-cxx library (NDN C++ library with eXperimental eXtensions).
6 *
7 * 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.
20 */
21
22/**
23 * Original copyright notice from NFD:
24 *
25 * Copyright (c) 2014, Regents of the University of California,
26 * Arizona Board of Regents,
27 * Colorado State University,
28 * University Pierre & Marie Curie, Sorbonne University,
29 * Washington University in St. Louis,
30 * Beijing Institute of Technology,
31 * The University of Memphis
32 *
33 * This file is part of NFD (Named Data Networking Forwarding Daemon).
34 * See AUTHORS.md for complete list of NFD authors and contributors.
35 *
36 * NFD is free software: you can redistribute it and/or modify it under the terms
37 * of the GNU General Public License as published by the Free Software Foundation,
38 * either version 3 of the License, or (at your option) any later version.
39 *
40 * NFD is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY;
41 * without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR
42 * PURPOSE. See the GNU General Public License for more details.
43 *
44 * You should have received a copy of the GNU General Public License along with
45 * NFD, e.g., in COPYING.md file. If not, see <http://www.gnu.org/licenses/>.
46 */
47
48#include "notification-subscriber.hpp"
49#include "random.hpp"
50
51namespace ndn {
52namespace util {
53
54NotificationSubscriberBase::NotificationSubscriberBase(Face& face, const Name& prefix,
55 time::milliseconds interestLifetime)
56 : m_face(face)
57 , m_prefix(prefix)
58 , m_isRunning(false)
59 , m_lastSequenceNo(std::numeric_limits<uint64_t>::max())
60 , m_lastNackSequenceNo(std::numeric_limits<uint64_t>::max())
61 , m_attempts(1)
62 , m_scheduler(face.getIoService())
63 , m_nackEvent(m_scheduler)
64 , m_interestLifetime(interestLifetime)
65{
66}
67
68NotificationSubscriberBase::~NotificationSubscriberBase() = default;
69
70void
71NotificationSubscriberBase::start()
72{
73 if (m_isRunning) // already running
74 return;
75 m_isRunning = true;
76
77 this->sendInitialInterest();
78}
79
80void
81NotificationSubscriberBase::stop()
82{
83 if (!m_isRunning) // not running
84 return;
85 m_isRunning = false;
86
87 if (m_lastInterestId != 0)
88 m_face.removePendingInterest(m_lastInterestId);
89 m_lastInterestId = 0;
90}
91
92void
93NotificationSubscriberBase::sendInitialInterest()
94{
95 if (this->shouldStop())
96 return;
97
98 auto interest = make_shared<Interest>(m_prefix);
99 interest->setMustBeFresh(true);
100 interest->setChildSelector(1);
101 interest->setInterestLifetime(getInterestLifetime());
102
103 m_lastInterestId = m_face.expressInterest(*interest,
104 bind(&NotificationSubscriberBase::afterReceiveData, this, _2),
105 bind(&NotificationSubscriberBase::afterReceiveNack, this, _2),
106 bind(&NotificationSubscriberBase::afterTimeout, this));
107}
108
109void
110NotificationSubscriberBase::sendNextInterest()
111{
112 if (this->shouldStop())
113 return;
114
115 BOOST_ASSERT(m_lastSequenceNo != std::numeric_limits<uint64_t>::max()); // overflow or missing initial reply
116
117 Name nextName = m_prefix;
118 nextName.appendSequenceNumber(m_lastSequenceNo + 1);
119
120 auto interest = make_shared<Interest>(nextName);
121 interest->setInterestLifetime(getInterestLifetime());
122
123 m_lastInterestId = m_face.expressInterest(*interest,
124 bind(&NotificationSubscriberBase::afterReceiveData, this, _2),
125 bind(&NotificationSubscriberBase::afterReceiveNack, this, _2),
126 bind(&NotificationSubscriberBase::afterTimeout, this));
127}
128
129bool
130NotificationSubscriberBase::shouldStop()
131{
132 if (!m_isRunning)
133 return true;
134 if (!this->hasSubscriber() && onNack.isEmpty()) {
135 this->stop();
136 return true;
137 }
138 return false;
139}
140
141void
142NotificationSubscriberBase::afterReceiveData(const Data& data)
143{
144 if (this->shouldStop())
145 return;
146
147 try {
148 m_lastSequenceNo = data.getName().get(-1).toSequenceNumber();
149 }
150 catch (const tlv::Error&) {
151 this->onDecodeError(data);
152 this->sendInitialInterest();
153 return;
154 }
155
156 if (!this->decodeAndDeliver(data)) {
157 this->onDecodeError(data);
158 this->sendInitialInterest();
159 return;
160 }
161
162 this->sendNextInterest();
163}
164
165void
166NotificationSubscriberBase::afterReceiveNack(const lp::Nack& nack)
167{
168 if (this->shouldStop())
169 return;
170
171 this->onNack(nack);
172
173 time::milliseconds delay = exponentialBackoff(nack);
174 m_nackEvent = m_scheduler.scheduleEvent(delay, [this] {this->sendInitialInterest();});
175}
176
177void
178NotificationSubscriberBase::afterTimeout()
179{
180 if (this->shouldStop())
181 return;
182
183 this->onTimeout();
184
185 this->sendInitialInterest();
186}
187
188time::milliseconds
189NotificationSubscriberBase::exponentialBackoff(lp::Nack nack)
190{
191 uint64_t nackSequenceNo;
192
193 try {
194 nackSequenceNo = nack.getInterest().getName().get(-1).toSequenceNumber();
195 }
196 catch (const tlv::Error&) {
197 nackSequenceNo = 0;
198 }
199
200 if (m_lastNackSequenceNo == nackSequenceNo) {
201 ++m_attempts;
202 }
203 else {
204 m_attempts = 1;
205 }
206
207 m_lastNackSequenceNo = nackSequenceNo;
208
209 return time::milliseconds(static_cast<uint32_t>(pow(2, m_attempts) * 100 +
210 random::generateWord32() % 100));
211}
212
213} // namespace util
214} // namespace ndn