Kea 3.1.1
d2_client_mgr.cc
Go to the documentation of this file.
1// Copyright (C) 2014-2025 Internet Systems Consortium, Inc. ("ISC")
2//
3// This Source Code Form is subject to the terms of the Mozilla Public
4// License, v. 2.0. If a copy of the MPL was not distributed with this
5// file, You can obtain one at http://mozilla.org/MPL/2.0/.
6
7#include <config.h>
8
9#include <dhcp/iface_mgr.h>
10#include <dhcp_ddns/ncr_udp.h>
12#include <dhcpsrv/dhcpsrv_log.h>
13
14#include <functional>
15#include <string>
16
17using namespace std;
18
19namespace isc {
20namespace dhcp {
21
22D2ClientMgr::D2ClientMgr() : d2_client_config_(new D2ClientConfig()),
23 name_change_sender_(), private_io_service_(),
24 registered_select_fd_(util::WatchSocket::SOCKET_NOT_VALID) {
25 // Default constructor initializes with a disabled configuration.
26}
27
32
33void
35 if (ddnsEnabled()) {
39 d2_client_config_->enableUpdates(false);
40 if (name_change_sender_) {
41 stopSender();
42 }
43 }
44}
45
46void
48 if (!new_config) {
50 "D2ClientMgr cannot set DHCP-DDNS configuration to NULL.");
51 }
52
53 // Don't do anything unless configuration values are actually different.
54 if (*d2_client_config_ != *new_config) {
55 // Make sure we stop sending first.
56 stopSender();
57 if (!new_config->getEnableUpdates()) {
58 // Updating has been turned off.
59 // Destroy current sender (any queued requests are tossed).
60 name_change_sender_.reset();
61 } else {
63 switch (new_config->getNcrProtocol()) {
64 case dhcp_ddns::NCR_UDP: {
65 // Instantiate a new sender.
66 new_sender.reset(new dhcp_ddns::NameChangeUDPSender(
67 new_config->getSenderIp(),
68 new_config->getSenderPort(),
69 new_config->getServerIp(),
70 new_config->getServerPort(),
71 new_config->getNcrFormat(),
72 shared_from_this(),
73 new_config->getMaxQueueSize()));
74 break;
75 }
76 default:
77 // In theory you can't get here.
78 isc_throw(D2ClientError, "Invalid sender Protocol: "
79 << new_config->getNcrProtocol());
80 break;
81 }
82
83 // Transfer queued requests from previous sender to the new one.
89 if (name_change_sender_) {
90 new_sender->assumeQueue(*name_change_sender_);
91 }
92
93 // Replace the old sender with the new one.
94 name_change_sender_ = new_sender;
95 }
96 }
97
98 // Update the configuration.
99 d2_client_config_ = new_config;
101 .arg(!ddnsEnabled() ? "DHCP-DDNS updates disabled" :
102 "DHCP_DDNS updates enabled");
103}
104
105bool
107 return (d2_client_config_->getEnableUpdates());
108}
109
112 return (d2_client_config_);
113}
114
115void
116D2ClientMgr::analyzeFqdn(const bool client_s, const bool client_n,
117 bool& server_s, bool& server_n,
118 const DdnsParams& ddns_params) const {
119 // Per RFC 4702 & 4704, the client N and S flags allow the client to
120 // request one of three options:
121 //
122 // N flag S flag Option
123 // ------------------------------------------------------------------
124 // 0 0 client wants to do forward updates (section 3.2)
125 // 0 1 client wants server to do forward updates (section 3.3)
126 // 1 0 client wants no one to do updates (section 3.4)
127 // 1 1 invalid combination
128 // (Note section numbers cited are for 4702, for 4704 see 5.1, 5.2, and 5.3)
129 //
130 // Make a bit mask from the client's flags and use it to set the response
131 // flags accordingly.
132 const uint8_t mask = ((client_n ? 2 : 0) + (client_s ? 1 : 0));
133
134 switch (mask) {
135 case 0:
136 if (!ddns_params.getEnableUpdates()) {
137 server_s = false;
138 server_n = true;
139 } else {
140 // If updates are enabled and we are overriding client delegation
141 // then S flag should be true. N-flag should be false.
142 server_s = ddns_params.getOverrideClientUpdate();
143 server_n = false;
144 }
145 break;
146
147 case 1:
148 server_s = ddns_params.getEnableUpdates();
149 server_n = !server_s;
150 break;
151
152 case 2:
153 // If updates are enabled and we are overriding "no updates" then
154 // S flag should be true.
155 server_s = (ddns_params.getEnableUpdates() &&
156 ddns_params.getOverrideNoUpdate());
157 server_n = !server_s;
158 break;
159
160 default:
161 // RFCs declare this an invalid combination.
163 "Invalid client FQDN - N and S cannot both be 1");
164 break;
165 }
166}
167
168std::string
170 const DdnsParams& ddns_params,
171 const bool trailing_dot) const {
172 std::string hostname = address.toText();
173 std::replace(hostname.begin(), hostname.end(),
174 (address.isV4() ? '.' : ':'), '-');
175
176 if (*(hostname.rbegin()) == '-') {
177 hostname.append("0");
178 }
179
180 std::ostringstream gen_name;
181 gen_name << ddns_params.getGeneratedPrefix() << "-" << hostname;
182 return (qualifyName(gen_name.str(), ddns_params, trailing_dot));
183}
184
185std::string
186D2ClientMgr::qualifyName(const std::string& partial_name,
187 const DdnsParams& ddns_params,
188 const bool trailing_dot) const {
189 std::ostringstream gen_name;
190 gen_name << partial_name;
191 std::string suffix = ddns_params.getQualifyingSuffix();
192 if (!suffix.empty() && partial_name.back() != '.') {
193 bool suffix_present = true;
194 std::string str = gen_name.str();
195 auto suffix_rit = suffix.rbegin();
196 if (*suffix_rit == '.') {
197 ++suffix_rit;
198 }
199
200 auto gen_rit = str.rbegin();
201 while (suffix_rit != suffix.rend()) {
202 if ((gen_rit == str.rend()) || (*suffix_rit != *gen_rit)) {
203 // They don't match.
204 suffix_present = false;
205 break;
206 }
207
208 ++suffix_rit;
209 ++gen_rit;
210 }
211
212 // Catch the case where name has suffix embedded.
213 // input: foo.barexample.com suffix: example.com
214 if ((suffix_present) && (suffix_rit == suffix.rend())) {
215 if ((gen_rit != str.rend()) && (*gen_rit != '.')) {
216 suffix_present = false;
217 }
218 }
219
220 if (!suffix_present) {
221 size_t len = str.length();
222 if ((len > 0) && (str[len - 1] != '.')) {
223 gen_name << ".";
224 }
225
226 gen_name << suffix;
227 }
228 }
229
230 std::string str = gen_name.str();
231 size_t len = str.length();
232
233 if (trailing_dot) {
234 // If trailing dot should be added but there is no trailing dot,
235 // append it.
236 if ((len > 0) && (str[len - 1] != '.')) {
237 gen_name << ".";
238 }
239
240 } else {
241 // If the trailing dot should not be appended but it is present,
242 // remove it.
243 if ((len > 0) && (str[len - 1] == '.')) {
244 gen_name.str(str.substr(0,len-1));
245 }
246
247 }
248
249 return (gen_name.str());
250}
251
252void
254 if (amSending()) {
255 return;
256 }
257
258 // Create a our own service instance when we are not being multiplexed
259 // into an external service..
260 private_io_service_.reset(new asiolink::IOService());
261 startSender(error_handler, private_io_service_);
263 .arg(d2_client_config_->toText());
264}
265
266void
268 const isc::asiolink::IOServicePtr& io_service) {
269 if (amSending()) {
270 return;
271 }
272
273 if (!name_change_sender_) {
274 isc_throw(D2ClientError, "D2ClientMgr::startSender sender is null");
275 }
276
277 if (!error_handler) {
278 isc_throw(D2ClientError, "D2ClientMgr::startSender handler is null");
279 }
280
281 // Set the error handler.
282 client_error_handler_ = error_handler;
283
284 // Start the sender on the given service.
285 name_change_sender_->startSending(io_service);
286
287 // Register sender's select-fd with IfaceMgr.
288 // We need to remember the fd that is registered so we can unregister later.
289 // IO error handling in the sender may alter its select-fd.
290 registered_select_fd_ = name_change_sender_->getSelectFd();
291 IfaceMgr::instance().addExternalSocket(registered_select_fd_,
292 std::bind(&D2ClientMgr::runReadyIO,
293 this));
294}
295
296bool
298 return (name_change_sender_ && name_change_sender_->amSending());
299}
300
301void
304 if (registered_select_fd_ != util::WatchSocket::SOCKET_NOT_VALID) {
305 IfaceMgr::instance().deleteExternalSocket(registered_select_fd_);
306 registered_select_fd_ = util::WatchSocket::SOCKET_NOT_VALID;
307 }
308
309 // If its not null, call stop.
310 if (amSending()) {
311 name_change_sender_->stopSending();
313 }
314
315 if (private_io_service_) {
316 private_io_service_->stopAndPoll();
317 }
318}
319
320void
322 if (!amSending()) {
323 // This is programmatic error so bust them for it.
324 isc_throw(D2ClientError, "D2ClientMgr::sendRequest not in send mode");
325 }
326
327 try {
328 name_change_sender_->sendRequest(ncr);
329 } catch (const std::exception& ex) {
331 .arg(ex.what()).arg((ncr ? ncr->toText() : " NULL "));
333 }
334}
335
336void
338 Result result,
340 // Handler is mandatory to enter send mode but test it just to be safe.
341 if (!client_error_handler_) {
343 } else {
344 // Handler is not supposed to throw, but catch just in case.
345 try {
346 (client_error_handler_)(result, ncr);
347 } catch (const std::exception& ex) {
349 .arg(ex.what());
350 }
351 }
352}
353
354size_t
356 if (!name_change_sender_) {
357 isc_throw(D2ClientError, "D2ClientMgr::getQueueSize sender is null");
358 }
359
360 return (name_change_sender_->getQueueSize());
361}
362
363size_t
365 if (!name_change_sender_) {
366 isc_throw(D2ClientError, "D2ClientMgr::getQueueMaxSize sender is null");
367 }
368
369 return (name_change_sender_->getQueueMaxSize());
370}
371
373D2ClientMgr::peekAt(const size_t index) const {
374 if (!name_change_sender_) {
375 isc_throw(D2ClientError, "D2ClientMgr::peekAt sender is null");
376 }
377
378 return (name_change_sender_->peekAt(index));
379}
380
381void
383 if (!name_change_sender_) {
384 isc_throw(D2ClientError, "D2ClientMgr::clearQueue sender is null");
385 }
386
387 name_change_sender_->clearSendQueue();
388}
389
390void
400
401void
403 name_change_sender_.reset();
404}
405
406int
408 if (!amSending()) {
409 isc_throw (D2ClientError, "D2ClientMgr::getSelectFd "
410 " not in send mode");
411 }
412
413 return (name_change_sender_->getSelectFd());
414}
415
416void
418 if (!name_change_sender_) {
419 // This should never happen.
420 isc_throw(D2ClientError, "D2ClientMgr::runReadyIO"
421 " name_change_sender is null");
422 }
423
424 name_change_sender_->runReadyIO();
425}
426
427} // namespace dhcp
428} // namespace isc
A generic exception that is thrown if a parameter given to a method is considered invalid in that con...
Acts as a storage vault for D2 client configuration.
An exception that is thrown if an error occurs while configuring the D2 DHCP DDNS client.
void invokeClientErrorHandler(const dhcp_ddns::NameChangeSender::Result result, dhcp_ddns::NameChangeRequestPtr &ncr)
Calls the client's error handler.
std::string generateFqdn(const asiolink::IOAddress &address, const DdnsParams &ddns_params, const bool trailing_dot=true) const
Builds a FQDN based on the configuration and given IP address.
void analyzeFqdn(const bool client_s, const bool client_n, bool &server_s, bool &server_n, const DdnsParams &ddns_params) const
Determines server flags based on configuration and client flags.
bool ddnsEnabled()
Convenience method for checking if DHCP-DDNS is enabled.
const D2ClientConfigPtr & getD2ClientConfig() const
Fetches the DHCP-DDNS configuration pointer.
void stop()
Stop the sender.
void suspendUpdates()
Suspends sending requests.
void sendRequest(dhcp_ddns::NameChangeRequestPtr &ncr)
Send the given NameChangeRequests to kea-dhcp-ddns.
size_t getQueueSize() const
Returns the number of NCRs queued for transmission.
void clearQueue()
Removes all NCRs queued for transmission.
void stopSender()
Disables sending NameChangeRequests to kea-dhcp-ddns.
void setD2ClientConfig(D2ClientConfigPtr &new_config)
Updates the DHCP-DDNS client configuration to the given value.
bool amSending() const
Returns true if the sender is in send mode, false otherwise.
int getSelectFd()
Fetches the sender's select-fd.
void runReadyIO()
Processes sender IO events.
D2ClientMgr()
Constructor.
size_t getQueueMaxSize() const
Returns the maximum number of NCRs allowed in the queue.
virtual void operator()(const dhcp_ddns::NameChangeSender::Result result, dhcp_ddns::NameChangeRequestPtr &ncr)
Function operator implementing the NCR sender callback.
const dhcp_ddns::NameChangeRequestPtr & peekAt(const size_t index) const
Returns the nth NCR queued for transmission.
std::string qualifyName(const std::string &partial_name, const DdnsParams &ddns_params, const bool trailing_dot) const
Adds a qualifying suffix to a given domain name.
void startSender(D2ClientErrorHandler error_handler, const isc::asiolink::IOServicePtr &io_service)
Enables sending NameChangeRequests to kea-dhcp-ddns.
Convenience container for conveying DDNS behavioral parameters It is intended to be created per Packe...
Definition ddns_params.h:23
std::string getGeneratedPrefix() const
Returns the Prefix Kea should use when generating domain-names.
std::string getQualifyingSuffix() const
Returns the suffix Kea should use when to qualify partial domain-names.
bool getOverrideNoUpdate() const
Returns whether or not Kea should perform updates, even if client requested no updates.
bool getEnableUpdates() const
Returns whether or not DHCP DDNS updating is enabled.
bool getOverrideClientUpdate() const
Returns whether or not Kea should perform updates, even if client requested delegation.
void deleteExternalSocket(int socketfd)
Deletes external socket.
Definition iface_mgr.cc:352
static IfaceMgr & instance()
IfaceMgr is a singleton class.
Definition iface_mgr.cc:54
void addExternalSocket(int socketfd, SocketCallback callback)
Adds external socket and a callback.
Definition iface_mgr.cc:329
Abstract interface for sending NameChangeRequests.
Definition ncr_io.h:468
Result
Defines the outcome of an asynchronous NCR send.
Definition ncr_io.h:478
Provides the ability to send NameChangeRequests via UDP socket.
Definition ncr_udp.h:442
Provides an IO "ready" semaphore for use with select() or poll() WatchSocket exposes a single open fi...
static const int SOCKET_NOT_VALID
Value used to signify an invalid descriptor.
Defines the D2ClientMgr class.
#define isc_throw(type, stream)
A shortcut macro to insert known values into exception arguments.
#define LOG_ERROR(LOGGER, MESSAGE)
Macro to conveniently test error output and log it.
Definition macros.h:32
#define LOG_INFO(LOGGER, MESSAGE)
Macro to conveniently test info output and log it.
Definition macros.h:20
#define LOG_WARN(LOGGER, MESSAGE)
Macro to conveniently test warn output and log it.
Definition macros.h:26
#define LOG_DEBUG(LOGGER, LEVEL, MESSAGE)
Macro to conveniently test debug output and log it.
Definition macros.h:14
boost::shared_ptr< NameChangeSender > NameChangeSenderPtr
Defines a smart pointer to an instance of a sender.
Definition ncr_io.h:857
boost::shared_ptr< NameChangeRequest > NameChangeRequestPtr
Defines a pointer to a NameChangeRequest.
Definition ncr_msg.h:241
isc::log::Logger dhcpsrv_logger("dhcpsrv")
DHCP server library Logger.
Definition dhcpsrv_log.h:56
const isc::log::MessageID DHCPSRV_DHCP_DDNS_NCR_SENT
boost::shared_ptr< D2ClientConfig > D2ClientConfigPtr
Defines a pointer for D2ClientConfig instances.
const isc::log::MessageID DHCPSRV_DHCP_DDNS_SENDER_STOPPED
const isc::log::MessageID DHCPSRV_DHCP_DDNS_SUSPEND_UPDATES
const isc::log::MessageID DHCPSRV_CFGMGR_CFG_DHCP_DDNS
const isc::log::MessageID DHCPSRV_DHCP_DDNS_SENDER_STARTED
const int DHCPSRV_DBG_TRACE_DETAIL
Additional information.
Definition dhcpsrv_log.h:38
const isc::log::MessageID DHCPSRV_DHCP_DDNS_NCR_REJECTED
std::function< void(const dhcp_ddns::NameChangeSender::Result result, dhcp_ddns::NameChangeRequestPtr &ncr)> D2ClientErrorHandler
Defines the type for D2 IO error handler.
const int DHCPSRV_DBG_TRACE
DHCP server library logging levels.
Definition dhcpsrv_log.h:26
const isc::log::MessageID DHCPSRV_DHCP_DDNS_ERROR_EXCEPTION
const isc::log::MessageID DHCPSRV_DHCP_DDNS_HANDLER_NULL
Defines the logger used by the top-level component of kea-lfc.
This file provides UDP socket based implementation for sending and receiving NameChangeRequests.