Kea 3.1.8
client_class_def_parser.cc
Go to the documentation of this file.
1// Copyright (C) 2015-2026 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#include <dhcp/libdhcp++.h>
9#include <dhcpsrv/cfgmgr.h>
11#include <dhcpsrv/dhcpsrv_log.h>
16#include <eval/eval_context.h>
17#include <asiolink/io_address.h>
18#include <asiolink/io_error.h>
19
20#include <algorithm>
21#include <sstream>
22
23using namespace isc::data;
24using namespace isc::asiolink;
25using namespace isc::util;
26using namespace std;
27
31
32namespace isc {
33namespace dhcp {
34
35// ********************** ExpressionParser ****************************
36
37void
39 ConstElementPtr expression_cfg,
40 uint16_t family,
41 EvalContext::CheckDefined check_defined,
42 EvalContext::ParserType parser_type) {
43 if (expression_cfg->getType() != Element::string) {
44 isc_throw(DhcpConfigError, "expression ["
45 << expression_cfg->str() << "] must be a string, at ("
46 << expression_cfg->getPosition() << ")");
47 }
48
49 // Get the expression's text via getValue() as the text returned
50 // by str() enclosed in quotes.
51 std::string value;
52 expression_cfg->getValue(value);
53
54 if (parser_type == EvalContext::PARSER_STRING && value.empty()) {
55 isc_throw(DhcpConfigError, "expression can not be empty at ("
56 << expression_cfg->getPosition() << ")");
57 }
58
59 try {
60 EvalContext eval_ctx(family == AF_INET ? Option::V4 : Option::V6,
61 check_defined);
62 eval_ctx.parseString(value, parser_type);
63 expression.reset(new Expression());
64 *expression = eval_ctx.expression_;
65 } catch (const std::exception& ex) {
66 // Append position if there is a failure.
68 "expression: [" << value
69 << "] error: " << ex.what() << " at ("
70 << expression_cfg->getPosition() << ")");
71 }
72}
73
74// ********************** ClientClassDefParser ****************************
75
76void
78 ConstElementPtr class_def_cfg,
79 uint16_t family,
80 bool append_error_position,
81 bool check_dependencies) {
82 // name is now mandatory, so let's deal with it first.
83 std::string name = getString(class_def_cfg, "name");
84 if (name.empty()) {
86 "not empty parameter 'name' is required "
87 << getPosition("name", class_def_cfg) << ")");
88 }
89
91
92 // Let's try to parse the template-test expression
93 bool is_template = false;
94
95 // Parse matching expression
96 ExpressionPtr match_expr;
97 ConstElementPtr test_cfg = class_def_cfg->get("test");
98 ConstElementPtr template_test_cfg = class_def_cfg->get("template-test");
99 if (test_cfg && template_test_cfg) {
100 isc_throw(DhcpConfigError, "can not use both 'test' and 'template-test' ("
101 << test_cfg->getPosition() << ") and ("
102 << template_test_cfg->getPosition() << ")");
103 }
104 std::string test;
105 bool depend_on_known = false;
107 if (template_test_cfg) {
108 test_cfg = template_test_cfg;
109 parser_type = EvalContext::PARSER_STRING;
110 is_template = true;
111 // We need to check for dependency on KNOWN, other than that we do not require
112 // classes in a template expression to be defined. This permits them to
113 // reference spawned classes. In other words, one template can depend on
114 // membership in the spawn of another template.
115 check_defined = [&depend_on_known](const ClientClass& cclass) {
116 // Check direct dependency on [UN]KNOWN
117 if ((cclass == "KNOWN") || (cclass == "UNKNOWN")) {
118 depend_on_known = true;
119 }
120
121 return (true);
122 };
123 } else {
124 check_defined =
125 [&class_dictionary, &depend_on_known, check_dependencies](const ClientClass& cclass) {
126 return (!check_dependencies || isClientClassDefined(class_dictionary,
127 depend_on_known, cclass));
128 };
129 }
130
131 if (test_cfg) {
132 ExpressionParser parser;
133 parser.parse(match_expr, test_cfg, family, check_defined, parser_type);
134 test = test_cfg->stringValue();
135 }
136
137 // Parse option def
138 CfgOptionDefPtr defs(new CfgOptionDef());
139 ConstElementPtr option_defs = class_def_cfg->get("option-def");
140 if (option_defs) {
141 // Apply defaults
143 family == AF_INET ?
146
147 OptionDefParser parser(family);
148 for (auto const& option_def : option_defs->listValue()) {
149 OptionDefinitionPtr def = parser.parse(option_def);
150
151 // Verify if the definition is for an option which is in a deferred
152 // processing list.
153 if (!LibDHCP::shouldDeferOptionUnpack(def->getOptionSpaceName(),
154 def->getCode())) {
156 "Not allowed option definition for code '"
157 << def->getCode() << "' in space '"
158 << def->getOptionSpaceName() << "' at ("
159 << option_def->getPosition() << ")");
160 }
161 try {
162 defs->add(def);
163 } catch (const std::exception& ex) {
164 // Sanity check: it should never happen
165 isc_throw(DhcpConfigError, ex.what() << " ("
166 << option_def->getPosition() << ")");
167 }
168 }
169 }
170
171 // Parse option data
172 CfgOptionPtr options(new CfgOption());
173 ConstElementPtr option_data = class_def_cfg->get("option-data");
174 if (option_data) {
175 auto opts_parser = createOptionDataListParser(family, defs);
176 opts_parser->parse(options, option_data);
177 }
178
179 // Parse user context
180 ConstElementPtr user_context = class_def_cfg->get("user-context");
181 if (user_context) {
182 if (user_context->getType() != Element::map) {
183 isc_throw(isc::dhcp::DhcpConfigError, "User context has to be a map ("
184 << user_context->getPosition() << ")");
185 }
186 }
187
188 // Let's try to parse the only-in-additional-list/only-if-required flag
189 auto required_elem = class_def_cfg->get("only-if-required");
190 auto additional_elem = class_def_cfg->get("only-in-additional-list");
191 if (required_elem) {
192 if (!additional_elem) {
194 additional_elem = required_elem;
195 } else {
197 "cannot specify both 'only-if-required' and "
198 "'only-in-additional-list'. Use only the latter.");
199 }
200 }
201
202 bool additional = false;
203 if (additional_elem) {
204 if (additional_elem->getType() == Element::boolean) {
205 additional = additional_elem->boolValue();
206 } else {
208 "'only-in-additional-list' must be boolean"
209 << additional_elem->getPosition());
210 }
211 }
212
213 // Let's try to parse the next-server field
214 IOAddress next_server("0.0.0.0");
215 if (class_def_cfg->contains("next-server")) {
216 std::string next_server_txt = getString(class_def_cfg, "next-server");
217 try {
218 next_server = IOAddress(next_server_txt);
219 } catch (const IOError& ex) {
221 "Invalid next-server value specified: '"
222 << next_server_txt << "' ("
223 << getPosition("next-server", class_def_cfg) << ")");
224 }
225
226 if (next_server.getFamily() != AF_INET) {
227 isc_throw(DhcpConfigError, "Invalid next-server value: '"
228 << next_server_txt << "', must be IPv4 address ("
229 << getPosition("next-server", class_def_cfg) << ")");
230 }
231
232 if (next_server.isV4Bcast()) {
233 isc_throw(DhcpConfigError, "Invalid next-server value: '"
234 << next_server_txt << "', must not be a broadcast ("
235 << getPosition("next-server", class_def_cfg) << ")");
236 }
237 }
238
239 // Let's try to parse server-hostname
240 std::string sname;
241 if (class_def_cfg->contains("server-hostname")) {
242 sname = getString(class_def_cfg, "server-hostname");
243
244 if (sname.length() >= Pkt4::MAX_SNAME_LEN) {
245 isc_throw(DhcpConfigError, "server-hostname must be at most "
246 << Pkt4::MAX_SNAME_LEN - 1 << " bytes long, it is "
247 << sname.length() << " ("
248 << getPosition("server-hostname", class_def_cfg) << ")");
249 }
250 }
251
252 // Let's try to parse boot-file-name
253 std::string filename;
254 if (class_def_cfg->contains("boot-file-name")) {
255 filename = getString(class_def_cfg, "boot-file-name");
256
257 if (filename.length() > Pkt4::MAX_FILE_LEN) {
258 isc_throw(DhcpConfigError, "boot-file-name must be at most "
259 << Pkt4::MAX_FILE_LEN - 1 << " bytes long, it is "
260 << filename.length() << " ("
261 << getPosition("boot-file-name", class_def_cfg) << ")");
262 }
263 }
264
265 Optional<uint32_t> offer_lft;
266 if (class_def_cfg->contains("offer-lifetime")) {
267 auto value = getInteger(class_def_cfg, "offer-lifetime");
268 if (value < 0) {
269 isc_throw(DhcpConfigError, "the value of offer-lifetime '"
270 << value << "' must be a positive number ("
271 << getPosition("offer-lifetime", class_def_cfg) << ")");
272 }
273
274 offer_lft = value;
275 }
276
277 // Parse valid lifetime triplet.
278 Triplet<uint32_t> valid_lft = parseIntTriplet(class_def_cfg, "valid-lifetime");
279
280 Triplet<uint32_t> preferred_lft;
281 if (family != AF_INET) {
282 // Parse preferred lifetime triplet.
283 preferred_lft = parseIntTriplet(class_def_cfg, "preferred-lifetime");
284 }
285
286 // Sanity checks on built-in classes
287 for (auto const& bn : builtinNames) {
288 if (name == bn) {
289 if (additional) {
290 isc_throw(DhcpConfigError, "built-in class '" << name
291 << "' only-in-additional-list flag must be false");
292 }
293 if (!test.empty()) {
294 isc_throw(DhcpConfigError, "built-in class '" << name
295 << "' test expression must be empty");
296 }
297 }
298 }
299
300 // Sanity checks on DROP
301 if (name == "DROP") {
302 if (additional) {
303 isc_throw(DhcpConfigError, "special class '" << name
304 << "' only-in-additional-list flag must be false");
305 }
306 // depend_on_known is now allowed
307 }
308
309 if (additional &&
310 (!valid_lft.unspecified() ||
311 !preferred_lft.unspecified() ||
312 !offer_lft.unspecified())) {
314 .arg(name);
315 }
316
317 // Add the client class definition
318 try {
319 class_dictionary->addClass(name, match_expr, test, additional,
320 depend_on_known, options, defs,
321 user_context, next_server, sname, filename,
322 valid_lft, preferred_lft, is_template, offer_lft);
323 } catch (const std::exception& ex) {
324 std::ostringstream s;
325 s << "Can't add class: " << ex.what();
326 // Append position of the error in JSON string if required.
327 if (append_error_position) {
328 s << " (" << class_def_cfg->getPosition() << ")";
329 }
330 isc_throw(DhcpConfigError, s.str());
331 }
332}
333
334void
336 const uint16_t family) {
337 // Make sure that the client class definition is stored in a map.
338 if (!class_def_cfg || (class_def_cfg->getType() != Element::map)) {
339 isc_throw(DhcpConfigError, "client class definition is not a map");
340 }
341
342 // Common v4 and v6 parameters supported for the client class.
343 static std::set<std::string> supported_params = { "name",
344 "test",
345 "option-data",
346 "user-context",
347 "only-if-required", // deprecated
348 "only-in-additional-list",
349 "valid-lifetime",
350 "min-valid-lifetime",
351 "max-valid-lifetime",
352 "template-test"};
353
354 // The v4 client class supports additional parameters.
355 static std::set<std::string> supported_params_v4 = { "option-def",
356 "next-server",
357 "server-hostname",
358 "boot-file-name" };
359
360 // The v6 client class supports additional parameters.
361 static std::set<std::string> supported_params_v6 = { "preferred-lifetime",
362 "min-preferred-lifetime",
363 "max-preferred-lifetime" };
364
365 // Iterate over the specified parameters and check if they are all supported.
366 for (auto const& name_value_pair : class_def_cfg->mapValue()) {
367 if ((supported_params.count(name_value_pair.first) > 0) ||
368 ((family == AF_INET) && (supported_params_v4.count(name_value_pair.first) > 0)) ||
369 ((family != AF_INET) && (supported_params_v6.count(name_value_pair.first) > 0))) {
370 continue;
371 } else {
372 isc_throw(DhcpConfigError, "unsupported client class parameter '"
373 << name_value_pair.first << "'");
374 }
375 }
376}
377
378boost::shared_ptr<OptionDataListParser>
380 CfgOptionDefPtr cfg_option_def) const {
381 auto parser = boost::make_shared<OptionDataListParser>(address_family, cfg_option_def);
382 return (parser);
383}
384
385// ****************** ClientClassDefListParser ************************
386
389 uint16_t family, bool check_dependencies) {
391 for (auto const& client_class_def : client_class_def_list->listValue()) {
393 parser.parse(dictionary, client_class_def, family, true, check_dependencies);
394 }
395 return (dictionary);
396}
397
398} // end of namespace isc::dhcp
399} // end of namespace isc
static size_t setListDefaults(isc::data::ConstElementPtr list, const SimpleDefaults &default_values)
Sets the default values for all entries in a list.
static const data::Element::Position & getPosition(const std::string &name, const data::ConstElementPtr parent)
Utility method that returns position of an element.
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 int64_t getInteger(isc::data::ConstElementPtr scope, const std::string &name)
Returns an integer parameter from a scope.
Represents option definitions used by the DHCP server.
Represents option data configuration for the DHCP server.
Definition cfg_option.h:404
ClientClassDictionaryPtr parse(isc::data::ConstElementPtr class_def_list, uint16_t family, bool check_dependencies=true)
Parse configuration entries.
Parser for a single client class definition.
virtual boost::shared_ptr< OptionDataListParser > createOptionDataListParser(const uint16_t address_family, CfgOptionDefPtr cfg_option_def) const
Returns an instance of the OptionDataListParser to be used in parsing the option-data structure.
void parse(ClientClassDictionaryPtr &class_dictionary, isc::data::ConstElementPtr client_class_def, uint16_t family, bool append_error_position=true, bool check_dependencies=true)
Parses an entry that describes single client class definition.
void checkParametersSupported(const isc::data::ConstElementPtr &class_def_cfg, const uint16_t family)
Iterates over class parameters and checks if they are supported.
Maintains a list of ClientClassDef's.
To be removed. Please use ConfigError instead.
Parser for a logical expression.
void parse(ExpressionPtr &expression, isc::data::ConstElementPtr expression_cfg, uint16_t family, isc::eval::EvalContext::CheckDefined check_defined=isc::eval::EvalContext::acceptAll, isc::eval::EvalContext::ParserType parser_type=isc::eval::EvalContext::PARSER_BOOL)
Parses an expression configuration element into an Expression.
static bool shouldDeferOptionUnpack(const std::string &space, const uint16_t code)
Checks if an option unpacking has to be deferred.
Definition libdhcp++.cc:286
Parser for a single option definition.
OptionDefinitionPtr parse(isc::data::ConstElementPtr option_def)
Parses an entry that describes single option definition.
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
static const isc::data::SimpleDefaults OPTION4_DEF_DEFAULTS
This table defines default values for option definitions in DHCPv4.
static const isc::data::SimpleDefaults OPTION6_DEF_DEFAULTS
This table defines default values for option definitions in DHCPv6.
Evaluation context, an interface to the expression evaluation.
std::function< bool(const ClientClass &)> CheckDefined
Type of the check defined function.
bool parseString(const std::string &str, ParserType type=PARSER_BOOL)
Run the parser on the string specified.
ParserType
Specifies what type of expression the parser is expected to see.
@ PARSER_BOOL
expression is expected to evaluate to bool
@ PARSER_STRING
expression is expected to evaluate to string
static bool acceptAll(const ClientClass &client_class)
Accept all client class names.
isc::dhcp::Expression expression_
Parsed expression (output tokens are stored here)
A template representing an optional value.
Definition optional.h:36
void unspecified(bool unspecified)
Modifies the flag that indicates whether the value is specified or unspecified.
Definition optional.h:136
This template specifies a parameter value.
Definition triplet.h:37
Defines classes for storing client class definitions.
Parsers for client class definitions.
#define isc_throw(type, stream)
A shortcut macro to insert known values into exception arguments.
#define LOG_WARN(LOGGER, MESSAGE)
Macro to conveniently test warn output and log it.
Definition macros.h:26
boost::shared_ptr< const Element > ConstElementPtr
Definition data.h:30
isc::log::Logger dhcpsrv_logger("dhcpsrv")
DHCP server library Logger.
Definition dhcpsrv_log.h:56
std::string ClientClass
Defines a single class name.
Definition classify.h:44
boost::shared_ptr< CfgOption > CfgOptionPtr
Non-const pointer.
Definition cfg_option.h:973
boost::shared_ptr< CfgOptionDef > CfgOptionDefPtr
Non-const pointer.
const isc::log::MessageID DHCPSRV_CLASS_WITH_ADDITIONAL_AND_LIFETIMES
boost::shared_ptr< OptionDefinition > OptionDefinitionPtr
Pointer to option definition object.
boost::shared_ptr< Expression > ExpressionPtr
Definition token.h:31
boost::shared_ptr< ClientClassDictionary > ClientClassDictionaryPtr
Defines a pointer to a ClientClassDictionary.
bool isClientClassDefined(ClientClassDictionaryPtr &class_dictionary, bool &depend_on_known, const ClientClass &client_class)
Check if a client class name is already defined, i.e.
const isc::log::MessageID DHCPSRV_ONLY_IF_REQUIRED_DEPRECATED
std::vector< TokenPtr > Expression
This is a structure that holds an expression converted to RPN.
Definition token.h:29
std::list< std::string > builtinNames
List of classes for which test expressions cannot be defined.
Defines the logger used by the top-level component of kea-lfc.