29 #include <ndn-cxx/face.hpp>
30 #include <ndn-cxx/name.hpp>
31 #include <ndn-cxx/encoding/buffer-stream.hpp>
32 #include <ndn-cxx/mgmt/nfd/controller.hpp>
33 #include <ndn-cxx/mgmt/nfd/face-monitor.hpp>
34 #include <ndn-cxx/mgmt/nfd/face-status.hpp>
35 #include <ndn-cxx/net/face-uri.hpp>
36 #include <ndn-cxx/security/key-chain.hpp>
38 #include <boost/exception/diagnostic_information.hpp>
39 #include <boost/program_options/options_description.hpp>
40 #include <boost/program_options/variables_map.hpp>
41 #include <boost/program_options/parsers.hpp>
46 namespace nfd_autoreg {
50 class AutoregServer : boost::noncopyable
54 : m_controller(m_face, m_keyChain)
55 , m_faceMonitor(m_face)
61 onRegisterCommandSuccess(uint64_t faceId,
const Name& prefix)
63 std::cerr <<
"SUCCEED: register " << prefix <<
" on face " << faceId << std::endl;
67 onRegisterCommandFailure(uint64_t faceId,
const Name& prefix,
68 const nfd::ControlResponse& response)
70 std::cerr <<
"FAILED: register " << prefix <<
" on face " << faceId
71 <<
" (code: " << response.getCode() <<
", reason: " << response.getText() <<
")"
79 hasAllowedSchema(
const FaceUri& uri)
81 const std::string& scheme = uri.getScheme();
82 return scheme ==
"udp4" || scheme ==
"tcp4" ||
83 scheme ==
"udp6" || scheme ==
"tcp6";
90 isBlacklisted(
const boost::asio::ip::address& address)
const
92 return std::any_of(m_blackList.begin(), m_blackList.end(),
93 std::bind(&Network::doesContain, _1, address));
100 isWhitelisted(
const boost::asio::ip::address& address)
const
102 return std::any_of(m_whiteList.begin(), m_whiteList.end(),
103 std::bind(&Network::doesContain, _1, address));
107 registerPrefixesForFace(uint64_t faceId,
const std::vector<Name>& prefixes)
109 for (
const Name& prefix : prefixes) {
110 m_controller.start<nfd::RibRegisterCommand>(
111 nfd::ControlParameters()
114 .setOrigin(nfd::ROUTE_ORIGIN_AUTOREG)
116 .setExpirationPeriod(time::milliseconds::max()),
117 std::bind(&AutoregServer::onRegisterCommandSuccess,
this, faceId, prefix),
118 std::bind(&AutoregServer::onRegisterCommandFailure,
this, faceId, prefix, _1));
123 registerPrefixesIfNeeded(uint64_t faceId,
const FaceUri& uri, nfd::FacePersistency facePersistency)
125 if (hasAllowedSchema(uri)) {
126 boost::system::error_code ec;
127 auto address = boost::asio::ip::address::from_string(uri.getHost(), ec);
129 if (!address.is_multicast()) {
131 registerPrefixesForFace(faceId, m_allFacesPrefixes);
134 if (facePersistency == nfd::FACE_PERSISTENCY_ON_DEMAND &&
135 !isBlacklisted(address) && isWhitelisted(address)) {
136 registerPrefixesForFace(faceId, m_autoregPrefixes);
143 onNotification(
const nfd::FaceEventNotification& notification)
145 if (notification.getKind() == nfd::FACE_EVENT_CREATED &&
146 notification.getFaceScope() != nfd::FACE_SCOPE_LOCAL) {
147 std::cerr <<
"PROCESSING: " << notification << std::endl;
149 registerPrefixesIfNeeded(notification.getFaceId(), FaceUri(notification.getRemoteUri()),
150 notification.getFacePersistency());
153 std::cerr <<
"IGNORED: " << notification << std::endl;
158 usage(std::ostream& os,
159 const boost::program_options::options_description& desc,
160 const char* programName)
162 os <<
"Usage: " << programName <<
" [--prefix=</autoreg/prefix>]... [options]\n"
170 std::cerr <<
"AUTOREG prefixes: " << std::endl;
171 for (
const Name& prefix : m_autoregPrefixes) {
172 std::cout <<
" " << prefix << std::endl;
174 std::cerr <<
"ALL-FACES-AUTOREG prefixes: " << std::endl;
175 for (
const Name& prefix : m_allFacesPrefixes) {
176 std::cout <<
" " << prefix << std::endl;
179 if (!m_blackList.empty()) {
180 std::cerr <<
"Blacklisted networks: " << std::endl;
181 for (
const Network& network : m_blackList) {
182 std::cout <<
" " << network << std::endl;
186 std::cerr <<
"Whitelisted networks: " << std::endl;
187 for (
const Network& network : m_whiteList) {
188 std::cout <<
" " << network << std::endl;
191 m_faceMonitor.onNotification.connect(std::bind(&AutoregServer::onNotification,
this, _1));
192 m_faceMonitor.start();
194 boost::asio::signal_set signalSet(m_face.getIoService(), SIGINT, SIGTERM);
195 signalSet.async_wait([
this] (
auto&&...) { m_face.shutdown(); });
197 m_face.processEvents();
201 startFetchingFaceStatusDataset()
203 m_controller.fetch<nfd::FaceDataset>(
204 [
this] (
const auto& faces) {
205 for (
const auto& faceStatus : faces) {
206 registerPrefixesIfNeeded(faceStatus.getFaceId(), FaceUri(faceStatus.getRemoteUri()),
207 faceStatus.getFacePersistency());
214 main(
int argc,
char* argv[])
216 namespace po = boost::program_options;
218 po::options_description optionsDesc(
"Options");
219 optionsDesc.add_options()
220 (
"help,h",
"print this message and exit")
221 (
"version,V",
"show version information and exit")
222 (
"prefix,i", po::value<std::vector<Name>>(&m_autoregPrefixes)->composing(),
223 "prefix that should be automatically registered when a new non-local face is created")
224 (
"all-faces-prefix,a", po::value<std::vector<Name>>(&m_allFacesPrefixes)->composing(),
225 "prefix that should be automatically registered for all TCP and UDP non-local faces "
226 "(blacklists and whitelists do not apply to this prefix)")
227 (
"cost,c", po::value<uint64_t>(&m_cost)->default_value(255),
228 "FIB cost that should be assigned to autoreg nexthops")
229 (
"whitelist,w", po::value<std::vector<Network>>(&m_whiteList)->composing(),
230 "Whitelisted network, e.g., 192.168.2.0/24 or ::1/128")
231 (
"blacklist,b", po::value<std::vector<Network>>(&m_blackList)->composing(),
232 "Blacklisted network, e.g., 192.168.2.32/30 or ::1/128")
235 po::variables_map options;
237 po::store(po::parse_command_line(argc, argv, optionsDesc), options);
240 catch (
const std::exception& e) {
241 std::cerr <<
"ERROR: " << e.what() << std::endl << std::endl;
242 usage(std::cerr, optionsDesc, argv[0]);
246 if (options.count(
"help") > 0) {
247 usage(std::cout, optionsDesc, argv[0]);
251 if (options.count(
"version") > 0) {
256 if (m_autoregPrefixes.empty() && m_allFacesPrefixes.empty()) {
257 std::cerr <<
"ERROR: at least one --prefix or --all-faces-prefix must be specified"
258 << std::endl << std::endl;
259 usage(std::cerr, optionsDesc, argv[0]);
263 if (m_whiteList.empty()) {
265 m_whiteList.push_back(Network::getMaxRangeV4());
266 m_whiteList.push_back(Network::getMaxRangeV6());
270 startFetchingFaceStatusDataset();
273 catch (
const std::exception& e) {
274 std::cerr <<
"ERROR: " << boost::diagnostic_information(e);
284 nfd::Controller m_controller;
285 nfd::FaceMonitor m_faceMonitor;
286 std::vector<Name> m_autoregPrefixes;
287 std::vector<Name> m_allFacesPrefixes;
289 std::vector<Network> m_whiteList;
290 std::vector<Network> m_blackList;
299 ndn::nfd_autoreg::AutoregServer server;
300 return server.main(argc, argv);
Copyright (c) 2014-2017, Regents of the University of California, Arizona Board of Regents,...
int main(int argc, char *argv[])
const char NFD_VERSION_BUILD_STRING[]
NFD version string, including git commit information if NFD is build from a specific git commit.