blob: 2be2e0b3a0d6b320e3eb0a8ad54f81dac81c9f13 [file] [log] [blame]
Yanbiao Li73860e32015-08-19 16:30:16 -07001/* -*- Mode:C++; c-file-style:"gnu"; indent-tabs-mode:nil; -*- */
Eric Newberry944f38b2017-07-20 20:54:22 -04002/*
Davide Pesavento15b55052018-01-27 19:09:28 -05003 * Copyright (c) 2014-2018, Regents of the University of California,
Yanbiao Li73860e32015-08-19 16:30:16 -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
26#include "face-manager.hpp"
Junxiao Shi40cb61c2015-09-23 18:36:45 -070027#include "face/generic-link-service.hpp"
Junxiao Shib8590312016-12-29 21:22:25 +000028#include "face/protocol-factory.hpp"
Yukai Tu0a49d342015-09-13 12:54:22 +080029#include "fw/face-table.hpp"
Yanbiao Li73860e32015-08-19 16:30:16 -070030
Eric Newberry0c3e57b2018-01-25 20:54:46 -070031#include <boost/logic/tribool.hpp>
32
Junxiao Shicbc8e942016-09-06 03:17:45 +000033#include <ndn-cxx/lp/tags.hpp>
Junxiao Shi25c6ce42016-09-09 13:49:59 +000034#include <ndn-cxx/mgmt/nfd/channel-status.hpp>
Yanbiao Li73860e32015-08-19 16:30:16 -070035
Yanbiao Li73860e32015-08-19 16:30:16 -070036namespace nfd {
37
38NFD_LOG_INIT("FaceManager");
39
Junxiao Shiea47bde2017-01-26 17:49:16 +000040FaceManager::FaceManager(FaceSystem& faceSystem,
41 Dispatcher& dispatcher, CommandAuthenticator& authenticator)
Junxiao Shi9ddf1b52016-08-22 03:58:55 +000042 : NfdManagerBase(dispatcher, authenticator, "faces")
Junxiao Shiea47bde2017-01-26 17:49:16 +000043 , m_faceSystem(faceSystem)
44 , m_faceTable(faceSystem.getFaceTable())
Yanbiao Li73860e32015-08-19 16:30:16 -070045{
Yanbiao Li58ba3f92017-02-15 14:27:18 +000046 // register handlers for ControlCommand
Yanbiao Li73860e32015-08-19 16:30:16 -070047 registerCommandHandler<ndn::nfd::FaceCreateCommand>("create",
48 bind(&FaceManager::createFace, this, _2, _3, _4, _5));
49
Eric Newberryb5aa7f52016-09-03 20:36:12 -070050 registerCommandHandler<ndn::nfd::FaceUpdateCommand>("update",
51 bind(&FaceManager::updateFace, this, _2, _3, _4, _5));
52
Yanbiao Li73860e32015-08-19 16:30:16 -070053 registerCommandHandler<ndn::nfd::FaceDestroyCommand>("destroy",
54 bind(&FaceManager::destroyFace, this, _2, _3, _4, _5));
55
Yanbiao Li58ba3f92017-02-15 14:27:18 +000056 // register handlers for StatusDataset
Yanbiao Li73860e32015-08-19 16:30:16 -070057 registerStatusDatasetHandler("list", bind(&FaceManager::listFaces, this, _1, _2, _3));
58 registerStatusDatasetHandler("channels", bind(&FaceManager::listChannels, this, _1, _2, _3));
59 registerStatusDatasetHandler("query", bind(&FaceManager::queryFaces, this, _1, _2, _3));
60
Yanbiao Li58ba3f92017-02-15 14:27:18 +000061 // register notification stream
Junxiao Shiae04d342016-07-19 13:20:22 +000062 m_postNotification = registerNotificationStream("events");
Eric Newberry1b4ba052016-10-07 23:04:07 -070063 m_faceAddConn = m_faceTable.afterAdd.connect([this] (const Face& face) {
64 connectFaceStateChangeSignal(face);
65 notifyFaceEvent(face, ndn::nfd::FACE_EVENT_CREATED);
66 });
67 m_faceRemoveConn = m_faceTable.beforeRemove.connect([this] (const Face& face) {
68 notifyFaceEvent(face, ndn::nfd::FACE_EVENT_DESTROYED);
69 });
Yanbiao Li73860e32015-08-19 16:30:16 -070070}
71
72void
73FaceManager::setConfigFile(ConfigFile& configFile)
74{
Junxiao Shib8590312016-12-29 21:22:25 +000075 m_faceSystem.setConfigFile(configFile);
Yanbiao Li73860e32015-08-19 16:30:16 -070076}
77
78void
79FaceManager::createFace(const Name& topPrefix, const Interest& interest,
80 const ControlParameters& parameters,
81 const ndn::mgmt::CommandContinuation& done)
82{
Eric Newberry78e32b02017-04-01 14:34:44 +000083 FaceUri remoteUri;
84 if (!remoteUri.parse(parameters.getUri())) {
85 NFD_LOG_TRACE("failed to parse remote URI: " << parameters.getUri());
Eric Newberry42602412016-08-27 09:33:18 -070086 done(ControlResponse(400, "Malformed command"));
87 return;
Yanbiao Li73860e32015-08-19 16:30:16 -070088 }
89
Eric Newberry78e32b02017-04-01 14:34:44 +000090 if (!remoteUri.isCanonical()) {
91 NFD_LOG_TRACE("received non-canonical remote URI: " << remoteUri.toString());
92 done(ControlResponse(400, "Non-canonical remote URI"));
Eric Newberry42602412016-08-27 09:33:18 -070093 return;
Yanbiao Li73860e32015-08-19 16:30:16 -070094 }
95
Eric Newberry78e32b02017-04-01 14:34:44 +000096 ndn::optional<FaceUri> localUri;
97 if (parameters.hasLocalUri()) {
98 localUri = FaceUri{};
99
100 if (!localUri->parse(parameters.getLocalUri())) {
101 NFD_LOG_TRACE("failed to parse local URI: " << parameters.getLocalUri());
102 done(ControlResponse(400, "Malformed command"));
103 return;
104 }
105
106 if (!localUri->isCanonical()) {
107 NFD_LOG_TRACE("received non-canonical local URI: " << localUri->toString());
108 done(ControlResponse(400, "Non-canonical local URI"));
109 return;
110 }
111 }
112
Davide Pesaventoe5eebad2017-04-06 20:23:26 -0400113 face::ProtocolFactory* factory = m_faceSystem.getFactoryByScheme(remoteUri.getScheme());
Junxiao Shib8590312016-12-29 21:22:25 +0000114 if (factory == nullptr) {
Eric Newberry78e32b02017-04-01 14:34:44 +0000115 NFD_LOG_TRACE("received create request for unsupported protocol: " << remoteUri.getScheme());
Eric Newberry42602412016-08-27 09:33:18 -0700116 done(ControlResponse(406, "Unsupported protocol"));
117 return;
Yanbiao Li73860e32015-08-19 16:30:16 -0700118 }
119
Davide Pesavento15b55052018-01-27 19:09:28 -0500120 face::FaceParams faceParams;
121 faceParams.persistency = parameters.getFacePersistency();
Eric Newberry0c3e57b2018-01-25 20:54:46 -0700122 if (parameters.hasBaseCongestionMarkingInterval()) {
123 faceParams.baseCongestionMarkingInterval = parameters.getBaseCongestionMarkingInterval();
124 }
125 if (parameters.hasDefaultCongestionThreshold()) {
126 faceParams.defaultCongestionThreshold = parameters.getDefaultCongestionThreshold();
127 }
Davide Pesavento15b55052018-01-27 19:09:28 -0500128 faceParams.wantLocalFields = parameters.hasFlagBit(ndn::nfd::BIT_LOCAL_FIELDS_ENABLED) &&
129 parameters.getFlagBit(ndn::nfd::BIT_LOCAL_FIELDS_ENABLED);
130 faceParams.wantLpReliability = parameters.hasFlagBit(ndn::nfd::BIT_LP_RELIABILITY_ENABLED) &&
131 parameters.getFlagBit(ndn::nfd::BIT_LP_RELIABILITY_ENABLED);
Eric Newberry0c3e57b2018-01-25 20:54:46 -0700132 if (parameters.hasFlagBit(ndn::nfd::BIT_CONGESTION_MARKING_ENABLED)) {
133 faceParams.wantCongestionMarking = parameters.getFlagBit(ndn::nfd::BIT_CONGESTION_MARKING_ENABLED);
134 }
Yanbiao Li73860e32015-08-19 16:30:16 -0700135 try {
Davide Pesavento15b55052018-01-27 19:09:28 -0500136 factory->createFace({remoteUri, localUri, faceParams},
137 bind(&FaceManager::afterCreateFaceSuccess, this, parameters, _1, done),
138 bind(&FaceManager::afterCreateFaceFailure, this, _1, _2, done));
Yanbiao Li73860e32015-08-19 16:30:16 -0700139 }
140 catch (const std::runtime_error& error) {
Eric Newberry42602412016-08-27 09:33:18 -0700141 NFD_LOG_ERROR("Face creation failed: " << error.what());
142 done(ControlResponse(500, "Face creation failed due to internal error"));
143 return;
Yanbiao Li73860e32015-08-19 16:30:16 -0700144 }
145 catch (const std::logic_error& error) {
Eric Newberry42602412016-08-27 09:33:18 -0700146 NFD_LOG_ERROR("Face creation failed: " << error.what());
147 done(ControlResponse(500, "Face creation failed due to internal error"));
148 return;
Yanbiao Li73860e32015-08-19 16:30:16 -0700149 }
150}
151
152void
Eric Newberry42602412016-08-27 09:33:18 -0700153FaceManager::afterCreateFaceSuccess(const ControlParameters& parameters,
Eric Newberryf40551a2016-09-05 15:41:16 -0700154 const shared_ptr<Face>& face,
Yanbiao Li73860e32015-08-19 16:30:16 -0700155 const ndn::mgmt::CommandContinuation& done)
156{
Yanbiao Li58ba3f92017-02-15 14:27:18 +0000157 if (face->getId() != face::INVALID_FACEID) {// Face already exists
158 NFD_LOG_TRACE("Attempted to create duplicate face of " << face->getId());
159
Junxiao Shi1cce2a32017-05-02 02:39:55 +0000160 ControlParameters response = collectFaceProperties(*face, true);
Yanbiao Li58ba3f92017-02-15 14:27:18 +0000161 done(ControlResponse(409, "Face with remote URI already exists").setBody(response.wireEncode()));
162 return;
163 }
164
Eric Newberryf40551a2016-09-05 15:41:16 -0700165 // If scope non-local and flags set to enable local fields, request shouldn't
166 // have made it this far
167 BOOST_ASSERT(face->getScope() == ndn::nfd::FACE_SCOPE_LOCAL ||
168 !parameters.hasFlagBit(ndn::nfd::BIT_LOCAL_FIELDS_ENABLED) ||
169 (parameters.hasFlagBit(ndn::nfd::BIT_LOCAL_FIELDS_ENABLED) &&
170 !parameters.getFlagBit(ndn::nfd::BIT_LOCAL_FIELDS_ENABLED)));
171
172 m_faceTable.add(face);
173
Junxiao Shi1cce2a32017-05-02 02:39:55 +0000174 ControlParameters response = collectFaceProperties(*face, true);
Eric Newberry42602412016-08-27 09:33:18 -0700175 done(ControlResponse(200, "OK").setBody(response.wireEncode()));
176}
177
178void
179FaceManager::afterCreateFaceFailure(uint32_t status,
180 const std::string& reason,
181 const ndn::mgmt::CommandContinuation& done)
182{
183 NFD_LOG_DEBUG("Face creation failed: " << reason);
184
185 done(ControlResponse(status, reason));
Yanbiao Li73860e32015-08-19 16:30:16 -0700186}
187
188void
Eric Newberryb5aa7f52016-09-03 20:36:12 -0700189FaceManager::updateFace(const Name& topPrefix, const Interest& interest,
190 const ControlParameters& parameters,
191 const ndn::mgmt::CommandContinuation& done)
192{
193 FaceId faceId = parameters.getFaceId();
194 if (faceId == 0) {
195 // Self-updating
196 shared_ptr<lp::IncomingFaceIdTag> incomingFaceIdTag = interest.getTag<lp::IncomingFaceIdTag>();
197 if (incomingFaceIdTag == nullptr) {
198 NFD_LOG_TRACE("unable to determine face for self-update");
199 done(ControlResponse(404, "No FaceId specified and IncomingFaceId not available"));
200 return;
201 }
202 faceId = *incomingFaceIdTag;
203 }
204
205 Face* face = m_faceTable.get(faceId);
206
207 if (face == nullptr) {
208 NFD_LOG_TRACE("invalid face specified");
209 done(ControlResponse(404, "Specified face does not exist"));
210 return;
211 }
212
213 // Verify validity of requested changes
214 ControlParameters response;
215 bool areParamsValid = true;
216
Eric Newberryb5aa7f52016-09-03 20:36:12 -0700217 if (parameters.hasFlagBit(ndn::nfd::BIT_LOCAL_FIELDS_ENABLED) &&
218 parameters.getFlagBit(ndn::nfd::BIT_LOCAL_FIELDS_ENABLED) &&
219 face->getScope() != ndn::nfd::FACE_SCOPE_LOCAL) {
220 NFD_LOG_TRACE("received request to enable local fields on non-local face");
221 areParamsValid = false;
222 response.setFlagBit(ndn::nfd::BIT_LOCAL_FIELDS_ENABLED,
223 parameters.getFlagBit(ndn::nfd::BIT_LOCAL_FIELDS_ENABLED));
224 }
225
Yanbiao Li58ba3f92017-02-15 14:27:18 +0000226 // check whether the requested FacePersistency change is valid if it's present
227 if (parameters.hasFacePersistency()) {
228 auto persistency = parameters.getFacePersistency();
229 if (!face->getTransport()->canChangePersistencyTo(persistency)) {
230 NFD_LOG_TRACE("cannot change face persistency to " << persistency);
231 areParamsValid = false;
232 response.setFacePersistency(persistency);
233 }
234 }
235
Eric Newberryb5aa7f52016-09-03 20:36:12 -0700236 if (!areParamsValid) {
237 done(ControlResponse(409, "Invalid properties specified").setBody(response.wireEncode()));
238 return;
239 }
240
241 // All specified properties are valid, so make changes
Yanbiao Li58ba3f92017-02-15 14:27:18 +0000242 if (parameters.hasFacePersistency()) {
243 face->setPersistency(parameters.getFacePersistency());
244 }
Eric Newberry2642cd22017-07-13 21:34:53 -0400245 setLinkServiceOptions(*face, parameters);
Eric Newberryb5aa7f52016-09-03 20:36:12 -0700246
Eric Newberryf40551a2016-09-05 15:41:16 -0700247 // Set ControlResponse fields
Junxiao Shi1cce2a32017-05-02 02:39:55 +0000248 response = collectFaceProperties(*face, false);
Eric Newberryb5aa7f52016-09-03 20:36:12 -0700249
250 done(ControlResponse(200, "OK").setBody(response.wireEncode()));
251}
252
253void
Junxiao Shi40cb61c2015-09-23 18:36:45 -0700254FaceManager::destroyFace(const Name& topPrefix, const Interest& interest,
255 const ControlParameters& parameters,
256 const ndn::mgmt::CommandContinuation& done)
257{
Junxiao Shi5b43f9a2016-07-19 13:15:56 +0000258 Face* face = m_faceTable.get(parameters.getFaceId());
259 if (face != nullptr) {
260 face->close();
Junxiao Shi40cb61c2015-09-23 18:36:45 -0700261 }
262
263 done(ControlResponse(200, "OK").setBody(parameters.wireEncode()));
264}
265
266void
Eric Newberryb5aa7f52016-09-03 20:36:12 -0700267FaceManager::setLinkServiceOptions(Face& face,
Eric Newberry2642cd22017-07-13 21:34:53 -0400268 const ControlParameters& parameters)
Eric Newberryb5aa7f52016-09-03 20:36:12 -0700269{
270 auto linkService = dynamic_cast<face::GenericLinkService*>(face.getLinkService());
271 BOOST_ASSERT(linkService != nullptr);
272
273 auto options = linkService->getOptions();
274 if (parameters.hasFlagBit(ndn::nfd::BIT_LOCAL_FIELDS_ENABLED) &&
275 face.getScope() == ndn::nfd::FACE_SCOPE_LOCAL) {
276 options.allowLocalFields = parameters.getFlagBit(ndn::nfd::BIT_LOCAL_FIELDS_ENABLED);
277 }
Eric Newberry2642cd22017-07-13 21:34:53 -0400278 if (parameters.hasFlagBit(ndn::nfd::BIT_LP_RELIABILITY_ENABLED)) {
279 options.reliabilityOptions.isEnabled = parameters.getFlagBit(ndn::nfd::BIT_LP_RELIABILITY_ENABLED);
280 }
Eric Newberry0c3e57b2018-01-25 20:54:46 -0700281 if (parameters.hasFlagBit(ndn::nfd::BIT_CONGESTION_MARKING_ENABLED)) {
282 options.allowCongestionMarking = parameters.getFlagBit(ndn::nfd::BIT_CONGESTION_MARKING_ENABLED);
283 }
284 if (parameters.hasBaseCongestionMarkingInterval()) {
285 options.baseCongestionMarkingInterval = parameters.getBaseCongestionMarkingInterval();
286 }
287 if (parameters.hasDefaultCongestionThreshold()) {
288 options.defaultCongestionThreshold = parameters.getDefaultCongestionThreshold();
289 }
Eric Newberryb5aa7f52016-09-03 20:36:12 -0700290 linkService->setOptions(options);
Eric Newberryb5aa7f52016-09-03 20:36:12 -0700291}
292
Yanbiao Li58ba3f92017-02-15 14:27:18 +0000293ControlParameters
Junxiao Shi1cce2a32017-05-02 02:39:55 +0000294FaceManager::collectFaceProperties(const Face& face, bool wantUris)
Yanbiao Li58ba3f92017-02-15 14:27:18 +0000295{
296 auto linkService = dynamic_cast<face::GenericLinkService*>(face.getLinkService());
297 BOOST_ASSERT(linkService != nullptr);
298 auto options = linkService->getOptions();
299
Junxiao Shi1cce2a32017-05-02 02:39:55 +0000300 ControlParameters params;
301 params.setFaceId(face.getId())
302 .setFacePersistency(face.getPersistency())
Eric Newberry0c3e57b2018-01-25 20:54:46 -0700303 .setBaseCongestionMarkingInterval(options.baseCongestionMarkingInterval)
304 .setDefaultCongestionThreshold(options.defaultCongestionThreshold)
Eric Newberry2642cd22017-07-13 21:34:53 -0400305 .setFlagBit(ndn::nfd::BIT_LOCAL_FIELDS_ENABLED, options.allowLocalFields, false)
Eric Newberry0c3e57b2018-01-25 20:54:46 -0700306 .setFlagBit(ndn::nfd::BIT_LP_RELIABILITY_ENABLED, options.reliabilityOptions.isEnabled, false)
307 .setFlagBit(ndn::nfd::BIT_CONGESTION_MARKING_ENABLED, options.allowCongestionMarking, false);
Junxiao Shi1cce2a32017-05-02 02:39:55 +0000308 if (wantUris) {
309 params.setUri(face.getRemoteUri().toString())
310 .setLocalUri(face.getLocalUri().toString());
311 }
312 return params;
Yanbiao Li58ba3f92017-02-15 14:27:18 +0000313}
314
Eric Newberryb5aa7f52016-09-03 20:36:12 -0700315void
Yanbiao Li73860e32015-08-19 16:30:16 -0700316FaceManager::listFaces(const Name& topPrefix, const Interest& interest,
317 ndn::mgmt::StatusDatasetContext& context)
318{
Eric Newberryc64d30a2015-12-26 11:07:27 -0700319 auto now = time::steady_clock::now();
Junxiao Shib84e6742016-07-19 13:16:22 +0000320 for (const Face& face : m_faceTable) {
321 ndn::nfd::FaceStatus status = collectFaceStatus(face, now);
Junxiao Shida93f1f2015-11-11 06:13:16 -0700322 context.append(status.wireEncode());
Yanbiao Li73860e32015-08-19 16:30:16 -0700323 }
324 context.end();
325}
326
327void
328FaceManager::listChannels(const Name& topPrefix, const Interest& interest,
329 ndn::mgmt::StatusDatasetContext& context)
330{
Davide Pesaventoe5eebad2017-04-06 20:23:26 -0400331 std::set<const face::ProtocolFactory*> factories = m_faceSystem.listProtocolFactories();
332 for (const auto* factory : factories) {
Junxiao Shib8590312016-12-29 21:22:25 +0000333 for (const auto& channel : factory->getChannels()) {
334 ndn::nfd::ChannelStatus entry;
335 entry.setLocalUri(channel->getUri().toString());
336 context.append(entry.wireEncode());
Yanbiao Li73860e32015-08-19 16:30:16 -0700337 }
338 }
Yanbiao Li73860e32015-08-19 16:30:16 -0700339 context.end();
340}
341
342void
343FaceManager::queryFaces(const Name& topPrefix, const Interest& interest,
344 ndn::mgmt::StatusDatasetContext& context)
345{
346 ndn::nfd::FaceQueryFilter faceFilter;
347 const Name& query = interest.getName();
348 try {
349 faceFilter.wireDecode(query[-1].blockFromValue());
350 }
Davide Pesavento1d7e7af2015-10-10 23:54:08 +0200351 catch (const tlv::Error& e) {
352 NFD_LOG_DEBUG("Malformed query filter: " << e.what());
353 return context.reject(ControlResponse(400, "Malformed filter"));
Yanbiao Li73860e32015-08-19 16:30:16 -0700354 }
355
Eric Newberryc64d30a2015-12-26 11:07:27 -0700356 auto now = time::steady_clock::now();
Junxiao Shib84e6742016-07-19 13:16:22 +0000357 for (const Face& face : m_faceTable) {
358 if (!matchFilter(faceFilter, face)) {
Junxiao Shida93f1f2015-11-11 06:13:16 -0700359 continue;
Yanbiao Li73860e32015-08-19 16:30:16 -0700360 }
Junxiao Shib84e6742016-07-19 13:16:22 +0000361 ndn::nfd::FaceStatus status = collectFaceStatus(face, now);
Junxiao Shida93f1f2015-11-11 06:13:16 -0700362 context.append(status.wireEncode());
Yanbiao Li73860e32015-08-19 16:30:16 -0700363 }
Davide Pesavento1d7e7af2015-10-10 23:54:08 +0200364
Yanbiao Li73860e32015-08-19 16:30:16 -0700365 context.end();
366}
367
368bool
Junxiao Shib84e6742016-07-19 13:16:22 +0000369FaceManager::matchFilter(const ndn::nfd::FaceQueryFilter& filter, const Face& face)
Yanbiao Li73860e32015-08-19 16:30:16 -0700370{
371 if (filter.hasFaceId() &&
Junxiao Shib84e6742016-07-19 13:16:22 +0000372 filter.getFaceId() != static_cast<uint64_t>(face.getId())) {
Yanbiao Li73860e32015-08-19 16:30:16 -0700373 return false;
374 }
375
376 if (filter.hasUriScheme() &&
Junxiao Shib84e6742016-07-19 13:16:22 +0000377 filter.getUriScheme() != face.getRemoteUri().getScheme() &&
378 filter.getUriScheme() != face.getLocalUri().getScheme()) {
Yanbiao Li73860e32015-08-19 16:30:16 -0700379 return false;
380 }
381
382 if (filter.hasRemoteUri() &&
Junxiao Shib84e6742016-07-19 13:16:22 +0000383 filter.getRemoteUri() != face.getRemoteUri().toString()) {
Yanbiao Li73860e32015-08-19 16:30:16 -0700384 return false;
385 }
386
387 if (filter.hasLocalUri() &&
Junxiao Shib84e6742016-07-19 13:16:22 +0000388 filter.getLocalUri() != face.getLocalUri().toString()) {
Yanbiao Li73860e32015-08-19 16:30:16 -0700389 return false;
390 }
391
392 if (filter.hasFaceScope() &&
Junxiao Shib84e6742016-07-19 13:16:22 +0000393 filter.getFaceScope() != face.getScope()) {
Yanbiao Li73860e32015-08-19 16:30:16 -0700394 return false;
395 }
396
397 if (filter.hasFacePersistency() &&
Junxiao Shib84e6742016-07-19 13:16:22 +0000398 filter.getFacePersistency() != face.getPersistency()) {
Yanbiao Li73860e32015-08-19 16:30:16 -0700399 return false;
400 }
401
402 if (filter.hasLinkType() &&
Junxiao Shib84e6742016-07-19 13:16:22 +0000403 filter.getLinkType() != face.getLinkType()) {
Yanbiao Li73860e32015-08-19 16:30:16 -0700404 return false;
405 }
406
407 return true;
408}
409
Eric Newberryc64d30a2015-12-26 11:07:27 -0700410ndn::nfd::FaceStatus
411FaceManager::collectFaceStatus(const Face& face, const time::steady_clock::TimePoint& now)
412{
413 ndn::nfd::FaceStatus status;
414
415 collectFaceProperties(face, status);
416
417 time::steady_clock::TimePoint expirationTime = face.getExpirationTime();
418 if (expirationTime != time::steady_clock::TimePoint::max()) {
Eric Newberry0c3e57b2018-01-25 20:54:46 -0700419 status.setExpirationPeriod(std::max(0_ms,
Eric Newberryc64d30a2015-12-26 11:07:27 -0700420 time::duration_cast<time::milliseconds>(expirationTime - now)));
421 }
422
Eric Newberry0c3e57b2018-01-25 20:54:46 -0700423 // Get LinkService options
424 auto linkService = dynamic_cast<face::GenericLinkService*>(face.getLinkService());
425 if (linkService != nullptr) {
426 auto linkServiceOptions = linkService->getOptions();
427 status.setBaseCongestionMarkingInterval(linkServiceOptions.baseCongestionMarkingInterval);
428 status.setDefaultCongestionThreshold(linkServiceOptions.defaultCongestionThreshold);
429 }
430
Eric Newberryc64d30a2015-12-26 11:07:27 -0700431 const face::FaceCounters& counters = face.getCounters();
432 status.setNInInterests(counters.nInInterests)
433 .setNOutInterests(counters.nOutInterests)
Junxiao Shif03d4792017-04-06 16:41:22 +0000434 .setNInData(counters.nInData)
435 .setNOutData(counters.nOutData)
Eric Newberryc64d30a2015-12-26 11:07:27 -0700436 .setNInNacks(counters.nInNacks)
437 .setNOutNacks(counters.nOutNacks)
438 .setNInBytes(counters.nInBytes)
439 .setNOutBytes(counters.nOutBytes);
440
441 return status;
442}
443
Junxiao Shida93f1f2015-11-11 06:13:16 -0700444template<typename FaceTraits>
Junxiao Shicde37ad2015-12-24 01:02:05 -0700445void
446FaceManager::collectFaceProperties(const Face& face, FaceTraits& traits)
Junxiao Shida93f1f2015-11-11 06:13:16 -0700447{
448 traits.setFaceId(face.getId())
449 .setRemoteUri(face.getRemoteUri().toString())
450 .setLocalUri(face.getLocalUri().toString())
451 .setFaceScope(face.getScope())
452 .setFacePersistency(face.getPersistency())
453 .setLinkType(face.getLinkType());
Eric Newberry1b4ba052016-10-07 23:04:07 -0700454
455 // Set Flag bits
456 auto linkService = dynamic_cast<face::GenericLinkService*>(face.getLinkService());
457 if (linkService != nullptr) {
458 auto linkServiceOptions = linkService->getOptions();
459 traits.setFlagBit(ndn::nfd::BIT_LOCAL_FIELDS_ENABLED, linkServiceOptions.allowLocalFields);
Eric Newberry84d3adc2017-08-09 23:31:40 -0400460 traits.setFlagBit(ndn::nfd::BIT_LP_RELIABILITY_ENABLED,
461 linkServiceOptions.reliabilityOptions.isEnabled);
Eric Newberryde332452018-01-30 11:45:32 -0700462 traits.setFlagBit(ndn::nfd::BIT_CONGESTION_MARKING_ENABLED,
463 linkServiceOptions.allowCongestionMarking);
Eric Newberry1b4ba052016-10-07 23:04:07 -0700464 }
Junxiao Shida93f1f2015-11-11 06:13:16 -0700465}
466
467void
Eric Newberry1b4ba052016-10-07 23:04:07 -0700468FaceManager::notifyFaceEvent(const Face& face, ndn::nfd::FaceEventKind kind)
Yanbiao Li73860e32015-08-19 16:30:16 -0700469{
470 ndn::nfd::FaceEventNotification notification;
Eric Newberry1b4ba052016-10-07 23:04:07 -0700471 notification.setKind(kind);
Junxiao Shiae04d342016-07-19 13:20:22 +0000472 collectFaceProperties(face, notification);
Yanbiao Li73860e32015-08-19 16:30:16 -0700473
Junxiao Shiae04d342016-07-19 13:20:22 +0000474 m_postNotification(notification.wireEncode());
Yanbiao Li73860e32015-08-19 16:30:16 -0700475}
476
477void
Eric Newberry1b4ba052016-10-07 23:04:07 -0700478FaceManager::connectFaceStateChangeSignal(const Face& face)
Yanbiao Li73860e32015-08-19 16:30:16 -0700479{
Eric Newberry1b4ba052016-10-07 23:04:07 -0700480 FaceId faceId = face.getId();
481 m_faceStateChangeConn[faceId] = face.afterStateChange.connect(
482 [this, faceId] (face::FaceState oldState, face::FaceState newState) {
483 const Face& face = *m_faceTable.get(faceId);
Yanbiao Li73860e32015-08-19 16:30:16 -0700484
Eric Newberry1b4ba052016-10-07 23:04:07 -0700485 if (newState == face::FaceState::UP) {
486 notifyFaceEvent(face, ndn::nfd::FACE_EVENT_UP);
487 }
488 else if (newState == face::FaceState::DOWN) {
489 notifyFaceEvent(face, ndn::nfd::FACE_EVENT_DOWN);
490 }
491 else if (newState == face::FaceState::CLOSED) {
492 m_faceStateChangeConn.erase(faceId);
493 }
494 });
Yanbiao Li73860e32015-08-19 16:30:16 -0700495}
496
Davide Pesavento1d7e7af2015-10-10 23:54:08 +0200497} // namespace nfd