summaryrefslogtreecommitdiffstats
path: root/test-dnsdist-lua-ffi.cc
diff options
context:
space:
mode:
Diffstat (limited to '')
-rw-r--r--test-dnsdist-lua-ffi.cc782
1 files changed, 782 insertions, 0 deletions
diff --git a/test-dnsdist-lua-ffi.cc b/test-dnsdist-lua-ffi.cc
new file mode 100644
index 0000000..776aee0
--- /dev/null
+++ b/test-dnsdist-lua-ffi.cc
@@ -0,0 +1,782 @@
+/*
+ * This file is part of PowerDNS or dnsdist.
+ * Copyright -- PowerDNS.COM B.V. and its contributors
+ *
+ * This program is free software; you can redistribute it and/or modify
+ * it under the terms of version 2 of the GNU General Public License as
+ * published by the Free Software Foundation.
+ *
+ * In addition, for the avoidance of any doubt, permission is granted to
+ * link this program with OpenSSL and to (re)distribute the binaries
+ * produced as the result of such linking.
+ *
+ * This program is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+ * GNU General Public License for more details.
+ *
+ * You should have received a copy of the GNU General Public License
+ * along with this program; if not, write to the Free Software
+ * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
+ */
+#define BOOST_TEST_DYN_LINK
+#define BOOST_TEST_NO_MAIN
+
+#include <boost/test/unit_test.hpp>
+
+#include "dnsdist-lua-ffi.hh"
+#include "dnsdist-rings.hh"
+#include "dnsparser.hh"
+#include "dnswriter.hh"
+
+bool addMetricDefinition(const std::string& name, const std::string& type, const std::string& description, const std::string& customPrometheusName)
+{
+ return true;
+}
+
+BOOST_AUTO_TEST_SUITE(test_dnsdist_lua_ffi)
+
+BOOST_AUTO_TEST_CASE(test_Query)
+{
+ InternalQueryState ids;
+ ids.origRemote = ComboAddress("192.0.2.1:4242");
+ ids.origDest = ComboAddress("192.0.2.255:53");
+ ids.qtype = QType::A;
+ ids.qclass = QClass::IN;
+ ids.protocol = dnsdist::Protocol::DoUDP;
+ ids.qname = DNSName("www.powerdns.com.");
+ ids.queryRealTime.start();
+ PacketBuffer query;
+ GenericDNSPacketWriter<PacketBuffer> pwQ(query, ids.qname, QType::A, QClass::IN, 0);
+ pwQ.getHeader()->rd = 1;
+ pwQ.getHeader()->id = htons(42);
+
+ DNSQuestion dq(ids, query);
+ dnsdist_ffi_dnsquestion_t lightDQ(&dq);
+
+ {
+ // dnsdist_ffi_dnsquestion_get_qtype
+ BOOST_CHECK_EQUAL(dnsdist_ffi_dnsquestion_get_qtype(&lightDQ), ids.qtype);
+ }
+
+ {
+ // dnsdist_ffi_dnsquestion_get_qclass
+ BOOST_CHECK_EQUAL(dnsdist_ffi_dnsquestion_get_qclass(&lightDQ), ids.qclass);
+ }
+
+ {
+ // dnsdist_ffi_dnsquestion_get_id
+ BOOST_CHECK_EQUAL(dnsdist_ffi_dnsquestion_get_id(&lightDQ), ntohs(pwQ.getHeader()->id));
+ BOOST_CHECK_EQUAL(dnsdist_ffi_dnsquestion_get_id(nullptr), 0U);
+ }
+
+ {
+ // dnsdist_ffi_dnsquestion_get_localaddr, dnsdist_ffi_dnsquestion_get_local_port
+ const char* buffer = nullptr;
+ size_t bufferSize = 0;
+ dnsdist_ffi_dnsquestion_get_localaddr(&lightDQ, reinterpret_cast<const void**>(&buffer), &bufferSize);
+ BOOST_REQUIRE(buffer != nullptr);
+ BOOST_REQUIRE_EQUAL(bufferSize, sizeof(ids.origDest.sin4.sin_addr.s_addr));
+ BOOST_CHECK(memcmp(buffer, &ids.origDest.sin4.sin_addr.s_addr, sizeof(ids.origDest.sin4.sin_addr.s_addr)) == 0);
+ BOOST_CHECK_EQUAL(dnsdist_ffi_dnsquestion_get_local_port(&lightDQ), 53U);
+ }
+
+ {
+ // dnsdist_ffi_dnsquestion_get_remoteaddr, dnsdist_ffi_dnsquestion_get_remote_port
+ const char* buffer = nullptr;
+ size_t bufferSize = 0;
+ dnsdist_ffi_dnsquestion_get_remoteaddr(&lightDQ, reinterpret_cast<const void**>(&buffer), &bufferSize);
+ BOOST_REQUIRE(buffer != nullptr);
+ BOOST_REQUIRE_EQUAL(bufferSize, sizeof(ids.origRemote.sin4.sin_addr.s_addr));
+ BOOST_CHECK(memcmp(buffer, &ids.origRemote.sin4.sin_addr.s_addr, sizeof(ids.origRemote.sin4.sin_addr.s_addr)) == 0);
+ BOOST_CHECK_EQUAL(dnsdist_ffi_dnsquestion_get_remote_port(&lightDQ), 4242U);
+ }
+
+ {
+ // dnsdist_ffi_dnsquestion_get_masked_remoteaddr
+ const char* buffer = nullptr;
+ size_t bufferSize = 0;
+ dnsdist_ffi_dnsquestion_get_masked_remoteaddr(&lightDQ, reinterpret_cast<const void**>(&buffer), &bufferSize, 16);
+ BOOST_REQUIRE(buffer != nullptr);
+ auto masked = Netmask(ids.origRemote, 16).getMaskedNetwork();
+ BOOST_REQUIRE_EQUAL(bufferSize, sizeof(masked.sin4.sin_addr.s_addr));
+ BOOST_CHECK(memcmp(buffer, &masked.sin4.sin_addr.s_addr, sizeof(masked.sin4.sin_addr.s_addr)) == 0);
+ }
+
+ {
+ const char* buffer[6];
+ size_t bufferSize = 6;
+
+ // invalid
+ BOOST_CHECK_EQUAL(dnsdist_ffi_dnsquestion_get_mac_addr(nullptr, buffer, 0), 0U);
+ // too small
+ BOOST_CHECK_EQUAL(dnsdist_ffi_dnsquestion_get_mac_addr(&lightDQ, buffer, 0), 0U);
+
+ // we will not find the correspondig MAC address in /proc/net/arp, unfortunately, especially not on !linux
+ BOOST_CHECK_EQUAL(dnsdist_ffi_dnsquestion_get_mac_addr(&lightDQ, buffer, bufferSize), 0U);
+ }
+
+ {
+ // dnsdist_ffi_dnsquestion_get_qname_raw
+ const char* buffer = nullptr;
+ size_t bufferSize = 0;
+ dnsdist_ffi_dnsquestion_get_qname_raw(&lightDQ, &buffer, &bufferSize);
+ BOOST_REQUIRE(buffer != nullptr);
+ BOOST_REQUIRE_EQUAL(bufferSize, ids.qname.getStorage().size());
+ BOOST_CHECK(memcmp(buffer, ids.qname.getStorage().data(), ids.qname.getStorage().size()) == 0);
+ }
+
+ {
+ // test V6 as well
+ ids.origRemote = ComboAddress("[2001:db8::1]:65535");
+ ids.origDest = ComboAddress("[2001:db8::2]:53");
+
+ const char* buffer = nullptr;
+ size_t bufferSize = 0;
+ dnsdist_ffi_dnsquestion_get_remoteaddr(&lightDQ, reinterpret_cast<const void**>(&buffer), &bufferSize);
+ BOOST_REQUIRE(buffer != nullptr);
+ BOOST_REQUIRE_EQUAL(bufferSize, sizeof(ids.origRemote.sin6.sin6_addr.s6_addr));
+ BOOST_CHECK(memcmp(buffer, &ids.origRemote.sin6.sin6_addr.s6_addr, sizeof(ids.origRemote.sin6.sin6_addr.s6_addr)) == 0);
+ BOOST_CHECK_EQUAL(dnsdist_ffi_dnsquestion_get_remote_port(&lightDQ), 65535U);
+ }
+
+ {
+ // dnsdist_ffi_dnsquestion_get_qname_hash
+ BOOST_CHECK_EQUAL(dnsdist_ffi_dnsquestion_get_qname_hash(&lightDQ, 42), ids.qname.hash(42));
+ }
+
+ {
+ // dnsdist_ffi_dnsquestion_get_rcode
+ BOOST_CHECK_EQUAL(dnsdist_ffi_dnsquestion_get_rcode(&lightDQ), RCode::NoError);
+ }
+
+ {
+ // dnsdist_ffi_dnsquestion_get_header
+ BOOST_CHECK(memcmp(dnsdist_ffi_dnsquestion_get_header(&lightDQ), pwQ.getHeader(), sizeof(dnsheader)) == 0);
+ }
+
+ {
+ // dnsdist_ffi_dnsquestion_get_len, dnsdist_ffi_dnsquestion_get_size
+ BOOST_CHECK_EQUAL(dnsdist_ffi_dnsquestion_get_len(&lightDQ), query.size());
+ BOOST_CHECK_EQUAL(dnsdist_ffi_dnsquestion_get_size(&lightDQ), query.size());
+
+ auto oldSize = query.size();
+ BOOST_CHECK(dnsdist_ffi_dnsquestion_set_size(&lightDQ, oldSize + 1));
+ BOOST_CHECK_EQUAL(dnsdist_ffi_dnsquestion_get_size(&lightDQ), oldSize + 1);
+ BOOST_CHECK_EQUAL(dnsdist_ffi_dnsquestion_get_len(&lightDQ), oldSize + 1);
+ dnsdist_ffi_dnsquestion_set_len(&lightDQ, oldSize);
+
+ auto max = std::numeric_limits<size_t>::max();
+ BOOST_CHECK_EQUAL(dnsdist_ffi_dnsquestion_set_size(&lightDQ, max), 0U);
+ }
+
+ {
+ // dnsdist_ffi_dnsquestion_get_opcode
+ BOOST_CHECK_EQUAL(dnsdist_ffi_dnsquestion_get_opcode(&lightDQ), Opcode::Query);
+ }
+
+ {
+ // dnsdist_ffi_dnsquestion_get_tcp
+ BOOST_CHECK_EQUAL(dnsdist_ffi_dnsquestion_get_tcp(&lightDQ), false);
+ }
+
+ {
+ // dnsdist_ffi_dnsquestion_get_protocol
+ BOOST_CHECK(static_cast<uint8_t>(dnsdist_ffi_dnsquestion_get_protocol(nullptr)) == dnsdist::Protocol(dnsdist::Protocol::DoUDP).toNumber());
+
+ BOOST_CHECK(static_cast<uint8_t>(dnsdist_ffi_dnsquestion_get_protocol(&lightDQ)) == dnsdist::Protocol(dnsdist::Protocol::DoUDP).toNumber());
+ for (const auto protocol : {dnsdist::Protocol::DoUDP, dnsdist::Protocol::DoTCP, dnsdist::Protocol::DNSCryptUDP, dnsdist::Protocol::DNSCryptTCP, dnsdist::Protocol::DoT, dnsdist::Protocol::DoH}) {
+ dq.ids.protocol = protocol;
+ BOOST_CHECK(static_cast<uint8_t>(dnsdist_ffi_dnsquestion_get_protocol(&lightDQ)) == protocol);
+ }
+ }
+
+ {
+ BOOST_CHECK_EQUAL(dnsdist_ffi_dnsquestion_get_elapsed_us(nullptr), 0U);
+ BOOST_CHECK_GT(dnsdist_ffi_dnsquestion_get_elapsed_us(&lightDQ), 0U);
+ }
+
+ {
+ BOOST_CHECK_EQUAL(dnsdist_ffi_dnsquestion_get_skip_cache(&lightDQ), false);
+ dnsdist_ffi_dnsquestion_set_skip_cache(&lightDQ, true);
+ BOOST_CHECK_EQUAL(dnsdist_ffi_dnsquestion_get_skip_cache(&lightDQ), true);
+ }
+
+ {
+ BOOST_CHECK_EQUAL(dnsdist_ffi_dnsquestion_get_use_ecs(&lightDQ), true);
+ dnsdist_ffi_dnsquestion_set_use_ecs(&lightDQ, false);
+ BOOST_CHECK_EQUAL(dnsdist_ffi_dnsquestion_get_use_ecs(&lightDQ), false);
+ }
+
+ {
+ BOOST_CHECK_EQUAL(dnsdist_ffi_dnsquestion_get_add_xpf(&lightDQ), true);
+ }
+
+ {
+ BOOST_CHECK_EQUAL(dnsdist_ffi_dnsquestion_get_ecs_override(&lightDQ), false);
+ dnsdist_ffi_dnsquestion_set_ecs_override(&lightDQ, true);
+ BOOST_CHECK_EQUAL(dnsdist_ffi_dnsquestion_get_ecs_override(&lightDQ), true);
+ }
+
+ {
+ BOOST_CHECK_EQUAL(dnsdist_ffi_dnsquestion_is_temp_failure_ttl_set(&lightDQ), false);
+ BOOST_CHECK_EQUAL(dnsdist_ffi_dnsquestion_get_temp_failure_ttl(&lightDQ), 0U);
+
+ dnsdist_ffi_dnsquestion_set_temp_failure_ttl(&lightDQ, 42);
+ BOOST_CHECK_EQUAL(dnsdist_ffi_dnsquestion_is_temp_failure_ttl_set(&lightDQ), true);
+ BOOST_CHECK_EQUAL(dnsdist_ffi_dnsquestion_get_temp_failure_ttl(&lightDQ), 42U);
+ dnsdist_ffi_dnsquestion_unset_temp_failure_ttl(&lightDQ);
+ BOOST_CHECK_EQUAL(dnsdist_ffi_dnsquestion_is_temp_failure_ttl_set(&lightDQ), false);
+ }
+
+ {
+ BOOST_CHECK(!dnsdist_ffi_dnsquestion_get_do(&lightDQ));
+ }
+
+ {
+ BOOST_CHECK_EQUAL(dnsdist_ffi_dnsquestion_get_ecs_prefix_length(&lightDQ), g_ECSSourcePrefixV4);
+ dnsdist_ffi_dnsquestion_set_ecs_prefix_length(&lightDQ, 65535);
+ BOOST_CHECK_EQUAL(dnsdist_ffi_dnsquestion_get_ecs_prefix_length(&lightDQ), 65535U);
+ }
+
+ {
+ const char* buffer = nullptr;
+ size_t bufferSize = 0;
+ dnsdist_ffi_dnsquestion_get_sni(&lightDQ, &buffer, &bufferSize);
+ BOOST_CHECK_EQUAL(bufferSize, 0U);
+ }
+
+ {
+ BOOST_CHECK_EQUAL(dnsdist_ffi_dnsquestion_get_trailing_data(&lightDQ, nullptr), 0U);
+#if 0
+ // DNSQuestion::setTrailingData() and DNSQuestion::getTrailingData() are currently stubs in the test runner
+ std::string garbage("thisissomegarbagetrailingdata");
+ BOOST_CHECK_EQUAL(dnsdist_ffi_dnsquestion_set_trailing_data(&lightDQ, garbage.data(), garbage.size()), true);
+ const char* buffer = nullptr;
+ BOOST_REQUIRE_EQUAL(dnsdist_ffi_dnsquestion_get_trailing_data(&lightDQ, &buffer), garbage.size());
+ BOOST_CHECK_EQUAL(garbage, std::string(buffer));
+#endif
+ }
+
+ {
+#if 0
+ // SpoofAction::operator() is a stub in the test runner
+ auto oldData = dq.getData();
+ std::vector<dnsdist_ffi_raw_value> values;
+ ComboAddress v4("192.0.2.1");
+ ComboAddress v6("[2001:db8::42]");
+ values.push_back({ reinterpret_cast<const char*>(&v4.sin4.sin_addr.s_addr), sizeof(v4.sin4.sin_addr.s_addr)});
+ values.push_back({ reinterpret_cast<const char*>(&v6.sin6.sin6_addr.s6_addr), sizeof(v6.sin6.sin6_addr.s6_addr)});
+
+ dnsdist_ffi_dnsquestion_spoof_addrs(&lightDQ, values.data(), values.size());
+ BOOST_CHECK(dq.getData().size() > oldData.size());
+
+ MOADNSParser mdp(false, reinterpret_cast<const char*>(dq.getData().data()), dq.getData().size());
+ BOOST_CHECK_EQUAL(mdp.d_qname, ids.qname);
+ BOOST_CHECK_EQUAL(mdp.d_header.qdcount, 1U);
+ BOOST_CHECK_EQUAL(mdp.d_header.ancount, values.size());
+ BOOST_CHECK_EQUAL(mdp.d_header.nscount, 0U);
+ BOOST_CHECK_EQUAL(mdp.d_header.arcount, 0U);
+
+ BOOST_REQUIRE_EQUAL(mdp.d_answers.size(), 1U);
+ BOOST_CHECK_EQUAL(mdp.d_answers.at(0).first.d_type, static_cast<uint16_t>(QType::A));
+ BOOST_CHECK_EQUAL(mdp.d_answers.at(0).first.d_class, QClass::IN);
+ BOOST_CHECK_EQUAL(mdp.d_answers.at(0).first.d_name, ids.qname);
+ BOOST_CHECK_EQUAL(mdp.d_answers.at(1).first.d_type, static_cast<uint16_t>(QType::AAAA));
+ BOOST_CHECK_EQUAL(mdp.d_answers.at(1).first.d_class, QClass::IN);
+ BOOST_CHECK_EQUAL(mdp.d_answers.at(1).first.d_name, ids.qname);
+
+ dq.getMutableData() = oldData;
+#endif
+ }
+
+ {
+ BOOST_CHECK(!dnsdist_ffi_dnsquestion_set_restartable(nullptr));
+ BOOST_CHECK(dnsdist_ffi_dnsquestion_set_restartable(&lightDQ));
+ }
+
+ {
+ BOOST_CHECK_EQUAL(ids.ttlCap, 0U);
+ dnsdist_ffi_dnsquestion_set_max_returned_ttl(&lightDQ, 42U);
+ BOOST_CHECK_EQUAL(ids.ttlCap, 42U);
+ }
+
+ {
+ const std::string tagName("my-tag");
+ const std::string tagValue("my-value");
+ const std::string tagRawValue("my-\0-binary-value");
+ std::string buffer;
+ buffer.resize(512);
+ BOOST_CHECK(dnsdist_ffi_dnsquestion_get_tag(nullptr, nullptr) == nullptr);
+ BOOST_CHECK(dnsdist_ffi_dnsquestion_get_tag(&lightDQ, nullptr) == nullptr);
+ BOOST_CHECK(dnsdist_ffi_dnsquestion_get_tag(&lightDQ, tagName.c_str()) == nullptr);
+
+ BOOST_CHECK_EQUAL(dnsdist_ffi_dnsquestion_get_tag_raw(nullptr, nullptr, nullptr, 0), 0U);
+ BOOST_CHECK_EQUAL(dnsdist_ffi_dnsquestion_get_tag_raw(&lightDQ, tagName.c_str(), buffer.data(), buffer.size()), 0U);
+
+ dnsdist_ffi_dnsquestion_set_tag(&lightDQ, tagName.c_str(), tagValue.c_str());
+
+ auto got = dnsdist_ffi_dnsquestion_get_tag(&lightDQ, tagName.c_str());
+ BOOST_CHECK(got != nullptr);
+ BOOST_CHECK_EQUAL(got, tagValue.c_str());
+
+ const dnsdist_ffi_tag_t* tags = nullptr;
+ BOOST_CHECK_EQUAL(dnsdist_ffi_dnsquestion_get_tag_array(nullptr, nullptr), 0U);
+ BOOST_REQUIRE_EQUAL(dnsdist_ffi_dnsquestion_get_tag_array(&lightDQ, &tags), 1U);
+ BOOST_CHECK_EQUAL(std::string(tags[0].name), tagName.c_str());
+ BOOST_CHECK_EQUAL(std::string(tags[0].value), tagValue.c_str());
+
+ dnsdist_ffi_dnsquestion_set_tag_raw(&lightDQ, tagName.c_str(), tagRawValue.c_str(), tagRawValue.size());
+
+ // too small
+ buffer.resize(1);
+ BOOST_CHECK_EQUAL(dnsdist_ffi_dnsquestion_get_tag_raw(&lightDQ, tagName.c_str(), buffer.data(), buffer.size()), 0U);
+
+ buffer.resize(tagRawValue.size());
+ BOOST_CHECK_EQUAL(dnsdist_ffi_dnsquestion_get_tag_raw(&lightDQ, tagName.c_str(), buffer.data(), buffer.size()), tagRawValue.size());
+ BOOST_CHECK_EQUAL(buffer, tagRawValue);
+
+ BOOST_CHECK_EQUAL(dnsdist_ffi_dnsquestion_get_tag_raw(&lightDQ, "wrong tag name", buffer.data(), buffer.size()), 0U);
+
+ // dnsdist_ffi_dnsquestion_get_tag_array
+
+ {
+ // no DOHUnit attached
+ BOOST_CHECK(dnsdist_ffi_dnsquestion_get_http_path(&lightDQ) == nullptr);
+ BOOST_CHECK(dnsdist_ffi_dnsquestion_get_http_query_string(&lightDQ) == nullptr);
+ BOOST_CHECK(dnsdist_ffi_dnsquestion_get_http_host(&lightDQ) == nullptr);
+ BOOST_CHECK(dnsdist_ffi_dnsquestion_get_http_scheme(&lightDQ) == nullptr);
+ BOOST_CHECK_EQUAL(dnsdist_ffi_dnsquestion_get_http_headers(&lightDQ, nullptr), 0U);
+ dnsdist_ffi_dnsquestion_set_http_response(&lightDQ, 0U, nullptr, 0U, nullptr);
+ }
+ }
+
+ const std::string deviceID{"my-device-id"};
+ const std::string deviceName{"my-device-name"};
+ const std::string requestorID{"my-requestor-ID"};
+ dnsdist_ffi_dnsquestion_set_device_id(nullptr, nullptr, 0);
+ dnsdist_ffi_dnsquestion_set_device_id(&lightDQ, nullptr, 0);
+ dnsdist_ffi_dnsquestion_set_device_id(&lightDQ, deviceID.c_str(), deviceID.size());
+ dnsdist_ffi_dnsquestion_set_device_name(nullptr, nullptr, 0);
+ dnsdist_ffi_dnsquestion_set_device_name(&lightDQ, nullptr, 0);
+ dnsdist_ffi_dnsquestion_set_device_name(&lightDQ, deviceName.c_str(), deviceName.size());
+ dnsdist_ffi_dnsquestion_set_requestor_id(nullptr, nullptr, 0);
+ dnsdist_ffi_dnsquestion_set_requestor_id(&lightDQ, nullptr, 0);
+ dnsdist_ffi_dnsquestion_set_requestor_id(&lightDQ, requestorID.c_str(), requestorID.size());
+ BOOST_REQUIRE(ids.d_protoBufData != nullptr);
+ BOOST_CHECK_EQUAL(ids.d_protoBufData->d_deviceID, deviceID);
+ BOOST_CHECK_EQUAL(ids.d_protoBufData->d_deviceName, deviceName);
+ BOOST_CHECK_EQUAL(ids.d_protoBufData->d_requestorID, requestorID);
+}
+
+BOOST_AUTO_TEST_CASE(test_Response)
+{
+ InternalQueryState ids;
+ ids.origRemote = ComboAddress("192.0.2.1:4242");
+ ids.origDest = ComboAddress("192.0.2.255:53");
+ ids.qtype = QType::A;
+ ids.qclass = QClass::IN;
+ ids.protocol = dnsdist::Protocol::DoUDP;
+ ids.qname = DNSName("www.powerdns.com.");
+ ids.queryRealTime.start();
+
+ PacketBuffer response;
+ GenericDNSPacketWriter<PacketBuffer> pwR(response, ids.qname, QType::A, QClass::IN, 0);
+ pwR.getHeader()->qr = 1;
+ pwR.getHeader()->rd = 1;
+ pwR.getHeader()->id = htons(42);
+
+ ComboAddress dsAddr("192.0.2.1:53");
+ auto ds = std::make_shared<DownstreamState>(dsAddr);
+
+ DNSResponse dr(ids, response, ds);
+ dnsdist_ffi_dnsresponse_t lightDR(&dr);
+
+ {
+ dnsdist_ffi_dnsresponse_set_min_ttl(&lightDR, 42);
+ dnsdist_ffi_dnsresponse_set_max_ttl(&lightDR, 84);
+ dnsdist_ffi_dnsresponse_limit_ttl(&lightDR, 42, 84);
+ }
+
+ {
+ BOOST_CHECK_EQUAL(ids.ttlCap, 0U);
+ dnsdist_ffi_dnsresponse_set_max_returned_ttl(&lightDR, 42);
+ BOOST_CHECK_EQUAL(ids.ttlCap, 42U);
+ }
+
+ {
+ /* invalid parameters */
+ BOOST_CHECK(!dnsdist_ffi_dnsresponse_rebase(&lightDR, nullptr, 0));
+
+ /* invalid name */
+ BOOST_CHECK(!dnsdist_ffi_dnsresponse_rebase(&lightDR, "\5AAAA", 5));
+
+ DNSName newName("not-powerdns.com.");
+ BOOST_CHECK(dnsdist_ffi_dnsresponse_rebase(&lightDR, newName.getStorage().data(), newName.getStorage().size()));
+ BOOST_CHECK_EQUAL(ids.qname.toString(), newName.toString());
+ }
+
+ {
+ dnsdist_ffi_dnsresponse_clear_records_type(nullptr, QType::A);
+ dnsdist_ffi_dnsresponse_clear_records_type(&lightDR, QType::A);
+ }
+}
+
+BOOST_AUTO_TEST_CASE(test_Server)
+{
+ ComboAddress dsAddr("192.0.2.1:53");
+ auto ds = std::make_shared<DownstreamState>(dsAddr);
+ dnsdist_ffi_server_t server(ds);
+
+ BOOST_CHECK_EQUAL(dnsdist_ffi_server_get_outstanding(&server), 0U);
+ BOOST_CHECK_EQUAL(dnsdist_ffi_server_is_up(&server), false);
+ BOOST_CHECK_EQUAL(dnsdist_ffi_server_get_name(&server), "");
+ BOOST_CHECK_EQUAL(dnsdist_ffi_server_get_name_with_addr(&server), dsAddr.toStringWithPort());
+ BOOST_CHECK_EQUAL(dnsdist_ffi_server_get_weight(&server), 1U);
+ BOOST_CHECK_EQUAL(dnsdist_ffi_server_get_order(&server), 1U);
+ BOOST_CHECK_EQUAL(dnsdist_ffi_server_get_latency(&server), 0.0);
+}
+
+BOOST_AUTO_TEST_CASE(test_PacketCache)
+{
+ auto packetCache = std::make_shared<DNSDistPacketCache>(10);
+
+ ComboAddress ipv4("192.0.2.1");
+ InternalQueryState ids;
+ ids.qtype = QType::A;
+ ids.qclass = QClass::IN;
+ ids.protocol = dnsdist::Protocol::DoUDP;
+ ids.qname = DNSName("powerdns.com.");
+ PacketBuffer query;
+ GenericDNSPacketWriter<PacketBuffer> pwQ(query, ids.qname, QType::A, QClass::IN, 0);
+ pwQ.getHeader()->rd = 1;
+
+ PacketBuffer response;
+ GenericDNSPacketWriter<PacketBuffer> pwR(response, ids.qname, QType::A, QClass::IN, 0);
+ pwR.getHeader()->id = pwQ.getHeader()->id;
+ pwR.startRecord(ids.qname, QType::A, 7200, QClass::IN, DNSResourceRecord::ANSWER);
+ pwR.xfrCAWithoutPort(4, ipv4);
+ pwR.commit();
+
+ bool dnssecOK = true;
+ bool receivedOverUDP = true;
+ uint32_t key = 0;
+ boost::optional<Netmask> subnet;
+ ids.queryRealTime.start();
+ DNSQuestion dq(ids, query);
+ packetCache->get(dq, 0, &key, subnet, dnssecOK, receivedOverUDP);
+ packetCache->insert(key, subnet, *(getFlagsFromDNSHeader(dq.getHeader())), dnssecOK, ids.qname, QType::A, QClass::IN, response, receivedOverUDP, 0, boost::none);
+
+ std::string poolName("test-pool");
+ auto testPool = std::make_shared<ServerPool>();
+ testPool->packetCache = packetCache;
+ std::string poolWithNoCacheName("test-pool-without-cache");
+ auto testPoolWithNoCache = std::make_shared<ServerPool>();
+ auto localPools = g_pools.getCopy();
+ localPools.emplace(poolName, testPool);
+ localPools.emplace(poolWithNoCacheName, testPoolWithNoCache);
+ g_pools.setState(localPools);
+
+ {
+ dnsdist_ffi_domain_list_t* list = nullptr;
+ {
+ // invalid parameters
+ BOOST_CHECK_EQUAL(dnsdist_ffi_packetcache_get_domain_list_by_addr(nullptr, nullptr, nullptr), 0U);
+ BOOST_CHECK_EQUAL(dnsdist_ffi_packetcache_get_domain_list_by_addr("not-existing-pool", ipv4.toString().c_str(), &list), 0U);
+ BOOST_CHECK_EQUAL(dnsdist_ffi_packetcache_get_domain_list_by_addr(poolName.c_str(), "invalid-address", &list), 0U);
+ BOOST_CHECK_EQUAL(dnsdist_ffi_packetcache_get_domain_list_by_addr(poolWithNoCacheName.c_str(), ipv4.toString().c_str(), &list), 0U);
+ }
+
+ {
+ // no match
+ BOOST_CHECK_EQUAL(dnsdist_ffi_packetcache_get_domain_list_by_addr(poolName.c_str(), ComboAddress("192.0.2.254").toString().c_str(), &list), 0U);
+ }
+
+ auto got = dnsdist_ffi_packetcache_get_domain_list_by_addr(poolName.c_str(), ipv4.toString().c_str(), &list);
+ BOOST_REQUIRE_EQUAL(got, 1U);
+ BOOST_REQUIRE(list != nullptr);
+
+ {
+ // invalid parameters
+ BOOST_CHECK(dnsdist_ffi_domain_list_get(nullptr, 0) == nullptr);
+ BOOST_CHECK(dnsdist_ffi_domain_list_get(list, 1) == nullptr);
+ }
+
+ {
+ const char* domain = dnsdist_ffi_domain_list_get(list, 0);
+ BOOST_CHECK(domain == ids.qname.toString());
+ }
+
+ dnsdist_ffi_domain_list_free(list);
+ }
+
+ {
+ dnsdist_ffi_address_list_t* addresses = nullptr;
+ {
+ // invalid parameters
+ BOOST_CHECK_EQUAL(dnsdist_ffi_packetcache_get_address_list_by_domain(nullptr, nullptr, nullptr), 0U);
+ BOOST_CHECK_EQUAL(dnsdist_ffi_packetcache_get_address_list_by_domain("not-existing-pool", ids.qname.toString().c_str(), &addresses), 0U);
+ BOOST_CHECK_EQUAL(dnsdist_ffi_packetcache_get_address_list_by_domain(poolName.c_str(), "invalid-dns...name", &addresses), 0U);
+ BOOST_CHECK_EQUAL(dnsdist_ffi_packetcache_get_address_list_by_domain(poolWithNoCacheName.c_str(), ipv4.toString().c_str(), &addresses), 0U);
+ }
+
+ {
+ // no match
+ BOOST_CHECK_EQUAL(dnsdist_ffi_packetcache_get_address_list_by_domain(poolName.c_str(), "wrong.name.", &addresses), 0U);
+ }
+
+ auto got = dnsdist_ffi_packetcache_get_address_list_by_domain(poolName.c_str(), ids.qname.toString().c_str(), &addresses);
+ BOOST_REQUIRE_EQUAL(got, 1U);
+ BOOST_REQUIRE(addresses != nullptr);
+
+ {
+ // invalid parameters
+ BOOST_CHECK(dnsdist_ffi_address_list_get(nullptr, 0) == nullptr);
+ BOOST_CHECK(dnsdist_ffi_address_list_get(addresses, 1) == nullptr);
+ }
+
+ {
+ const char* addr = dnsdist_ffi_address_list_get(addresses, 0);
+ BOOST_CHECK(addr == ipv4.toString());
+ }
+
+ dnsdist_ffi_address_list_free(addresses);
+ }
+}
+
+BOOST_AUTO_TEST_CASE(test_ProxyProtocol)
+{
+ ComboAddress v4("192.0.2.1");
+ ComboAddress v6("[2001:db8::42]");
+
+ std::vector<dnsdist_ffi_proxy_protocol_value> values;
+ values.push_back({"test-value", 10U, 1U});
+
+ std::vector<uint8_t> output;
+ output.resize(4096);
+
+ {
+ // too small buffer
+ auto got = dnsdist_ffi_generate_proxy_protocol_payload(sizeof(v4.sin4.sin_addr.s_addr), &v4.sin4.sin_addr.s_addr, &v4.sin4.sin_addr.s_addr, 4242U, 53U, true, values.size(), values.data(), output.data(), 0);
+ BOOST_CHECK_EQUAL(got, 0U);
+ }
+
+ {
+ // invalid address size
+ auto got = dnsdist_ffi_generate_proxy_protocol_payload(0U, &v4.sin4.sin_addr.s_addr, &v4.sin4.sin_addr.s_addr, 4242U, 53U, true, values.size(), values.data(), output.data(), 0);
+ BOOST_CHECK_EQUAL(got, 0U);
+ }
+
+ {
+ auto got = dnsdist_ffi_generate_proxy_protocol_payload(sizeof(v4.sin4.sin_addr.s_addr), &v4.sin4.sin_addr.s_addr, &v4.sin4.sin_addr.s_addr, 4242U, 53U, true, values.size(), values.data(), output.data(), output.size());
+ BOOST_CHECK_EQUAL(got, 41U);
+ }
+}
+
+BOOST_AUTO_TEST_CASE(test_PacketOverlay)
+{
+ const DNSName target("powerdns.com.");
+ PacketBuffer response;
+ GenericDNSPacketWriter<PacketBuffer> pwR(response, target, QType::A, QClass::IN, 0);
+ pwR.getHeader()->qr = 1;
+ pwR.getHeader()->rd = 1;
+ pwR.getHeader()->ra = 1;
+ pwR.getHeader()->id = htons(42);
+ pwR.startRecord(target, QType::A, 7200, QClass::IN, DNSResourceRecord::ANSWER);
+ ComboAddress v4("192.0.2.1");
+ pwR.xfrCAWithoutPort(4, v4);
+ pwR.commit();
+ pwR.startRecord(target, QType::AAAA, 7200, QClass::IN, DNSResourceRecord::ADDITIONAL);
+ ComboAddress v6("2001:db8::1");
+ pwR.xfrCAWithoutPort(6, v6);
+ pwR.commit();
+ pwR.addOpt(4096, 0, 0);
+ pwR.commit();
+
+ /* invalid parameters */
+ BOOST_CHECK(!dnsdist_ffi_dnspacket_parse(nullptr, 0, nullptr));
+
+ dnsdist_ffi_dnspacket_t* packet = nullptr;
+ // invalid packet
+ BOOST_CHECK(!dnsdist_ffi_dnspacket_parse(reinterpret_cast<const char*>(response.data()), response.size() - 1, &packet));
+ BOOST_REQUIRE(dnsdist_ffi_dnspacket_parse(reinterpret_cast<const char*>(response.data()), response.size(), &packet));
+ BOOST_REQUIRE(packet != nullptr);
+
+ const char* qname = nullptr;
+ size_t qnameSize = 0;
+
+ // invalid parameters
+ dnsdist_ffi_dnspacket_get_qname_raw(nullptr, nullptr, 0);
+ BOOST_CHECK_EQUAL(dnsdist_ffi_dnspacket_get_qtype(nullptr), 0U);
+ BOOST_CHECK_EQUAL(dnsdist_ffi_dnspacket_get_qclass(nullptr), 0U);
+ BOOST_CHECK_EQUAL(dnsdist_ffi_dnspacket_get_qtype(packet), QType::A);
+ BOOST_CHECK_EQUAL(dnsdist_ffi_dnspacket_get_qclass(packet), QClass::IN);
+
+ BOOST_CHECK_EQUAL(dnsdist_ffi_dnspacket_get_records_count_in_section(nullptr, 0), 0U);
+ BOOST_CHECK_EQUAL(dnsdist_ffi_dnspacket_get_records_count_in_section(packet, 0), 0U);
+ BOOST_CHECK_EQUAL(dnsdist_ffi_dnspacket_get_records_count_in_section(packet, 1), 1U);
+ BOOST_CHECK_EQUAL(dnsdist_ffi_dnspacket_get_records_count_in_section(packet, 2), 0U);
+ BOOST_CHECK_EQUAL(dnsdist_ffi_dnspacket_get_records_count_in_section(packet, 3), 2U);
+ BOOST_CHECK_EQUAL(dnsdist_ffi_dnspacket_get_records_count_in_section(packet, 4), 0U);
+
+ dnsdist_ffi_dnspacket_get_qname_raw(packet, &qname, &qnameSize);
+ BOOST_REQUIRE(qname != nullptr);
+ BOOST_REQUIRE_EQUAL(qnameSize, target.wirelength());
+ BOOST_CHECK_EQUAL(memcmp(qname, target.getStorage().data(), target.getStorage().size()), 0);
+
+ {
+ std::string parsedName;
+ parsedName.resize(1024);
+
+ // too small
+ BOOST_CHECK_EQUAL(dnsdist_ffi_dnspacket_get_name_at_offset_raw(reinterpret_cast<const char*>(response.data()), response.size(), sizeof(dnsheader), parsedName.data(), 1U), 0U);
+ // invalid parameters
+ BOOST_CHECK_EQUAL(dnsdist_ffi_dnspacket_get_name_at_offset_raw(nullptr, 0, sizeof(dnsheader), parsedName.data(), parsedName.size()), 0U);
+ // invalid packet
+ BOOST_CHECK_EQUAL(dnsdist_ffi_dnspacket_get_name_at_offset_raw(reinterpret_cast<const char*>(response.data()), sizeof(dnsheader) + 2, sizeof(dnsheader), parsedName.data(), parsedName.size()), 0U);
+
+ auto parsedNameSize = dnsdist_ffi_dnspacket_get_name_at_offset_raw(reinterpret_cast<const char*>(response.data()), response.size(), sizeof(dnsheader), parsedName.data(), parsedName.size());
+ BOOST_REQUIRE_GT(parsedNameSize, 0U);
+ BOOST_REQUIRE_EQUAL(parsedNameSize, target.wirelength());
+ BOOST_CHECK_EQUAL(memcmp(parsedName.c_str(), target.getStorage().data(), target.getStorage().size()), 0);
+ }
+
+ const char* name = nullptr;
+ size_t nameSize = 0;
+ dnsdist_ffi_dnspacket_get_record_name_raw(nullptr, 0, nullptr, 0);
+ BOOST_REQUIRE(name == nullptr);
+
+ // invalid parameters
+ BOOST_CHECK_EQUAL(dnsdist_ffi_dnspacket_get_record_type(nullptr, 0), 0U);
+ BOOST_CHECK_EQUAL(dnsdist_ffi_dnspacket_get_record_class(nullptr, 0), 0U);
+ BOOST_CHECK_EQUAL(dnsdist_ffi_dnspacket_get_record_ttl(nullptr, 0), 0U);
+ BOOST_CHECK_EQUAL(dnsdist_ffi_dnspacket_get_record_content_length(nullptr, 0), 0U);
+ BOOST_CHECK_EQUAL(dnsdist_ffi_dnspacket_get_record_content_offset(nullptr, 0), 0U);
+ BOOST_CHECK_EQUAL(dnsdist_ffi_dnspacket_get_name_at_offset_raw(nullptr, 0, 0, nullptr, 0), 0U);
+
+ // first record */
+ dnsdist_ffi_dnspacket_get_record_name_raw(packet, 0, &name, &nameSize);
+ BOOST_REQUIRE(name != nullptr);
+ BOOST_REQUIRE_EQUAL(nameSize, target.wirelength());
+ BOOST_CHECK_EQUAL(memcmp(name, target.getStorage().data(), target.getStorage().size()), 0);
+ BOOST_CHECK_EQUAL(dnsdist_ffi_dnspacket_get_record_type(packet, 0), QType::A);
+ BOOST_CHECK_EQUAL(dnsdist_ffi_dnspacket_get_record_class(packet, 0), QClass::IN);
+ BOOST_CHECK_EQUAL(dnsdist_ffi_dnspacket_get_record_ttl(packet, 0), 7200U);
+ BOOST_CHECK_EQUAL(dnsdist_ffi_dnspacket_get_record_content_length(packet, 0), sizeof(v4.sin4.sin_addr.s_addr));
+ BOOST_CHECK_EQUAL(dnsdist_ffi_dnspacket_get_record_content_offset(packet, 0), 42U);
+
+ // second record
+ dnsdist_ffi_dnspacket_get_record_name_raw(packet, 1, &name, &nameSize);
+ BOOST_REQUIRE(name != nullptr);
+ BOOST_REQUIRE_EQUAL(nameSize, target.wirelength());
+ BOOST_CHECK_EQUAL(memcmp(name, target.getStorage().data(), target.getStorage().size()), 0);
+ BOOST_CHECK_EQUAL(dnsdist_ffi_dnspacket_get_record_type(packet, 1), QType::AAAA);
+ BOOST_CHECK_EQUAL(dnsdist_ffi_dnspacket_get_record_class(packet, 1), QClass::IN);
+ BOOST_CHECK_EQUAL(dnsdist_ffi_dnspacket_get_record_ttl(packet, 1), 7200U);
+ BOOST_CHECK_EQUAL(dnsdist_ffi_dnspacket_get_record_content_length(packet, 1), sizeof(v6.sin6.sin6_addr.s6_addr));
+ BOOST_CHECK_EQUAL(dnsdist_ffi_dnspacket_get_record_content_offset(packet, 1), 58U);
+
+ dnsdist_ffi_dnspacket_free(packet);
+}
+
+BOOST_AUTO_TEST_CASE(test_RingBuffers)
+{
+ dnsheader dh;
+ memset(&dh, 0, sizeof(dh));
+ DNSName qname("rings.luaffi.powerdns.com.");
+ ComboAddress requestor1("192.0.2.1");
+ ComboAddress backend("192.0.2.42");
+ uint16_t qtype = QType::AAAA;
+ uint16_t size = 42;
+ dnsdist::Protocol protocol = dnsdist::Protocol::DoUDP;
+ dnsdist::Protocol outgoingProtocol = dnsdist::Protocol::DoUDP;
+ unsigned int responseTime = 0;
+ struct timespec now;
+ gettime(&now);
+
+ g_rings.reset();
+ g_rings.init();
+ BOOST_CHECK_EQUAL(g_rings.getNumberOfQueryEntries(), 0U);
+
+ g_rings.insertQuery(now, requestor1, qname, qtype, size, dh, protocol);
+ g_rings.insertResponse(now, requestor1, qname, qtype, responseTime, size, dh, backend, outgoingProtocol);
+
+ dnsdist_ffi_ring_entry_list_t* list = nullptr;
+
+ {
+ // invalid
+ BOOST_CHECK_EQUAL(dnsdist_ffi_ring_get_entries(nullptr), 0U);
+ BOOST_CHECK(list == nullptr);
+ BOOST_CHECK_EQUAL(dnsdist_ffi_ring_get_entries_by_addr(requestor1.toString().c_str(), nullptr), 0U);
+ BOOST_CHECK_EQUAL(dnsdist_ffi_ring_get_entries_by_addr(nullptr, &list), 0U);
+ BOOST_CHECK(list == nullptr);
+ BOOST_CHECK_EQUAL(dnsdist_ffi_ring_get_entries_by_addr("invalid-address", &list), 0U);
+ BOOST_CHECK(list == nullptr);
+ BOOST_CHECK_EQUAL(dnsdist_ffi_ring_get_entries_by_mac(nullptr, nullptr), 0U);
+ BOOST_CHECK(list == nullptr);
+ BOOST_CHECK(!dnsdist_ffi_ring_entry_is_response(nullptr, 0));
+ BOOST_CHECK(dnsdist_ffi_ring_entry_get_name(nullptr, 0) == nullptr);
+ BOOST_CHECK(dnsdist_ffi_ring_entry_get_type(nullptr, 0) == 0);
+ BOOST_CHECK(dnsdist_ffi_ring_entry_get_requestor(nullptr, 0) == nullptr);
+ BOOST_CHECK(dnsdist_ffi_ring_entry_get_protocol(nullptr, 0) == 0);
+ BOOST_CHECK(dnsdist_ffi_ring_entry_get_size(nullptr, 0) == 0);
+ BOOST_CHECK(!dnsdist_ffi_ring_entry_has_mac_address(nullptr, 0));
+ BOOST_CHECK(dnsdist_ffi_ring_entry_get_mac_address(nullptr, 0) == nullptr);
+ }
+
+ BOOST_REQUIRE_EQUAL(dnsdist_ffi_ring_get_entries(&list), 2U);
+ BOOST_CHECK(list != nullptr);
+
+ BOOST_CHECK(!dnsdist_ffi_ring_entry_is_response(list, 0));
+ BOOST_CHECK(dnsdist_ffi_ring_entry_is_response(list, 1));
+
+ for (size_t idx = 0; idx < 2; idx++) {
+ BOOST_CHECK(dnsdist_ffi_ring_entry_get_name(list, idx) == qname.toString());
+ BOOST_CHECK(dnsdist_ffi_ring_entry_get_type(list, idx) == qtype);
+ BOOST_CHECK(dnsdist_ffi_ring_entry_get_requestor(list, idx) == requestor1.toString());
+ BOOST_CHECK(dnsdist_ffi_ring_entry_get_protocol(list, idx) == protocol.toNumber());
+ BOOST_CHECK_EQUAL(dnsdist_ffi_ring_entry_get_size(list, idx), size);
+ BOOST_CHECK(!dnsdist_ffi_ring_entry_has_mac_address(list, idx));
+ BOOST_CHECK(dnsdist_ffi_ring_entry_get_mac_address(list, idx) == std::string());
+ }
+
+ dnsdist_ffi_ring_entry_list_free(list);
+ list = nullptr;
+
+ // no the right requestor
+ BOOST_REQUIRE_EQUAL(dnsdist_ffi_ring_get_entries_by_addr("192.0.2.2", &list), 0U);
+ BOOST_CHECK(list == nullptr);
+
+ BOOST_REQUIRE_EQUAL(dnsdist_ffi_ring_get_entries_by_addr(requestor1.toString().c_str(), &list), 2U);
+ BOOST_CHECK(list != nullptr);
+ dnsdist_ffi_ring_entry_list_free(list);
+ list = nullptr;
+}
+
+BOOST_AUTO_TEST_CASE(test_NetworkEndpoint)
+{
+ {
+ dnsdist_ffi_network_endpoint_t* endpoint = nullptr;
+ BOOST_CHECK(!dnsdist_ffi_network_endpoint_new("a", 1, nullptr));
+ BOOST_CHECK(!dnsdist_ffi_network_endpoint_new(nullptr, 1, &endpoint));
+ BOOST_CHECK(!dnsdist_ffi_network_endpoint_new("a", 0, &endpoint));
+ // the path does not exist
+ BOOST_CHECK(!dnsdist_ffi_network_endpoint_new("a", 1, &endpoint));
+ }
+
+ {
+ BOOST_CHECK(!dnsdist_ffi_network_endpoint_is_valid(nullptr));
+ }
+
+ {
+ dnsdist_ffi_network_endpoint_t* endpoint = nullptr;
+ BOOST_CHECK(!dnsdist_ffi_network_endpoint_send(nullptr, "a", 1));
+ BOOST_CHECK(!dnsdist_ffi_network_endpoint_send(endpoint, nullptr, 1));
+ }
+
+ {
+ dnsdist_ffi_network_endpoint_free(nullptr);
+ }
+}
+
+BOOST_AUTO_TEST_SUITE_END();