// Copyright (C) 2016-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 #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include using namespace isc; using namespace isc::asiolink; using namespace isc::db; using namespace isc::dhcp; using namespace isc::util; using namespace isc::data; using namespace std; namespace { /// @brief Maximum length of option value. /// The maximum size of the raw option data that may be read from the /// database. const size_t OPTION_VALUE_MAX_LEN = 4096; /// @brief Numeric value representing last supported identifier. /// /// This value is used to validate whether the identifier type stored in /// a database is within bounds. of supported identifiers. const uint8_t MAX_IDENTIFIER_TYPE = static_cast(Host::LAST_IDENTIFIER_TYPE); /// @brief Maximum length of DHCP identifier value. const size_t DHCP_IDENTIFIER_MAX_LEN = ClientId::MAX_CLIENT_ID_LEN; /// @brief This class provides mechanisms for sending and retrieving /// information from the 'hosts' table. /// /// This class is used to insert and retrieve entries from the 'hosts' table. /// The queries used with this class do not retrieve IPv6 reservations or /// options associated with a host to minimize impact on performance. Other /// classes derived from @ref PgSqlHostExchange should be used to retrieve /// information about IPv6 reservations and options. /// /// Database schema contains several unique indexes to guard against adding /// multiple hosts for the same client identifier in a single subnet and for /// adding multiple hosts with a reservation for the same IPv4 address in a /// single subnet. The exceptions that have to be taken into account are /// listed below: /// - zero or null IPv4 address indicates that there is no reservation for the /// IPv4 address for the host, /// - null subnet identifier (either IPv4 or IPv6) indicates that /// this subnet identifier must be ignored. Specifically, this is the case /// when host reservation is created for the DHCPv4 server, the IPv6 subnet id /// should be ignored. Conversely, when host reservation is created for the /// DHCPv6 server, the IPv4 subnet id should be ignored. /// NOTE! Zero is the "global" subnet id as Kea 1.5.0 /// /// To exclude those special case values, the Postgres backend uses partial /// indexes, i.e. the only values that are included in the index are those that /// are non-zero and non-null. class PgSqlHostExchange : public PgSqlExchange { private: /// @brief Column numbers for each column in the hosts table. /// These are used for both retrieving data and for looking up /// column labels for logging. Note that their numeric order /// MUST match that of the column order in the hosts table. static const size_t HOST_ID_COL = 0; static const size_t DHCP_IDENTIFIER_COL = 1; static const size_t DHCP_IDENTIFIER_TYPE_COL = 2; static const size_t DHCP4_SUBNET_ID_COL = 3; static const size_t DHCP6_SUBNET_ID_COL = 4; static const size_t IPV4_ADDRESS_COL = 5; static const size_t HOSTNAME_COL = 6; static const size_t DHCP4_CLIENT_CLASSES_COL = 7; static const size_t DHCP6_CLIENT_CLASSES_COL = 8; static const size_t USER_CONTEXT_COL = 9; static const size_t DHCP4_NEXT_SERVER_COL = 10; static const size_t DHCP4_SERVER_HOSTNAME_COL = 11; static const size_t DHCP4_BOOT_FILE_NAME_COL = 12; static const size_t AUTH_KEY_COL = 13; /// @brief Number of columns returned for SELECT queries sent by this class. static const size_t HOST_COLUMNS = 14; public: /// @brief Constructor /// /// @param additional_columns_num This value is set by the derived classes /// to indicate how many additional columns will be returned by SELECT /// queries performed by the derived class. This constructor will allocate /// resources for these columns, e.g. binding table, error indicators. PgSqlHostExchange(const size_t additional_columns_num = 0) : PgSqlExchange(HOST_COLUMNS + additional_columns_num) { // Set the column names for use by this class. This only comprises // names used by the PgSqlHostExchange class. Derived classes will // need to set names for the columns they use. Currently these are // only used for logging purposes. columns_[HOST_ID_COL] = "host_id"; columns_[DHCP_IDENTIFIER_COL] = "dhcp_identifier"; columns_[DHCP_IDENTIFIER_TYPE_COL] = "dhcp_identifier_type"; columns_[DHCP4_SUBNET_ID_COL] = "dhcp4_subnet_id"; columns_[DHCP6_SUBNET_ID_COL] = "dhcp6_subnet_id"; columns_[IPV4_ADDRESS_COL] = "ipv4_address"; columns_[HOSTNAME_COL] = "hostname"; columns_[DHCP4_CLIENT_CLASSES_COL] = "dhcp4_client_classes"; columns_[DHCP6_CLIENT_CLASSES_COL] = "dhcp6_client_classes"; columns_[USER_CONTEXT_COL] = "user_context"; columns_[DHCP4_NEXT_SERVER_COL] = "dhcp4_next_server"; columns_[DHCP4_SERVER_HOSTNAME_COL] = "dhcp4_server_hostname"; columns_[DHCP4_BOOT_FILE_NAME_COL] = "dhcp4_boot_file_name"; columns_[AUTH_KEY_COL] = "auth_key"; BOOST_STATIC_ASSERT(12 < HOST_COLUMNS); }; /// @brief Virtual destructor. virtual ~PgSqlHostExchange() { } /// @brief Reinitializes state information /// /// This function should be called in between statement executions. /// Deriving classes should invoke this method as well as be reset /// all of their own stateful values. virtual void clear() { host_.reset(); }; /// @brief Returns index of the first uninitialized column name. /// /// This method is called by the derived classes to determine which /// column indexes are available for the derived classes within a /// binding array, error array and column names. This method /// determines the first available index by searching the first /// empty value within the columns_ vector. Previously we relied on /// the fixed values set for each class, but this was hard to maintain /// when new columns were added to the SELECT queries. It required /// modifying indexes in all derived classes. /// /// Derived classes must call this method in their constructors and /// use returned value as an index for the first column used by the /// derived class and increment this value for each subsequent column. size_t findAvailColumn() const { std::vector::const_iterator empty_column = std::find(columns_.begin(), columns_.end(), std::string()); return (std::distance(columns_.begin(), empty_column)); } /// @brief Returns value of host id in the given row. /// /// This method is used to "look ahead" at the host_id in a row /// without having to call retrieveHost() HostID getHostId(const PgSqlResult& r, int row) { HostID host_id; getColumnValue(r, row, HOST_ID_COL, host_id); return (host_id); } /// @brief Populate a bind array from a host /// /// Constructs a PsqlBindArray for sending data stored in a Host object /// to the database. /// /// @param host Host object to be added to the database. /// None of the fields in the host reservation are modified - /// the host data is only read. /// @param unique_ip boolean value indicating if multiple reservations for the /// same IP address are allowed (false) or not (true). /// /// @return pointer to newly constructed bind_array containing the /// bound values extracted from host /// /// @throw DbOperationError if bind_array cannot be populated. PsqlBindArrayPtr createBindForSend(const HostPtr& host, const bool unique_ip) { if (!host) { isc_throw(BadValue, "createBindForSend:: host object is NULL"); } // Store the host to ensure bound values remain in scope host_ = host; // Bind the host data to the array PsqlBindArrayPtr bind_array(new PsqlBindArray()); try { // host_id : is auto_incremented skip it // dhcp_identifier : BYTEA NOT NULL bind_array->add(host->getIdentifier()); // dhcp_identifier_type : SMALLINT NOT NULL bind_array->add(host->getIdentifierType()); // dhcp4_subnet_id : INT NULL if (host->getIPv4SubnetID() == SUBNET_ID_UNUSED) { bind_array->addNull(); } else { bind_array->add(host->getIPv4SubnetID()); } // dhcp6_subnet_id : INT NULL if (host->getIPv6SubnetID() == SUBNET_ID_UNUSED) { bind_array->addNull(); } else { bind_array->add(host->getIPv6SubnetID()); } // ipv4_address : BIGINT NULL bind_array->add((host->getIPv4Reservation())); // hostname : VARCHAR(255) NULL bind_array->add(host->getHostname()); // dhcp4_client_classes : VARCHAR(255) NULL // Override default separator to not include space after comma. bind_array->addTempString(host->getClientClasses4().toText(",")); // dhcp6_client_classes : VARCHAR(255) NULL bind_array->addTempString(host->getClientClasses6().toText(",")); // user_context: TEXT NULL ConstElementPtr ctx = host->getContext(); if (ctx) { std::string user_context_ = ctx->str(); bind_array->addTempString(user_context_); } else { bind_array->addNull(); } // dhcp4_next_server : BIGINT NULL bind_array->add((host->getNextServer())); // dhcp4_server_hostname : VARCHAR(64) bind_array->add(host->getServerHostname()); // dhcp4_boot_file_name : VARCHAR(128) bind_array->add(host->getBootFileName()); // add auth keys std::string key = host->getKey().toText(); if (key.empty()) { bind_array->addNull(); } else { bind_array->addTempString(key); } // When checking whether the IP is unique we need to bind the IPv4 address // at the end of the query as it has additional binding for the IPv4 // address. if (unique_ip) { bind_array->add(host->getIPv4Reservation()); // ipv4_address bind_array->add(host->getIPv4SubnetID()); // subnet_id } } catch (const std::exception& ex) { host_.reset(); isc_throw(DbOperationError, "Could not create bind array from Host: " << host->getHostname() << ", reason: " << ex.what()); } return (bind_array); }; /// @brief Processes one row of data fetched from a database. /// /// The processed data must contain host id, which uniquely identifies a /// host. This method creates a host and inserts it to the hosts collection /// only if the last inserted host has a different host id. This prevents /// adding duplicated hosts to the collection, assuming that processed /// rows are primarily ordered by host id column. /// /// This method must be overridden in the derived classes to also /// retrieve IPv6 reservations and DHCP options associated with a host. /// /// @param [out] hosts Collection of hosts to which a new host created /// from the processed data should be inserted. virtual void processRowData(ConstHostCollection& hosts, const PgSqlResult& r, int row) { // Peek at the host id , so we can skip it if we already have it // This lets us avoid constructing a copy of host for each // of its sub-rows (options, etc...) HostID row_host_id = getHostId(r, row); // Add new host only if there are no hosts or the host id of the // most recently added host is different than the host id of the // currently processed host. if (hosts.empty() || row_host_id != hosts.back()->getHostId()) { HostPtr host = retrieveHost(r, row, row_host_id); hosts.push_back(host); } } /// @brief Creates a Host object from a given row in a result set. /// /// @param r result set containing one or more rows from the hosts table /// @param row index within the result set of the row to process /// @param peeked_host_id if the caller has peeked ahead at the row's /// host_id, it can be passed in here to avoid fetching it from the row /// a second time. /// /// @return HostPtr to the newly created Host object /// /// @throw DbOperationError if the host cannot be created. HostPtr retrieveHost(const PgSqlResult& r, int row, const HostID& peeked_host_id = 0) { // If the caller peeked ahead at the host_id use that, otherwise // read it from the row. HostID host_id = (peeked_host_id ? peeked_host_id : getHostId(r,row)); // dhcp_identifier : BYTEA NOT NULL uint8_t identifier_value[DHCP_IDENTIFIER_MAX_LEN]; size_t identifier_len; convertFromBytea(r, row, DHCP_IDENTIFIER_COL, identifier_value, sizeof(identifier_value), identifier_len); // dhcp_identifier_type : SMALLINT NOT NULL uint8_t type; getColumnValue(r, row, DHCP_IDENTIFIER_TYPE_COL, type); if (type > MAX_IDENTIFIER_TYPE) { isc_throw(BadValue, "invalid dhcp identifier type returned: " << static_cast(type)); } Host::IdentifierType identifier_type = static_cast(type); // dhcp4_subnet_id : INT NULL uint32_t subnet_id(SUBNET_ID_UNUSED); if (!isColumnNull(r, row, DHCP4_SUBNET_ID_COL)) { getColumnValue(r, row, DHCP4_SUBNET_ID_COL, subnet_id); } SubnetID dhcp4_subnet_id = static_cast(subnet_id); // dhcp6_subnet_id : INT NULL subnet_id = SUBNET_ID_UNUSED; if (!isColumnNull(r, row, DHCP6_SUBNET_ID_COL)) { getColumnValue(r, row, DHCP6_SUBNET_ID_COL, subnet_id); } SubnetID dhcp6_subnet_id = static_cast(subnet_id); // ipv4_address : BIGINT NULL uint32_t addr4(0); if (!isColumnNull(r, row, IPV4_ADDRESS_COL)) { getColumnValue(r, row, IPV4_ADDRESS_COL, addr4); } isc::asiolink::IOAddress ipv4_reservation(addr4); // hostname : VARCHAR(255) NULL std::string hostname; if (!isColumnNull(r, row, HOSTNAME_COL)) { getColumnValue(r, row, HOSTNAME_COL, hostname); } // dhcp4_client_classes : VARCHAR(255) NULL std::string dhcp4_client_classes; if (!isColumnNull(r, row, DHCP4_CLIENT_CLASSES_COL)) { getColumnValue(r, row, DHCP4_CLIENT_CLASSES_COL, dhcp4_client_classes); } // dhcp6_client_classes : VARCHAR(255) NULL std::string dhcp6_client_classes; if (!isColumnNull(r, row, DHCP6_CLIENT_CLASSES_COL)) { getColumnValue(r, row, DHCP6_CLIENT_CLASSES_COL, dhcp6_client_classes); } // user_context: TEXT std::string user_context; if (!isColumnNull(r, row, USER_CONTEXT_COL)) { getColumnValue(r, row, USER_CONTEXT_COL, user_context); } // dhcp4_next_server : BIGINT NULL uint32_t dhcp4_next_server_as_uint32(0); if (!isColumnNull(r, row, DHCP4_NEXT_SERVER_COL)) { getColumnValue(r, row, DHCP4_NEXT_SERVER_COL, dhcp4_next_server_as_uint32); } isc::asiolink::IOAddress dhcp4_next_server(dhcp4_next_server_as_uint32); // dhcp4_server_hostname : VARCHAR(64) std::string dhcp4_server_hostname; if (!isColumnNull(r, row, DHCP4_SERVER_HOSTNAME_COL)) { getColumnValue(r, row, DHCP4_SERVER_HOSTNAME_COL, dhcp4_server_hostname); } // dhcp4_boot_file_name : VARCHAR(128) std::string dhcp4_boot_file_name; if (!isColumnNull(r, row, DHCP4_BOOT_FILE_NAME_COL)) { getColumnValue(r, row, DHCP4_BOOT_FILE_NAME_COL, dhcp4_boot_file_name); } // auth_key : VARCHAR(16) std::string auth_key; if (!isColumnNull(r, row, AUTH_KEY_COL)) { getColumnValue(r, row, AUTH_KEY_COL, auth_key); } // Finally, attempt to create the new host. HostPtr host; try { host.reset(new Host(identifier_value, identifier_len, identifier_type, dhcp4_subnet_id, dhcp6_subnet_id, ipv4_reservation, hostname, dhcp4_client_classes, dhcp6_client_classes, dhcp4_next_server, dhcp4_server_hostname, dhcp4_boot_file_name, AuthKey(auth_key))); // Set the user context if there is one. if (!user_context.empty()) { try { ConstElementPtr ctx = Element::fromJSON(user_context); if (!ctx || (ctx->getType() != Element::map)) { isc_throw(BadValue, "user context '" << user_context << "' is not a JSON map"); } host->setContext(ctx); } catch (const isc::data::JSONError& ex) { isc_throw(BadValue, "user context '" << user_context << "' is invalid JSON: " << ex.what()); } } host->setHostId(host_id); } catch (const isc::Exception& ex) { isc_throw(DbOperationError, "Could not create host: " << ex.what()); } return(host); }; protected: /// Pointer to Host object holding information to be inserted into /// Hosts table. This is used to retain scope. HostPtr host_; }; /// @brief Extends base exchange class with ability to retrieve DHCP options /// from the 'dhcp4_options' and 'dhcp6_options' tables. /// /// This class provides means to retrieve both DHCPv4 and DHCPv6 options /// along with the host information. It is not used to retrieve IPv6 /// reservations. The following types of queries are supported: /// - SELECT ? FROM hosts LEFT JOIN dhcp4_options LEFT JOIN dhcp6_options ... /// - SELECT ? FROM hosts LEFT JOIN dhcp4_options ... /// - SELECT ? FROM hosts LEFT JOIN dhcp6_options ... class PgSqlHostWithOptionsExchange : public PgSqlHostExchange { private: /// @brief Number of columns holding DHCPv4 or DHCPv6 option information. static const size_t OPTION_COLUMNS = 8; /// @brief Receives DHCPv4 or DHCPv6 options information from the /// dhcp4_options or dhcp6_options tables respectively. /// /// The PgSqlHostWithOptionsExchange class holds two respective instances /// of this class, one for receiving DHCPv4 options, one for receiving /// DHCPv6 options. /// /// The following are the basic functions of this class: /// - bind class members to specific columns in PgSQL binding tables, /// - set DHCP options specific column names, /// - create instances of options retrieved from the database. /// /// The reason for isolating those functions in a separate C++ class is /// to prevent code duplication for handling DHCPv4 and DHCPv6 options. class OptionProcessor { public: /// @brief Constructor. /// /// @param universe V4 or V6. The type of the options' instances /// created by this class depends on this parameter. /// @param start_column Index of the first column to be used by this /// class. OptionProcessor(const Option::Universe& universe, const size_t start_column) : universe_(universe), start_column_(start_column), option_id_index_(start_column), code_index_(start_column_ + 1), value_index_(start_column_ + 2), formatted_value_index_(start_column_ + 3), space_index_(start_column_ + 4), persistent_index_(start_column_ + 5), cancelled_index_(start_column_ + 6), user_context_index_(start_column_ + 7), most_recent_option_id_(0) { } /// @brief Reinitializes state information /// /// This function should be called prior to processing a fetched /// set of options. void clear() { most_recent_option_id_ = 0; } /// @brief Creates instance of the currently processed option. /// /// This method detects if the currently processed option is a new /// instance. It makes its determination by comparing the identifier /// of the currently processed option, with the most recently processed /// option. If the current value is greater than the id of the recently /// processed option it is assumed that the processed row holds new /// option information. In such case the option instance is created and /// inserted into the configuration passed as argument. /// /// This logic is necessary to deal with result sets made from multiple /// left joins which contain duplicated data. For instance queries /// returning both v4 and v6 options for a host would generate result /// sets similar to this: /// @code /// /// row 0: host-1 v4-opt-1 v6-opt-1 /// row 1: host-1 v4-opt-1 v6-opt-2 /// row 2: host-1 v4-opt-1 v6-opt-3 /// row 4: host-1 v4-opt-2 v6-opt-1 /// row 5: host-1 v4-opt-2 v6-opt-2 /// row 6: host-1 v4-opt-2 v6-opt-3 /// row 7: host-2 v4-opt-1 v6-opt-1 /// row 8: host-2 v4-opt-2 v6-opt-1 /// : /// @endcode /// /// @param cfg Pointer to the configuration object into which new /// option instances should be inserted. /// @param r result set containing one or more rows from a dhcp /// options table. /// @param row index within the result set of the row to process void retrieveOption(const CfgOptionPtr& cfg, const PgSqlResult& r, int row) { // If the option id on this row is NULL, then there's no // option of this type (4/6) on this row to fetch, so bail. if (PgSqlExchange::isColumnNull(r, row, option_id_index_)) { return; } // option_id: INT uint64_t option_id; PgSqlExchange::getColumnValue(r, row, option_id_index_, option_id); // The row option id must be greater than id if the most recent // option because they are ordered by option id. Otherwise // we assume that we have already processed this option. if (most_recent_option_id_ >= option_id) { return; } // Remember current option id as the most recent processed one. We // will be comparing it with option ids in subsequent rows. most_recent_option_id_ = option_id; // code: SMALLINT NOT NULL uint16_t code; PgSqlExchange::getColumnValue(r, row, code_index_, code); // value: BYTEA uint8_t value[OPTION_VALUE_MAX_LEN]; size_t value_len(0); if (!isColumnNull(r, row, value_index_)) { PgSqlExchange::convertFromBytea(r, row, value_index_, value, sizeof(value), value_len); } // formatted_value: TEXT std::string formatted_value; if (!isColumnNull(r, row, formatted_value_index_)) { PgSqlExchange::getColumnValue(r, row, formatted_value_index_, formatted_value); } // space: VARCHAR(128) std::string space; if (!isColumnNull(r, row, space_index_)) { PgSqlExchange::getColumnValue(r, row, space_index_, space); } // If empty or null space provided, use a default top level space. if (space.empty()) { space = (universe_ == Option::V4 ? DHCP4_OPTION_SPACE : DHCP6_OPTION_SPACE); } // persistent: BOOL default false bool persistent; PgSqlExchange::getColumnValue(r, row, persistent_index_, persistent); // cancelled: BOOL default false bool cancelled; PgSqlExchange::getColumnValue(r, row, cancelled_index_, cancelled); // user_context: TEXT std::string user_context; if (!isColumnNull(r, row, user_context_index_)) { PgSqlExchange::getColumnValue(r, row, user_context_index_, user_context); } // Options are held in a binary or textual format in the database. // This is similar to having an option specified in a server // configuration file. Such option is converted to appropriate C++ // class, using option definition. Thus, we need to find the // option definition for this option code and option space. // If the option space is a standard DHCPv4 or DHCPv6 option space, // this is most likely a standard option, for which we have a // definition created within libdhcp++. OptionDefinitionPtr def = LibDHCP::getOptionDef(space, code); // Otherwise, we may check if this an option encapsulated within the // vendor space. if (!def && (space != DHCP4_OPTION_SPACE) && (space != DHCP6_OPTION_SPACE)) { uint32_t vendor_id = LibDHCP::optionSpaceToVendorId(space); if (vendor_id > 0) { def = LibDHCP::getVendorOptionDef(universe_, vendor_id, code); } } // In all other cases, we use runtime option definitions, which // should be also registered within the libdhcp++. if (!def) { def = LibDHCP::getRuntimeOptionDef(space, code); } // Finish with a last resort option definition. if (!def) { def = LibDHCP::getLastResortOptionDef(space, code); } OptionPtr option; if (!def) { // If no definition found, we use generic option type. OptionBuffer buf(value, value + value_len); option.reset(new Option(universe_, code, buf.begin(), buf.end())); } else { // The option value may be specified in textual or binary format // in the database. If formatted_value is empty, the binary // format is used. Depending on the format we use a different // variant of the optionFactory function. if (formatted_value.empty()) { OptionBuffer buf(value, value + value_len); option = def->optionFactory(universe_, code, buf.begin(), buf.end()); } else { // Spit the value specified in comma separated values // format. std::vector split_vec; boost::split(split_vec, formatted_value, boost::is_any_of(",")); option = def->optionFactory(universe_, code, split_vec); } } OptionDescriptor desc(option, persistent, cancelled, formatted_value); // Set the user context if there is one into the option descriptor. if (!user_context.empty()) { try { ConstElementPtr ctx = Element::fromJSON(user_context); if (!ctx || (ctx->getType() != Element::map)) { isc_throw(BadValue, "user context '" << user_context << "' is no a JSON map"); } desc.setContext(ctx); } catch (const isc::data::JSONError& ex) { isc_throw(BadValue, "user context '" << user_context << "' is invalid JSON: " << ex.what()); } } cfg->add(desc, space); } /// @brief Specify column names. /// /// @param [out] columns Reference to a vector holding names of option /// specific columns. void setColumnNames(std::vector& columns) { columns[option_id_index_] = "option_id"; columns[code_index_] = "code"; columns[value_index_] = "value"; columns[formatted_value_index_] = "formatted_value"; columns[space_index_] = "space"; columns[persistent_index_] = "persistent"; columns[cancelled_index_] = "cancelled"; columns[user_context_index_] = "user_context"; } private: /// @brief Universe: V4 or V6. Option::Universe universe_; /// @brief Index of first column used by this class. size_t start_column_; //@} /// @name Indexes of the specific columns //@{ /// @brief Option id size_t option_id_index_; /// @brief Code size_t code_index_; /// @brief Value size_t value_index_; /// @brief Formatted value size_t formatted_value_index_; /// @brief Space size_t space_index_; /// @brief Persistent size_t persistent_index_; /// @brief Cancelled size_t cancelled_index_; //@} /// @brief User context size_t user_context_index_; /// @brief Option id for last processed row. uint64_t most_recent_option_id_; }; /// @brief Pointer to the @ref OptionProcessor class. typedef boost::shared_ptr OptionProcessorPtr; public: /// @brief DHCP option types to be fetched from the database. /// /// Supported types are: /// - Only DHCPv4 options, /// - Only DHCPv6 options, /// - Both DHCPv4 and DHCPv6 options. enum FetchedOptions { DHCP4_ONLY, DHCP6_ONLY, DHCP4_AND_DHCP6 }; /// @brief Constructor. /// /// @param fetched_options Specifies if DHCPv4, DHCPv6 or both should /// be fetched from the database for a host. /// @param additional_columns_num Number of additional columns for which /// resources should be allocated, e.g. binding table, column names etc. /// This parameter should be set to a non zero value by derived classes to /// allocate resources for the columns supported by derived classes. PgSqlHostWithOptionsExchange(const FetchedOptions& fetched_options, const size_t additional_columns_num = 0) : PgSqlHostExchange(getRequiredColumnsNum(fetched_options) + additional_columns_num), opt_proc4_(), opt_proc6_() { // Create option processor for DHCPv4 options, if required. if ((fetched_options == DHCP4_ONLY) || (fetched_options == DHCP4_AND_DHCP6)) { opt_proc4_.reset(new OptionProcessor(Option::V4, findAvailColumn())); opt_proc4_->setColumnNames(columns_); } // Create option processor for DHCPv6 options, if required. if ((fetched_options == DHCP6_ONLY) || (fetched_options == DHCP4_AND_DHCP6)) { opt_proc6_.reset(new OptionProcessor(Option::V6, findAvailColumn())); opt_proc6_->setColumnNames(columns_); } } /// @brief Clears state information /// /// This function should be called in between statement executions. /// Deriving classes should invoke this method as well as be reset /// all of their own stateful values. virtual void clear() { PgSqlHostExchange::clear(); if (opt_proc4_) { opt_proc4_->clear(); } if (opt_proc6_) { opt_proc6_->clear(); } } /// @brief Processes the current row. /// /// The fetched row includes both host information and DHCP option /// information. Because the SELECT queries use one or more LEFT JOIN /// clauses, the result set may contain duplicated host or options /// entries. This method detects duplicated information and discards such /// entries. /// /// @param [out] hosts Container holding parsed hosts and options. virtual void processRowData(ConstHostCollection& hosts, const PgSqlResult& r, int row) { HostPtr current_host; if (hosts.empty()) { // Must be the first one, fetch it. current_host = retrieveHost(r, row); hosts.push_back(current_host); } else { // Peek at the host id so we can skip it if we already have // this host. This lets us avoid retrieving the host needlessly // for each of its sub-rows (options, etc...). HostID row_host_id = getHostId(r, row); current_host = boost::const_pointer_cast(hosts.back()); // if the row's host id is greater than the one we've been // working on we're starting a new host, so fetch it. if (row_host_id > current_host->getHostId()) { current_host = retrieveHost(r, row, row_host_id); hosts.push_back(current_host); } } // Parse DHCPv4 options if required to do so. if (opt_proc4_) { CfgOptionPtr cfg = current_host->getCfgOption4(); opt_proc4_->retrieveOption(cfg, r, row); } // Parse DHCPv6 options if required to do so. if (opt_proc6_) { CfgOptionPtr cfg = current_host->getCfgOption6(); opt_proc6_->retrieveOption(cfg, r, row); } } private: /// @brief Returns a number of columns required to retrieve option data. /// /// Depending if we need DHCPv4/DHCPv6 options only, or both DHCPv4 and /// DHCPv6 a different number of columns is required in the binding array. /// This method returns the number of required columns, according to the /// value of @c fetched_columns passed in the constructor. /// /// @param fetched_columns A value which specifies whether DHCPv4, DHCPv6 or /// both types of options should be retrieved. /// /// @return Number of required columns. static size_t getRequiredColumnsNum(const FetchedOptions& fetched_options) { return (fetched_options == DHCP4_AND_DHCP6 ? 2 * OPTION_COLUMNS : OPTION_COLUMNS); } /// @brief Pointer to DHCPv4 options processor. /// /// If this object is NULL, the DHCPv4 options are not fetched. OptionProcessorPtr opt_proc4_; /// @brief Pointer to DHCPv6 options processor. /// /// If this object is NULL, the DHCPv6 options are not fetched. OptionProcessorPtr opt_proc6_; }; /// @brief This class provides mechanisms for sending and retrieving /// host information, DHCPv4 options, DHCPv6 options and IPv6 reservations. /// /// This class extends the @ref PgSqlHostWithOptionsExchange class with the /// mechanisms to retrieve IPv6 reservations. This class is used in situations /// when it is desired to retrieve DHCPv6 specific information about the host /// (DHCPv6 options and reservations), or entire information about the host /// (DHCPv4 options, DHCPv6 options and reservations). The following are the /// queries used with this class: /// - SELECT ? FROM hosts LEFT JOIN dhcp4_options LEFT JOIN dhcp6_options /// LEFT JOIN ipv6_reservations ... /// - SELECT ? FROM hosts LEFT JOIN dhcp6_options LEFT JOIN ipv6_reservations .. class PgSqlHostIPv6Exchange : public PgSqlHostWithOptionsExchange { private: /// @brief Number of columns holding IPv6 reservation information. static const size_t RESERVATION_COLUMNS = 5; public: /// @brief Constructor. /// /// Apart from initializing the base class data structures it also /// initializes values representing IPv6 reservation information. PgSqlHostIPv6Exchange(const FetchedOptions& fetched_options) : PgSqlHostWithOptionsExchange(fetched_options, RESERVATION_COLUMNS), reservation_id_index_(findAvailColumn()), address_index_(reservation_id_index_ + 1), prefix_len_index_(reservation_id_index_ + 2), type_index_(reservation_id_index_ + 3), iaid_index_(reservation_id_index_ + 4), most_recent_reservation_id_(0) { // Provide names of additional columns returned by the queries. columns_[reservation_id_index_] = "reservation_id"; columns_[address_index_] = "address"; columns_[prefix_len_index_] = "prefix_len"; columns_[type_index_] = "type"; columns_[iaid_index_] = "dhcp6_iaid"; BOOST_STATIC_ASSERT(4 < RESERVATION_COLUMNS); } /// @brief Reinitializes state information /// /// This function should be called in between statement executions. /// Deriving classes should invoke this method as well as be reset /// all of their own stateful values. void clear() { PgSqlHostWithOptionsExchange::clear(); most_recent_reservation_id_ = 0; } /// @brief Returns reservation id from the row. /// /// @return Reservation id or 0 if no reservation data is fetched. uint64_t getReservationId(const PgSqlResult& r, int row) const { uint64_t resv_id = 0; if (!isColumnNull(r, row, reservation_id_index_)) { getColumnValue(r, row, reservation_id_index_, resv_id); } return (resv_id); }; /// @brief Creates IPv6 reservation from the data contained in the /// currently processed row. /// /// @return IPv6Resrv object (containing IPv6 address or prefix reservation) IPv6Resrv retrieveReservation(const PgSqlResult& r, int row) { // type: SMALLINT NOT NULL uint16_t tmp; getColumnValue(r, row, type_index_, tmp); // Convert it to IPv6 Reservation type (0 = IA_NA, 2 = IA_PD) IPv6Resrv::Type resv_type; switch (tmp) { case 0: resv_type = IPv6Resrv::TYPE_NA; break; case 2: resv_type = IPv6Resrv::TYPE_PD; break; default: isc_throw(BadValue, "invalid IPv6 reservation type returned: " << tmp << ". Only 0 or 2 are allowed."); } // address VARCHAR(39) NOT NULL isc::asiolink::IOAddress address(getIPv6Value(r, row, address_index_)); // prefix_len: SMALLINT NOT NULL uint16_t prefix_len; getColumnValue(r, row, prefix_len_index_, prefix_len); // @todo once we support populating iaid // iaid: INT // int iaid; // getColumnValue(r, row, iaid_index_, iaid); // Create the reservation. IPv6Resrv reservation(resv_type, IOAddress(address), prefix_len); return (reservation); }; /// @brief Processes one row of data fetched from a database. /// /// The processed data must contain host id, which uniquely identifies a /// host. This method creates a host and inserts it to the hosts collection /// only if the last inserted host has a different host id. This prevents /// adding duplicated hosts to the collection, assuming that processed /// rows are primarily ordered by host id column. /// /// Depending on the value of the @c fetched_options specified in the /// constructor, this method also parses options returned as a result /// of SELECT queries. /// /// For any returned row which contains IPv6 reservation information it /// checks if the reservation is not a duplicate of previously parsed /// reservation and appends the IPv6Resrv object into the host object /// if the parsed row contains new reservation information. /// /// @param [out] hosts Collection of hosts to which a new host created /// from the processed data should be inserted. /// @param r result set containing one or more rows of fetched data /// @param row index within the result set of the row to process virtual void processRowData(ConstHostCollection& hosts, const PgSqlResult& r, int row) { // Call parent class to fetch host information and options. PgSqlHostWithOptionsExchange::processRowData(hosts, r, row); // Shouldn't happen but just in case if (hosts.empty()) { isc_throw(Unexpected, "no host information while retrieving" " IPv6 reservation"); } // If we have reservation id we haven't seen yet, retrieve the // the reservation, adding it to the current host uint64_t reservation_id = getReservationId(r, row); if (reservation_id && (reservation_id > most_recent_reservation_id_)) { HostPtr host = boost::const_pointer_cast(hosts.back()); host->addReservation(retrieveReservation(r, row)); most_recent_reservation_id_ = reservation_id; } } private: /// @name Indexes of columns holding information about IPv6 reservations. //@{ /// @brief Index of reservation_id column. size_t reservation_id_index_; /// @brief Index of address column. size_t address_index_; /// @brief Index of prefix_len column. size_t prefix_len_index_; /// @brief Index of type column. size_t type_index_; /// @brief Index of IAID column. size_t iaid_index_; //@} /// @brief Reservation id for last processed row. uint64_t most_recent_reservation_id_; }; /// @brief This class is used for storing IPv6 reservations in a PgSQL database. /// /// This class is only used to insert IPv6 reservations into the /// ipv6_reservations table. It is not used to retrieve IPv6 reservations. To /// retrieve IPv6 reservation the @ref PgSqlIPv6HostExchange class should be /// used instead. /// /// When a new IPv6 reservation is inserted into the database, an appropriate /// host must be defined in the hosts table. An attempt to insert IPv6 /// reservation for non-existing host will result in failure. class PgSqlIPv6ReservationExchange : public PgSqlExchange { private: /// @brief Set number of columns for ipv6_reservation table. static const size_t RESRV_COLUMNS = 6; public: /// @brief Constructor /// /// Initialize class members representing a single IPv6 reservation. PgSqlIPv6ReservationExchange() : PgSqlExchange(RESRV_COLUMNS), resv_(IPv6Resrv::TYPE_NA, asiolink::IOAddress("::"), 128) { // Set the column names (for error messages) columns_[0] = "host_id"; columns_[1] = "address"; columns_[2] = "prefix_len"; columns_[3] = "type"; columns_[4] = "dhcp6_iaid"; BOOST_STATIC_ASSERT(5 < RESRV_COLUMNS); } /// @brief Populate a bind array representing an IPv6 reservation /// /// Constructs a PsqlBindArray for an IPv6 reservation to the database. /// /// @param resv The IPv6 reservation to be added to the database. /// None of the fields in the reservation are modified - /// @param host_id ID of the host to which this reservation belongs. /// @param unique_ip boolean value indicating if multiple reservations for the /// same IP address are allowed (false) or not (true). /// /// @return pointer to newly constructed bind_array containing the /// bound values extracted the IPv6 reservation /// /// @throw DbOperationError if bind_array cannot be populated. PsqlBindArrayPtr createBindForSend(const IPv6Resrv& resv, const HostID& host_id, const bool unique_ip) { // Store the values to ensure they remain valid. // Technically we don't need this, as currently all the values // are converted to strings and stored by the bind array. resv_ = resv; PsqlBindArrayPtr bind_array(new PsqlBindArray()); try { // address VARCHAR(39) NOT NULL bind_array->add(resv.getPrefix()); // prefix_len: SMALLINT NOT NULL bind_array->add(resv.getPrefixLen()); // type: SMALLINT NOT NULL // See lease6_types table for values (0 = IA_NA, 2 = IA_PD) uint16_t type = resv.getType() == IPv6Resrv::TYPE_NA ? 0 : 2; bind_array->add(type); // dhcp6_iaid: INT UNSIGNED /// @todo: We don't support iaid in the IPv6Resrv yet. bind_array->addNull(); // host_id: BIGINT NOT NULL bind_array->add(host_id); // When checking whether the IP is unique we need to bind the IPv6 address // and prefix length at the end of the query as it has additional binding // for the IPv6 address and prefix length. if (unique_ip) { bind_array->add(resv.getPrefix()); // address bind_array->add(resv.getPrefixLen()); // prefix_len } } catch (const std::exception& ex) { isc_throw(DbOperationError, "Could not create bind array from IPv6 Reservation: " << resv_.toText() << ", reason: " << ex.what()); } return (bind_array); } private: /// @brief Object holding reservation being sent to the database. IPv6Resrv resv_; }; /// @brief This class is used for inserting options into a database. /// /// This class supports inserting both DHCPv4 and DHCPv6 options. class PgSqlOptionExchange : public PgSqlExchange { private: static const size_t OPTION_ID_COL = 0; static const size_t CODE_COL = 1; static const size_t VALUE_COL = 2; static const size_t FORMATTED_VALUE_COL = 3; static const size_t SPACE_COL = 4; static const size_t PERSISTENT_COL = 5; static const size_t CANCELLED_COL = 6; static const size_t USER_CONTEXT_COL = 7; static const size_t DHCP_SUBNET_ID_COL = 8; static const size_t HOST_ID_COL = 9; /// @brief Number of columns in the option tables holding bindable values. static const size_t OPTION_COLUMNS = 10; public: /// @brief Constructor. PgSqlOptionExchange() : PgSqlExchange(OPTION_COLUMNS), value_(), value_len_(0), option_() { columns_[OPTION_ID_COL] = "option_id"; columns_[CODE_COL] = "code"; columns_[VALUE_COL] = "value"; columns_[FORMATTED_VALUE_COL] = "formatted_value"; columns_[SPACE_COL] = "space"; columns_[PERSISTENT_COL] = "persistent"; columns_[CANCELLED_COL] = "cancelled"; columns_[USER_CONTEXT_COL] = "user_context"; columns_[DHCP_SUBNET_ID_COL] = "dhcp_subnet_id"; columns_[HOST_ID_COL] = "host_id"; BOOST_STATIC_ASSERT(10 <= OPTION_COLUMNS); } /// @brief Creates binding array to insert option data into database. /// /// @param opt_desc option descriptor of the option to write /// @param opt_space name of the option space to which the option belongs /// @param host_id host id of the host to which the option belongs /// /// @return pointer to newly constructed bind_array containing the /// bound values extracted from host PsqlBindArrayPtr createBindForSend(const OptionDescriptor& opt_desc, const std::string& opt_space, const HostID& host_id) { // Hold pointer to the option to make sure it remains valid until // we complete a query. option_ = opt_desc.option_; // Create the bind-array PsqlBindArrayPtr bind_array(new PsqlBindArray()); try { // option_id: is auto_incremented so skip it // code: SMALLINT UNSIGNED NOT NULL bind_array->add(option_->getType()); // value: BYTEA NULL if (opt_desc.formatted_value_.empty() && (opt_desc.option_->len() > opt_desc.option_->getHeaderLen())) { // The formatted_value is empty and the option value is // non-empty so we need to prepare on-wire format for the // option and store it in the database as a BYTEA. OutputBuffer buf(opt_desc.option_->len()); opt_desc.option_->pack(buf); const char* buf_ptr = static_cast(buf.getData()); value_.assign(buf_ptr + opt_desc.option_->getHeaderLen(), buf_ptr + buf.getLength()); value_len_ = value_.size(); bind_array->add(value_); } else { // No value or formatted_value specified. In this case, the // value BYTEA should be NULL. bind_array->addNull(PsqlBindArray::BINARY_FMT); } // formatted_value: TEXT NULL, if (!opt_desc.formatted_value_.empty()) { bind_array->addTempString(opt_desc.formatted_value_); } else { bind_array->addNull(); } // space: VARCHAR(128) NULL if (!opt_space.empty()) { bind_array->addTempString(opt_space); } else { bind_array->addNull(); } // persistent: BOOLEAN DEFAULT false bind_array->add(opt_desc.persistent_); // cancelled: BOOLEAN DEFAULT false bind_array->add(opt_desc.cancelled_); // user_context: TEXT NULL, ConstElementPtr ctx = opt_desc.getContext(); if (ctx) { std::string user_context_ = ctx->str(); bind_array->addTempString(user_context_); } else { bind_array->addNull(); } // host_id: INT NULL if (!host_id) { isc_throw(BadValue, "host_id cannot be null"); } bind_array->add(host_id); } catch (const std::exception& ex) { isc_throw(DbOperationError, "Could not create bind array for inserting DHCP " "host option: " << option_->toText() << ", reason: " << ex.what()); } return (bind_array); } private: /// @brief Option value as binary. std::vector value_; /// @brief Option value length. size_t value_len_; /// @brief Pointer to currently parsed option. OptionPtr option_; }; } // namespace namespace isc { namespace dhcp { /// @brief PostgreSQL Host Context /// /// This class stores the thread context for the manager pool. /// The class is needed by all get/update/delete functions which must use one /// or more exchanges to perform database operations. /// Each context provides a set of such exchanges for each thread. /// The context instances are lazy initialized by the requesting thread by using /// the manager's createContext function and are destroyed when the manager's /// pool instance is destroyed. class PgSqlHostContext { public: /// @brief Constructor /// /// @param parameters See PgSqlHostMgr constructor. /// @param io_service_accessor The IOService accessor function. /// @param db_reconnect_callback The connection recovery callback. PgSqlHostContext(const DatabaseConnection::ParameterMap& parameters, IOServiceAccessorPtr io_service_accessor, db::DbCallback db_reconnect_callback); /// The exchange objects are used for transfer of data to/from the database. /// They are pointed-to objects as the contents may change in "const" calls, /// while the rest of this object does not. (At alternative would be to /// declare them as "mutable".) /// @brief Pointer to the object representing an exchange which /// can be used to retrieve hosts and DHCPv4 options. boost::shared_ptr host_ipv4_exchange_; /// @brief Pointer to an object representing an exchange which can /// be used to retrieve hosts, DHCPv6 options and IPv6 reservations. boost::shared_ptr host_ipv6_exchange_; /// @brief Pointer to an object representing an exchange which can /// be used to retrieve hosts, DHCPv4 and DHCPv6 options, and /// IPv6 reservations using a single query. boost::shared_ptr host_ipv46_exchange_; /// @brief Pointer to an object representing an exchange which can /// be used to insert new IPv6 reservation. boost::shared_ptr host_ipv6_reservation_exchange_; /// @brief Pointer to an object representing an exchange which can /// be used to insert DHCPv4 or DHCPv6 option into dhcp4_options /// or dhcp6_options table. boost::shared_ptr host_option_exchange_; /// @brief PostgreSQL connection PgSqlConnection conn_; /// @brief Indicates if the database is opened in read only mode. bool is_readonly_; }; /// @brief PostgreSQL Host Context Pool /// /// This class provides a pool of contexts. /// The manager will use this class to handle available contexts. /// There is only one ContextPool per manager per back-end, which is created /// and destroyed by the respective manager factory class. class PgSqlHostContextPool { public: /// @brief The vector of available contexts. std::vector pool_; /// @brief The mutex to protect pool access. std::mutex mutex_; }; /// @brief Type of pointers to context pools. typedef boost::shared_ptr PgSqlHostContextPoolPtr; /// @brief Implementation of the @ref PgSqlHostDataSource. class PgSqlHostDataSourceImpl { public: /// @brief Statement Tags /// /// The contents of the enum are indexes into the list of SQL statements. /// It is assumed that the order is such that the indices of statements /// reading the database are less than those of statements modifying the /// database. /// @note: please add new statements doing read only operations before /// the WRITE_STMTS_BEGIN position. enum StatementIndex { GET_HOST_DHCPID, // Gets hosts by host identifier GET_HOST_ADDR, // Gets hosts by IPv4 address GET_HOST_SUBID4_DHCPID, // Gets host by IPv4 SubnetID, HW address/DUID GET_HOST_SUBID6_DHCPID, // Gets host by IPv6 SubnetID, HW address/DUID GET_HOST_SUBID_ADDR, // Gets host by IPv4 SubnetID and IPv4 address GET_HOST_PREFIX, // Gets host by IPv6 prefix GET_HOST_SUBID6_ADDR, // Gets host by IPv6 SubnetID and IPv6 prefix GET_HOST_ADDR6, // Gets hosts by IPv6 address/prefix GET_HOST_SUBID4, // Gets hosts by IPv4 SubnetID GET_HOST_SUBID6, // Gets hosts by IPv6 SubnetID GET_HOST_HOSTNAME, // Gets hosts by hostname GET_HOST_HOSTNAME_SUBID4, // Gets hosts by hostname and IPv4 SubnetID GET_HOST_HOSTNAME_SUBID6, // Gets hosts by hostname and IPv6 SubnetID GET_HOST_SUBID4_PAGE, // Gets hosts by IPv4 SubnetID beginning by HID GET_HOST_SUBID6_PAGE, // Gets hosts by IPv6 SubnetID beginning by HID GET_HOST_PAGE4, // Gets v4 hosts beginning by HID GET_HOST_PAGE6, // Gets v6 hosts beginning by HID INSERT_HOST_NON_UNIQUE_IP, // Insert new host to collection with allowing IP duplicates INSERT_HOST_UNIQUE_IP, // Insert new host to collection with checking for IP duplicates INSERT_V6_RESRV_NON_UNIQUE,// Insert v6 reservation without checking that it is unique INSERT_V6_RESRV_UNIQUE, // Insert v6 reservation with checking that it is unique INSERT_V4_HOST_OPTION, // Insert DHCPv4 option INSERT_V6_HOST_OPTION, // Insert DHCPv6 option DEL_HOST_ADDR4, // Delete v4 host (subnet-id, addr4) DEL_HOST_ADDR6, // Delete v6 host (subnet-id, addr6) DEL_HOST_SUBID4_ID, // Delete v4 host (subnet-id, ident.type, identifier) DEL_HOST_SUBID6_ID, // Delete v6 host (subnet-id, ident.type, identifier) NUM_STATEMENTS // Number of statements }; /// @brief Index of first statement performing write to the database. /// /// This value is used to mark border line between queries and other /// statements and statements performing write operation on the database, /// such as INSERT, DELETE, UPDATE. static const StatementIndex WRITE_STMTS_BEGIN = INSERT_HOST_NON_UNIQUE_IP; /// @brief Constructor. /// /// This constructor opens database connection and initializes prepared /// statements used in the queries. PgSqlHostDataSourceImpl(const DatabaseConnection::ParameterMap& parameters); /// @brief Destructor. ~PgSqlHostDataSourceImpl(); /// @brief Attempts to reconnect the server to the host DB backend manager. /// /// This is a self-rescheduling function that attempts to reconnect to the /// server's host DB backends after connectivity to one or more have been /// lost. Upon entry it will attempt to reconnect via /// @ref HostDataSourceFactory::add. /// If this is successful, DHCP servicing is re-enabled and server returns /// to normal operation. /// /// If reconnection fails and the maximum number of retries has not been /// exhausted, it will schedule a call to itself to occur at the /// configured retry interval. DHCP service remains disabled. /// /// If the maximum number of retries has been exhausted an error is logged /// and the server shuts down. /// /// This function is passed to the connection recovery mechanism. It will be /// invoked when a connection loss is detected. /// /// @param db_reconnect_ctl pointer to the ReconnectCtl containing the /// configured reconnect parameters. /// @return true if connection has been recovered, false otherwise. static bool dbReconnect(ReconnectCtlPtr db_reconnect_ctl); /// @brief Create a new context. /// /// The database is opened with all the SQL commands pre-compiled. /// /// @return A new (never null) context. /// /// @throw isc::dhcp::NoDatabaseName Mandatory database name not given. /// @throw isc::db::DbOperationError An operation on the open database has /// failed. PgSqlHostContextPtr createContext() const; /// @brief Executes statements which insert a row into one of the tables. /// /// @param ctx Context /// @param stindex Index of a statement being executed. /// @param bind Vector of PgsqlBindArray objects to be used for the query /// @param return_last_id flag indicating whether or not the insert /// returns the primary key of from the row inserted via " RETURNING /// as pid" clause on the INSERT statement. The RETURNING /// clause causes the INSERT to return a result set that should consist /// of a single row with one column, the value of the primary key. /// Defaults to false. /// /// @return 0 if return_last_id is false, otherwise it returns the /// the value in the result set in the first col of the first row. /// /// @throw isc::db::DuplicateEntry Database throws duplicate entry error uint64_t addStatement(PgSqlHostContextPtr& ctx, PgSqlHostDataSourceImpl::StatementIndex stindex, PsqlBindArrayPtr& bind, const bool return_last_id = false); /// @brief Executes statements that delete records. /// /// @param ctx Context /// @param stindex Index of a statement being executed. /// @param bind pointer to PsqlBindArray objects to be used for the query /// /// @return true if any records were deleted, false otherwise bool delStatement(PgSqlHostContextPtr& ctx, PgSqlHostDataSourceImpl::StatementIndex stindex, PsqlBindArrayPtr& bind); /// @brief Inserts IPv6 Reservation into ipv6_reservation table. /// /// @param ctx Context /// @param resv IPv6 Reservation to be added /// @param id ID of a host owning this reservation void addResv(PgSqlHostContextPtr& ctx, const IPv6Resrv& resv, const HostID& id); /// @brief Inserts a single DHCP option into the database. /// /// @param ctx Context /// @param stindex Index of a statement being executed. /// @param opt_desc Option descriptor holding information about an option /// to be inserted into the database. /// @param opt_space Option space name. /// @param subnet_id Subnet identifier. /// @param host_id Host identifier. void addOption(PgSqlHostContextPtr& ctx, const PgSqlHostDataSourceImpl::StatementIndex& stindex, const OptionDescriptor& opt_desc, const std::string& opt_space, const Optional& subnet_id, const HostID& host_id); /// @brief Inserts multiple options into the database. /// /// @param ctx Context /// @param stindex Index of a statement being executed. /// @param options_cfg An object holding a collection of options to be /// inserted into the database. /// @param host_id Host identifier retrieved using getColumnValue /// in addStatement method void addOptions(PgSqlHostContextPtr& ctx, const StatementIndex& stindex, const ConstCfgOptionPtr& options_cfg, const uint64_t host_id); /// @brief Creates collection of @ref Host objects with associated /// information such as IPv6 reservations and/or DHCP options. /// /// This method performs a query which returns host information from /// the 'hosts' table. The query may also use LEFT JOIN clause to /// retrieve information from other tables, e.g. ipv6_reservations, /// dhcp4_options and dhcp6_options. /// Whether IPv6 reservations and/or options are assigned to the /// @ref Host objects depends on the type of the exchange object. /// /// @param ctx Context /// @param stindex Statement index. /// @param bind Pointer to an array of PgSQL bindings. /// @param exchange Pointer to the exchange object used for the /// particular query. /// @param [out] result Reference to the collection of hosts returned. /// @param single A boolean value indicating if a single host is /// expected to be returned, or multiple hosts. void getHostCollection(PgSqlHostContextPtr& ctx, StatementIndex stindex, PsqlBindArrayPtr bind, boost::shared_ptr exchange, ConstHostCollection& result, bool single) const; /// @brief Retrieves a host by subnet and client's unique identifier. /// /// This method is used by both PgSqlHostDataSource::get4 and /// PgSqlHostDataSource::get6 methods. /// /// @param ctx Context /// @param subnet_id Subnet identifier. /// @param identifier_type Identifier type. /// @param identifier_begin Pointer to a beginning of a buffer containing /// an identifier. /// @param identifier_len Identifier length. /// @param stindex Statement index. /// @param exchange Pointer to the exchange object used for the /// particular query. /// /// @return Pointer to const instance of Host or null pointer if /// no host found. ConstHostPtr getHost(PgSqlHostContextPtr& ctx, const SubnetID& subnet_id, const Host::IdentifierType& identifier_type, const uint8_t* identifier_begin, const size_t identifier_len, StatementIndex stindex, boost::shared_ptr exchange) const; /// @brief Throws exception if database is read only. /// /// This method should be called by the methods which write to the /// database. If the backend is operating in read-only mode this /// method will throw exception. /// /// @param ctx Context /// /// @throw DbReadOnly if backend is operating in read only mode. void checkReadOnly(PgSqlHostContextPtr& ctx) const; /// @brief Returns PostgreSQL schema version of the open database /// /// @return Version number stored in the database, as a pair of unsigned /// integers. "first" is the major version number, "second" the /// minor number. /// /// @throw isc::db::DbOperationError An operation on the open database /// has failed. std::pair getVersion() const; /// @brief The parameters DatabaseConnection::ParameterMap parameters_; /// @brief Holds the setting whether the IP reservations must be unique or /// may be non-unique. bool ip_reservations_unique_; /// @brief The pool of contexts PgSqlHostContextPoolPtr pool_; /// @brief Indicates if there is at least one connection that can no longer /// be used for normal operations. bool unusable_; /// @brief Timer name used to register database reconnect timer. std::string timer_name_; }; namespace { /// @brief Array of tagged statements. typedef boost::array TaggedStatementArray; /// @brief Prepared PosgreSQL statements used by the backend to insert and /// retrieve reservation data from the database. TaggedStatementArray tagged_statements = { { // PgSqlHostDataSourceImpl::GET_HOST_DHCPID // Retrieves host information, IPv6 reservations and both DHCPv4 and // DHCPv6 options associated with the host. The LEFT JOIN clause is used // to retrieve information from 4 different tables using a single query. // Hence, this query returns multiple rows for a single host. {2, { OID_BYTEA, OID_INT2 }, "get_host_dhcpid", "SELECT h.host_id, h.dhcp_identifier, h.dhcp_identifier_type, " " h.dhcp4_subnet_id, h.dhcp6_subnet_id, h.ipv4_address, " " h.hostname, h.dhcp4_client_classes, h.dhcp6_client_classes, " " h.user_context, " " h.dhcp4_next_server, h.dhcp4_server_hostname, " " h.dhcp4_boot_file_name, h.auth_key, " " o4.option_id, o4.code, o4.value, o4.formatted_value, o4.space, " " o4.persistent, o4.cancelled, o4.user_context, " " o6.option_id, o6.code, o6.value, o6.formatted_value, o6.space, " " o6.persistent, o6.cancelled, o6.user_context, " " r.reservation_id, host(r.address), r.prefix_len, r.type, r.dhcp6_iaid " "FROM hosts AS h " "LEFT JOIN dhcp4_options AS o4 ON h.host_id = o4.host_id " "LEFT JOIN dhcp6_options AS o6 ON h.host_id = o6.host_id " "LEFT JOIN ipv6_reservations AS r ON h.host_id = r.host_id " "WHERE dhcp_identifier = $1 AND dhcp_identifier_type = $2 " "ORDER BY h.host_id, o4.option_id, o6.option_id, r.reservation_id" }, // PgSqlHostDataSourceImpl::GET_HOST_ADDR // Retrieves host information along with the DHCPv4 options associated with // it. Left joining the dhcp4_options table results in multiple rows being // returned for the same host. The host is retrieved by IPv4 address. {1, { OID_INT8 }, "get_host_addr", "SELECT h.host_id, h.dhcp_identifier, h.dhcp_identifier_type, " " h.dhcp4_subnet_id, h.dhcp6_subnet_id, h.ipv4_address, h.hostname, " " h.dhcp4_client_classes, h.dhcp6_client_classes, h.user_context, " " h.dhcp4_next_server, h.dhcp4_server_hostname, " " h.dhcp4_boot_file_name, h.auth_key, " " o.option_id, o.code, o.value, o.formatted_value, o.space, " " o.persistent, o.cancelled, o.user_context " "FROM hosts AS h " "LEFT JOIN dhcp4_options AS o ON h.host_id = o.host_id " "WHERE ipv4_address = $1 " "ORDER BY h.host_id, o.option_id" }, // PgSqlHostDataSourceImpl::GET_HOST_SUBID4_DHCPID // Retrieves host information and DHCPv4 options using subnet identifier // and client's identifier. Left joining the dhcp4_options table results in // multiple rows being returned for the same host. {3, { OID_INT8, OID_INT2, OID_BYTEA }, "get_host_subid4_dhcpid", "SELECT h.host_id, h.dhcp_identifier, h.dhcp_identifier_type, " " h.dhcp4_subnet_id, h.dhcp6_subnet_id, h.ipv4_address, h.hostname, " " h.dhcp4_client_classes, h.dhcp6_client_classes, h.user_context, " " h.dhcp4_next_server, h.dhcp4_server_hostname, " " h.dhcp4_boot_file_name, h.auth_key, " " o.option_id, o.code, o.value, o.formatted_value, o.space, " " o.persistent, o.cancelled, o.user_context " "FROM hosts AS h " "LEFT JOIN dhcp4_options AS o ON h.host_id = o.host_id " "WHERE h.dhcp4_subnet_id = $1 AND h.dhcp_identifier_type = $2 " " AND h.dhcp_identifier = $3 " "ORDER BY h.host_id, o.option_id" }, // PgSqlHostDataSourceImpl::GET_HOST_SUBID6_DHCPID // Retrieves host information, IPv6 reservations and DHCPv6 options // associated with a host. The number of rows returned is a multiplication // of number of IPv6 reservations and DHCPv6 options. {3, { OID_INT8, OID_INT2, OID_BYTEA }, "get_host_subid6_dhcpid", "SELECT h.host_id, h.dhcp_identifier, " " h.dhcp_identifier_type, h.dhcp4_subnet_id, " " h.dhcp6_subnet_id, h.ipv4_address, h.hostname, " " h.dhcp4_client_classes, h.dhcp6_client_classes, h.user_context, " " h.dhcp4_next_server, h.dhcp4_server_hostname, " " h.dhcp4_boot_file_name, h.auth_key, " " o.option_id, o.code, o.value, o.formatted_value, o.space, " " o.persistent, o.cancelled, o.user_context, " " r.reservation_id, host(r.address), r.prefix_len, r.type, r.dhcp6_iaid " "FROM hosts AS h " "LEFT JOIN dhcp6_options AS o ON h.host_id = o.host_id " "LEFT JOIN ipv6_reservations AS r ON h.host_id = r.host_id " "WHERE h.dhcp6_subnet_id = $1 AND h.dhcp_identifier_type = $2 " " AND h.dhcp_identifier = $3 " "ORDER BY h.host_id, o.option_id, r.reservation_id" }, // PgSqlHostDataSourceImpl::GET_HOST_SUBID_ADDR // Retrieves host information and DHCPv4 options for the host using subnet // identifier and IPv4 reservation. Left joining the dhcp4_options table // results in multiple rows being returned for the host. The number of // rows depends on the number of options defined for the host. {2, { OID_INT8, OID_INT8 }, "get_host_subid_addr", "SELECT h.host_id, h.dhcp_identifier, h.dhcp_identifier_type, " " h.dhcp4_subnet_id, h.dhcp6_subnet_id, h.ipv4_address, h.hostname, " " h.dhcp4_client_classes, h.dhcp6_client_classes, h.user_context, " " h.dhcp4_next_server, h.dhcp4_server_hostname, " " h.dhcp4_boot_file_name, h.auth_key, " " o.option_id, o.code, o.value, o.formatted_value, o.space, " " o.persistent, o.cancelled, o.user_context " "FROM hosts AS h " "LEFT JOIN dhcp4_options AS o ON h.host_id = o.host_id " "WHERE h.dhcp4_subnet_id = $1 AND h.ipv4_address = $2 " "ORDER BY h.host_id, o.option_id" }, // PgSqlHostDataSourceImpl::GET_HOST_PREFIX // Retrieves host information, IPv6 reservations and DHCPv6 options // associated with a host using prefix and prefix length. This query // returns host information for a single host. However, multiple rows // are returned due to left joining IPv6 reservations and DHCPv6 options. // The number of rows returned is multiplication of number of existing // IPv6 reservations and DHCPv6 options. {2, { OID_VARCHAR, OID_INT2 }, "get_host_prefix", "SELECT h.host_id, h.dhcp_identifier, " " h.dhcp_identifier_type, h.dhcp4_subnet_id, " " h.dhcp6_subnet_id, h.ipv4_address, h.hostname, " " h.dhcp4_client_classes, h.dhcp6_client_classes, h.user_context, " " h.dhcp4_next_server, h.dhcp4_server_hostname, " " h.dhcp4_boot_file_name, h.auth_key, " " o.option_id, o.code, o.value, o.formatted_value, o.space, " " o.persistent, o.cancelled, o.user_context, " " r.reservation_id, host(r.address), r.prefix_len, r.type, " " r.dhcp6_iaid " "FROM hosts AS h " "LEFT JOIN dhcp6_options AS o ON h.host_id = o.host_id " "LEFT JOIN ipv6_reservations AS r ON h.host_id = r.host_id " "WHERE h.host_id = " " (SELECT host_id FROM ipv6_reservations " " WHERE address = cast($1 as inet) AND prefix_len = $2) " "ORDER BY h.host_id, o.option_id, r.reservation_id" }, // PgSqlHostDataSourceImpl::GET_HOST_SUBID6_ADDR // Retrieves host information, IPv6 reservations and DHCPv6 options // associated with a host using IPv6 subnet id and prefix. This query // returns host information for a single host. However, multiple rows // are returned due to left joining IPv6 reservations and DHCPv6 options. // The number of rows returned is multiplication of number of existing // IPv6 reservations and DHCPv6 options. {2, { OID_INT8, OID_VARCHAR }, "get_host_subid6_addr", "SELECT h.host_id, h.dhcp_identifier, " " h.dhcp_identifier_type, h.dhcp4_subnet_id, " " h.dhcp6_subnet_id, h.ipv4_address, h.hostname, " " h.dhcp4_client_classes, h.dhcp6_client_classes, h.user_context, " " h.dhcp4_next_server, h.dhcp4_server_hostname, " " h.dhcp4_boot_file_name, h.auth_key, " " o.option_id, o.code, o.value, o.formatted_value, o.space, " " o.persistent, o.cancelled, o.user_context, " " r.reservation_id, host(r.address), r.prefix_len, r.type, " " r.dhcp6_iaid " "FROM hosts AS h " "LEFT JOIN dhcp6_options AS o ON h.host_id = o.host_id " "LEFT JOIN ipv6_reservations AS r ON h.host_id = r.host_id " "WHERE h.dhcp6_subnet_id = $1 AND h.host_id IN " " (SELECT host_id FROM ipv6_reservations " " WHERE address = cast($2 as inet)) " "ORDER BY h.host_id, o.option_id, r.reservation_id" }, // PgSqlHostDataSourceImpl::GET_HOST_ADDR6 // Retrieves host information, IPv6 reservations and DHCPv6 options // associated with a host using IPv6 address/prefix. This query // may return host information for one or more host reservations. Even // if only one host is found, multiple rows // are returned due to left joining IPv6 reservations and DHCPv6 options. // The number of rows returned is multiplication of number of existing // IPv6 reservations and DHCPv6 options. {1, { OID_VARCHAR }, "get_host_addr6", "SELECT h.host_id, h.dhcp_identifier, " " h.dhcp_identifier_type, h.dhcp4_subnet_id, " " h.dhcp6_subnet_id, h.ipv4_address, h.hostname, " " h.dhcp4_client_classes, h.dhcp6_client_classes, h.user_context, " " h.dhcp4_next_server, h.dhcp4_server_hostname, " " h.dhcp4_boot_file_name, h.auth_key, " " o.option_id, o.code, o.value, o.formatted_value, o.space, " " o.persistent, o.cancelled, o.user_context, " " r.reservation_id, r.address, r.prefix_len, r.type, " " r.dhcp6_iaid " "FROM hosts AS h " "LEFT JOIN dhcp6_options AS o ON h.host_id = o.host_id " "LEFT JOIN ipv6_reservations AS r ON h.host_id = r.host_id " "WHERE h.host_id IN " " (SELECT host_id FROM ipv6_reservations " " WHERE address = cast($1 as inet)) " "ORDER BY h.host_id, o.option_id, r.reservation_id" }, // PgSqlHostDataSourceImpl::GET_HOST_SUBID4 // // Retrieves host information for all hosts in a subnet, along with the // DHCPv4 options associated with it. Left joining the dhcp4_options table // results in multiple rows being returned for the same host. The hosts are // retrieved by subnet id. {1, { OID_INT8 }, "get_host_subid4", "SELECT h.host_id, h.dhcp_identifier, h.dhcp_identifier_type, " " h.dhcp4_subnet_id, h.dhcp6_subnet_id, h.ipv4_address, h.hostname, " " h.dhcp4_client_classes, h.dhcp6_client_classes, h.user_context, " " h.dhcp4_next_server, h.dhcp4_server_hostname, " " h.dhcp4_boot_file_name, h.auth_key, " " o.option_id, o.code, o.value, o.formatted_value, o.space, " " o.persistent, o.cancelled, o.user_context " "FROM hosts AS h " "LEFT JOIN dhcp4_options AS o ON h.host_id = o.host_id " "WHERE h.dhcp4_subnet_id = $1 " "ORDER BY h.host_id, o.option_id" }, // PgSqlHostDataSourceImpl::GET_HOST_SUBID6 // // Retrieves host information, IPv6 reservations and DHCPv6 options // associated with all hosts using the IPv6 subnet id. This query returns // host information for many hosts. However, multiple rows are // returned due to left joining IPv6 reservations and DHCPv6 options. // The number of rows returned is multiplication of number of existing // IPv6 reservations and DHCPv6 options for each host in a subnet. There // are usually many hosts in a subnet. The amount of returned data may // be huge. {1, { OID_INT8 }, "get_host_subid6", "SELECT h.host_id, h.dhcp_identifier, " " h.dhcp_identifier_type, h.dhcp4_subnet_id, " " h.dhcp6_subnet_id, h.ipv4_address, h.hostname, " " h.dhcp4_client_classes, h.dhcp6_client_classes, h.user_context, " " h.dhcp4_next_server, h.dhcp4_server_hostname, " " h.dhcp4_boot_file_name, h.auth_key, " " o.option_id, o.code, o.value, o.formatted_value, o.space, " " o.persistent, o.cancelled, o.user_context, " " r.reservation_id, host(r.address), r.prefix_len, r.type, r.dhcp6_iaid " "FROM hosts AS h " "LEFT JOIN dhcp6_options AS o ON h.host_id = o.host_id " "LEFT JOIN ipv6_reservations AS r ON h.host_id = r.host_id " "WHERE h.dhcp6_subnet_id = $1 " "ORDER BY h.host_id, o.option_id, r.reservation_id" }, // PgSqlHostDataSourceImpl::GET_HOST_HOSTNAME // Retrieves host information, IPv6 reservations and both DHCPv4 and // DHCPv6 options associated with all hosts using the hostname. // The LEFT JOIN clause is used to retrieve information from 4 different // tables using a single query. Hence, this query returns multiple rows // for a single host. {1, { OID_VARCHAR }, "get_host_hostname", "SELECT h.host_id, h.dhcp_identifier, h.dhcp_identifier_type, " " h.dhcp4_subnet_id, h.dhcp6_subnet_id, h.ipv4_address, " " h.hostname, h.dhcp4_client_classes, h.dhcp6_client_classes, " " h.user_context, " " h.dhcp4_next_server, h.dhcp4_server_hostname, " " h.dhcp4_boot_file_name, h.auth_key, " " o4.option_id, o4.code, o4.value, o4.formatted_value, o4.space, " " o4.persistent, o4.cancelled, o4.user_context, " " o6.option_id, o6.code, o6.value, o6.formatted_value, o6.space, " " o6.persistent, o6.cancelled, o6.user_context, " " r.reservation_id, host(r.address), r.prefix_len, r.type, r.dhcp6_iaid " "FROM hosts AS h " "LEFT JOIN dhcp4_options AS o4 ON h.host_id = o4.host_id " "LEFT JOIN dhcp6_options AS o6 ON h.host_id = o6.host_id " "LEFT JOIN ipv6_reservations AS r ON h.host_id = r.host_id " "WHERE lower(h.hostname) = $1 " "ORDER BY h.host_id, o4.option_id, o6.option_id, r.reservation_id" }, // PgSqlHostDataSourceImpl::GET_HOST_HOSTNAME_SUBID4 // Retrieves host information for all hosts with a hostname in a subnet, // along with the DHCPv4 options associated with it. Left joining // the dhcp4_options table results in multiple rows being returned for // the same host. {2, { OID_VARCHAR, OID_INT8 }, "get_host_hostname_subid4", "SELECT h.host_id, h.dhcp_identifier, h.dhcp_identifier_type, " " h.dhcp4_subnet_id, h.dhcp6_subnet_id, h.ipv4_address, h.hostname, " " h.dhcp4_client_classes, h.dhcp6_client_classes, h.user_context, " " h.dhcp4_next_server, h.dhcp4_server_hostname, " " h.dhcp4_boot_file_name, h.auth_key, " " o.option_id, o.code, o.value, o.formatted_value, o.space, " " o.persistent, o.cancelled, o.user_context " "FROM hosts AS h " "LEFT JOIN dhcp4_options AS o ON h.host_id = o.host_id " "WHERE lower(h.hostname) = $1 AND h.dhcp4_subnet_id = $2 " "ORDER BY h.host_id, o.option_id" }, // PgSqlHostDataSourceImpl::GET_HOST_HOSTNAME_SUBID6 // Retrieves host information, IPv6 reservations and DHCPv6 options // associated with all hosts using the hostname and the IPv6 subnet id. // This query returns host information for many hosts. However, multiple // rows are returned due to left joining IPv6 reservations and DHCPv6 // options. The number of rows returned is multiplication of number of // existing IPv6 reservations and DHCPv6 options for each host in a subnet. {2, { OID_VARCHAR, OID_INT8 }, "get_host_hostname_subid6", "SELECT h.host_id, h.dhcp_identifier, " " h.dhcp_identifier_type, h.dhcp4_subnet_id, " " h.dhcp6_subnet_id, h.ipv4_address, h.hostname, " " h.dhcp4_client_classes, h.dhcp6_client_classes, h.user_context, " " h.dhcp4_next_server, h.dhcp4_server_hostname, " " h.dhcp4_boot_file_name, h.auth_key, " " o.option_id, o.code, o.value, o.formatted_value, o.space, " " o.persistent, o.cancelled, o.user_context, " " r.reservation_id, host(r.address), r.prefix_len, r.type, r.dhcp6_iaid " "FROM hosts AS h " "LEFT JOIN dhcp6_options AS o ON h.host_id = o.host_id " "LEFT JOIN ipv6_reservations AS r ON h.host_id = r.host_id " "WHERE lower(h.hostname) = $1 AND h.dhcp6_subnet_id = $2 " "ORDER BY h.host_id, o.option_id, r.reservation_id" }, // PgSqlHostDataSourceImpl::GET_HOST_SUBID4_PAGE // Retrieves host information along with the DHCPv4 options associated with // it. Left joining the dhcp4_options table results in multiple rows being // returned for the same host. The hosts are retrieved by subnet id, // starting from specified host id. Specified number of hosts is returned. {3, { OID_INT8, OID_INT8, OID_INT8 }, "get_host_subid4_page", "SELECT h.host_id, h.dhcp_identifier, h.dhcp_identifier_type, " " h.dhcp4_subnet_id, h.dhcp6_subnet_id, h.ipv4_address, h.hostname, " " h.dhcp4_client_classes, h.dhcp6_client_classes, h.user_context, " " h.dhcp4_next_server, h.dhcp4_server_hostname, " " h.dhcp4_boot_file_name, h.auth_key, " " o.option_id, o.code, o.value, o.formatted_value, o.space, " " o.persistent, o.cancelled, o.user_context " "FROM ( SELECT * FROM hosts AS h " " WHERE h.dhcp4_subnet_id = $1 AND h.host_id > $2 " " ORDER BY h.host_id " " LIMIT $3 ) AS h " "LEFT JOIN dhcp4_options AS o ON h.host_id = o.host_id " "ORDER BY h.host_id, o.option_id" }, // PgSqlHostDataSourceImpl::GET_HOST_SUBID6_PAGE // Retrieves host information, IPv6 reservations and DHCPv6 options // associated with a host using IPv6 subnet id. This query returns // host information for a single host. However, multiple rows are // returned due to left joining IPv6 reservations and DHCPv6 options. // The number of rows returned is multiplication of number of existing // IPv6 reservations and DHCPv6 options. {3, { OID_INT8, OID_INT8, OID_INT8 }, "get_host_subid6_page", "SELECT h.host_id, h.dhcp_identifier, " " h.dhcp_identifier_type, h.dhcp4_subnet_id, " " h.dhcp6_subnet_id, h.ipv4_address, h.hostname, " " h.dhcp4_client_classes, h.dhcp6_client_classes, h.user_context, " " h.dhcp4_next_server, h.dhcp4_server_hostname, " " h.dhcp4_boot_file_name, h.auth_key, " " o.option_id, o.code, o.value, o.formatted_value, o.space, " " o.persistent, o.cancelled, o.user_context, " " r.reservation_id, host(r.address), r.prefix_len, r.type, r.dhcp6_iaid " "FROM ( SELECT * FROM hosts AS h " " WHERE h.dhcp6_subnet_id = $1 AND h.host_id > $2 " " ORDER BY h.host_id " " LIMIT $3 ) AS h " "LEFT JOIN dhcp6_options AS o ON h.host_id = o.host_id " "LEFT JOIN ipv6_reservations AS r ON h.host_id = r.host_id " "ORDER BY h.host_id, o.option_id, r.reservation_id" }, // PgSqlHostDataSourceImpl::GET_HOST_PAGE4 // Retrieves host information along with the DHCPv4 options associated with // it. Left joining the dhcp4_options table results in multiple rows being // returned for the same host. The hosts are retrieved starting from // specified host id. Specified number of hosts is returned. {2, { OID_INT8, OID_INT8 }, "get_host_page4", "SELECT h.host_id, h.dhcp_identifier, h.dhcp_identifier_type, " " h.dhcp4_subnet_id, h.dhcp6_subnet_id, h.ipv4_address, h.hostname, " " h.dhcp4_client_classes, h.dhcp6_client_classes, h.user_context, " " h.dhcp4_next_server, h.dhcp4_server_hostname, " " h.dhcp4_boot_file_name, h.auth_key, " " o.option_id, o.code, o.value, o.formatted_value, o.space, " " o.persistent, o.cancelled, o.user_context " "FROM ( SELECT * FROM hosts AS h " " WHERE h.host_id > $1 " " ORDER BY h.host_id " " LIMIT $2 ) AS h " "LEFT JOIN dhcp4_options AS o ON h.host_id = o.host_id " "ORDER BY h.host_id, o.option_id" }, // PgSqlHostDataSourceImpl::GET_HOST_PAGE6 // Retrieves host information, IPv6 reservations and DHCPv6 options // associated with a host using IPv6 subnet id. This query returns // host information for a single host. However, multiple rows are // returned due to left joining IPv6 reservations and DHCPv6 options. // The number of rows returned is multiplication of number of existing // IPv6 reservations and DHCPv6 options. {2, { OID_INT8, OID_INT8 }, "get_host_page6", "SELECT h.host_id, h.dhcp_identifier, " " h.dhcp_identifier_type, h.dhcp4_subnet_id, " " h.dhcp6_subnet_id, h.ipv4_address, h.hostname, " " h.dhcp4_client_classes, h.dhcp6_client_classes, h.user_context, " " h.dhcp4_next_server, h.dhcp4_server_hostname, " " h.dhcp4_boot_file_name, h.auth_key, " " o.option_id, o.code, o.value, o.formatted_value, o.space, " " o.persistent, o.cancelled, o.user_context, " " r.reservation_id, host(r.address), r.prefix_len, r.type, r.dhcp6_iaid " "FROM ( SELECT * FROM hosts AS h " " WHERE h.host_id > $1 " " ORDER BY h.host_id " " LIMIT $2 ) AS h " "LEFT JOIN dhcp6_options AS o ON h.host_id = o.host_id " "LEFT JOIN ipv6_reservations AS r ON h.host_id = r.host_id " "ORDER BY h.host_id, o.option_id, r.reservation_id" }, // PgSqlHostDataSourceImpl::INSERT_HOST_NON_UNIQUE_IP // Inserts a host into the 'hosts' table without checking that there is // a reservation for the IP address. {13, { OID_BYTEA, OID_INT2, OID_INT8, OID_INT8, OID_INT8, OID_VARCHAR, OID_VARCHAR, OID_VARCHAR, OID_TEXT, OID_INT8, OID_VARCHAR, OID_VARCHAR, OID_VARCHAR}, "insert_host_non_unique_ip", "INSERT INTO hosts(dhcp_identifier, dhcp_identifier_type, " " dhcp4_subnet_id, dhcp6_subnet_id, ipv4_address, hostname, " " dhcp4_client_classes, dhcp6_client_classes, user_context, " " dhcp4_next_server, dhcp4_server_hostname, dhcp4_boot_file_name, auth_key)" "VALUES ( $1, $2, $3, $4, $5, $6, $7, $8, $9, $10, $11, $12, $13 ) " "RETURNING host_id" }, // PgSqlHostDataSourceImpl::INSERT_HOST_UNIQUE_IP // Inserts a host into the 'hosts' table with checking that reserved IP // address is unique. The innermost query checks if there is at least // one host for the given IP/subnet combination. For checking whether // hosts exists or not it doesn't matter if we select actual columns, // thus SELECT 1 was used as an optimization to avoid selecting data // that will be ignored anyway. If it does not exist the new host is // inserted. If the host with the given IP address already exists the // new host won't be inserted. The caller can check the number of // affected rows to detect that there was a duplicate host in the // database. Returns the inserted host id. {15, { OID_BYTEA, OID_INT2, OID_INT8, OID_INT8, OID_INT8, OID_VARCHAR, OID_VARCHAR, OID_VARCHAR, OID_TEXT, OID_INT8, OID_VARCHAR, OID_VARCHAR, OID_VARCHAR, OID_INT8, OID_INT8}, "insert_host_unique_ip", "INSERT INTO hosts(dhcp_identifier, dhcp_identifier_type, " " dhcp4_subnet_id, dhcp6_subnet_id, ipv4_address, hostname, " " dhcp4_client_classes, dhcp6_client_classes, user_context, " " dhcp4_next_server, dhcp4_server_hostname, dhcp4_boot_file_name, auth_key)" " SELECT $1, $2, $3, $4, $5, $6, $7, $8, $9, $10, $11, $12, $13" " WHERE NOT EXISTS (" " SELECT 1 FROM hosts WHERE ipv4_address = $14 AND dhcp4_subnet_id = $15" " LIMIT 1" " ) " "RETURNING host_id" }, // PgSqlHostDataSourceImpl::INSERT_V6_RESRV_NON_UNIQUE // Inserts a single IPv6 reservation into 'reservations' table without // checking that the inserted reservation is unique. {5, { OID_VARCHAR, OID_INT2, OID_INT4, OID_INT4, OID_INT4 }, "insert_v6_resrv_non_unique", "INSERT INTO ipv6_reservations(address, prefix_len, type, " " dhcp6_iaid, host_id) " "VALUES (cast($1 as inet), $2, $3, $4, $5)" }, // PgSqlHostDataSourceImpl::INSERT_V6_RESRV_UNIQUE // Inserts a single IPv6 reservation into 'reservations' table with // checking that the inserted reservation is unique. {7, { OID_VARCHAR, OID_INT2, OID_INT4, OID_INT4, OID_INT4, OID_VARCHAR, OID_INT2 }, "insert_v6_resrv_unique", "INSERT INTO ipv6_reservations(address, prefix_len, type, " " dhcp6_iaid, host_id) " "SELECT cast($1 as inet), $2, $3, $4, $5 " " WHERE NOT EXISTS (" " SELECT 1 FROM ipv6_reservations" " WHERE address = cast($6 as inet) AND prefix_len = $7" " LIMIT 1" " )" }, // PgSqlHostDataSourceImpl::INSERT_V4_HOST_OPTION // Inserts a single DHCPv4 option into 'dhcp4_options' table. // Using fixed scope_id = 3, which associates an option with host. {8, { OID_INT2, OID_BYTEA, OID_TEXT, OID_VARCHAR, OID_BOOL, OID_BOOL, OID_TEXT, OID_INT8 }, "insert_v4_host_option", "INSERT INTO dhcp4_options(code, value, formatted_value, space, " " persistent, cancelled, user_context, host_id, scope_id) " "VALUES ($1, $2, $3, $4, $5, $6, $7, $8, 3)" }, // PgSqlHostDataSourceImpl::INSERT_V6_HOST_OPTION // Inserts a single DHCPv6 option into 'dhcp6_options' table. // Using fixed scope_id = 3, which associates an option with host. {8, { OID_INT2, OID_BYTEA, OID_TEXT, OID_VARCHAR, OID_BOOL, OID_BOOL, OID_TEXT, OID_INT8 }, "insert_v6_host_option", "INSERT INTO dhcp6_options(code, value, formatted_value, space, " " persistent, cancelled, user_context, host_id, scope_id) " "VALUES ($1, $2, $3, $4, $5, $6, $7, $8, 3)" }, // PgSqlHostDataSourceImpl::DEL_HOST_ADDR4 // Deletes a v4 host that matches (subnet-id, addr4) {2, { OID_INT8, OID_INT8 }, "del_host_addr4", "DELETE FROM hosts WHERE dhcp4_subnet_id = $1 AND ipv4_address = $2" }, // PgSqlHostDataSourceImpl::DEL_HOST_ADDR6 // Deletes a v6 host that matches (subnet-id, addr6) {2, { OID_INT8, OID_VARCHAR }, "del_host_addr6", "DELETE FROM hosts USING ipv6_reservations " " WHERE dhcp6_subnet_id = $1 AND ipv6_reservations.address = cast($2 as inet)" }, // PgSqlHostDataSourceImpl::DEL_HOST_SUBID4_ID // Deletes a v4 host that matches (subnet4-id, identifier-type, identifier) {3, { OID_INT8, OID_INT2, OID_BYTEA }, "del_host_subid4_id", "DELETE FROM hosts WHERE dhcp4_subnet_id = $1 " "AND dhcp_identifier_type = $2 " "AND dhcp_identifier = $3" }, // PgSqlHostDataSourceImpl::DEL_HOST_SUBID6_ID // Deletes a v6 host that matches (subnet6-id, identifier-type, identifier) {3, { OID_INT8, OID_INT2, OID_BYTEA }, "del_host_subid6_id", "DELETE FROM hosts WHERE dhcp6_subnet_id = $1 " "AND dhcp_identifier_type = $2 " "AND dhcp_identifier = $3" } } }; } // namespace // PgSqlHostContext Constructor PgSqlHostContext::PgSqlHostContext(const DatabaseConnection::ParameterMap& parameters, IOServiceAccessorPtr io_service_accessor, db::DbCallback db_reconnect_callback) : conn_(parameters, io_service_accessor, db_reconnect_callback), is_readonly_(true) { } // PgSqlHostContextAlloc Constructor and Destructor PgSqlHostDataSource::PgSqlHostContextAlloc::PgSqlHostContextAlloc( PgSqlHostDataSourceImpl& mgr) : ctx_(), mgr_(mgr) { if (MultiThreadingMgr::instance().getMode()) { // multi-threaded { // we need to protect the whole pool_ operation, hence extra scope {} lock_guard lock(mgr_.pool_->mutex_); if (!mgr_.pool_->pool_.empty()) { ctx_ = mgr_.pool_->pool_.back(); mgr_.pool_->pool_.pop_back(); } } if (!ctx_) { ctx_ = mgr_.createContext(); } } else { // single-threaded if (mgr_.pool_->pool_.empty()) { isc_throw(Unexpected, "No available PostgreSQL host context?!"); } ctx_ = mgr_.pool_->pool_.back(); } } PgSqlHostDataSource::PgSqlHostContextAlloc::~PgSqlHostContextAlloc() { if (MultiThreadingMgr::instance().getMode()) { // multi-threaded lock_guard lock(mgr_.pool_->mutex_); mgr_.pool_->pool_.push_back(ctx_); if (ctx_->conn_.isUnusable()) { mgr_.unusable_ = true; } } else if (ctx_->conn_.isUnusable()) { mgr_.unusable_ = true; } } PgSqlHostDataSourceImpl::PgSqlHostDataSourceImpl(const DatabaseConnection::ParameterMap& parameters) : parameters_(parameters), ip_reservations_unique_(true), unusable_(false), timer_name_("") { // Create unique timer name per instance. timer_name_ = "PgSqlHostMgr["; timer_name_ += boost::lexical_cast(reinterpret_cast(this)); timer_name_ += "]DbReconnectTimer"; // Check TLS support. size_t tls(0); tls += parameters.count("trust-anchor"); tls += parameters.count("cert-file"); tls += parameters.count("key-file"); tls += parameters.count("cipher-list"); #ifdef HAVE_PGSQL_SSL if ((tls > 0) && !PgSqlConnection::warned_about_tls) { PgSqlConnection::warned_about_tls = true; LOG_INFO(dhcpsrv_logger, DHCPSRV_PGSQL_TLS_SUPPORT) .arg(DatabaseConnection::redactedAccessString(parameters_)); PQinitSSL(1); } #else if (tls > 0) { LOG_ERROR(dhcpsrv_logger, DHCPSRV_PGSQL_NO_TLS_SUPPORT) .arg(DatabaseConnection::redactedAccessString(parameters_)); isc_throw(DbOpenError, "Attempt to configure TLS for PostgreSQL " << "backend (built with this feature disabled)"); } #endif // Validate the schema version first. std::pair code_version(PGSQL_SCHEMA_VERSION_MAJOR, PGSQL_SCHEMA_VERSION_MINOR); std::pair db_version = getVersion(); if (code_version != db_version) { isc_throw(DbOpenError, "PostgreSQL schema version mismatch: need version: " << code_version.first << "." << code_version.second << " found version: " << db_version.first << "." << db_version.second); } // Create an initial context. pool_.reset(new PgSqlHostContextPool()); pool_->pool_.push_back(createContext()); } // Create context. PgSqlHostContextPtr PgSqlHostDataSourceImpl::createContext() const { PgSqlHostContextPtr ctx(new PgSqlHostContext(parameters_, IOServiceAccessorPtr(new IOServiceAccessor(&HostMgr::getIOService)), &PgSqlHostDataSourceImpl::dbReconnect)); // Open the database. ctx->conn_.openDatabase(); // Now prepare the SQL statements. ctx->conn_.prepareStatements(tagged_statements.begin(), tagged_statements.begin() + WRITE_STMTS_BEGIN); // Check if the backend is explicitly configured to operate with // read only access to the database. ctx->is_readonly_ = ctx->conn_.configuredReadOnly(); // If we are using read-write mode for the database we also prepare // statements for INSERTS etc. if (!ctx->is_readonly_) { ctx->conn_.prepareStatements(tagged_statements.begin() + WRITE_STMTS_BEGIN, tagged_statements.end()); } else { LOG_INFO(dhcpsrv_logger, DHCPSRV_PGSQL_HOST_DB_READONLY); } ctx->host_ipv4_exchange_.reset(new PgSqlHostWithOptionsExchange(PgSqlHostWithOptionsExchange::DHCP4_ONLY)); ctx->host_ipv6_exchange_.reset(new PgSqlHostIPv6Exchange(PgSqlHostWithOptionsExchange::DHCP6_ONLY)); ctx->host_ipv46_exchange_.reset(new PgSqlHostIPv6Exchange(PgSqlHostWithOptionsExchange::DHCP4_AND_DHCP6)); ctx->host_ipv6_reservation_exchange_.reset(new PgSqlIPv6ReservationExchange()); ctx->host_option_exchange_.reset(new PgSqlOptionExchange()); // Create ReconnectCtl for this connection. ctx->conn_.makeReconnectCtl(timer_name_); return (ctx); } PgSqlHostDataSourceImpl::~PgSqlHostDataSourceImpl() { } bool PgSqlHostDataSourceImpl::dbReconnect(ReconnectCtlPtr db_reconnect_ctl) { MultiThreadingCriticalSection cs; // Invoke application layer connection lost callback. if (!DatabaseConnection::invokeDbLostCallback(db_reconnect_ctl)) { return (false); } bool reopened = false; const std::string timer_name = db_reconnect_ctl->timerName(); // At least one connection was lost. try { CfgDbAccessPtr cfg_db = CfgMgr::instance().getCurrentCfg()->getCfgDbAccess(); std::list host_db_access_list = cfg_db->getHostDbAccessStringList(); for (std::string& hds : host_db_access_list) { auto parameters = DatabaseConnection::parse(hds); if (HostMgr::delBackend("postgresql", hds, true)) { HostMgr::addBackend(hds); } } reopened = true; } catch (const std::exception& ex) { LOG_ERROR(dhcpsrv_logger, DHCPSRV_PGSQL_HOST_DB_RECONNECT_ATTEMPT_FAILED) .arg(ex.what()); } if (reopened) { // Cancel the timer. if (TimerMgr::instance()->isTimerRegistered(timer_name)) { TimerMgr::instance()->unregisterTimer(timer_name); } // Invoke application layer connection recovered callback. if (!DatabaseConnection::invokeDbRecoveredCallback(db_reconnect_ctl)) { return (false); } } else { if (!db_reconnect_ctl->checkRetries()) { // We're out of retries, log it and initiate shutdown. LOG_ERROR(dhcpsrv_logger, DHCPSRV_PGSQL_HOST_DB_RECONNECT_FAILED) .arg(db_reconnect_ctl->maxRetries()); // Cancel the timer. if (TimerMgr::instance()->isTimerRegistered(timer_name)) { TimerMgr::instance()->unregisterTimer(timer_name); } // Invoke application layer connection failed callback. DatabaseConnection::invokeDbFailedCallback(db_reconnect_ctl); return (false); } LOG_INFO(dhcpsrv_logger, DHCPSRV_PGSQL_HOST_DB_RECONNECT_ATTEMPT_SCHEDULE) .arg(db_reconnect_ctl->maxRetries() - db_reconnect_ctl->retriesLeft() + 1) .arg(db_reconnect_ctl->maxRetries()) .arg(db_reconnect_ctl->retryInterval()); // Start the timer. if (!TimerMgr::instance()->isTimerRegistered(timer_name)) { TimerMgr::instance()->registerTimer(timer_name, std::bind(&PgSqlHostDataSourceImpl::dbReconnect, db_reconnect_ctl), db_reconnect_ctl->retryInterval(), asiolink::IntervalTimer::ONE_SHOT); } TimerMgr::instance()->setup(timer_name); } return (true); } uint64_t PgSqlHostDataSourceImpl::addStatement(PgSqlHostContextPtr& ctx, StatementIndex stindex, PsqlBindArrayPtr& bind_array, const bool return_last_id) { uint64_t last_id = 0; PgSqlResult r(PQexecPrepared(ctx->conn_, tagged_statements[stindex].name, tagged_statements[stindex].nbparams, &bind_array->values_[0], &bind_array->lengths_[0], &bind_array->formats_[0], 0)); int s = PQresultStatus(r); if (s != PGRES_COMMAND_OK) { // Failure: check for the special case of duplicate entry. if (ctx->conn_.compareError(r, PgSqlConnection::DUPLICATE_KEY)) { isc_throw(DuplicateEntry, "Database duplicate entry error"); } // Connection determines if the error is fatal or not, and // throws the appropriate exception ctx->conn_.checkStatementError(r, tagged_statements[stindex]); } // Get the number of affected rows. char* rows_affected = PQcmdTuples(r); if (!rows_affected) { isc_throw(DbOperationError, "Could not retrieve the number of affected rows."); } // If the number of rows inserted is 0 it means that the query detected // an attempt to insert duplicated data for which there is no unique // index in the database. Unique indexes are not created in the database // when it may be sometimes allowed to insert duplicated records per // server's configuration. if (rows_affected[0] == '0') { isc_throw(DuplicateEntry, "Database duplicate entry error"); } if (return_last_id) { PgSqlExchange::getColumnValue(r, 0, 0, last_id); } return (last_id); } bool PgSqlHostDataSourceImpl::delStatement(PgSqlHostContextPtr& ctx, StatementIndex stindex, PsqlBindArrayPtr& bind_array) { PgSqlResult r(PQexecPrepared(ctx->conn_, tagged_statements[stindex].name, tagged_statements[stindex].nbparams, &bind_array->values_[0], &bind_array->lengths_[0], &bind_array->formats_[0], 0)); int s = PQresultStatus(r); if (s != PGRES_COMMAND_OK) { // Connection determines if the error is fatal or not, and // throws the appropriate exception ctx->conn_.checkStatementError(r, tagged_statements[stindex]); } // Now check how many rows (hosts) were deleted. This should be either // "0" or "1". char* rows_deleted = PQcmdTuples(r); if (!rows_deleted) { isc_throw(DbOperationError, "Could not retrieve the number of deleted rows."); } return (rows_deleted[0] != '0'); } void PgSqlHostDataSourceImpl::addResv(PgSqlHostContextPtr& ctx, const IPv6Resrv& resv, const HostID& id) { PsqlBindArrayPtr bind_array = ctx->host_ipv6_reservation_exchange_-> createBindForSend(resv, id, ip_reservations_unique_); addStatement(ctx, ip_reservations_unique_ ? INSERT_V6_RESRV_UNIQUE : INSERT_V6_RESRV_NON_UNIQUE, bind_array); } void PgSqlHostDataSourceImpl::addOption(PgSqlHostContextPtr& ctx, const StatementIndex& stindex, const OptionDescriptor& opt_desc, const std::string& opt_space, const Optional&, const HostID& id) { PsqlBindArrayPtr bind_array = ctx->host_option_exchange_->createBindForSend(opt_desc, opt_space, id); addStatement(ctx, stindex, bind_array); } void PgSqlHostDataSourceImpl::addOptions(PgSqlHostContextPtr& ctx, const StatementIndex& stindex, const ConstCfgOptionPtr& options_cfg, const uint64_t host_id) { // Get option space names and vendor space names and combine them within a // single list. std::list option_spaces = options_cfg->getOptionSpaceNames(); std::list vendor_spaces = options_cfg->getVendorIdsSpaceNames(); option_spaces.insert(option_spaces.end(), vendor_spaces.begin(), vendor_spaces.end()); // For each option space retrieve all options and insert them into the // database. for (auto space = option_spaces.begin(); space != option_spaces.end(); ++space) { OptionContainerPtr options = options_cfg->getAllCombined(*space); if (options && !options->empty()) { for (auto opt = options->begin(); opt != options->end(); ++opt) { addOption(ctx, stindex, *opt, *space, Optional(), host_id); } } } } void PgSqlHostDataSourceImpl::getHostCollection(PgSqlHostContextPtr& ctx, StatementIndex stindex, PsqlBindArrayPtr bind_array, boost::shared_ptr exchange, ConstHostCollection& result, bool single) const { exchange->clear(); PgSqlResult r(PQexecPrepared(ctx->conn_, tagged_statements[stindex].name, tagged_statements[stindex].nbparams, &bind_array->values_[0], &bind_array->lengths_[0], &bind_array->formats_[0], 0)); ctx->conn_.checkStatementError(r, tagged_statements[stindex]); int rows = r.getRows(); for (int row = 0; row < rows; ++row) { exchange->processRowData(result, r, row); if (single && result.size() > 1) { isc_throw(MultipleRecords, "multiple records were found in the " "database where only one was expected for query " << tagged_statements[stindex].name); } } } ConstHostPtr PgSqlHostDataSourceImpl::getHost(PgSqlHostContextPtr& ctx, const SubnetID& subnet_id, const Host::IdentifierType& identifier_type, const uint8_t* identifier_begin, const size_t identifier_len, StatementIndex stindex, boost::shared_ptr exchange) const { // Set up the WHERE clause value PsqlBindArrayPtr bind_array(new PsqlBindArray()); // Add the subnet id. bind_array->add(subnet_id); // Add the Identifier type. bind_array->add(static_cast(identifier_type)); // Add the identifier value. bind_array->add(identifier_begin, identifier_len); ConstHostCollection collection; getHostCollection(ctx, stindex, bind_array, exchange, collection, true); // Return single record if present, else clear the host. ConstHostPtr result; if (!collection.empty()) { result = *collection.begin(); } return (result); } std::pair PgSqlHostDataSourceImpl::getVersion() const { LOG_DEBUG(dhcpsrv_logger, DHCPSRV_DBG_TRACE_DETAIL, DHCPSRV_PGSQL_HOST_DB_GET_VERSION); return (PgSqlConnection::getVersion(parameters_)); } void PgSqlHostDataSourceImpl::checkReadOnly(PgSqlHostContextPtr& ctx) const { if (ctx->is_readonly_) { isc_throw(ReadOnlyDb, "PostgreSQL host database backend is configured" " to operate in read only mode"); } } /*********** PgSqlHostDataSource *********************/ PgSqlHostDataSource::PgSqlHostDataSource(const DatabaseConnection::ParameterMap& parameters) : impl_(new PgSqlHostDataSourceImpl(parameters)) { } PgSqlHostDataSource::~PgSqlHostDataSource() { } DatabaseConnection::ParameterMap PgSqlHostDataSource::getParameters() const { return (impl_->parameters_); } void PgSqlHostDataSource::add(const HostPtr& host) { // Get a context PgSqlHostContextAlloc get_context(*impl_); PgSqlHostContextPtr ctx = get_context.ctx_; // If operating in read-only mode, throw exception. impl_->checkReadOnly(ctx); // Initiate PostgreSQL transaction as we will have to make multiple queries // to insert host information into multiple tables. If that fails on // any stage, the transaction will be rolled back by the destructor of // the PgSqlTransaction class. PgSqlTransaction transaction(ctx->conn_); // If we're configured to check that an IP reservation within a given subnet // is unique, the IP reservation exists and the subnet is actually set // we will be using a special query that checks for uniqueness. Otherwise, // we will use a regular insert statement. bool unique_ip = impl_->ip_reservations_unique_ && !host->getIPv4Reservation().isV4Zero() && host->getIPv4SubnetID() != SUBNET_ID_UNUSED; // Create the PgSQL Bind array for the host PsqlBindArrayPtr bind_array = ctx->host_ipv4_exchange_->createBindForSend(host, unique_ip); // ... and insert the host. uint32_t host_id = impl_->addStatement(ctx, unique_ip ? PgSqlHostDataSourceImpl::INSERT_HOST_UNIQUE_IP : PgSqlHostDataSourceImpl::INSERT_HOST_NON_UNIQUE_IP, bind_array, true); // Insert DHCPv4 options. ConstCfgOptionPtr cfg_option4 = host->getCfgOption4(); if (cfg_option4) { impl_->addOptions(ctx, PgSqlHostDataSourceImpl::INSERT_V4_HOST_OPTION, cfg_option4, host_id); } // Insert DHCPv6 options. ConstCfgOptionPtr cfg_option6 = host->getCfgOption6(); if (cfg_option6) { impl_->addOptions(ctx, PgSqlHostDataSourceImpl::INSERT_V6_HOST_OPTION, cfg_option6, host_id); } // Insert IPv6 reservations. IPv6ResrvRange v6resv = host->getIPv6Reservations(); if (std::distance(v6resv.first, v6resv.second) > 0) { for (IPv6ResrvIterator resv = v6resv.first; resv != v6resv.second; ++resv) { impl_->addResv(ctx, resv->second, host_id); } } // Everything went fine, so explicitly commit the transaction. transaction.commit(); } bool PgSqlHostDataSource::del(const SubnetID& subnet_id, const asiolink::IOAddress& addr) { // Get a context PgSqlHostContextAlloc get_context(*impl_); PgSqlHostContextPtr ctx = get_context.ctx_; // If operating in read-only mode, throw exception. impl_->checkReadOnly(ctx); PsqlBindArrayPtr bind_array(new PsqlBindArray()); bind_array->add(subnet_id); // v4 if (addr.isV4()) { bind_array->add(addr); return (impl_->delStatement(ctx, PgSqlHostDataSourceImpl::DEL_HOST_ADDR4, bind_array)); } // v6 bind_array->addTempString(addr.toText()); return (impl_->delStatement(ctx, PgSqlHostDataSourceImpl::DEL_HOST_ADDR6, bind_array)); } bool PgSqlHostDataSource::del4(const SubnetID& subnet_id, const Host::IdentifierType& identifier_type, const uint8_t* identifier_begin, const size_t identifier_len) { // Get a context PgSqlHostContextAlloc get_context(*impl_); PgSqlHostContextPtr ctx = get_context.ctx_; // If operating in read-only mode, throw exception. impl_->checkReadOnly(ctx); PsqlBindArrayPtr bind_array(new PsqlBindArray()); // Subnet-id bind_array->add(subnet_id); // identifier-type bind_array->add(static_cast(identifier_type)); // identifier bind_array->add(identifier_begin, identifier_len); return (impl_->delStatement(ctx, PgSqlHostDataSourceImpl::DEL_HOST_SUBID4_ID, bind_array)); } bool PgSqlHostDataSource::del6(const SubnetID& subnet_id, const Host::IdentifierType& identifier_type, const uint8_t* identifier_begin, const size_t identifier_len) { // Get a context PgSqlHostContextAlloc get_context(*impl_); PgSqlHostContextPtr ctx = get_context.ctx_; // If operating in read-only mode, throw exception. impl_->checkReadOnly(ctx); PsqlBindArrayPtr bind_array(new PsqlBindArray()); // Subnet-id bind_array->add(subnet_id); // identifier-type bind_array->add(static_cast(identifier_type)); // identifier bind_array->add(identifier_begin, identifier_len); return (impl_->delStatement(ctx, PgSqlHostDataSourceImpl::DEL_HOST_SUBID6_ID, bind_array)); } ConstHostCollection PgSqlHostDataSource::getAll(const Host::IdentifierType& identifier_type, const uint8_t* identifier_begin, const size_t identifier_len) const { // Get a context PgSqlHostContextAlloc get_context(*impl_); PgSqlHostContextPtr ctx = get_context.ctx_; // Set up the WHERE clause value PsqlBindArrayPtr bind_array(new PsqlBindArray()); // Identifier value. bind_array->add(identifier_begin, identifier_len); // Identifier type. bind_array->add(static_cast(identifier_type)); ConstHostCollection result; impl_->getHostCollection(ctx, PgSqlHostDataSourceImpl::GET_HOST_DHCPID, bind_array, ctx->host_ipv46_exchange_, result, false); return (result); } ConstHostCollection PgSqlHostDataSource::getAll4(const SubnetID& subnet_id) const { // Get a context PgSqlHostContextAlloc get_context(*impl_); PgSqlHostContextPtr ctx = get_context.ctx_; // Set up the WHERE clause value PsqlBindArrayPtr bind_array(new PsqlBindArray()); // Add the subnet id. bind_array->add(subnet_id); ConstHostCollection result; impl_->getHostCollection(ctx, PgSqlHostDataSourceImpl::GET_HOST_SUBID4, bind_array, ctx->host_ipv4_exchange_, result, false); return (result); } ConstHostCollection PgSqlHostDataSource::getAll6(const SubnetID& subnet_id) const { // Get a context PgSqlHostContextAlloc get_context(*impl_); PgSqlHostContextPtr ctx = get_context.ctx_; // Set up the WHERE clause value PsqlBindArrayPtr bind_array(new PsqlBindArray()); // Add the subnet id. bind_array->add(subnet_id); ConstHostCollection result; impl_->getHostCollection(ctx, PgSqlHostDataSourceImpl::GET_HOST_SUBID6, bind_array, ctx->host_ipv6_exchange_, result, false); return (result); } ConstHostCollection PgSqlHostDataSource::getAllbyHostname(const std::string& hostname) const { // Get a context PgSqlHostContextAlloc get_context(*impl_); PgSqlHostContextPtr ctx = get_context.ctx_; // Set up the WHERE clause value PsqlBindArrayPtr bind_array(new PsqlBindArray()); // Add the hostname. bind_array->add(hostname); ConstHostCollection result; impl_->getHostCollection(ctx, PgSqlHostDataSourceImpl::GET_HOST_HOSTNAME, bind_array, ctx->host_ipv46_exchange_, result, false); return (result); } ConstHostCollection PgSqlHostDataSource::getAllbyHostname4(const std::string& hostname, const SubnetID& subnet_id) const { // Get a context PgSqlHostContextAlloc get_context(*impl_); PgSqlHostContextPtr ctx = get_context.ctx_; // Set up the WHERE clause value PsqlBindArrayPtr bind_array(new PsqlBindArray()); // Add the hostname. bind_array->add(hostname); // Add the subnet id. bind_array->add(subnet_id); ConstHostCollection result; impl_->getHostCollection(ctx, PgSqlHostDataSourceImpl::GET_HOST_HOSTNAME_SUBID4, bind_array, ctx->host_ipv4_exchange_, result, false); return (result); } ConstHostCollection PgSqlHostDataSource::getAllbyHostname6(const std::string& hostname, const SubnetID& subnet_id) const { // Get a context PgSqlHostContextAlloc get_context(*impl_); PgSqlHostContextPtr ctx = get_context.ctx_; // Set up the WHERE clause value PsqlBindArrayPtr bind_array(new PsqlBindArray()); // Add the hostname. bind_array->add(hostname); // Add the subnet id. bind_array->add(subnet_id); ConstHostCollection result; impl_->getHostCollection(ctx, PgSqlHostDataSourceImpl::GET_HOST_HOSTNAME_SUBID6, bind_array, ctx->host_ipv6_exchange_, result, false); return (result); } ConstHostCollection PgSqlHostDataSource::getPage4(const SubnetID& subnet_id, size_t& /*source_index*/, uint64_t lower_host_id, const HostPageSize& page_size) const { // Get a context PgSqlHostContextAlloc get_context(*impl_); PgSqlHostContextPtr ctx = get_context.ctx_; // Set up the WHERE clause value PsqlBindArrayPtr bind_array(new PsqlBindArray()); // Add the subnet id. bind_array->add(subnet_id); // Add the lower bound host id. bind_array->add(lower_host_id); // Add the page size value. string page_size_data = boost::lexical_cast(page_size.page_size_); bind_array->add(page_size_data); ConstHostCollection result; impl_->getHostCollection(ctx, PgSqlHostDataSourceImpl::GET_HOST_SUBID4_PAGE, bind_array, ctx->host_ipv4_exchange_, result, false); return (result); } ConstHostCollection PgSqlHostDataSource::getPage6(const SubnetID& subnet_id, size_t& /*source_index*/, uint64_t lower_host_id, const HostPageSize& page_size) const { // Get a context PgSqlHostContextAlloc get_context(*impl_); PgSqlHostContextPtr ctx = get_context.ctx_; // Set up the WHERE clause value PsqlBindArrayPtr bind_array(new PsqlBindArray()); // Add the subnet id. bind_array->add(subnet_id); // Add the lower bound host id. bind_array->add(lower_host_id); // Add the page size value. string page_size_data = boost::lexical_cast(page_size.page_size_); bind_array->add(page_size_data); ConstHostCollection result; impl_->getHostCollection(ctx, PgSqlHostDataSourceImpl::GET_HOST_SUBID6_PAGE, bind_array, ctx->host_ipv6_exchange_, result, false); return (result); } ConstHostCollection PgSqlHostDataSource::getPage4(size_t& /*source_index*/, uint64_t lower_host_id, const HostPageSize& page_size) const { // Get a context PgSqlHostContextAlloc get_context(*impl_); PgSqlHostContextPtr ctx = get_context.ctx_; // Set up the WHERE clause value PsqlBindArrayPtr bind_array(new PsqlBindArray()); // Add the lower bound host id. bind_array->add(lower_host_id); // Add the page size value. string page_size_data = boost::lexical_cast(page_size.page_size_); bind_array->add(page_size_data); ConstHostCollection result; impl_->getHostCollection(ctx, PgSqlHostDataSourceImpl::GET_HOST_PAGE4, bind_array, ctx->host_ipv4_exchange_, result, false); return (result); } ConstHostCollection PgSqlHostDataSource::getPage6(size_t& /*source_index*/, uint64_t lower_host_id, const HostPageSize& page_size) const { // Get a context PgSqlHostContextAlloc get_context(*impl_); PgSqlHostContextPtr ctx = get_context.ctx_; // Set up the WHERE clause value PsqlBindArrayPtr bind_array(new PsqlBindArray()); // Add the lower bound host id. bind_array->add(lower_host_id); // Add the page size value. string page_size_data = boost::lexical_cast(page_size.page_size_); bind_array->add(page_size_data); ConstHostCollection result; impl_->getHostCollection(ctx, PgSqlHostDataSourceImpl::GET_HOST_PAGE6, bind_array, ctx->host_ipv6_exchange_, result, false); return (result); } ConstHostCollection PgSqlHostDataSource::getAll4(const asiolink::IOAddress& address) const { // Get a context PgSqlHostContextAlloc get_context(*impl_); PgSqlHostContextPtr ctx = get_context.ctx_; // Set up the WHERE clause value PsqlBindArrayPtr bind_array(new PsqlBindArray()); // v4 Reservation address bind_array->add(address); ConstHostCollection result; impl_->getHostCollection(ctx, PgSqlHostDataSourceImpl::GET_HOST_ADDR, bind_array, ctx->host_ipv4_exchange_, result, false); return (result); } ConstHostPtr PgSqlHostDataSource::get4(const SubnetID& subnet_id, const Host::IdentifierType& identifier_type, const uint8_t* identifier_begin, const size_t identifier_len) const { // Get a context PgSqlHostContextAlloc get_context(*impl_); PgSqlHostContextPtr ctx = get_context.ctx_; return (impl_->getHost(ctx, subnet_id, identifier_type, identifier_begin, identifier_len, PgSqlHostDataSourceImpl::GET_HOST_SUBID4_DHCPID, ctx->host_ipv4_exchange_)); } ConstHostPtr PgSqlHostDataSource::get4(const SubnetID& subnet_id, const asiolink::IOAddress& address) const { // Get a context PgSqlHostContextAlloc get_context(*impl_); PgSqlHostContextPtr ctx = get_context.ctx_; if (!address.isV4()) { isc_throw(BadValue, "PgSqlHostDataSource::get4(id, address) - " " wrong address type, address supplied is an IPv6 address"); } // Set up the WHERE clause value PsqlBindArrayPtr bind_array(new PsqlBindArray()); // Add the subnet id bind_array->add(subnet_id); // Add the address bind_array->add(address); ConstHostCollection collection; impl_->getHostCollection(ctx, PgSqlHostDataSourceImpl::GET_HOST_SUBID_ADDR, bind_array, ctx->host_ipv4_exchange_, collection, true); // Return single record if present, else clear the host. ConstHostPtr result; if (!collection.empty()) { result = *collection.begin(); } return (result); } ConstHostCollection PgSqlHostDataSource::getAll4(const SubnetID& subnet_id, const asiolink::IOAddress& address) const { // Get a context PgSqlHostContextAlloc get_context(*impl_); PgSqlHostContextPtr ctx = get_context.ctx_; if (!address.isV4()) { isc_throw(BadValue, "PgSqlHostDataSource::get4(id, address) - " " wrong address type, address supplied is an IPv6 address"); } // Set up the WHERE clause value PsqlBindArrayPtr bind_array(new PsqlBindArray()); // Add the subnet id bind_array->add(subnet_id); // Add the address bind_array->add(address); ConstHostCollection collection; impl_->getHostCollection(ctx, PgSqlHostDataSourceImpl::GET_HOST_SUBID_ADDR, bind_array, ctx->host_ipv4_exchange_, collection, false); return (collection); } ConstHostPtr PgSqlHostDataSource::get6(const SubnetID& subnet_id, const Host::IdentifierType& identifier_type, const uint8_t* identifier_begin, const size_t identifier_len) const { // Get a context PgSqlHostContextAlloc get_context(*impl_); PgSqlHostContextPtr ctx = get_context.ctx_; return (impl_->getHost(ctx, subnet_id, identifier_type, identifier_begin, identifier_len, PgSqlHostDataSourceImpl::GET_HOST_SUBID6_DHCPID, ctx->host_ipv6_exchange_)); } ConstHostPtr PgSqlHostDataSource::get6(const asiolink::IOAddress& prefix, const uint8_t prefix_len) const { if (!prefix.isV6()) { isc_throw(BadValue, "PgSqlHostDataSource::get6(prefix, prefix_len): " "wrong address type, address supplied is an IPv4 address"); } // Get a context PgSqlHostContextAlloc get_context(*impl_); PgSqlHostContextPtr ctx = get_context.ctx_; // Set up the WHERE clause value PsqlBindArrayPtr bind_array(new PsqlBindArray()); // Add the prefix bind_array->add(prefix); // Add the prefix length bind_array->add(prefix_len); ConstHostCollection collection; impl_->getHostCollection(ctx, PgSqlHostDataSourceImpl::GET_HOST_PREFIX, bind_array, ctx->host_ipv6_exchange_, collection, true); // Return single record if present, else clear the host. ConstHostPtr result; if (!collection.empty()) { result = *collection.begin(); } return (result); } ConstHostPtr PgSqlHostDataSource::get6(const SubnetID& subnet_id, const asiolink::IOAddress& address) const { if (!address.isV6()) { isc_throw(BadValue, "PgSqlHostDataSource::get6(id, address): " "wrong address type, address supplied is an IPv4 address"); } // Get a context PgSqlHostContextAlloc get_context(*impl_); PgSqlHostContextPtr ctx = get_context.ctx_; // Set up the WHERE clause value PsqlBindArrayPtr bind_array(new PsqlBindArray()); // Add the subnet id bind_array->add(subnet_id); // Add the prefix bind_array->add(address); ConstHostCollection collection; impl_->getHostCollection(ctx, PgSqlHostDataSourceImpl::GET_HOST_SUBID6_ADDR, bind_array, ctx->host_ipv6_exchange_, collection, true); // Return single record if present, else clear the host. ConstHostPtr result; if (!collection.empty()) { result = *collection.begin(); } return (result); } ConstHostCollection PgSqlHostDataSource::getAll6(const SubnetID& subnet_id, const asiolink::IOAddress& address) const { if (!address.isV6()) { isc_throw(BadValue, "PgSqlHostDataSource::get6(id, address): " "wrong address type, address supplied is an IPv4 address"); } // Get a context PgSqlHostContextAlloc get_context(*impl_); PgSqlHostContextPtr ctx = get_context.ctx_; // Set up the WHERE clause value PsqlBindArrayPtr bind_array(new PsqlBindArray()); // Add the subnet id bind_array->add(subnet_id); // Add the prefix bind_array->add(address); ConstHostCollection collection; impl_->getHostCollection(ctx, PgSqlHostDataSourceImpl::GET_HOST_SUBID6_ADDR, bind_array, ctx->host_ipv6_exchange_, collection, false); return (collection); } ConstHostCollection PgSqlHostDataSource::getAll6(const IOAddress& address) const { if (!address.isV6()) { isc_throw(BadValue, "PgSqlHostDataSource::get6(address): " "wrong address type, address supplied is an IPv4 address"); } // Get a context PgSqlHostContextAlloc get_context(*impl_); PgSqlHostContextPtr ctx = get_context.ctx_; // Set up the WHERE clause value PsqlBindArrayPtr bind_array(new PsqlBindArray()); // Add the prefix bind_array->add(address); ConstHostCollection collection; impl_->getHostCollection(ctx, PgSqlHostDataSourceImpl::GET_HOST_ADDR6, bind_array, ctx->host_ipv6_exchange_, collection, false); return (collection); } void PgSqlHostDataSource::update(HostPtr const& host) { // Get a context. PgSqlHostContextAlloc const context(*impl_); PgSqlHostContextPtr ctx(context.ctx_); // If operating in read-only mode, throw exception. impl_->checkReadOnly(ctx); // Initiate PostgreSQL transaction as we will have to make multiple queries // to update host information into multiple tables. If that fails on // any stage, the transaction will be rolled back by the destructor of // the PgSqlTransaction class. PgSqlTransaction transaction(ctx->conn_); // As much as having dedicated prepared statements for updating tables would be consistent with // the implementation of other commands, it's difficult if not impossible to cover all cases for // updating the host to exactly as is described in the command, which may involve inserts and // deletes alongside updates. So let's delete and add. The delete cascades into all tables. The // add explicitly adds into all tables. BaseHostDataSource::update(host); // Everything went fine, so explicitly commit the transaction. transaction.commit(); } // Miscellaneous database methods. std::string PgSqlHostDataSource::getName() const { std::string name = ""; // Get a context PgSqlHostContextAlloc get_context(*impl_); PgSqlHostContextPtr ctx = get_context.ctx_; try { name = ctx->conn_.getParameter("name"); } catch (...) { // Return an empty name } return (name); } std::string PgSqlHostDataSource::getDescription() const { return (std::string("Host data source that stores host information" "in PostgreSQL database")); } std::pair PgSqlHostDataSource::getVersion() const { return(impl_->getVersion()); } void PgSqlHostDataSource::commit() { // Get a context PgSqlHostContextAlloc get_context(*impl_); PgSqlHostContextPtr ctx = get_context.ctx_; // If operating in read-only mode, throw exception. impl_->checkReadOnly(ctx); ctx->conn_.commit(); } void PgSqlHostDataSource::rollback() { // Get a context PgSqlHostContextAlloc get_context(*impl_); PgSqlHostContextPtr ctx = get_context.ctx_; // If operating in read-only mode, throw exception. impl_->checkReadOnly(ctx); ctx->conn_.rollback(); } bool PgSqlHostDataSource::setIPReservationsUnique(const bool unique) { impl_->ip_reservations_unique_ = unique; return (true); } bool PgSqlHostDataSource::isUnusable() { return (impl_->unusable_); } } // namespace dhcp } // namespace isc