blob: ce2c2e3e7962227bdc299331b5fee4e215582e26 [file] [log] [blame]
/* -*- Mode:C++; c-file-style:"gnu"; indent-tabs-mode:nil; -*- */
/*
* Copyright (c) 2014-2023, Arizona Board of Regents.
*
* This program 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.
*
* This program 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 this program. If not, see <http://www.gnu.org/licenses/>.
*
* Author: Jerald Paul Abraham <jeraldabraham@email.arizona.edu>
*/
#include "util.hpp"
#include <ndn-cxx/data.hpp>
#include <ndn-cxx/face.hpp>
#include <ndn-cxx/interest.hpp>
#include <ndn-cxx/security/key-chain.hpp>
#include <ndn-cxx/security/signing-info.hpp>
#include <ndn-cxx/util/random.hpp>
#include <ndn-cxx/util/time.hpp>
#include <chrono>
#include <limits>
#include <optional>
#include <sstream>
#include <thread>
#include <vector>
#include <boost/asio/io_context.hpp>
#include <boost/asio/signal_set.hpp>
#include <boost/core/noncopyable.hpp>
#include <boost/program_options/options_description.hpp>
#include <boost/program_options/parsers.hpp>
#include <boost/program_options/variables_map.hpp>
using namespace std::chrono_literals;
namespace ndntg {
using namespace ndn::time_literals;
using namespace std::string_literals;
class NdnTrafficServer : boost::noncopyable
{
public:
explicit
NdnTrafficServer(std::string configFile)
: m_configurationFile(std::move(configFile))
{
}
void
setMaximumInterests(uint64_t maxInterests)
{
m_nMaximumInterests = maxInterests;
}
void
setContentDelay(std::chrono::milliseconds delay)
{
BOOST_ASSERT(delay >= 0ms);
m_contentDelay = delay;
}
void
setTimestampFormat(std::string format)
{
m_timestampFormat = std::move(format);
}
void
setQuietLogging()
{
m_wantQuiet = true;
}
int
run()
{
m_logger.initialize(std::to_string(ndn::random::generateWord32()), m_timestampFormat);
if (!readConfigurationFile(m_configurationFile, m_trafficPatterns, m_logger)) {
return 2;
}
if (!checkTrafficPatternCorrectness()) {
m_logger.log("ERROR: Traffic configuration provided is not proper", false, true);
return 2;
}
m_logger.log("Traffic configuration file processing completed\n", true, false);
for (std::size_t i = 0; i < m_trafficPatterns.size(); i++) {
m_logger.log("Traffic Pattern Type #" + std::to_string(i + 1), false, false);
m_trafficPatterns[i].printTrafficConfiguration(m_logger);
m_logger.log("", false, false);
}
if (m_nMaximumInterests == 0) {
logStatistics();
return 0;
}
m_signalSet.async_wait([this] (const boost::system::error_code&, int) {
if (m_nMaximumInterests && m_nInterestsReceived < *m_nMaximumInterests) {
m_hasError = true;
}
stop();
});
for (std::size_t id = 0; id < m_trafficPatterns.size(); id++) {
m_registeredPrefixes.push_back(
m_face.setInterestFilter(m_trafficPatterns[id].m_name,
[this, id] (auto&&, const auto& interest) { onInterest(interest, id); },
nullptr,
[this, id] (auto&&, const auto& reason) { onRegisterFailed(reason, id); }));
}
try {
m_face.processEvents();
return m_hasError ? 1 : 0;
}
catch (const std::exception& e) {
m_logger.log("ERROR: "s + e.what(), true, true);
m_io.stop();
return 1;
}
}
private:
class DataTrafficConfiguration
{
public:
void
printTrafficConfiguration(Logger& logger) const
{
std::ostringstream os;
if (!m_name.empty()) {
os << "Name=" << m_name << ", ";
}
if (m_contentDelay >= 0ms) {
os << "ContentDelay=" << m_contentDelay.count() << ", ";
}
if (m_freshnessPeriod >= 0_ms) {
os << "FreshnessPeriod=" << m_freshnessPeriod.count() << ", ";
}
if (m_contentType) {
os << "ContentType=" << *m_contentType << ", ";
}
if (m_contentLength) {
os << "ContentBytes=" << *m_contentLength << ", ";
}
if (!m_content.empty()) {
os << "Content=" << m_content << ", ";
}
os << "SigningInfo=" << m_signingInfo;
logger.log(os.str(), false, false);
}
bool
parseConfigurationLine(const std::string& line, Logger& logger, int lineNumber)
{
std::string parameter, value;
if (!extractParameterAndValue(line, parameter, value)) {
logger.log("Line " + std::to_string(lineNumber) + " - Invalid syntax: " + line,
false, true);
return false;
}
if (parameter == "Name") {
m_name = value;
}
else if (parameter == "ContentDelay") {
m_contentDelay = std::chrono::milliseconds(std::stoul(value));
}
else if (parameter == "FreshnessPeriod") {
m_freshnessPeriod = ndn::time::milliseconds(std::stoul(value));
}
else if (parameter == "ContentType") {
m_contentType = std::stoul(value);
}
else if (parameter == "ContentBytes") {
m_contentLength = std::stoul(value);
}
else if (parameter == "Content") {
m_content = value;
}
else if (parameter == "SigningInfo") {
m_signingInfo = ndn::security::SigningInfo(value);
}
else {
logger.log("Line " + std::to_string(lineNumber) + " - Ignoring unknown parameter: " + parameter,
false, true);
}
return true;
}
bool
checkTrafficDetailCorrectness() const
{
return true;
}
public:
std::string m_name;
std::chrono::milliseconds m_contentDelay{-1};
ndn::time::milliseconds m_freshnessPeriod{-1};
std::optional<uint32_t> m_contentType;
std::optional<std::size_t> m_contentLength;
std::string m_content;
ndn::security::SigningInfo m_signingInfo;
uint64_t m_nInterestsReceived = 0;
};
void
logStatistics()
{
using std::to_string;
m_logger.log("\n\n== Traffic Report ==\n", false, true);
m_logger.log("Total Traffic Pattern Types = " + to_string(m_trafficPatterns.size()), false, true);
m_logger.log("Total Interests Received = " + to_string(m_nInterestsReceived) + "\n", false, true);
for (std::size_t patternId = 0; patternId < m_trafficPatterns.size(); patternId++) {
const auto& pattern = m_trafficPatterns[patternId];
m_logger.log("Traffic Pattern Type #" + to_string(patternId + 1), false, true);
pattern.printTrafficConfiguration(m_logger);
m_logger.log("Total Interests Received = " +
to_string(pattern.m_nInterestsReceived) + "\n", false, true);
}
}
bool
checkTrafficPatternCorrectness() const
{
// TODO
return true;
}
static std::string
getRandomByteString(std::size_t length)
{
// per ISO C++ std, cannot instantiate uniform_int_distribution with char
static std::uniform_int_distribution<short> dist(std::numeric_limits<char>::min(),
std::numeric_limits<char>::max());
std::string s;
s.reserve(length);
for (std::size_t i = 0; i < length; i++) {
s += static_cast<char>(dist(ndn::random::getRandomNumberEngine()));
}
return s;
}
void
onInterest(const ndn::Interest& interest, std::size_t patternId)
{
auto& pattern = m_trafficPatterns[patternId];
if (!m_nMaximumInterests || m_nInterestsReceived < *m_nMaximumInterests) {
ndn::Data data(interest.getName());
if (pattern.m_freshnessPeriod >= 0_ms)
data.setFreshnessPeriod(pattern.m_freshnessPeriod);
if (pattern.m_contentType)
data.setContentType(*pattern.m_contentType);
std::string content;
if (pattern.m_contentLength > 0)
content = getRandomByteString(*pattern.m_contentLength);
if (!pattern.m_content.empty())
content = pattern.m_content;
data.setContent(ndn::makeStringBlock(ndn::tlv::Content, content));
m_keyChain.sign(data, pattern.m_signingInfo);
m_nInterestsReceived++;
pattern.m_nInterestsReceived++;
if (!m_wantQuiet) {
auto logLine = "Interest Received - PatternType=" + std::to_string(patternId + 1) +
", GlobalID=" + std::to_string(m_nInterestsReceived) +
", LocalID=" + std::to_string(pattern.m_nInterestsReceived) +
", Name=" + interest.getName().toUri();
m_logger.log(logLine, true, false);
}
if (pattern.m_contentDelay > 0ms)
std::this_thread::sleep_for(pattern.m_contentDelay);
if (m_contentDelay > 0ms)
std::this_thread::sleep_for(m_contentDelay);
m_face.put(data);
}
if (m_nMaximumInterests && m_nInterestsReceived >= *m_nMaximumInterests) {
logStatistics();
m_registeredPrefixes.clear();
m_signalSet.cancel();
}
}
void
onRegisterFailed(const std::string& reason, std::size_t patternId)
{
auto logLine = "Prefix registration failed - PatternType=" + std::to_string(patternId + 1) +
", Name=" + m_trafficPatterns[patternId].m_name +
", Reason=" + reason;
m_logger.log(logLine, true, true);
m_nRegistrationsFailed++;
if (m_nRegistrationsFailed == m_trafficPatterns.size()) {
m_hasError = true;
stop();
}
}
void
stop()
{
logStatistics();
m_face.shutdown();
m_io.stop();
}
private:
Logger m_logger{"NdnTrafficServer"};
boost::asio::io_context m_io;
boost::asio::signal_set m_signalSet{m_io, SIGINT, SIGTERM};
ndn::Face m_face{m_io};
ndn::KeyChain m_keyChain;
std::string m_configurationFile;
std::string m_timestampFormat;
std::optional<uint64_t> m_nMaximumInterests;
std::chrono::milliseconds m_contentDelay{0};
std::vector<DataTrafficConfiguration> m_trafficPatterns;
std::vector<ndn::ScopedRegisteredPrefixHandle> m_registeredPrefixes;
uint64_t m_nRegistrationsFailed = 0;
uint64_t m_nInterestsReceived = 0;
bool m_wantQuiet = false;
bool m_hasError = false;
};
} // namespace ndntg
namespace po = boost::program_options;
static void
usage(std::ostream& os, std::string_view programName, const po::options_description& desc)
{
os << "Usage: " << programName << " [options] <Traffic_Configuration_File>\n"
<< "\n"
<< "Respond to Interests as per provided Traffic_Configuration_File.\n"
<< "Multiple prefixes can be configured for handling.\n"
<< "Set the environment variable NDN_TRAFFIC_LOGFOLDER to redirect output to a log file.\n"
<< "\n"
<< desc;
}
int
main(int argc, char* argv[])
{
std::string configFile;
std::string timestampFormat;
po::options_description visibleOptions("Options");
visibleOptions.add_options()
("help,h", "print this help message and exit")
("count,c", po::value<int64_t>(), "maximum number of Interests to respond to")
("delay,d", po::value<std::chrono::milliseconds::rep>()->default_value(0),
"wait this amount of milliseconds before responding to each Interest")
("timestamp-format,t", po::value<std::string>(&timestampFormat), "format string for timestamp output")
("quiet,q", po::bool_switch(), "turn off logging of Interest reception and Data generation")
;
po::options_description hiddenOptions;
hiddenOptions.add_options()
("config-file", po::value<std::string>(&configFile))
;
po::positional_options_description posOptions;
posOptions.add("config-file", -1);
po::options_description allOptions;
allOptions.add(visibleOptions).add(hiddenOptions);
po::variables_map vm;
try {
po::store(po::command_line_parser(argc, argv).options(allOptions).positional(posOptions).run(), vm);
po::notify(vm);
}
catch (const po::error& e) {
std::cerr << "ERROR: " << e.what() << std::endl;
return 2;
}
catch (const boost::bad_any_cast& e) {
std::cerr << "ERROR: " << e.what() << std::endl;
return 2;
}
if (vm.count("help") > 0) {
usage(std::cout, argv[0], visibleOptions);
return 0;
}
if (configFile.empty()) {
usage(std::cerr, argv[0], visibleOptions);
return 2;
}
ndntg::NdnTrafficServer server(std::move(configFile));
if (vm.count("count") > 0) {
auto count = vm["count"].as<int64_t>();
if (count < 0) {
std::cerr << "ERROR: the argument for option '--count' cannot be negative\n";
return 2;
}
server.setMaximumInterests(static_cast<uint64_t>(count));
}
if (vm.count("delay") > 0) {
std::chrono::milliseconds delay(vm["delay"].as<std::chrono::milliseconds::rep>());
if (delay < 0ms) {
std::cerr << "ERROR: the argument for option '--delay' cannot be negative\n";
return 2;
}
server.setContentDelay(delay);
}
if (!timestampFormat.empty()) {
server.setTimestampFormat(std::move(timestampFormat));
}
if (vm["quiet"].as<bool>()) {
server.setQuietLogging();
}
return server.run();
}