2013-12-17 09:08:59 +11:00
|
|
|
/*
|
2018-02-23 09:53:12 +01:00
|
|
|
* Copyright (C) Internet Systems Consortium, Inc. ("ISC")
|
2013-12-17 09:08:59 +11:00
|
|
|
*
|
2016-06-27 14:56:38 +10:00
|
|
|
* This Source Code Form is subject to the terms of the Mozilla Public
|
|
|
|
* License, v. 2.0. If a copy of the MPL was not distributed with this
|
|
|
|
* file, You can obtain one at http://mozilla.org/MPL/2.0/.
|
2018-02-23 09:53:12 +01:00
|
|
|
*
|
|
|
|
* See the COPYRIGHT file distributed with this work for additional
|
|
|
|
* information regarding copyright ownership.
|
2013-12-17 09:08:59 +11:00
|
|
|
*/
|
|
|
|
|
|
|
|
/*! \file */
|
|
|
|
|
|
|
|
#include <config.h>
|
|
|
|
|
|
|
|
#include <atf-c.h>
|
|
|
|
|
|
|
|
#include <unistd.h>
|
|
|
|
#include <stdlib.h>
|
|
|
|
|
|
|
|
#include <dns/db.h>
|
|
|
|
#include <dns/dbiterator.h>
|
|
|
|
#include <dns/journal.h>
|
2017-09-06 09:58:29 +10:00
|
|
|
#include <dns/name.h>
|
|
|
|
#include <dns/rdatalist.h>
|
2013-12-17 09:08:59 +11:00
|
|
|
|
|
|
|
#include "dnstest.h"
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Helper functions
|
|
|
|
*/
|
|
|
|
|
|
|
|
#define BUFLEN 255
|
|
|
|
#define BIGBUFLEN (64 * 1024)
|
|
|
|
#define TEST_ORIGIN "test"
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Individual unit tests
|
|
|
|
*/
|
|
|
|
|
|
|
|
ATF_TC(getoriginnode);
|
|
|
|
ATF_TC_HEAD(getoriginnode, tc) {
|
|
|
|
atf_tc_set_md_var(tc, "descr",
|
|
|
|
"test multiple calls to dns_db_getoriginnode");
|
|
|
|
}
|
|
|
|
ATF_TC_BODY(getoriginnode, tc) {
|
|
|
|
dns_db_t *db = NULL;
|
|
|
|
dns_dbnode_t *node = NULL;
|
2015-01-20 13:29:18 -08:00
|
|
|
isc_mem_t *mymctx = NULL;
|
2013-12-17 09:08:59 +11:00
|
|
|
isc_result_t result;
|
|
|
|
|
2015-01-20 13:29:18 -08:00
|
|
|
result = isc_mem_create(0, 0, &mymctx);
|
2013-12-17 09:08:59 +11:00
|
|
|
ATF_REQUIRE_EQ(result, ISC_R_SUCCESS);
|
|
|
|
|
2015-01-20 13:29:18 -08:00
|
|
|
result = dns_db_create(mymctx, "rbt", dns_rootname, dns_dbtype_zone,
|
2017-09-06 09:58:29 +10:00
|
|
|
dns_rdataclass_in, 0, NULL, &db);
|
2013-12-17 09:08:59 +11:00
|
|
|
ATF_REQUIRE_EQ(result, ISC_R_SUCCESS);
|
|
|
|
|
|
|
|
result = dns_db_getoriginnode(db, &node);
|
2017-09-06 09:58:29 +10:00
|
|
|
ATF_CHECK_EQ(result, ISC_R_SUCCESS);
|
2013-12-17 09:08:59 +11:00
|
|
|
dns_db_detachnode(db, &node);
|
|
|
|
|
|
|
|
result = dns_db_getoriginnode(db, &node);
|
2017-09-06 09:58:29 +10:00
|
|
|
ATF_CHECK_EQ(result, ISC_R_SUCCESS);
|
2013-12-17 09:08:59 +11:00
|
|
|
dns_db_detachnode(db, &node);
|
|
|
|
|
|
|
|
dns_db_detach(&db);
|
2015-01-20 13:29:18 -08:00
|
|
|
isc_mem_detach(&mymctx);
|
2013-12-17 09:08:59 +11:00
|
|
|
}
|
|
|
|
|
2017-09-06 09:58:29 +10:00
|
|
|
ATF_TC(getsetservestalettl);
|
|
|
|
ATF_TC_HEAD(getsetservestalettl, tc) {
|
|
|
|
atf_tc_set_md_var(tc, "descr",
|
|
|
|
"test getservestalettl and setservestalettl");
|
|
|
|
}
|
|
|
|
ATF_TC_BODY(getsetservestalettl, tc) {
|
|
|
|
dns_db_t *db = NULL;
|
|
|
|
isc_mem_t *mymctx = NULL;
|
|
|
|
isc_result_t result;
|
|
|
|
dns_ttl_t ttl;
|
|
|
|
|
|
|
|
result = isc_mem_create(0, 0, &mymctx);
|
|
|
|
ATF_REQUIRE_EQ(result, ISC_R_SUCCESS);
|
|
|
|
|
|
|
|
result = dns_db_create(mymctx, "rbt", dns_rootname, dns_dbtype_cache,
|
|
|
|
dns_rdataclass_in, 0, NULL, &db);
|
|
|
|
ATF_REQUIRE_EQ(result, ISC_R_SUCCESS);
|
|
|
|
|
|
|
|
ttl = 5000;
|
|
|
|
result = dns_db_getservestalettl(db, &ttl);
|
|
|
|
ATF_CHECK_EQ_MSG(result, ISC_R_SUCCESS, "dns_db_getservestalettl");
|
|
|
|
ATF_CHECK_EQ_MSG(ttl, 0, "dns_db_getservestalettl initial value");
|
|
|
|
|
|
|
|
ttl = 6 * 3600;
|
|
|
|
result = dns_db_setservestalettl(db, ttl);
|
|
|
|
ATF_CHECK_EQ_MSG(result, ISC_R_SUCCESS, "dns_db_setservestalettl");
|
|
|
|
|
|
|
|
ttl = 5000;
|
|
|
|
result = dns_db_getservestalettl(db, &ttl);
|
|
|
|
ATF_CHECK_EQ_MSG(result, ISC_R_SUCCESS, "dns_db_getservestalettl");
|
|
|
|
ATF_CHECK_EQ_MSG(ttl, 6 * 3600, "dns_db_getservestalettl update value");
|
|
|
|
|
|
|
|
dns_db_detach(&db);
|
|
|
|
isc_mem_detach(&mymctx);
|
|
|
|
}
|
|
|
|
|
|
|
|
ATF_TC(dns_dbfind_staleok);
|
|
|
|
ATF_TC_HEAD(dns_dbfind_staleok, tc) {
|
|
|
|
atf_tc_set_md_var(tc, "descr",
|
|
|
|
"check DNS_DBFIND_STALEOK works");
|
|
|
|
}
|
|
|
|
ATF_TC_BODY(dns_dbfind_staleok, tc) {
|
|
|
|
dns_db_t *db = NULL;
|
|
|
|
dns_dbnode_t *node = NULL;
|
|
|
|
dns_fixedname_t example_fixed;
|
|
|
|
dns_fixedname_t found_fixed;
|
|
|
|
dns_name_t *example;
|
|
|
|
dns_name_t *found;
|
|
|
|
dns_rdatalist_t rdatalist;
|
|
|
|
dns_rdataset_t rdataset;
|
|
|
|
int count;
|
|
|
|
int pass;
|
|
|
|
isc_mem_t *mymctx = NULL;
|
|
|
|
isc_result_t result;
|
|
|
|
unsigned char data[] = { 0x0a, 0x00, 0x00, 0x01 };
|
|
|
|
|
|
|
|
result = isc_mem_create(0, 0, &mymctx);
|
|
|
|
ATF_REQUIRE_EQ(result, ISC_R_SUCCESS);
|
|
|
|
|
|
|
|
result = dns_db_create(mymctx, "rbt", dns_rootname, dns_dbtype_cache,
|
|
|
|
dns_rdataclass_in, 0, NULL, &db);
|
|
|
|
ATF_REQUIRE_EQ(result, ISC_R_SUCCESS);
|
|
|
|
|
|
|
|
dns_fixedname_init(&example_fixed);
|
|
|
|
example = dns_fixedname_name(&example_fixed);
|
|
|
|
|
|
|
|
dns_fixedname_init(&found_fixed);
|
|
|
|
found = dns_fixedname_name(&found_fixed);
|
|
|
|
|
|
|
|
result = dns_name_fromstring(example, "example", 0, NULL);
|
|
|
|
ATF_REQUIRE_EQ(result, ISC_R_SUCCESS);
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Pass 0: default; no stale processing permitted.
|
|
|
|
* Pass 1: stale processing for 1 second.
|
|
|
|
* Pass 2: stale turned off after being on.
|
|
|
|
*/
|
|
|
|
for (pass = 0; pass < 3; pass++) {
|
|
|
|
dns_rdata_t rdata = DNS_RDATA_INIT;
|
|
|
|
|
|
|
|
/* 10.0.0.1 */
|
|
|
|
rdata.data = data;
|
|
|
|
rdata.length = 4;
|
|
|
|
rdata.rdclass = dns_rdataclass_in;
|
|
|
|
rdata.type = dns_rdatatype_a;
|
|
|
|
|
|
|
|
dns_rdatalist_init(&rdatalist);
|
|
|
|
rdatalist.ttl = 2;
|
|
|
|
rdatalist.type = dns_rdatatype_a;
|
|
|
|
rdatalist.rdclass = dns_rdataclass_in;
|
|
|
|
ISC_LIST_APPEND(rdatalist.rdata, &rdata, link);
|
|
|
|
|
|
|
|
switch (pass) {
|
|
|
|
case 0:
|
|
|
|
/* default: stale processing off */
|
|
|
|
break;
|
|
|
|
case 1:
|
|
|
|
/* turn on stale processing */
|
|
|
|
result = dns_db_setservestalettl(db, 1);
|
|
|
|
ATF_REQUIRE_EQ(result, ISC_R_SUCCESS);
|
|
|
|
break;
|
|
|
|
case 2:
|
|
|
|
/* turn off stale processing */
|
|
|
|
result = dns_db_setservestalettl(db, 0);
|
|
|
|
ATF_REQUIRE_EQ(result, ISC_R_SUCCESS);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
dns_rdataset_init(&rdataset);
|
|
|
|
result = dns_rdatalist_tordataset(&rdatalist, &rdataset);
|
|
|
|
ATF_REQUIRE_EQ(result, ISC_R_SUCCESS);
|
|
|
|
|
|
|
|
result = dns_db_findnode(db, example, ISC_TRUE, &node);
|
|
|
|
ATF_REQUIRE_EQ(result, ISC_R_SUCCESS);
|
|
|
|
|
|
|
|
result = dns_db_addrdataset(db, node, NULL, 0, &rdataset, 0,
|
|
|
|
NULL);
|
|
|
|
ATF_REQUIRE_EQ(result, ISC_R_SUCCESS);
|
|
|
|
|
|
|
|
dns_db_detachnode(db, &node);
|
|
|
|
dns_rdataset_disassociate(&rdataset);
|
|
|
|
|
|
|
|
result = dns_db_find(db, example, NULL, dns_rdatatype_a,
|
|
|
|
0, 0, &node, found, &rdataset, NULL);
|
|
|
|
ATF_REQUIRE_EQ(result, ISC_R_SUCCESS);
|
|
|
|
|
|
|
|
/*
|
|
|
|
* May loop for up to 2 seconds performing non stale lookups.
|
|
|
|
*/
|
|
|
|
count = 0;
|
|
|
|
do {
|
|
|
|
count++;
|
|
|
|
ATF_REQUIRE(count < 21); /* loop sanity */
|
|
|
|
ATF_CHECK_EQ(rdataset.attributes &
|
|
|
|
DNS_RDATASETATTR_STALE, 0);
|
|
|
|
ATF_CHECK(rdataset.ttl > 0);
|
|
|
|
dns_db_detachnode(db, &node);
|
|
|
|
dns_rdataset_disassociate(&rdataset);
|
|
|
|
|
|
|
|
usleep(100000); /* 100 ms */
|
|
|
|
|
|
|
|
result = dns_db_find(db, example, NULL,
|
|
|
|
dns_rdatatype_a, 0, 0,
|
|
|
|
&node, found, &rdataset, NULL);
|
|
|
|
} while (result == ISC_R_SUCCESS);
|
|
|
|
|
|
|
|
ATF_CHECK_EQ(result, ISC_R_NOTFOUND);
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Check whether we can get stale data.
|
|
|
|
*/
|
|
|
|
result = dns_db_find(db, example, NULL, dns_rdatatype_a,
|
|
|
|
DNS_DBFIND_STALEOK, 0,
|
|
|
|
&node, found, &rdataset, NULL);
|
|
|
|
switch (pass) {
|
|
|
|
case 0:
|
|
|
|
ATF_CHECK_EQ(result, ISC_R_NOTFOUND);
|
|
|
|
break;
|
|
|
|
case 1:
|
|
|
|
/*
|
|
|
|
* Should loop for 1 second with stale lookups then
|
|
|
|
* stop.
|
|
|
|
*/
|
|
|
|
count = 0;
|
|
|
|
do {
|
|
|
|
count++;
|
|
|
|
ATF_REQUIRE(count < 50); /* loop sanity */
|
|
|
|
ATF_CHECK_EQ(result, ISC_R_SUCCESS);
|
|
|
|
ATF_CHECK_EQ(rdataset.ttl, 0);
|
|
|
|
ATF_CHECK_EQ(rdataset.attributes &
|
|
|
|
DNS_RDATASETATTR_STALE,
|
|
|
|
DNS_RDATASETATTR_STALE);
|
|
|
|
dns_db_detachnode(db, &node);
|
|
|
|
dns_rdataset_disassociate(&rdataset);
|
|
|
|
|
|
|
|
usleep(100000); /* 100 ms */
|
|
|
|
|
|
|
|
result = dns_db_find(db, example, NULL,
|
|
|
|
dns_rdatatype_a,
|
|
|
|
DNS_DBFIND_STALEOK,
|
|
|
|
0, &node, found,
|
|
|
|
&rdataset, NULL);
|
|
|
|
} while (result == ISC_R_SUCCESS);
|
|
|
|
ATF_CHECK(count > 1);
|
|
|
|
ATF_CHECK(count < 11);
|
|
|
|
ATF_CHECK_EQ(result, ISC_R_NOTFOUND);
|
|
|
|
break;
|
|
|
|
case 2:
|
|
|
|
ATF_CHECK_EQ(result, ISC_R_NOTFOUND);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
dns_db_detach(&db);
|
|
|
|
isc_mem_detach(&mymctx);
|
|
|
|
}
|
|
|
|
|
2018-02-28 20:18:27 -08:00
|
|
|
ATF_TC(class);
|
|
|
|
ATF_TC_HEAD(class, tc) {
|
|
|
|
atf_tc_set_md_var(tc, "descr", "database class");
|
|
|
|
}
|
|
|
|
ATF_TC_BODY(class, tc) {
|
|
|
|
isc_result_t result;
|
|
|
|
dns_db_t *db = NULL;
|
|
|
|
|
|
|
|
result = dns_test_begin(NULL, ISC_FALSE);
|
|
|
|
ATF_REQUIRE_EQ(result, ISC_R_SUCCESS);
|
|
|
|
|
|
|
|
result = dns_db_create(mctx, "rbt", dns_rootname, dns_dbtype_zone,
|
|
|
|
dns_rdataclass_in, 0, NULL, &db);
|
|
|
|
ATF_REQUIRE_EQ(result, ISC_R_SUCCESS);
|
|
|
|
|
|
|
|
result = dns_db_load(db, "testdata/db/data.db");
|
|
|
|
ATF_REQUIRE_EQ(result, ISC_R_SUCCESS);
|
|
|
|
|
|
|
|
ATF_CHECK_EQ(dns_db_class(db), dns_rdataclass_in);
|
|
|
|
|
|
|
|
dns_db_detach(&db);
|
|
|
|
}
|
|
|
|
|
|
|
|
ATF_TC(dbtype);
|
|
|
|
ATF_TC_HEAD(dbtype, tc) {
|
|
|
|
atf_tc_set_md_var(tc, "descr", "database type");
|
|
|
|
}
|
|
|
|
ATF_TC_BODY(dbtype, tc) {
|
|
|
|
isc_result_t result;
|
|
|
|
dns_db_t *db = NULL;
|
|
|
|
|
|
|
|
result = dns_test_begin(NULL, ISC_FALSE);
|
|
|
|
ATF_REQUIRE_EQ(result, ISC_R_SUCCESS);
|
|
|
|
|
|
|
|
/* DB has zone semantics */
|
|
|
|
result = dns_db_create(mctx, "rbt", dns_rootname, dns_dbtype_zone,
|
|
|
|
dns_rdataclass_in, 0, NULL, &db);
|
|
|
|
ATF_REQUIRE_EQ(result, ISC_R_SUCCESS);
|
|
|
|
result = dns_db_load(db, "testdata/db/data.db");
|
|
|
|
ATF_REQUIRE_EQ(result, ISC_R_SUCCESS);
|
|
|
|
ATF_CHECK(dns_db_iszone(db));
|
|
|
|
ATF_CHECK(!dns_db_iscache(db));
|
|
|
|
dns_db_detach(&db);
|
|
|
|
|
|
|
|
/* DB has cache semantics */
|
|
|
|
result = dns_db_create(mctx, "rbt", dns_rootname, dns_dbtype_cache,
|
|
|
|
dns_rdataclass_in, 0, NULL, &db);
|
|
|
|
ATF_REQUIRE_EQ(result, ISC_R_SUCCESS);
|
|
|
|
result = dns_db_load(db, "testdata/db/data.db");
|
|
|
|
ATF_REQUIRE_EQ(result, ISC_R_SUCCESS);
|
|
|
|
ATF_CHECK(dns_db_iscache(db));
|
|
|
|
ATF_CHECK(!dns_db_iszone(db));
|
|
|
|
dns_db_detach(&db);
|
|
|
|
|
|
|
|
dns_test_end();
|
|
|
|
}
|
|
|
|
|
|
|
|
ATF_TC(version);
|
|
|
|
ATF_TC_HEAD(version, tc) {
|
|
|
|
atf_tc_set_md_var(tc, "descr", "database versions");
|
|
|
|
}
|
|
|
|
ATF_TC_BODY(version, tc) {
|
|
|
|
isc_result_t result;
|
|
|
|
dns_fixedname_t fname, ffound;
|
|
|
|
dns_name_t *name, *foundname;
|
|
|
|
dns_db_t *db = NULL;
|
|
|
|
dns_dbversion_t *ver = NULL, *new = NULL;
|
|
|
|
dns_dbnode_t *node = NULL;
|
|
|
|
dns_rdataset_t rdataset;
|
|
|
|
|
|
|
|
result = dns_test_begin(NULL, ISC_FALSE);
|
|
|
|
ATF_REQUIRE_EQ(result, ISC_R_SUCCESS);
|
|
|
|
|
|
|
|
result = dns_test_loaddb(&db, dns_dbtype_zone, "test.test",
|
|
|
|
"testdata/db/data.db");
|
|
|
|
ATF_REQUIRE_EQ(result, ISC_R_SUCCESS);
|
|
|
|
|
|
|
|
/* Open current version for reading */
|
|
|
|
dns_db_currentversion(db, &ver);
|
|
|
|
dns_test_namefromstring("b.test.test", &fname);
|
|
|
|
name = dns_fixedname_name(&fname);
|
|
|
|
dns_fixedname_init(&ffound);
|
|
|
|
foundname = dns_fixedname_name(&ffound);
|
|
|
|
dns_rdataset_init(&rdataset);
|
|
|
|
result = dns_db_find(db, name , ver, dns_rdatatype_a, 0, 0, &node,
|
|
|
|
foundname, &rdataset, NULL);
|
|
|
|
ATF_REQUIRE_EQ(result, ISC_R_SUCCESS);
|
|
|
|
dns_rdataset_disassociate(&rdataset);
|
|
|
|
dns_db_detachnode(db, &node);
|
|
|
|
dns_db_closeversion(db, &ver, ISC_FALSE);
|
|
|
|
|
|
|
|
/* Open new version for writing */
|
|
|
|
dns_db_currentversion(db, &ver);
|
|
|
|
dns_test_namefromstring("b.test.test", &fname);
|
|
|
|
name = dns_fixedname_name(&fname);
|
|
|
|
dns_fixedname_init(&ffound);
|
|
|
|
foundname = dns_fixedname_name(&ffound);
|
|
|
|
dns_rdataset_init(&rdataset);
|
|
|
|
result = dns_db_find(db, name , ver, dns_rdatatype_a, 0, 0, &node,
|
|
|
|
foundname, &rdataset, NULL);
|
|
|
|
ATF_REQUIRE_EQ(result, ISC_R_SUCCESS);
|
|
|
|
|
|
|
|
result = dns_db_newversion(db, &new);
|
|
|
|
ATF_REQUIRE_EQ(result, ISC_R_SUCCESS);
|
|
|
|
|
|
|
|
/* Delete the rdataset from the new verison */
|
|
|
|
result = dns_db_deleterdataset(db, node, new, dns_rdatatype_a, 0);
|
|
|
|
ATF_REQUIRE_EQ(result, ISC_R_SUCCESS);
|
|
|
|
|
|
|
|
dns_rdataset_disassociate(&rdataset);
|
|
|
|
dns_db_detachnode(db, &node);
|
|
|
|
|
|
|
|
/* This should fail now */
|
|
|
|
result = dns_db_find(db, name, new, dns_rdatatype_a, 0, 0, &node,
|
|
|
|
foundname, &rdataset, NULL);
|
|
|
|
ATF_REQUIRE_EQ(result, DNS_R_NXDOMAIN);
|
|
|
|
|
|
|
|
dns_db_closeversion(db, &new, ISC_TRUE);
|
|
|
|
|
|
|
|
/* But this should still succeed */
|
|
|
|
result = dns_db_find(db, name, ver, dns_rdatatype_a, 0, 0, &node,
|
|
|
|
foundname, &rdataset, NULL);
|
|
|
|
ATF_REQUIRE_EQ(result, ISC_R_SUCCESS);
|
|
|
|
dns_rdataset_disassociate(&rdataset);
|
|
|
|
dns_db_detachnode(db, &node);
|
|
|
|
dns_db_closeversion(db, &ver, ISC_FALSE);
|
|
|
|
|
|
|
|
dns_db_detach(&db);
|
|
|
|
dns_test_end();
|
|
|
|
}
|
|
|
|
|
2013-12-17 09:08:59 +11:00
|
|
|
/*
|
|
|
|
* Main
|
|
|
|
*/
|
|
|
|
ATF_TP_ADD_TCS(tp) {
|
|
|
|
ATF_TP_ADD_TC(tp, getoriginnode);
|
2017-09-06 09:58:29 +10:00
|
|
|
ATF_TP_ADD_TC(tp, getsetservestalettl);
|
|
|
|
ATF_TP_ADD_TC(tp, dns_dbfind_staleok);
|
2018-02-28 20:18:27 -08:00
|
|
|
ATF_TP_ADD_TC(tp, class);
|
|
|
|
ATF_TP_ADD_TC(tp, dbtype);
|
|
|
|
ATF_TP_ADD_TC(tp, version);
|
|
|
|
|
2013-12-17 09:08:59 +11:00
|
|
|
return (atf_no_error());
|
|
|
|
}
|