diff options
Diffstat (limited to 'src/lib/dhcpsrv/cb_ctl_dhcp4.cc')
-rw-r--r-- | src/lib/dhcpsrv/cb_ctl_dhcp4.cc | 360 |
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 |