44 << expression_cfg->str() <<
"] must be a string, at ("
45 << expression_cfg->getPosition() <<
")");
51 expression_cfg->getValue(value);
55 << expression_cfg->getPosition() <<
")");
63 *expression = eval_ctx.expression_;
64 }
catch (
const std::exception& ex) {
67 "expression: [" << value
68 <<
"] error: " << ex.what() <<
" at ("
69 << expression_cfg->getPosition() <<
")");
79 bool append_error_position,
80 bool check_dependencies) {
82 std::string name =
getString(class_def_cfg,
"name");
85 "not empty parameter 'name' is required "
92 bool is_template =
false;
97 ConstElementPtr template_test_cfg = class_def_cfg->get(
"template-test");
98 if (test_cfg && template_test_cfg) {
100 << test_cfg->getPosition() <<
") and ("
101 << template_test_cfg->getPosition() <<
")");
104 bool depend_on_known =
false;
106 if (template_test_cfg) {
107 test_cfg = template_test_cfg;
111 check_defined = [&class_dictionary, &depend_on_known, check_dependencies](
const ClientClass& cclass) {
112 return (!check_dependencies ||
isClientClassDefined(class_dictionary, depend_on_known, cclass));
118 parser.
parse(match_expr, test_cfg, family, check_defined, parser_type);
119 test = test_cfg->stringValue();
133 for (
auto const& option_def : option_defs->listValue()) {
141 "Not allowed option definition for code '"
142 << def->getCode() <<
"' in space '"
143 << def->getOptionSpaceName() <<
"' at ("
144 << option_def->getPosition() <<
")");
148 }
catch (
const std::exception& ex) {
151 << option_def->getPosition() <<
")");
161 opts_parser->parse(options, option_data);
169 << user_context->getPosition() <<
")");
174 bool required =
false;
175 if (class_def_cfg->contains(
"only-if-required")) {
176 required =
getBoolean(class_def_cfg,
"only-if-required");
181 if (class_def_cfg->contains(
"next-server")) {
182 std::string next_server_txt =
getString(class_def_cfg,
"next-server");
184 next_server =
IOAddress(next_server_txt);
187 "Invalid next-server value specified: '"
188 << next_server_txt <<
"' ("
189 <<
getPosition(
"next-server", class_def_cfg) <<
")");
192 if (next_server.getFamily() != AF_INET) {
194 << next_server_txt <<
"', must be IPv4 address ("
195 <<
getPosition(
"next-server", class_def_cfg) <<
")");
198 if (next_server.isV4Bcast()) {
200 << next_server_txt <<
"', must not be a broadcast ("
201 <<
getPosition(
"next-server", class_def_cfg) <<
")");
207 if (class_def_cfg->contains(
"server-hostname")) {
208 sname =
getString(class_def_cfg,
"server-hostname");
213 << sname.length() <<
" ("
214 <<
getPosition(
"server-hostname", class_def_cfg) <<
")");
219 std::string filename;
220 if (class_def_cfg->contains(
"boot-file-name")) {
221 filename =
getString(class_def_cfg,
"boot-file-name");
226 << filename.length() <<
" ("
227 <<
getPosition(
"boot-file-name", class_def_cfg) <<
")");
231 Optional<uint32_t> offer_lft;
232 if (class_def_cfg->contains(
"offer-lifetime")) {
233 auto value =
getInteger(class_def_cfg,
"offer-lifetime");
236 << value <<
"' must be a positive number ("
237 <<
getPosition(
"offer-lifetime", class_def_cfg) <<
")");
244 Triplet<uint32_t> valid_lft =
parseIntTriplet(class_def_cfg,
"valid-lifetime");
246 Triplet<uint32_t> preferred_lft;
247 if (family != AF_INET) {
257 <<
"' only-if-required flag must be false");
261 <<
"' test expression must be empty");
267 if (name ==
"DROP") {
270 <<
"' only-if-required flag must be false");
277 class_dictionary->addClass(name, match_expr, test, required,
278 depend_on_known, options, defs,
279 user_context, next_server, sname, filename,
280 valid_lft, preferred_lft, is_template, offer_lft);
281 }
catch (
const std::exception& ex) {
282 std::ostringstream s;
283 s <<
"Can't add class: " << ex.what();
285 if (append_error_position) {
286 s <<
" (" << class_def_cfg->getPosition() <<
")";
294 const uint16_t family) {
296 if (!class_def_cfg || (class_def_cfg->getType() !=
Element::map)) {
301 static std::set<std::string> supported_params = {
"name",
307 "min-valid-lifetime",
308 "max-valid-lifetime",
312 static std::set<std::string> supported_params_v4 = {
"option-def",
318 static std::set<std::string> supported_params_v6 = {
"preferred-lifetime",
319 "min-preferred-lifetime",
320 "max-preferred-lifetime" };
323 for (
auto const& name_value_pair : class_def_cfg->mapValue()) {
324 if ((supported_params.count(name_value_pair.first) > 0) ||
325 ((family == AF_INET) && (supported_params_v4.count(name_value_pair.first) > 0)) ||
326 ((family != AF_INET) && (supported_params_v6.count(name_value_pair.first) > 0))) {
330 << name_value_pair.first <<
"'");
335boost::shared_ptr<OptionDataListParser>
338 auto parser = boost::make_shared<OptionDataListParser>(address_family, cfg_option_def);
346 uint16_t family,
bool check_dependencies) {
348 for (
auto const& client_class_def : client_class_def_list->listValue()) {
350 parser.
parse(dictionary, client_class_def, family,
true, check_dependencies);
virtual const char * what() const
Returns a C-style character string of the cause of the exception.
The IOAddress class represents an IP addresses (version agnostic)
An exception that is thrown if an error occurs within the IO module.
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 bool getBoolean(isc::data::ConstElementPtr scope, const std::string &name)
Returns a boolean parameter from a scope.
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.
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.
Parser for a single option definition.
static const size_t MAX_SNAME_LEN
length of the SNAME field in DHCPv4 message
static const size_t MAX_FILE_LEN
length of the FILE field in DHCPv4 message
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.
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
std::function< bool(const ClientClass &) CheckDefined)
Type of the check defined function.
static bool acceptAll(const ClientClass &client_class)
Accept all client class names.
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.
boost::shared_ptr< const Element > ConstElementPtr
std::string ClientClass
Defines a single class name.
boost::shared_ptr< CfgOption > CfgOptionPtr
Non-const pointer.
boost::shared_ptr< CfgOptionDef > CfgOptionDefPtr
Non-const pointer.
boost::shared_ptr< OptionDefinition > OptionDefinitionPtr
Pointer to option definition object.
boost::shared_ptr< Expression > ExpressionPtr
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.
std::vector< TokenPtr > Expression
This is a structure that holds an expression converted to RPN.
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.