summaryrefslogtreecommitdiffstats
path: root/src/lib/dhcpsrv/cb_ctl_dhcp4.cc
diff options
context:
space:
mode:
Diffstat (limited to 'src/lib/dhcpsrv/cb_ctl_dhcp4.cc')
-rw-r--r--src/lib/dhcpsrv/cb_ctl_dhcp4.cc360
1 files changed, 360 insertions, 0 deletions
diff --git a/src/lib/dhcpsrv/cb_ctl_dhcp4.cc b/src/lib/dhcpsrv/cb_ctl_dhcp4.cc
new file mode 100644
index 0000000..5ba211f
--- /dev/null
+++ b/src/lib/dhcpsrv/cb_ctl_dhcp4.cc
@@ -0,0 +1,360 @@
+// Copyright (C) 2019-2023 Internet Systems Consortium, Inc. ("ISC")
+//
+// This Source Code Form is subject to the terms of the Mozilla Public
+// License, v. 2.0. If a copy of the MPL was not distributed with this
+// file, You can obtain one at http://mozilla.org/MPL/2.0/.
+
+#include <config.h>
+#include <dhcpsrv/cb_ctl_dhcp4.h>
+#include <dhcpsrv/cfgmgr.h>
+#include <dhcpsrv/client_class_def.h>
+#include <dhcpsrv/dhcpsrv_log.h>
+#include <dhcpsrv/host_mgr.h>
+#include <dhcpsrv/parsers/simple_parser4.h>
+#include <hooks/callout_handle.h>
+#include <hooks/hooks_manager.h>
+
+using namespace isc::db;
+using namespace isc::data;
+using namespace isc::process;
+using namespace isc::hooks;
+
+namespace {
+
+/// Structure that holds registered hook indexes.
+struct CbCtlHooks {
+ int hook_index_cb4_updated_; ///< index for "cb4_updated" hook point.
+
+ /// Constructor that registers hook points for CBControlDHCPv4.
+ CbCtlHooks() {
+ hook_index_cb4_updated_ = HooksManager::registerHook("cb4_updated");
+ }
+};
+
+// Declare a Hooks object. As this is outside any function or method, it
+// will be instantiated (and the constructor run) when the module is loaded.
+// As a result, the hook indexes will be defined before any method in this
+// module is called.
+CbCtlHooks hooks_;
+
+}; // anonymous namespace
+
+namespace isc {
+namespace dhcp {
+
+void
+CBControlDHCPv4::databaseConfigApply(const BackendSelector& backend_selector,
+ const ServerSelector& server_selector,
+ const boost::posix_time::ptime& lb_modification_time,
+ const AuditEntryCollection& audit_entries) {
+
+ auto globals_fetched = false;
+ auto reconfig = audit_entries.empty();
+ auto cb_update = !reconfig;
+ auto current_cfg = CfgMgr::instance().getCurrentCfg();
+ auto staging_cfg = CfgMgr::instance().getStagingCfg();
+
+ // Let's first delete all the configuration elements for which DELETE audit
+ // entries are found. Although, this may break chronology of the audit in
+ // some cases it should not affect the end result of the data fetch. If the
+ // object was created and then subsequently deleted, we will first try to
+ // delete this object from the local configuration (which will fail because
+ // the object does not exist) and then we will try to fetch it from the
+ // database which will return no result.
+ if (cb_update) {
+
+ auto external_cfg = CfgMgr::instance().createExternalCfg();
+
+ // Get audit entries for deleted global parameters.
+ const auto& index = audit_entries.get<AuditEntryObjectTypeTag>();
+ auto range = index.equal_range(boost::make_tuple("dhcp4_global_parameter",
+ AuditEntry::ModificationType::DELETE));
+ if (range.first != range.second) {
+ // Some globals have been deleted. Since we currently don't track database
+ // identifiers of the global parameters we have to fetch all global
+ // parameters for this server. Next, we simply replace existing
+ // global parameters with the new parameters. This is slightly
+ // inefficient but only slightly. Note that this is a single
+ // database query and the number of global parameters is small.
+ data::StampedValueCollection globals;
+ globals = getMgr().getPool()->getAllGlobalParameters4(backend_selector, server_selector);
+ addGlobalsToConfig(external_cfg, globals);
+
+ // Add defaults.
+ external_cfg->applyDefaultsConfiguredGlobals(SimpleParser4::GLOBAL4_DEFAULTS);
+
+ // Sanity check it.
+ external_cfg->sanityChecksLifetime("valid-lifetime");
+
+ // Now that we successfully fetched the new global parameters, let's
+ // remove existing ones and merge them into the current configuration.
+ current_cfg->clearConfiguredGlobals();
+ CfgMgr::instance().mergeIntoCurrentCfg(external_cfg->getSequence());
+ globals_fetched = true;
+ }
+
+ try {
+ // Get audit entries for deleted option definitions and delete each
+ // option definition from the current configuration for which the
+ // audit entry is found.
+ range = index.equal_range(boost::make_tuple("dhcp4_option_def",
+ AuditEntry::ModificationType::DELETE));
+ for (auto entry = range.first; entry != range.second; ++entry) {
+ current_cfg->getCfgOptionDef()->del((*entry)->getObjectId());
+ }
+
+ // Repeat the same for other configuration elements.
+
+ range = index.equal_range(boost::make_tuple("dhcp4_options",
+ AuditEntry::ModificationType::DELETE));
+ for (auto entry = range.first; entry != range.second; ++entry) {
+ current_cfg->getCfgOption()->del((*entry)->getObjectId());
+ }
+
+ range = index.equal_range(boost::make_tuple("dhcp4_client_class",
+ AuditEntry::ModificationType::DELETE));
+ for (auto entry = range.first; entry != range.second; ++entry) {
+ current_cfg->getClientClassDictionary()->removeClass((*entry)->getObjectId());
+ }
+
+ range = index.equal_range(boost::make_tuple("dhcp4_shared_network",
+ AuditEntry::ModificationType::DELETE));
+ for (auto entry = range.first; entry != range.second; ++entry) {
+ current_cfg->getCfgSharedNetworks4()->del((*entry)->getObjectId());
+ }
+
+ range = index.equal_range(boost::make_tuple("dhcp4_subnet",
+ AuditEntry::ModificationType::DELETE));
+ for (auto entry = range.first; entry != range.second; ++entry) {
+ // If the deleted subnet belongs to a shared network and the
+ // shared network is not being removed, we need to detach the
+ // subnet from the shared network.
+ auto subnet = current_cfg->getCfgSubnets4()->getBySubnetId((*entry)->getObjectId());
+ if (subnet) {
+ // Check if the subnet belongs to a shared network.
+ SharedNetwork4Ptr network;
+ subnet->getSharedNetwork(network);
+ if (network) {
+ // Detach the subnet from the shared network.
+ network->del(subnet->getID());
+ }
+ // Actually delete the subnet from the configuration.
+ current_cfg->getCfgSubnets4()->del((*entry)->getObjectId());
+ }
+ }
+
+ } catch (...) {
+ // Ignore errors thrown when attempting to delete a non-existing
+ // configuration entry. There is no guarantee that the deleted
+ // entry is actually there as we're not processing the audit
+ // chronologically.
+ }
+ }
+
+ // Create the external config into which we'll fetch backend config data.
+ auto external_cfg = CfgMgr::instance().createExternalCfg();
+
+ // First let's fetch the globals and add them to external config.
+ AuditEntryCollection updated_entries;
+ if (!globals_fetched) {
+ if (cb_update) {
+ updated_entries = fetchConfigElement(audit_entries, "dhcp4_global_parameter");
+ }
+ if (reconfig || !updated_entries.empty()) {
+ data::StampedValueCollection globals;
+ globals = getMgr().getPool()->getModifiedGlobalParameters4(backend_selector, server_selector,
+ lb_modification_time);
+ addGlobalsToConfig(external_cfg, globals);
+ globals_fetched = true;
+ }
+ }
+
+ // Now we fetch the option definitions and add them.
+ if (cb_update) {
+ updated_entries = fetchConfigElement(audit_entries, "dhcp4_option_def");
+ }
+ if (reconfig || !updated_entries.empty()) {
+ OptionDefContainer option_defs =
+ getMgr().getPool()->getModifiedOptionDefs4(backend_selector, server_selector,
+ lb_modification_time);
+ for (auto option_def = option_defs.begin(); option_def != option_defs.end(); ++option_def) {
+ if (!audit_entries.empty() && !hasObjectId(updated_entries, (*option_def)->getId())) {
+ continue;
+ }
+ external_cfg->getCfgOptionDef()->add(*option_def);
+ }
+ }
+
+ // Next fetch the options. They are returned as a container of OptionDescriptors.
+ if (cb_update) {
+ updated_entries = fetchConfigElement(audit_entries, "dhcp4_options");
+ }
+ if (reconfig || !updated_entries.empty()) {
+ OptionContainer options = getMgr().getPool()->getModifiedOptions4(backend_selector,
+ server_selector,
+ lb_modification_time);
+ for (auto option = options.begin(); option != options.end(); ++option) {
+ if (!audit_entries.empty() && !hasObjectId(updated_entries, (*option).getId())) {
+ continue;
+ }
+ external_cfg->getCfgOption()->add((*option), (*option).space_name_);
+ }
+ }
+
+ // Fetch client classes. They are returned in a ClientClassDictionary.
+ if (cb_update) {
+ updated_entries = fetchConfigElement(audit_entries, "dhcp4_client_class");
+ }
+ if (reconfig || !updated_entries.empty()) {
+ ClientClassDictionary client_classes = getMgr().getPool()->getAllClientClasses4(backend_selector,
+ server_selector);
+ // Match expressions are not initialized for classes returned from the config backend.
+ // We have to ensure to initialize them before they can be used by the server.
+ client_classes.initMatchExpr(AF_INET);
+
+ // Class options also need to be created when returned from the config backend.
+ client_classes.createOptions(external_cfg->getCfgOptionDef());
+
+ external_cfg->setClientClassDictionary(boost::make_shared<ClientClassDictionary>(client_classes));
+ }
+
+ // Allocator selection at the global level can affect subnets and shared networks
+ // for which the allocator hasn't been specified explicitly. Let's see if the
+ // allocator has been specified at the global level.
+ std::string global_allocator;
+ auto allocator = external_cfg->getConfiguredGlobal(CfgGlobals::ALLOCATOR);
+ if (allocator && (allocator->getType() == Element::string)) {
+ global_allocator = allocator->stringValue();
+ }
+
+ // If we're fetching the changes from the config backend we also want
+ // to see if the global allocator has changed. Let's get the currently
+ // used allocator too.
+ auto allocator_changed = false;
+ // We're only affected by the allocator change if this is the update from
+ // the configuration backend.
+ if (cb_update) {
+ auto allocator = CfgMgr::instance().getCurrentCfg()->getConfiguredGlobal(CfgGlobals::ALLOCATOR);
+ if (allocator && (allocator->getType() == Element::string)) {
+ allocator_changed = (global_allocator != allocator->stringValue());
+ }
+ }
+
+ // Now fetch the shared networks.
+ if (cb_update) {
+ updated_entries = fetchConfigElement(audit_entries, "dhcp4_shared_network");
+ }
+ SharedNetwork4Collection networks;
+ if (allocator_changed || reconfig) {
+ // A change of the allocator or the server reconfiguration can affect all
+ // shared networks. Get all shared networks.
+ networks = getMgr().getPool()->getAllSharedNetworks4(backend_selector, server_selector);
+
+ } else if (!updated_entries.empty()) {
+ // An update from the config backend when the global allocator hasn't changed
+ // means that we only need to handle the modified subnets.
+ networks = getMgr().getPool()->getModifiedSharedNetworks4(backend_selector, server_selector,
+ lb_modification_time);
+ }
+ // Iterate over all shared networks that may require reconfiguration.
+ for (auto network = networks.begin(); network != networks.end(); ++network) {
+ if (!allocator_changed && cb_update && !hasObjectId(updated_entries, (*network)->getId())) {
+ continue;
+ }
+ // In order to take advantage of the dynamic inheritance of global
+ // parameters to a shared network we need to set a callback function
+ // for each network to allow for fetching global parameters.
+ (*network)->setFetchGlobalsFn([] () -> ConstCfgGlobalsPtr {
+ return (CfgMgr::instance().getCurrentCfg()->getConfiguredGlobals());
+ });
+ (*network)->setDefaultAllocatorType(global_allocator);
+ external_cfg->getCfgSharedNetworks4()->add((*network));
+ }
+
+ // Next, fetch the subnets.
+ if (cb_update) {
+ updated_entries = fetchConfigElement(audit_entries, "dhcp4_subnet");
+ }
+ Subnet4Collection subnets;
+ if (allocator_changed || reconfig) {
+ // A change of the allocator or the server reconfiguration can affect all
+ // subnets. Get all subnets.
+ subnets = getMgr().getPool()->getAllSubnets4(backend_selector, server_selector);
+
+ } else if (!updated_entries.empty()) {
+ // An update from the config backend when the global allocator hasn't changed
+ // means that we only need to handle the modified subnets.
+ subnets = getMgr().getPool()->getModifiedSubnets4(backend_selector,
+ server_selector,
+ lb_modification_time);
+ }
+ // Iterate over all subnets that may require reconfiguration.
+ for (auto subnet = subnets.begin(); subnet != subnets.end(); ++subnet) {
+ if (!allocator_changed && cb_update && !hasObjectId(updated_entries, (*subnet)->getID())) {
+ continue;
+ }
+ // In order to take advantage of the dynamic inheritance of global
+ // parameters to a subnet we need to set a callback function for each
+ // subnet to allow for fetching global parameters.
+ (*subnet)->setFetchGlobalsFn([] () -> ConstCfgGlobalsPtr {
+ return (CfgMgr::instance().getCurrentCfg()->getConfiguredGlobals());
+ });
+ (*subnet)->setDefaultAllocatorType(global_allocator);
+ external_cfg->getCfgSubnets4()->add((*subnet));
+ }
+
+ if (reconfig) {
+ // If we're configuring the server after startup, we do not apply the
+ // ip-reservations-unique setting here. It will be applied when the
+ // configuration is committed.
+ external_cfg->sanityChecksLifetime(*staging_cfg, "valid-lifetime");
+ CfgMgr::instance().mergeIntoStagingCfg(external_cfg->getSequence());
+
+ } else {
+ if (globals_fetched) {
+ // ip-reservations-unique parameter requires special handling because
+ // setting it to false may be unsupported by some host backends.
+ bool ip_unique = true;
+ auto ip_unique_param = external_cfg->getConfiguredGlobal("ip-reservations-unique");
+ if (ip_unique_param && (ip_unique_param->getType() == Element::boolean)) {
+ ip_unique = ip_unique_param->boolValue();
+ }
+ // First try to use the new setting to configure the HostMgr because it
+ // may fail if the backend does not support it.
+ if (!HostMgr::instance().setIPReservationsUnique(ip_unique)) {
+ // The new setting is unsupported by the backend, so do not apply this
+ // setting at all.
+ LOG_WARN(dhcpsrv_logger, DHCPSRV_CFGMGR_IPV4_RESERVATIONS_NON_UNIQUE_IGNORED);
+ external_cfg->addConfiguredGlobal("ip-reservations-unique", Element::create(true));
+ }
+ }
+ external_cfg->sanityChecksLifetime(*current_cfg, "valid-lifetime");
+ CfgMgr::instance().mergeIntoCurrentCfg(external_cfg->getSequence());
+ CfgMgr::instance().getCurrentCfg()->getCfgSubnets4()->initAllocatorsAfterConfigure();
+ }
+
+ LOG_INFO(dhcpsrv_logger, DHCPSRV_CFGMGR_CONFIG4_MERGED);
+
+ if (cb_update &&
+ HooksManager::calloutsPresent(hooks_.hook_index_cb4_updated_)) {
+ CalloutHandlePtr callout_handle = HooksManager::createCalloutHandle();
+
+ // Use the RAII wrapper to make sure that the callout handle state is
+ // reset when this object goes out of scope. All hook points must do
+ // it to prevent possible circular dependency between the callout
+ // handle and its arguments.
+ ScopedCalloutHandleState callout_handle_state(callout_handle);
+
+ // Pass a shared pointer to audit entries.
+ AuditEntryCollectionPtr ptr(new AuditEntryCollection(audit_entries));
+ callout_handle->setArgument("audit_entries", ptr);
+
+ // Call the callouts
+ HooksManager::callCallouts(hooks_.hook_index_cb4_updated_, *callout_handle);
+
+ // Ignore the result.
+ }
+}
+
+} // end of namespace isc::dhcp
+} // end of namespace isc