blob: 38179470400d0a81372f1e11eaffb0f1e613af34 [file] [log] [blame]
Alexander Afanasyevc169a812014-05-20 20:37:29 -04001/* -*- Mode:C++; c-file-style:"gnu"; indent-tabs-mode:nil; -*- */
Jeff Thompson47eecfc2013-07-07 22:56:46 -07002/**
Alexander Afanasyev4f512fb2016-05-18 10:47:53 -07003 * Copyright (c) 2013-2016 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.
Alexander Afanasyevdfa52c42014-04-24 21:10:11 -070020 *
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/>
Jeff Thompson9c41dfe2013-06-27 12:10:25 -070024 */
25
26#ifndef NDN_NAME_HPP
Jeff Thompson2d27e2f2013-08-09 12:55:00 -070027#define NDN_NAME_HPP
Jeff Thompson9c41dfe2013-06-27 12:10:25 -070028
Alexander Afanasyev95e8c2f2014-02-06 17:29:30 -080029#include "common.hpp"
30#include "name-component.hpp"
Jeff Thompson25b4e612013-10-10 16:03:24 -070031
Alexander Afanasyev95e8c2f2014-02-06 17:29:30 -080032#include <boost/iterator/reverse_iterator.hpp>
33
Jeff Thompson9c41dfe2013-06-27 12:10:25 -070034namespace ndn {
Alexander Afanasyev95e8c2f2014-02-06 17:29:30 -080035
Joao Pereira6f7cfd02015-06-15 11:36:26 -040036class Name;
37
Jeff Thompsonc7d65502013-11-06 17:22:26 -080038/**
Joao Pereira6f7cfd02015-06-15 11:36:26 -040039 * @brief Partial name abstraction to represent an arbitrary sequence of name components
40 */
41typedef Name PartialName;
42
43/**
44 * @brief Name abstraction to represent an absolute name
Jeff Thompsonc7d65502013-11-06 17:22:26 -080045 */
Alexander Afanasyevf73f0632014-05-12 18:02:37 -070046class Name : public enable_shared_from_this<Name>
Alexander Afanasyev2a7f7202014-04-23 14:25:29 -070047{
Jeff Thompson9c41dfe2013-06-27 12:10:25 -070048public:
Joao Pereira6f7cfd02015-06-15 11:36:26 -040049 /**
50 * @brief Error that can be thrown from Name
51 */
Alexander Afanasyev2a7f7202014-04-23 14:25:29 -070052 class Error : public name::Component::Error
53 {
Alexander Afanasyev1dd95c52014-03-22 19:11:36 -070054 public:
Alexander Afanasyev2a7f7202014-04-23 14:25:29 -070055 explicit
Alexander Afanasyev1dd95c52014-03-22 19:11:36 -070056 Error(const std::string& what)
Alexander Afanasyev2a7f7202014-04-23 14:25:29 -070057 : name::Component::Error(what)
58 {
59 }
Alexander Afanasyev1dd95c52014-03-22 19:11:36 -070060 };
Alexander Afanasyevaf283d82014-01-03 13:23:34 -080061
Alexander Afanasyev95e8c2f2014-02-06 17:29:30 -080062 typedef name::Component Component;
63
64 typedef std::vector<Component> component_container;
65
66 typedef Component value_type;
67 typedef void allocator_type;
68 typedef Component& reference;
69 typedef const Component const_reference;
70 typedef Component* pointer;
71 typedef const Component* const_pointer;
72 typedef Component* iterator;
73 typedef const Component* const_iterator;
Alexander Afanasyev4b98e8c2014-03-22 19:10:19 -070074
Alexander Afanasyev95e8c2f2014-02-06 17:29:30 -080075 typedef boost::reverse_iterator<iterator> reverse_iterator;
76 typedef boost::reverse_iterator<const_iterator> const_reverse_iterator;
77
78 typedef component_container::difference_type difference_type;
79 typedef component_container::size_type size_type;
Alexander Afanasyev4b98e8c2014-03-22 19:10:19 -070080
Jeff Thompson443398d2013-07-02 19:45:46 -070081 /**
Alexander Afanasyevc89efb42015-02-10 18:26:42 -080082 * @brief Create a new Name with no components.
Jeff Thompson443398d2013-07-02 19:45:46 -070083 */
Alexander Afanasyevc89efb42015-02-10 18:26:42 -080084 Name();
Alexander Afanasyev52eb20d2014-02-06 18:25:54 -080085
86 /**
87 * @brief Create Name object from wire block
88 *
89 * This is a more efficient equivalent for
90 * @code
91 * Name name;
92 * name.wireDecode(wire);
93 * @endcode
94 */
Alexander Afanasyev6d48bc12014-02-18 00:10:51 -080095 explicit
Alexander Afanasyevc89efb42015-02-10 18:26:42 -080096 Name(const Block& wire);
Alexander Afanasyev4b98e8c2014-03-22 19:10:19 -070097
Jeff Thompson3f2175b2013-07-31 17:12:47 -070098 /**
Alexander Afanasyevc89efb42015-02-10 18:26:42 -080099 * @brief Create name from @p uri (NDN URI scheme)
100 * @param uri The null-terminated URI string
Jeff Thompson443398d2013-07-02 19:45:46 -0700101 */
Alexander Afanasyevc89efb42015-02-10 18:26:42 -0800102 Name(const char* uri);
Alexander Afanasyev4b98e8c2014-03-22 19:10:19 -0700103
Jeff Thompsone5f839b2013-06-28 12:50:38 -0700104 /**
Alexander Afanasyevc89efb42015-02-10 18:26:42 -0800105 * @brief Create name from @p uri (NDN URI scheme)
106 * @param uri The URI string
Jeff Thompson3549ef32013-09-25 14:05:17 -0700107 */
Junxiao Shi3188c4032016-07-18 20:53:56 +0000108 Name(std::string uri);
Jeff Thompsonec39fbd2013-10-04 10:56:23 -0700109
Alexander Afanasyev380420b2014-02-09 20:52:29 -0800110 /**
Alexander Afanasyev4f512fb2016-05-18 10:47:53 -0700111 * @brief Make a deep copy of the name, reallocating the underlying memory buffer
112 */
113 Name
114 deepCopy() const;
115
116 /**
Alexander Afanasyev380420b2014-02-09 20:52:29 -0800117 * @brief Fast encoding or block size estimation
118 */
Alexander Afanasyev74633892015-02-08 18:08:46 -0800119 template<encoding::Tag TAG>
Wentao Shang77949212014-02-01 23:42:24 -0800120 size_t
Alexander Afanasyevd5c48e02015-06-24 11:58:14 -0700121 wireEncode(EncodingImpl<TAG>& encoder) const;
Alexander Afanasyev4b98e8c2014-03-22 19:10:19 -0700122
Alexander Afanasyev1dd95c52014-03-22 19:11:36 -0700123 const Block&
Alexander Afanasyevaf283d82014-01-03 13:23:34 -0800124 wireEncode() const;
Alexander Afanasyev848c61a2014-01-03 13:52:04 -0800125
126 void
Alexander Afanasyev1dd95c52014-03-22 19:11:36 -0700127 wireDecode(const Block& wire);
Alexander Afanasyev6d48bc12014-02-18 00:10:51 -0800128
129 /**
130 * @brief Check if already has wire
131 */
132 bool
133 hasWire() const;
Alexander Afanasyev4b98e8c2014-03-22 19:10:19 -0700134
Jeff Thompsonb468c312013-07-01 17:50:14 -0700135 /**
Alexander Afanasyevc89efb42015-02-10 18:26:42 -0800136 * @brief Append a new component, copying from value of length valueLength.
Jeff Thompson26b0d792013-09-23 16:19:01 -0700137 * @return This name so that you can chain calls to append.
Jeff Thompsone5f839b2013-06-28 12:50:38 -0700138 */
Alexander Afanasyev4b98e8c2014-03-22 19:10:19 -0700139 Name&
Alexander Afanasyev1dd95c52014-03-22 19:11:36 -0700140 append(const uint8_t* value, size_t valueLength)
Jeff Thompson0f743452013-09-12 14:23:18 -0700141 {
Alexander Afanasyev29e5c3d2014-02-11 00:01:10 -0800142 m_nameBlock.push_back(Component(value, valueLength));
Jeff Thompson26b0d792013-09-23 16:19:01 -0700143 return *this;
Jeff Thompsone5f839b2013-06-28 12:50:38 -0700144 }
Jeff Thompsonf72b1ac2013-08-16 16:44:41 -0700145
Alexander Afanasyevbf9671d2014-02-11 13:44:13 -0800146 /**
Alexander Afanasyev74633892015-02-08 18:08:46 -0800147 * @brief Append a new component, copying from value frome the range [@p first, @p last) of bytes
148 * @param first Iterator pointing to the beginning of the buffer
149 * @param last Iterator pointing to the ending of the buffer
150 * @tparam Iterator iterator type satisfying at least InputIterator concept. Implementation
151 * is more optimal when the iterator type satisfies RandomAccessIterator concept.
152 * It is required that sizeof(std::iterator_traits<Iterator>::value_type) == 1.
Alexander Afanasyevbf9671d2014-02-11 13:44:13 -0800153 * @return This name so that you can chain calls to append.
154 */
Alexander Afanasyev74633892015-02-08 18:08:46 -0800155 template<class Iterator>
Alexander Afanasyevbf9671d2014-02-11 13:44:13 -0800156 Name&
Alexander Afanasyev74633892015-02-08 18:08:46 -0800157 append(Iterator first, Iterator last)
Alexander Afanasyevbf9671d2014-02-11 13:44:13 -0800158 {
Alexander Afanasyev74633892015-02-08 18:08:46 -0800159 m_nameBlock.push_back(Component(first, last));
Alexander Afanasyevbf9671d2014-02-11 13:44:13 -0800160 return *this;
161 }
162
Alexander Afanasyevc89efb42015-02-10 18:26:42 -0800163 /**
164 * @brief Append component @p value
165 */
Alexander Afanasyev4b98e8c2014-03-22 19:10:19 -0700166 Name&
Alexander Afanasyev1dd95c52014-03-22 19:11:36 -0700167 append(const Component& value)
Jeff Thompson21eb7212013-09-26 09:05:40 -0700168 {
Alexander Afanasyev29e5c3d2014-02-11 00:01:10 -0800169 m_nameBlock.push_back(value);
Jeff Thompson21eb7212013-09-26 09:05:40 -0700170 return *this;
171 }
Alexander Afanasyevaf283d82014-01-03 13:23:34 -0800172
Alexander Afanasyev594cdb22014-01-03 15:11:33 -0800173 /**
174 * @brief Append name component that represented as a string
175 *
176 * Note that this method is necessary to ensure correctness and unambiguity of
177 * ``append("string")`` operations (both Component and Name can be implicitly
178 * converted from string, each having different outcomes
179 */
Alexander Afanasyev4b98e8c2014-03-22 19:10:19 -0700180 Name&
Alexander Afanasyev1dd95c52014-03-22 19:11:36 -0700181 append(const char* value)
Alexander Afanasyev594cdb22014-01-03 15:11:33 -0800182 {
Alexander Afanasyev29e5c3d2014-02-11 00:01:10 -0800183 m_nameBlock.push_back(Component(value));
Alexander Afanasyev594cdb22014-01-03 15:11:33 -0800184 return *this;
185 }
Alexander Afanasyev4b98e8c2014-03-22 19:10:19 -0700186
Alexander Afanasyevaf283d82014-01-03 13:23:34 -0800187 Name&
Alexander Afanasyev1dd95c52014-03-22 19:11:36 -0700188 append(const Block& value)
Alexander Afanasyevaf283d82014-01-03 13:23:34 -0800189 {
Steve DiBenedetto54ce6682014-07-22 13:22:57 -0600190 if (value.type() == tlv::NameComponent)
Alexander Afanasyev29e5c3d2014-02-11 00:01:10 -0800191 m_nameBlock.push_back(value);
Alexander Afanasyev380420b2014-02-09 20:52:29 -0800192 else
Steve DiBenedetto54ce6682014-07-22 13:22:57 -0600193 m_nameBlock.push_back(Block(tlv::NameComponent, value));
Alexander Afanasyev380420b2014-02-09 20:52:29 -0800194
Alexander Afanasyevaf283d82014-01-03 13:23:34 -0800195 return *this;
196 }
Alexander Afanasyev4b98e8c2014-03-22 19:10:19 -0700197
Jeff Thompsone5f839b2013-06-28 12:50:38 -0700198 /**
Joao Pereira6f7cfd02015-06-15 11:36:26 -0400199 * @brief append a PartialName to this Name.
200 * @param name the components to append
201 * @return this name
Jeff Thompson0aa66f22013-09-23 13:02:13 -0700202 */
Jeff Thompson26b0d792013-09-23 16:19:01 -0700203 Name&
Joao Pereira6f7cfd02015-06-15 11:36:26 -0400204 append(const PartialName& name);
Alexander Afanasyev4b98e8c2014-03-22 19:10:19 -0700205
Jeff Thompson0aa66f22013-09-23 13:02:13 -0700206 /**
Jeff Thompsone5f839b2013-06-28 12:50:38 -0700207 * Clear all the components.
208 */
Alexander Afanasyev4b98e8c2014-03-22 19:10:19 -0700209 void
Alexander Afanasyev95e8c2f2014-02-06 17:29:30 -0800210 clear()
211 {
Steve DiBenedetto54ce6682014-07-22 13:22:57 -0600212 m_nameBlock = Block(tlv::Name);
Jeff Thompsone5f839b2013-06-28 12:50:38 -0700213 }
Alexander Afanasyev4b98e8c2014-03-22 19:10:19 -0700214
Jeff Thompsone5f839b2013-06-28 12:50:38 -0700215 /**
Joao Pereira6f7cfd02015-06-15 11:36:26 -0400216 * @brief Extract a sub-name (PartialName) of @p nComponents components starting
217 * from @p iStartComponent
218 * @param iStartComponent index of the first component;
219 * if iStartComponent is negative, size()+iStartComponent is used instead
Jeff Thompsond0159d72013-09-23 13:34:15 -0700220 * @param nComponents The number of components starting at iStartComponent.
Joao Pereira6f7cfd02015-06-15 11:36:26 -0400221 * Use npos to get the Partial Name until the end of this Name.
Davide Pesavento18cf81b2015-09-12 23:36:43 +0200222 * @details If iStartComponent is out of bounds and is negative, returns the components
223 * starting from the beginning of the Name.
224 * If iStartComponent is out of bounds and is positive, returns the component "/".
225 * If nComponents is out of bounds, returns the components until the end of
226 * this Name
Joao Pereira6f7cfd02015-06-15 11:36:26 -0400227 * @return A new partial name
Jeff Thompsond0159d72013-09-23 13:34:15 -0700228 */
Joao Pereira6f7cfd02015-06-15 11:36:26 -0400229 PartialName
230 getSubName(ssize_t iStartComponent, size_t nComponents = npos) const;
Alexander Afanasyev4b98e8c2014-03-22 19:10:19 -0700231
Jeff Thompsond0159d72013-09-23 13:34:15 -0700232 /**
Joao Pereira6f7cfd02015-06-15 11:36:26 -0400233 * @brief Extract a prefix (PartialName) of the name, containing first @p nComponents components
Alexander Afanasyev15f67312014-07-22 15:11:09 -0700234 *
235 * @param nComponents The number of prefix components. If nComponents is -N then return
236 * the prefix up to name.size() - N. For example getPrefix(-1)
237 * returns the name without the final component.
Joao Pereira6f7cfd02015-06-15 11:36:26 -0400238 * @return A new partial name
Jeff Thompsond0159d72013-09-23 13:34:15 -0700239 */
Joao Pereira6f7cfd02015-06-15 11:36:26 -0400240 PartialName
Alexander Afanasyev15f67312014-07-22 15:11:09 -0700241 getPrefix(ssize_t nComponents) const
Jeff Thompsond0159d72013-09-23 13:34:15 -0700242 {
Jeff Thompsoneb0358f2013-12-17 10:59:53 -0800243 if (nComponents < 0)
Alexander Afanasyev29e5c3d2014-02-11 00:01:10 -0800244 return getSubName(0, m_nameBlock.elements_size() + nComponents);
Jeff Thompsoneb0358f2013-12-17 10:59:53 -0800245 else
246 return getSubName(0, nComponents);
Jeff Thompsond0159d72013-09-23 13:34:15 -0700247 }
Alexander Afanasyev4b98e8c2014-03-22 19:10:19 -0700248
Jeff Thompsond0159d72013-09-23 13:34:15 -0700249 /**
Jeff Thompsone6063512013-07-01 15:11:28 -0700250 * Encode this name as a URI.
Jeff Thompson3f2175b2013-07-31 17:12:47 -0700251 * @return The encoded URI.
Jeff Thompsone6063512013-07-01 15:11:28 -0700252 */
Alexander Afanasyev4b98e8c2014-03-22 19:10:19 -0700253 std::string
Jeff Thompson0050abe2013-09-17 12:50:25 -0700254 toUri() const;
Jeff Thompsond129ac12013-10-11 14:30:12 -0700255
256 /**
Alexander Afanasyev4b98e8c2014-03-22 19:10:19 -0700257 * @brief Append a component with the number encoded as nonNegativeInteger
258 *
259 * @see http://named-data.net/doc/ndn-tlv/tlv.html#non-negative-integer-encoding
260 *
Steve DiBenedettoc145d492014-03-11 16:35:45 -0600261 * @param number The non-negative number
262 * @return This name so that you can chain calls to append.
263 */
264 Name&
Alexander Afanasyev15f67312014-07-22 15:11:09 -0700265 appendNumber(uint64_t number);
Steve DiBenedettoc145d492014-03-11 16:35:45 -0600266
Jeff Thompsonec7789a2013-08-21 11:08:36 -0700267 /**
Alexander Afanasyev15f67312014-07-22 15:11:09 -0700268 * @brief Create a component encoded as NameComponentWithMarker
269 *
270 * @see http://named-data.net/doc/tech-memos/naming-conventions.pdf
271 *
272 * @param marker 1-byte marker octet
273 * @param number The non-negative number
Alexander Afanasyev4b98e8c2014-03-22 19:10:19 -0700274 */
275 Name&
Alexander Afanasyev15f67312014-07-22 15:11:09 -0700276 appendNumberWithMarker(uint8_t marker, uint64_t number);
Alexander Afanasyev4b98e8c2014-03-22 19:10:19 -0700277
278 /**
Alexander Afanasyev15f67312014-07-22 15:11:09 -0700279 * @brief Append version using NDN naming conventions
280 *
281 * @see http://named-data.net/doc/tech-memos/naming-conventions.pdf
282 */
283 Name&
284 appendVersion(uint64_t version);
285
286 /**
287 * @brief Append version using NDN naming conventions based on current UNIX timestamp
288 * in milliseconds
289 *
290 * @see http://named-data.net/doc/tech-memos/naming-conventions.pdf
Alexander Afanasyev4b98e8c2014-03-22 19:10:19 -0700291 */
292 Name&
293 appendVersion();
294
295 /**
Alexander Afanasyev15f67312014-07-22 15:11:09 -0700296 * @brief Append segment number (sequential) using NDN naming conventions
297 *
298 * @see http://named-data.net/doc/tech-memos/naming-conventions.pdf
Alexander Afanasyev4b98e8c2014-03-22 19:10:19 -0700299 */
300 Name&
Alexander Afanasyev15f67312014-07-22 15:11:09 -0700301 appendSegment(uint64_t segmentNo);
Alexander Afanasyev4b98e8c2014-03-22 19:10:19 -0700302
303 /**
Alexander Afanasyev15f67312014-07-22 15:11:09 -0700304 * @brief Append segment byte offset using NDN naming conventions
305 *
306 * @see http://named-data.net/doc/tech-memos/naming-conventions.pdf
307 */
308 Name&
309 appendSegmentOffset(uint64_t offset);
310
311 /**
312 * @brief Append timestamp using NDN naming conventions
313 *
314 * @see http://named-data.net/doc/tech-memos/naming-conventions.pdf
315 */
316 Name&
317 appendTimestamp(const time::system_clock::TimePoint& timePoint = time::system_clock::now());
318
319 /**
320 * @brief Append sequence number using NDN naming conventions
321 *
322 * @see http://named-data.net/doc/tech-memos/naming-conventions.pdf
323 */
324 Name&
325 appendSequenceNumber(uint64_t seqNo);
326
327 /**
Alexander Afanasyev6486d522014-10-23 14:14:11 -0700328 * @brief Append ImplicitSha256Digest
329 */
330 Name&
331 appendImplicitSha256Digest(const ConstBufferPtr& digest);
332
333 /**
334 * @brief Append ImplicitSha256Digest
335 */
336 Name&
337 appendImplicitSha256Digest(const uint8_t* digest, size_t digestSize);
338
339 /**
Alexander Afanasyev15f67312014-07-22 15:11:09 -0700340 * @brief Get the successor of a name
341 *
342 * The successor of a name is defined as follows:
343 *
344 * N represents the set of NDN Names, and X,Y ∈ N.
345 * Operator < is defined by canonical order on N.
346 * Y is the successor of X, if (a) X < Y, and (b) ∄ Z ∈ N s.t. X < Z < Y.
347 *
348 * In plain words, successor of a name is the same name, but with its last component
349 * advanced to a next possible value.
350 *
351 * Examples:
352 *
353 * - successor for / is /%00
354 * - successor for /%00%01/%01%02 is /%00%01/%01%03
355 * - successor for /%00%01/%01%FF is /%00%01/%02%00
356 * - successor for /%00%01/%FF%FF is /%00%01/%00%00%00
357 *
Shuo Chen5aa8c742014-06-22 15:00:02 +0800358 * @return a new name
359 */
360 Name
361 getSuccessor() const;
362
363 /**
Jeff Thompson3c2ab012013-10-02 14:18:16 -0700364 * Check if this name has the same component count and components as the given name.
365 * @param name The Name to check.
366 * @return true if the names are equal, otherwise false.
367 */
368 bool
369 equals(const Name& name) const;
Alexander Afanasyev4b98e8c2014-03-22 19:10:19 -0700370
Jeff Thompson3c2ab012013-10-02 14:18:16 -0700371 /**
Alexander Afanasyev15f67312014-07-22 15:11:09 -0700372 * @brief Check if the N components of this name are the same as the first N components
373 * of the given name.
374 *
Jeff Thompsonec7789a2013-08-21 11:08:36 -0700375 * @param name The Name to check.
Alexander Afanasyev15f67312014-07-22 15:11:09 -0700376 * @return true if this matches the given name, otherwise false. This always returns
377 * true if this name is empty.
Jeff Thompsonec7789a2013-08-21 11:08:36 -0700378 */
Alexander Afanasyev4b98e8c2014-03-22 19:10:19 -0700379 bool
Alexander Afanasyevaf283d82014-01-03 13:23:34 -0800380 isPrefixOf(const Name& name) const;
381
Jeff Thompsonec39fbd2013-10-04 10:56:23 -0700382 //
383 // vector equivalent interface.
384 //
Alexander Afanasyevaf283d82014-01-03 13:23:34 -0800385
386 /**
387 * @brief Check if name is emtpy
388 */
389 bool
Alexander Afanasyevfdbfc6d2014-04-14 15:12:11 -0700390 empty() const
391 {
392 return m_nameBlock.elements().empty();
393 }
Alexander Afanasyev4b98e8c2014-03-22 19:10:19 -0700394
Jeff Thompsonec39fbd2013-10-04 10:56:23 -0700395 /**
396 * Get the number of components.
397 * @return The number of components.
398 */
Alexander Afanasyev4b98e8c2014-03-22 19:10:19 -0700399 size_t
Alexander Afanasyevfdbfc6d2014-04-14 15:12:11 -0700400 size() const
401 {
402 return m_nameBlock.elements_size();
403 }
Jeff Thompsonec39fbd2013-10-04 10:56:23 -0700404
405 /**
406 * Get the component at the given index.
407 * @param i The index of the component, starting from 0.
408 * @return The name component at the index.
409 */
Alexander Afanasyev4b98e8c2014-03-22 19:10:19 -0700410 const Component&
Alexander Afanasyev8f9aa8bed2013-12-30 15:58:57 -0800411 get(ssize_t i) const
412 {
413 if (i >= 0)
Alexander Afanasyev95e8c2f2014-02-06 17:29:30 -0800414 return reinterpret_cast<const Component&>(m_nameBlock.elements()[i]);
Alexander Afanasyev8f9aa8bed2013-12-30 15:58:57 -0800415 else
Alexander Afanasyev95e8c2f2014-02-06 17:29:30 -0800416 return reinterpret_cast<const Component&>(m_nameBlock.elements()[size() + i]);
417 }
418
419 const Component&
Alexander Afanasyev1dd95c52014-03-22 19:11:36 -0700420 operator[](ssize_t i) const
Alexander Afanasyev95e8c2f2014-02-06 17:29:30 -0800421 {
422 return get(i);
Alexander Afanasyev8f9aa8bed2013-12-30 15:58:57 -0800423 }
Alexander Afanasyevc2344292014-03-02 00:08:00 +0000424
425 /**
426 * @brief Get component at the specified index
427 *
428 * Unlike get() and operator[] methods, at() checks for out of bounds
429 * and will throw Name::Error when it happens
430 *
431 * @throws Name::Error if index out of bounds
432 */
Alexander Afanasyev95e8c2f2014-02-06 17:29:30 -0800433 const Component&
434 at(ssize_t i) const
435 {
Alexander Afanasyev1dd95c52014-03-22 19:11:36 -0700436 if ((i >= 0 && static_cast<size_t>(i) >= size()) ||
437 (i < 0 && static_cast<size_t>(-i) > size()))
Spyridon Mastorakis0d2ed2e2015-07-27 19:09:12 -0700438 BOOST_THROW_EXCEPTION(Error("Requested component does not exist (out of bounds)"));
Alexander Afanasyevc2344292014-03-02 00:08:00 +0000439
Alexander Afanasyev95e8c2f2014-02-06 17:29:30 -0800440 return get(i);
441 }
442
Jeff Thompsonafc45a92014-01-15 12:42:45 -0800443 /**
Alexander Afanasyev4b98e8c2014-03-22 19:10:19 -0700444 * @brief Compare this to the other Name using NDN canonical ordering.
Jeff Thompsonafc45a92014-01-15 12:42:45 -0800445 *
Joao Pereiraaa8fd162015-06-05 16:35:15 -0400446 * If the first components of each name are not equal, this returns a negative value if
447 * the first comes before the second using the NDN canonical ordering for name
448 * components, or a positive value if it comes after. If they are equal, this compares
449 * the second components of each name, etc. If both names are the same up to the size
450 * of the shorter name, this returns a negative value if the first name is shorter than
451 * the second or a positive value if it is longer. For example, if you std::sort gives:
452 * /a/b/d /a/b/cc /c /c/a /bb .
453 * This is intuitive because all names with the prefix /a are next to each other.
454 * But it may be also be counter-intuitive because /c comes before /bb according
455 * to NDN canonical ordering since it is shorter.
Junxiao Shia6452ac2015-01-23 11:21:06 -0700456 *
Joao Pereiraaa8fd162015-06-05 16:35:15 -0400457 * @param other The other Name to compare with.
458 *
459 * @retval negative this comes before other in canonical ordering
460 * @retval zero this equals other
461 * @retval positive this comes after other in canonical ordering
Alexander Afanasyev4b98e8c2014-03-22 19:10:19 -0700462 *
463 * @see http://named-data.net/doc/ndn-tlv/name.html#canonical-order
Jeff Thompsonafc45a92014-01-15 12:42:45 -0800464 */
465 int
Junxiao Shia6452ac2015-01-23 11:21:06 -0700466 compare(const Name& other) const
467 {
468 return this->compare(0, npos, other);
469 }
470
471 /** \brief compares [pos1, pos1+count1) components in this Name
472 * to [pos2, pos2+count2) components in \p other
473 *
474 * This is equivalent to this->getSubName(pos1, count1).compare(other.getSubName(pos2, count2));
475 */
476 int
477 compare(size_t pos1, size_t count1,
478 const Name& other, size_t pos2 = 0, size_t count2 = npos) const;
Jeff Thompsonec39fbd2013-10-04 10:56:23 -0700479
Jeff Thompsonec39fbd2013-10-04 10:56:23 -0700480 /**
481 * Append the component
482 * @param component The component of type T.
483 */
484 template<class T> void
Alexander Afanasyev1dd95c52014-03-22 19:11:36 -0700485 push_back(const T& component)
Jeff Thompsonec39fbd2013-10-04 10:56:23 -0700486 {
487 append(component);
488 }
Alexander Afanasyev4b98e8c2014-03-22 19:10:19 -0700489
Jeff Thompson91737f52013-10-04 11:07:24 -0700490 /**
491 * Check if this name has the same component count and components as the given name.
492 * @param name The Name to check.
493 * @return true if the names are equal, otherwise false.
494 */
495 bool
Alexander Afanasyevfdbfc6d2014-04-14 15:12:11 -0700496 operator==(const Name& name) const
497 {
498 return equals(name);
499 }
Jeff Thompson91737f52013-10-04 11:07:24 -0700500
501 /**
502 * Check if this name has the same component count and components as the given name.
503 * @param name The Name to check.
504 * @return true if the names are not equal, otherwise false.
505 */
506 bool
Alexander Afanasyevfdbfc6d2014-04-14 15:12:11 -0700507 operator!=(const Name& name) const
508 {
509 return !equals(name);
510 }
Alexander Afanasyev4b98e8c2014-03-22 19:10:19 -0700511
Jeff Thompson82568ad2013-12-17 15:17:40 -0800512 /**
Jeff Thompsonafc45a92014-01-15 12:42:45 -0800513 * Return true if this is less than or equal to the other Name in the NDN canonical ordering.
514 * @param other The other Name to compare with.
515 *
Alexander Afanasyev4b98e8c2014-03-22 19:10:19 -0700516 * @see http://named-data.net/doc/ndn-tlv/name.html#canonical-order
Jeff Thompson82568ad2013-12-17 15:17:40 -0800517 */
Jeff Thompsonafc45a92014-01-15 12:42:45 -0800518 bool
Alexander Afanasyevfdbfc6d2014-04-14 15:12:11 -0700519 operator<=(const Name& other) const
520 {
521 return compare(other) <= 0;
522 }
Jeff Thompsonafc45a92014-01-15 12:42:45 -0800523
524 /**
525 * Return true if this is less than the other Name in the NDN canonical ordering.
526 * @param other The other Name to compare with.
527 *
Alexander Afanasyev4b98e8c2014-03-22 19:10:19 -0700528 * @see http://named-data.net/doc/ndn-tlv/name.html#canonical-order
Jeff Thompsonafc45a92014-01-15 12:42:45 -0800529 */
530 bool
Alexander Afanasyevfdbfc6d2014-04-14 15:12:11 -0700531 operator<(const Name& other) const
532 {
533 return compare(other) < 0;
534 }
Jeff Thompsonafc45a92014-01-15 12:42:45 -0800535
536 /**
537 * Return true if this is less than or equal to the other Name in the NDN canonical ordering.
538 * @param other The other Name to compare with.
539 *
Alexander Afanasyev4b98e8c2014-03-22 19:10:19 -0700540 * @see http://named-data.net/doc/ndn-tlv/name.html#canonical-order
Jeff Thompsonafc45a92014-01-15 12:42:45 -0800541 */
542 bool
Alexander Afanasyevfdbfc6d2014-04-14 15:12:11 -0700543 operator>=(const Name& other) const
544 {
545 return compare(other) >= 0;
546 }
Jeff Thompsonafc45a92014-01-15 12:42:45 -0800547
548 /**
549 * Return true if this is greater than the other Name in the NDN canonical ordering.
550 * @param other The other Name to compare with.
551 *
Alexander Afanasyev4b98e8c2014-03-22 19:10:19 -0700552 * @see http://named-data.net/doc/ndn-tlv/name.html#canonical-order
Jeff Thompsonafc45a92014-01-15 12:42:45 -0800553 */
554 bool
Alexander Afanasyevfdbfc6d2014-04-14 15:12:11 -0700555 operator>(const Name& other) const
556 {
557 return compare(other) > 0;
558 }
Alexander Afanasyev4b98e8c2014-03-22 19:10:19 -0700559
Jeff Thompsonec39fbd2013-10-04 10:56:23 -0700560 //
561 // Iterator interface to name components.
562 //
Jeff Thompsonec39fbd2013-10-04 10:56:23 -0700563
564 /**
565 * Begin iterator (const).
566 */
567 const_iterator
Alexander Afanasyev95e8c2f2014-02-06 17:29:30 -0800568 begin() const
569 {
570 return reinterpret_cast<const_iterator>(&*m_nameBlock.elements().begin());
571 }
Jeff Thompsonec39fbd2013-10-04 10:56:23 -0700572
573 /**
Jeff Thompsonec39fbd2013-10-04 10:56:23 -0700574 * End iterator (const).
Alexander Afanasyev95e8c2f2014-02-06 17:29:30 -0800575 *
576 * @todo Check if this crash when there are no elements in the buffer
Jeff Thompsonec39fbd2013-10-04 10:56:23 -0700577 */
578 const_iterator
Alexander Afanasyev29e5c3d2014-02-11 00:01:10 -0800579 end() const
580 {
581 return reinterpret_cast<const_iterator>(&*m_nameBlock.elements().end());
582 }
Jeff Thompsonec39fbd2013-10-04 10:56:23 -0700583
584 /**
Jeff Thompsonec39fbd2013-10-04 10:56:23 -0700585 * Reverse begin iterator (const).
586 */
587 const_reverse_iterator
Alexander Afanasyev29e5c3d2014-02-11 00:01:10 -0800588 rbegin() const
589 {
590 return const_reverse_iterator(end());
591 }
Jeff Thompsonec39fbd2013-10-04 10:56:23 -0700592
593 /**
594 * Reverse end iterator (const).
595 */
596 const_reverse_iterator
Alexander Afanasyev29e5c3d2014-02-11 00:01:10 -0800597 rend() const
598 {
599 return const_reverse_iterator(begin());
600 }
Jeff Thompsonec39fbd2013-10-04 10:56:23 -0700601
Junxiao Shia6452ac2015-01-23 11:21:06 -0700602public:
603 /** \brief indicates "until the end" in getSubName and compare
604 */
605 static const size_t npos;
606
Jeff Thompson9c41dfe2013-06-27 12:10:25 -0700607private:
Alexander Afanasyev95e8c2f2014-02-06 17:29:30 -0800608 mutable Block m_nameBlock;
Alexander Afanasyevaf283d82014-01-03 13:23:34 -0800609};
610
Alexander Afanasyev15f67312014-07-22 15:11:09 -0700611std::ostream&
612operator<<(std::ostream& os, const Name& name);
Alexander Afanasyev4b98e8c2014-03-22 19:10:19 -0700613
Alexander Afanasyev15f67312014-07-22 15:11:09 -0700614std::istream&
615operator>>(std::istream& is, Name& name);
Alexander Afanasyev52eb20d2014-02-06 18:25:54 -0800616
Alexander Afanasyev6d48bc12014-02-18 00:10:51 -0800617inline bool
618Name::hasWire() const
Alexander Afanasyev52eb20d2014-02-06 18:25:54 -0800619{
Alexander Afanasyev6d48bc12014-02-18 00:10:51 -0800620 return m_nameBlock.hasWire();
Alexander Afanasyev52eb20d2014-02-06 18:25:54 -0800621}
622
Alexander Afanasyev95e8c2f2014-02-06 17:29:30 -0800623} // namespace ndn
Jeff Thompson9c41dfe2013-06-27 12:10:25 -0700624
Yingdi Yu90e23582014-11-06 14:21:04 -0800625namespace std {
626template<>
627struct hash<ndn::Name>
628{
629 size_t
630 operator()(const ndn::Name& name) const;
631};
632
633} // namespace std
634
Jeff Thompson9c41dfe2013-06-27 12:10:25 -0700635#endif