123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387 |
- // Copyright (C) 2012-2014 Internet Systems Consortium, Inc. ("ISC")
- //
- // Permission to use, copy, modify, and/or distribute this software for any
- // purpose with or without fee is hereby granted, provided that the above
- // copyright notice and this permission notice appear in all copies.
- //
- // THE SOFTWARE IS PROVIDED "AS IS" AND ISC DISCLAIMS ALL WARRANTIES WITH
- // REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF MERCHANTABILITY
- // AND FITNESS. IN NO EVENT SHALL ISC BE LIABLE FOR ANY SPECIAL, DIRECT,
- // INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING FROM
- // LOSS OF USE, DATA OR PROFITS, WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE
- // OR OTHER TORTIOUS ACTION, ARISING OUT OF OR IN CONNECTION WITH THE USE OR
- // PERFORMANCE OF THIS SOFTWARE.
- #include <asiolink/io_address.h>
- #include <dhcp/libdhcp++.h>
- #include <dhcpsrv/cfgmgr.h>
- #include <dhcpsrv/dhcpsrv_log.h>
- #include <string>
- using namespace isc::asiolink;
- using namespace isc::util;
- namespace isc {
- namespace dhcp {
- CfgMgr&
- CfgMgr::instance() {
- static CfgMgr cfg_mgr;
- return (cfg_mgr);
- }
- void
- CfgMgr::addOptionSpace4(const OptionSpacePtr& space) {
- if (!space) {
- isc_throw(InvalidOptionSpace,
- "provided option space object is NULL.");
- }
- OptionSpaceCollection::iterator it = spaces4_.find(space->getName());
- if (it != spaces4_.end()) {
- isc_throw(InvalidOptionSpace, "option space " << space->getName()
- << " already added.");
- }
- spaces4_.insert(make_pair(space->getName(), space));
- }
- void
- CfgMgr::addOptionSpace6(const OptionSpacePtr& space) {
- if (!space) {
- isc_throw(InvalidOptionSpace,
- "provided option space object is NULL.");
- }
- OptionSpaceCollection::iterator it = spaces6_.find(space->getName());
- if (it != spaces6_.end()) {
- isc_throw(InvalidOptionSpace, "option space " << space->getName()
- << " already added.");
- }
- spaces6_.insert(make_pair(space->getName(), space));
- }
- void
- CfgMgr::addOptionDef(const OptionDefinitionPtr& def,
- const std::string& option_space) {
- // @todo we need better validation of the provided option space name here.
- // This will be implemented when #2313 is merged.
- if (option_space.empty()) {
- isc_throw(BadValue, "option space name must not be empty");
- } else if (!def) {
- // Option definition must point to a valid object.
- isc_throw(MalformedOptionDefinition, "option definition must not be NULL");
- } else if (getOptionDef(option_space, def->getCode())) {
- // Option definition must not be overriden.
- isc_throw(DuplicateOptionDefinition, "option definition already added"
- << " to option space " << option_space);
- // We must not override standard (assigned) option for which there is a
- // definition in libdhcp++. The standard options belong to dhcp4 or dhcp6
- // option space.
- } else if ((option_space == "dhcp4" &&
- LibDHCP::isStandardOption(Option::V4, def->getCode()) &&
- LibDHCP::getOptionDef(Option::V4, def->getCode())) ||
- (option_space == "dhcp6" &&
- LibDHCP::isStandardOption(Option::V6, def->getCode()) &&
- LibDHCP::getOptionDef(Option::V6, def->getCode()))) {
- isc_throw(BadValue, "unable to override definition of option '"
- << def->getCode() << "' in standard option space '"
- << option_space << "'.");
- }
- // Actually add a new item.
- option_def_spaces_.addItem(def, option_space);
- }
- OptionDefContainerPtr
- CfgMgr::getOptionDefs(const std::string& option_space) const {
- // @todo Validate the option space once the #2313 is implemented.
- return (option_def_spaces_.getItems(option_space));
- }
- OptionDefinitionPtr
- CfgMgr::getOptionDef(const std::string& option_space,
- const uint16_t option_code) const {
- // @todo Validate the option space once the #2313 is implemented.
- // Get a reference to option definitions for a particular option space.
- OptionDefContainerPtr defs = getOptionDefs(option_space);
- // If there are no matching option definitions then return the empty pointer.
- if (!defs || defs->empty()) {
- return (OptionDefinitionPtr());
- }
- // If there are some option definitions for a particular option space
- // use an option code to get the one we want.
- const OptionDefContainerTypeIndex& idx = defs->get<1>();
- const OptionDefContainerTypeRange& range = idx.equal_range(option_code);
- // If there is no definition that matches option code, return empty pointer.
- if (std::distance(range.first, range.second) == 0) {
- return (OptionDefinitionPtr());
- }
- // If there is more than one definition matching an option code, return
- // the first one. This should not happen because we check for duplicates
- // when addOptionDef is called.
- return (*range.first);
- }
- Subnet6Ptr
- CfgMgr::getSubnet6(const std::string& iface) {
- if (!iface.length()) {
- return (Subnet6Ptr());
- }
- // If there is more than one, we need to choose the proper one
- for (Subnet6Collection::iterator subnet = subnets6_.begin();
- subnet != subnets6_.end(); ++subnet) {
- if (iface == (*subnet)->getIface()) {
- LOG_DEBUG(dhcpsrv_logger, DHCPSRV_DBG_TRACE,
- DHCPSRV_CFGMGR_SUBNET6_IFACE)
- .arg((*subnet)->toText()).arg(iface);
- return (*subnet);
- }
- }
- return (Subnet6Ptr());
- }
- Subnet6Ptr
- CfgMgr::getSubnet6(const isc::asiolink::IOAddress& hint) {
- // If there's only one subnet configured, let's just use it
- // The idea is to keep small deployments easy. In a small network - one
- // router that also runs DHCPv6 server. User specifies a single pool and
- // expects it to just work. Without this, the server would complain that it
- // doesn't have IP address on its interfaces that matches that
- // configuration. Such requirement makes sense in IPv4, but not in IPv6.
- // The server does not need to have a global address (using just link-local
- // is ok for DHCPv6 server) from the pool it serves.
- if ((subnets6_.size() == 1) && hint.isV6LinkLocal()) {
- LOG_DEBUG(dhcpsrv_logger, DHCPSRV_DBG_TRACE,
- DHCPSRV_CFGMGR_ONLY_SUBNET6)
- .arg(subnets6_[0]->toText()).arg(hint.toText());
- return (subnets6_[0]);
- }
- // If there is more than one, we need to choose the proper one
- for (Subnet6Collection::iterator subnet = subnets6_.begin();
- subnet != subnets6_.end(); ++subnet) {
- if ((*subnet)->inRange(hint)) {
- LOG_DEBUG(dhcpsrv_logger, DHCPSRV_DBG_TRACE,
- DHCPSRV_CFGMGR_SUBNET6)
- .arg((*subnet)->toText()).arg(hint.toText());
- return (*subnet);
- }
- }
- // sorry, we don't support that subnet
- LOG_DEBUG(dhcpsrv_logger, DHCPSRV_DBG_TRACE, DHCPSRV_CFGMGR_NO_SUBNET6)
- .arg(hint.toText());
- return (Subnet6Ptr());
- }
- Subnet6Ptr CfgMgr::getSubnet6(OptionPtr iface_id_option) {
- if (!iface_id_option) {
- return (Subnet6Ptr());
- }
- // Let's iterate over all subnets and for those that have interface-id
- // defined, check if the interface-id is equal to what we are looking for
- for (Subnet6Collection::iterator subnet = subnets6_.begin();
- subnet != subnets6_.end(); ++subnet) {
- if ( (*subnet)->getInterfaceId() &&
- ((*subnet)->getInterfaceId()->equal(iface_id_option))) {
- LOG_DEBUG(dhcpsrv_logger, DHCPSRV_DBG_TRACE,
- DHCPSRV_CFGMGR_SUBNET6_IFACE_ID)
- .arg((*subnet)->toText());
- return (*subnet);
- }
- }
- return (Subnet6Ptr());
- }
- void CfgMgr::addSubnet6(const Subnet6Ptr& subnet) {
- /// @todo: Check that this new subnet does not cross boundaries of any
- /// other already defined subnet.
- /// @todo: Check that there is no subnet with the same interface-id
- LOG_DEBUG(dhcpsrv_logger, DHCPSRV_DBG_TRACE, DHCPSRV_CFGMGR_ADD_SUBNET6)
- .arg(subnet->toText());
- subnets6_.push_back(subnet);
- }
- Subnet4Ptr
- CfgMgr::getSubnet4(const isc::asiolink::IOAddress& hint) {
- // If there's only one subnet configured, let's just use it
- // The idea is to keep small deployments easy. In a small network - one
- // router that also runs DHCPv6 server. Users specifies a single pool and
- // expects it to just work. Without this, the server would complain that it
- // doesn't have IP address on its interfaces that matches that
- // configuration. Such requirement makes sense in IPv4, but not in IPv6.
- // The server does not need to have a global address (using just link-local
- // is ok for DHCPv6 server) from the pool it serves.
- if (subnets4_.size() == 1) {
- LOG_DEBUG(dhcpsrv_logger, DHCPSRV_DBG_TRACE,
- DHCPSRV_CFGMGR_ONLY_SUBNET4)
- .arg(subnets4_[0]->toText()).arg(hint.toText());
- return (subnets4_[0]);
- }
- // If there is more than one, we need to choose the proper one
- for (Subnet4Collection::iterator subnet = subnets4_.begin();
- subnet != subnets4_.end(); ++subnet) {
- if ((*subnet)->inRange(hint)) {
- LOG_DEBUG(dhcpsrv_logger, DHCPSRV_DBG_TRACE,
- DHCPSRV_CFGMGR_SUBNET4)
- .arg((*subnet)->toText()).arg(hint.toText());
- return (*subnet);
- }
- }
- // sorry, we don't support that subnet
- LOG_DEBUG(dhcpsrv_logger, DHCPSRV_DBG_TRACE, DHCPSRV_CFGMGR_NO_SUBNET4)
- .arg(hint.toText());
- return (Subnet4Ptr());
- }
- void CfgMgr::addSubnet4(const Subnet4Ptr& subnet) {
- /// @todo: Check that this new subnet does not cross boundaries of any
- /// other already defined subnet.
- LOG_DEBUG(dhcpsrv_logger, DHCPSRV_DBG_TRACE, DHCPSRV_CFGMGR_ADD_SUBNET4)
- .arg(subnet->toText());
- subnets4_.push_back(subnet);
- }
- void CfgMgr::deleteOptionDefs() {
- option_def_spaces_.clearItems();
- }
- void CfgMgr::deleteSubnets4() {
- LOG_DEBUG(dhcpsrv_logger, DHCPSRV_DBG_TRACE, DHCPSRV_CFGMGR_DELETE_SUBNET4);
- subnets4_.clear();
- }
- void CfgMgr::deleteSubnets6() {
- LOG_DEBUG(dhcpsrv_logger, DHCPSRV_DBG_TRACE, DHCPSRV_CFGMGR_DELETE_SUBNET6);
- subnets6_.clear();
- }
- std::string CfgMgr::getDataDir() {
- return (datadir_);
- }
- void
- CfgMgr::addActiveIface(const std::string& iface) {
- size_t pos = iface.find("/");
- std::string iface_copy = iface;
- if (pos != std::string::npos) {
- std::string addr_string = iface.substr(pos + 1);
- try {
- IOAddress addr(addr_string);
- iface_copy = iface.substr(0,pos);
- unicast_addrs_.insert(make_pair(iface_copy, addr));
- } catch (...) {
- isc_throw(BadValue, "Can't convert '" << addr_string
- << "' into address in interface defition ('"
- << iface << "')");
- }
- }
- if (isIfaceListedActive(iface_copy)) {
- isc_throw(DuplicateListeningIface,
- "attempt to add duplicate interface '" << iface_copy << "'"
- " to the set of interfaces on which server listens");
- }
- LOG_DEBUG(dhcpsrv_logger, DHCPSRV_DBG_TRACE, DHCPSRV_CFGMGR_ADD_IFACE)
- .arg(iface_copy);
- active_ifaces_.push_back(iface_copy);
- }
- void
- CfgMgr::activateAllIfaces() {
- LOG_DEBUG(dhcpsrv_logger, DHCPSRV_DBG_TRACE,
- DHCPSRV_CFGMGR_ALL_IFACES_ACTIVE);
- all_ifaces_active_ = true;
- }
- void
- CfgMgr::deleteActiveIfaces() {
- LOG_DEBUG(dhcpsrv_logger, DHCPSRV_DBG_TRACE,
- DHCPSRV_CFGMGR_CLEAR_ACTIVE_IFACES);
- active_ifaces_.clear();
- all_ifaces_active_ = false;
- unicast_addrs_.clear();
- }
- bool
- CfgMgr::isActiveIface(const std::string& iface) const {
- // @todo Verify that the interface with the specified name is
- // present in the system.
- // If all interfaces are marked active, there is no need to check that
- // the name of this interface has been explicitly listed.
- if (all_ifaces_active_) {
- return (true);
- }
- return (isIfaceListedActive(iface));
- }
- bool
- CfgMgr::isIfaceListedActive(const std::string& iface) const {
- for (ActiveIfacesCollection::const_iterator it = active_ifaces_.begin();
- it != active_ifaces_.end(); ++it) {
- if (iface == *it) {
- return (true);
- }
- }
- return (false);
- }
- const isc::asiolink::IOAddress*
- CfgMgr::getUnicast(const std::string& iface) const {
- UnicastIfacesCollection::const_iterator addr = unicast_addrs_.find(iface);
- if (addr == unicast_addrs_.end()) {
- return (NULL);
- }
- return (&(*addr).second);
- }
- void
- CfgMgr::setD2ClientConfig(D2ClientConfigPtr& new_config) {
- d2_client_mgr_.setD2ClientConfig(new_config);
- }
- bool
- CfgMgr::ddnsEnabled() {
- return (d2_client_mgr_.ddnsEnabled());
- }
- const D2ClientConfigPtr&
- CfgMgr::getD2ClientConfig() const {
- return (d2_client_mgr_.getD2ClientConfig());
- }
- D2ClientMgr&
- CfgMgr::getD2ClientMgr() {
- return (d2_client_mgr_);
- }
- CfgMgr::CfgMgr()
- : datadir_(DHCP_DATA_DIR),
- all_ifaces_active_(false), echo_v4_client_id_(true),
- d2_client_mgr_() {
- // DHCP_DATA_DIR must be set set with -DDHCP_DATA_DIR="..." in Makefile.am
- // Note: the definition of DHCP_DATA_DIR needs to include quotation marks
- // See AM_CPPFLAGS definition in Makefile.am
- }
- CfgMgr::~CfgMgr() {
- }
- }; // end of isc::dhcp namespace
- }; // end of isc namespace
|