--- /dev/null
+#define BOOST_TEST_DYN_LINK
+#include <boost/test/unit_test.hpp>
+
+#include <stdio.h>
+
+#include "dnsname.hh"
+#include "qtype.hh"
+#include "taskqueue.hh"
+#include "rec-taskqueue.hh"
+#include "test-syncres_cc.hh"
+
+BOOST_AUTO_TEST_SUITE(rec_taskqueue)
+
+BOOST_AUTO_TEST_CASE(test_almostexpired_queue_no_dups)
+{
+ taskQueueClear();
+ pushAlmostExpiredTask(DNSName("foo"), QType::AAAA, 0);
+ pushAlmostExpiredTask(DNSName("foo"), QType::AAAA, 0);
+ pushAlmostExpiredTask(DNSName("foo"), QType::A, 0);
+
+ BOOST_CHECK_EQUAL(getTaskSize(), 2U);
+ taskQueuePop();
+ taskQueuePop();
+ BOOST_CHECK_EQUAL(getTaskSize(), 0U);
+ // AE queue is not rate limited
+ pushAlmostExpiredTask(DNSName("foo"), QType::A, 0);
+ BOOST_CHECK_EQUAL(getTaskSize(), 1U);
+}
+
+BOOST_AUTO_TEST_CASE(test_resolve_queue_rate_limit)
+{
+ taskQueueClear();
+ pushResolveTask(DNSName("foo"), QType::AAAA, 0, 1);
+ BOOST_CHECK_EQUAL(getTaskSize(), 1U);
+ taskQueuePop();
+ BOOST_CHECK_EQUAL(getTaskSize(), 0U);
+
+ // Should hit rate limiting
+ pushResolveTask(DNSName("foo"), QType::AAAA, 0, 1);
+ BOOST_CHECK_EQUAL(getTaskSize(), 0U);
+
+ // Should not hit rate limiting as time has passed
+ pushResolveTask(DNSName("foo"), QType::AAAA, 61, 62);
+ BOOST_CHECK_EQUAL(getTaskSize(), 1U);
+}
+
+BOOST_AUTO_TEST_SUITE_END()
#include <boost/test/unit_test.hpp>
#include "test-syncres_cc.hh"
+#include "taskqueue.hh"
+#include "rec-taskqueue.hh"
BOOST_AUTO_TEST_SUITE(syncres_cc1)
BOOST_CHECK_EQUAL(ret[0].d_name, target);
}
+BOOST_AUTO_TEST_CASE(test_glueless_referral_aaaa_task)
+{
+ std::unique_ptr<SyncRes> sr;
+ initSR(sr);
+
+ primeHints();
+
+ const DNSName target("powerdns.com.");
+
+ sr->setAsyncCallback([target](const ComboAddress& ip, const DNSName& domain, int type, bool doTCP, bool sendRDQuery, int EDNS0Level, struct timeval* now, boost::optional<Netmask>& srcmask, boost::optional<const ResolveContext&> context, LWResult* res, bool* chained) {
+ if (isRootServer(ip)) {
+ setLWResult(res, 0, false, false, true);
+
+ if (domain.isPartOf(DNSName("com."))) {
+ addRecordToLW(res, "com.", QType::NS, "a.gtld-servers.net.", DNSResourceRecord::AUTHORITY, 172800);
+ }
+ else if (domain.isPartOf(DNSName("org."))) {
+ addRecordToLW(res, "org.", QType::NS, "a.gtld-servers.net.", DNSResourceRecord::AUTHORITY, 172800);
+ }
+ else {
+ setLWResult(res, RCode::NXDomain, false, false, true);
+ return LWResult::Result::Success;
+ }
+
+ addRecordToLW(res, "a.gtld-servers.net.", QType::A, "192.0.2.1", DNSResourceRecord::ADDITIONAL, 3600);
+ addRecordToLW(res, "a.gtld-servers.net.", QType::AAAA, "2001:DB8::1", DNSResourceRecord::ADDITIONAL, 3600);
+ return LWResult::Result::Success;
+ }
+ else if (ip == ComboAddress("192.0.2.1:53") || ip == ComboAddress("[2001:DB8::1]:53")) {
+ if (domain == target) {
+ setLWResult(res, 0, false, false, true);
+ addRecordToLW(res, "powerdns.com.", QType::NS, "pdns-public-ns1.powerdns.org.", DNSResourceRecord::AUTHORITY, 172800);
+ addRecordToLW(res, "powerdns.com.", QType::NS, "pdns-public-ns2.powerdns.org.", DNSResourceRecord::AUTHORITY, 172800);
+ return LWResult::Result::Success;
+ }
+ else if (domain == DNSName("pdns-public-ns1.powerdns.org.")) {
+ setLWResult(res, 0, true, false, true);
+ if (type == QType::A) {
+ addRecordToLW(res, "pdns-public-ns1.powerdns.org.", QType::A, "192.0.2.2");
+ }
+ else {
+ addRecordToLW(res, "pdns-public-ns1.powerdns.org.", QType::AAAA, "2001:DB8::2");
+ }
+ return LWResult::Result::Success;
+ }
+ else if (domain == DNSName("pdns-public-ns2.powerdns.org.")) {
+ setLWResult(res, 0, true, false, true);
+ if (type == QType::A) {
+ addRecordToLW(res, "pdns-public-ns2.powerdns.org.", QType::A, "192.0.2.3");
+ }
+ else {
+ addRecordToLW(res, "pdns-public-ns2.powerdns.org.", QType::AAAA, "2001:DB8::3");
+ }
+ return LWResult::Result::Success;
+ }
+
+ setLWResult(res, RCode::NXDomain, false, false, true);
+ return LWResult::Result::Success;
+ }
+ else if (ip == ComboAddress("192.0.2.2:53") || ip == ComboAddress("192.0.2.3:53") || ip == ComboAddress("[2001:DB8::2]:53") || ip == ComboAddress("[2001:DB8::3]:53")) {
+ setLWResult(res, 0, true, false, true);
+ addRecordToLW(res, target, QType::A, "192.0.2.4");
+ return LWResult::Result::Success;
+ }
+ else {
+ return LWResult::Result::Timeout;
+ }
+ });
+
+ vector<DNSRecord> ret;
+ int res = sr->beginResolve(target, QType(QType::A), QClass::IN, ret);
+ BOOST_CHECK_EQUAL(res, RCode::NoError);
+ BOOST_REQUIRE_EQUAL(ret.size(), 1U);
+ BOOST_CHECK(ret[0].d_type == QType::A);
+ BOOST_CHECK_EQUAL(ret[0].d_name, target);
+
+ // One task should be submitted
+ BOOST_REQUIRE_EQUAL(getTaskSize(), 1U);
+ auto task = taskQueuePop();
+ BOOST_CHECK(task.d_qname == DNSName("pdns-public-ns1.powerdns.org") || task.d_qname == DNSName("pdns-public-ns2.powerdns.org"));
+ BOOST_CHECK_EQUAL(task.d_qtype, QType::AAAA);
+}
+
BOOST_AUTO_TEST_CASE(test_edns_subnet_by_domain)
{
std::unique_ptr<SyncRes> sr;
sr->setAsyncCallback([dnameOwner, dnameTarget, target, cnameTarget, keys, &queries](const ComboAddress& ip, const DNSName& domain, int type, bool doTCP, bool sendRDQuery, int EDNS0Level, struct timeval* now, boost::optional<Netmask>& srcmask, boost::optional<const ResolveContext&> context, LWResult* res, bool* chained) {
queries++;
/* We don't use the genericDSAndDNSKEYHandler here, as it would deny names existing at the wrong level of the tree, due to the way computeZoneCuts works
- * As such, we need to do some more work to make the answers correct.
- */
+ * As such, we need to do some more work to make the answers correct.
+ */
if (isRootServer(ip)) {
if (domain.countLabels() == 0 && type == QType::DNSKEY) { // .|DNSKEY