blob: 4bf50cc56b7567465370508e19d37855c4344b06 [file] [log] [blame]
/* -*- Mode:C++; c-file-style:"gnu"; indent-tabs-mode:nil -*- */
/**
* Copyright (C) 2014 University of Arizona.
*
* GNU 3.0 License, see the LICENSE file for more information
*
* Author: Jerald Paul Abraham <jeraldabraham@email.arizona.edu>
*/
#include <sstream>
#include <boost/asio.hpp>
#include <boost/filesystem.hpp>
#include <boost/lexical_cast.hpp>
#include <boost/noncopyable.hpp>
#include <ndn-cxx/face.hpp>
#include <ndn-cxx/security/key-chain.hpp>
#include "logger.hpp"
namespace ndn {
class NdnTrafficServer : boost::noncopyable
{
public:
explicit
NdnTrafficServer(char* programName)
: m_logger("NdnTrafficServer")
, m_programName(programName)
, m_hasError(false)
, m_hasQuietLogging(false)
, m_nRegistrationsFailed(0)
, m_nMaximumInterests(-1)
, m_nInterestsReceived(0)
, m_contentDelay(time::milliseconds(-1))
, m_face(m_ioService)
{
m_instanceId = boost::lexical_cast<std::string>(std::rand());
}
class DataTrafficConfiguration
{
public:
DataTrafficConfiguration()
: m_contentType(-1)
, m_freshnessPeriod(time::milliseconds(-1))
, m_contentBytes(-1)
, m_contentDelay(time::milliseconds(-1))
, m_nInterestsReceived(0)
{
}
void
printTrafficConfiguration(Logger& logger)
{
std::string detail = "";
if (m_name != "")
detail += "Name=" + m_name + ", ";
if (m_contentType >= 0)
detail += "ContentType=" + boost::lexical_cast<std::string>(m_contentType) + ", ";
if (m_freshnessPeriod >= time::milliseconds(0))
detail += "FreshnessPeriod=" +
boost::lexical_cast<std::string>(static_cast<int>(m_freshnessPeriod.count())) + ", ";
if (m_contentBytes >= 0)
detail += "ContentBytes=" + boost::lexical_cast<std::string>(m_contentBytes) + ", ";
if (m_contentDelay >= time::milliseconds(0))
detail += "ContentDelay=" +
boost::lexical_cast<std::string>(m_contentDelay.count()) + ", ";
if (m_content != "")
detail += "Content=" + m_content + ", ";
if (detail.length() >= 2)
detail = detail.substr(0, detail.length() - 2);
logger.log(detail, false, false);
}
bool
extractParameterValue(const std::string& detail,
std::string& parameter,
std::string& value)
{
std::string allowedCharacters = ":/+._-%";
parameter = "";
value = "";
int i = 0;
while (detail[i] != '=' && i < detail.length())
{
parameter += detail[i];
i++;
}
if (i == detail.length())
return false;
i++;
while ((std::isalnum(detail[i]) ||
allowedCharacters.find(detail[i]) != std::string::npos) &&
i < detail.length())
{
value += detail[i];
i++;
}
if(parameter == "" || value == "")
return false;
return true;
}
bool
processConfigurationDetail(const std::string& detail,
Logger& logger,
int lineNumber)
{
std::string parameter, value;
if (extractParameterValue(detail, parameter, value))
{
if (parameter == "Name")
m_name = value;
else if (parameter == "ContentType")
m_contentType = boost::lexical_cast<int>(value);
else if (parameter == "FreshnessPeriod")
m_freshnessPeriod = time::milliseconds(boost::lexical_cast<int>(value));
else if (parameter == "ContentDelay")
m_contentDelay = time::milliseconds(boost::lexical_cast<int>(value));
else if (parameter == "ContentBytes")
m_contentBytes = boost::lexical_cast<int>(value);
else if (parameter == "Content")
m_content = value;
else
logger.log("Line " + boost::lexical_cast<std::string>(lineNumber) +
" \t- Invalid Parameter='" + parameter + "'", false, true);
}
else
{
logger.log("Line " + boost::lexical_cast<std::string>(lineNumber) +
" \t- Improper Traffic Configuration Line - " + detail, false, true);
return false;
}
return true;
}
bool
checkTrafficDetailCorrectness()
{
return true;
}
std::string m_name;
int m_contentType;
time::milliseconds m_freshnessPeriod;
int m_contentBytes;
time::milliseconds m_contentDelay;
std::string m_content;
int m_nInterestsReceived;
};
void
usage()
{
std::cout << "\nUsage: " << m_programName << " [options] <Traffic_Configuration_File>\n"
"Respond to Interest as per provided Traffic Configuration File\n"
"Multiple Prefixes can be configured for handling.\n"
"Set environment variable NDN_TRAFFIC_LOGFOLDER for redirecting output to a log.\n"
" [-d interval] - set delay before responding to interest in milliseconds\n"
" [-c count] - specify maximum number of interests to be satisfied\n"
" [-q] - quiet logging - no interest reception/data generation messages\n"
" [-h] - print help and exit\n\n";
exit(1);
}
void
setMaximumInterests(int maximumInterests)
{
if (maximumInterests < 0)
usage();
m_nMaximumInterests = maximumInterests;
}
bool
hasError() const
{
return m_hasError;
}
void
setContentDelay(int contentDelay)
{
if (contentDelay < 0)
usage();
m_contentDelay = time::milliseconds(contentDelay);
}
void
setConfigurationFile(char* configurationFile)
{
m_configurationFile = configurationFile;
}
void
setQuietLogging()
{
m_hasQuietLogging = true;
}
void
signalHandler()
{
logStatistics();
m_logger.shutdownLogger();
m_face.shutdown();
m_ioService.stop();
if (m_hasError)
exit(1);
else
exit(0);
}
void
logStatistics()
{
m_logger.log("\n\n== Interest Traffic Report ==\n", false, true);
m_logger.log("Total Traffic Pattern Types = " +
boost::lexical_cast<std::string>(m_trafficPatterns.size()), false, true);
m_logger.log("Total Interests Received = " +
boost::lexical_cast<std::string>(m_nInterestsReceived), false, true);
if (m_nInterestsReceived < m_nMaximumInterests)
m_hasError = true;
for (int patternId = 0; patternId < m_trafficPatterns.size(); patternId++)
{
m_logger.log("\nTraffic Pattern Type #" +
boost::lexical_cast<std::string>(patternId + 1), false, true);
m_trafficPatterns[patternId].printTrafficConfiguration(m_logger);
m_logger.log("Total Interests Received = " +
boost::lexical_cast<std::string>(
m_trafficPatterns[patternId].m_nInterestsReceived) + "\n", false, true);
}
}
bool
checkTrafficPatternCorrectness()
{
return true;
}
void
parseConfigurationFile()
{
std::string patternLine;
std::ifstream patternFile;
m_logger.log("Analyzing Traffic Configuration File: " + m_configurationFile, true, true);
patternFile.open(m_configurationFile.c_str());
if (patternFile.is_open())
{
int patternId = 0;
int lineNumber = 0;
while (getline(patternFile, patternLine))
{
lineNumber++;
if (std::isalpha(patternLine[0]))
{
DataTrafficConfiguration dataData;
bool shouldSkipLine = false;
patternId++;
if (dataData.processConfigurationDetail(patternLine, m_logger, lineNumber))
{
while (getline(patternFile, patternLine) && std::isalpha(patternLine[0]))
{
lineNumber++;
if (!dataData.processConfigurationDetail(patternLine,
m_logger,
lineNumber))
{
shouldSkipLine = true;
break;
}
}
lineNumber++;
}
else
shouldSkipLine = true;
if (!shouldSkipLine)
{
if (dataData.checkTrafficDetailCorrectness())
m_trafficPatterns.push_back(dataData);
}
}
}
patternFile.close();
if (!checkTrafficPatternCorrectness())
{
m_logger.log("ERROR - Traffic Configuration Provided Is Not Proper- " +
m_configurationFile, false, true);
m_logger.shutdownLogger();
exit(1);
}
m_logger.log("Traffic Configuration File Processing Completed\n", true, false);
for (patternId = 0; patternId < m_trafficPatterns.size(); patternId++)
{
m_logger.log("Traffic Pattern Type #" +
boost::lexical_cast<std::string>(patternId + 1), false, false);
m_trafficPatterns[patternId].printTrafficConfiguration(m_logger);
m_logger.log("", false, false);
}
}
else
{
m_logger.log("ERROR - Unable To Open Traffic Configuration File: " +
m_configurationFile, false, true);
m_logger.shutdownLogger();
exit(1);
}
}
void
initializeTrafficConfiguration()
{
if (boost::filesystem::exists(boost::filesystem::path(m_configurationFile)))
{
if (boost::filesystem::is_regular_file(boost::filesystem::path(m_configurationFile)))
{
parseConfigurationFile();
}
else
{
m_logger.log("ERROR - Traffic Configuration File Is Not A Regular File: " +
m_configurationFile, false, true);
m_logger.shutdownLogger();
exit(1);
}
}
else
{
m_logger.log("ERROR - Traffic Configuration File Does Not Exist: " +
m_configurationFile, false, true);
m_logger.shutdownLogger();
exit(1);
}
}
static std::string
getRandomByteString(int randomSize)
{
std::string randomString;
for (int i = 0; i < randomSize; i++)
randomString += static_cast<char>(std::rand() % 128);
return randomString;
}
void
onInterest(const Name& name, const Interest& interest, int patternId)
{
if (m_nMaximumInterests < 0 || m_nInterestsReceived < m_nMaximumInterests)
{
Data data(interest.getName());
if (m_trafficPatterns[patternId].m_contentType >= 0)
data.setContentType(m_trafficPatterns[patternId].m_contentType);
if (m_trafficPatterns[patternId].m_freshnessPeriod >= time::milliseconds(0))
data.setFreshnessPeriod(m_trafficPatterns[patternId].m_freshnessPeriod);
std::string content = "";
if (m_trafficPatterns[patternId].m_contentBytes >= 0)
content = getRandomByteString(m_trafficPatterns[patternId].m_contentBytes);
if (m_trafficPatterns[patternId].m_content != "")
content = m_trafficPatterns[patternId].m_content;
data.setContent(reinterpret_cast<const uint8_t*>(content.c_str()), content.length());
m_keyChain.sign(data);
m_nInterestsReceived++;
m_trafficPatterns[patternId].m_nInterestsReceived++;
std::string logLine = "Interest Received - PatternType=" +
boost::lexical_cast<std::string>(patternId + 1);
logLine += ", GlobalID=" + boost::lexical_cast<std::string>(m_nInterestsReceived);
logLine += ", LocalID=" +
boost::lexical_cast<std::string>(m_trafficPatterns[patternId].m_nInterestsReceived);
logLine += ", Name=" + m_trafficPatterns[patternId].m_name;
if (!m_hasQuietLogging)
m_logger.log(logLine, true, false);
if (m_trafficPatterns[patternId].m_contentDelay > time::milliseconds(-1))
usleep(m_trafficPatterns[patternId].m_contentDelay.count() * 1000);
if (m_contentDelay > time::milliseconds(-1))
usleep(m_contentDelay.count() * 1000);
m_face.put(data);
}
if (m_nMaximumInterests >= 0 && m_nInterestsReceived == m_nMaximumInterests)
{
logStatistics();
m_logger.shutdownLogger();
m_face.shutdown();
m_ioService.stop();
}
}
void
onRegisterFailed(const ndn::Name& prefix, const std::string& reason, int patternId)
{
std::string logLine = "";
logLine += "Prefix Registration Failed - PatternType=" +
boost::lexical_cast<std::string>(patternId + 1);
logLine += ", Name=" + m_trafficPatterns[patternId].m_name;
m_logger.log(logLine, true, true);
m_nRegistrationsFailed++;
if (m_nRegistrationsFailed == m_trafficPatterns.size())
{
m_hasError = true;
signalHandler();
}
}
void
run()
{
boost::asio::signal_set signalSet(m_ioService, SIGINT, SIGTERM);
signalSet.async_wait(boost::bind(&NdnTrafficServer::signalHandler, this));
m_logger.initializeLog(m_instanceId);
initializeTrafficConfiguration();
if (m_nMaximumInterests == 0)
{
logStatistics();
m_logger.shutdownLogger();
return;
}
for (int patternId = 0; patternId < m_trafficPatterns.size(); patternId++)
{
m_face.setInterestFilter(m_trafficPatterns[patternId].m_name,
bind(&NdnTrafficServer::onInterest,
this, _1, _2,
patternId),
bind(&NdnTrafficServer::onRegisterFailed,
this, _1, _2,
patternId));
}
try {
m_face.processEvents();
}
catch (std::exception& e) {
m_logger.log("ERROR: " + static_cast<std::string>(e.what()), true, true);
m_logger.shutdownLogger();
m_hasError = true;
m_ioService.stop();
}
}
private:
KeyChain m_keyChain;
std::string m_programName;
bool m_hasError;
bool m_hasQuietLogging;
std::string m_instanceId;
time::milliseconds m_contentDelay;
int m_nRegistrationsFailed;
Logger m_logger;
std::string m_configurationFile;
boost::asio::io_service m_ioService;
Face m_face;
std::vector<DataTrafficConfiguration> m_trafficPatterns;
int m_nMaximumInterests;
int m_nInterestsReceived;
};
} // namespace ndn
int
main(int argc, char* argv[])
{
std::srand(std::time(0));
ndn::NdnTrafficServer ndnTrafficServer(argv[0]);
int option;
while ((option = getopt(argc, argv, "hqc:d:")) != -1) {
switch (option) {
case 'h':
ndnTrafficServer.usage();
break;
case 'c':
ndnTrafficServer.setMaximumInterests(atoi(optarg));
break;
case 'd':
ndnTrafficServer.setContentDelay(atoi(optarg));
break;
case 'q':
ndnTrafficServer.setQuietLogging();
break;
default:
ndnTrafficServer.usage();
break;
}
}
argc -= optind;
argv += optind;
if (argv[0] == 0)
ndnTrafficServer.usage();
ndnTrafficServer.setConfigurationFile(argv[0]);
ndnTrafficServer.run();
if (ndnTrafficServer.hasError())
return 1;
else
return 0;
}