blob: beca2cda31bae05a33eca856ebb4dd9b86d3df61 [file] [log] [blame]
Alexander Afanasyevc169a812014-05-20 20:37:29 -04001/* -*- Mode:C++; c-file-style:"gnu"; indent-tabs-mode:nil; -*- */
Junxiao Shi81206d52017-07-23 12:43:22 +00002/*
Eric Newberrya3c8bd12020-05-15 17:27:07 -07003 * Copyright (c) 2013-2020 Regents of the University of California.
Alexander Afanasyevdfa52c42014-04-24 21:10:11 -07004 *
5 * This file is part of ndn-cxx library (NDN C++ library with eXperimental eXtensions).
Alexander Afanasyevdfa52c42014-04-24 21:10:11 -07006 *
Alexander Afanasyevc169a812014-05-20 20:37:29 -04007 * ndn-cxx library is free software: you can redistribute it and/or modify it under the
8 * terms of the GNU Lesser General Public License as published by the Free Software
9 * Foundation, either version 3 of the License, or (at your option) any later version.
10 *
11 * ndn-cxx library is distributed in the hope that it will be useful, but WITHOUT ANY
12 * WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A
13 * PARTICULAR PURPOSE. See the GNU Lesser General Public License for more details.
14 *
15 * You should have received copies of the GNU General Public License and GNU Lesser
16 * General Public License along with ndn-cxx, e.g., in COPYING.md file. If not, see
17 * <http://www.gnu.org/licenses/>.
18 *
19 * See AUTHORS.md for complete list of ndn-cxx authors and contributors.
Jeff Thompson5cae5e52013-07-10 19:41:20 -070020 */
21
Davide Pesavento905d40f2020-06-09 21:33:25 -040022#ifndef NDN_CXX_DATA_HPP
23#define NDN_CXX_DATA_HPP
Jeff Thompson5cae5e52013-07-10 19:41:20 -070024
Davide Pesavento14c56cd2020-05-21 01:44:03 -040025#include "ndn-cxx/detail/packet-base.hpp"
26#include "ndn-cxx/encoding/block.hpp"
Davide Pesavento7e780642018-11-24 15:51:34 -050027#include "ndn-cxx/meta-info.hpp"
28#include "ndn-cxx/name.hpp"
Eric Newberrye36aae12020-08-31 22:23:13 -070029#include "ndn-cxx/security/security-common.hpp"
Davide Pesavento81bd6962020-06-17 16:03:23 -040030#include "ndn-cxx/signature-info.hpp"
Jeff Thompson5cae5e52013-07-10 19:41:20 -070031
32namespace ndn {
Alexander Afanasyevaa0e7da2014-03-17 14:37:33 -070033
Davide Pesavento905d40f2020-06-09 21:33:25 -040034/** @brief Represents a %Data packet.
35 * @sa https://named-data.net/doc/NDN-packet-spec/0.3/data.html
Junxiao Shic2b8d242014-11-04 08:35:29 -070036 */
Davide Pesavento1fd00242018-05-20 00:11:01 -040037class Data : public PacketBase, public std::enable_shared_from_this<Data>
Alexander Afanasyev0222fba2014-02-09 23:16:02 -080038{
Jeff Thompson5cae5e52013-07-10 19:41:20 -070039public:
Junxiao Shic2b8d242014-11-04 08:35:29 -070040 class Error : public tlv::Error
Alexander Afanasyevff2d08f2014-04-07 18:28:25 -070041 {
42 public:
Junxiao Shi68b53852018-07-25 13:56:38 -060043 using tlv::Error::Error;
Alexander Afanasyevff2d08f2014-04-07 18:28:25 -070044 };
Alexander Afanasyevaa0e7da2014-03-17 14:37:33 -070045
Junxiao Shi7d9039b2018-04-14 15:56:28 +000046 /** @brief Construct an unsigned Data packet with given @p name and empty Content.
Davide Pesavento905d40f2020-06-09 21:33:25 -040047 * @warning In certain contexts that use `Data::shared_from_this()`, Data must be created using
48 * `std::make_shared`. Otherwise, `shared_from_this()` may trigger undefined behavior.
49 * One example where this is necessary is storing Data into a subclass of InMemoryStorage.
Jeff Thompson20af0732013-09-12 17:01:45 -070050 */
Junxiao Shi9b845352017-07-24 05:04:56 +000051 explicit
Junxiao Shi81206d52017-07-23 12:43:22 +000052 Data(const Name& name = Name());
Alexander Afanasyevaa0e7da2014-03-17 14:37:33 -070053
Junxiao Shi7d9039b2018-04-14 15:56:28 +000054 /** @brief Construct a Data packet by decoding from @p wire.
Davide Pesavento905d40f2020-06-09 21:33:25 -040055 * @param wire TLV block of type tlv::Data; may be signed or unsigned.
56 * @warning In certain contexts that use `Data::shared_from_this()`, Data must be created using
57 * `std::make_shared`. Otherwise, `shared_from_this()` may trigger undefined behavior.
58 * One example where this is necessary is storing Data into a subclass of InMemoryStorage.
Alexander Afanasyev6d48bc12014-02-18 00:10:51 -080059 */
60 explicit
Alexander Afanasyev197e5652014-06-13 16:56:31 -070061 Data(const Block& wire);
Alexander Afanasyevaa0e7da2014-03-17 14:37:33 -070062
Davide Pesavento905d40f2020-06-09 21:33:25 -040063 /** @brief Prepend wire encoding to @p encoder
64 * @param encoder EncodingEstimator or EncodingBuffer instance.
65 * @param wantUnsignedPortionOnly If true, prepend only Name, MetaInfo, Content, and
66 * SignatureInfo to @p encoder, but omit SignatureValue and the outermost TLV
67 * Type and Length of the Data element. This is intended to be used with
68 * wireEncode(EncodingBuffer&, const Block&) const.
69 * @throw Error %Signature is not present and @p wantUnsignedPortionOnly is false.
Jeff Thompsonb7aefa002013-09-16 18:22:00 -070070 */
Alexander Afanasyev74633892015-02-08 18:08:46 -080071 template<encoding::Tag TAG>
Alexander Afanasyevff2d08f2014-04-07 18:28:25 -070072 size_t
Alexander Afanasyevd5c48e02015-06-24 11:58:14 -070073 wireEncode(EncodingImpl<TAG>& encoder, bool wantUnsignedPortionOnly = false) const;
Alexander Afanasyevaa0e7da2014-03-17 14:37:33 -070074
Junxiao Shi81206d52017-07-23 12:43:22 +000075 /** @brief Finalize Data packet encoding with the specified SignatureValue
76 * @param encoder EncodingBuffer containing Name, MetaInfo, Content, and SignatureInfo, but
Davide Pesavento905d40f2020-06-09 21:33:25 -040077 * without SignatureValue and the outermost Type-Length of the Data element.
78 * @param signatureValue SignatureValue element.
Alexander Afanasyev197e5652014-06-13 16:56:31 -070079 *
Davide Pesavento905d40f2020-06-09 21:33:25 -040080 * This method is intended to be used in concert with `wireEncode(encoder, true)`, e.g.:
Junxiao Shi81206d52017-07-23 12:43:22 +000081 * @code
Alexander Afanasyev197e5652014-06-13 16:56:31 -070082 * Data data;
83 * ...
84 * EncodingBuffer encoder;
85 * data.wireEncode(encoder, true);
86 * ...
87 * Block signatureValue = <sign_over_unsigned_portion>(encoder.buf(), encoder.size());
88 * data.wireEncode(encoder, signatureValue)
Junxiao Shi81206d52017-07-23 12:43:22 +000089 * @endcode
Alexander Afanasyev197e5652014-06-13 16:56:31 -070090 */
91 const Block&
92 wireEncode(EncodingBuffer& encoder, const Block& signatureValue) const;
93
Davide Pesavento905d40f2020-06-09 21:33:25 -040094 /** @brief Encode into a Block.
95 * @pre Data must be signed.
Junxiao Shi81206d52017-07-23 12:43:22 +000096 */
97 const Block&
98 wireEncode() const;
99
Davide Pesavento905d40f2020-06-09 21:33:25 -0400100 /** @brief Decode from @p wire.
Alexander Afanasyev809805d2014-02-17 17:20:33 -0800101 */
Alexander Afanasyevff2d08f2014-04-07 18:28:25 -0700102 void
103 wireDecode(const Block& wire);
Alexander Afanasyevfadc97d2014-01-03 13:22:10 -0800104
Junxiao Shi7d9039b2018-04-14 15:56:28 +0000105 /** @brief Check if this instance has cached wire encoding.
Alexander Afanasyev6d48bc12014-02-18 00:10:51 -0800106 */
Alexander Afanasyevff2d08f2014-04-07 18:28:25 -0700107 bool
Davide Pesavento14c56cd2020-05-21 01:44:03 -0400108 hasWire() const noexcept
Junxiao Shi81206d52017-07-23 12:43:22 +0000109 {
110 return m_wire.hasWire();
111 }
Alexander Afanasyevaa0e7da2014-03-17 14:37:33 -0700112
Junxiao Shi81206d52017-07-23 12:43:22 +0000113 /** @brief Get full name including implicit digest
114 * @pre hasWire() == true; i.e. wireEncode() must have been called
115 * @throw Error Data has no wire encoding
Alexander Afanasyev3b703102014-06-13 17:01:14 -0700116 */
117 const Name&
118 getFullName() const;
119
Junxiao Shi81206d52017-07-23 12:43:22 +0000120public: // Data fields
121 /** @brief Get name
122 */
123 const Name&
Davide Pesavento14c56cd2020-05-21 01:44:03 -0400124 getName() const noexcept
Junxiao Shi81206d52017-07-23 12:43:22 +0000125 {
126 return m_name;
127 }
128
129 /** @brief Set name
130 * @return a reference to this Data, to allow chaining
131 */
132 Data&
133 setName(const Name& name);
134
135 /** @brief Get MetaInfo
Alexander Afanasyev770827c2014-05-13 17:42:55 -0700136 */
Alexander Afanasyevff2d08f2014-04-07 18:28:25 -0700137 const MetaInfo&
Davide Pesavento14c56cd2020-05-21 01:44:03 -0400138 getMetaInfo() const noexcept
Junxiao Shi81206d52017-07-23 12:43:22 +0000139 {
140 return m_metaInfo;
141 }
Alexander Afanasyevaa0e7da2014-03-17 14:37:33 -0700142
Junxiao Shi81206d52017-07-23 12:43:22 +0000143 /** @brief Set MetaInfo
144 * @return a reference to this Data, to allow chaining
Jeff Thompson0cd8c4a2013-09-13 17:46:40 -0700145 */
Alexander Afanasyev770827c2014-05-13 17:42:55 -0700146 Data&
Alexander Afanasyev2ba8f662014-01-05 22:53:18 -0800147 setMetaInfo(const MetaInfo& metaInfo);
Jeff Thompson46bd45f2013-08-08 16:46:41 -0700148
Davide Pesavento81bd6962020-06-17 16:03:23 -0400149 /**
150 * @brief Return whether this Data has a Content element
151 */
152 bool
153 hasContent() const noexcept
154 {
155 return m_content.isValid();
156 }
157
158 /**
159 * @brief Get the Content element
Alexander Afanasyev2ba8f662014-01-05 22:53:18 -0800160 *
Davide Pesavento81bd6962020-06-17 16:03:23 -0400161 * If the element is not present (hasContent() == false), an invalid Block will be returned.
162 *
163 * The value of the returned Content Block (if valid) can be accessed through
164 * Block::value() / Block::value_size() or Block::value_begin() / Block::value_end().
165 *
166 * @sa hasContent()
167 * @sa Block::blockFromValue(), Block::parse()
Alexander Afanasyev2ba8f662014-01-05 22:53:18 -0800168 */
Alexander Afanasyevff2d08f2014-04-07 18:28:25 -0700169 const Block&
Davide Pesavento81bd6962020-06-17 16:03:23 -0400170 getContent() const noexcept
171 {
172 return m_content;
173 }
Alexander Afanasyevfadc97d2014-01-03 13:22:10 -0800174
Davide Pesavento81bd6962020-06-17 16:03:23 -0400175 /**
176 * @brief Set Content from a Block
177 * @param block TLV block to be used as Content; must be valid
178 * @return a reference to this Data, to allow chaining
Alexander Afanasyev770827c2014-05-13 17:42:55 -0700179 *
Davide Pesavento81bd6962020-06-17 16:03:23 -0400180 * If the block's TLV-TYPE is tlv::Content, it will be used directly as this Data's
181 * Content element. Otherwise, the block will be nested into a Content element.
Alexander Afanasyev770827c2014-05-13 17:42:55 -0700182 */
183 Data&
184 setContent(const Block& block);
Alexander Afanasyev2ba8f662014-01-05 22:53:18 -0800185
Davide Pesavento81bd6962020-06-17 16:03:23 -0400186 /**
187 * @brief Set Content by copying from a raw buffer
188 * @param value buffer with the TLV-VALUE of the content; may be nullptr if @p length is zero
189 * @param length size of the buffer
190 * @return a reference to this Data, to allow chaining
Alexander Afanasyev770827c2014-05-13 17:42:55 -0700191 */
192 Data&
Davide Pesavento81bd6962020-06-17 16:03:23 -0400193 setContent(const uint8_t* value, size_t length);
Alexander Afanasyev809805d2014-02-17 17:20:33 -0800194
Davide Pesavento81bd6962020-06-17 16:03:23 -0400195 /**
196 * @brief Set Content from a shared buffer
197 * @param value buffer with the TLV-VALUE of the content; must not be nullptr
198 * @return a reference to this Data, to allow chaining
Junxiao Shi81206d52017-07-23 12:43:22 +0000199 */
200 Data&
Davide Pesavento3b101d02018-07-21 22:44:09 -0400201 setContent(ConstBufferPtr value);
Alexander Afanasyevaa0e7da2014-03-17 14:37:33 -0700202
Davide Pesavento81bd6962020-06-17 16:03:23 -0400203 /**
204 * @brief Remove the Content element
205 * @return a reference to this Data, to allow chaining
206 * @post hasContent() == false
207 */
208 Data&
209 unsetContent();
210
Eric Newberrya3c8bd12020-05-15 17:27:07 -0700211 /** @brief Get SignatureInfo
212 */
213 const SignatureInfo&
Davide Pesavento14c56cd2020-05-21 01:44:03 -0400214 getSignatureInfo() const noexcept
Eric Newberrya3c8bd12020-05-15 17:27:07 -0700215 {
216 return m_signatureInfo;
217 }
218
219 /** @brief Set SignatureInfo
Davide Pesavento14c56cd2020-05-21 01:44:03 -0400220 *
221 * This is a low-level function that should not normally be called directly by applications.
222 * Instead, provide a SignatureInfo to the SigningInfo object passed to KeyChain::sign().
223 *
Eric Newberrya3c8bd12020-05-15 17:27:07 -0700224 * @return a reference to this Data, to allow chaining
Davide Pesavento14c56cd2020-05-21 01:44:03 -0400225 * @warning SignatureInfo is overwritten when the packet is signed via KeyChain::sign().
226 * @sa SigningInfo
Eric Newberrya3c8bd12020-05-15 17:27:07 -0700227 */
228 Data&
229 setSignatureInfo(const SignatureInfo& info);
230
231 /** @brief Get SignatureValue
232 */
233 const Block&
Davide Pesavento14c56cd2020-05-21 01:44:03 -0400234 getSignatureValue() const noexcept
Eric Newberrya3c8bd12020-05-15 17:27:07 -0700235 {
236 return m_signatureValue;
237 }
238
Junxiao Shi81206d52017-07-23 12:43:22 +0000239 /** @brief Set SignatureValue
Davide Pesavento14c56cd2020-05-21 01:44:03 -0400240 * @param value buffer containing the TLV-VALUE of the SignatureValue; must not be nullptr
241 *
242 * This is a low-level function that should not normally be called directly by applications.
243 * Instead, use KeyChain::sign() to sign the packet.
244 *
Junxiao Shi81206d52017-07-23 12:43:22 +0000245 * @return a reference to this Data, to allow chaining
Davide Pesavento14c56cd2020-05-21 01:44:03 -0400246 * @warning SignatureValue is overwritten when the packet is signed via KeyChain::sign().
Junxiao Shi81206d52017-07-23 12:43:22 +0000247 */
Alexander Afanasyev770827c2014-05-13 17:42:55 -0700248 Data&
Davide Pesavento14c56cd2020-05-21 01:44:03 -0400249 setSignatureValue(ConstBufferPtr value);
Yingdi Yua4e57672014-02-06 11:16:17 -0800250
Eric Newberrye36aae12020-08-31 22:23:13 -0700251 /** @brief Extract ranges of Data covered by the signature
252 * @throw Error Data cannot be encoded or is missing ranges necessary for signing
253 * @warning The returned pointers will be invalidated if wireDecode() or wireEncode() are called.
254 */
255 InputBuffers
256 extractSignedRanges() const;
257
Junxiao Shi81206d52017-07-23 12:43:22 +0000258public: // MetaInfo fields
259 uint32_t
260 getContentType() const
261 {
262 return m_metaInfo.getType();
263 }
Alexander Afanasyevff2d08f2014-04-07 18:28:25 -0700264
Junxiao Shi81206d52017-07-23 12:43:22 +0000265 Data&
266 setContentType(uint32_t type);
267
Junxiao Shiebfe4a22018-04-01 23:53:40 +0000268 time::milliseconds
Junxiao Shi81206d52017-07-23 12:43:22 +0000269 getFreshnessPeriod() const
270 {
271 return m_metaInfo.getFreshnessPeriod();
272 }
273
274 Data&
Junxiao Shiebfe4a22018-04-01 23:53:40 +0000275 setFreshnessPeriod(time::milliseconds freshnessPeriod);
Junxiao Shi81206d52017-07-23 12:43:22 +0000276
Junxiao Shiebfe4a22018-04-01 23:53:40 +0000277 const optional<name::Component>&
278 getFinalBlock() const
Junxiao Shi81206d52017-07-23 12:43:22 +0000279 {
Junxiao Shiebfe4a22018-04-01 23:53:40 +0000280 return m_metaInfo.getFinalBlock();
Junxiao Shi81206d52017-07-23 12:43:22 +0000281 }
282
283 Data&
Junxiao Shiebfe4a22018-04-01 23:53:40 +0000284 setFinalBlock(optional<name::Component> finalBlockId);
285
Davide Pesavento14c56cd2020-05-21 01:44:03 -0400286public: // SignatureInfo fields
287 /** @brief Get SignatureType
288 * @return tlv::SignatureTypeValue, or -1 to indicate the signature is invalid
289 */
290 int32_t
291 getSignatureType() const noexcept
292 {
293 return m_signatureInfo.getSignatureType();
294 }
295
296 /** @brief Get KeyLocator
297 */
298 optional<KeyLocator>
299 getKeyLocator() const noexcept
300 {
301 return m_signatureInfo.hasKeyLocator() ? make_optional(m_signatureInfo.getKeyLocator()) : nullopt;
302 }
303
Spyridon Mastorakis3b54e852015-04-07 08:03:25 -0700304protected:
Junxiao Shi81206d52017-07-23 12:43:22 +0000305 /** @brief Clear wire encoding and cached FullName
306 * @note This does not clear the SignatureValue.
Jeff Thompsonb7aefa002013-09-16 18:22:00 -0700307 */
Alexander Afanasyevff2d08f2014-04-07 18:28:25 -0700308 void
Junxiao Shi81206d52017-07-23 12:43:22 +0000309 resetWire();
Alexander Afanasyevfadc97d2014-01-03 13:22:10 -0800310
311private:
Alexander Afanasyev809805d2014-02-17 17:20:33 -0800312 Name m_name;
313 MetaInfo m_metaInfo;
Junxiao Shi81206d52017-07-23 12:43:22 +0000314 Block m_content;
Eric Newberrya3c8bd12020-05-15 17:27:07 -0700315 SignatureInfo m_signatureInfo;
316 Block m_signatureValue;
Alexander Afanasyevfadc97d2014-01-03 13:22:10 -0800317
Alexander Afanasyev809805d2014-02-17 17:20:33 -0800318 mutable Block m_wire;
Davide Pesaventoa86300a2020-12-05 14:55:45 -0500319 mutable Name m_fullName; // cached FullName computed from m_wire
Jeff Thompson5cae5e52013-07-10 19:41:20 -0700320};
Alexander Afanasyevfadc97d2014-01-03 13:22:10 -0800321
Davide Pesavento88a0d812017-08-19 21:31:42 -0400322#ifndef DOXYGEN
323extern template size_t
324Data::wireEncode<encoding::EncoderTag>(EncodingBuffer&, bool) const;
325
326extern template size_t
327Data::wireEncode<encoding::EstimatorTag>(EncodingEstimator&, bool) const;
328#endif
329
Alexander Afanasyeva0c5f832014-06-19 13:27:56 -0700330std::ostream&
331operator<<(std::ostream& os, const Data& data);
Alexander Afanasyev809805d2014-02-17 17:20:33 -0800332
Junxiao Shi81206d52017-07-23 12:43:22 +0000333bool
334operator==(const Data& lhs, const Data& rhs);
335
Alexander Afanasyev6d48bc12014-02-18 00:10:51 -0800336inline bool
Junxiao Shi81206d52017-07-23 12:43:22 +0000337operator!=(const Data& lhs, const Data& rhs)
Alexander Afanasyev6d48bc12014-02-18 00:10:51 -0800338{
Junxiao Shi81206d52017-07-23 12:43:22 +0000339 return !(lhs == rhs);
Alexander Afanasyev2ba8f662014-01-05 22:53:18 -0800340}
Alexander Afanasyevaa0e7da2014-03-17 14:37:33 -0700341
Alexander Afanasyevfadc97d2014-01-03 13:22:10 -0800342} // namespace ndn
343
Davide Pesavento905d40f2020-06-09 21:33:25 -0400344#endif // NDN_CXX_DATA_HPP