name-component.cpp
Go to the documentation of this file.
1 /* -*- Mode:C++; c-file-style:"gnu"; indent-tabs-mode:nil; -*- */
2 /*
3  * Copyright (c) 2013-2019 Regents of the University of California.
4  *
5  * This file is part of ndn-cxx library (NDN C++ library with eXperimental eXtensions).
6  *
7  * 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.
20  *
21  * @author Jeff Thompson <jefft0@remap.ucla.edu>
22  * @author Alexander Afanasyev <http://lasr.cs.ucla.edu/afanasyev/index.html>
23  * @author Zhenkai Zhu <http://irl.cs.ucla.edu/~zhenkai/>
24  */
25 
27 #include "ndn-cxx/impl/name-component-types.hpp"
28 
29 #include <cstdlib>
30 #include <cstring>
31 #include <sstream>
32 
33 namespace ndn {
34 namespace name {
35 
36 BOOST_CONCEPT_ASSERT((boost::EqualityComparable<Component>));
37 BOOST_CONCEPT_ASSERT((WireEncodable<Component>));
38 BOOST_CONCEPT_ASSERT((WireEncodableWithEncodingBuffer<Component>));
39 BOOST_CONCEPT_ASSERT((WireDecodable<Component>));
40 static_assert(std::is_base_of<tlv::Error, Component::Error>::value,
41  "name::Component::Error must inherit from tlv::Error");
42 
45 
48 {
49  return g_conventionEncoding;
50 }
51 
52 void
54 {
55  switch (convention) {
56  case Convention::MARKER:
57  case Convention::TYPED:
58  g_conventionEncoding = convention;
59  break;
60  default:
61  NDN_THROW(std::invalid_argument("Unknown naming convention"));
62  }
63 }
64 
67 {
68  return g_conventionDecoding;
69 }
70 
71 void
73 {
74  g_conventionDecoding = convention;
75 }
76 
77 static bool
79 {
80  return (to_underlying(g_conventionDecoding) & to_underlying(Convention::MARKER)) != 0;
81 }
82 
83 static bool
85 {
86  return (to_underlying(g_conventionDecoding) & to_underlying(Convention::TYPED)) != 0;
87 }
88 
89 static bool
91 {
92  if (format == UriFormat::DEFAULT) {
93  static const char* env = std::getenv("NDN_NAME_ALT_URI");
94  static bool defaultSetting = env == nullptr || env[0] != '0';
95  return defaultSetting;
96  }
97  return format == UriFormat::ALTERNATE;
98 }
99 
100 void
101 Component::ensureValid() const
102 {
104  NDN_THROW(Error("TLV-TYPE " + to_string(type()) + " is not a valid NameComponent"));
105  }
106  detail::getComponentTypeTable().get(type()).check(*this);
107 }
108 
110  : Block(type)
111 {
112  ensureValid();
113 }
114 
116  : Block(wire)
117 {
118  ensureValid();
119 }
120 
122  : Block(type, std::move(buffer))
123 {
124  ensureValid();
125 }
126 
127 Component::Component(uint32_t type, const uint8_t* value, size_t valueLen)
128  : Block(makeBinaryBlock(type, value, valueLen))
129 {
130  ensureValid();
131 }
132 
133 Component::Component(const char* str)
134  : Block(makeBinaryBlock(tlv::GenericNameComponent, str, std::char_traits<char>::length(str)))
135 {
136 }
137 
138 Component::Component(const std::string& str)
140 {
141 }
142 
143 static Component
144 parseUriEscapedValue(uint32_t type, const char* input, size_t len)
145 {
146  std::ostringstream oss;
147  unescape(oss, input, len);
148  std::string value = oss.str();
149  if (value.find_first_not_of('.') == std::string::npos) { // all periods
150  if (value.size() < 3) {
151  NDN_THROW(Component::Error("Illegal URI (name component cannot be . or ..)"));
152  }
153  return Component(type, reinterpret_cast<const uint8_t*>(value.data()), value.size() - 3);
154  }
155  return Component(type, reinterpret_cast<const uint8_t*>(value.data()), value.size());
156 }
157 
158 Component
159 Component::fromEscapedString(const std::string& input)
160 {
161  size_t equalPos = input.find('=');
162  if (equalPos == std::string::npos) {
163  return parseUriEscapedValue(tlv::GenericNameComponent, input.data(), input.size());
164  }
165 
166  auto typePrefix = input.substr(0, equalPos);
167  auto type = std::strtoul(typePrefix.data(), nullptr, 10);
169  to_string(type) == typePrefix) {
170  size_t valuePos = equalPos + 1;
171  return parseUriEscapedValue(static_cast<uint32_t>(type),
172  input.data() + valuePos, input.size() - valuePos);
173  }
174 
175  auto ct = detail::getComponentTypeTable().findByUriPrefix(typePrefix);
176  if (ct == nullptr) {
177  NDN_THROW(Error("Unknown TLV-TYPE '" + typePrefix + "' in NameComponent URI"));
178  }
179  return ct->parseAltUriValue(input.substr(equalPos + 1));
180 }
181 
182 void
183 Component::toUri(std::ostream& os, UriFormat format) const
184 {
185  if (chooseAltUri(format)) {
186  detail::getComponentTypeTable().get(type()).writeUri(os, *this);
187  }
188  else {
189  detail::ComponentType().writeUri(os, *this);
190  }
191 }
192 
193 std::string
195 {
196  std::ostringstream os;
197  toUri(os, format);
198  return os.str();
199 }
200 
202 
203 bool
205 {
206  return value_size() == 1 || value_size() == 2 ||
207  value_size() == 4 || value_size() == 8;
208 }
209 
210 bool
211 Component::isNumberWithMarker(uint8_t marker) const
212 {
213  return (value_size() == 2 || value_size() == 3 ||
214  value_size() == 5 || value_size() == 9) && value()[0] == marker;
215 }
216 
217 bool
219 {
222 }
223 
224 bool
226 {
229 }
230 
231 bool
233 {
236 }
237 
238 bool
240 {
243 }
244 
245 bool
247 {
250 }
251 
253 
254 uint64_t
256 {
257  if (!isNumber())
258  NDN_THROW(Error("Name component does not have nonNegativeInteger value"));
259 
260  return readNonNegativeInteger(*this);
261 }
262 
263 uint64_t
264 Component::toNumberWithMarker(uint8_t marker) const
265 {
266  if (!isNumberWithMarker(marker))
267  NDN_THROW(Error("Name component does not have the requested marker "
268  "or the value is not a nonNegativeInteger"));
269 
270  Buffer::const_iterator valueBegin = value_begin() + 1;
271  return tlv::readNonNegativeInteger(value_size() - 1, valueBegin, value_end());
272 }
273 
274 uint64_t
276 {
279  }
281  return toNumber();
282  }
283  NDN_THROW(Error("Not a Version component"));
284 }
285 
286 uint64_t
288 {
291  }
293  return toNumber();
294  }
295  NDN_THROW(Error("Not a Segment component"));
296 }
297 
298 uint64_t
300 {
303  }
305  return toNumber();
306  }
307  NDN_THROW(Error("Not a ByteOffset component"));
308 }
309 
312 {
313  uint64_t value = 0;
316  }
318  value = toNumber();
319  }
320  else {
321  NDN_THROW(Error("Not a Timestamp component"));
322  }
323  return time::getUnixEpoch() + time::microseconds(value);
324 }
325 
326 uint64_t
328 {
331  }
333  return toNumber();
334  }
335  NDN_THROW(Error("Not a SequenceNumber component"));
336 }
337 
339 
340 Component
341 Component::fromNumber(uint64_t number, uint32_t type)
342 {
343  return makeNonNegativeIntegerBlock(type, number);
344 }
345 
346 Component
347 Component::fromNumberWithMarker(uint8_t marker, uint64_t number)
348 {
349  EncodingEstimator estimator;
350 
351  size_t valueLength = estimator.prependNonNegativeInteger(number);
352  valueLength += estimator.prependByteArray(&marker, 1);
353  size_t totalLength = valueLength;
354  totalLength += estimator.prependVarNumber(valueLength);
355  totalLength += estimator.prependVarNumber(tlv::GenericNameComponent);
356 
357  EncodingBuffer encoder(totalLength, 0);
358  encoder.prependNonNegativeInteger(number);
359  encoder.prependByteArray(&marker, 1);
360  encoder.prependVarNumber(valueLength);
361  encoder.prependVarNumber(tlv::GenericNameComponent);
362 
363  return encoder.block();
364 }
365 
366 Component
367 Component::fromVersion(uint64_t version)
368 {
369  return g_conventionEncoding == Convention::MARKER ?
372 }
373 
374 Component
375 Component::fromSegment(uint64_t segmentNo)
376 {
377  return g_conventionEncoding == Convention::MARKER ?
380 }
381 
382 Component
384 {
385  return g_conventionEncoding == Convention::MARKER ?
388 }
389 
390 Component
392 {
393  uint64_t value = time::duration_cast<time::microseconds>(timePoint - time::getUnixEpoch()).count();
394  return g_conventionEncoding == Convention::MARKER ?
397 }
398 
399 Component
401 {
402  return g_conventionEncoding == Convention::MARKER ?
405 }
406 
408 
409 bool
411 {
412  return type() == tlv::GenericNameComponent;
413 }
414 
415 bool
417 {
418  return detail::getComponentType1().match(*this);
419 }
420 
421 Component
423 {
424  return detail::getComponentType1().create(digest);
425 }
426 
427 Component
428 Component::fromImplicitSha256Digest(const uint8_t* digest, size_t digestSize)
429 {
430  return detail::getComponentType1().create(digest, digestSize);
431 }
432 
433 bool
435 {
436  return detail::getComponentType2().match(*this);
437 }
438 
439 Component
441 {
442  return detail::getComponentType2().create(digest);
443 }
444 
445 Component
446 Component::fromParametersSha256Digest(const uint8_t* digest, size_t digestSize)
447 {
448  return detail::getComponentType2().create(digest, digestSize);
449 }
450 
452 
453 bool
454 Component::equals(const Component& other) const
455 {
456  return type() == other.type() &&
457  value_size() == other.value_size() &&
458  (empty() || // needed with Apple clang < 9.0.0 due to libc++ bug
459  std::equal(value_begin(), value_end(), other.value_begin()));
460 }
461 
462 int
463 Component::compare(const Component& other) const
464 {
465  if (this->hasWire() && other.hasWire()) {
466  // In the common case where both components have wire encoding,
467  // it's more efficient to simply compare the wire encoding.
468  // This works because lexical order of TLV encoding happens to be
469  // the same as canonical order of the value.
470  return std::memcmp(wire(), other.wire(), std::min(size(), other.size()));
471  }
472 
473  int cmpType = type() - other.type();
474  if (cmpType != 0)
475  return cmpType;
476 
477  int cmpSize = value_size() - other.value_size();
478  if (cmpSize != 0)
479  return cmpSize;
480 
481  if (empty())
482  return 0;
483 
484  return std::memcmp(value(), other.value(), value_size());
485 }
486 
487 Component
489 {
490  bool isOverflow = false;
491  Component successor;
492  std::tie(isOverflow, successor) =
493  detail::getComponentTypeTable().get(type()).getSuccessor(*this);
494  if (!isOverflow) {
495  return successor;
496  }
497 
498  uint32_t type = this->type() + 1;
499  const std::vector<uint8_t>& value = detail::getComponentTypeTable().get(type).getMinValue();
500  return Component(type, value.data(), value.size());
501 }
502 
503 template<encoding::Tag TAG>
504 size_t
506 {
507  size_t totalLength = 0;
508  if (value_size() > 0)
509  totalLength += encoder.prependByteArray(value(), value_size());
510  totalLength += encoder.prependVarNumber(value_size());
511  totalLength += encoder.prependVarNumber(type());
512  return totalLength;
513 }
514 
516 
517 const Block&
519 {
520  if (this->hasWire())
521  return *this;
522 
523  EncodingEstimator estimator;
524  size_t estimatedSize = wireEncode(estimator);
525 
526  EncodingBuffer buffer(estimatedSize, 0);
527  wireEncode(buffer);
528 
529  const_cast<Component&>(*this) = buffer.block();
530  return *this;
531 }
532 
533 void
535 {
536  *this = wire;
537  // validity check is done within Component(const Block& wire)
538 }
539 
540 } // namespace name
541 } // namespace ndn
static Component fromParametersSha256Digest(ConstBufferPtr digest)
Create ParametersSha256DigestComponent component.
void setConventionDecoding(Convention convention)
Set which Naming Conventions style(s) to accept while decoding.
static Component fromSequenceNumber(uint64_t seqNo)
Create sequence number component using NDN naming conventions.
Definition: data.cpp:26
bool isSequenceNumber() const
Check if the component is a sequence number per NDN naming conventions.
static Convention g_conventionEncoding
uint64_t toSequenceNumber() const
Interpret as sequence number component using NDN naming conventions.
static Component fromNumberWithMarker(uint8_t marker, uint64_t number)
Create a component encoded as NameComponentWithMarker.
UriFormat
Identify a format of URI representation.
std::string to_string(const T &val)
Definition: backports.hpp:102
static bool canDecodeMarkerConvention()
bool isByteOffset() const
Check if the component is a byte offset per NDN naming conventions.
Component(uint32_t type=tlv::GenericNameComponent)
Construct a NameComponent of TLV-TYPE type, using empty TLV-VALUE.
bool isNumberWithMarker(uint8_t marker) const
Check if the component is a NameComponentWithMarker per NDN naming conventions rev1.
static bool chooseAltUri(UriFormat format)
static Component fromTimestamp(const time::system_clock::TimePoint &timePoint)
Create sequence number component using NDN naming conventions.
const uint8_t * wire() const
Return a raw pointer to the beginning of the encoded wire.
Definition: block.cpp:281
STL namespace.
size_t value_size() const noexcept
Return the size of TLV-VALUE, aka TLV-LENGTH.
Definition: block.cpp:308
static Convention g_conventionDecoding
void setConventionEncoding(Convention convention)
Set which Naming Conventions style to use while encoding.
Represents a TLV element of NDN packet format.
Definition: block.hpp:42
uint64_t readNonNegativeInteger(size_t size, Iterator &begin, Iterator end)
Read nonNegativeInteger in NDN-TLV encoding.
Definition: tlv.hpp:486
bool hasWire() const noexcept
Check if the Block contains a fully encoded wire representation.
Definition: block.hpp:230
int compare(const Component &other) const
Compare this to the other Component using NDN canonical ordering.
Block makeNonNegativeIntegerBlock(uint32_t type, uint64_t value)
Create a TLV block containing a non-negative integer.
uint64_t readNonNegativeInteger(const Block &block)
Read a non-negative integer from a TLV element.
#define NDN_THROW(e)
Definition: exception.hpp:61
typed name components (revision 2)
bool isNumber() const
Check if the component is a nonNegativeInteger.
Buffer::const_iterator value_begin() const
Get begin iterator of TLV-VALUE.
Definition: block.hpp:296
bool isParametersSha256Digest() const
Check if the component is ParametersSha256DigestComponent.
static Component fromSegment(uint64_t segmentNo)
Create segment number component using NDN naming conventions.
Component getSuccessor() const
Get the successor of this name component.
ALTERNATE, unless NDN_NAME_ALT_URI environment variable is set to &#39;0&#39;.
a concept check for TLV abstraction with .wireEncode method
Definition: concepts.hpp:60
Convention getConventionDecoding()
Return which Naming Conventions style(s) to accept while decoding.
size_t size() const
Return the size of the encoded wire, i.e.
Definition: block.cpp:290
const Block & get(uint32_t type) const
Return the first sub-element of the specified TLV-TYPE.
Definition: block.cpp:414
element_const_iterator find(uint32_t type) const
Find the first sub-element of the specified TLV-TYPE.
Definition: block.cpp:426
static Component fromByteOffset(uint64_t offset)
Create byte offset component using NDN naming conventions.
static Component fromEscapedString(const char *input, size_t beginOffset, size_t endOffset)
Decode NameComponent from a URI component.
Block makeBinaryBlock(uint32_t type, const uint8_t *value, size_t length)
Create a TLV block copying TLV-VALUE from raw buffer.
bool isSegment() const
Check if the component is a segment number per NDN naming conventions.
static bool canDecodeTypedConvention()
Block makeStringBlock(uint32_t type, const std::string &value)
Create a TLV block containing a string.
Buffer::const_iterator value_end() const
Get end iterator of TLV-VALUE.
Definition: block.hpp:305
#define NDN_CXX_DEFINE_WIRE_ENCODE_INSTANTIATIONS(ClassName)
std::string unescape(const std::string &str)
Decode a percent-encoded string.
bool isImplicitSha256Digest() const
Check if the component is ImplicitSha256DigestComponent.
bool equals(const Component &other) const
Check if this is the same component as other.
const Block & wireEncode() const
Encode to a wire format.
prefer alternate format when available
uint64_t toNumber() const
Interpret this name component as nonNegativeInteger.
time_point TimePoint
Definition: time.hpp:195
static Component parseUriEscapedValue(uint32_t type, const char *input, size_t len)
const uint8_t * value() const noexcept
Return a raw pointer to the beginning of TLV-VALUE.
Definition: block.cpp:302
Represents a name component.
void wireDecode(const Block &wire)
Decode from the wire format.
uint64_t toVersion() const
Interpret as version component using NDN naming conventions.
uint64_t toSegment() const
Interpret as segment number component using NDN naming conventions.
static Component fromNumber(uint64_t number, uint32_t type=tlv::GenericNameComponent)
Create a component encoded as nonNegativeInteger.
component markers (revision 1)
bool isTimestamp() const
Check if the component is a timestamp per NDN naming conventions.
Convention getConventionEncoding()
Return which Naming Conventions style to use while encoding.
static Component fromVersion(uint64_t version)
Create version component using NDN naming conventions.
uint64_t toNumberWithMarker(uint8_t marker) const
Interpret this name component as NameComponentWithMarker.
bool isGeneric() const
Check if the component is GenericComponent.
Convention
Identify a style of NDN Naming Conventions.
bool isVersion() const
Check if the component is a version per NDN naming conventions.
uint64_t toByteOffset() const
Interpret as byte offset component using NDN naming conventions.
static Component fromImplicitSha256Digest(ConstBufferPtr digest)
Create ImplicitSha256DigestComponent component.
time::system_clock::TimePoint toTimestamp() const
Interpret as timestamp component using NDN naming conventions.
a concept check for TLV abstraction with .wireEncode method
Definition: concepts.hpp:44
constexpr std::underlying_type_t< T > to_underlying(T val) noexcept
Definition: backports.hpp:141
a concept check for TLV abstraction with .wireDecode method and constructible from Block ...
Definition: concepts.hpp:80
void toUri(std::ostream &os, UriFormat format=UriFormat::DEFAULT) const
Write *this to the output stream, escaping characters according to the NDN URI format.
EncodingImpl< EncoderTag > EncodingBuffer
uint32_t type() const
Return the TLV-TYPE of the Block.
Definition: block.hpp:274
const system_clock::TimePoint & getUnixEpoch()
Get system_clock::TimePoint representing UNIX time epoch (00:00:00 on Jan 1, 1970) ...
Definition: time.cpp:106
size_t prependNonNegativeInteger(uint64_t integer) const noexcept
Prepend non-negative integer integer of NDN TLV encoding.
Definition: estimator.cpp:51
EncodingImpl< EstimatorTag > EncodingEstimator
shared_ptr< const Buffer > ConstBufferPtr
Definition: buffer.hpp:126