blob: 07ac0d963545a1782f0dc16b4c88e32fd522e2ef [file] [log] [blame]
Spyridon Mastorakis3b54e852015-04-07 08:03:25 -07001/* -*- Mode:C++; c-file-style:"gnu"; indent-tabs-mode:nil; -*- */
2/**
3 * Copyright (c) 2013-2015 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
22#include "link.hpp"
23#include "interest.hpp"
24#include "encoding/block-helpers.hpp"
25#include "util/crypto.hpp"
26#include "security/key-chain.hpp"
27#include "util/concepts.hpp"
28
29#include <algorithm>
30
31#include <boost/range/adaptors.hpp>
32
33namespace ndn {
34
35BOOST_CONCEPT_ASSERT((boost::EqualityComparable<Link>));
36BOOST_CONCEPT_ASSERT((WireEncodable<Link>));
37BOOST_CONCEPT_ASSERT((WireDecodable<Link>));
38static_assert(std::is_base_of<Data::Error, Link::Error>::value,
39 "Link::Error should inherit from Data::Error");
40
41Link::Link(const Block& block)
42{
43 wireDecode(block);
44}
45
46Link::Link(const Name& name)
47 : Data(name)
48{
49}
50
51Link::Link(const Name& name, std::initializer_list<std::pair<uint32_t, Name>> links)
52 : Data(name)
53{
54 m_delegations.insert(links);
55 encodeContent();
56}
57
58void
59Link::addDelegation(uint32_t preference, const Name& name)
60{
61 this->removeDelegationNoEncode(name);
62 m_delegations.insert({preference, name});
63 encodeContent();
64}
65
66bool
67Link::removeDelegation(const Name& name)
68{
69 bool hasRemovedDelegation = this->removeDelegationNoEncode(name);
70 if (hasRemovedDelegation) {
71 encodeContent();
72 }
73 return hasRemovedDelegation;
74}
75
76const Link::DelegationSet&
77Link::getDelegations() const
78{
79 return m_delegations;
80}
81
82template<encoding::Tag TAG>
83size_t
84Link::encodeContent(EncodingImpl<TAG>& encoder) const
85{
86 // LinkContent ::= CONTENT-TYPE TLV-LENGTH
87 // Delegation+
88
89 // Delegation ::= LINK-DELEGATION-TYPE TLV-LENGTH
90 // Preference
91 // Name
92
93 // Preference ::= LINK-PREFERENCE-TYPE TLV-LENGTH
94 // nonNegativeInteger
95
96 size_t totalLength = 0;
97 for (const auto& delegation : m_delegations | boost::adaptors::reversed) {
98 size_t delegationLength = 0;
99 delegationLength += std::get<1>(delegation).wireEncode(encoder);
100 delegationLength += prependNonNegativeIntegerBlock(encoder, tlv::LinkPreference,
101 std::get<0>(delegation));
102 delegationLength += encoder.prependVarNumber(delegationLength);
103 delegationLength += encoder.prependVarNumber(tlv::LinkDelegation);
104 totalLength += delegationLength;
105 }
106 totalLength += encoder.prependVarNumber(totalLength);
107 totalLength += encoder.prependVarNumber(tlv::Content);
108 return totalLength;
109}
110
111template size_t
112Link::encodeContent<encoding::EncoderTag>(EncodingImpl<encoding::EncoderTag>&) const;
113
114template size_t
115Link::encodeContent<encoding::EstimatorTag>(EncodingImpl<encoding::EstimatorTag>&) const;
116
117void
118Link::encodeContent()
119{
120 onChanged();
121
122 EncodingEstimator estimator;
123 size_t estimatedSize = encodeContent(estimator);
124
125 EncodingBuffer buffer(estimatedSize, 0);
126 encodeContent(buffer);
127
128 setContentType(tlv::ContentType_Link);
129 setContent(buffer.block());
130}
131
132void
133Link::decodeContent()
134{
135 // LinkContent ::= CONTENT-TYPE TLV-LENGTH
136 // Delegation+
137
138 // Delegation ::= LINK-DELEGATION-TYPE TLV-LENGTH
139 // Preference
140 // Name
141
142 // Preference ::= LINK-PREFERENCE-TYPE TLV-LENGTH
143 // nonNegativeInteger
144
145 if (getContentType() != tlv::ContentType_Link)
146 {
147 throw Error("Expected Content Type Link");
148 }
149
150 const Block& content = getContent();
151 content.parse();
152
153 for (auto& delegation : content.elements()) {
154 delegation.parse();
155 Block::element_const_iterator val = delegation.elements_begin();
156 if (val == delegation.elements_end()) {
157 throw Error("Unexpected Link Encoding");
158 }
159 uint32_t preference;
160 try {
161 preference = static_cast<uint32_t>(readNonNegativeInteger(*val));
162 }
163 catch (tlv::Error&) {
164 throw Error("Missing preference field in Link Encoding");
165 }
166 ++val;
167 if (val == delegation.elements_end()) {
168 throw Error("Missing name field in Link Encoding");
169 }
170 Name name(*val);
171 m_delegations.insert({preference, name});
172 }
173}
174
175void
176Link::wireDecode(const Block& wire)
177{
178 Data::wireDecode(wire);
179 decodeContent();
180}
181
182bool
183Link::removeDelegationNoEncode(const Name& name)
184{
185 bool hasRemoved = false;
186 auto i = m_delegations.begin();
187 while (i != m_delegations.end()) {
188 if (i->second == name) {
189 hasRemoved = true;
190 i = m_delegations.erase(i);
191 }
192 else {
193 ++i;
194 }
195 }
196 return hasRemoved;
197}
198
199} // namespace ndn