Alexander Afanasyev | c169a81 | 2014-05-20 20:37:29 -0400 | [diff] [blame] | 1 | /* -*- Mode:C++; c-file-style:"gnu"; indent-tabs-mode:nil; -*- */ |
Alexander Afanasyev | 52eb20d | 2014-02-06 18:25:54 -0800 | [diff] [blame] | 2 | /** |
Alexander Afanasyev | c169a81 | 2014-05-20 20:37:29 -0400 | [diff] [blame] | 3 | * Copyright (c) 2013-2014 Regents of the University of California. |
Alexander Afanasyev | dfa52c4 | 2014-04-24 21:10:11 -0700 | [diff] [blame] | 4 | * |
| 5 | * This file is part of ndn-cxx library (NDN C++ library with eXperimental eXtensions). |
Alexander Afanasyev | dfa52c4 | 2014-04-24 21:10:11 -0700 | [diff] [blame] | 6 | * |
Alexander Afanasyev | c169a81 | 2014-05-20 20:37:29 -0400 | [diff] [blame] | 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. |
Alexander Afanasyev | 52eb20d | 2014-02-06 18:25:54 -0800 | [diff] [blame] | 20 | */ |
| 21 | |
Alexander Afanasyev | 52eb20d | 2014-02-06 18:25:54 -0800 | [diff] [blame] | 22 | #include "name.hpp" |
| 23 | |
Alexander Afanasyev | b1db7c6 | 2014-04-03 14:57:25 -0700 | [diff] [blame] | 24 | #include "boost-test.hpp" |
Alexander Afanasyev | 15f6731 | 2014-07-22 15:11:09 -0700 | [diff] [blame] | 25 | #include <boost/tuple/tuple.hpp> |
| 26 | #include <boost/mpl/vector.hpp> |
Yingdi Yu | 90e2358 | 2014-11-06 14:21:04 -0800 | [diff] [blame] | 27 | #include <unordered_map> |
Alexander Afanasyev | b1db7c6 | 2014-04-03 14:57:25 -0700 | [diff] [blame] | 28 | |
Alexander Afanasyev | 52eb20d | 2014-02-06 18:25:54 -0800 | [diff] [blame] | 29 | namespace ndn { |
| 30 | |
| 31 | BOOST_AUTO_TEST_SUITE(TestName) |
| 32 | |
| 33 | static const uint8_t TestName[] = { |
Alexander Afanasyev | 4b45628 | 2014-02-13 00:34:34 -0800 | [diff] [blame] | 34 | 0x7, 0x14, // Name |
| 35 | 0x8, 0x5, // NameComponent |
Alexander Afanasyev | 52eb20d | 2014-02-06 18:25:54 -0800 | [diff] [blame] | 36 | 0x6c, 0x6f, 0x63, 0x61, 0x6c, |
Alexander Afanasyev | 4b45628 | 2014-02-13 00:34:34 -0800 | [diff] [blame] | 37 | 0x8, 0x3, // NameComponent |
Alexander Afanasyev | 52eb20d | 2014-02-06 18:25:54 -0800 | [diff] [blame] | 38 | 0x6e, 0x64, 0x6e, |
Alexander Afanasyev | 4b45628 | 2014-02-13 00:34:34 -0800 | [diff] [blame] | 39 | 0x8, 0x6, // NameComponent |
Alexander Afanasyev | 52eb20d | 2014-02-06 18:25:54 -0800 | [diff] [blame] | 40 | 0x70, 0x72, 0x65, 0x66, 0x69, 0x78 |
| 41 | }; |
| 42 | |
Alexander Afanasyev | 4b45628 | 2014-02-13 00:34:34 -0800 | [diff] [blame] | 43 | const uint8_t Name1[] = {0x7, 0x7, // Name |
| 44 | 0x8, 0x5, // NameComponent |
| 45 | 0x6c, 0x6f, 0x63, 0x61, 0x6c}; |
Alexander Afanasyev | b1db7c6 | 2014-04-03 14:57:25 -0700 | [diff] [blame] | 46 | |
Alexander Afanasyev | 4b45628 | 2014-02-13 00:34:34 -0800 | [diff] [blame] | 47 | const uint8_t Name2[] = {0x7, 0xc, // Name |
| 48 | 0x8, 0x5, // NameComponent |
| 49 | 0x6c, 0x6f, 0x63, 0x61, 0x6c, |
| 50 | 0x8, 0x3, // NameComponent |
| 51 | 0x6e, 0x64, 0x6e}; |
| 52 | |
| 53 | |
Alexander Afanasyev | c234429 | 2014-03-02 00:08:00 +0000 | [diff] [blame] | 54 | BOOST_AUTO_TEST_CASE(Basic) |
| 55 | { |
| 56 | Name name("/hello/world"); |
| 57 | |
| 58 | BOOST_CHECK_NO_THROW(name.at(0)); |
| 59 | BOOST_CHECK_NO_THROW(name.at(1)); |
| 60 | BOOST_CHECK_NO_THROW(name.at(-1)); |
| 61 | BOOST_CHECK_NO_THROW(name.at(-2)); |
| 62 | |
| 63 | BOOST_CHECK_THROW(name.at(2), Name::Error); |
| 64 | BOOST_CHECK_THROW(name.at(-3), Name::Error); |
| 65 | } |
| 66 | |
| 67 | BOOST_AUTO_TEST_CASE(Encode) |
Alexander Afanasyev | 52eb20d | 2014-02-06 18:25:54 -0800 | [diff] [blame] | 68 | { |
| 69 | Name name("/local/ndn/prefix"); |
| 70 | |
| 71 | const Block &wire = name.wireEncode(); |
| 72 | |
| 73 | // for (Buffer::const_iterator i = wire.begin(); |
| 74 | // i != wire.end(); |
| 75 | // ++i) |
| 76 | // { |
| 77 | // std::ios::fmtflags saveFlags = std::cout.flags(std::ios::hex); |
| 78 | |
| 79 | // if (i != wire.begin()) |
| 80 | // std::cout << ", "; |
| 81 | // std::cout << "0x" << static_cast<uint32_t>(*i); |
Alexander Afanasyev | b1db7c6 | 2014-04-03 14:57:25 -0700 | [diff] [blame] | 82 | |
Alexander Afanasyev | 52eb20d | 2014-02-06 18:25:54 -0800 | [diff] [blame] | 83 | // std::cout.flags(saveFlags); |
| 84 | // } |
| 85 | // std::cout << std::endl; |
Alexander Afanasyev | b1db7c6 | 2014-04-03 14:57:25 -0700 | [diff] [blame] | 86 | |
Alexander Afanasyev | 29e5c3d | 2014-02-11 00:01:10 -0800 | [diff] [blame] | 87 | BOOST_CHECK_EQUAL_COLLECTIONS(TestName, TestName+sizeof(TestName), |
| 88 | wire.begin(), wire.end()); |
Alexander Afanasyev | 52eb20d | 2014-02-06 18:25:54 -0800 | [diff] [blame] | 89 | } |
| 90 | |
| 91 | |
Alexander Afanasyev | c234429 | 2014-03-02 00:08:00 +0000 | [diff] [blame] | 92 | BOOST_AUTO_TEST_CASE(Decode) |
Alexander Afanasyev | 52eb20d | 2014-02-06 18:25:54 -0800 | [diff] [blame] | 93 | { |
| 94 | Block block(TestName, sizeof(TestName)); |
| 95 | |
| 96 | Name name(block); |
| 97 | |
| 98 | BOOST_CHECK_EQUAL(name.toUri(), "/local/ndn/prefix"); |
| 99 | } |
| 100 | |
Alexander Afanasyev | c234429 | 2014-03-02 00:08:00 +0000 | [diff] [blame] | 101 | BOOST_AUTO_TEST_CASE(AppendsAndMultiEncode) |
Alexander Afanasyev | 29e5c3d | 2014-02-11 00:01:10 -0800 | [diff] [blame] | 102 | { |
| 103 | Name name("/local"); |
Alexander Afanasyev | b1db7c6 | 2014-04-03 14:57:25 -0700 | [diff] [blame] | 104 | |
Alexander Afanasyev | 29e5c3d | 2014-02-11 00:01:10 -0800 | [diff] [blame] | 105 | BOOST_CHECK_EQUAL_COLLECTIONS(name.wireEncode().begin(), name.wireEncode().end(), |
Alexander Afanasyev | 4b45628 | 2014-02-13 00:34:34 -0800 | [diff] [blame] | 106 | Name1, Name1 + sizeof(Name1)); |
Alexander Afanasyev | 29e5c3d | 2014-02-11 00:01:10 -0800 | [diff] [blame] | 107 | |
| 108 | name.append("ndn"); |
Alexander Afanasyev | b1db7c6 | 2014-04-03 14:57:25 -0700 | [diff] [blame] | 109 | |
Alexander Afanasyev | 29e5c3d | 2014-02-11 00:01:10 -0800 | [diff] [blame] | 110 | BOOST_CHECK_EQUAL_COLLECTIONS(name.wireEncode().begin(), name.wireEncode().end(), |
Alexander Afanasyev | 4b45628 | 2014-02-13 00:34:34 -0800 | [diff] [blame] | 111 | Name2, Name2 + sizeof(Name2)); |
Alexander Afanasyev | 29e5c3d | 2014-02-11 00:01:10 -0800 | [diff] [blame] | 112 | |
| 113 | name.append("prefix"); |
| 114 | BOOST_CHECK_EQUAL_COLLECTIONS(name.wireEncode().begin(), name.wireEncode().end(), |
| 115 | TestName, TestName+sizeof(TestName)); |
| 116 | } |
| 117 | |
Steve DiBenedetto | c145d49 | 2014-03-11 16:35:45 -0600 | [diff] [blame] | 118 | BOOST_AUTO_TEST_CASE(AppendNumber) |
| 119 | { |
| 120 | Name name; |
Alexander Afanasyev | b1db7c6 | 2014-04-03 14:57:25 -0700 | [diff] [blame] | 121 | for (uint32_t i = 0; i < 10; i++) |
Steve DiBenedetto | c145d49 | 2014-03-11 16:35:45 -0600 | [diff] [blame] | 122 | { |
| 123 | name.appendNumber(i); |
| 124 | } |
| 125 | |
| 126 | BOOST_CHECK_EQUAL(name.size(), 10); |
| 127 | |
Alexander Afanasyev | b1db7c6 | 2014-04-03 14:57:25 -0700 | [diff] [blame] | 128 | for (uint32_t i = 0; i < 10; i++) |
Steve DiBenedetto | c145d49 | 2014-03-11 16:35:45 -0600 | [diff] [blame] | 129 | { |
| 130 | BOOST_CHECK_EQUAL(name[i].toNumber(), i); |
| 131 | } |
| 132 | } |
| 133 | |
Alexander Afanasyev | 15f6731 | 2014-07-22 15:11:09 -0700 | [diff] [blame] | 134 | class Numeric |
| 135 | { |
| 136 | public: |
| 137 | typedef std::list<boost::tuple<function<name::Component(uint64_t)>, |
| 138 | function<uint64_t(const name::Component&)>, |
| 139 | function<Name&(Name&, uint64_t)>, |
| 140 | Name/*expected*/, |
Alexander Afanasyev | 0f232c5 | 2014-10-23 13:07:31 -0700 | [diff] [blame] | 141 | uint64_t/*value*/, |
| 142 | function<bool(const name::Component&)> > > Dataset; |
Alexander Afanasyev | 15f6731 | 2014-07-22 15:11:09 -0700 | [diff] [blame] | 143 | |
| 144 | Numeric() |
| 145 | { |
| 146 | dataset.push_back(boost::make_tuple(bind(&name::Component::fromNumberWithMarker, |
| 147 | 0xAA, _1), |
| 148 | bind(&name::Component::toNumberWithMarker, _1, 0xAA), |
| 149 | bind(&Name::appendNumberWithMarker, _1, 0xAA, _2), |
| 150 | Name("/%AA%03%E8"), |
Alexander Afanasyev | 0f232c5 | 2014-10-23 13:07:31 -0700 | [diff] [blame] | 151 | 1000, |
| 152 | bind(&name::Component::isNumberWithMarker, _1, 0xAA))); |
Alexander Afanasyev | 15f6731 | 2014-07-22 15:11:09 -0700 | [diff] [blame] | 153 | dataset.push_back(boost::make_tuple(&name::Component::fromSegment, |
| 154 | bind(&name::Component::toSegment, _1), |
| 155 | bind(&Name::appendSegment, _1, _2), |
| 156 | Name("/%00%27%10"), |
Alexander Afanasyev | 0f232c5 | 2014-10-23 13:07:31 -0700 | [diff] [blame] | 157 | 10000, |
| 158 | bind(&name::Component::isSegment, _1))); |
Alexander Afanasyev | 15f6731 | 2014-07-22 15:11:09 -0700 | [diff] [blame] | 159 | dataset.push_back(boost::make_tuple(&name::Component::fromSegmentOffset, |
| 160 | bind(&name::Component::toSegmentOffset, _1), |
| 161 | bind(&Name::appendSegmentOffset, _1, _2), |
| 162 | Name("/%FB%00%01%86%A0"), |
Alexander Afanasyev | 0f232c5 | 2014-10-23 13:07:31 -0700 | [diff] [blame] | 163 | 100000, |
| 164 | bind(&name::Component::isSegmentOffset, _1))); |
Alexander Afanasyev | 15f6731 | 2014-07-22 15:11:09 -0700 | [diff] [blame] | 165 | dataset.push_back(boost::make_tuple(&name::Component::fromVersion, |
| 166 | bind(&name::Component::toVersion, _1), |
Davide Pesavento | dfe9c6b | 2014-08-25 21:17:10 +0200 | [diff] [blame] | 167 | bind(static_cast<Name&(Name::*)(uint64_t)>( |
| 168 | &Name::appendVersion), _1, _2), |
Alexander Afanasyev | 15f6731 | 2014-07-22 15:11:09 -0700 | [diff] [blame] | 169 | Name("/%FD%00%0FB%40"), |
Alexander Afanasyev | 0f232c5 | 2014-10-23 13:07:31 -0700 | [diff] [blame] | 170 | 1000000, |
| 171 | bind(&name::Component::isVersion, _1))); |
Alexander Afanasyev | 15f6731 | 2014-07-22 15:11:09 -0700 | [diff] [blame] | 172 | dataset.push_back(boost::make_tuple(&name::Component::fromSequenceNumber, |
| 173 | bind(&name::Component::toSequenceNumber, _1), |
| 174 | bind(&Name::appendSequenceNumber, _1, _2), |
| 175 | Name("/%FE%00%98%96%80"), |
Alexander Afanasyev | 0f232c5 | 2014-10-23 13:07:31 -0700 | [diff] [blame] | 176 | 10000000, |
| 177 | bind(&name::Component::isSequenceNumber, _1))); |
Alexander Afanasyev | 15f6731 | 2014-07-22 15:11:09 -0700 | [diff] [blame] | 178 | } |
| 179 | |
| 180 | Dataset dataset; |
| 181 | }; |
| 182 | |
| 183 | class Timestamp |
| 184 | { |
| 185 | public: |
| 186 | typedef std::list<boost::tuple<function<name::Component(const time::system_clock::TimePoint&)>, |
| 187 | function<time::system_clock::TimePoint(const name::Component&)>, |
| 188 | function<Name&(Name&, const time::system_clock::TimePoint&)>, |
| 189 | Name/*expected*/, |
Alexander Afanasyev | 0f232c5 | 2014-10-23 13:07:31 -0700 | [diff] [blame] | 190 | time::system_clock::TimePoint/*value*/, |
| 191 | function<bool(const name::Component&)> > > Dataset; |
Alexander Afanasyev | 15f6731 | 2014-07-22 15:11:09 -0700 | [diff] [blame] | 192 | Timestamp() |
| 193 | { |
| 194 | dataset.push_back(boost::make_tuple(&name::Component::fromTimestamp, |
Alexander Afanasyev | 6f9ec93 | 2014-10-31 10:34:00 -0700 | [diff] [blame] | 195 | ndn::bind(&name::Component::toTimestamp, _1), |
| 196 | ndn::bind(&Name::appendTimestamp, _1, _2), |
Alexander Afanasyev | 15f6731 | 2014-07-22 15:11:09 -0700 | [diff] [blame] | 197 | Name("/%FC%00%04%7BE%E3%1B%00%00"), |
Davide Pesavento | dfe9c6b | 2014-08-25 21:17:10 +0200 | [diff] [blame] | 198 | time::getUnixEpoch() + time::days(14600/*40 years*/), |
Alexander Afanasyev | 0f232c5 | 2014-10-23 13:07:31 -0700 | [diff] [blame] | 199 | bind(&name::Component::isTimestamp, _1))); |
Alexander Afanasyev | 15f6731 | 2014-07-22 15:11:09 -0700 | [diff] [blame] | 200 | } |
| 201 | |
| 202 | Dataset dataset; |
| 203 | }; |
| 204 | |
| 205 | typedef boost::mpl::vector<Numeric, Timestamp> ConventionsDatasets; |
| 206 | |
| 207 | BOOST_FIXTURE_TEST_CASE_TEMPLATE(NamingConventions, T, ConventionsDatasets, T) |
| 208 | { |
| 209 | // // These octets are obtained by the snippet below. |
| 210 | // // This check is intended to detect unexpected encoding change in the future. |
| 211 | // for (typename T::Dataset::const_iterator it = this->dataset.begin(); |
| 212 | // it != this->dataset.end(); ++it) { |
| 213 | // Name name; |
| 214 | // name.append(it->template get<0>()(it->template get<4>())); |
| 215 | // std::cout << name << std::endl; |
| 216 | // } |
| 217 | |
Alexander Afanasyev | 0f232c5 | 2014-10-23 13:07:31 -0700 | [diff] [blame] | 218 | name::Component invalidComponent1; |
| 219 | name::Component invalidComponent2("1234567890"); |
| 220 | |
Alexander Afanasyev | 15f6731 | 2014-07-22 15:11:09 -0700 | [diff] [blame] | 221 | for (typename T::Dataset::const_iterator it = this->dataset.begin(); |
| 222 | it != this->dataset.end(); ++it) { |
| 223 | const Name& expected = it->template get<3>(); |
Alexander Afanasyev | 0f232c5 | 2014-10-23 13:07:31 -0700 | [diff] [blame] | 224 | BOOST_TEST_MESSAGE("Check " << expected[0].toUri()); |
Alexander Afanasyev | 15f6731 | 2014-07-22 15:11:09 -0700 | [diff] [blame] | 225 | |
| 226 | name::Component actualComponent = it->template get<0>()(it->template get<4>()); |
| 227 | BOOST_CHECK_EQUAL(actualComponent, expected[0]); |
| 228 | |
| 229 | Name actualName; |
| 230 | it->template get<2>()(actualName, it->template get<4>()); |
| 231 | BOOST_CHECK_EQUAL(actualName, expected); |
| 232 | |
Alexander Afanasyev | 0f232c5 | 2014-10-23 13:07:31 -0700 | [diff] [blame] | 233 | BOOST_CHECK_EQUAL(it->template get<5>()(expected[0]), true); |
Alexander Afanasyev | 15f6731 | 2014-07-22 15:11:09 -0700 | [diff] [blame] | 234 | BOOST_REQUIRE_NO_THROW(it->template get<1>()(expected[0])); |
| 235 | BOOST_CHECK_EQUAL(it->template get<1>()(expected[0]), it->template get<4>()); |
Alexander Afanasyev | 0f232c5 | 2014-10-23 13:07:31 -0700 | [diff] [blame] | 236 | |
| 237 | BOOST_CHECK_EQUAL(it->template get<5>()(invalidComponent1), false); |
| 238 | BOOST_CHECK_EQUAL(it->template get<5>()(invalidComponent2), false); |
| 239 | |
| 240 | BOOST_REQUIRE_THROW(it->template get<1>()(invalidComponent1), name::Component::Error); |
| 241 | BOOST_REQUIRE_THROW(it->template get<1>()(invalidComponent2), name::Component::Error); |
Alexander Afanasyev | 15f6731 | 2014-07-22 15:11:09 -0700 | [diff] [blame] | 242 | } |
| 243 | } |
| 244 | |
Shuo Chen | 5aa8c74 | 2014-06-22 15:00:02 +0800 | [diff] [blame] | 245 | BOOST_AUTO_TEST_CASE(GetSuccessor) |
| 246 | { |
| 247 | BOOST_CHECK_EQUAL(Name("ndn:/%00%01/%01%02").getSuccessor(), Name("ndn:/%00%01/%01%03")); |
| 248 | BOOST_CHECK_EQUAL(Name("ndn:/%00%01/%01%FF").getSuccessor(), Name("ndn:/%00%01/%02%00")); |
| 249 | BOOST_CHECK_EQUAL(Name("ndn:/%00%01/%FF%FF").getSuccessor(), Name("ndn:/%00%01/%00%00%00")); |
| 250 | BOOST_CHECK_EQUAL(Name().getSuccessor(), Name("ndn:/%00")); |
| 251 | } |
| 252 | |
Junxiao Shi | 937e461 | 2014-10-22 15:39:07 -0700 | [diff] [blame] | 253 | BOOST_AUTO_TEST_CASE(Markers) |
| 254 | { |
| 255 | Name name; |
| 256 | uint64_t number; |
| 257 | |
| 258 | BOOST_REQUIRE_NO_THROW(number = name.appendSegment(30923).at(-1).toSegment()); |
| 259 | BOOST_CHECK_EQUAL(number, 30923); |
| 260 | |
| 261 | BOOST_REQUIRE_NO_THROW(number = name.appendSegmentOffset(589).at(-1).toSegmentOffset()); |
| 262 | BOOST_CHECK_EQUAL(number, 589); |
| 263 | |
| 264 | BOOST_REQUIRE_NO_THROW(number = name.appendVersion().at(-1).toVersion()); |
| 265 | |
| 266 | BOOST_REQUIRE_NO_THROW(number = name.appendVersion(25912).at(-1).toVersion()); |
| 267 | BOOST_CHECK_EQUAL(number, 25912); |
| 268 | |
| 269 | const time::system_clock::TimePoint tp = time::system_clock::now(); |
| 270 | time::system_clock::TimePoint tp2; |
| 271 | BOOST_REQUIRE_NO_THROW(tp2 = name.appendTimestamp(tp).at(-1).toTimestamp()); |
| 272 | BOOST_CHECK_LE(std::abs(time::duration_cast<time::microseconds>(tp2 - tp).count()), 1); |
| 273 | |
| 274 | BOOST_REQUIRE_NO_THROW(number = name.appendSequenceNumber(11676).at(-1).toSequenceNumber()); |
| 275 | BOOST_CHECK_EQUAL(number, 11676); |
| 276 | } |
| 277 | |
Yingdi Yu | 90e2358 | 2014-11-06 14:21:04 -0800 | [diff] [blame] | 278 | BOOST_AUTO_TEST_CASE(UnorderedMap) |
| 279 | { |
| 280 | std::unordered_map<Name, int> map; |
| 281 | Name name1("/1"); |
| 282 | Name name2("/2"); |
| 283 | Name name3("/3"); |
| 284 | map[name1] = 1; |
| 285 | map[name2] = 2; |
| 286 | map[name3] = 3; |
| 287 | |
| 288 | BOOST_CHECK_EQUAL(map[name1], 1); |
| 289 | BOOST_CHECK_EQUAL(map[name2], 2); |
| 290 | BOOST_CHECK_EQUAL(map[name3], 3); |
| 291 | } |
| 292 | |
Alexander Afanasyev | 52eb20d | 2014-02-06 18:25:54 -0800 | [diff] [blame] | 293 | BOOST_AUTO_TEST_SUITE_END() |
| 294 | |
| 295 | } // namespace ndn |