Kea 3.1.8
dhcp4/json_config_parser.cc
Go to the documentation of this file.
1// Copyright (C) 2012-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
11#include <config/command_mgr.h>
19#include <dhcp4/dhcp4_log.h>
20#include <dhcp4/dhcp4_srv.h>
22#include <dhcp/libdhcp++.h>
27#include <dhcpsrv/cfg_option.h>
28#include <dhcpsrv/cfgmgr.h>
30#include <dhcpsrv/db_type.h>
45#include <dhcpsrv/host_mgr.h>
46#include <dhcpsrv/timer_mgr.h>
47#include <hooks/hooks_manager.h>
48#include <hooks/hooks_parser.h>
50#include <util/encode/encode.h>
52#include <boost/algorithm/string.hpp>
53#include <boost/lexical_cast.hpp>
54
55#include <iomanip>
56#include <iostream>
57#include <limits>
58#include <map>
59#include <netinet/in.h>
60#include <vector>
61
62using namespace isc::asiolink;
63using namespace isc::config;
64using namespace isc::data;
65using namespace isc::db;
66using namespace isc::dhcp;
67using namespace isc::hooks;
68using namespace isc::process;
69using namespace isc::util;
70using namespace isc;
71using namespace std;
72
73//
74// Register database backends
75//
76namespace {
77
86class Dhcp4ConfigParser : public isc::data::SimpleParser {
87public:
88
103 void parse(const SrvConfigPtr& cfg, const ConstElementPtr& global) {
104
105 // Set whether v4 server is supposed to echo back client-id
106 // (yes = RFC6842 compatible, no = backward compatibility)
107 bool echo_client_id = getBoolean(global, "echo-client-id");
108 cfg->setEchoClientId(echo_client_id);
109
110 // Set the probation period for decline handling.
111 uint32_t probation_period =
112 getUint32(global, "decline-probation-period");
113 cfg->setDeclinePeriod(probation_period);
114
115 // Set the DHCPv4-over-DHCPv6 interserver port.
116 uint16_t dhcp4o6_port = getUint16(global, "dhcp4o6-port");
117 cfg->setDhcp4o6Port(dhcp4o6_port);
118
119 // Set the global user context.
120 ConstElementPtr user_context = global->get("user-context");
121 if (user_context) {
122 cfg->setContext(user_context);
123 }
124
125 // Set the server's logical name
126 std::string server_tag = getString(global, "server-tag");
127 cfg->setServerTag(server_tag);
128 }
129
141 void parseEarly(const SrvConfigPtr& cfg, const ConstElementPtr& global) {
142 // Set ip-reservations-unique flag.
143 bool ip_reservations_unique = getBoolean(global, "ip-reservations-unique");
144 cfg->setIPReservationsUnique(ip_reservations_unique);
145 }
146
153 void
154 copySubnets4(const CfgSubnets4Ptr& dest, const CfgSharedNetworks4Ptr& from) {
155
156 if (!dest || !from) {
157 isc_throw(BadValue, "Unable to copy subnets: at least one pointer is null");
158 }
159
160 const SharedNetwork4Collection* networks = from->getAll();
161 if (!networks) {
162 // Nothing to copy. Technically, it should return a pointer to empty
163 // container, but let's handle null pointer as well.
164 return;
165 }
166
167 // Let's go through all the networks one by one
168 for (auto const& net : *networks) {
169
170 // For each network go through all the subnets in it.
171 const Subnet4SimpleCollection* subnets = net->getAllSubnets();
172 if (!subnets) {
173 // Shared network without subnets it weird, but we decided to
174 // accept such configurations.
175 continue;
176 }
177
178 // For each subnet, add it to a list of regular subnets.
179 for (auto const& subnet : *subnets) {
180 dest->add(subnet);
181 }
182 }
183 }
184
193 void
194 sanityChecks(const SrvConfigPtr& cfg, const ConstElementPtr& global) {
195
197 cfg->sanityChecksLifetime("valid-lifetime");
198
200 cfg->sanityChecksDdnsTtlParameters();
201
203 const SharedNetwork4Collection* networks = cfg->getCfgSharedNetworks4()->getAll();
204 if (networks) {
205 sharedNetworksSanityChecks(*networks, global->get("shared-networks"));
206 }
207
208 // Sanity check use of SFLQ allocator.
209 cfg->sanityChecksSflqAllocator();
210 }
211
218 void
219 sharedNetworksSanityChecks(const SharedNetwork4Collection& networks,
220 ConstElementPtr json) {
221
223 if (!json) {
224 // No json? That means that the shared-networks was never specified
225 // in the config.
226 return;
227 }
228
229 // Used for names uniqueness checks.
230 std::set<string> names;
231
232 // Let's go through all the networks one by one
233 for (auto const& net : networks) {
234 string txt;
235
236 // Let's check if all subnets have either the same interface
237 // or don't have the interface specified at all.
238 bool authoritative = net->getAuthoritative();
239 string iface = net->getIface();
240
241 const Subnet4SimpleCollection* subnets = net->getAllSubnets();
242 if (subnets) {
243 // For each subnet, add it to a list of regular subnets.
244 for (auto const& subnet : *subnets) {
245 if (subnet->getAuthoritative() != authoritative) {
246 isc_throw(DhcpConfigError, "Subnet " << boolalpha
247 << subnet->toText()
248 << " has different authoritative setting "
249 << subnet->getAuthoritative()
250 << " than the shared-network itself: "
251 << authoritative);
252 }
253
254 if (iface.empty()) {
255 iface = subnet->getIface();
256 continue;
257 }
258
259 if (subnet->getIface().empty()) {
260 continue;
261 }
262
263 if (subnet->getIface() != iface) {
264 isc_throw(DhcpConfigError, "Subnet " << subnet->toText()
265 << " has specified interface " << subnet->getIface()
266 << ", but earlier subnet in the same shared-network"
267 << " or the shared-network itself used " << iface);
268 }
269
270 // Let's collect the subnets in case we later find out the
271 // subnet doesn't have a mandatory name.
272 txt += subnet->toText() + " ";
273 }
274 }
275
276 // Next, let's check name of the shared network.
277 if (net->getName().empty()) {
278 isc_throw(DhcpConfigError, "Shared-network with subnets "
279 << txt << " is missing mandatory 'name' parameter");
280 }
281
282 // Is it unique?
283 if (names.find(net->getName()) != names.end()) {
284 isc_throw(DhcpConfigError, "A shared-network with "
285 "name " << net->getName() << " defined twice.");
286 }
287 names.insert(net->getName());
288
289 }
290 }
291};
292
293} // anonymous namespace
294
295namespace isc {
296namespace dhcp {
297
306 // Get new UNIX socket configuration.
307 ConstElementPtr unix_config =
308 CfgMgr::instance().getStagingCfg()->getUnixControlSocketInfo();
309
310 // Get current UNIX socket configuration.
311 ConstElementPtr current_unix_config =
312 CfgMgr::instance().getCurrentCfg()->getUnixControlSocketInfo();
313
314 // Determine if the socket configuration has changed. It has if
315 // both old and new configuration is specified but respective
316 // data elements aren't equal.
317 bool sock_changed = (unix_config && current_unix_config &&
318 !unix_config->equals(*current_unix_config));
319
320 // If the previous or new socket configuration doesn't exist or
321 // the new configuration differs from the old configuration we
322 // close the existing socket and open a new socket as appropriate.
323 // Note that closing an existing socket means the client will not
324 // receive the configuration result.
325 if (!unix_config || !current_unix_config || sock_changed) {
326 if (unix_config) {
327 // This will create a control socket and install the external
328 // socket in IfaceMgr. That socket will be monitored when
329 // Dhcp4Srv::receivePacket() calls IfaceMgr::receive4() and
330 // callback in CommandMgr will be called, if necessary.
332 } else if (current_unix_config) {
334 }
335 }
336
337 // Get new HTTP/HTTPS socket configuration.
338 ConstElementPtr http_config =
339 CfgMgr::instance().getStagingCfg()->getHttpControlSocketInfo();
340
341 // Get current HTTP/HTTPS socket configuration.
342 ConstElementPtr current_http_config =
343 CfgMgr::instance().getCurrentCfg()->getHttpControlSocketInfo();
344
345 if (http_config) {
347 } else if (current_http_config) {
349 }
350}
351
358 // Revert any runtime option definitions configured so far and not committed.
360 // Let's set empty container in case a user hasn't specified any configuration
361 // for option definitions. This is equivalent to committing empty container.
363
365
366 // Answer will hold the result.
367 ConstElementPtr answer;
368
369 // Global parameter name in case of an error.
370 string parameter_name;
371 ElementPtr mutable_cfg;
372 SrvConfigPtr srv_config;
373 try {
374 // Get the staging configuration.
375 srv_config = CfgMgr::instance().getStagingCfg();
376
377 // This is a way to convert ConstElementPtr to ElementPtr.
378 // We need a config that can be edited, because we will insert
379 // default values and will insert derived values as well.
380 mutable_cfg = boost::const_pointer_cast<Element>(config_set);
381
382 // Set all default values if not specified by the user.
384
385 // And now derive (inherit) global parameters to subnets, if not specified.
387
388 // In principle we could have the following code structured as a series
389 // of long if else if clauses. That would give a marginal performance
390 // boost, but would make the code less readable. We had serious issues
391 // with the parser code debugability, so I decided to keep it as a
392 // series of independent ifs.
393
394 // This parser is used in several places.
395 Dhcp4ConfigParser global_parser;
396
397 // Apply global options in the staging config, e.g. ip-reservations-unique
398 global_parser.parseEarly(srv_config, mutable_cfg);
399
400 // We need definitions first
401 ConstElementPtr option_defs = mutable_cfg->get("option-def");
402 if (option_defs) {
403 parameter_name = "option-def";
404 OptionDefListParser parser(AF_INET);
405 CfgOptionDefPtr cfg_option_def = srv_config->getCfgOptionDef();
406 parser.parse(cfg_option_def, option_defs);
407 }
408
409 ConstElementPtr option_datas = mutable_cfg->get("option-data");
410 if (option_datas) {
411 parameter_name = "option-data";
412 OptionDataListParser parser(AF_INET);
413 CfgOptionPtr cfg_option = srv_config->getCfgOption();
414 parser.parse(cfg_option, option_datas);
415 }
416
417 ConstElementPtr control_socket = mutable_cfg->get("control-socket");
418 if (control_socket) {
419 mutable_cfg->remove("control-socket");
420 ElementPtr l = Element::createList(control_socket->getPosition());
421 l->add(UserContext::toElement(control_socket));
422 mutable_cfg->set("control-sockets", l);
423 }
424
425 ConstElementPtr control_sockets = mutable_cfg->get("control-sockets");
426 if (control_sockets) {
427 parameter_name = "control-sockets";
429 parser.parse(*srv_config, control_sockets);
430 }
431
432 ConstElementPtr multi_threading = mutable_cfg->get("multi-threading");
433 if (multi_threading) {
434 parameter_name = "multi-threading";
436 parser.parse(*srv_config, multi_threading);
437 }
438
439 bool multi_threading_enabled = true;
440 uint32_t thread_count = 0;
441 uint32_t queue_size = 0;
442 CfgMultiThreading::extract(CfgMgr::instance().getStagingCfg()->getDHCPMultiThreading(),
443 multi_threading_enabled, thread_count, queue_size);
444
446 ConstElementPtr queue_control = mutable_cfg->get("dhcp-queue-control");
447 if (queue_control) {
448 parameter_name = "dhcp-queue-control";
450 srv_config->setDHCPQueueControl(parser.parse(queue_control, multi_threading_enabled));
451 }
452
454 ConstElementPtr reservations_lookup_first = mutable_cfg->get("reservations-lookup-first");
455 if (reservations_lookup_first) {
456 parameter_name = "reservations-lookup-first";
457 if (multi_threading_enabled) {
459 }
460 srv_config->setReservationsLookupFirst(reservations_lookup_first->boolValue());
461 }
462
463 ConstElementPtr hr_identifiers =
464 mutable_cfg->get("host-reservation-identifiers");
465 if (hr_identifiers) {
466 parameter_name = "host-reservation-identifiers";
467 HostReservationIdsParser4 parser(srv_config->getCfgHostOperations4());
468 parser.parse(hr_identifiers);
469 }
470
471 ConstElementPtr sanity_checks = mutable_cfg->get("sanity-checks");
472 if (sanity_checks) {
473 parameter_name = "sanity-checks";
474 SanityChecksParser parser;
475 parser.parse(*srv_config, sanity_checks);
476 }
477
478 ConstElementPtr expiration_cfg =
479 mutable_cfg->get("expired-leases-processing");
480 if (expiration_cfg) {
481 parameter_name = "expired-leases-processing";
483 parser.parse(expiration_cfg, CfgMgr::instance().getStagingCfg()->getCfgExpiration());
484 }
485
486 // The hooks-libraries configuration must be parsed after parsing
487 // multi-threading configuration so that libraries are checked
488 // for multi-threading compatibility.
489 ConstElementPtr hooks_libraries = mutable_cfg->get("hooks-libraries");
490 if (hooks_libraries) {
491 parameter_name = "hooks-libraries";
492 HooksLibrariesParser hooks_parser;
493 HooksConfig& libraries = srv_config->getHooksConfig();
494 hooks_parser.parse(libraries, hooks_libraries);
495 libraries.verifyLibraries(hooks_libraries->getPosition(),
496 multi_threading_enabled);
497 }
498
499 // D2 client configuration.
500 D2ClientConfigPtr d2_client_cfg;
501
502 // Legacy DhcpConfigParser stuff below.
503 ConstElementPtr dhcp_ddns = mutable_cfg->get("dhcp-ddns");
504 if (dhcp_ddns) {
505 parameter_name = "dhcp-ddns";
506 // Apply defaults
509 d2_client_cfg = parser.parse(dhcp_ddns);
510 }
511
512 ConstElementPtr client_classes = mutable_cfg->get("client-classes");
513 if (client_classes) {
514 parameter_name = "client-classes";
516 ClientClassDictionaryPtr dictionary =
517 parser.parse(client_classes, AF_INET);
518 srv_config->setClientClassDictionary(dictionary);
519 }
520
521 // Please move at the end when migration will be finished.
522 ConstElementPtr lease_database = mutable_cfg->get("lease-database");
523 if (lease_database) {
524 parameter_name = "lease-database";
525 db::DbAccessParser parser;
526 std::string access_string;
527 parser.parse(access_string, lease_database);
528 CfgDbAccessPtr cfg_db_access = srv_config->getCfgDbAccess();
529 cfg_db_access->setLeaseDbAccessString(access_string);
530 }
531
532 ConstElementPtr hosts_database = mutable_cfg->get("hosts-database");
533 if (hosts_database) {
534 parameter_name = "hosts-database";
535 db::DbAccessParser parser;
536 std::string access_string;
537 parser.parse(access_string, hosts_database);
538 CfgDbAccessPtr cfg_db_access = srv_config->getCfgDbAccess();
539 cfg_db_access->setHostDbAccessString(access_string);
540 }
541
542 ConstElementPtr hosts_databases = mutable_cfg->get("hosts-databases");
543 if (hosts_databases) {
544 parameter_name = "hosts-databases";
545 CfgDbAccessPtr cfg_db_access = srv_config->getCfgDbAccess();
546 for (auto const& it : hosts_databases->listValue()) {
547 db::DbAccessParser parser;
548 std::string access_string;
549 parser.parse(access_string, it);
550 cfg_db_access->setHostDbAccessString(access_string);
551 }
552 }
553
554 // Keep relative orders of shared networks and subnets.
555 ConstElementPtr shared_networks = mutable_cfg->get("shared-networks");
556 if (shared_networks) {
557 parameter_name = "shared-networks";
564 CfgSharedNetworks4Ptr cfg = srv_config->getCfgSharedNetworks4();
565 parser.parse(cfg, shared_networks);
566
567 // We also need to put the subnets it contains into normal
568 // subnets list.
569 global_parser.copySubnets4(srv_config->getCfgSubnets4(), cfg);
570 }
571
572 ConstElementPtr subnet4 = mutable_cfg->get("subnet4");
573 if (subnet4) {
574 parameter_name = "subnet4";
575 Subnets4ListConfigParser subnets_parser;
576 // parse() returns number of subnets parsed. We may log it one day.
577 subnets_parser.parse(srv_config, subnet4);
578 }
579
580 ConstElementPtr reservations = mutable_cfg->get("reservations");
581 if (reservations) {
582 parameter_name = "reservations";
583 HostCollection hosts;
585 parser.parse(SUBNET_ID_GLOBAL, reservations, hosts);
586 for (auto const& h : hosts) {
587 srv_config->getCfgHosts()->add(h);
588 }
589 }
590
591 ConstElementPtr config_control = mutable_cfg->get("config-control");
592 if (config_control) {
593 parameter_name = "config-control";
594 ConfigControlParser parser;
595 ConfigControlInfoPtr config_ctl_info = parser.parse(config_control);
596 CfgMgr::instance().getStagingCfg()->setConfigControlInfo(config_ctl_info);
597 }
598
599 ConstElementPtr compatibility = mutable_cfg->get("compatibility");
600 if (compatibility) {
601 CompatibilityParser parser;
602 parser.parse(compatibility, *CfgMgr::instance().getStagingCfg());
603 }
604
605 // Make parsers grouping.
606 const std::map<std::string, ConstElementPtr>& values_map =
607 mutable_cfg->mapValue();
608
609 for (auto const& config_pair : values_map) {
610 parameter_name = config_pair.first;
611
612 // These are converted to SimpleParser and are handled already above.
613 if ((config_pair.first == "option-def") ||
614 (config_pair.first == "option-data") ||
615 (config_pair.first == "control-socket") ||
616 (config_pair.first == "control-sockets") ||
617 (config_pair.first == "multi-threading") ||
618 (config_pair.first == "dhcp-queue-control") ||
619 (config_pair.first == "host-reservation-identifiers") ||
620 (config_pair.first == "interfaces-config") ||
621 (config_pair.first == "sanity-checks") ||
622 (config_pair.first == "expired-leases-processing") ||
623 (config_pair.first == "hooks-libraries") ||
624 (config_pair.first == "dhcp-ddns") ||
625 (config_pair.first == "client-classes") ||
626 (config_pair.first == "lease-database") ||
627 (config_pair.first == "hosts-database") ||
628 (config_pair.first == "hosts-databases") ||
629 (config_pair.first == "subnet4") ||
630 (config_pair.first == "shared-networks") ||
631 (config_pair.first == "reservations") ||
632 (config_pair.first == "config-control") ||
633 (config_pair.first == "loggers") ||
634 (config_pair.first == "compatibility")) {
635 continue;
636 }
637
638 // As of Kea 1.6.0 we have two ways of inheriting the global parameters.
639 // The old method is used in JSON configuration parsers when the global
640 // parameters are derived into the subnets and shared networks and are
641 // being treated as explicitly specified. The new way used by the config
642 // backend is the dynamic inheritance whereby each subnet and shared
643 // network uses a callback function to return global parameter if it
644 // is not specified at lower level. This callback uses configured globals.
645 // We deliberately include both default and explicitly specified globals
646 // so as the callback can access the appropriate global values regardless
647 // whether they are set to a default or other value.
648 if ( (config_pair.first == "renew-timer") ||
649 (config_pair.first == "rebind-timer") ||
650 (config_pair.first == "valid-lifetime") ||
651 (config_pair.first == "min-valid-lifetime") ||
652 (config_pair.first == "max-valid-lifetime") ||
653 (config_pair.first == "decline-probation-period") ||
654 (config_pair.first == "dhcp4o6-port") ||
655 (config_pair.first == "echo-client-id") ||
656 (config_pair.first == "match-client-id") ||
657 (config_pair.first == "authoritative") ||
658 (config_pair.first == "next-server") ||
659 (config_pair.first == "server-hostname") ||
660 (config_pair.first == "boot-file-name") ||
661 (config_pair.first == "server-tag") ||
662 (config_pair.first == "reservations-global") ||
663 (config_pair.first == "reservations-in-subnet") ||
664 (config_pair.first == "reservations-out-of-pool") ||
665 (config_pair.first == "calculate-tee-times") ||
666 (config_pair.first == "t1-percent") ||
667 (config_pair.first == "t2-percent") ||
668 (config_pair.first == "cache-threshold") ||
669 (config_pair.first == "cache-max-age") ||
670 (config_pair.first == "adaptive-lease-time-threshold") ||
671 (config_pair.first == "hostname-char-set") ||
672 (config_pair.first == "hostname-char-replacement") ||
673 (config_pair.first == "ddns-send-updates") ||
674 (config_pair.first == "ddns-override-no-update") ||
675 (config_pair.first == "ddns-override-client-update") ||
676 (config_pair.first == "ddns-replace-client-name") ||
677 (config_pair.first == "ddns-generated-prefix") ||
678 (config_pair.first == "ddns-qualifying-suffix") ||
679 (config_pair.first == "ddns-update-on-renew") ||
680 (config_pair.first == "ddns-use-conflict-resolution") ||
681 (config_pair.first == "ddns-conflict-resolution-mode") ||
682 (config_pair.first == "ddns-ttl-percent") ||
683 (config_pair.first == "store-extended-info") ||
684 (config_pair.first == "statistic-default-sample-count") ||
685 (config_pair.first == "statistic-default-sample-age") ||
686 (config_pair.first == "early-global-reservations-lookup") ||
687 (config_pair.first == "ip-reservations-unique") ||
688 (config_pair.first == "reservations-lookup-first") ||
689 (config_pair.first == "parked-packet-limit") ||
690 (config_pair.first == "allocator") ||
691 (config_pair.first == "offer-lifetime") ||
692 (config_pair.first == "ddns-ttl") ||
693 (config_pair.first == "ddns-ttl-min") ||
694 (config_pair.first == "ddns-ttl-max") ||
695 (config_pair.first == "stash-agent-options")) {
696 CfgMgr::instance().getStagingCfg()->addConfiguredGlobal(config_pair.first,
697 config_pair.second);
698 continue;
699 }
700
701 // Nothing to configure for the user-context.
702 if (config_pair.first == "user-context") {
703 continue;
704 }
705
706 // If we got here, no code handled this parameter, so we bail out.
708 "unsupported global configuration parameter: " << config_pair.first
709 << " (" << config_pair.second->getPosition() << ")");
710 }
711
712 // Reset parameter name.
713 parameter_name = "<post parsing>";
714
715 // Apply global options in the staging config.
716 global_parser.parse(srv_config, mutable_cfg);
717
718 // This method conducts final sanity checks and tweaks. In particular,
719 // it checks that there is no conflict between plain subnets and those
720 // defined as part of shared networks.
721 global_parser.sanityChecks(srv_config, mutable_cfg);
722
723 // Validate D2 client configuration.
724 if (!d2_client_cfg) {
725 d2_client_cfg.reset(new D2ClientConfig());
726 }
727 d2_client_cfg->validateContents();
728 srv_config->setD2ClientConfig(d2_client_cfg);
729 } catch (const isc::Exception& ex) {
731 .arg(parameter_name).arg(ex.what());
733 } catch (...) {
734 // For things like bad_cast in boost::lexical_cast
735 LOG_ERROR(dhcp4_logger, DHCP4_PARSER_EXCEPTION).arg(parameter_name);
736 answer = isc::config::createAnswer(CONTROL_RESULT_ERROR, "undefined configuration "
737 "processing error");
738 }
739
740 if (!answer) {
741 answer = isc::config::createAnswer(CONTROL_RESULT_SUCCESS, "Configuration seems sane. "
742 "Control-socket, hook-libraries, and D2 configuration "
743 "were sanity checked, but not applied.");
744 }
745
746 return (answer);
747}
748
751 bool check_only, bool extra_checks) {
752 if (!config_set) {
754 "Can't parse NULL config");
755 return (answer);
756 }
757
759 .arg(server.redactConfig(config_set)->str());
760
761 // Resource managers usually check for @ref MultiThreadingMgr::isTestMode
762 // value and do not open UNIX or TCP/UDP sockets, lock files, nor do they open
763 // or rotate files, as any of these actions could interfere with a running
764 // process on the same machine.
765 std::unique_ptr<MtTestMode> mt_test_mode;
766 if (check_only) {
767 mt_test_mode.reset(new MtTestMode());
768 }
769
770 auto answer = processDhcp4Config(config_set);
771
772 int status_code = CONTROL_RESULT_SUCCESS;
773 isc::config::parseAnswer(status_code, answer);
774
775 SrvConfigPtr srv_config;
776
777 if (status_code == CONTROL_RESULT_SUCCESS) {
778 if (check_only) {
779 if (extra_checks) {
780 std::ostringstream err;
781 // Configure DHCP packet queueing
782 try {
784 qc = CfgMgr::instance().getStagingCfg()->getDHCPQueueControl();
785 if (IfaceMgr::instance().configureDHCPPacketQueue(AF_INET, qc)) {
787 .arg(IfaceMgr::instance().getPacketQueue4()->getInfoStr());
788 }
789
790 } catch (const std::exception& ex) {
791 err << "Error setting packet queue controls after server reconfiguration: "
792 << ex.what();
794 status_code = CONTROL_RESULT_ERROR;
795 }
796 }
797 } else {
798
799 // Usually unit tests create managers before calling configureDhcp4Server and
800 // do not call ControlledDhcpv4Srv::processConfig.
801 // Runtime code path creates the managers after calling configureDhcp4Server
802 // and they need to be reset just after successful configuration parsing.
803 if (!IfaceMgr::instance().isTestMode()) {
804 // Destroy lease manager before hooks unload.
806
807 // Destroy host manager before hooks unload.
809 }
810
811 // disable multi-threading (it will be applied by new configuration)
812 // this must be done in order to properly handle MT to ST transition
813 // when 'multi-threading' structure is missing from new config and
814 // to properly drop any task items stored in the thread pool which
815 // might reference some handles to loaded hooks, preventing them
816 // from being unloaded.
817 MultiThreadingMgr::instance().apply(false, 0, 0);
818
819 // Close DHCP sockets and remove any existing timers.
821 TimerMgr::instance()->unregisterTimers();
822 server.discardPackets();
823 server.getCBControl()->reset();
824 }
825 }
826
827 // Parsing stage is complete. The current configuration has not been altered.
828 // From this stage on, every error might have irreversible consequences and the
829 // configuration might not be restored to a working state.
830 if (status_code == CONTROL_RESULT_SUCCESS) {
831 string parameter_name;
832 ElementPtr mutable_cfg;
833 try {
834 // Get the staging configuration.
835 srv_config = CfgMgr::instance().getStagingCfg();
836
837 // This is a way to convert ConstElementPtr to ElementPtr.
838 // We need a config that can be edited, because we will insert
839 // default values and will insert derived values as well.
840 mutable_cfg = boost::const_pointer_cast<Element>(config_set);
841
842 ConstElementPtr ifaces_config = mutable_cfg->get("interfaces-config");
843 if (ifaces_config) {
844 parameter_name = "interfaces-config";
845 IfacesConfigParser parser(AF_INET, check_only);
846 CfgIfacePtr cfg_iface = srv_config->getCfgIface();
847 cfg_iface->reset();
848 parser.parse(cfg_iface, ifaces_config);
849 }
850 } catch (const isc::Exception& ex) {
852 .arg(parameter_name).arg(ex.what());
853 if (!check_only || extra_checks) {
854 status_code = CONTROL_RESULT_FATAL_ERROR;
855 } else {
856 status_code = CONTROL_RESULT_ERROR;
857 }
858 answer = isc::config::createAnswer(status_code, ex.what());
859 } catch (...) {
860 // For things like bad_cast in boost::lexical_cast
861 LOG_ERROR(dhcp4_logger, DHCP4_PARSER_EXCEPTION).arg(parameter_name);
862 if (!check_only || extra_checks) {
863 status_code = CONTROL_RESULT_FATAL_ERROR;
864 } else {
865 status_code = CONTROL_RESULT_ERROR;
866 }
867 answer = isc::config::createAnswer(status_code, "undefined configuration"
868 " processing error");
869 }
870 }
871
872 // So far so good, there was no parsing error so let's commit the
873 // configuration. This will add created subnets and option values into
874 // the server's configuration.
875 // This operation should be exception safe but let's make sure.
876 if (status_code == CONTROL_RESULT_SUCCESS && !check_only) {
877 try {
878
879 // Setup the command channel.
881 } catch (const isc::Exception& ex) {
883 status_code = CONTROL_RESULT_FATAL_ERROR;
884 answer = isc::config::createAnswer(status_code, ex.what());
885 } catch (...) {
886 // For things like bad_cast in boost::lexical_cast
888 status_code = CONTROL_RESULT_FATAL_ERROR;
889 answer = isc::config::createAnswer(status_code, "undefined configuration"
890 " parsing error");
891 }
892 }
893
894 if (status_code == CONTROL_RESULT_SUCCESS && (!check_only || extra_checks)) {
895 try {
896 // No need to commit interface names as this is handled by the
897 // CfgMgr::commit() function.
898
899 // Apply the staged D2ClientConfig, used to be done by parser commit
901 cfg = CfgMgr::instance().getStagingCfg()->getD2ClientConfig();
903 } catch (const isc::Exception& ex) {
905 status_code = CONTROL_RESULT_FATAL_ERROR;
906 answer = isc::config::createAnswer(status_code, ex.what());
907 } catch (...) {
908 // For things like bad_cast in boost::lexical_cast
910 status_code = CONTROL_RESULT_FATAL_ERROR;
911 answer = isc::config::createAnswer(status_code, "undefined configuration"
912 " parsing error");
913 }
914 }
915
916 if (status_code == CONTROL_RESULT_SUCCESS && (!check_only || extra_checks)) {
917 try {
918 // This occurs last as if it succeeds, there is no easy way to
919 // revert it. As a result, the failure to commit a subsequent
920 // change causes problems when trying to roll back.
922 static_cast<void>(HooksManager::unloadLibraries());
924 const HooksConfig& libraries =
925 CfgMgr::instance().getStagingCfg()->getHooksConfig();
926 bool multi_threading_enabled = true;
927 uint32_t thread_count = 0;
928 uint32_t queue_size = 0;
929 CfgMultiThreading::extract(CfgMgr::instance().getStagingCfg()->getDHCPMultiThreading(),
930 multi_threading_enabled, thread_count, queue_size);
931 libraries.loadLibraries(multi_threading_enabled);
932 } catch (const isc::Exception& ex) {
934 status_code = CONTROL_RESULT_FATAL_ERROR;
935 answer = isc::config::createAnswer(status_code, ex.what());
936 } catch (...) {
937 // For things like bad_cast in boost::lexical_cast
939 status_code = CONTROL_RESULT_FATAL_ERROR;
940 answer = isc::config::createAnswer(status_code, "undefined configuration"
941 " parsing error");
942 }
943
944 if (extra_checks && status_code == CONTROL_RESULT_SUCCESS) {
945 // Re-open lease and host database with new parameters.
946 try {
947 // Get the staging configuration.
948 srv_config = CfgMgr::instance().getStagingCfg();
949
950 // Create managers like @ref ControlledDhcpv4Srv::processConfig
951 // does. No need to change "persist" value here because
952 // @ref MultiThreadingMgr::isTestMode is used instead.
953 // This will also make log messages consistent with the checked
954 // config values.
955 CfgDbAccessPtr cfg_db = CfgMgr::instance().getStagingCfg()->getCfgDbAccess();
956 string params = "universe=4";
957 cfg_db->setAppendedParameters(params);
958 cfg_db->createManagers();
959 } catch (const std::exception& ex) {
960 status_code = CONTROL_RESULT_FATAL_ERROR;
961 answer = isc::config::createAnswer(status_code, ex.what());
962 }
963 }
964 }
965
966 // Log the list of known backends.
968
969 // Log the list of known backends.
971
972 // Log the list of known backends.
974
975 // Log the list of known backends.
977
978 // Moved from the commit block to add the config backend indication.
979 if (status_code == CONTROL_RESULT_SUCCESS && (!check_only || extra_checks)) {
980 try {
981 // If there are config backends, fetch and merge into staging config
982 server.getCBControl()->databaseConfigFetch(srv_config,
983 CBControlDHCPv4::FetchMode::FETCH_ALL);
984 } catch (const isc::Exception& ex) {
985 std::ostringstream err;
986 err << "during update from config backend database: " << ex.what();
988 status_code = CONTROL_RESULT_FATAL_ERROR;
989 answer = isc::config::createAnswer(status_code, err.str());
990 } catch (...) {
991 // For things like bad_cast in boost::lexical_cast
992 std::ostringstream err;
993 err << "during update from config backend database: "
994 << "undefined configuration parsing error";
996 status_code = CONTROL_RESULT_FATAL_ERROR;
997 answer = isc::config::createAnswer(status_code, err.str());
998 }
999 }
1000
1001 // Rollback changes as the configuration parsing failed.
1002 if (check_only || status_code != CONTROL_RESULT_SUCCESS) {
1003 // Revert to original configuration of runtime option definitions
1004 // in the libdhcp++.
1006
1007 if (status_code == CONTROL_RESULT_SUCCESS && extra_checks) {
1008 auto notify_libraries = ControlledDhcpv4Srv::finishConfigHookLibraries(config_set);
1009 if (notify_libraries) {
1010 return (notify_libraries);
1011 }
1012
1014 try {
1015 // Handle events registered by hooks using external IOService objects.
1017 } catch (const std::exception& ex) {
1018 std::ostringstream err;
1019 err << "Error initializing hooks: "
1020 << ex.what();
1022 }
1023 }
1024
1025 return (answer);
1026 }
1027
1029 .arg(CfgMgr::instance().getStagingCfg()->
1030 getConfigSummary(SrvConfig::CFGSEL_ALL4));
1031
1032 // Also calculate SHA256 hash of the config that was just set and
1033 // append it to the response.
1035 string hash = BaseCommandMgr::getHash(config);
1036 ElementPtr hash_map = Element::createMap();
1037 hash_map->set("hash", Element::create(hash));
1038
1039 // Everything was fine. Configuration is successful.
1040 answer = isc::config::createAnswer(CONTROL_RESULT_SUCCESS, "Configuration successful.", hash_map);
1041 return (answer);
1042}
1043
1044} // namespace dhcp
1045} // namespace isc
This is a base class for exceptions thrown from the DNS library module.
virtual const char * what() const
Returns a C-style character string of the cause of the exception.
void logRegistered()
Logs out all registered backends.
static std::string getHash(const isc::data::ConstElementPtr &config)
returns a hash of a given Element structure
void closeCommandSockets()
Close http control sockets.
static HttpCommandMgr & instance()
HttpCommandMgr is a singleton class.
void openCommandSockets(const isc::data::ConstElementPtr config)
Open http control sockets using configuration.
static UnixCommandMgr & instance()
UnixCommandMgr is a singleton class.
void openCommandSockets(const isc::data::ConstElementPtr config)
Opens unix control socket with parameters specified in socket_info (required parameters: socket-type:...
void closeCommandSockets()
Shuts down any open unix control sockets.
static ElementPtr create(const Position &pos=ZERO_POSITION())
Create a NullElement.
Definition data.cc:299
static ElementPtr createMap(const Position &pos=ZERO_POSITION())
Creates an empty MapElement type ElementPtr.
Definition data.cc:354
static ElementPtr createList(const Position &pos=ZERO_POSITION())
Creates an empty ListElement type ElementPtr.
Definition data.cc:349
Parse Database Parameters.
void parse(std::string &access_string, isc::data::ConstElementPtr database_config)
Parse configuration value.
void setD2ClientConfig(D2ClientConfigPtr &new_config)
Updates the DHCP-DDNS client configuration to the given value.
Definition cfgmgr.cc:45
static CfgMgr & instance()
returns a single instance of Configuration Manager
Definition cfgmgr.cc:29
SrvConfigPtr getStagingCfg()
Returns a pointer to the staging configuration.
Definition cfgmgr.cc:121
SrvConfigPtr getCurrentCfg()
Returns a pointer to the current configuration.
Definition cfgmgr.cc:116
static void extract(data::ConstElementPtr value, bool &enabled, uint32_t &thread_count, uint32_t &queue_size)
Extract multi-threading parameters from a given configuration.
Parser for a list of client class definitions.
ClientClassDictionaryPtr parse(isc::data::ConstElementPtr class_def_list, uint16_t family, bool check_dependencies=true)
Parse configuration entries.
void parse(isc::data::ConstElementPtr cfg, isc::dhcp::SrvConfig &srv_cfg)
Parse compatibility flags.
static ConfigBackendDHCPv4Mgr & instance()
Returns a sole instance of the ConfigBackendDHCPv4Mgr.
Parser for the control-sockets structure.
void parse(SrvConfig &srv_cfg, isc::data::ConstElementPtr value)
"Parses" control-sockets structure
static isc::data::ConstElementPtr finishConfigHookLibraries(isc::data::ConstElementPtr config)
Configuration checker for hook libraries.
Parser for D2ClientConfig.
D2ClientConfigPtr parse(isc::data::ConstElementPtr d2_client_cfg)
Parses a given dhcp-ddns element into D2ClientConfig.
static size_t setAllDefaults(isc::data::ConstElementPtr d2_config)
Sets all defaults for D2 client configuration.
Acts as a storage vault for D2 client configuration.
Parser for the configuration of DHCP packet queue controls.
data::ElementPtr parse(const isc::data::ConstElementPtr &control_elem, bool multi_threading_enabled)
Parses content of the "dhcp-queue-control".
To be removed. Please use ConfigError instead.
DHCPv4 server service.
Definition dhcp4_srv.h:268
CBControlDHCPv4Ptr getCBControl() const
Returns an object which controls access to the configuration backends.
Definition dhcp4_srv.h:333
void discardPackets()
Discards parked packets Clears the packet parking lots of all packets.
Parser for the configuration parameters pertaining to the processing of expired leases.
void parse(isc::data::ConstElementPtr expiration_config, isc::dhcp::CfgExpirationPtr expiration)
Parses parameters in the JSON map, pertaining to the processing of the expired leases.
static void logRegistered()
Logs out all registered backends.
static void create()
Creates new instance of the HostMgr.
Definition host_mgr.cc:52
Parser for a list of host identifiers for DHCPv4.
void parse(isc::data::ConstElementPtr ids_list)
Parses a list of host identifiers.
Parser for a list of host reservations for a subnet.
void parse(const SubnetID &subnet_id, isc::data::ConstElementPtr hr_list, HostCollection &hosts_list)
Parses a list of host reservation entries for a subnet.
void detectIfaces(bool update_only=false)
Detects network interfaces.
static IfaceMgr & instance()
IfaceMgr is a singleton class.
Definition iface_mgr.cc:49
void closeSockets()
Closes all open sockets.
Definition iface_mgr.cc:287
Parser for the configuration of interfaces.
void parse(const CfgIfacePtr &config, const isc::data::ConstElementPtr &values)
Parses content of the "interfaces-config".
static void destroy()
Destroy lease manager.
static void logRegistered()
Logs out all registered backends.
static void logRegistered()
Logs out all registered backends.
static void setRuntimeOptionDefs(const OptionDefSpaceContainer &defs)
Copies option definitions created at runtime.
Definition libdhcp++.cc:224
static void revertRuntimeOptionDefs()
Reverts uncommitted changes to runtime option definitions.
Definition libdhcp++.cc:243
Simple parser for multi-threading structure.
void parse(SrvConfig &srv_cfg, const isc::data::ConstElementPtr &value)
parses JSON structure.
Parser for option data values within a subnet.
void parse(const CfgOptionPtr &cfg, isc::data::ConstElementPtr option_data_list, bool encapsulate=true)
Parses a list of options, instantiates them and stores in cfg.
Parser for a list of option definitions.
void parse(CfgOptionDefPtr cfg, isc::data::ConstElementPtr def_list)
Parses a list of option definitions, create them and store in cfg.
Class of option definition space container.
Simple parser for sanity-checks structure.
void parse(SrvConfig &srv_cfg, const isc::data::ConstElementPtr &value)
parses JSON structure
void parse(CfgSharedNetworksTypePtr &cfg, const data::ConstElementPtr &shared_networks_list_data)
Parses a list of shared networks.
static size_t deriveParameters(isc::data::ElementPtr global)
Derives (inherits) all parameters from global to more specific scopes.
static size_t setAllDefaults(isc::data::ElementPtr global)
Sets all defaults for DHCPv4 configuration.
static const uint32_t CFGSEL_ALL4
IPv4 related config.
Definition srv_config.h:74
this class parses list of DHCP4 subnets
size_t parse(SrvConfigPtr cfg, data::ConstElementPtr subnets_list, bool encapsulate_options=true)
parses contents of the list
static const TimerMgrPtr & instance()
Returns pointer to the sole instance of the TimerMgr.
Definition timer_mgr.cc:446
Wrapper class that holds hooks libraries configuration.
void verifyLibraries(const isc::data::Element::Position &position, bool multi_threading_enabled) const
Verifies that libraries stored in libraries_ are valid.
void loadLibraries(bool multi_threading_enabled) const
Commits hooks libraries configuration.
Parser for hooks library list.
void parse(HooksConfig &libraries, isc::data::ConstElementPtr value)
Parses parameters value.
static bool unloadLibraries()
Unload libraries.
static void prepareUnloadLibraries()
Prepare the unloading of libraries.
Implements parser for config control information, "config-control".
ConfigControlInfoPtr parse(const data::ConstElementPtr &config_control)
Parses a configuration control Element.
isc::data::ConstElementPtr redactConfig(isc::data::ConstElementPtr const &config)
Redact a configuration.
Definition daemon.cc:297
RAII wrapper for MT test mode.
static MultiThreadingMgr & instance()
Returns a single instance of Multi Threading Manager.
void apply(bool enabled, uint32_t thread_count, uint32_t queue_size)
Apply the multi-threading related settings.
Parsers for client class definitions.
This file contains several functions and constants that are used for handling commands and responses ...
Contains declarations for loggers used by the DHCPv4 server component.
#define isc_throw(type, stream)
A shortcut macro to insert known values into exception arguments.
#define LOG_ERROR(LOGGER, MESSAGE)
Macro to conveniently test error output and log it.
Definition macros.h:32
#define LOG_INFO(LOGGER, MESSAGE)
Macro to conveniently test info output and log it.
Definition macros.h:20
#define LOG_WARN(LOGGER, MESSAGE)
Macro to conveniently test warn output and log it.
Definition macros.h:26
#define LOG_DEBUG(LOGGER, LEVEL, MESSAGE)
Macro to conveniently test debug output and log it.
Definition macros.h:14
ConstElementPtr parseAnswer(int &rcode, const ConstElementPtr &msg)
Parses a standard config/command level answer and returns arguments or text status code.
const int CONTROL_RESULT_ERROR
Status code indicating a general failure.
ConstElementPtr createAnswer(const int status_code, const std::string &text, const ConstElementPtr &arg)
Creates a standard config/command level answer message.
ConstElementPtr createAnswer()
Creates a standard config/command level success answer message (i.e.
const int CONTROL_RESULT_FATAL_ERROR
Status code indicating that the command was unsuccessful and the configuration could not be reverted ...
const int CONTROL_RESULT_SUCCESS
Status code indicating a successful operation.
boost::shared_ptr< const Element > ConstElementPtr
Definition data.h:30
boost::shared_ptr< Element > ElementPtr
Definition data.h:29
void configureCommandChannel()
Initialize the command channel based on the staging configuration.
const isc::log::MessageID DHCP4_CONFIG_START
boost::shared_ptr< D2ClientConfig > D2ClientConfigPtr
Defines a pointer for D2ClientConfig instances.
const isc::log::MessageID DHCP4_PARSER_EXCEPTION
boost::shared_ptr< CfgOption > CfgOptionPtr
Non-const pointer.
Definition cfg_option.h:973
SharedNetworksListParser< SharedNetwork4Parser > SharedNetworks4ListParser
Type of the shared networks list parser for IPv4.
boost::shared_ptr< CfgOptionDef > CfgOptionDefPtr
Non-const pointer.
boost::shared_ptr< CfgDbAccess > CfgDbAccessPtr
A pointer to the CfgDbAccess.
const isc::log::MessageID DHCP4_RESERVATIONS_LOOKUP_FIRST_ENABLED
isc::data::ConstElementPtr processDhcp4Config(isc::data::ConstElementPtr config_set)
Process a DHCPv4 configuration and return an answer stating if the configuration is valid,...
boost::shared_ptr< CfgIface > CfgIfacePtr
A pointer to the CfgIface .
Definition cfg_iface.h:501
const isc::log::MessageID DHCP4_CONFIG_PACKET_QUEUE
const isc::log::MessageID DHCP4_PARSER_COMMIT_EXCEPTION
boost::shared_ptr< SrvConfig > SrvConfigPtr
Non-const pointer to the SrvConfig.
const isc::log::MessageID DHCP4_PARSER_FAIL
std::vector< HostPtr > HostCollection
Collection of the Host objects.
Definition host.h:846
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 > > > > Subnet4SimpleCollection
A simple collection of Subnet4 objects.
Definition subnet.h:810
const int DBG_DHCP4_COMMAND
Debug level used to log receiving commands.
Definition dhcp4_log.h:30
const isc::log::MessageID DHCP4_PARSER_COMMIT_FAIL
boost::shared_ptr< ClientClassDictionary > ClientClassDictionaryPtr
Defines a pointer to a ClientClassDictionary.
isc::data::ConstElementPtr configureDhcp4Server(Dhcpv4Srv &server, isc::data::ConstElementPtr config_set, bool check_only, bool extra_checks)
Configure DHCPv4 server (Dhcpv4Srv) with a set of configuration values.
boost::shared_ptr< CfgSubnets4 > CfgSubnets4Ptr
Non-const pointer.
const isc::log::MessageID DHCP4_CONFIG_COMPLETE
boost::multi_index_container< SharedNetwork4Ptr, boost::multi_index::indexed_by< boost::multi_index::random_access< boost::multi_index::tag< SharedNetworkRandomAccessIndexTag > >, boost::multi_index::hashed_non_unique< boost::multi_index::tag< SharedNetworkIdIndexTag >, boost::multi_index::const_mem_fun< data::BaseStampedElement, uint64_t, &data::BaseStampedElement::getId > >, boost::multi_index::ordered_unique< boost::multi_index::tag< SharedNetworkNameIndexTag >, boost::multi_index::const_mem_fun< SharedNetwork4, std::string, &SharedNetwork4::getName > >, boost::multi_index::ordered_non_unique< boost::multi_index::tag< SharedNetworkServerIdIndexTag >, boost::multi_index::const_mem_fun< Network4, asiolink::IOAddress, &Network4::getServerId > >, boost::multi_index::ordered_non_unique< boost::multi_index::tag< SharedNetworkModificationTimeIndexTag >, boost::multi_index::const_mem_fun< data::BaseStampedElement, boost::posix_time::ptime, &data::BaseStampedElement::getModificationTime > > > > SharedNetwork4Collection
Multi index container holding shared networks.
boost::shared_ptr< CfgSharedNetworks4 > CfgSharedNetworks4Ptr
Pointer to the configuration of IPv4 shared networks.
isc::log::Logger dhcp4_logger(DHCP4_APP_LOGGER_NAME)
Base logger for DHCPv4 server.
Definition dhcp4_log.h:90
boost::shared_ptr< ConfigControlInfo > ConfigControlInfoPtr
Defines a pointer to a ConfigControlInfo.
Defines the logger used by the top-level component of kea-lfc.
static data::ElementPtr toElement(data::ConstElementPtr map)
Copy an Element map.