blob: 8535924c44a717d93ef4abf771f1faa2d070d055 [file] [log] [blame]
Alexander Afanasyevc169a812014-05-20 20:37:29 -04001/* -*- Mode:C++; c-file-style:"gnu"; indent-tabs-mode:nil; -*- */
Jeff Thompsonaa4e6db2013-07-15 17:25:23 -07002/**
Alexander Afanasyevc169a812014-05-20 20:37:29 -04003 * Copyright (c) 2013-2014 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.
Alexander Afanasyevdfa52c42014-04-24 21:10:11 -070020 *
21 * Based on code originally written by Jeff Thompson <jefft0@remap.ucla.edu>
Jeff Thompsonaa4e6db2013-07-15 17:25:23 -070022 */
23
Alexander Afanasyev09c613f2014-01-29 00:23:58 -080024#include "face.hpp"
Alexander Afanasyev258ec2b2014-05-14 16:15:37 -070025#include "detail/face-impl.hpp"
Jeff Thompsonaa4e6db2013-07-15 17:25:23 -070026
Junxiao Shi468abc32014-11-04 09:12:47 -070027#include "encoding/tlv.hpp"
Junxiao Shiedd834e2014-10-28 20:28:58 -070028#include "security/key-chain.hpp"
Alexander Afanasyev0222fba2014-02-09 23:16:02 -080029#include "util/time.hpp"
30#include "util/random.hpp"
Steve DiBenedettoa8659ff2014-12-04 14:50:28 -070031#include "util/face-uri.hpp"
Jeff Thompsonb982b6d2013-07-15 18:15:45 -070032
Jeff Thompsonaa4e6db2013-07-15 17:25:23 -070033namespace ndn {
Alexander Afanasyevb790d952014-01-24 12:07:53 -080034
Alexander Afanasyevf7ca3202014-02-14 22:28:31 -080035Face::Face()
Junxiao Shi2cced062014-11-02 21:27:38 -070036 : m_internalIoService(new boost::asio::io_service())
37 , m_ioService(*m_internalIoService)
38 , m_internalKeyChain(new KeyChain())
Alexander Afanasyevee8bb1e2014-05-02 17:39:54 -070039 , m_isDirectNfdFibManagementRequested(false)
Junxiao Shiedd834e2014-10-28 20:28:58 -070040 , m_impl(new Impl(*this))
Jeff Thompsonfb29cda2013-08-24 10:26:54 -070041{
Steve DiBenedettoa8659ff2014-12-04 14:50:28 -070042 construct(m_internalKeyChain);
Alexander Afanasyev0222fba2014-02-09 23:16:02 -080043}
44
Alexander Afanasyev691c3ce2014-04-23 14:28:04 -070045Face::Face(boost::asio::io_service& ioService)
Junxiao Shi2cced062014-11-02 21:27:38 -070046 : m_ioService(ioService)
47 , m_internalKeyChain(new KeyChain())
Alexander Afanasyevee8bb1e2014-05-02 17:39:54 -070048 , m_isDirectNfdFibManagementRequested(false)
Junxiao Shiedd834e2014-10-28 20:28:58 -070049 , m_impl(new Impl(*this))
Alexander Afanasyev691c3ce2014-04-23 14:28:04 -070050{
Steve DiBenedettoa8659ff2014-12-04 14:50:28 -070051 construct(m_internalKeyChain);
Alexander Afanasyev691c3ce2014-04-23 14:28:04 -070052}
53
Alexander Afanasyev7682ccb2014-02-20 10:29:35 -080054Face::Face(const std::string& host, const std::string& port/* = "6363"*/)
Junxiao Shi2cced062014-11-02 21:27:38 -070055 : m_internalIoService(new boost::asio::io_service())
56 , m_ioService(*m_internalIoService)
57 , m_internalKeyChain(new KeyChain())
Junxiao Shiedd834e2014-10-28 20:28:58 -070058 , m_impl(new Impl(*this))
Alexander Afanasyev0222fba2014-02-09 23:16:02 -080059{
Alexander Afanasyevf73f0632014-05-12 18:02:37 -070060 construct(make_shared<TcpTransport>(host, port),
Junxiao Shiedd834e2014-10-28 20:28:58 -070061 m_internalKeyChain);
Alexander Afanasyev0222fba2014-02-09 23:16:02 -080062}
63
Alexander Afanasyevf7ca3202014-02-14 22:28:31 -080064Face::Face(const shared_ptr<Transport>& transport)
Junxiao Shi2cced062014-11-02 21:27:38 -070065 : m_internalIoService(new boost::asio::io_service())
66 , m_ioService(*m_internalIoService)
67 , m_internalKeyChain(new KeyChain())
Alexander Afanasyevee8bb1e2014-05-02 17:39:54 -070068 , m_isDirectNfdFibManagementRequested(false)
Junxiao Shiedd834e2014-10-28 20:28:58 -070069 , m_impl(new Impl(*this))
Alexander Afanasyev0222fba2014-02-09 23:16:02 -080070{
71 construct(transport,
Junxiao Shiedd834e2014-10-28 20:28:58 -070072 m_internalKeyChain);
Alexander Afanasyev0222fba2014-02-09 23:16:02 -080073}
74
75Face::Face(const shared_ptr<Transport>& transport,
Alexander Afanasyev691c3ce2014-04-23 14:28:04 -070076 boost::asio::io_service& ioService)
Junxiao Shi2cced062014-11-02 21:27:38 -070077 : m_ioService(ioService)
78 , m_internalKeyChain(new KeyChain())
Alexander Afanasyevee8bb1e2014-05-02 17:39:54 -070079 , m_isDirectNfdFibManagementRequested(false)
Junxiao Shiedd834e2014-10-28 20:28:58 -070080 , m_impl(new Impl(*this))
Alexander Afanasyev0222fba2014-02-09 23:16:02 -080081{
Alexander Afanasyev691c3ce2014-04-23 14:28:04 -070082 construct(transport,
Junxiao Shiedd834e2014-10-28 20:28:58 -070083 m_internalKeyChain);
84}
85
86Face::Face(shared_ptr<Transport> transport,
87 boost::asio::io_service& ioService,
88 KeyChain& keyChain)
Junxiao Shi2cced062014-11-02 21:27:38 -070089 : m_ioService(ioService)
90 , m_internalKeyChain(nullptr)
Junxiao Shiedd834e2014-10-28 20:28:58 -070091 , m_isDirectNfdFibManagementRequested(false)
92 , m_impl(new Impl(*this))
93{
94 construct(transport,
Junxiao Shiedd834e2014-10-28 20:28:58 -070095 &keyChain);
Alexander Afanasyev7682ccb2014-02-20 10:29:35 -080096}
97
Alexander Afanasyev0222fba2014-02-09 23:16:02 -080098void
Steve DiBenedettoa8659ff2014-12-04 14:50:28 -070099Face::construct(KeyChain* keyChain)
100{
101 // transport=unix:///var/run/nfd.sock
102 // transport=tcp://localhost:6363
103
104 const ConfigFile::Parsed& parsed = m_impl->m_config.getParsedConfiguration();
105
106 const auto transportType = parsed.get_optional<std::string>("transport");
107 if (!transportType)
108 {
109 // transport not specified, use default Unix transport.
110 construct(UnixTransport::create(m_impl->m_config), keyChain);
111 return;
112 }
113
114 unique_ptr<util::FaceUri> uri;
115 try
116 {
117 uri.reset(new util::FaceUri(*transportType));
118 }
119 catch (const util::FaceUri::Error& error)
120 {
121 throw ConfigFile::Error(error.what());
122 }
123
124 shared_ptr<Transport> transport;
125 const std::string protocol = uri->getScheme();
126
127 if (protocol == "unix")
128 {
129 construct(UnixTransport::create(m_impl->m_config), keyChain);
130
131 }
132 else if (protocol == "tcp" || protocol == "tcp4" || protocol == "tcp6")
133 {
134 construct(TcpTransport::create(m_impl->m_config), keyChain);
135 }
136 else
137 {
138 throw ConfigFile::Error("Unsupported transport protocol \"" + protocol + "\"");
139 }
140}
141
142void
Junxiao Shiedd834e2014-10-28 20:28:58 -0700143Face::construct(shared_ptr<Transport> transport,
Junxiao Shiedd834e2014-10-28 20:28:58 -0700144 KeyChain* keyChain)
Alexander Afanasyev0222fba2014-02-09 23:16:02 -0800145{
Junxiao Shiedd834e2014-10-28 20:28:58 -0700146 m_nfdController = new nfd::Controller(*this, *keyChain);
147
Alexander Afanasyev258ec2b2014-05-14 16:15:37 -0700148 m_impl->m_pitTimeoutCheckTimerActive = false;
Alexander Afanasyevf39c5372014-02-17 19:42:56 -0800149 m_transport = transport;
Alexander Afanasyev0222fba2014-02-09 23:16:02 -0800150
Junxiao Shi2cced062014-11-02 21:27:38 -0700151 m_impl->m_pitTimeoutCheckTimer = make_shared<monotonic_deadline_timer>(ref(m_ioService));
152 m_impl->m_processEventsTimeoutTimer = make_shared<monotonic_deadline_timer>(ref(m_ioService));
Alexander Afanasyeva68aa7f2014-02-11 15:42:33 -0800153
Steve DiBenedettoc07b3a22014-03-19 12:32:52 -0600154 std::string protocol = "nrd-0.1";
155
156 try
Alexander Afanasyevefe3ab22014-02-19 14:57:50 -0800157 {
Alexander Afanasyev258ec2b2014-05-14 16:15:37 -0700158 protocol = m_impl->m_config.getParsedConfiguration().get<std::string>("protocol");
Steve DiBenedettoc07b3a22014-03-19 12:32:52 -0600159 }
Alexander Afanasyev2a7f7202014-04-23 14:25:29 -0700160 catch (boost::property_tree::ptree_bad_path& error)
Steve DiBenedettoc07b3a22014-03-19 12:32:52 -0600161 {
162 // protocol not specified
163 }
Alexander Afanasyev2a7f7202014-04-23 14:25:29 -0700164 catch (boost::property_tree::ptree_bad_data& error)
Steve DiBenedettoc07b3a22014-03-19 12:32:52 -0600165 {
166 throw ConfigFile::Error(error.what());
167 }
168
169 if (isSupportedNrdProtocol(protocol))
170 {
Alexander Afanasyevee8bb1e2014-05-02 17:39:54 -0700171 // do nothing
Steve DiBenedettoc07b3a22014-03-19 12:32:52 -0600172 }
Steve DiBenedettoacab8802014-03-24 11:15:57 -0600173 else if (isSupportedNfdProtocol(protocol))
Steve DiBenedettoc07b3a22014-03-19 12:32:52 -0600174 {
Alexander Afanasyevee8bb1e2014-05-02 17:39:54 -0700175 m_isDirectNfdFibManagementRequested = true;
Alexander Afanasyevefe3ab22014-02-19 14:57:50 -0800176 }
Jeff Thompsonfb29cda2013-08-24 10:26:54 -0700177 else
Steve DiBenedettoc07b3a22014-03-19 12:32:52 -0600178 {
179 throw Face::Error("Cannot create controller for unsupported protocol \"" + protocol + "\"");
180 }
Alexander Afanasyev0222fba2014-02-09 23:16:02 -0800181}
182
Junxiao Shiedd834e2014-10-28 20:28:58 -0700183Face::~Face()
184{
185 if (m_internalKeyChain != nullptr) {
186 delete m_internalKeyChain;
187 }
188
189 delete m_nfdController;
190 delete m_impl;
191}
192
Alexander Afanasyev0222fba2014-02-09 23:16:02 -0800193const PendingInterestId*
194Face::expressInterest(const Interest& interest, const OnData& onData, const OnTimeout& onTimeout)
195{
Alexander Afanasyevf73f0632014-05-12 18:02:37 -0700196 shared_ptr<Interest> interestToExpress = make_shared<Interest>(interest);
Alexander Afanasyeva68aa7f2014-02-11 15:42:33 -0800197
Alexander Afanasyev49bb1fb2014-07-21 12:54:01 -0700198 // Use `interestToExpress` to avoid wire format creation for the original Interest
199 if (interestToExpress->wireEncode().size() > MAX_NDN_PACKET_SIZE)
200 throw Error("Interest size exceeds maximum limit");
201
Alexander Afanasyev4e50b972014-03-25 10:57:50 -0700202 // If the same ioService thread, dispatch directly calls the method
Junxiao Shi2cced062014-11-02 21:27:38 -0700203 m_ioService.dispatch(bind(&Impl::asyncExpressInterest, m_impl,
204 interestToExpress, onData, onTimeout));
Alexander Afanasyeva68aa7f2014-02-11 15:42:33 -0800205
Alexander Afanasyev0222fba2014-02-09 23:16:02 -0800206 return reinterpret_cast<const PendingInterestId*>(interestToExpress.get());
207}
208
209const PendingInterestId*
210Face::expressInterest(const Name& name,
Alexander Afanasyev7682ccb2014-02-20 10:29:35 -0800211 const Interest& tmpl,
Alexander Afanasyev0222fba2014-02-09 23:16:02 -0800212 const OnData& onData, const OnTimeout& onTimeout/* = OnTimeout()*/)
213{
Alexander Afanasyev9c578182014-05-14 17:28:28 -0700214 return expressInterest(Interest(tmpl)
Steve DiBenedettoa8659ff2014-12-04 14:50:28 -0700215 .setName(name)
216 .setNonce(0),
Alexander Afanasyev0222fba2014-02-09 23:16:02 -0800217 onData, onTimeout);
218}
219
220void
Alexander Afanasyev7682ccb2014-02-20 10:29:35 -0800221Face::put(const Data& data)
Alexander Afanasyev0222fba2014-02-09 23:16:02 -0800222{
Alexander Afanasyev49bb1fb2014-07-21 12:54:01 -0700223 // Use original `data`, since wire format should already exist for the original Data
224 if (data.wireEncode().size() > MAX_NDN_PACKET_SIZE)
225 throw Error("Data size exceeds maximum limit");
226
Alexander Afanasyev6a05b4b2014-07-18 17:23:00 -0700227 shared_ptr<const Data> dataPtr;
228 try {
229 dataPtr = data.shared_from_this();
230 }
231 catch (const bad_weak_ptr& e) {
232 std::cerr << "Face::put WARNING: the supplied Data should be created using make_shared<Data>()"
233 << std::endl;
234 dataPtr = make_shared<Data>(data);
235 }
Alexander Afanasyev0222fba2014-02-09 23:16:02 -0800236
Alexander Afanasyev6a05b4b2014-07-18 17:23:00 -0700237 // If the same ioService thread, dispatch directly calls the method
Junxiao Shi2cced062014-11-02 21:27:38 -0700238 m_ioService.dispatch(bind(&Impl::asyncPutData, m_impl, dataPtr));
Alexander Afanasyev0222fba2014-02-09 23:16:02 -0800239}
240
Alexander Afanasyev0222fba2014-02-09 23:16:02 -0800241void
Alexander Afanasyev7682ccb2014-02-20 10:29:35 -0800242Face::removePendingInterest(const PendingInterestId* pendingInterestId)
Alexander Afanasyev0222fba2014-02-09 23:16:02 -0800243{
Junxiao Shi2cced062014-11-02 21:27:38 -0700244 m_ioService.post(bind(&Impl::asyncRemovePendingInterest, m_impl, pendingInterestId));
Alexander Afanasyev0222fba2014-02-09 23:16:02 -0800245}
246
Alexander Afanasyev6fcdde22014-08-22 19:03:36 -0700247size_t
248Face::getNPendingInterests() const
249{
250 return m_impl->m_pendingInterestTable.size();
251}
252
Alexander Afanasyev0222fba2014-02-09 23:16:02 -0800253const RegisteredPrefixId*
Alexander Afanasyev90164962014-03-06 08:29:59 +0000254Face::setInterestFilter(const InterestFilter& interestFilter,
Alexander Afanasyev0222fba2014-02-09 23:16:02 -0800255 const OnInterest& onInterest,
Alexander Afanasyev984ad192014-05-02 19:11:15 -0700256 const RegisterPrefixSuccessCallback& onSuccess,
257 const RegisterPrefixFailureCallback& onFailure,
Alexander Afanasyev0866f512014-08-11 13:25:09 -0700258 const IdentityCertificate& certificate,
259 uint64_t flags)
Yingdi Yue66bf2a2014-04-28 17:07:36 -0700260{
Alexander Afanasyev984ad192014-05-02 19:11:15 -0700261 shared_ptr<InterestFilterRecord> filter =
262 make_shared<InterestFilterRecord>(interestFilter, onInterest);
263
Junxiao Shi388ec252014-11-02 15:19:57 -0700264 nfd::CommandOptions options;
265 if (certificate.getName().empty()) {
266 options.setSigningDefault();
267 }
268 else {
269 options.setSigningCertificate(certificate);
270 }
271
Alexander Afanasyev258ec2b2014-05-14 16:15:37 -0700272 return m_impl->registerPrefix(interestFilter.getPrefix(), filter,
273 onSuccess, onFailure,
Junxiao Shi388ec252014-11-02 15:19:57 -0700274 flags, options);
Yingdi Yue66bf2a2014-04-28 17:07:36 -0700275}
276
277const RegisteredPrefixId*
Alexander Afanasyev90164962014-03-06 08:29:59 +0000278Face::setInterestFilter(const InterestFilter& interestFilter,
Yingdi Yue66bf2a2014-04-28 17:07:36 -0700279 const OnInterest& onInterest,
Alexander Afanasyev984ad192014-05-02 19:11:15 -0700280 const RegisterPrefixFailureCallback& onFailure,
Alexander Afanasyev0866f512014-08-11 13:25:09 -0700281 const IdentityCertificate& certificate,
282 uint64_t flags)
Alexander Afanasyev984ad192014-05-02 19:11:15 -0700283{
284 shared_ptr<InterestFilterRecord> filter =
285 make_shared<InterestFilterRecord>(interestFilter, onInterest);
286
Junxiao Shi388ec252014-11-02 15:19:57 -0700287 nfd::CommandOptions options;
288 if (certificate.getName().empty()) {
289 options.setSigningDefault();
290 }
291 else {
292 options.setSigningCertificate(certificate);
293 }
294
Alexander Afanasyev258ec2b2014-05-14 16:15:37 -0700295 return m_impl->registerPrefix(interestFilter.getPrefix(), filter,
296 RegisterPrefixSuccessCallback(), onFailure,
Junxiao Shi388ec252014-11-02 15:19:57 -0700297 flags, options);
Alexander Afanasyev984ad192014-05-02 19:11:15 -0700298}
299
300const RegisteredPrefixId*
301Face::setInterestFilter(const InterestFilter& interestFilter,
302 const OnInterest& onInterest,
303 const RegisterPrefixSuccessCallback& onSuccess,
304 const RegisterPrefixFailureCallback& onFailure,
Alexander Afanasyev0866f512014-08-11 13:25:09 -0700305 const Name& identity,
306 uint64_t flags)
Yingdi Yue66bf2a2014-04-28 17:07:36 -0700307{
Alexander Afanasyev984ad192014-05-02 19:11:15 -0700308 shared_ptr<InterestFilterRecord> filter =
309 make_shared<InterestFilterRecord>(interestFilter, onInterest);
310
Junxiao Shi388ec252014-11-02 15:19:57 -0700311 nfd::CommandOptions options;
312 options.setSigningIdentity(identity);
313
Alexander Afanasyev258ec2b2014-05-14 16:15:37 -0700314 return m_impl->registerPrefix(interestFilter.getPrefix(), filter,
315 onSuccess, onFailure,
Junxiao Shi388ec252014-11-02 15:19:57 -0700316 flags, options);
Alexander Afanasyev984ad192014-05-02 19:11:15 -0700317}
318
319const RegisteredPrefixId*
320Face::setInterestFilter(const InterestFilter& interestFilter,
321 const OnInterest& onInterest,
322 const RegisterPrefixFailureCallback& onFailure,
Alexander Afanasyev0866f512014-08-11 13:25:09 -0700323 const Name& identity,
324 uint64_t flags)
Alexander Afanasyev984ad192014-05-02 19:11:15 -0700325{
326 shared_ptr<InterestFilterRecord> filter =
327 make_shared<InterestFilterRecord>(interestFilter, onInterest);
328
Junxiao Shi388ec252014-11-02 15:19:57 -0700329 nfd::CommandOptions options;
330 options.setSigningIdentity(identity);
331
Alexander Afanasyev258ec2b2014-05-14 16:15:37 -0700332 return m_impl->registerPrefix(interestFilter.getPrefix(), filter,
333 RegisterPrefixSuccessCallback(), onFailure,
Junxiao Shi388ec252014-11-02 15:19:57 -0700334 flags, options);
Alexander Afanasyev984ad192014-05-02 19:11:15 -0700335}
336
337
338const InterestFilterId*
339Face::setInterestFilter(const InterestFilter& interestFilter,
340 const OnInterest& onInterest)
341{
342 shared_ptr<InterestFilterRecord> filter =
343 make_shared<InterestFilterRecord>(interestFilter, onInterest);
344
Alexander Afanasyev258ec2b2014-05-14 16:15:37 -0700345 getIoService().post(bind(&Impl::asyncSetInterestFilter, m_impl, filter));
Alexander Afanasyev984ad192014-05-02 19:11:15 -0700346
347 return reinterpret_cast<const InterestFilterId*>(filter.get());
Yingdi Yue66bf2a2014-04-28 17:07:36 -0700348}
349
Alexander Afanasyev984ad192014-05-02 19:11:15 -0700350const RegisteredPrefixId*
351Face::registerPrefix(const Name& prefix,
352 const RegisterPrefixSuccessCallback& onSuccess,
353 const RegisterPrefixFailureCallback& onFailure,
Alexander Afanasyev0866f512014-08-11 13:25:09 -0700354 const IdentityCertificate& certificate,
355 uint64_t flags)
Alexander Afanasyev984ad192014-05-02 19:11:15 -0700356{
Junxiao Shi388ec252014-11-02 15:19:57 -0700357 nfd::CommandOptions options;
358 if (certificate.getName().empty()) {
359 options.setSigningDefault();
360 }
361 else {
362 options.setSigningCertificate(certificate);
363 }
364
Alexander Afanasyev258ec2b2014-05-14 16:15:37 -0700365 return m_impl->registerPrefix(prefix, shared_ptr<InterestFilterRecord>(),
366 onSuccess, onFailure,
Junxiao Shi388ec252014-11-02 15:19:57 -0700367 flags, options);
Alexander Afanasyev984ad192014-05-02 19:11:15 -0700368}
369
370const RegisteredPrefixId*
371Face::registerPrefix(const Name& prefix,
372 const RegisterPrefixSuccessCallback& onSuccess,
373 const RegisterPrefixFailureCallback& onFailure,
Alexander Afanasyev0866f512014-08-11 13:25:09 -0700374 const Name& identity,
375 uint64_t flags)
Alexander Afanasyev984ad192014-05-02 19:11:15 -0700376{
Junxiao Shi388ec252014-11-02 15:19:57 -0700377 nfd::CommandOptions options;
378 options.setSigningIdentity(identity);
379
Alexander Afanasyev258ec2b2014-05-14 16:15:37 -0700380 return m_impl->registerPrefix(prefix, shared_ptr<InterestFilterRecord>(),
381 onSuccess, onFailure,
Junxiao Shi388ec252014-11-02 15:19:57 -0700382 flags, options);
Alexander Afanasyev984ad192014-05-02 19:11:15 -0700383}
384
Alexander Afanasyev984ad192014-05-02 19:11:15 -0700385void
Alexander Afanasyev7682ccb2014-02-20 10:29:35 -0800386Face::unsetInterestFilter(const RegisteredPrefixId* registeredPrefixId)
Alexander Afanasyev0222fba2014-02-09 23:16:02 -0800387{
Junxiao Shi2cced062014-11-02 21:27:38 -0700388 m_ioService.post(bind(&Impl::asyncUnregisterPrefix, m_impl, registeredPrefixId,
389 UnregisterPrefixSuccessCallback(), UnregisterPrefixFailureCallback()));
Alexander Afanasyev0222fba2014-02-09 23:16:02 -0800390}
391
Alexander Afanasyev258ec2b2014-05-14 16:15:37 -0700392void
393Face::unsetInterestFilter(const InterestFilterId* interestFilterId)
394{
Junxiao Shi2cced062014-11-02 21:27:38 -0700395 m_ioService.post(bind(&Impl::asyncUnsetInterestFilter, m_impl, interestFilterId));
Alexander Afanasyev258ec2b2014-05-14 16:15:37 -0700396}
Yingdi Yue66bf2a2014-04-28 17:07:36 -0700397
398void
Alexander Afanasyev984ad192014-05-02 19:11:15 -0700399Face::unregisterPrefix(const RegisteredPrefixId* registeredPrefixId,
400 const UnregisterPrefixSuccessCallback& onSuccess,
401 const UnregisterPrefixFailureCallback& onFailure)
402{
Junxiao Shi2cced062014-11-02 21:27:38 -0700403 m_ioService.post(bind(&Impl::asyncUnregisterPrefix, m_impl, registeredPrefixId,
404 onSuccess, onFailure));
Alexander Afanasyev984ad192014-05-02 19:11:15 -0700405}
406
Alexander Afanasyeva68aa7f2014-02-11 15:42:33 -0800407void
Alexander Afanasyevaa0e7da2014-03-17 14:37:33 -0700408Face::processEvents(const time::milliseconds& timeout/* = time::milliseconds::zero()*/,
409 bool keepThread/* = false*/)
Alexander Afanasyev0222fba2014-02-09 23:16:02 -0800410{
Alexander Afanasyev8e158542014-11-18 00:47:18 -0500411 if (m_ioService.stopped()) {
412 m_ioService.reset(); // ensure that run()/poll() will do some work
413 }
414
Alexander Afanasyev49bb1fb2014-07-21 12:54:01 -0700415 try {
416 if (timeout < time::milliseconds::zero())
417 {
418 // do not block if timeout is negative, but process pending events
Junxiao Shi2cced062014-11-02 21:27:38 -0700419 m_ioService.poll();
Alexander Afanasyev49bb1fb2014-07-21 12:54:01 -0700420 return;
Alexander Afanasyev0222fba2014-02-09 23:16:02 -0800421 }
Alexander Afanasyeva68aa7f2014-02-11 15:42:33 -0800422
Alexander Afanasyev49bb1fb2014-07-21 12:54:01 -0700423 if (timeout > time::milliseconds::zero())
424 {
425 m_impl->m_processEventsTimeoutTimer->expires_from_now(time::milliseconds(timeout));
426 m_impl->m_processEventsTimeoutTimer->async_wait(&fireProcessEventsTimeout);
427 }
428
429 if (keepThread) {
430 // work will ensure that m_ioService is running until work object exists
Junxiao Shi2cced062014-11-02 21:27:38 -0700431 m_impl->m_ioServiceWork = make_shared<boost::asio::io_service::work>(ref(m_ioService));
Alexander Afanasyev0222fba2014-02-09 23:16:02 -0800432 }
Alexander Afanasyev49bb1fb2014-07-21 12:54:01 -0700433
Junxiao Shi2cced062014-11-02 21:27:38 -0700434 m_ioService.run();
Alexander Afanasyev49bb1fb2014-07-21 12:54:01 -0700435 }
436 catch (Face::ProcessEventsTimeout&) {
437 // break
438 m_impl->m_ioServiceWork.reset();
Alexander Afanasyev49bb1fb2014-07-21 12:54:01 -0700439 }
440 catch (...) {
441 m_impl->m_ioServiceWork.reset();
Alexander Afanasyev49bb1fb2014-07-21 12:54:01 -0700442 m_impl->m_pendingInterestTable.clear();
443 m_impl->m_registeredPrefixTable.clear();
444 throw;
445 }
Jeff Thompsonfb29cda2013-08-24 10:26:54 -0700446}
447
Alexander Afanasyeva68aa7f2014-02-11 15:42:33 -0800448void
Jeff Thompson0050abe2013-09-17 12:50:25 -0700449Face::shutdown()
Jeff Thompson517ffa82013-08-05 16:04:34 -0700450{
Junxiao Shi2cced062014-11-02 21:27:38 -0700451 m_ioService.post(bind(&Face::asyncShutdown, this));
Alexander Afanasyev7dced462014-03-19 15:12:32 -0700452}
453
454void
455Face::asyncShutdown()
456{
Alexander Afanasyev258ec2b2014-05-14 16:15:37 -0700457 m_impl->m_pendingInterestTable.clear();
458 m_impl->m_registeredPrefixTable.clear();
Alexander Afanasyev0222fba2014-02-09 23:16:02 -0800459
Alexander Afanasyev984ad192014-05-02 19:11:15 -0700460 if (m_transport->isConnected())
461 m_transport->close();
462
Alexander Afanasyev258ec2b2014-05-14 16:15:37 -0700463 m_impl->m_pitTimeoutCheckTimer->cancel();
464 m_impl->m_processEventsTimeoutTimer->cancel();
465 m_impl->m_pitTimeoutCheckTimerActive = false;
Alexander Afanasyev1f5486e2014-07-10 17:45:49 -0700466
467 m_impl->m_ioServiceWork.reset();
Jeff Thompsonaa4e6db2013-07-15 17:25:23 -0700468}
469
Alexander Afanasyev0222fba2014-02-09 23:16:02 -0800470void
471Face::fireProcessEventsTimeout(const boost::system::error_code& error)
472{
473 if (!error) // can fire for some other reason, e.g., cancelled
474 throw Face::ProcessEventsTimeout();
Jeff Thompsonaa4e6db2013-07-15 17:25:23 -0700475}
Alexander Afanasyev0222fba2014-02-09 23:16:02 -0800476
Alexander Afanasyev0222fba2014-02-09 23:16:02 -0800477
Alexander Afanasyeva68aa7f2014-02-11 15:42:33 -0800478void
Alexander Afanasyev6d48bc12014-02-18 00:10:51 -0800479Face::onReceiveElement(const Block& blockFromDaemon)
Alexander Afanasyev0222fba2014-02-09 23:16:02 -0800480{
Alexander Afanasyev6d48bc12014-02-18 00:10:51 -0800481 const Block& block = nfd::LocalControlHeader::getPayload(blockFromDaemon);
482
Steve DiBenedetto54ce6682014-07-22 13:22:57 -0600483 if (block.type() == tlv::Interest)
Alexander Afanasyev0222fba2014-02-09 23:16:02 -0800484 {
Alexander Afanasyevf73f0632014-05-12 18:02:37 -0700485 shared_ptr<Interest> interest = make_shared<Interest>();
Alexander Afanasyev0222fba2014-02-09 23:16:02 -0800486 interest->wireDecode(block);
Alexander Afanasyev6d48bc12014-02-18 00:10:51 -0800487 if (&block != &blockFromDaemon)
488 interest->getLocalControlHeader().wireDecode(blockFromDaemon);
Alexander Afanasyeva68aa7f2014-02-11 15:42:33 -0800489
Alexander Afanasyev258ec2b2014-05-14 16:15:37 -0700490 m_impl->processInterestFilters(*interest);
Alexander Afanasyev0222fba2014-02-09 23:16:02 -0800491 }
Steve DiBenedetto54ce6682014-07-22 13:22:57 -0600492 else if (block.type() == tlv::Data)
Alexander Afanasyev0222fba2014-02-09 23:16:02 -0800493 {
Alexander Afanasyevf73f0632014-05-12 18:02:37 -0700494 shared_ptr<Data> data = make_shared<Data>();
Alexander Afanasyev0222fba2014-02-09 23:16:02 -0800495 data->wireDecode(block);
Alexander Afanasyev6d48bc12014-02-18 00:10:51 -0800496 if (&block != &blockFromDaemon)
497 data->getLocalControlHeader().wireDecode(blockFromDaemon);
Alexander Afanasyev0222fba2014-02-09 23:16:02 -0800498
Alexander Afanasyev258ec2b2014-05-14 16:15:37 -0700499 m_impl->satisfyPendingInterests(*data);
Alexander Afanasyev0222fba2014-02-09 23:16:02 -0800500
Alexander Afanasyev258ec2b2014-05-14 16:15:37 -0700501 if (m_impl->m_pendingInterestTable.empty()) {
502 m_impl->m_pitTimeoutCheckTimer->cancel(); // this will cause checkPitExpire invocation
Alexander Afanasyev0222fba2014-02-09 23:16:02 -0800503 }
504 }
Yingdi Yuf9fa52f2014-02-06 12:27:32 -0800505 // ignore any other type
Alexander Afanasyev0222fba2014-02-09 23:16:02 -0800506}
507
Alexander Afanasyev0222fba2014-02-09 23:16:02 -0800508
Alexander Afanasyev0222fba2014-02-09 23:16:02 -0800509
510} // namespace ndn