blob: ed947de2c12da1a06a1c58f69a7b99970ad70890 [file] [log] [blame]
/* -*- Mode:C++; c-file-style:"gnu"; indent-tabs-mode:nil; -*- */
/*
* Copyright (c) 2014-2024, Regents of the University of California.
*
* This file is part of NDNS (Named Data Networking Domain Name Service).
* See AUTHORS.md for complete list of NDNS authors and contributors.
*
* NDNS 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.
*
* NDNS 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
* NDNS, e.g., in COPYING.md file. If not, see <http://www.gnu.org/licenses/>.
*/
#include "clients/response.hpp"
#include "clients/query.hpp"
#include "ndns-label.hpp"
#include "validator/validator.hpp"
#include "ndns-enum.hpp"
#include "ndns-tlv.hpp"
#include "logger.hpp"
#include "daemon/db-mgr.hpp"
#include "util/util.hpp"
#include "util/cert-helper.hpp"
#include <ndn-cxx/face.hpp>
#include <ndn-cxx/security/key-chain.hpp>
#include <ndn-cxx/security/signing-helpers.hpp>
#include <ndn-cxx/util/io.hpp>
#include <ndn-cxx/util/regex.hpp>
#include <boost/asio/io_service.hpp>
#include <boost/program_options.hpp>
#include <iostream>
#include <tuple>
namespace ndn {
namespace ndns {
NDNS_LOG_INIT(NdnsUpdate);
class NdnsUpdate : boost::noncopyable
{
public:
NdnsUpdate(const Name& zone, const shared_ptr<Data>& update, Face& face)
: m_zone(zone)
, m_interestLifetime(DEFAULT_INTEREST_LIFETIME)
, m_face(face)
, m_validator(NdnsValidatorBuilder::create(face))
, m_update(update)
, m_hasError(false)
{
}
void
start()
{
NDNS_LOG_INFO(" ================ "
<< "start to update RR at Zone = " << this->m_zone
<< " new RR is: " << m_update->getName()
<<" =================== ");
NDNS_LOG_INFO("new RR is signed by: " << m_update->getKeyLocator()->getName());
Interest interest = this->makeUpdateInterest();
NDNS_LOG_TRACE("[* <- *] send Update: " << m_update->getName().toUri());
m_face.expressInterest(interest,
bind(&NdnsUpdate::onData, this, _1, _2),
bind(&NdnsUpdate::onTimeout, this, _1), // nack
bind(&NdnsUpdate::onTimeout, this, _1) //dynamic binding
);
}
void
stop()
{
m_face.getIoContext().stop();
}
private:
void
onData(const Interest&, const Data& data)
{
NDNS_LOG_INFO("get response of Update");
auto [ret, msg] = parseResponse(data);
NDNS_LOG_INFO("Return Code: " << ret << ", and Update " << (ret == UPDATE_OK ? "succeeds" : "fails"));
if (ret != UPDATE_OK) {
m_hasError = true;
}
if (!msg.empty()) {
NDNS_LOG_INFO("Return Msg: " << msg);
}
NDNS_LOG_INFO("to verify the response");
m_validator->validate(data,
bind(&NdnsUpdate::onDataValidated, this, _1),
bind(&NdnsUpdate::onDataValidationFailed, this, _1, _2));
}
static std::tuple<int, std::string>
parseResponse(const Data& data)
{
int ret = -1;
std::string msg;
Block blk = data.getContent();
blk.parse();
Block block = blk.blockFromValue();
block.parse();
auto val = block.elements_begin();
for (; val != block.elements_end(); ++val) {
if (val->type() == ndns::tlv::UpdateReturnCode) { // the first must be return code
ret = readNonNegativeInteger(*val);
}
else if (val->type() == ndns::tlv::UpdateReturnMsg) {
msg = std::string(reinterpret_cast<const char*>(val->value()), val->value_size());
}
}
return {ret, msg};
}
/**
* @brief construct a query (interest) which contains the update information
*/
Interest
makeUpdateInterest()
{
Query q(m_zone, label::NDNS_ITERATIVE_QUERY);
q.setRrLabel(Name().append(ndn::tlv::GenericNameComponent, m_update->wireEncode()));
q.setRrType(label::NDNS_UPDATE_LABEL);
q.setInterestLifetime(m_interestLifetime);
return q.toInterest();
}
private:
void
onTimeout(const ndn::Interest&)
{
NDNS_LOG_TRACE("Update timeout");
m_hasError = true;
this->stop();
}
void
onDataValidated(const Data&)
{
NDNS_LOG_INFO("data pass verification");
this->stop();
}
void
onDataValidationFailed(const Data&, const security::ValidationError&)
{
NDNS_LOG_INFO("data does not pass verification");
m_hasError = true;
this->stop();
}
public:
void
setInterestLifetime(const time::milliseconds& interestLifetime)
{
m_interestLifetime = interestLifetime;
}
bool
hasError() const
{
return m_hasError;
}
private:
Name m_zone;
time::milliseconds m_interestLifetime;
Face& m_face;
unique_ptr<security::Validator> m_validator;
KeyChain m_keyChain;
shared_ptr<Data> m_update;
bool m_hasError;
};
} // namespace ndns
} // namespace ndn
int
main(int argc, char* argv[])
{
using std::string;
using namespace ndn;
using namespace ndn::ndns;
Name zone;
int ttl = 4;
Name rrLabel;
string rrType = "TXT";
string contentTypeStr = "resp";
Name certName;
std::vector<string> contents;
string contentFile;
shared_ptr<Data> update;
try {
namespace po = boost::program_options;
po::variables_map vm;
po::options_description generic("Generic Options");
generic.add_options()("help,h", "print help message");
po::options_description config("Configuration");
config.add_options()
("ttl,T", po::value<int>(&ttl), "TTL of query. default: 4 sec")
("rrtype,t", po::value<string>(&rrType), "set request RR Type. default: TXT")
("contentType,n", po::value<string>(&contentTypeStr), "Set the contentType of the resource record. "
"Potential values are [blob|link|nack|auth|resp]. Default: resp")
("cert,c", po::value<Name>(&certName), "set the name of certificate to sign the update")
("content,o", po::value<std::vector<string>>(&contents)->multitoken(),
"set the content of the RR")
("contentFile,f", po::value<string>(&contentFile), "set the path of file which contain"
" Response packet in base64 format")
;
po::options_description hidden("Hidden Options");
hidden.add_options()
("zone,z", po::value<Name>(&zone), "zone the record is delegated")
("rrlabel,l", po::value<Name>(&rrLabel), "set request RR Label")
;
po::positional_options_description postion;
postion.add("zone", 1);
postion.add("rrlabel", 1);
po::options_description cmdline_options;
cmdline_options.add(generic).add(config).add(hidden);
po::options_description config_file_options;
config_file_options.add(config).add(hidden);
po::options_description visible("Usage: ndns-update zone rrLabel [-t rrType] [-T TTL] "
"[-n NdnsContentType] [-c cert] "
"[-f contentFile]|[-o content]\n"
"Allowed options");
visible.add(generic).add(config);
po::parsed_options parsed =
po::command_line_parser(argc, argv).options(cmdline_options).positional(postion).run();
po::store(parsed, vm);
po::notify(vm);
if (vm.count("help")) {
std::cout << visible << std::endl;
return 0;
}
if (vm.count("content") && vm.count("contentFile")) {
std::cerr << "both -o content and -f contentFile are set. Only one is allowed" << std::endl;
return 1;
}
if (!vm.count("contentFile")) {
NDNS_LOG_TRACE("content option is set. try to figure out the certificate");
if (!vm.count("zone") || !vm.count("rrlabel")) {
std::cerr << "-o option must be set together with -z zone and -r rrLabel" << std::endl;
return 1;
}
KeyChain keyChain;
if (certName.empty()) {
Name name = Name().append(zone).append(rrLabel);
// choosing the longest match of the identity who also have default certificate
for (size_t i = name.size() + 1; i > 0; --i) { // i >=0 will present warnning
Name tmp = name.getPrefix(i - 1);
if (CertHelper::doesIdentityExist(keyChain, tmp)) {
try {
certName = CertHelper::getDefaultCertificateNameOfIdentity(keyChain, tmp);
break;
}
catch (const std::exception&) {
// If it cannot get a default certificate from one identity,
// just ignore this one try next identity.
}
}
} // for
if (certName.empty()) {
std::cerr << "cannot figure out the certificate automatically. "
<< "please set it with -c CERT_NAEME" << std::endl;
return 1;
}
}
NdnsContentType contentType = toNdnsContentType(contentTypeStr);
if (contentType == ndns::NDNS_UNKNOWN) {
std::cerr << "unknown NdnsContentType: " << contentTypeStr << std::endl;
return 1;
}
Response re;
re.setZone(zone);
re.setRrLabel(rrLabel);
name::Component qType = ndns::label::NDNS_ITERATIVE_QUERY;
re.setQueryType(qType);
re.setRrType(name::Component(rrType));
re.setContentType(contentType);
for (const auto& content : contents) {
re.addRr(makeStringBlock(ndns::tlv::RrData, content));
// re.addRr(content);
}
update = re.toData();
keyChain.sign(*update, security::signingByCertificate(certName));
}
else {
try {
update = ndn::io::load<ndn::Data>(contentFile);
NDNS_LOG_TRACE("load data " << update->getName() << " from content file: " << contentFile);
}
catch (const std::exception& e) {
std::cerr << "Error: load Data packet from file: " << contentFile
<< ". Due to: " << e.what() << std::endl;
return 1;
}
try {
// must check the Data is a legal Response with right name
shared_ptr<Regex> regex = make_shared<Regex>("(<>*)<NDNS>(<>+)<CERT><>*");
shared_ptr<Regex> regex2 = make_shared<Regex>("(<>*)<NDNS>(<>+)");
Name zone2;
if (regex->match(update->getName())) {
zone2 = regex->expand("\\1");
}
else if (regex2->match(update->getName())) {
zone2 = regex2->expand("\\1");
}
else {
std::cerr << "The loaded Data packet cannot be stored in NDNS "
"since its does not have a proper name" << std::endl;
return 1;
}
if (vm.count("zone") && zone != zone2) {
std::cerr << "The loaded Data packet is supposed to be stored at zone: " << zone2
<< " instead of zone: " << zone << std::endl;
return 1;
}
else {
zone = zone2;
}
Response re;
re.fromData(zone, *update);
if (vm.count("rrlabel") && rrLabel != re.getRrLabel()) {
std::cerr << "The loaded Data packet is supposed to have rrLabel: " << re.getRrLabel()
<< " instead of label: " << rrLabel << std::endl;
return 1;
}
if (vm.count("rrtype") && name::Component(rrType) != re.getRrType()) {
std::cerr << "The loaded Data packet is supposed to have rrType: " << re.getRrType()
<< " instead of label: " << rrType << std::endl;
return 1;
}
}
catch (const std::exception&) {
std::cerr << "Error: the loaded Data packet cannot parse to a Response stored at zone: "
<< zone << std::endl;
return 1;
}
}
}
catch (const std::exception& ex) {
std::cerr << "Parameter Error: " << ex.what() << std::endl;
return 1;
}
Face face;
try {
NdnsUpdate updater(zone, update, face);
updater.setInterestLifetime(ndn::time::seconds(ttl));
updater.start();
face.processEvents();
if (updater.hasError())
return 1;
else
return 0;
}
catch (const std::exception& e) {
std::cerr << "Error: " << e.what() << std::endl;
return 1;
}
}