8#include <kea_version.h>
115#include <boost/algorithm/string/erase.hpp>
116#include <boost/algorithm/string/join.hpp>
117#include <boost/algorithm/string/split.hpp>
118#include <boost/foreach.hpp>
119#include <boost/tokenizer.hpp>
133namespace ph = std::placeholders;
139 int hook_index_buffer6_receive_;
140 int hook_index_pkt6_receive_;
141 int hook_index_subnet6_select_;
142 int hook_index_leases6_committed_;
143 int hook_index_lease6_release_;
144 int hook_index_pkt6_send_;
145 int hook_index_buffer6_send_;
146 int hook_index_lease6_decline_;
147 int hook_index_host6_identifier_;
148 int hook_index_ddns6_update_;
149 int hook_index_addr6_register_;
185createStatusCode(
const Pkt6& pkt,
const uint16_t status_code,
186 const std::string& status_message) {
191 .arg(option_status->dataToText());
192 return (option_status);
210createStatusCode(
const Pkt6& pkt,
const Option6IA& ia,
const uint16_t status_code,
211 const std::string& status_message) {
217 .arg(option_status->dataToText());
218 return (option_status);
223std::set<std::string> dhcp6_statistics = {
225 "pkt6-solicit-received",
226 "pkt6-advertise-received",
227 "pkt6-request-received",
228 "pkt6-reply-received",
229 "pkt6-renew-received",
230 "pkt6-rebind-received",
231 "pkt6-decline-received",
232 "pkt6-release-received",
233 "pkt6-infrequest-received",
234 "pkt6-dhcpv4-query-received",
235 "pkt6-dhcpv4-response-received",
236 "pkt6-addr-reg-inform-received",
237 "pkt6-addr-reg-reply-received",
238 "pkt6-unknown-received",
240 "pkt6-advertise-sent",
242 "pkt6-dhcpv4-response-sent",
243 "pkt6-addr-reg-reply-sent",
244 "pkt6-service-disabled",
247 "v6-allocation-fail",
248 "v6-allocation-fail-shared-network",
249 "v6-allocation-fail-subnet",
250 "v6-allocation-fail-no-pools",
251 "v6-allocation-fail-classes",
252 "v6-ia-na-lease-reuses",
253 "v6-ia-pd-lease-reuses",
296 }
catch (
const std::exception &e) {
312 for (
auto const& it : dhcp6_statistics) {
314 stats_mgr.
setValue(it,
static_cast<int64_t
>(0));
324 }
catch (
const std::exception& ex) {
331 }
catch (
const std::exception& ex) {
350 if (!names.empty()) {
352 for (
size_t i = 1; i < names.size(); ++i) {
353 msg += std::string(
", ") + names[i];
359 io_service_->stopAndPoll();
385 if (
getServerID()->getData() != server_id->getData()){
387 .arg(pkt->getLabel())
399 switch (pkt->getType()) {
404 if (pkt->relay_info_.empty() && !pkt->getLocalAddr().isV6Multicast()) {
406 .arg(pkt->getLabel())
407 .arg(pkt->getName());
422 for (
auto const& id_type : cfg->getIdentifierTypes()) {
443 std::vector<uint8_t> id;
452 callout_handle->setArgument(
"query6", ctx.
query_);
453 callout_handle->setArgument(
"id_type", type);
454 callout_handle->setArgument(
"id_value",
id);
460 callout_handle->getArgument(
"id_type", type);
461 callout_handle->getArgument(
"id_value",
id);
467 .arg(ctx.
query_->getLabel())
487 ctx.
duid_ = query->getClientId();
514 if (global_host && !global_host->getClientClasses6().empty()) {
519 const ClientClasses& classes = global_host->getClientClasses6();
520 for (
auto const& cclass : classes) {
521 query->addClass(cclass);
530 query->addClass(
"KNOWN");
532 .arg(query->getLabel())
539 if (query->inClass(
"DROP")) {
542 .arg(query->makeLabel(query->getClientId(),
nullptr))
543 .arg(query->toText());
545 static_cast<int64_t
>(1));
550 ctx.
hosts_[SUBNET_ID_GLOBAL] = global_host;
584 ctx.
subnet_->getSharedNetwork(sn);
596 global_host && !global_host->getClientClasses6().empty()) ||
597 (!sn && current_host && !current_host->getClientClasses6().empty())) {
618 if (!ctx.
hosts_.empty()) {
619 ctx.
query_->addClass(
"KNOWN");
621 .arg(ctx.
query_->getLabel())
624 ctx.
query_->addClass(
"UNKNOWN");
626 .arg(ctx.
query_->getLabel())
635 .arg(ctx.
query_->getLabel())
639 if (ctx.
query_->inClass(
"DROP")) {
641 .arg(ctx.
query_->makeLabel(ctx.
query_->getClientId(), 0))
642 .arg(ctx.
query_->toText());
644 static_cast<int64_t
>(1));
656 const char*
interface = getenv(
"KEA_AFL_INTERFACE");
658 isc_throw(FuzzInitFail,
"no fuzzing interface has been set");
662 const char* address = getenv(
"KEA_AFL_ADDRESS");
664 isc_throw(FuzzInitFail,
"no fuzzing address has been set");
672 while (__AFL_LOOP(fuzzer.maxLoopCount())) {
684 }
catch (
const std::exception& e) {
715 uint32_t timeout = 1;
726 .arg(query->getRemoteAddr().toText())
727 .arg(query->getRemotePort())
728 .arg(query->getLocalAddr().toText())
729 .arg(query->getLocalPort())
730 .arg(query->getIface());
752 }
catch (
const std::exception& e) {
766 .arg(query->getLabel());
769 static_cast<int64_t
>(1));
771 static_cast<int64_t
>(1));
775 query->addPktEvent(
"mt_queued");
776 typedef function<void()> CallBack;
777 boost::shared_ptr<CallBack> call_back =
793 }
catch (
const std::exception& e) {
795 .arg(query->getLabel())
799 .arg(query->getLabel());
816 query->addPktEvent(
"process_started");
819 query->addClass(
"ALL");
821 bool skip_unpack =
false;
838 callout_handle->setArgument(
"query6", query);
849 .arg(query->getRemoteAddr().toText())
850 .arg(query->getLocalAddr().toText())
851 .arg(query->getIface());
860 .arg(query->getRemoteAddr().toText())
861 .arg(query->getLocalAddr().toText())
862 .arg(query->getIface());
872 callout_handle->getArgument(
"query6", query);
884 .arg(query->getRemoteAddr().toText())
885 .arg(query->getLocalAddr().toText())
886 .arg(query->getIface());
893 .arg(query->getLabel())
895 }
catch (
const std::exception &e) {
898 .arg(query->getLabel())
899 .arg(query->getRemoteAddr().toText())
900 .arg(query->getLocalAddr().toText())
901 .arg(query->getIface())
903 .arg(query->makeLabel(query->getClientId(),
nullptr));
907 static_cast<int64_t
>(1));
909 static_cast<int64_t
>(1));
916 .arg(query->getLabel());
919 processStatsReceived(query);
944 .arg(query->getLabel())
945 .arg(query->getName())
946 .arg(
static_cast<int>(query->getType()))
947 .arg(query->getRemoteAddr())
948 .arg(query->getLocalAddr())
949 .arg(query->getIface());
951 .arg(query->getLabel())
952 .arg(query->toText());
970 callout_handle->setArgument(
"query6", query);
981 .arg(query->getLabel());
990 callout_handle->getArgument(
"query6", query);
1003 if (query->inClass(
"DROP")) {
1005 .arg(query->makeLabel(query->getClientId(),
nullptr))
1006 .arg(query->toText());
1008 static_cast<int64_t
>(1));
1025 }
catch (
const std::exception& e) {
1027 .arg(query->getLabel())
1031 .arg(query->getLabel());
1052 if (!client_handler.
tryLock(query, cont)) {
1093 }
catch (
const std::exception& e) {
1095 .arg(query->getLabel())
1099 .arg(query->getLabel());
1112 callout_handle->getContext(
"subnet6", ctx.
subnet_);
1132 switch (query->getType()) {
1173 }
catch (
const std::exception& e) {
1183 .arg(query->getLabel())
1184 .arg(query->getName())
1185 .arg(query->getRemoteAddr().toText())
1208 rsp->setRemoteAddr(query->getRemoteAddr());
1209 rsp->setLocalAddr(query->getLocalAddr());
1214 }
else if (rsp->relay_info_.empty()) {
1216 rsp->setRemotePort(DHCP6_CLIENT_PORT);
1220 rsp->setRemotePort(relay_port ? relay_port : DHCP6_SERVER_PORT);
1226 rsp->setLocalPort(DHCP6_SERVER_PORT);
1228 rsp->setIndex(query->getIndex());
1229 rsp->setIface(query->getIface());
1258 std::shared_ptr<ScopedCalloutHandleState> callout_handle_state =
1259 std::make_shared<ScopedCalloutHandleState>(callout_handle);
1264 callout_handle->setArgument(
"query6", query);
1268 callout_handle->setArgument(
"response6", rsp);
1274 if (new_lease->reuseable_valid_lft_ == 0) {
1275 new_leases->push_back(new_lease);
1279 callout_handle->setArgument(
"leases6", new_leases);
1284 for (
auto const& iac : ctx.
ias_) {
1285 if (!iac.old_leases_.empty()) {
1286 for (
auto const& old_lease : iac.old_leases_) {
1288 deleted_leases->push_back(old_lease);
1291 bool in_new =
false;
1293 if ((new_lease->addr_ == old_lease->addr_) &&
1295 (new_lease->prefixlen_ == old_lease->prefixlen_))) {
1301 deleted_leases->push_back(old_lease);
1306 callout_handle->setArgument(
"deleted_leases6", deleted_leases);
1309 uint32_t parked_packet_limit = 0;
1313 parked_packet_limit = ppl->intValue();
1316 if (parked_packet_limit) {
1318 getParkingLotPtr(
"leases6_committed");
1319 if (parking_lot && (parking_lot->size() >= parked_packet_limit)) {
1323 .arg(parked_packet_limit)
1324 .arg(query->getLabel());
1326 static_cast<int64_t
>(1));
1338 [
this, callout_handle, query, rsp, callout_handle_state, subnet]()
mutable {
1340 typedef function<void()> CallBack;
1341 boost::shared_ptr<CallBack> call_back =
1343 this, callout_handle, query, rsp, subnet));
1344 callout_handle_state->on_completion_ = [call_back]() {
1365 .arg(query->getLabel());
1376 .arg(query->getLabel());
1397 }
catch (
const std::exception& e) {
1399 .arg(query->getLabel())
1403 .arg(query->getLabel());
1411 query->addPktEvent(
"process_completed");
1417 bool skip_pack =
false;
1435 callout_handle->setArgument(
"query6", query);
1438 callout_handle->setArgument(
"response6", rsp);
1441 callout_handle->setArgument(
"subnet6", subnet);
1453 .arg(rsp->getLabel());
1460 .arg(rsp->getLabel());
1470 }
catch (
const std::exception& e) {
1472 .arg(query->getLabel())
1504 callout_handle->setArgument(
"response6", rsp);
1517 .arg(rsp->getLabel());
1521 callout_handle->getArgument(
"response6", rsp);
1525 .arg(rsp->getLabel())
1526 .arg(rsp->getName())
1527 .arg(
static_cast<int>(rsp->getType()))
1528 .arg(rsp->getLocalAddr().isV6Zero() ?
"*" : rsp->getLocalAddr().toText())
1529 .arg(rsp->getLocalPort())
1530 .arg(rsp->getRemoteAddr())
1531 .arg(rsp->getRemotePort())
1532 .arg(rsp->getIface());
1535 .arg(rsp->getLabel())
1536 .arg(rsp->getName())
1537 .arg(
static_cast<int>(rsp->getType()))
1538 .arg(rsp->toText());
1544 }
catch (
const std::exception& e) {
1546 .arg(rsp->getLabel())
1558 for (
auto const& it :
data) {
1562 tmp << hex << setw(2) << setfill('0') << static_cast<uint16_t>(it);
1576 answer->addOption(clientid);
1581 if (!question->relay_info_.empty()) {
1582 answer->copyRelayInfo(question);
1600 co_list.push_back(ctx.
currentHost()->getCfgOption6());
1608 ctx.
subnet_->getPool(resource.getPrefixLength() == 128 ?
1610 resource.getAddress(),
1612 if (pool && !pool->getCfgOption()->empty()) {
1613 co_list.push_back(pool->getCfgOption());
1618 if (!ctx.
subnet_->getCfgOption()->empty()) {
1619 co_list.push_back(ctx.
subnet_->getCfgOption());
1624 ctx.
subnet_->getSharedNetwork(network);
1625 if (network && !network->getCfgOption()->empty()) {
1626 co_list.push_back(network->getCfgOption());
1632 for (
auto const& cclass : classes) {
1635 getClientClassDictionary()->findClass(cclass);
1640 .arg(question->getLabel())
1647 if (ccdef->getCfgOption()->empty()) {
1652 co_list.push_back(ccdef->getCfgOption());
1665 if (co_list.empty()) {
1669 set<uint16_t> requested_opts;
1678 for (uint16_t code : option_oro->getValues()) {
1679 static_cast<void>(requested_opts.insert(code));
1683 set<uint16_t> cancelled_opts;
1687 for (
auto const& copts : co_list) {
1695 BOOST_FOREACH(
auto const& desc, prange) {
1698 uint16_t code = desc.option_->getType();
1699 static_cast<void>(requested_opts.insert(code));
1705 BOOST_FOREACH(
auto const& desc, crange) {
1708 uint16_t code = desc.option_->getType();
1709 static_cast<void>(cancelled_opts.insert(code));
1714 const auto& cclasses = question->getClasses();
1717 for (uint16_t opt : requested_opts) {
1719 if (cancelled_opts.count(opt) > 0) {
1727 if (!answer->getOption(opt)) {
1729 for (
auto const& copts : co_list) {
1734 answer->addOption(desc.
option_);
1747 set<uint32_t> vendor_ids;
1751 vendor_class = boost::dynamic_pointer_cast<OptionVendorClass>(opt.second);
1753 uint32_t vendor_id = vendor_class->getVendorId();
1754 static_cast<void>(vendor_ids.insert(vendor_id));
1758 for (
auto const& copts : co_list) {
1761 if (!desc.option_ || !desc.allowedForClientClasses(cclasses)) {
1765 boost::dynamic_pointer_cast<OptionVendorClass>(desc.option_);
1766 if (!vendor_class) {
1770 uint32_t vendor_id = vendor_class->getVendorId();
1771 if (vendor_ids.count(vendor_id) > 0) {
1775 answer->addOption(desc.option_);
1776 static_cast<void>(vendor_ids.insert(vendor_id));
1785 set<uint32_t> vendor_ids;
1789 vendor_opts = boost::dynamic_pointer_cast<OptionVendor>(opt.second);
1791 uint32_t vendor_id = vendor_opts->getVendorId();
1792 static_cast<void>(vendor_ids.insert(vendor_id));
1796 for (
auto const& copts : co_list) {
1799 if (!desc.option_ || !desc.allowedForClientClasses(cclasses)) {
1803 boost::dynamic_pointer_cast<OptionVendor>(desc.option_);
1808 uint32_t vendor_id = vendor_opts->getVendorId();
1809 if (vendor_ids.count(vendor_id) > 0) {
1815 answer->addOption(vendor_opts);
1816 static_cast<void>(vendor_ids.insert(vendor_id));
1835 if (!ctx.
subnet_ || co_list.empty()) {
1839 set<uint32_t> vendor_ids;
1843 map<uint32_t, OptionVendorPtr> vendor_rsps;
1846 vendor_rsp = boost::dynamic_pointer_cast<OptionVendor>(opt.second);
1848 uint32_t vendor_id = vendor_rsp->getVendorId();
1849 vendor_rsps[vendor_id] = vendor_rsp;
1850 static_cast<void>(vendor_ids.insert(vendor_id));
1856 map<uint32_t, OptionVendorPtr> vendor_reqs;
1859 vendor_req = boost::dynamic_pointer_cast<OptionVendor>(opt.second);
1861 uint32_t vendor_id = vendor_req->getVendorId();
1862 vendor_reqs[vendor_id] = vendor_req;
1863 static_cast<void>(vendor_ids.insert(vendor_id));
1871 vendor_class = boost::dynamic_pointer_cast<OptionVendorClass>(opt.second);
1873 uint32_t vendor_id = vendor_class->getVendorId();
1874 static_cast<void>(vendor_ids.insert(vendor_id));
1880 if (vendor_ids.empty()) {
1884 map<uint32_t, set<uint16_t> > requested_opts;
1898 oro = boost::dynamic_pointer_cast<OptionUint16Array>(oro_generic);
1901 set<uint16_t> oro_req_opts;
1902 for (uint16_t code : oro->getValues()) {
1903 static_cast<void>(oro_req_opts.insert(code));
1909 map<uint32_t, set<uint16_t> > cancelled_opts;
1910 const auto& cclasses = question->getClasses();
1914 for (uint32_t vendor_id : vendor_ids) {
1915 for (
auto const& copts : co_list) {
1923 BOOST_FOREACH(
auto const& desc, prange) {
1924 if (!desc.option_) {
1928 uint16_t code = desc.option_->getType();
1929 static_cast<void>(requested_opts[vendor_id].insert(code));
1934 BOOST_FOREACH(
auto const& desc, crange) {
1935 if (!desc.option_) {
1939 uint16_t code = desc.option_->getType();
1940 static_cast<void>(cancelled_opts[vendor_id].insert(code));
1949 if (requested_opts[vendor_id].empty()) {
1956 if (vendor_rsps.count(vendor_id) > 0) {
1957 vendor_rsp = vendor_rsps[vendor_id];
1965 for (uint16_t opt : requested_opts[vendor_id]) {
1966 if (cancelled_opts[vendor_id].count(opt) > 0) {
1969 if (!vendor_rsp->getOption(opt)) {
1970 for (
auto const& copts : co_list) {
1975 vendor_rsp->addOption(desc.
option_);
1985 if (added && (vendor_rsps.count(vendor_id) == 0)) {
1986 answer->addOption(vendor_rsp);
1994 switch (pkt->getType()) {
2017 .arg(pkt->getLabel())
2018 .arg(
static_cast<int>(pkt->getType()))
2019 .arg(pkt->getIface());
2024 .arg(pkt->getLabel())
2025 .arg(pkt->getName())
2026 .arg(pkt->getRemoteAddr().toText())
2041 if (client_ids.size() != 1) {
2043 << pkt->getName() <<
", but " << client_ids.size()
2050 if (client_ids.size() > 1) {
2052 <<
") client-id options received in " << pkt->getName());
2054 if (!client_ids.empty()) {
2067 if (!server_ids.empty()) {
2069 << server_ids.size() <<
" received in " << pkt->getName());
2074 if (server_ids.size() != 1) {
2076 << server_ids.size() <<
"), exactly 1 expected in message "
2083 if (server_ids.size() > 1) {
2085 <<
") server-id options received in " << pkt->getName());
2087 if (!server_ids.empty()) {
2100 uint16_t len = opt->len() - opt->getHeaderLen();
2113 getCfgSubnets6()->selectSubnet(selector);
2123 shared_ptr<ScopedCalloutHandleState> callout_handle_state(
2124 std::make_shared<ScopedCalloutHandleState>(callout_handle));
2130 callout_handle->setArgument(
"query6", question);
2131 callout_handle->setArgument(
"subnet6", subnet);
2136 callout_handle->setArgument(
"subnet6collection",
2138 getCfgSubnets6()->getAll());
2140 auto const tpl(parkingLimitExceeded(
"subnet6_select"));
2141 bool const exceeded(
get<0>(tpl));
2143 uint32_t
const limit(
get<1>(tpl));
2148 .arg(question->getLabel());
2155 HooksManager::park(
"subnet6_select", question, [
this, question, callout_handle_state]() {
2157 boost::shared_ptr<function<void()>> callback(
2158 boost::make_shared<function<
void()>>([
this, question]()
mutable {
2161 callout_handle_state->on_completion_ = [callback]() {
2183 .arg(question->getLabel());
2196 .arg(question->getLabel());
2204 .arg(question->getLabel());
2210 callout_handle->getArgument(
"subnet6", subnet);
2216 .arg(question->getLabel())
2217 .arg(subnet->getID());
2221 .arg(question->getLabel())
2222 .arg(subnet->toText());
2226 .arg(question->getLabel());
2251 for (
auto const& opt : question->options_) {
2252 switch (opt.second->getType()) {
2255 boost::dynamic_pointer_cast<
2258 answer->addOption(answer_opt);
2264 boost::dynamic_pointer_cast<
2267 answer->addOption(answer_opt);
2293 if (ddns_params->getEnableUpdates() &&
2303 .arg(question->getLabel());
2316 .arg(question->getLabel())
2317 .arg(fqdn->toText());
2336 *ddns_params,
true),
2346 ctx.
hostname_ = fqdn_resp->getDomainName();
2353 .arg(question->getLabel())
2354 .arg(fqdn_resp->toText());
2355 answer->addOption(fqdn_resp);
2370 callout_handle->setArgument(
"query6", question);
2371 callout_handle->setArgument(
"response6", answer);
2372 callout_handle->setArgument(
"subnet6", subnet);
2373 callout_handle->setArgument(
"hostname", ctx.
hostname_);
2376 callout_handle->setArgument(
"ddns-params", ddns_params);
2382 string hook_hostname;
2383 bool hook_fwd_dns_update;
2384 bool hook_rev_dns_update;
2385 callout_handle->getArgument(
"hostname", hook_hostname);
2386 callout_handle->getArgument(
"fwd-update", hook_fwd_dns_update);
2387 callout_handle->getArgument(
"rev-update", hook_rev_dns_update);
2398 fqdn_resp = boost::dynamic_pointer_cast<Option6ClientFqdn>(answer->getOption(
D6O_CLIENT_FQDN));
2401 if (!(hook_fwd_dns_update || hook_rev_dns_update)) {
2428 <<
" encapsulating server's message must not be"
2429 <<
" NULL when creating DNS NameChangeRequest");
2442 bool do_fwd =
false;
2443 bool do_rev =
false;
2453 "client identifier is required when creating a new"
2454 " DNS NameChangeRequest");
2461 opt_fqdn->packDomainName(name_buf);
2462 const std::vector<uint8_t>& buf_vec = name_buf.
getVector();
2486 bool extended_only =
false;
2488 IOAddress ia_address = iaaddr->getAddress();
2489 for (
auto const& l : ia_ctx.reused_leases_) {
2490 if (l->addr_ == ia_address) {
2491 extended_only =
true;
2496 if (extended_only) {
2503 for (
auto const& l : ia_ctx.changed_leases_) {
2505 if (l->addr_ == ia_address) {
2509 if ((l->reuseable_valid_lft_ > 0) ||
2511 (l->hostname_ == opt_fqdn->getDomainName() &&
2512 l->fqdn_fwd_ == do_fwd && l->fqdn_rev_ == do_rev))) {
2513 extended_only =
true;
2525 if (!(do_fwd || do_rev) || (extended_only)) {
2541 opt_fqdn->getDomainName(),
2542 iaaddr->getAddress().toText(),
2551 .arg(answer->getLabel())
2552 .arg(ncr->toText());
2568 getMACSources().get();
2570 for (
auto const& it : mac_sources) {
2571 hwaddr = pkt->getMAC(it);
2587 BOOST_FOREACH(
auto const& resv, resvs) {
2588 if ((resv.second.getPrefix() == lease->addr_) &&
2589 (resv.second.getPrefixLen() == lease->prefixlen_)) {
2590 return (resv.second.getPDExclude());
2597 Pool6Ptr pool = boost::dynamic_pointer_cast<Pool6>(
2600 return (pool->getPrefixExcludeOption());
2608 boost::shared_ptr<Option6IA> ia) {
2614 boost::dynamic_pointer_cast<Option6IAAddr>(ia->getOption(
D6O_IAADDR));
2617 hint = hint_opt->getAddress();
2622 .arg(query->getLabel())
2624 .arg(hint_opt ? hint.
toText() :
"(no hint)");
2627 .arg(query->getLabel())
2629 .arg(hint_opt ? hint.
toText() :
"(no hint)");
2649 "Server could not select subnet for"
2672 if (!leases.empty()) {
2673 lease = *leases.begin();
2687 .arg(query->getLabel())
2688 .arg(lease->addr_.toText())
2689 .arg(ia->getIAID());
2690 }
else if (lease->reuseable_valid_lft_ == 0) {
2692 .arg(query->getLabel())
2693 .arg(lease->addr_.toText())
2697 lease->valid_lft_ = lease->reuseable_valid_lft_;
2698 lease->preferred_lft_ = lease->reuseable_preferred_lft_;
2701 .arg(query->getLabel())
2702 .arg(lease->addr_.toText())
2709 "v6-ia-na-lease-reuses"),
2713 .arg(query->getLabel())
2715 .arg(lease->toText());
2718 setTeeTimes(lease->preferred_lft_, subnet, ia_rsp);
2721 lease->preferred_lft_,
2722 lease->valid_lft_));
2723 ia_rsp->addOption(addr);
2736 .arg(query->getLabel())
2737 .arg(ia->getIAID());
2739 ia_rsp->addOption(createStatusCode(*query, *ia_rsp,
2741 "Sorry, no address could be"
2750 boost::shared_ptr<Option6IA> ia) {
2757 boost::dynamic_pointer_cast<Option6IAPrefix>(ia->getOption(
D6O_IAPREFIX));
2760 hint = hint_opt->getAddress();
2765 .arg(query->getLabel())
2767 .arg(hint_opt ? hint.
toText() :
"(no hint)");
2770 .arg(query->getLabel())
2772 .arg(hint_opt ? hint.
toText() :
"(no hint)");
2790 "Sorry, no subnet available."));
2811 if (!leases.empty()) {
2815 uint32_t min_preferred_lft = (*leases.begin())->preferred_lft_;
2817 const bool pd_exclude_requested = requestedInORO(query,
D6O_PD_EXCLUDE);
2818 for (
auto const& l : leases) {
2824 .arg(query->getLabel())
2825 .arg(l->addr_.toText())
2826 .arg(
static_cast<int>(l->prefixlen_))
2827 .arg(ia->getIAID());
2828 }
else if (l->reuseable_valid_lft_ == 0) {
2830 .arg(query->getLabel())
2831 .arg(l->addr_.toText())
2832 .arg(
static_cast<int>(l->prefixlen_))
2836 l->valid_lft_ = l->reuseable_valid_lft_;
2837 l->preferred_lft_ = l->reuseable_preferred_lft_;
2839 .arg(query->getLabel())
2840 .arg(l->addr_.toText())
2841 .arg(
static_cast<int>(l->prefixlen_))
2848 "v6-ia-pd-lease-reuses"),
2853 if ((l->preferred_lft_ > 0) && (min_preferred_lft > l->preferred_lft_)) {
2854 min_preferred_lft = l->preferred_lft_;
2857 boost::shared_ptr<Option6IAPrefix>
2859 l->prefixlen_, l->preferred_lft_,
2861 ia_rsp->addOption(addr);
2863 if (pd_exclude_requested) {
2865 if (pd_exclude_option) {
2866 addr->addOption(pd_exclude_option);
2885 .arg(query->getLabel())
2886 .arg(ia->getIAID());
2888 ia_rsp->addOption(createStatusCode(*query, *ia_rsp,
2890 "Sorry, no prefixes could"
2899 boost::shared_ptr<Option6IA> ia) {
2902 .arg(query->getLabel())
2903 .arg(ia->getIAID());
2922 "Sorry, no known leases for this duid/iaid."));
2934 for (
auto const& it : addrs) {
2938 Option6IAAddrPtr iaaddr = boost::dynamic_pointer_cast<Option6IAAddr>(it.second);
2965 uint32_t min_preferred_lft = std::numeric_limits<uint32_t>::max();
2968 for (
auto const& l : leases) {
2969 if (l->reuseable_valid_lft_ == 0) {
2971 .arg(query->getLabel())
2972 .arg(l->addr_.toText())
2973 .arg(ia->getIAID());
2975 l->valid_lft_ = l->reuseable_valid_lft_;
2976 l->preferred_lft_ = l->reuseable_preferred_lft_;
2978 .arg(query->getLabel())
2979 .arg(l->addr_.toText())
2986 "v6-ia-na-lease-reuses"),
2991 l->addr_, l->preferred_lft_, l->valid_lft_));
2992 ia_rsp->addOption(iaaddr);
2995 if ((l->preferred_lft_ > 0) && (min_preferred_lft > l->preferred_lft_)) {
2996 min_preferred_lft = l->preferred_lft_;
3001 hints.erase(std::remove(hints.begin(), hints.end(), hint_type),
3011 if (
equalValues(query->getClientId(), l->duid_)) {
3014 ia_rsp->addOption(iaaddr);
3019 hints.erase(std::remove(hints.begin(), hints.end(), hint_type), hints.end());
3027 .arg(query->getLabel())
3039 for (
auto const& hint : hints) {
3041 hint.getAddress(), 0, 0));
3042 ia_rsp->addOption(iaaddr);
3045 if (!leases.empty()) {
3051 ia_rsp->addOption(createStatusCode(*query, *ia_rsp,
3053 "Sorry, no addresses could be"
3054 " assigned at this time."));
3063 boost::shared_ptr<Option6IA> ia) {
3066 .arg(query->getLabel())
3067 .arg(ia->getIAID());
3084 "Sorry, no known PD leases"
3085 " for this duid/iaid."));
3105 " client sending Rebind to extend lifetime of the"
3106 " prefix (DUID=" << duid->toText() <<
", IAID="
3107 << ia->getIAID() <<
")");
3119 for (
auto const& it : addrs) {
3153 const bool pd_exclude_requested = requestedInORO(query,
D6O_PD_EXCLUDE);
3157 uint32_t min_preferred_lft = std::numeric_limits<uint32_t>::max();
3159 for (
auto const& l : leases) {
3160 if (l->reuseable_valid_lft_ == 0) {
3162 .arg(query->getLabel())
3163 .arg(l->addr_.toText())
3164 .arg(
static_cast<int>(l->prefixlen_))
3165 .arg(ia->getIAID());
3167 l->valid_lft_ = l->reuseable_valid_lft_;
3168 l->preferred_lft_ = l->reuseable_preferred_lft_;
3170 .arg(query->getLabel())
3171 .arg(l->addr_.toText())
3172 .arg(
static_cast<int>(l->prefixlen_))
3179 "v6-ia-pd-lease-reuses"),
3184 l->addr_, l->prefixlen_,
3185 l->preferred_lft_, l->valid_lft_));
3186 ia_rsp->addOption(prf);
3188 if (pd_exclude_requested) {
3190 if (pd_exclude_option) {
3191 prf->addOption(pd_exclude_option);
3196 if ((l->preferred_lft_ > 0) && (l->preferred_lft_ < min_preferred_lft)) {
3197 min_preferred_lft = l->preferred_lft_;
3202 hints.erase(std::remove(hints.begin(), hints.end(), hint_type),
3212 if (
equalValues(query->getClientId(), l->duid_)) {
3214 l->prefixlen_, 0, 0));
3215 ia_rsp->addOption(prefix);
3220 hints.erase(std::remove(hints.begin(), hints.end(), hint_type), hints.end());
3225 for (
auto const& prefix : hints) {
3230 if (!prefix.getAddress().isV6Zero()) {
3232 prefix.getAddress(),
3233 prefix.getPrefixLength(),
3235 ia_rsp->addOption(prefix_opt);
3239 if (!leases.empty()) {
3246 ia_rsp->addOption(createStatusCode(*query, *ia_rsp,
3248 "Sorry, no prefixes could be"
3249 " assigned at this time."));
3270 for (
auto const& opt : query->options_) {
3271 switch (opt.second->getType()) {
3274 boost::dynamic_pointer_cast<
3277 reply->addOption(answer_opt);
3284 boost::dynamic_pointer_cast<
3287 reply->addOption(answer_opt);
3323 for (
auto const& opt : release->options_) {
3325 switch (opt.second->getType()) {
3328 boost::dynamic_pointer_cast<Option6IA>(opt.second),
3331 reply->addOption(answer_opt);
3337 boost::dynamic_pointer_cast<Option6IA>(opt.second),
3340 reply->addOption(answer_opt);
3357 reply->addOption(createStatusCode(*release, general_status,
3358 "Summary status for all processed IA_NAs"));
3363 int& general_status, boost::shared_ptr<Option6IA> ia,
3367 .arg(query->getLabel())
3368 .arg(ia->getIAID());
3384 if (!release_addr) {
3386 "You did not include an address in your RELEASE"));
3392 release_addr->getAddress());
3399 "Sorry, no known leases for this duid/iaid, can't release."));
3405 if (!lease->duid_) {
3411 .arg(query->getLabel())
3412 .arg(release_addr->getAddress().toText());
3416 "Database consistency check failed when trying to RELEASE"));
3420 if (*duid != *(lease->duid_)) {
3424 .arg(query->getLabel())
3425 .arg(release_addr->getAddress().toText())
3426 .arg(lease->duid_->toText());
3430 "This address does not belong to you, you can't release it"));
3434 if (ia->getIAID() != lease->iaid_) {
3437 .arg(query->getLabel())
3438 .arg(release_addr->getAddress().toText())
3440 .arg(ia->getIAID());
3442 "This is your address, but you used wrong IAID"));
3465 callout_handle->deleteAllArguments();
3468 callout_handle->setArgument(
"query6", query);
3471 callout_handle->setArgument(
"lease6", lease);
3483 .arg(query->getLabel());
3488 bool success =
false;
3489 bool expired =
false;
3496 if (expiration_cfg->getFlushReclaimedTimerWaitTime() &&
3497 expiration_cfg->getHoldReclaimedTime() &&
3500 lease->valid_lft_ = 0;
3501 lease->preferred_lft_ = 0;
3521 "Server failed to release a lease"));
3524 .arg(query->getLabel())
3525 .arg(lease->addr_.toText())
3534 .arg(query->getLabel())
3535 .arg(lease->addr_.toText())
3538 ia_rsp->addOption(createStatusCode(*query, *ia_rsp,
STATUS_Success,
3539 "Lease released. Thank you, please come again."));
3543 .arg(query->getLabel())
3544 .arg(lease->addr_.toText())
3548 .arg(query->getLabel())
3549 .arg(lease->addr_.toText())
3563 static_cast<int64_t
>(-1));
3567 auto const& pool = subnet->getPool(
Lease::TYPE_NA, lease->addr_,
false);
3572 static_cast<int64_t
>(-1));
3582 int& general_status, boost::shared_ptr<Option6IA> ia,
3597 boost::shared_ptr<Option6IAPrefix> release_prefix =
3598 boost::dynamic_pointer_cast<Option6IAPrefix>(ia->getOption(
D6O_IAPREFIX));
3599 if (!release_prefix) {
3601 "You did not include a prefix in your RELEASE"));
3607 release_prefix->getAddress());
3614 "Sorry, no known leases for this duid/iaid, can't release."));
3620 if (!lease->duid_) {
3625 .arg(query->getLabel())
3626 .arg(release_prefix->getAddress().toText())
3627 .arg(
static_cast<int>(release_prefix->getLength()));
3631 "Database consistency check failed when trying to RELEASE"));
3635 if (*duid != *(lease->duid_)) {
3638 .arg(query->getLabel())
3639 .arg(release_prefix->getAddress().toText())
3640 .arg(
static_cast<int>(release_prefix->getLength()))
3641 .arg(lease->duid_->toText());
3645 "This address does not belong to you, you can't release it"));
3649 if (ia->getIAID() != lease->iaid_) {
3652 .arg(query->getLabel())
3653 .arg(release_prefix->getAddress().toText())
3654 .arg(
static_cast<int>(release_prefix->getLength()))
3656 .arg(ia->getIAID());
3658 "This is your address, but you used wrong IAID"));
3681 callout_handle->setArgument(
"query6", query);
3684 callout_handle->setArgument(
"lease6", lease);
3696 .arg(query->getLabel());
3701 bool success =
false;
3702 bool expired =
false;
3709 if (expiration_cfg->getFlushReclaimedTimerWaitTime() &&
3710 expiration_cfg->getHoldReclaimedTime() &&
3713 lease->valid_lft_ = 0;
3714 lease->preferred_lft_ = 0;
3734 "Server failed to release a lease"));
3737 .arg(query->getLabel())
3738 .arg(lease->addr_.toText())
3739 .arg(
static_cast<int>(lease->prefixlen_))
3747 .arg(query->getLabel())
3748 .arg(lease->addr_.toText())
3749 .arg(
static_cast<int>(lease->prefixlen_))
3752 ia_rsp->addOption(createStatusCode(*query, *ia_rsp,
STATUS_Success,
3753 "Lease released. Thank you, please come again."));
3757 .arg(query->getLabel())
3758 .arg(lease->addr_.toText())
3759 .arg(
static_cast<int>(lease->prefixlen_))
3763 .arg(query->getLabel())
3764 .arg(lease->addr_.toText())
3765 .arg(
static_cast<int>(lease->prefixlen_))
3774 static_cast<int64_t
>(-1));
3778 auto const& pool = subnet->getPool(
Lease::TYPE_PD, lease->addr_,
false);
3783 static_cast<int64_t
>(-1));
3800 if (opt_rapid_commit) {
3803 .arg(solicit->getLabel());
3808 response->addOption(opt_rapid_commit);
3841 .arg(solicit->getLabel())
3842 .arg(solicit->getName())
3843 .arg(solicit->getClasses().toText());
3852 updateReservedFqdn(ctx, response);
3885 .arg(request->getLabel())
3886 .arg(request->getName())
3887 .arg(request->getClasses().toText());
3896 updateReservedFqdn(ctx, reply);
3897 generateFqdn(reply, ctx);
3925 .arg(renew->getLabel())
3926 .arg(renew->getName())
3927 .arg(renew->getClasses().toText());
3936 updateReservedFqdn(ctx, reply);
3937 generateFqdn(reply, ctx);
3965 .arg(rebind->getLabel())
3966 .arg(rebind->getName())
3967 .arg(rebind->getClasses().toText());
3976 updateReservedFqdn(ctx, reply);
3977 generateFqdn(reply, ctx);
3992 .arg(confirm->getLabel())
3993 .arg(confirm->getName())
3994 .arg(confirm->getClasses().toText());
4015 bool verified =
false;
4024 for (
auto const& ia : ias) {
4026 for (
auto const& opt : opts) {
4038 if (subnet && !subnet->inRange(iaaddr->getAddress())) {
4039 std::ostringstream status_msg;
4040 status_msg <<
"Address " << iaaddr->getAddress()
4041 <<
" is not on link.";
4042 reply->addOption(createStatusCode(*confirm,
4050 " to the Option6IAAddrPtr. This is programming"
4051 " error and should be reported");
4068 "All addresses are on-link"));
4071 "No subnet selected"));
4086 .arg(release->getLabel())
4087 .arg(release->getName())
4088 .arg(release->getClasses().toText());
4118 .arg(decline->getLabel())
4119 .arg(decline->getName())
4120 .arg(decline->getClasses().toText());
4159 for (
auto const& opt : decline->options_) {
4160 switch (opt.second->getType()) {
4163 boost::dynamic_pointer_cast<Option6IA>(opt.second),
4168 reply->addOption(answer_opt);
4189 int& general_status, boost::shared_ptr<Option6IA> ia,
4193 .arg(decline->getLabel())
4194 .arg(ia->getIAID());
4209 int total_addrs = 0;
4210 for (
auto const& opt : opts) {
4219 if (!decline_addr) {
4226 decline_addr->getAddress());
4231 .arg(decline->getLabel()).arg(decline_addr->getAddress().toText());
4239 "Server does not know about such an address."));
4247 if (!lease->duid_) {
4253 .arg(decline->getLabel())
4254 .arg(decline_addr->getAddress().toText());
4257 "Database consistency check failed when attempting Decline."));
4263 if (*duid != *(lease->duid_)) {
4267 .arg(decline->getLabel())
4268 .arg(decline_addr->getAddress().toText())
4269 .arg(lease->duid_->toText());
4272 "This address does not belong to you, you can't decline it"));
4278 if (ia->getIAID() != lease->iaid_) {
4281 .arg(decline->getLabel())
4282 .arg(lease->addr_.toText())
4286 "This is your address, but you used wrong IAID"));
4298 new_leases.push_back(lease);
4302 if (total_addrs == 0) {
4304 "No addresses sent in IA_NA"));
4317 container->addOption(status);
4322 boost::shared_ptr<Option6IA> ia_rsp) {
4346 callout_handle->setArgument(
"query6", decline);
4349 callout_handle->setArgument(
"lease6", lease);
4360 .arg(decline->getLabel())
4361 .arg(decline->getIface())
4362 .arg(lease->addr_.toText());
4370 .arg(decline->getLabel())
4371 .arg(decline->getIface())
4372 .arg(lease->addr_.toText());
4377 Lease6Ptr old_values = boost::make_shared<Lease6>(*lease);
4391 .arg(decline->getLabel())
4392 .arg(lease->addr_.toText())
4405 static_cast<int64_t
>(1));
4409 auto const& pool = subnet->getPool(
Lease::TYPE_NA, lease->addr_,
false);
4414 static_cast<int64_t
>(1));
4422 .arg(lease->addr_.toText()).arg(lease->valid_lft_);
4424 ia_rsp->addOption(createStatusCode(*decline, *ia_rsp,
STATUS_Success,
4425 "Lease declined. Hopefully the next one will be better."));
4439 .arg(inf_request->getLabel())
4440 .arg(inf_request->getName())
4441 .arg(inf_request->getClasses().toText());
4502 IOAddress addr = addr_reg_inf->getRemoteAddr();
4505 size_t relay_level = addr_reg_inf->relay_info_.size();
4506 if (relay_level > 0) {
4507 addr = addr_reg_inf->getRelay6LinkAddress(relay_level - 1);
4512 const uint32_t no_iaid = 0;
4517 if (addr_reg_inf->getOption(
D6O_IA_NA)) {
4522 if (addr_reg_inf->getOption(
D6O_IA_TA)) {
4527 if (addr_reg_inf->getOption(
D6O_IA_PD)) {
4534 if (addrs.size() != 1) {
4536 << addrs.size() <<
" received");
4538 iaaddr = boost::dynamic_pointer_cast<Option6IAAddr>(addrs.begin()->second);
4544 if (addr != iaaddr->getAddress()) {
4546 <<
" wants to register " << iaaddr->getAddress());
4550 if (!subnet->inRange(addr)) {
4552 << subnet->
toText() <<
" (id " << subnet->getID() <<
")");
4567 if (!hosts.empty()) {
4570 }
catch (
const std::exception &ex) {
4580 if (old_lease->duid_ && (*ctx.
duid_ != *(old_lease->duid_))) {
4583 .arg(ctx.
duid_->toText())
4584 .arg(old_lease->duid_->toText());
4590 addr_reg_inf->getTransid()));
4591 addr_reg_rep->addOption(iaaddr);
4599 ia->addOption(iaaddr);
4606 no_iaid, iaaddr->getPreferred(),
4607 iaaddr->getValid(), subnet->getID(),
4619 .arg(addr_reg_inf->getLabel())
4620 .arg(addr_reg_inf->getName())
4621 .arg(addr_reg_inf->getClasses().toText());
4639 callout_handle->setArgument(
"query6", addr_reg_inf);
4643 callout_handle->setArgument(
"response6", addr_reg_rep);
4646 callout_handle->setArgument(
"address6", addr);
4649 callout_handle->setArgument(
"old_lease6", old_lease);
4652 callout_handle->setArgument(
"new_lease6", lease);
4662 .arg(addr_reg_inf->getLabel())
4663 .arg(old_lease ?
"update" :
"add")
4672 .arg(addr_reg_inf->getLabel())
4683 }
catch (
const std::exception& ex) {
4693 if (old_lease->subnet_id_ != lease->subnet_id_) {
4697 static_cast<int64_t
>(-1));
4701 static_cast<int64_t
>(1));
4704 "cumulative-registered-nas"),
4705 static_cast<int64_t
>(1));
4718 static_cast<int64_t
>(1));
4721 "cumulative-registered-nas"),
4722 static_cast<int64_t
>(1));
4724 static_cast<int64_t
>(1));
4731 updateReservedFqdn(ctx, addr_reg_rep);
4732 generateFqdn(addr_reg_rep, ctx);
4735 return (addr_reg_rep);
4738void Dhcpv6Srv::classifyByVendor(
const Pkt6Ptr& pkt) {
4741 vclass = boost::dynamic_pointer_cast<OptionVendorClass>(opt.second);
4742 if (!vclass || vclass->getTuplesNum() == 0) {
4760 pkt->addClass(
"ALL");
4763 classifyByVendor(pkt);
4774 for (
auto const& it : *defs_ptr) {
4782 if (it->getAdditional()) {
4786 if (it->getDependOnKnown() != depend_on_known) {
4789 it->test(pkt, expr_ptr);
4798 for (
auto const& def : *defs_ptr) {
4802 if (def->getMatchExpr()) {
4803 pkt->classes_.erase(def->getName());
4813 for (
auto const& cclass : classes) {
4814 pkt->addClass(cclass);
4824 ctx.
subnet_->getSharedNetwork(shared_network);
4825 if (shared_network) {
4827 if (host && (host->getIPv6SubnetID() != SUBNET_ID_GLOBAL)) {
4846 ctx.
subnet_->getPool(resource.getPrefixLength() == 128 ?
4848 resource.getAddress(),
4851 const ClientClasses& pool_to_add = pool->getAdditionalClasses();
4852 for (
auto const& cclass : pool_to_add) {
4859 const ClientClasses& to_add = subnet->getAdditionalClasses();
4860 for (
auto const& cclass : to_add) {
4866 subnet->getSharedNetwork(network);
4868 const ClientClasses& net_to_add = network->getAdditionalClasses();
4869 for (
auto const& cclass : net_to_add) {
4879 for (
auto const& cclass : classes) {
4894 pkt->addClass(cclass);
4902 .arg(pkt->getLabel())
4904 .arg(status ?
"true" :
"false");
4907 pkt->addClass(cclass);
4911 .arg(pkt->getLabel())
4923 " a message must not be NULL when updating reserved FQDN");
4934 std::string name = fqdn->getDomainName();
4946 if (new_name != name) {
4951 answer->addOption(fqdn);
4957Dhcpv6Srv::generateFqdn(
const Pkt6Ptr& answer,
4958 AllocEngine::ClientContext6& ctx) {
4960 isc_throw(isc::Unexpected,
"an instance of the object encapsulating"
4961 " a message must not be NULL when generating FQDN");
4972 if (!fqdn || !fqdn->getDomainName().empty()) {
4986 if (!iaaddr || iaaddr->getValid() == 0) {
4991 IOAddress addr = iaaddr->getAddress();
4992 std::string generated_name =
4996 .arg(answer->getLabel())
4997 .arg(generated_name);
5014 lease->hostname_ = generated_name;
5015 lease->reuseable_valid_lft_ = 0;
5019 isc_throw(isc::Unexpected,
"there is no lease in the database "
5020 " for address " << addr <<
", so as it is impossible"
5021 " to update FQDN data. This is a programmatic error"
5022 " as the given address is now being handed to the"
5030 answer->addOption(fqdn);
5032 }
catch (
const Exception& ex) {
5034 .arg(answer->getLabel())
5048 this, ph::_1, ph::_2));
5067 arg(result).arg((ncr ? ncr->toText() :
" NULL "));
5076 std::stringstream tmp;
5080 tmp <<
" (" << EXTENDED_VERSION <<
")" << endl;
5081 tmp <<
"premium: " << PREMIUM_EXTENDED_VERSION << endl;
5082 tmp <<
"linked with:" << endl;
5087 tmp << endl <<
"lease backends:";
5089 tmp << endl <<
"- " <<
version;
5094 tmp << endl <<
"host backends:";
5096 tmp << endl <<
"- " <<
version;
5101 tmp << endl <<
"forensic backends:";
5103 tmp << endl <<
"- " <<
version;
5114 if (query->relay_info_.empty()) {
5125 for (
int i = query->relay_info_.size(); i > 0 ; --i) {
5127 if (rsoo_container) {
5132 for (
auto const& opt : rsoo) {
5136 if (cfg_rsoo->enabled(opt.second->getType()) &&
5137 !rsp->getOption(opt.second->getType())) {
5138 rsp->addOption(opt.second);
5147 if (query->relay_info_.empty()) {
5155 return (query->getRemotePort());
5161void Dhcpv6Srv::processStatsReceived(
const Pkt6Ptr& query) {
5165 string stat_name =
"pkt6-unknown-received";
5166 switch (query->getType()) {
5168 stat_name =
"pkt6-solicit-received";
5172 stat_name =
"pkt6-advertise-received";
5175 stat_name =
"pkt6-request-received";
5178 stat_name =
"pkt6-confirm-received";
5181 stat_name =
"pkt6-renew-received";
5184 stat_name =
"pkt6-rebind-received";
5188 stat_name =
"pkt6-reply-received";
5191 stat_name =
"pkt6-release-received";
5194 stat_name =
"pkt6-decline-received";
5197 stat_name =
"pkt6-reconfigure-received";
5200 stat_name =
"pkt6-infrequest-received";
5203 stat_name =
"pkt6-dhcpv4-query-received";
5207 stat_name =
"pkt6-dhcpv4-response-received";
5210 stat_name =
"pkt6-addr-reg-inform-received";
5214 stat_name =
"pkt6-addr-reg-reply-received";
5229 switch (response->getType()) {
5231 stat_name =
"pkt6-advertise-sent";
5234 stat_name =
"pkt6-reply-sent";
5237 stat_name =
"pkt6-dhcpv4-response-sent";
5240 stat_name =
"pkt6-addr-reg-reply-sent";
5251 return (
Hooks.hook_index_buffer6_send_);
5255Dhcpv6Srv::requestedInORO(
const Pkt6Ptr& query,
const uint16_t code)
const {
5257 boost::dynamic_pointer_cast<OptionUint16Array>(query->getOption(
D6O_ORO));
5260 const std::vector<uint16_t>& codes = oro->getValues();
5261 return (std::find(codes.begin(), codes.end(), code) != codes.end());
5267tuple<bool, uint32_t>
5268Dhcpv6Srv::parkingLimitExceeded(
string const& hook_label) {
5270 uint32_t parked_packet_limit(0);
5274 parked_packet_limit = ppl->intValue();
5277 if (parked_packet_limit) {
5281 if (parking_lot && parked_packet_limit <= parking_lot->size()) {
5282 return make_tuple(
true, parked_packet_limit);
5285 return make_tuple(
false, parked_packet_limit);
5296 char const*
const rotate(getenv(
"KEA_DHCP6_FUZZING_ROTATE_PORT"));
5300 while (!locker.
lock()) {
5301 this_thread::sleep_for(1s);
5304 port_file.open(
"/tmp/port6.txt", ios::in);
5307 getline(port_file, line);
5319 port_file.open(
"/tmp/port6.txt", ios::out | ios::trunc);
5320 port_file << to_string(port) << endl;
5335 uint32_t t2_time = 0;
5338 if (!subnet->getT2().unspecified()) {
5339 t2_time = subnet->getT2();
5340 }
else if (subnet->getCalculateTeeTimes()) {
5342 t2_time =
static_cast<uint32_t
>(round(subnet->getT2Percent() * preferred_lft));
5346 resp->setT2(t2_time);
5349 uint32_t t1_time = 0;
5352 if (!subnet->getT1().unspecified()) {
5353 t1_time = subnet->getT1();
5354 }
else if (subnet->getCalculateTeeTimes()) {
5356 t1_time =
static_cast<uint32_t
>(round(subnet->getT1Percent() * preferred_lft));
5360 if (t1_time < t2_time) {
5361 resp->setT1(t1_time);
5372 bool reprocess_client_name =
false;
5379 if (iaaddr && (iaaddr->getValid() > 0)) {
5381 auto pool = ddns_params->setPoolFromAddress(iaaddr->getAddress());
5384 reprocess_client_name = pool->hasDdnsParameters();
5390 if (ctx.
subnet_ && orig_subnet && (orig_subnet->getID() != ctx.
subnet_->getID())) {
5395 orig_subnet->getSharedNetwork(network);
5397 .arg(question->getLabel())
5398 .arg(orig_subnet->toText())
5400 .arg(network ? network->getName() :
"<no network?>");
5403 reprocess_client_name =
true;
5408 if (reprocess_client_name) {
5410 std::string prev_hostname = ctx.
hostname_;
5431 l->reuseable_valid_lft_ = 0;
5439 static std::list<std::list<std::string>>
const list({
5440 {
"config-control",
"config-databases",
"[]"},
5441 {
"hooks-libraries",
"[]",
"parameters",
"*"},
5443 {
"hosts-databases",
"[]"},
Defines elements for storing the names of client classes.
This is a base class for exceptions thrown from the DNS library module.
virtual const char * what() const
Returns a C-style character string of the cause of the exception.
A generic exception that is thrown when an unexpected error condition occurs.
The IOAddress class represents an IP addresses (version agnostic)
std::string toText() const
Convert the address to a string.
static const IOAddress & IPV6_ZERO_ADDRESS()
Returns an IPv6 zero address.
void clearIOServices()
Clear the list of IOService objects.
static IOServiceMgr & instance()
Access the IOServiceMgr singleton instance.
size_t pollIOServices()
Poll IOService objects.
The IOService class is a wrapper for the ASIO io_context class.
static std::string getVersion()
Get version string.
DHCPv4 and DHCPv6 allocation engine.
std::vector< Resource > HintContainer
Container for client's hints.
Implementation of the mechanisms to control the use of the Configuration Backends by the DHCPv6 serve...
@ EARLY_GLOBAL_RESERVATIONS_LOOKUP
D2ClientMgr & getD2ClientMgr()
Fetches the DHCP-DDNS manager.
static CfgMgr & instance()
returns a single instance of Configuration Manager
SrvConfigPtr getCurrentCfg()
Returns a pointer to the current configuration.
static SubnetSelector initSelector(const Pkt6Ptr &query)
Build selector from a client's message.
Container for storing client class names.
void insert(const ClientClass &class_name)
Insert an element.
std::string toText(const std::string &separator=", ") const
Returns all class names as text.
Client race avoidance RAII handler.
bool tryLock(Pkt4Ptr query, ContinuationPtr cont=ContinuationPtr())
Tries to acquires a client.
D2ClientMgr isolates Kea from the details of being a D2 client.
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.
bool ddnsEnabled()
Convenience method for checking if DHCP-DDNS is enabled.
void getUpdateDirections(const T &fqdn_resp, bool &forward, bool &reverse)
Get directional update flags based on server FQDN flags.
void stop()
Stop the sender.
void suspendUpdates()
Suspends sending requests.
void adjustDomainName(const T &fqdn, T &fqdn_resp, const DdnsParams &ddns_params)
Set server FQDN name based on configuration and a given FQDN.
void sendRequest(dhcp_ddns::NameChangeRequestPtr &ncr)
Send the given NameChangeRequests to kea-dhcp-ddns.
void stopSender()
Disables sending NameChangeRequests to kea-dhcp-ddns.
void adjustFqdnFlags(const T &fqdn, T &fqdn_resp, const DdnsParams &ddns_params)
Set server FQDN flags based on configuration and a given FQDN.
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.
This exception is thrown when DHCP server hits the error which should result in discarding the messag...
Factory for generating DUIDs (DHCP Unique Identifiers).
DuidPtr get()
Returns current DUID.
Holds DUID (DHCPv6 Unique Identifier)
static constexpr size_t MIN_DUID_LEN
minimum duid size
static constexpr size_t MAX_DUID_LEN
maximum duid size
void send(const Pkt6Ptr &pkt)
Send message over IPC.
void close()
Close communication socket.
static Dhcp6to4Ipc & instance()
Returns pointer to the sole instance of Dhcp6to4Ipc.
static uint16_t client_port
std::queue< isc::dhcp_ddns::NameChangeRequest > name_change_reqs_
Holds a list of isc::dhcp_ddns::NameChangeRequest objects, which are waiting for sending to kea-dhcp-...
void shutdown() override
Instructs the server to shut down.
RequirementLevel
defines if certain option may, must or must not appear
OptionPtr getServerID()
Returns server-identifier option.
Pkt6Ptr processPacket(Pkt6Ptr query)
Process a single incoming DHCPv6 packet.
Pkt6Ptr processLocalizedQuery6(AllocEngine::ClientContext6 &ctx)
Process a localized incoming DHCPv6 query.
void processPacketAndSendResponseNoThrow(Pkt6Ptr query)
Process a single incoming DHCPv6 packet and sends the response.
OptionPtr extendIA_PD(const Pkt6Ptr &query, AllocEngine::ClientContext6 &ctx, Option6IAPtr ia)
Extends lifetime of the prefix.
void setReservedClientClasses(const Pkt6Ptr &pkt, const AllocEngine::ClientContext6 &ctx)
Assigns classes retrieved from host reservation database.
Pkt6Ptr processDecline(AllocEngine::ClientContext6 &ctx)
Process incoming Decline message.
void evaluateClasses(const Pkt6Ptr &pkt, bool depend_on_known)
Evaluate classes.
Pkt6Ptr processRenew(AllocEngine::ClientContext6 &ctx)
Processes incoming Renew message.
static void processStatsSent(const Pkt6Ptr &response)
Updates statistics for transmitted packets.
void evaluateAdditionalClasses(const Pkt6Ptr &pkt, AllocEngine::ClientContext6 &ctx)
Evaluates classes in the additional classes lists.
void processLocalizedQuery6AndSendResponse(Pkt6Ptr query, AllocEngine::ClientContext6 &ctx)
Process a localized incoming DHCPv6 query.
int run()
Main server processing loop.
void setPacketStatisticsDefaults()
This function sets statistics related to DHCPv6 packets processing to their initial values.
bool sanityCheck(const Pkt6Ptr &pkt)
Verifies if specified packet meets RFC requirements.
static uint16_t checkRelaySourcePort(const Pkt6Ptr &query)
Used for DHCPv4-over-DHCPv6 too.
void assignLeases(const Pkt6Ptr &question, Pkt6Ptr &answer, AllocEngine::ClientContext6 &ctx)
Assigns leases.
void stopD2()
Stops DHCP_DDNS client IO if DDNS updates are enabled.
void copyClientOptions(const Pkt6Ptr &question, Pkt6Ptr &answer)
Copies required options from client message to server answer.
boost::shared_ptr< AllocEngine > alloc_engine_
Allocation Engine.
virtual void sendPacket(const Pkt6Ptr &pkt)
dummy wrapper around IfaceMgr::send()
bool testServerID(const Pkt6Ptr &pkt)
Compare received server id with our server id.
virtual void d2ClientErrorHandler(const dhcp_ddns::NameChangeSender::Result result, dhcp_ddns::NameChangeRequestPtr &ncr)
Implements the error handler for DHCP_DDNS IO errors.
OptionPtr declineIA(const Pkt6Ptr &decline, const DuidPtr &duid, int &general_status, boost::shared_ptr< Option6IA > ia, Lease6Collection &new_leases)
Declines leases in a single IA_NA option.
uint16_t getServerPort() const
Get UDP port on which server should listen.
void runOne()
Main server processing step.
virtual Pkt6Ptr receivePacket(int timeout)
dummy wrapper around IfaceMgr::receive6
void processPacketBufferSend(hooks::CalloutHandlePtr &callout_handle, Pkt6Ptr &rsp)
Executes buffer6_send callout and sends the response.
OptionPtr releaseIA_NA(const DuidPtr &duid, const Pkt6Ptr &query, int &general_status, boost::shared_ptr< Option6IA > ia, Lease6Ptr &old_lease)
Releases specific IA_NA option.
void buildCfgOptionList(const Pkt6Ptr &question, AllocEngine::ClientContext6 &ctx, CfgOptionList &co_list)
Build the configured option list.
void appendDefaultOptions(const Pkt6Ptr &question, Pkt6Ptr &answer, const CfgOptionList &co_list)
Appends default options to server's answer.
OptionPtr assignIA_NA(const isc::dhcp::Pkt6Ptr &query, AllocEngine::ClientContext6 &ctx, Option6IAPtr ia)
Processes IA_NA option (and assigns addresses if necessary).
static const std::string VENDOR_CLASS_PREFIX
this is a prefix added to the content of vendor-class option
OptionPtr serverid_
Server DUID (to be sent in server-identifier option)
void setTeeTimes(uint32_t preferred_lft, const ConstSubnet6Ptr &subnet, Option6IAPtr &resp)
Sets the T1 and T2 timers in the outbound IA.
void conditionallySetReservedClientClasses(const Pkt6Ptr &pkt, const AllocEngine::ClientContext6 &ctx)
Assigns classes retrieved from host reservation database if they haven't been yet set.
void processPacketAndSendResponse(Pkt6Ptr query)
Process a single incoming DHCPv6 packet and sends the response.
OptionPtr releaseIA_PD(const DuidPtr &duid, const Pkt6Ptr &query, int &general_status, boost::shared_ptr< Option6IA > ia, Lease6Ptr &old_lease)
Releases specific IA_PD option.
void processDhcp4Query(const Pkt6Ptr &dhcp4_query)
Processes incoming DHCPv4-query message.
Pkt6Ptr processRebind(AllocEngine::ClientContext6 &ctx)
Processes incoming Rebind message.
bool earlyGHRLookup(const Pkt6Ptr &query, AllocEngine::ClientContext6 &ctx)
Initialize client context and perform early global reservations lookup.
void initContext0(const Pkt6Ptr &query, AllocEngine::ClientContext6 &ctx)
Initialize client context (first part).
virtual ~Dhcpv6Srv()
Destructor. Used during DHCPv6 service shutdown.
void initContext(AllocEngine::ClientContext6 &ctx, bool &drop)
Initializes client context for specified packet.
Pkt6Ptr processRequest(AllocEngine::ClientContext6 &ctx)
Processes incoming Request and returns Reply response.
void sendResponseNoThrow(hooks::CalloutHandlePtr &callout_handle, Pkt6Ptr query, Pkt6Ptr &rsp, ConstSubnet6Ptr &subnet)
Process an unparked DHCPv6 packet and sends the response.
NetworkStatePtr network_state_
Holds information about disabled DHCP service and/or disabled subnet/network scopes.
std::list< std::list< std::string > > jsonPathsToRedact() const final override
Return a list of all paths that contain passwords or secrets for kea-dhcp6.
OptionPtr assignIA_PD(const Pkt6Ptr &query, AllocEngine::ClientContext6 &ctx, boost::shared_ptr< Option6IA > ia)
Processes IA_PD option (and assigns prefixes if necessary).
bool testUnicast(const Pkt6Ptr &pkt) const
Check if the message can be sent to unicast.
Pkt6Ptr processRelease(AllocEngine::ClientContext6 &ctx)
Process incoming Release message.
void processClientFqdn(const Pkt6Ptr &question, const Pkt6Ptr &answer, AllocEngine::ClientContext6 &ctx)
Processes Client FQDN Option.
void setStatusCode(boost::shared_ptr< Option6IA > &container, const OptionPtr &status)
A simple utility method that sets the status code.
static int getHookIndexBuffer6Send()
Returns the index of the buffer6_send hook.
void classifyPacket(const Pkt6Ptr &pkt)
Assigns incoming packet to zero or more classes.
static HWAddrPtr getMAC(const Pkt6Ptr &pkt)
Attempts to get a MAC/hardware address using configured sources.
Dhcpv6Srv(uint16_t server_port=DHCP6_SERVER_PORT, uint16_t client_port=0)
Default constructor.
bool declineLeases(const Pkt6Ptr &decline, Pkt6Ptr &reply, AllocEngine::ClientContext6 &ctx)
Attempts to decline all leases in specified Decline message.
void releaseLeases(const Pkt6Ptr &release, Pkt6Ptr &reply, AllocEngine::ClientContext6 &ctx)
Attempts to release received addresses.
void extendLeases(const Pkt6Ptr &query, Pkt6Ptr &reply, AllocEngine::ClientContext6 &ctx)
Attempts to extend the lifetime of IAs.
void processRSOO(const Pkt6Ptr &query, const Pkt6Ptr &rsp)
Processes Relay-supplied options, if present.
static std::string getVersion(bool extended)
returns Kea version on stdout and exit.
Pkt6Ptr processAddrRegInform(AllocEngine::ClientContext6 &ctx)
Processes incoming Addr-reg-inform message.
OptionPtr extendIA_NA(const Pkt6Ptr &query, AllocEngine::ClientContext6 &ctx, Option6IAPtr ia)
Extends lifetime of the specific IA_NA option.
Pkt6Ptr processConfirm(AllocEngine::ClientContext6 &ctx)
Processes incoming Confirm message and returns Reply.
void sanityCheckDUID(const OptionPtr &opt, const std::string &opt_name)
verifies if received DUID option (client-id or server-id) is sane
static void setHostIdentifiers(AllocEngine::ClientContext6 &ctx)
Set host identifiers within a context.
void processPacketPktSend(hooks::CalloutHandlePtr &callout_handle, Pkt6Ptr &query, Pkt6Ptr &rsp, ConstSubnet6Ptr &subnet)
Executes pkt6_send callout.
Pkt6Ptr processDhcp6Query(Pkt6Ptr query)
Process a single incoming DHCPv6 query.
void processDhcp6QueryAndSendResponse(Pkt6Ptr query)
Process a single incoming DHCPv6 query.
asiolink::IOServicePtr & getIOService()
Returns pointer to the IO service used by the server.
void appendRequestedOptions(const Pkt6Ptr &question, Pkt6Ptr &answer, const CfgOptionList &co_list)
Appends requested options to server's answer.
uint16_t client_port_
UDP port number to which server sends all responses.
CBControlDHCPv6Ptr cb_control_
Controls access to the configuration backends.
isc::dhcp::ConstSubnet6Ptr selectSubnet(const Pkt6Ptr &question, bool &drop)
Selects a subnet for a given client's packet.
volatile bool shutdown_
Indicates if shutdown is in progress.
void checkPostAssignmentChanges(const Pkt6Ptr &question, Pkt6Ptr &answer, AllocEngine::ClientContext6 &ctx, const ConstSubnet6Ptr orig_subnet)
Iterates over new leases, update stale DNS entries.
Pkt6Ptr processSolicit(AllocEngine::ClientContext6 &ctx)
Processes incoming Solicit and returns response.
void startD2()
Starts DHCP_DDNS client IO if DDNS updates are enabled.
static std::string duidToString(const OptionPtr &opt)
converts DUID to text Converts content of DUID option to a text representation, e....
OptionPtr getPDExclude(const AllocEngine::ClientContext6 &ctx, const Lease6Ptr &lease)
Return the PD exclude option to include.
static void removeDependentEvaluatedClasses(const Pkt6Ptr &pkt)
Removed evaluated client classes.
void createNameChangeRequests(const Pkt6Ptr &answer, AllocEngine::ClientContext6 &ctx)
Creates a number of isc::dhcp_ddns::NameChangeRequest objects based on the DHCPv6 Client FQDN Option.
Pkt6Ptr processInfRequest(AllocEngine::ClientContext6 &ctx)
Processes incoming Information-request message.
uint16_t server_port_
UDP port number on which server listens.
void appendRequestedVendorOptions(const Pkt6Ptr &question, Pkt6Ptr &answer, AllocEngine::ClientContext6 &ctx, const CfgOptionList &co_list)
Appends requested vendor options to server's answer.
bool declineLease(const Pkt6Ptr &decline, const Lease6Ptr lease, boost::shared_ptr< Option6IA > ia_rsp)
Declines specific IPv6 lease.
void discardPackets()
Discards parked packets Clears the packet parking lots of all packets.
static std::list< std::string > getDBVersions()
Return extended version info for registered backends.
ConstHostCollection getAll6(const SubnetID &subnet_id, const HostMgrOperationTarget target) const
Return all hosts in a DHCPv6 subnet.
static void create()
Creates new instance of the HostMgr.
static HostMgr & instance()
Returns a sole instance of the HostMgr.
IdentifierType
Type of the host identifier.
@ IDENT_FLEX
Flexible host identifier.
std::string getIdentifierAsText() const
Returns host identifier in a textual form.
static IfaceMgr & instance()
IfaceMgr is a singleton class.
bool send(const Pkt6Ptr &pkt)
Sends an IPv6 packet.
void closeSockets()
Closes all open sockets.
static TrackingLeaseMgr & instance()
Return current lease manager.
static std::list< std::string > getDBVersions()
Return extended version info for registered backends.
static void destroy()
Destroy lease manager.
virtual bool deleteLease(const Lease4Ptr &lease)=0
Deletes an IPv4 lease.
virtual Lease6Ptr getLease6(Lease::Type type, const isc::asiolink::IOAddress &addr) const =0
Returns existing IPv6 lease for a given IPv6 address.
virtual void updateLease6(const Lease6Ptr &lease6)=0
Updates IPv6 lease.
static std::list< std::string > getDBVersions()
Return extended version info for registered backends.
static void splitNtpServerOptions6(isc::dhcp::OptionCollection &options)
Split NTP server option to one suboption per instance.
Controls the DHCP service enabling status.
Represents DHCPv6 Client FQDN Option (code 39).
static const uint8_t FLAG_S
S bit.
static const uint8_t FLAG_N
N bit.
Class that represents IAPREFIX option in DHCPv6.
uint32_t getIAID() const
Returns IA identifier.
OptionPtr option_
Option instance.
bool allowedForClientClasses(const ClientClasses &cclasses) const
Validates an OptionDescriptor's client-classes against a list of classes.
This class represents vendor-specific information option.
const OptionCollection & getOptions() const
Returns all encapsulated options.
OptionPtr getOption(uint16_t type) const
Returns shared_ptr to suboption of specific type.
Represents a DHCPv6 packet.
virtual std::string getLabel() const
Returns text representation of the primary packet identifiers.
An exception that is thrown if a DHCPv6 protocol violation occurs while processing a message (e....
RAII object enabling copying options retrieved from the packet.
Exception thrown when a call to select is interrupted by a signal.
Exception thrown during option unpacking This exception is thrown when an error has occurred,...
Container class for handling the DHCID value within a NameChangeRequest.
Represents a DHCP-DDNS client request.
Result
Defines the outcome of an asynchronous NCR send.
@ NEXT_STEP_PARK
park the packet
@ NEXT_STEP_CONTINUE
continue normally
@ NEXT_STEP_DROP
drop the packet
@ NEXT_STEP_SKIP
skip the next processing step
static int registerHook(const std::string &name)
Register Hook.
static bool calloutsPresent(int index)
Are callouts present?
static std::vector< std::string > getLibraryNames()
Return list of loaded libraries.
static bool unloadLibraries()
Unload libraries.
static void park(const std::string &hook_name, T parked_object, std::function< void()> unpark_callback)
Park an object (packet).
static void callCallouts(int index, CalloutHandle &handle)
Calls the callouts for a given hook.
static void prepareUnloadLibraries()
Prepare the unloading of libraries.
static bool drop(const std::string &hook_name, T parked_object)
Removes parked object without calling a callback.
static void clearParkingLots()
Clears any parking packets.
Wrapper class around callout handle which automatically resets handle's state.
static ServerHooks & getServerHooks()
Return ServerHooks object.
static std::string getVersion()
Version.
File-based Interprocess Sync Class.
Interprocess Sync Locker Class.
bool unlock()
Release the lock.
bool lock()
Acquire the lock (blocks if something else has acquired a lock on the same task name)
int getExitValue()
Fetches the exit value.
Statistics Manager class.
static StatsMgr & instance()
Statistics Manager accessor method.
static std::string generateName(const std::string &context, Type index, const std::string &stat_name)
Generates statistic name in a given context.
RAII class creating a critical section.
static MultiThreadingMgr & instance()
Returns a single instance of Multi Threading Manager.
ThreadPool< std::function< void()> > & getThreadPool()
Get the dhcp thread pool.
void apply(bool enabled, uint32_t thread_count, uint32_t queue_size)
Apply the multi-threading related settings.
The OutputBuffer class is a buffer abstraction for manipulating mutable data.
const std::vector< uint8_t > & getVector() const
Return the buffer.
Defines classes for storing client class definitions.
int version()
returns Kea hooks version.
Defines the D2ClientConfig class.
Defines the D2ClientMgr class.
@ DHCPV6_INFORMATION_REQUEST
Defines the Dhcp6to4Ipc class.
#define VENDOR_ID_CABLE_LABS
#define isc_throw(type, stream)
A shortcut macro to insert known values into exception arguments.
boost::shared_ptr< OptionUint16Array > OptionUint16ArrayPtr
OptionIntArray< uint16_t > OptionUint16Array
void setValue(const std::string &name, const int64_t value)
Records absolute integer observation.
void addValue(const std::string &name, const int64_t value)
Records incremental integer observation.
int get(CalloutHandle &handle)
The gss-tsig-get command.
When a message is logged with DEBUG severity, the debug level associated with the message is also spe...
#define LOG_ERROR(LOGGER, MESSAGE)
Macro to conveniently test error output and log it.
#define LOG_INFO(LOGGER, MESSAGE)
Macro to conveniently test info output and log it.
#define LOG_WARN(LOGGER, MESSAGE)
Macro to conveniently test warn output and log it.
#define LOG_DEBUG(LOGGER, LEVEL, MESSAGE)
Macro to conveniently test debug output and log it.
boost::shared_ptr< const Element > ConstElementPtr
ConflictResolutionMode StringToConflictResolutionMode(const std::string &mode_str)
Function which converts string to ConflictResolutionMode enum values.
boost::shared_ptr< NameChangeRequest > NameChangeRequestPtr
Defines a pointer to a NameChangeRequest.
const isc::log::MessageID DHCP6_DDNS_REQUEST_SEND_FAILED
boost::shared_ptr< OptionVendor > OptionVendorPtr
Pointer to a vendor option.
const isc::log::MessageID DHCP6_PD_LEASE_ADVERT
const isc::log::MessageID DHCP6_BUFFER_RECEIVED
const isc::log::MessageID DHCP6_RELEASE_NA_DELETED
isc::log::Logger bad_packet6_logger(DHCP6_BAD_PACKET_LOGGER_NAME)
Logger for rejected packets.
const isc::log::MessageID DHCP6_PACKET_DROP_PARSE_FAIL
const isc::log::MessageID DHCP6_ADDITIONAL_CLASS_NO_TEST
const isc::log::MessageID DHCP6_ADDITIONAL_CLASS_EVAL_RESULT
const isc::log::MessageID DHCP6_RELEASE_PD_DELETED
const isc::log::MessageID DHCP6_LEASE_ALLOC
const isc::log::MessageID DHCP6_FLEX_ID
const isc::log::MessageID DHCP6_REGISTERED_LEASE_ADD_FAIL
uint32_t calculateDdnsTtl(uint32_t lease_lft, const util::Optional< double > &ddns_ttl_percent, const util::Optional< uint32_t > &ddns_ttl, const util::Optional< uint32_t > &ddns_ttl_min, const util::Optional< uint32_t > &ddns_ttl_max)
Calculates TTL for a DNS resource record based on lease life time.
const isc::log::MessageID DHCP6_PACKET_PROCESS_EXCEPTION_MAIN
const isc::log::MessageID DHCP6_HOOK_PACKET_SEND_SKIP
const isc::log::MessageID DHCP6_ADDR_REG_INFORM_CLIENT_CHANGE
const isc::log::MessageID DHCP6_SUBNET_SELECTION_FAILED
const isc::log::MessageID DHCP6_PACKET_DROP_SERVERID_MISMATCH
const isc::log::MessageID DHCP6_HOOK_DECLINE_SKIP
const isc::log::MessageID DHCP6_PD_LEASE_REUSE
const isc::log::MessageID DHCP6_PACKET_DROP_UNICAST
const isc::log::MessageID DHCP6_LEASE_PD_WITHOUT_DUID
const isc::log::MessageID DHCP6_LEASE_ALLOC_FAIL
const isc::log::MessageID DHCP6_ADDITIONAL_CLASS_UNDEFINED
const isc::log::MessageID DHCP6_PACKET_SEND_FAIL
const isc::log::MessageID DHCP6_PACKET_PROCESS_EXCEPTION
void queueNCR(const NameChangeType &chg_type, const Lease4Ptr &lease)
Creates name change request from the DHCPv4 lease.
const isc::log::MessageID DHCP6_QUERY_LABEL
const isc::log::MessageID DHCP6_BUFFER_UNPACK
boost::shared_ptr< const Subnet6 > ConstSubnet6Ptr
A const pointer to a Subnet6 object.
std::vector< uint32_t > CfgMACSources
Container for defined MAC/hardware address sources.
const isc::log::MessageID DHCP6_SRV_D2STOP_ERROR
const isc::log::MessageID DHCP6_REGISTERED_LEASE_UPDATE_FAIL
const isc::log::MessageID DHCP6_PACKET_SEND
const isc::log::MessageID DHCP6_DECLINE_FAIL_LEASE_WITHOUT_DUID
const int DBG_DHCP6_BASIC_DATA
Debug level used to log the traces with some basic data.
const isc::log::MessageID DHCP6_HOOK_PACKET_RCVD_SKIP
const isc::log::MessageID DHCP6_ADDITIONAL_CLASS_EVAL_ERROR
boost::shared_ptr< DUID > DuidPtr
const isc::log::MessageID DHCP6_OPEN_SOCKET
const isc::log::MessageID DHCP6_PACK_FAIL
const isc::log::MessageID DHCP6_RELEASE_PD_FAIL_WRONG_IAID
const isc::log::MessageID DHCP6_HOOK_DDNS_UPDATE
boost::shared_ptr< Lease6 > Lease6Ptr
Pointer to a Lease6 structure.
std::vector< Lease6Ptr > Lease6Collection
A collection of IPv6 leases.
const int DBG_DHCP6_HOOKS
Debug level used to trace hook related operations.
const isc::log::MessageID DHCP6_HOOK_LEASE6_RELEASE_NA_SKIP
ContinuationPtr makeContinuation(Continuation &&cont)
Continuation factory.
const int DBG_DHCP6_START
Debug level used to log information during server startup.
const isc::log::MessageID DHCP6_PD_LEASE_ALLOC
const isc::log::MessageID DHCP6_DDNS_GENERATE_FQDN
const isc::log::MessageID DHCP6_RELEASE_PD_EXPIRED
boost::shared_ptr< Option6IA > Option6IAPtr
A pointer to the Option6IA object.
const isc::log::MessageID DHCP6_DDNS_REMOVE_OLD_LEASE_FQDN
boost::shared_ptr< const CfgRSOO > ConstCfgRSOOPtr
Pointer to the const object.
boost::shared_ptr< const CfgHostOperations > ConstCfgHostOperationsPtr
Pointer to the const object.
std::multimap< unsigned int, OptionPtr > OptionCollection
A collection of DHCP (v4 or v6) options.
const isc::log::MessageID DHCP6_SUBNET_DATA
const isc::log::MessageID DHCP6_UNKNOWN_MSG_RECEIVED
boost::shared_ptr< ClientClassDef > ClientClassDefPtr
a pointer to an ClientClassDef
boost::shared_ptr< DdnsParams > DdnsParamsPtr
Defines a pointer for DdnsParams instances.
const isc::log::MessageID DHCP6_HOOK_SUBNET6_SELECT_DROP
const isc::log::MessageID DHCP6_ADD_GLOBAL_STATUS_CODE
boost::shared_ptr< Option6IAPrefix > Option6IAPrefixPtr
Pointer to the Option6IAPrefix object.
const isc::log::MessageID DHCP6_DDNS_RESPONSE_FQDN_DATA
const isc::log::MessageID DHCP6_RELEASE_NA
const isc::log::MessageID DHCP6_CLASSES_ASSIGNED
const isc::log::MessageID DHCP6_REQUIRED_OPTIONS_CHECK_FAIL
const isc::log::MessageID DHCP6_PROCESS_IA_NA_EXTEND
std::pair< IPv6ResrvIterator, IPv6ResrvIterator > IPv6ResrvRange
const isc::log::MessageID DHCP6_RELEASE_PD_FAIL
const isc::log::MessageID DHCP6_SRV_CONSTRUCT_ERROR
const isc::log::MessageID DHCP6_LEASE_RENEW
const char * DOCSIS3_CLASS_EROUTER
The class as specified in vendor-class option by the devices.
const isc::log::MessageID DHCP6_RELEASE_NA_FAIL
boost::shared_ptr< HWAddr > HWAddrPtr
Shared pointer to a hardware address structure.
const isc::log::MessageID DHCP6_PACKET_PROCESS_STD_EXCEPTION
const isc::log::MessageID DHCP6_PACKET_RECEIVE_FAIL
const isc::log::MessageID DHCP6_PROCESS_IA_NA_SOLICIT
const isc::log::MessageID DHCP6_ADDR_REG_INFORM_FAIL
const isc::log::MessageID DHCP6_DECLINE_FAIL_IAID_MISMATCH
const isc::log::MessageID DHCP6_HOOK_SUBNET6_SELECT_PARK
OptionContainer::nth_index< 5 >::type OptionContainerCancelIndex
Type of the index #5 - option cancellation flag.
boost::shared_ptr< Option6StatusCode > Option6StatusCodePtr
Pointer to the isc::dhcp::Option6StatusCode.
std::pair< OptionContainerPersistIndex::const_iterator, OptionContainerPersistIndex::const_iterator > OptionContainerPersistRange
Pair of iterators to represent the range of options having the same persistency flag.
boost::shared_ptr< SharedNetwork6 > SharedNetwork6Ptr
Pointer to SharedNetwork6 object.
const isc::log::MessageID DHCP4_HOOK_SUBNET6_SELECT_PARKING_LOT_FULL
isc::log::Logger packet6_logger(DHCP6_PACKET_LOGGER_NAME)
Logger for processed packets.
const isc::log::MessageID DHCP6_DECLINE_LEASE
boost::shared_ptr< Expression > ExpressionPtr
const isc::log::MessageID DHCP6_LEASE_ADVERT_FAIL
const isc::log::MessageID DHCP6_HOOK_LEASES6_PARKING_LOT_FULL
const isc::log::MessageID DHCP6_PD_LEASE_ADVERT_FAIL
boost::shared_ptr< Pool > PoolPtr
a pointer to either IPv4 or IPv6 Pool
isc::hooks::CalloutHandlePtr getCalloutHandle(const T &pktptr)
CalloutHandle Store.
const isc::log::MessageID DHCP6_PACKET_PROCESS_FAIL
const isc::log::MessageID DHCP6_RELEASE_NA_EXPIRED
boost::shared_ptr< ClientClassDictionary > ClientClassDictionaryPtr
Defines a pointer to a ClientClassDictionary.
const isc::log::MessageID DHCP6_PACKET_RECEIVED
const isc::log::MessageID DHCP6_RESPONSE_DATA
const isc::log::MessageID DHCP6_DDNS_RECEIVE_FQDN
const isc::log::MessageID DHCP6_PACKET_OPTIONS_SKIPPED
const isc::log::MessageID DHCP6_NO_INTERFACES
const isc::log::MessageID DHCP6_RELEASE_PD_FAIL_WRONG_DUID
const isc::log::MessageID DHCP6_LEASE_REUSE
isc::log::Logger ddns6_logger(DHCP6_DDNS_LOGGER_NAME)
Logger for Hostname or FQDN processing.
const isc::log::MessageID DHCP6_HOOK_ADDR6_REGISTER_SKIP
boost::shared_ptr< Continuation > ContinuationPtr
Define the type of shared pointers to continuations.
boost::shared_ptr< OptionContainer > OptionContainerPtr
Pointer to the OptionContainer object.
boost::shared_ptr< ClientClassDefList > ClientClassDefListPtr
Defines a pointer to a ClientClassDefList.
const isc::log::MessageID DHCP6_HOOK_LEASES6_COMMITTED_DROP
const isc::log::MessageID DHCP6_HOOK_PACKET_SEND_DROP
const isc::log::MessageID DHCP6_DDNS_GENERATED_FQDN_UPDATE_FAIL
const isc::log::MessageID DHCP6_PACKET_DROP_DROP_CLASS2
boost::shared_ptr< Option6IAAddr > Option6IAAddrPtr
A pointer to the isc::dhcp::Option6IAAddr object.
const char * DOCSIS3_CLASS_MODEM
DOCSIS3.0 compatible cable modem.
const isc::log::MessageID DHCP6_SHUTDOWN_REQUEST
const isc::log::MessageID DHCP6_HOOK_BUFFER_RCVD_SKIP
const isc::log::MessageID DHCP6_DECLINE_FAIL
bool evaluateBool(const Expression &expr, Pkt &pkt)
Evaluate a RPN expression for a v4 or v6 packet and return a true or false decision.
const isc::log::MessageID DHCP6_QUERY_DATA
const int DBG_DHCP6_DETAIL_DATA
This level is used to log the contents of packets received and sent.
boost::shared_ptr< const Host > ConstHostPtr
Const pointer to the Host object.
const isc::log::MessageID DHCP6_PROCESS_IA_PD_SOLICIT
const isc::log::MessageID DHCP6_PACKET_DROP_DROP_CLASS
boost::shared_ptr< Lease6Collection > Lease6CollectionPtr
A shared pointer to the collection of IPv6 leases.
const isc::log::MessageID DHCP6_DECLINE_PROCESS_IA
const isc::log::MessageID DHCP6_PROCESS_IA_NA_REQUEST
OptionContainer::nth_index< 2 >::type OptionContainerPersistIndex
Type of the index #2 - option persistency flag.
isc::log::Logger lease6_logger(DHCP6_LEASE_LOGGER_NAME)
Logger for lease allocation logic.
const isc::log::MessageID DHCP6_CLASS_UNCONFIGURED
boost::shared_ptr< OptionVendorClass > OptionVendorClassPtr
Defines a pointer to the OptionVendorClass.
const isc::log::MessageID DHCP6_LEASE_NA_WITHOUT_DUID
const isc::log::MessageID DHCP6_HOOK_DECLINE_DROP
const isc::log::MessageID DHCP6_PROCESS_IA_PD_REQUEST
const isc::log::MessageID DHCP6_HOOK_SUBNET6_SELECT_SKIP
const isc::log::MessageID DHCP6_PD_LEASE_RENEW
const isc::log::MessageID DHCP6_CLASS_ASSIGNED
boost::shared_ptr< const Subnet > ConstSubnetPtr
A generic pointer to either const Subnet4 or const Subnet6 object.
boost::shared_ptr< Pkt6 > Pkt6Ptr
A pointer to Pkt6 packet.
const isc::log::MessageID DHCP6_PROCESS_IA_NA_RELEASE
const isc::log::MessageID DHCP6_PACKET_DROP_DROP_CLASS_EARLY
const isc::log::MessageID DHCP6_LEASE_ADVERT
std::vector< uint8_t > OptionBuffer
buffer types used in DHCP code.
const isc::log::MessageID DHCP6_DECLINE_FAIL_DUID_MISMATCH
const isc::log::MessageID DHCP6_SRV_UNLOAD_LIBRARIES_ERROR
const isc::log::MessageID DHCP6_HOOK_LEASES6_COMMITTED_PARK
const isc::log::MessageID DHCP6_DECLINE_FAIL_NO_LEASE
const isc::log::MessageID DHCP6_RAPID_COMMIT
const isc::log::MessageID DHCP6_BUFFER_WAIT_SIGNAL
bool isClientClassBuiltIn(const ClientClass &client_class)
Check if a client class name is builtin.
boost::shared_ptr< Option6ClientFqdn > Option6ClientFqdnPtr
A pointer to the Option6ClientFqdn object.
const isc::log::MessageID DHCP6_PROCESS_IA_PD_EXTEND
std::pair< OptionContainerCancelIndex::const_iterator, OptionContainerCancelIndex::const_iterator > OptionContainerCancelRange
Pair of iterators to represent the range of options having the same cancellation flag.
const isc::log::MessageID DHCP6_CLASSES_ASSIGNED_AFTER_SUBNET_SELECTION
const isc::log::MessageID DHCP6_PACKET_PROCESS_STD_EXCEPTION_MAIN
const int DBG_DHCP6_BASIC
Debug level used to trace basic operations within the code.
const isc::log::MessageID DHCP6_HOOK_ADDR6_REGISTER_DROP
isc::log::Logger dhcp6_logger(DHCP6_APP_LOGGER_NAME)
Base logger for DHCPv6 server.
const isc::log::MessageID DHCP6_SUBNET_SELECTED
const isc::log::MessageID DHCP6_RELEASE_NA_FAIL_WRONG_IAID
const isc::log::MessageID DHCP6_HOOK_BUFFER_RCVD_DROP
boost::shared_ptr< Option > OptionPtr
const isc::log::MessageID DHCP6_DDNS_CREATE_ADD_NAME_CHANGE_REQUEST
const isc::log::MessageID DHCP6_RELEASE_NA_FAIL_WRONG_DUID
const isc::log::MessageID DHCP6_ADD_STATUS_CODE_FOR_IA
isc::log::Logger options6_logger(DHCP6_OPTIONS_LOGGER_NAME)
Logger for options parser.
const isc::log::MessageID DHCP6_LEASE_DATA
const isc::log::MessageID DHCP6_HOOK_BUFFER_SEND_SKIP
const int DBG_DHCP6_DETAIL
Debug level used to trace detailed errors.
const isc::log::MessageID DHCP6_SUBNET_DYNAMICALLY_CHANGED
const isc::log::MessageID DHCP6_PACKET_QUEUE_FULL
const isc::log::MessageID DHCP6_PD_LEASE_ALLOC_FAIL
const isc::log::MessageID DHCP6_HOOK_LEASE6_RELEASE_PD_SKIP
const isc::log::MessageID DHCP6_RELEASE_PD
std::list< ConstCfgOptionPtr > CfgOptionList
Const pointer list.
const isc::log::MessageID DHCP6_DDNS_FQDN_GENERATED
const isc::log::MessageID DHCP6_PACKET_DROP_DHCP_DISABLED
boost::shared_ptr< Pool6 > Pool6Ptr
a pointer an IPv6 Pool
isc::log::Logger hooks_logger("hooks")
Hooks Logger.
boost::shared_ptr< CalloutHandle > CalloutHandlePtr
A shared pointer to a CalloutHandle object.
boost::shared_ptr< ParkingLot > ParkingLotPtr
Type of the pointer to the parking lot.
const int DBGLVL_TRACE_BASIC
Trace basic operations.
const int DBGLVL_PKT_HANDLING
This debug level is reserved for logging the details of packet handling, such as dropping the packet ...
bool equalValues(const T &ptr1, const T &ptr2)
This function checks if two pointers are non-null and values are equal.
Defines the logger used by the top-level component of kea-lfc.
This file defines abstract classes for exchanging NameChangeRequests.
This file provides the classes needed to embody, compose, and decompose DNS update requests that are ...
Standard implementation of read-write mutexes with writer preference using C++11 mutex and condition ...
#define DHCP6_OPTION_SPACE
Lease6Collection old_leases_
A pointer to any old leases that the client had before update but are no longer valid after the updat...
Option6IAPtr ia_rsp_
A pointer to the IA_NA/IA_PD option to be sent in response.
Lease::Type type_
Lease type (IA or PD)
Lease6Collection changed_leases_
A pointer to any leases that have changed FQDN information.
void addHint(const asiolink::IOAddress &prefix, const uint8_t prefix_len=128, const uint32_t preferred=0, const uint32_t valid=0)
Convenience method adding new hint.
Lease6Collection reused_leases_
Set of leases marked for reuse by lease caching.
HintContainer hints_
Client's hints.
uint32_t iaid_
The IAID field from IA_NA or IA_PD that is being processed.
Context information for the DHCPv6 leases allocation.
IAContext & currentIA()
Returns IA specific context for the currently processed IA.
std::vector< IAContext > ias_
Container holding IA specific contexts.
void addHostIdentifier(const Host::IdentifierType &id_type, const std::vector< uint8_t > &identifier)
Convenience function adding host identifier into host_identifiers_ list.
bool fake_allocation_
Indicates if this is a real or fake allocation.
ConstHostPtr currentHost() const
Returns host from the most preferred subnet.
DuidPtr duid_
Client identifier.
Lease6Collection new_leases_
A collection of newly allocated leases.
std::vector< IAContext > & getIAContexts()
HWAddrPtr hwaddr_
Hardware/MAC address (if available, may be NULL)
hooks::CalloutHandlePtr callout_handle_
Callout handle associated with the client's message.
ResourceContainer allocated_resources_
Holds addresses and prefixes allocated for all IAs.
bool rev_dns_update_
A boolean value which indicates that server takes responsibility for the reverse DNS Update for this ...
DdnsParamsPtr getDdnsParams()
Returns the set of DDNS behavioral parameters based on the selected subnet.
ConstHostPtr globalHost() const
Returns global host reservation if there is one.
Pkt6Ptr query_
A pointer to the client's message.
bool early_global_reservations_lookup_
Indicates if early global reservation is enabled.
std::string hostname_
Hostname.
void createIAContext()
Creates new IA context.
ConstSubnet6Ptr subnet_
Subnet selected for the client by the server.
std::map< SubnetID, ConstHostPtr > hosts_
Holds a map of hosts belonging to the client within different subnets.
bool fwd_dns_update_
A boolean value which indicates that server takes responsibility for the forward DNS Update for this ...
Structure that holds a lease for IPv6 address and/or prefix.
static const uint32_t INFINITY_LFT
Infinity (means static, i.e. never expire)
static std::string lifetimeToText(uint32_t lifetime)
Print lifetime.
static const uint32_t STATE_DEFAULT
A lease in the default state.
static const uint32_t STATE_RELEASED
Released lease held in the database for lease affinity.
@ TYPE_PD
the lease contains IPv6 prefix (for prefix delegation)
@ TYPE_NA
the lease contains non-temporary IPv6 address
static const uint32_t STATE_REGISTERED
Registered self-generated lease.
Subnet selector used to specify parameters used to select a subnet.
bool add(const WorkItemPtr &item)
add a work item to the thread pool