blob: cf012d85147eb0f0dc27a2ffc067cb833ffbbe9e [file] [log] [blame]
/* -*- Mode:C++; c-file-style:"gnu"; indent-tabs-mode:nil; -*- */
/*
* Copyright (c) 2013-2021, Regents of the University of California.
*
* This file is part of ndn-tools (Named Data Networking Essential Tools).
* See AUTHORS.md for complete list of ndn-tools authors and contributors.
*
* ndn-tools 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.
*
* ndn-tools 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
* ndn-tools, e.g., in COPYING.md file. If not, see <http://www.gnu.org/licenses/>.
*
* See AUTHORS.md for complete list of ndn-cxx authors and contributors.
*
* @author Alexander Afanasyev <http://lasr.cs.ucla.edu/afanasyev/index.html>
*/
#include "ndn-dissect.hpp"
#include <map>
#include <ndn-cxx/encoding/tlv.hpp>
#include <ndn-cxx/name-component.hpp>
namespace ndn {
namespace dissect {
NdnDissect::NdnDissect(std::istream& input, std::ostream& output)
: m_in(input)
, m_out(output)
{
}
void
NdnDissect::dissect()
{
size_t offset = 0;
try {
while (m_in.peek() != std::istream::traits_type::eof()) {
auto block = Block::fromStream(m_in);
printBlock(block);
offset += block.size();
}
}
catch (const std::exception& e) {
std::cerr << "ERROR: " << e.what() << " at offset " << offset << "\n";
}
}
// http://git.altlinux.org/people/legion/packages/kbd.git?p=kbd.git;a=blob;f=data/consolefonts/README.eurlatgr
static const char GLYPH_VERTICAL[] = "\u2502 "; // "│ "
static const char GLYPH_VERTICAL_AND_RIGHT[] = "\u251c\u2500"; // "├─"
static const char GLYPH_UP_AND_RIGHT[] = "\u2514\u2500"; // "└─"
static const char GLYPH_SPACE[] = " ";
void
NdnDissect::printBranches()
{
for (size_t i = 0; i < m_branches.size(); ++i) {
if (i == m_branches.size() - 1) {
m_out << (m_branches[i] ? GLYPH_VERTICAL_AND_RIGHT : GLYPH_UP_AND_RIGHT);
}
else {
m_out << (m_branches[i] ? GLYPH_VERTICAL : GLYPH_SPACE);
}
}
}
static const std::map<uint32_t, const char*> TLV_DICT = {
{tlv::Interest , "Interest"},
{tlv::Data , "Data"},
{tlv::Name , "Name"},
{tlv::CanBePrefix , "CanBePrefix"},
{tlv::MustBeFresh , "MustBeFresh"},
//{tlv::ForwardingHint , "ForwardingHint"},
{tlv::Nonce , "Nonce"},
{tlv::InterestLifetime , "InterestLifetime"},
{tlv::HopLimit , "HopLimit"},
{tlv::ApplicationParameters , "ApplicationParameters"},
{tlv::MetaInfo , "MetaInfo"},
{tlv::Content , "Content"},
{tlv::SignatureInfo , "SignatureInfo"},
{tlv::SignatureValue , "SignatureValue"},
{tlv::ContentType , "ContentType"},
{tlv::FreshnessPeriod , "FreshnessPeriod"},
{tlv::FinalBlockId , "FinalBlockId"},
{tlv::SignatureType , "SignatureType"},
{tlv::KeyLocator , "KeyLocator"},
{tlv::KeyDigest , "KeyDigest"},
// Name components
{tlv::GenericNameComponent , "GenericNameComponent"},
{tlv::ImplicitSha256DigestComponent , "ImplicitSha256DigestComponent"},
{tlv::ParametersSha256DigestComponent, "ParametersSha256DigestComponent"},
{tlv::KeywordNameComponent , "KeywordNameComponent"},
//{tlv::SegmentNameComponent , "SegmentNameComponent"},
//{tlv::ByteOffsetNameComponent , "ByteOffsetNameComponent"},
//{tlv::VersionNameComponent , "VersionNameComponent"},
//{tlv::TimestampNameComponent , "TimestampNameComponent"},
//{tlv::SequenceNumNameComponent , "SequenceNumNameComponent"},
// Deprecated elements
{tlv::Selectors , "Selectors"},
{tlv::MinSuffixComponents , "MinSuffixComponents"},
{tlv::MaxSuffixComponents , "MaxSuffixComponents"},
{tlv::PublisherPublicKeyLocator , "PublisherPublicKeyLocator"},
{tlv::Exclude , "Exclude"},
{tlv::ChildSelector , "ChildSelector"},
{tlv::Any , "Any"},
};
void
NdnDissect::printType(uint32_t type)
{
m_out << type << " (";
auto it = TLV_DICT.find(type);
if (it != TLV_DICT.end()) {
m_out << it->second;
}
else if (type < tlv::AppPrivateBlock1) {
m_out << "RESERVED_1";
}
else if (tlv::AppPrivateBlock1 <= type && type < 253) {
m_out << "APP_TAG_1";
}
else if (253 <= type && type < tlv::AppPrivateBlock2) {
m_out << "RESERVED_3";
}
else {
m_out << "APP_TAG_3";
}
m_out << ")";
}
void
NdnDissect::printBlock(const Block& block)
{
printBranches();
printType(block.type());
m_out << " (size: " << block.value_size() << ")";
try {
// if (block.type() != tlv::Content && block.type() != tlv::SignatureValue)
block.parse();
}
catch (const tlv::Error&) {
// pass (e.g., leaf block reached)
// @todo: Figure how to deterministically figure out that value is not recursive TLV block
}
const auto& elements = block.elements();
if (elements.empty()) {
m_out << " [[";
name::Component(block.value(), block.value_size()).toUri(m_out);
m_out << "]]";
}
m_out << "\n";
m_branches.push_back(true);
for (size_t i = 0; i < elements.size(); ++i) {
if (i == elements.size() - 1) {
// no more branches to draw at this level of the tree
m_branches.back() = false;
}
printBlock(elements[i]);
}
m_branches.pop_back();
}
} // namespace dissect
} // namespace ndn