Kea 2.7.6
shared_network_parser.cc
Go to the documentation of this file.
1// Copyright (C) 2017-2024 Internet Systems Consortium, Inc. ("ISC")
2//
3// This Source Code Form is subject to the terms of the Mozilla Public
4// License, v. 2.0. If a copy of the MPL was not distributed with this
5// file, You can obtain one at http://mozilla.org/MPL/2.0/.
6
7#include <config.h>
8
10#include <cc/data.h>
11#include <dhcpsrv/cfgmgr.h>
12#include <dhcpsrv/cfg_option.h>
18#include <boost/make_shared.hpp>
19#include <boost/pointer_cast.hpp>
20#include <string>
21
22using namespace isc::asiolink;
23using namespace isc::data;
24using namespace isc::util;
25namespace ph = std::placeholders;
26
27namespace isc {
28namespace dhcp {
29
31 : check_iface_(check_iface) {
32}
33
36 bool encapsulate_options) {
37 SharedNetwork4Ptr shared_network;
38 try {
39
40 // Check parameters.
42 shared_network_data);
43
44 // Make sure that the network name has been specified. The name is required
45 // to create a SharedNetwork4 object.
46 std::string name = getString(shared_network_data, "name");
47 shared_network.reset(new SharedNetwork4(name));
48
49 ElementPtr mutable_params;
50 mutable_params = boost::const_pointer_cast<Element>(shared_network_data);
51
52 // Parse parameters common to all Network derivations.
53 NetworkPtr network = boost::dynamic_pointer_cast<Network>(shared_network);
54 parseCommon(mutable_params, network);
55
56 // interface is an optional parameter
57 if (shared_network_data->contains("interface")) {
58 std::string iface = getString(shared_network_data, "interface");
59 if (!iface.empty()) {
60 if (check_iface_ && !IfaceMgr::instance().getIface(iface)) {
62 shared_network_data->get("interface");
64 "Specified network interface name " << iface
65 << " for shared network " << name
66 << " is not present in the system ("
67 << error->getPosition() << ")");
68 }
69 shared_network->setIface(iface);
70 }
71 }
72
73 if (shared_network_data->contains("option-data")) {
74 auto json = shared_network_data->get("option-data");
75 // Create parser instance for option-data.
76 CfgOptionPtr cfg_option = shared_network->getCfgOption();
77 auto parser = createOptionDataListParser();
78 parser->parse(cfg_option, json, encapsulate_options);
79 }
80
81 if (shared_network_data->contains("subnet4")) {
82 auto json = shared_network_data->get("subnet4");
83
84 // Create parser instance of subnet4.
85 auto parser = createSubnetsListParser();
86 Subnet4Collection subnets;
87 parser->parse(subnets, json);
88
89 // Add all returned subnets into shared network.
90 for (auto const& subnet : subnets) {
91 shared_network->add(subnet);
92 }
93 }
94
95 if (shared_network_data->contains("match-client-id")) {
96 shared_network->setMatchClientId(getBoolean(shared_network_data,
97 "match-client-id"));
98 }
99
100 if (shared_network_data->contains("authoritative")) {
101 shared_network->setAuthoritative(getBoolean(shared_network_data,
102 "authoritative"));
103 }
104
105 // Set next-server
106 if (shared_network_data->contains("next-server")) {
107 std::string next_server;
108 try {
109 next_server = getString(shared_network_data, "next-server");
110 if (!next_server.empty()) {
111 shared_network->setSiaddr(IOAddress(next_server));
112 }
113 } catch (...) {
114 ConstElementPtr next = shared_network_data->get("next-server");
115 std::string pos;
116 if (next) {
117 pos = next->getPosition().str();
118 } else {
119 pos = shared_network_data->getPosition().str();
120 }
121 isc_throw(DhcpConfigError, "invalid parameter next-server : "
122 << next_server << "(" << pos << ")");
123 }
124 }
125
126 // Set server-hostname.
127 if (shared_network_data->contains("server-hostname")) {
128 std::string sname = getString(shared_network_data, "server-hostname");
129 if (!sname.empty()) {
130 if (sname.length() >= Pkt4::MAX_SNAME_LEN) {
131 ConstElementPtr error = shared_network_data->get("server-hostname");
132 isc_throw(DhcpConfigError, "server-hostname must be at most "
133 << Pkt4::MAX_SNAME_LEN - 1 << " bytes long, it is "
134 << sname.length() << " ("
135 << error->getPosition() << ")");
136 }
137 shared_network->setSname(sname);
138 }
139 }
140
141 // Set boot-file-name.
142 if (shared_network_data->contains("boot-file-name")) {
143 std::string filename = getString(shared_network_data, "boot-file-name");
144 if (!filename.empty()) {
145 if (filename.length() > Pkt4::MAX_FILE_LEN) {
146 ConstElementPtr error = shared_network_data->get("boot-file-name");
147 isc_throw(DhcpConfigError, "boot-file-name must be at most "
148 << Pkt4::MAX_FILE_LEN - 1 << " bytes long, it is "
149 << filename.length() << " ("
150 << error->getPosition() << ")");
151 }
152 shared_network->setFilename(filename);
153 }
154 }
155
156 ConstElementPtr user_context = shared_network_data->get("user-context");
157 if (user_context) {
158 shared_network->setContext(user_context);
159 }
160
161 // Setup additional class list.
162 getClientClassesElem(shared_network_data,
163 std::bind(&Network::allowClientClass,
164 shared_network, ph::_1));
165
166 // Setup additional class list.
167 getAdditionalClassesElem(shared_network_data,
169 shared_network, ph::_1));
170
171 if (shared_network_data->contains("relay")) {
172 auto relay_parms = shared_network_data->get("relay");
173 if (relay_parms) {
176 parser.parse(relay_info, relay_parms);
177 shared_network->setRelayInfo(*relay_info);
178 }
179 }
180
181 parseTeePercents(shared_network_data, network);
182
183 // Parse DDNS parameters
184 parseDdnsParams(shared_network_data, network);
185
186 // Parse lease cache parameters
187 parseCacheParams(shared_network_data, network);
188
189 // Parse allocator params.
190 parseAllocatorParams(shared_network_data, network);
191
192 // Parse offer-lifetime parameter.
193 Network4Ptr network4 = boost::dynamic_pointer_cast<Network4>(shared_network);
194 parseOfferLft(shared_network_data, network4);
195
196 } catch (const std::exception& ex) {
197 isc_throw(DhcpConfigError, ex.what() << " ("
198 << shared_network_data->getPosition() << ")");
199 }
200
201 // In order to take advantage of the dynamic inheritance of global
202 // parameters to a shared network we need to set a callback function
203 // for each shared network to allow for fetching global parameters.
204 shared_network->setFetchGlobalsFn([]() -> ConstCfgGlobalsPtr {
205 return (CfgMgr::instance().getCurrentCfg()->getConfiguredGlobals());
206 });
207
208 return (shared_network);
209}
210
211boost::shared_ptr<OptionDataListParser>
213 auto parser = boost::make_shared<OptionDataListParser>(AF_INET);
214 return (parser);
215}
216
217boost::shared_ptr<Subnets4ListConfigParser>
219 auto parser = boost::make_shared<Subnets4ListConfigParser>(check_iface_);
220 return (parser);
221}
222
224 : check_iface_(check_iface) {
225}
226
229 bool encapsulate_options) {
230 SharedNetwork6Ptr shared_network;
231 std::string name;
232 try {
233 // Check parameters.
235 shared_network_data);
236
237 // Make sure that the network name has been specified. The name is required
238 // to create a SharedNetwork6 object.
239 name = getString(shared_network_data, "name");
240 shared_network.reset(new SharedNetwork6(name));
241
242 ElementPtr mutable_params;
243 mutable_params = boost::const_pointer_cast<Element>(shared_network_data);
244
245 // Parse parameters common to all Network derivations.
246 NetworkPtr network = boost::dynamic_pointer_cast<Network>(shared_network);
247 parseCommon(mutable_params, network);
248
249 // preferred-lifetime
250 shared_network->setPreferred(parseIntTriplet(shared_network_data,
251 "preferred-lifetime"));
252
253 // Get interface-id option content. For now we support string
254 // representation only
255 Optional<std::string> ifaceid;
256 if (shared_network_data->contains("interface-id")) {
257 ifaceid = getString(shared_network_data, "interface-id");
258 }
259
260 // Interface is an optional parameter
262 if (shared_network_data->contains("interface")) {
263 iface = getString(shared_network_data, "interface");
264 }
265
266 // Specifying both interface for locally reachable subnets and
267 // interface id for relays is mutually exclusive. Need to test for
268 // this condition.
269 if (!ifaceid.unspecified() && !iface.unspecified() && !ifaceid.empty() &&
270 !iface.empty()) {
272 "parser error: interface (defined for locally reachable "
273 "subnets) and interface-id (defined for subnets reachable"
274 " via relays) cannot be defined at the same time for "
275 "shared network " << name << "("
276 << shared_network_data->getPosition() << ")");
277 }
278
279 // Configure interface-id for remote interfaces, if defined
280 if (!ifaceid.unspecified() && !ifaceid.empty()) {
281 std::string ifaceid_value = ifaceid.get();
282 OptionBuffer tmp(ifaceid_value.begin(), ifaceid_value.end());
284 shared_network->setInterfaceId(opt);
285 }
286
287 // Set interface name. If it is defined, then subnets are available
288 // directly over specified network interface.
289 if (!iface.unspecified() && !iface.empty()) {
290 if (check_iface_ && !IfaceMgr::instance().getIface(iface)) {
291 ConstElementPtr error = shared_network_data->get("interface");
293 "Specified network interface name " << iface
294 << " for shared network " << name
295 << " is not present in the system ("
296 << error->getPosition() << ")");
297 }
298 shared_network->setIface(iface);
299 }
300
301 if (shared_network_data->contains("rapid-commit")) {
302 shared_network->setRapidCommit(getBoolean(shared_network_data,
303 "rapid-commit"));
304 }
305
306 if (shared_network_data->contains("option-data")) {
307 auto json = shared_network_data->get("option-data");
308 // Create parser instance for option-data.
309 CfgOptionPtr cfg_option = shared_network->getCfgOption();
310 auto parser = createOptionDataListParser();
311 parser->parse(cfg_option, json, encapsulate_options);
312 }
313
314 ConstElementPtr user_context = shared_network_data->get("user-context");
315 if (user_context) {
316 shared_network->setContext(user_context);
317 }
318
319 // Setup additional class list.
320 getClientClassesElem(shared_network_data,
321 std::bind(&Network::allowClientClass,
322 shared_network, ph::_1));
323
324 // Setup additional class list.
325 getAdditionalClassesElem(shared_network_data,
327 shared_network, ph::_1));
328
329 if (shared_network_data->contains("subnet6")) {
330 auto json = shared_network_data->get("subnet6");
331
332 // Create parser instance of subnet6.
333 auto parser = createSubnetsListParser();
334 Subnet6Collection subnets;
335 parser->parse(subnets, json);
336
337 // Add all returned subnets into shared network.
338 for (auto const& subnet : subnets) {
339 shared_network->add(subnet);
340 }
341 }
342
343 if (shared_network_data->contains("relay")) {
344 auto relay_parms = shared_network_data->get("relay");
345 if (relay_parms) {
348 parser.parse(relay_info, relay_parms);
349 shared_network->setRelayInfo(*relay_info);
350 }
351 }
352
353 parseTeePercents(shared_network_data, network);
354
355 // Parse DDNS parameters
356 parseDdnsParams(shared_network_data, network);
357
358 // Parse lease cache parameters
359 parseCacheParams(shared_network_data, network);
360
361 // Parse allocator params.
362 parseAllocatorParams(shared_network_data, network);
363 if (network->getAllocatorType() == "flq") {
364 isc_throw(BadValue, "Free Lease Queue allocator is not supported for IPv6 address pools");
365 }
366
367 // Parse prefix delegation allocator params.
368 auto network6 = boost::dynamic_pointer_cast<Network6>(shared_network);
369 parsePdAllocatorParams(shared_network_data, network6);
370
371 } catch (const std::exception& ex) {
372 isc_throw(DhcpConfigError, ex.what() << " ("
373 << shared_network_data->getPosition() << ")");
374 }
375
376 // In order to take advantage of the dynamic inheritance of global
377 // parameters to a shared network we need to set a callback function
378 // for each shared network which can be used to fetch global parameters.
379 shared_network->setFetchGlobalsFn([]() -> ConstCfgGlobalsPtr {
380 return (CfgMgr::instance().getCurrentCfg()->getConfiguredGlobals());
381 });
382
383 return (shared_network);
384}
385
386boost::shared_ptr<OptionDataListParser>
388 auto parser = boost::make_shared<OptionDataListParser>(AF_INET6);
389 return (parser);
390}
391
392boost::shared_ptr<Subnets6ListConfigParser>
394 auto parser = boost::make_shared<Subnets6ListConfigParser>(check_iface_);
395 return (parser);
396}
397
398} // end of namespace isc::dhcp
399} // end of namespace isc
A generic exception that is thrown if a parameter given to a method is considered invalid in that con...
static void checkKeywords(const SimpleKeywords &keywords, isc::data::ConstElementPtr scope)
Checks acceptable keywords with their expected type.
static std::string getString(isc::data::ConstElementPtr scope, const std::string &name)
Returns a string parameter from a scope.
const isc::util::Triplet< uint32_t > parseIntTriplet(const data::ConstElementPtr &scope, const std::string &name)
Parses an integer triplet.
static bool getBoolean(isc::data::ConstElementPtr scope, const std::string &name)
Returns a boolean parameter from a scope.
void parseCacheParams(const data::ConstElementPtr &network_data, NetworkPtr &network)
Parses parameters related to lease cache settings.
void parseAllocatorParams(const data::ConstElementPtr &network_data, NetworkPtr &network)
Parses parameters pertaining to allocator selection.
static void getClientClassesElem(data::ConstElementPtr params, ClassAdderFunc adder_func)
Fetches the element for either 'client-classes' or deprecated 'client-class'.
void parseDdnsParams(const data::ConstElementPtr &network_data, NetworkPtr &network)
Parses parameters pertaining to DDNS behavior.
void parseCommon(const data::ConstElementPtr &network_data, NetworkPtr &network)
Parses common parameters.
static void getAdditionalClassesElem(data::ConstElementPtr params, ClassAdderFunc adder_func)
Fetches the element for either 'evaluate-additional-classes' or deprecated 'require-client-classes'.
void parseTeePercents(const data::ConstElementPtr &network_data, NetworkPtr &network)
Parses parameters related to "percent" timers settings.
void parsePdAllocatorParams(const data::ConstElementPtr &network_data, Network6Ptr &network)
Parses parameters pertaining to prefix delegation allocator selection.
void parseOfferLft(const data::ConstElementPtr &network_data, Network4Ptr &network)
Parses offer-lifetime parameter (v4 only)
static CfgMgr & instance()
returns a single instance of Configuration Manager
Definition cfgmgr.cc:28
To be removed. Please use ConfigError instead.
static IfaceMgr & instance()
IfaceMgr is a singleton class.
Definition iface_mgr.cc:54
Holds optional information about relay.
Definition network.h:162
boost::shared_ptr< Network::RelayInfo > RelayInfoPtr
Pointer to the RelayInfo structure.
Definition network.h:210
void allowClientClass(const isc::dhcp::ClientClass &class_name)
Adds class clas_name to the allowed client classes list.
Definition network.cc:79
void addAdditionalClass(const isc::dhcp::ClientClass &class_name)
Adds class class_name to the additional classes list.
Definition network.cc:86
static const size_t MAX_SNAME_LEN
length of the SNAME field in DHCPv4 message
Definition pkt4.h:44
static const size_t MAX_FILE_LEN
length of the FILE field in DHCPv4 message
Definition pkt4.h:47
parser for additional relay information
void parse(const isc::dhcp::Network::RelayInfoPtr &relay_info, isc::data::ConstElementPtr relay_elem)
parses the actual relay parameters
virtual boost::shared_ptr< Subnets4ListConfigParser > createSubnetsListParser() const
Returns an instance of the Subnets4ListConfigParser to be used for parsing the subnets within the sha...
SharedNetwork4Parser(bool check_iface=true)
Constructor.
virtual boost::shared_ptr< OptionDataListParser > createOptionDataListParser() const
Returns an instance of the OptionDataListParser to be used in parsing the option-data structure.
bool check_iface_
Check if the specified interface exists in the system.
SharedNetwork4Ptr parse(const data::ConstElementPtr &shared_network_data, bool encapsulate_options=true)
Parses shared configuration information for IPv4 shared network.
Shared network holding IPv4 subnets.
SharedNetwork6Parser(bool check_iface=true)
Constructor.
virtual boost::shared_ptr< Subnets6ListConfigParser > createSubnetsListParser() const
Returns an instance of the Subnets6ListConfigParser to be used for parsing the subnets within the sha...
bool check_iface_
Check if the specified interface exists in the system.
virtual boost::shared_ptr< OptionDataListParser > createOptionDataListParser() const
Returns an instance of the OptionDataListParser to be used in parsing the option-data structure.
SharedNetwork6Ptr parse(const data::ConstElementPtr &shared_network_data, bool encapsulate_options=true)
Parses shared configuration information for IPv6 shared network.
Shared network holding IPv6 subnets.
static const isc::data::SimpleKeywords SHARED_NETWORK4_PARAMETERS
This table defines all shared network parameters for DHCPv4.
static const isc::data::SimpleKeywords SHARED_NETWORK6_PARAMETERS
This table defines all shared network parameters for DHCPv6.
A template representing an optional value.
Definition optional.h:36
T get() const
Retrieves the encapsulated value.
Definition optional.h:114
bool empty() const
Checks if the encapsulated value is empty.
Definition optional.h:153
void unspecified(bool unspecified)
Modifies the flag that indicates whether the value is specified or unspecified.
Definition optional.h:136
@ D6O_INTERFACE_ID
Definition dhcp6.h:38
#define isc_throw(type, stream)
A shortcut macro to insert known values into exception arguments.
boost::shared_ptr< const Element > ConstElementPtr
Definition data.h:29
boost::shared_ptr< Element > ElementPtr
Definition data.h:28
@ error
Definition db_log.h:118
boost::shared_ptr< const CfgGlobals > ConstCfgGlobalsPtr
Const shared pointer to a CfgGlobals instance.
boost::shared_ptr< Network4 > Network4Ptr
Pointer to the Network4 object.
Definition network.h:1479
boost::shared_ptr< CfgOption > CfgOptionPtr
Non-const pointer.
Definition cfg_option.h:832
boost::multi_index_container< Subnet6Ptr, boost::multi_index::indexed_by< boost::multi_index::ordered_unique< boost::multi_index::tag< SubnetSubnetIdIndexTag >, boost::multi_index::const_mem_fun< Subnet, SubnetID, &Subnet::getID > >, boost::multi_index::ordered_unique< boost::multi_index::tag< SubnetPrefixIndexTag >, boost::multi_index::const_mem_fun< Subnet, std::string, &Subnet::toText > >, boost::multi_index::ordered_non_unique< boost::multi_index::tag< SubnetModificationTimeIndexTag >, boost::multi_index::const_mem_fun< data::BaseStampedElement, boost::posix_time::ptime, &data::BaseStampedElement::getModificationTime > > > > Subnet6Collection
A collection of Subnet6 objects.
Definition subnet.h:937
boost::shared_ptr< SharedNetwork6 > SharedNetwork6Ptr
Pointer to SharedNetwork6 object.
boost::multi_index_container< Subnet4Ptr, boost::multi_index::indexed_by< boost::multi_index::ordered_unique< boost::multi_index::tag< SubnetSubnetIdIndexTag >, boost::multi_index::const_mem_fun< Subnet, SubnetID, &Subnet::getID > >, boost::multi_index::ordered_unique< boost::multi_index::tag< SubnetPrefixIndexTag >, boost::multi_index::const_mem_fun< Subnet, std::string, &Subnet::toText > >, boost::multi_index::ordered_non_unique< boost::multi_index::tag< SubnetServerIdIndexTag >, boost::multi_index::const_mem_fun< Network4, asiolink::IOAddress, &Network4::getServerId > >, boost::multi_index::ordered_non_unique< boost::multi_index::tag< SubnetModificationTimeIndexTag >, boost::multi_index::const_mem_fun< data::BaseStampedElement, boost::posix_time::ptime, &data::BaseStampedElement::getModificationTime > > > > Subnet4Collection
A collection of Subnet4 objects.
Definition subnet.h:866
boost::shared_ptr< Network > NetworkPtr
Pointer to the Network object.
Definition network.h:73
std::vector< uint8_t > OptionBuffer
buffer types used in DHCP code.
Definition option.h:24
boost::shared_ptr< SharedNetwork4 > SharedNetwork4Ptr
Pointer to SharedNetwork4 object.
boost::shared_ptr< Option > OptionPtr
Definition option.h:37
Defines the logger used by the top-level component of kea-lfc.