| /* -*- Mode:C++; c-file-style:"gnu"; indent-tabs-mode:nil; -*- */ |
| /** |
| * Copyright (c) 2013-2016 Regents of the University of California. |
| * |
| * This file is part of ndn-cxx library (NDN C++ library with eXperimental eXtensions). |
| * |
| * ndn-cxx library is free software: you can redistribute it and/or modify it under the |
| * terms of the GNU Lesser General Public License as published by the Free Software |
| * Foundation, either version 3 of the License, or (at your option) any later version. |
| * |
| * ndn-cxx library 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 Lesser General Public License for more details. |
| * |
| * You should have received copies of the GNU General Public License and GNU Lesser |
| * General Public License along with ndn-cxx, 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. |
| */ |
| |
| #include "data.hpp" |
| #include "encoding/block-helpers.hpp" |
| #include "util/crypto.hpp" |
| |
| namespace ndn { |
| |
| BOOST_CONCEPT_ASSERT((boost::EqualityComparable<Data>)); |
| BOOST_CONCEPT_ASSERT((WireEncodable<Data>)); |
| BOOST_CONCEPT_ASSERT((WireEncodableWithEncodingBuffer<Data>)); |
| BOOST_CONCEPT_ASSERT((WireDecodable<Data>)); |
| static_assert(std::is_base_of<tlv::Error, Data::Error>::value, |
| "Data::Error must inherit from tlv::Error"); |
| |
| Data::Data() |
| : m_content(tlv::Content) // empty content |
| { |
| } |
| |
| Data::Data(const Name& name) |
| : m_name(name) |
| { |
| } |
| |
| Data::Data(const Block& wire) |
| { |
| wireDecode(wire); |
| } |
| |
| template<encoding::Tag TAG> |
| size_t |
| Data::wireEncode(EncodingImpl<TAG>& encoder, bool unsignedPortion/* = false*/) const |
| { |
| size_t totalLength = 0; |
| |
| // Data ::= DATA-TLV TLV-LENGTH |
| // Name |
| // MetaInfo |
| // Content |
| // Signature |
| |
| // (reverse encoding) |
| |
| if (!unsignedPortion && !m_signature) |
| { |
| BOOST_THROW_EXCEPTION(Error("Requested wire format, but data packet has not been signed yet")); |
| } |
| |
| if (!unsignedPortion) |
| { |
| // SignatureValue |
| totalLength += encoder.prependBlock(m_signature.getValue()); |
| } |
| |
| // SignatureInfo |
| totalLength += encoder.prependBlock(m_signature.getInfo()); |
| |
| // Content |
| totalLength += encoder.prependBlock(getContent()); |
| |
| // MetaInfo |
| totalLength += getMetaInfo().wireEncode(encoder); |
| |
| // Name |
| totalLength += getName().wireEncode(encoder); |
| |
| if (!unsignedPortion) |
| { |
| totalLength += encoder.prependVarNumber(totalLength); |
| totalLength += encoder.prependVarNumber(tlv::Data); |
| } |
| return totalLength; |
| } |
| |
| |
| template size_t |
| Data::wireEncode<encoding::EncoderTag>(EncodingImpl<encoding::EncoderTag>& encoder, |
| bool unsignedPortion) const; |
| |
| template size_t |
| Data::wireEncode<encoding::EstimatorTag>(EncodingImpl<encoding::EstimatorTag>& encoder, |
| bool unsignedPortion) const; |
| |
| |
| const Block& |
| Data::wireEncode(EncodingBuffer& encoder, const Block& signatureValue) const |
| { |
| size_t totalLength = encoder.size(); |
| totalLength += encoder.appendBlock(signatureValue); |
| |
| encoder.prependVarNumber(totalLength); |
| encoder.prependVarNumber(tlv::Data); |
| |
| const_cast<Data*>(this)->wireDecode(encoder.block()); |
| return m_wire; |
| } |
| |
| const Block& |
| Data::wireEncode() const |
| { |
| if (m_wire.hasWire()) |
| return m_wire; |
| |
| EncodingEstimator estimator; |
| size_t estimatedSize = wireEncode(estimator); |
| |
| EncodingBuffer buffer(estimatedSize, 0); |
| wireEncode(buffer); |
| |
| const_cast<Data*>(this)->wireDecode(buffer.block()); |
| return m_wire; |
| } |
| |
| void |
| Data::wireDecode(const Block& wire) |
| { |
| m_fullName.clear(); |
| m_wire = wire; |
| m_wire.parse(); |
| |
| // Data ::= DATA-TLV TLV-LENGTH |
| // Name |
| // MetaInfo |
| // Content |
| // Signature |
| |
| // Name |
| m_name.wireDecode(m_wire.get(tlv::Name)); |
| |
| // MetaInfo |
| m_metaInfo.wireDecode(m_wire.get(tlv::MetaInfo)); |
| |
| // Content |
| m_content = m_wire.get(tlv::Content); |
| |
| /////////////// |
| // Signature // |
| /////////////// |
| |
| // SignatureInfo |
| m_signature.setInfo(m_wire.get(tlv::SignatureInfo)); |
| |
| // SignatureValue |
| Block::element_const_iterator val = m_wire.find(tlv::SignatureValue); |
| if (val != m_wire.elements_end()) |
| m_signature.setValue(*val); |
| } |
| |
| Data& |
| Data::setName(const Name& name) |
| { |
| onChanged(); |
| m_name = name; |
| |
| return *this; |
| } |
| |
| const Name& |
| Data::getFullName() const |
| { |
| if (m_fullName.empty()) { |
| if (!m_wire.hasWire()) { |
| BOOST_THROW_EXCEPTION(Error("Full name requested, but Data packet does not have wire format " |
| "(e.g., not signed)")); |
| } |
| m_fullName = m_name; |
| m_fullName.appendImplicitSha256Digest(crypto::computeSha256Digest(m_wire.wire(), m_wire.size())); |
| } |
| |
| return m_fullName; |
| } |
| |
| Data& |
| Data::setMetaInfo(const MetaInfo& metaInfo) |
| { |
| onChanged(); |
| m_metaInfo = metaInfo; |
| |
| return *this; |
| } |
| |
| Data& |
| Data::setContentType(uint32_t type) |
| { |
| onChanged(); |
| m_metaInfo.setType(type); |
| |
| return *this; |
| } |
| |
| Data& |
| Data::setFreshnessPeriod(const time::milliseconds& freshnessPeriod) |
| { |
| onChanged(); |
| m_metaInfo.setFreshnessPeriod(freshnessPeriod); |
| |
| return *this; |
| } |
| |
| Data& |
| Data::setFinalBlockId(const name::Component& finalBlockId) |
| { |
| onChanged(); |
| m_metaInfo.setFinalBlockId(finalBlockId); |
| |
| return *this; |
| } |
| |
| const Block& |
| Data::getContent() const |
| { |
| if (m_content.empty()) |
| m_content = makeEmptyBlock(tlv::Content); |
| |
| if (!m_content.hasWire()) |
| m_content.encode(); |
| return m_content; |
| } |
| |
| Data& |
| Data::setContent(const uint8_t* content, size_t contentLength) |
| { |
| onChanged(); |
| |
| m_content = makeBinaryBlock(tlv::Content, content, contentLength); |
| |
| return *this; |
| } |
| |
| Data& |
| Data::setContent(const ConstBufferPtr& contentValue) |
| { |
| onChanged(); |
| |
| m_content = Block(tlv::Content, contentValue); // not a real wire encoding yet |
| |
| return *this; |
| } |
| |
| Data& |
| Data::setContent(const Block& content) |
| { |
| onChanged(); |
| |
| if (content.type() == tlv::Content) |
| m_content = content; |
| else { |
| m_content = Block(tlv::Content, content); |
| } |
| |
| return *this; |
| } |
| |
| Data& |
| Data::setSignature(const Signature& signature) |
| { |
| onChanged(); |
| m_signature = signature; |
| |
| return *this; |
| } |
| |
| Data& |
| Data::setSignatureValue(const Block& value) |
| { |
| onChanged(); |
| m_signature.setValue(value); |
| |
| return *this; |
| } |
| |
| void |
| Data::onChanged() |
| { |
| // The values have changed, so the wire format is invalidated |
| |
| // !!!Note!!! Signature is not invalidated and it is responsibility of |
| // the application to do proper re-signing if necessary |
| |
| m_wire.reset(); |
| m_fullName.clear(); |
| } |
| |
| bool |
| Data::operator==(const Data& other) const |
| { |
| return getName() == other.getName() && |
| getMetaInfo() == other.getMetaInfo() && |
| getContent() == other.getContent() && |
| getSignature() == other.getSignature(); |
| } |
| |
| bool |
| Data::operator!=(const Data& other) const |
| { |
| return !(*this == other); |
| } |
| |
| std::ostream& |
| operator<<(std::ostream& os, const Data& data) |
| { |
| os << "Name: " << data.getName() << "\n"; |
| os << "MetaInfo: " << data.getMetaInfo() << "\n"; |
| os << "Content: (size: " << data.getContent().value_size() << ")\n"; |
| os << "Signature: (type: " << data.getSignature().getType() << |
| ", value_length: "<< data.getSignature().getValue().value_size() << ")"; |
| os << std::endl; |
| |
| return os; |
| } |
| |
| #ifdef NDN_LP_KEEP_LOCAL_CONTROL_HEADER |
| |
| // Permit deprecated usage for gcc only. |
| // clang allows deprecated usage in deprecated functions, so it doesn't need this directive. |
| #pragma GCC diagnostic push |
| #pragma GCC diagnostic ignored "-Wdeprecated-declarations" |
| |
| lp::LocalControlHeaderFacade |
| Data::getLocalControlHeader() |
| { |
| return lp::LocalControlHeaderFacade(*this); |
| } |
| |
| const lp::LocalControlHeaderFacade |
| Data::getLocalControlHeader() const |
| { |
| return lp::LocalControlHeaderFacade(const_cast<Data&>(*this)); |
| } |
| |
| uint64_t |
| Data::getIncomingFaceId() const |
| { |
| return getLocalControlHeader().getIncomingFaceId(); |
| } |
| |
| Data& |
| Data::setIncomingFaceId(uint64_t incomingFaceId) |
| { |
| getLocalControlHeader().setIncomingFaceId(incomingFaceId); |
| return *this; |
| } |
| |
| lp::LocalControlHeaderFacade::CachingPolicy |
| Data::getCachingPolicy() const |
| { |
| return getLocalControlHeader().getCachingPolicy(); |
| } |
| |
| Data& |
| Data::setCachingPolicy(lp::LocalControlHeaderFacade::CachingPolicy cachingPolicy) |
| { |
| getLocalControlHeader().setCachingPolicy(cachingPolicy); |
| return *this; |
| } |
| |
| #pragma GCC diagnostic pop |
| |
| #endif // NDN_LP_KEEP_LOCAL_CONTROL_HEADER |
| |
| } // namespace ndn |