| /* -*- Mode:C++; c-file-style:"gnu"; indent-tabs-mode:nil; -*- */ |
| /** |
| * Copyright (c) 2014, Regents of the University of California, |
| * Arizona Board of Regents, |
| * Colorado State University, |
| * University Pierre & Marie Curie, Sorbonne University, |
| * Washington University in St. Louis, |
| * Beijing Institute of Technology, |
| * The University of Memphis |
| * |
| * This file is part of NFD (Named Data Networking Forwarding Daemon). |
| * See AUTHORS.md for complete list of NFD authors and contributors. |
| * |
| * NFD is free software: you can redistribute it and/or modify it under the terms |
| * of the GNU General Public License as published by the Free Software Foundation, |
| * either version 3 of the License, or (at your option) any later version. |
| * |
| * NFD is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; |
| * without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR |
| * PURPOSE. See the GNU General Public License for more details. |
| * |
| * You should have received a copy of the GNU General Public License along with |
| * NFD, e.g., in COPYING.md file. If not, see <http://www.gnu.org/licenses/>. |
| */ |
| |
| #include "nfdc.hpp" |
| #include "version.hpp" |
| |
| #include <boost/lexical_cast.hpp> |
| #include <boost/algorithm/string.hpp> |
| #include <boost/algorithm/string/regex_find_format.hpp> |
| #include <boost/regex.hpp> |
| |
| void |
| usage(const char* programName) |
| { |
| std::cout << "Usage:\n" << programName << " [-h] [-V] COMMAND [<Command Options>]\n" |
| " -h print usage and exit\n" |
| " -V print version and exit\n" |
| "\n" |
| " COMMAND can be one of the following:\n" |
| " register [-I] [-C] [-c cost] [-e expiration time] [-o origin] name <faceId | faceUri>\n" |
| " register name to the given faceId or faceUri\n" |
| " -I: unset CHILD_INHERIT flag\n" |
| " -C: set CAPTURE flag\n" |
| " -c: specify cost (default 0)\n" |
| " -e: specify expiration time in ms\n" |
| " (by default the entry remains in FIB for the lifetime of the associated face)\n" |
| " -o: specify origin\n" |
| " 0 for Local producer applications, 128 for NLSR, 255(default) for static routes\n" |
| " unregister [-o origin] name <faceId>\n" |
| " unregister name from the given faceId\n" |
| " create <faceUri> \n" |
| " Create a face in one of the following formats:\n" |
| " UDP unicast: udp[4|6]://<remote-IP-or-host>[:<remote-port>]\n" |
| " TCP: tcp[4|6]://<remote-IP-or-host>[:<remote-port>] \n" |
| " destroy <faceId | faceUri> \n" |
| " Destroy a face\n" |
| " set-strategy <name> <strategy> \n" |
| " Set the strategy for a namespace \n" |
| " unset-strategy <name> \n" |
| " Unset the strategy for a namespace \n" |
| " add-nexthop [-c <cost>] <name> <faceId | faceUri>\n" |
| " Add a nexthop to a FIB entry\n" |
| " -c: specify cost (default 0)\n" |
| " remove-nexthop <name> <faceId> \n" |
| " Remove a nexthop from a FIB entry\n" |
| << std::endl; |
| } |
| |
| namespace nfdc { |
| |
| using ndn::bind; |
| using ndn::_1; |
| using ndn::_2; |
| |
| const ndn::time::milliseconds Nfdc::DEFAULT_EXPIRATION_PERIOD = ndn::time::milliseconds::max(); |
| const uint64_t Nfdc::DEFAULT_COST = 0; |
| |
| Nfdc::Nfdc(ndn::Face& face) |
| : m_flags(ROUTE_FLAG_CHILD_INHERIT) |
| , m_cost(DEFAULT_COST) |
| , m_origin(ROUTE_ORIGIN_STATIC) |
| , m_expires(DEFAULT_EXPIRATION_PERIOD) |
| , m_controller(face) |
| , m_ioService(face.getIoService()) |
| { |
| } |
| |
| Nfdc::~Nfdc() |
| { |
| } |
| |
| bool |
| Nfdc::dispatch(const std::string& command) |
| { |
| if (command == "add-nexthop") { |
| if (m_nOptions != 2) |
| return false; |
| fibAddNextHop(); |
| } |
| else if (command == "remove-nexthop") { |
| if (m_nOptions != 2) |
| return false; |
| fibRemoveNextHop(); |
| } |
| else if (command == "register") { |
| if (m_nOptions != 2) |
| return false; |
| ribRegisterPrefix(); |
| } |
| else if (command == "unregister") { |
| if (m_nOptions != 2) |
| return false; |
| ribUnregisterPrefix(); |
| } |
| else if (command == "create") { |
| if (m_nOptions != 1) |
| return false; |
| faceCreate(); |
| } |
| else if (command == "destroy") { |
| if (m_nOptions != 1) |
| return false; |
| faceDestroy(); |
| } |
| else if (command == "set-strategy") { |
| if (m_nOptions != 2) |
| return false; |
| strategyChoiceSet(); |
| } |
| else if (command == "unset-strategy") { |
| if (m_nOptions != 1) |
| return false; |
| strategyChoiceUnset(); |
| } |
| else |
| usage(m_programName); |
| |
| return true; |
| } |
| |
| namespace { |
| |
| inline bool |
| isValidUri(const std::string& input) |
| { |
| // an extended regex to support the validation of uri structure |
| // boost::regex e("^[a-z0-9]+-?+[a-z0-9]+\\:\\/\\/.*"); |
| boost::regex e("^[a-z0-9]+\\:.*"); |
| return boost::regex_match(input, e); |
| } |
| |
| } // anonymous namespace |
| |
| void |
| Nfdc::fibAddNextHop() |
| { |
| m_name = m_commandLineArguments[0]; |
| ControlParameters parameters; |
| parameters |
| .setName(m_name) |
| .setCost(m_cost); |
| |
| if (!isValidUri(m_commandLineArguments[1])) { |
| try { //So the uri is not valid, may be a faceId is provided. |
| m_faceId = boost::lexical_cast<int>(m_commandLineArguments[1]); |
| } |
| catch (const std::exception& e) { |
| std::cerr << "No valid faceUri or faceId is provided"<< std::endl; |
| return; |
| } |
| parameters.setFaceId(m_faceId); |
| fibAddNextHop(parameters); |
| } |
| else { |
| ndn::util::FaceUri faceUri; |
| faceUri.parse(m_commandLineArguments[1]); |
| |
| faceUri.canonize(bind(&Nfdc::startFibAddNextHop, this, _1), |
| bind(&Nfdc::onCanonizeFailure, this, _1), |
| m_ioService, ndn::time::seconds(4)); |
| } |
| } |
| |
| void |
| Nfdc::startFibAddNextHop(const ndn::util::FaceUri& canonicalUri) |
| { |
| ControlParameters parameters; |
| parameters.setUri(canonicalUri.toString()); |
| |
| m_controller.start<FaceCreateCommand>(parameters, |
| [this](const ControlParameters& result) { |
| fibAddNextHop(result); |
| }, |
| bind(&Nfdc::onError, this, _1, _2, |
| "Face creation failed")); |
| } |
| |
| void |
| Nfdc::fibAddNextHop(const ControlParameters& faceCreateResult) |
| { |
| ControlParameters ribParameters; |
| ribParameters |
| .setName(m_name) |
| .setCost(m_cost) |
| .setFaceId(faceCreateResult.getFaceId()); |
| |
| m_controller.start<FibAddNextHopCommand>(ribParameters, |
| bind(&Nfdc::onSuccess, this, _1, |
| "Nexthop insertion succeeded"), |
| bind(&Nfdc::onError, this, _1, _2, |
| "Nexthop insertion failed")); |
| } |
| |
| void |
| Nfdc::fibRemoveNextHop() |
| { |
| m_name = m_commandLineArguments[0]; |
| try { |
| m_faceId = boost::lexical_cast<int>(m_commandLineArguments[1]); |
| } |
| catch (const std::exception& e) { |
| std::cerr << "No valid faceUri or faceId is provided"<< std::endl; |
| return; |
| } |
| |
| ControlParameters parameters; |
| parameters |
| .setName(m_name) |
| .setFaceId(m_faceId); |
| |
| m_controller.start<FibRemoveNextHopCommand>(parameters, |
| bind(&Nfdc::onSuccess, this, _1, |
| "Nexthop removal succeeded"), |
| bind(&Nfdc::onError, this, _1, _2, |
| "Nexthop removal failed")); |
| } |
| |
| void |
| Nfdc::ribRegisterPrefix() |
| { |
| m_name = m_commandLineArguments[0]; |
| ControlParameters parameters; |
| parameters |
| .setName(m_name) |
| .setCost(m_cost) |
| .setFlags(m_flags) |
| .setOrigin(m_origin); |
| |
| if (m_expires != DEFAULT_EXPIRATION_PERIOD) |
| parameters.setExpirationPeriod(m_expires); |
| |
| if (!isValidUri(m_commandLineArguments[1])) { |
| try { //So the uri is not valid, may be a faceId is provided. |
| m_faceId = boost::lexical_cast<int>(m_commandLineArguments[1]); |
| } |
| catch (const std::exception& e) { |
| std::cerr << "No valid faceUri or faceId is provided"<< std::endl; |
| return; |
| } |
| parameters.setFaceId(m_faceId); |
| ribRegisterPrefix(parameters); |
| } |
| else { |
| ndn::util::FaceUri faceUri; |
| faceUri.parse(m_commandLineArguments[1]); |
| |
| faceUri.canonize(bind(&Nfdc::startRibRegisterPrefix, this, _1), |
| bind(&Nfdc::onCanonizeFailure, this, _1), |
| m_ioService, ndn::time::seconds(4)); |
| } |
| } |
| |
| void |
| Nfdc::startRibRegisterPrefix(const ndn::util::FaceUri& canonicalUri) |
| { |
| ControlParameters parameters; |
| parameters.setUri(canonicalUri.toString()); |
| |
| m_controller.start<FaceCreateCommand>(parameters, |
| [this](const ControlParameters& result) { |
| ribRegisterPrefix(result); |
| }, |
| bind(&Nfdc::onError, this, _1, _2, |
| "Face creation failed")); |
| } |
| |
| void |
| Nfdc::ribRegisterPrefix(const ControlParameters& faceCreateResult) |
| { |
| ControlParameters ribParameters; |
| ribParameters |
| .setName(m_name) |
| .setCost(m_cost) |
| .setFlags(m_flags) |
| .setOrigin(m_origin) |
| .setFaceId(faceCreateResult.getFaceId()); |
| |
| if (m_expires != DEFAULT_EXPIRATION_PERIOD) |
| ribParameters.setExpirationPeriod(m_expires); |
| |
| m_controller.start<RibRegisterCommand>(ribParameters, |
| bind(&Nfdc::onSuccess, this, _1, |
| "Successful in name registration"), |
| bind(&Nfdc::onError, this, _1, _2, |
| "Failed in name registration")); |
| } |
| |
| void |
| Nfdc::ribUnregisterPrefix() |
| { |
| m_name = m_commandLineArguments[0]; |
| try { |
| m_faceId = boost::lexical_cast<int>(m_commandLineArguments[1]); |
| } |
| catch (const std::exception& e) { |
| std::cerr << "No valid faceId is provided" << std::endl; |
| return; |
| } |
| |
| ControlParameters parameters; |
| parameters |
| .setName(m_name) |
| .setFaceId(m_faceId) |
| .setOrigin(m_origin); |
| |
| m_controller.start<RibUnregisterCommand>(parameters, |
| bind(&Nfdc::onSuccess, this, _1, |
| "Successful in unregistering name"), |
| bind(&Nfdc::onError, this, _1, _2, |
| "Failed in unregistering name")); |
| } |
| |
| void |
| Nfdc::onCanonizeFailure(const std::string& reason) |
| { |
| std::cerr << reason << std::endl; |
| } |
| |
| void |
| Nfdc::faceCreate() |
| { |
| if (!isValidUri(m_commandLineArguments[0])) |
| throw Error("invalid uri format"); |
| |
| ndn::util::FaceUri faceUri; |
| faceUri.parse(m_commandLineArguments[0]); |
| |
| faceUri.canonize(bind(&Nfdc::startFaceCreate, this, _1), |
| bind(&Nfdc::onCanonizeFailure, this, _1), |
| m_ioService, ndn::time::seconds(4)); |
| } |
| |
| void |
| Nfdc::startFaceCreate(const ndn::util::FaceUri& canonicalUri) |
| { |
| ControlParameters parameters; |
| parameters.setUri(canonicalUri.toString()); |
| |
| m_controller.start<FaceCreateCommand>(parameters, |
| bind(&Nfdc::onSuccess, this, _1, |
| "Face creation succeeded"), |
| bind(&Nfdc::onError, this, _1, _2, |
| "Face creation failed")); |
| } |
| |
| void |
| Nfdc::faceDestroy() |
| { |
| ControlParameters parameters; |
| if (!isValidUri(m_commandLineArguments[0])) { |
| try { //So the uri is not valid, may be a faceId is provided. |
| m_faceId = boost::lexical_cast<int>(m_commandLineArguments[0]); |
| } |
| catch (const std::exception& e) { |
| std::cerr << "No valid faceUri or faceId is provided" << std::endl; |
| return; |
| } |
| parameters.setFaceId(m_faceId); |
| faceDestroy(parameters); |
| } |
| else{ |
| ndn::util::FaceUri faceUri; |
| faceUri.parse(m_commandLineArguments[0]); |
| |
| faceUri.canonize(bind(&Nfdc::startFaceDestroy, this, _1), |
| bind(&Nfdc::onCanonizeFailure, this, _1), |
| m_ioService, ndn::time::seconds(4)); |
| } |
| } |
| |
| void |
| Nfdc::startFaceDestroy(const ndn::util::FaceUri& canonicalUri) |
| { |
| ControlParameters parameters; |
| parameters.setUri(canonicalUri.toString()); |
| |
| m_controller.start<FaceCreateCommand>(parameters, |
| [this](const ControlParameters& result) { |
| faceDestroy(result); |
| }, |
| bind(&Nfdc::onError, this, _1, _2, |
| "Face destroy failed")); |
| } |
| |
| void |
| Nfdc::faceDestroy(const ControlParameters& faceCreateResult) |
| { |
| ControlParameters faceParameters; |
| faceParameters.setFaceId(faceCreateResult.getFaceId()); |
| |
| m_controller.start<FaceDestroyCommand>(faceParameters, |
| bind(&Nfdc::onSuccess, this, _1, |
| "Face destroy succeeded"), |
| bind(&Nfdc::onError, this, _1, _2, |
| "Face destroy failed")); |
| } |
| |
| void |
| Nfdc::strategyChoiceSet() |
| { |
| const std::string& name = m_commandLineArguments[0]; |
| const std::string& strategy = m_commandLineArguments[1]; |
| |
| ControlParameters parameters; |
| parameters |
| .setName(name) |
| .setStrategy(strategy); |
| |
| m_controller.start<StrategyChoiceSetCommand>(parameters, |
| bind(&Nfdc::onSuccess, this, _1, |
| "Successfully set strategy choice"), |
| bind(&Nfdc::onError, this, _1, _2, |
| "Failed to set strategy choice")); |
| } |
| |
| void |
| Nfdc::strategyChoiceUnset() |
| { |
| const std::string& name = m_commandLineArguments[0]; |
| |
| ControlParameters parameters; |
| parameters.setName(name); |
| |
| m_controller.start<StrategyChoiceUnsetCommand>(parameters, |
| bind(&Nfdc::onSuccess, this, _1, |
| "Successfully unset strategy choice"), |
| bind(&Nfdc::onError, this, _1, _2, |
| "Failed to unset strategy choice")); |
| } |
| |
| void |
| Nfdc::onSuccess(const ControlParameters& commandSuccessResult, const std::string& message) |
| { |
| std::cout << message << ": " << commandSuccessResult << std::endl; |
| } |
| |
| void |
| Nfdc::onError(uint32_t code, const std::string& error, const std::string& message) |
| { |
| std::ostringstream os; |
| os << message << ": " << error << " (code: " << code << ")"; |
| throw Error(os.str()); |
| } |
| |
| |
| |
| } // namespace nfdc |
| |
| int |
| main(int argc, char** argv) |
| { |
| ndn::Face face; |
| nfdc::Nfdc p(face); |
| |
| p.m_programName = argv[0]; |
| |
| if (argc < 2) { |
| usage(p.m_programName); |
| return 0; |
| } |
| |
| if (!strcmp(argv[1], "-h")) { |
| usage(p.m_programName); |
| return 0; |
| } |
| |
| if (!strcmp(argv[1], "-V")) { |
| std::cout << NFD_VERSION_BUILD_STRING << std::endl; |
| return 0; |
| } |
| |
| ::optind = 2; //start reading options from 2nd argument i.e. Command |
| int opt; |
| while ((opt = ::getopt(argc, argv, "ICc:e:o:")) != -1) { |
| switch (opt) { |
| case 'I': |
| p.m_flags = p.m_flags & ~(nfdc::ROUTE_FLAG_CHILD_INHERIT); |
| break; |
| |
| case 'C': |
| p.m_flags = p.m_flags | nfdc::ROUTE_FLAG_CAPTURE; |
| break; |
| |
| case 'c': |
| try { |
| p.m_cost = boost::lexical_cast<uint64_t>(::optarg); |
| } |
| catch (boost::bad_lexical_cast&) { |
| std::cerr << "Error: cost must be in unsigned integer format" << std::endl; |
| return 1; |
| } |
| break; |
| |
| case 'e': |
| uint64_t expires; |
| try { |
| expires = boost::lexical_cast<uint64_t>(::optarg); |
| } |
| catch (boost::bad_lexical_cast&) { |
| std::cerr << "Error: expiration time must be in unsigned integer format" << std::endl; |
| return 1; |
| } |
| p.m_expires = ndn::time::milliseconds(expires); |
| break; |
| |
| case 'o': |
| try { |
| p.m_origin = boost::lexical_cast<uint64_t>(::optarg); |
| } |
| catch (boost::bad_lexical_cast&) { |
| std::cerr << "Error: origin must be in unsigned integer format" << std::endl; |
| return 1; |
| } |
| break; |
| |
| default: |
| usage(p.m_programName); |
| return 1; |
| } |
| } |
| |
| if (argc == ::optind) { |
| usage(p.m_programName); |
| return 1; |
| } |
| |
| try { |
| p.m_commandLineArguments = argv + ::optind; |
| p.m_nOptions = argc - ::optind; |
| |
| //argv[1] points to the command, so pass it to the dispatch |
| bool isOk = p.dispatch(argv[1]); |
| if (!isOk) { |
| usage(p.m_programName); |
| return 1; |
| } |
| face.processEvents(); |
| } |
| catch (const std::exception& e) { |
| std::cerr << "ERROR: " << e.what() << std::endl; |
| return 2; |
| } |
| return 0; |
| } |