| /* -*- 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 |