mirror of
https://github.com/NLnetLabs/unbound.git
synced 2025-12-20 23:00:56 -05:00
(configurable) infra-cache-max-rtt value. That could result in infra-keep-probing not working if an infra-cache-max-rtt value was chosen that was below 12000 ms. This fix still uses a default value of 12000 ms for the probe but caps it to the infra-cache-max-rtt if that is lower.
209 lines
7.8 KiB
C
209 lines
7.8 KiB
C
/*
|
|
* testcode/unitinfra.c - unit test for infra cache.
|
|
*
|
|
* Copyright (c) 2025, NLnet Labs. All rights reserved.
|
|
*
|
|
* This software is open source.
|
|
*
|
|
* Redistribution and use in source and binary forms, with or without
|
|
* modification, are permitted provided that the following conditions
|
|
* are met:
|
|
*
|
|
* Redistributions of source code must retain the above copyright notice,
|
|
* this list of conditions and the following disclaimer.
|
|
*
|
|
* Redistributions in binary form must reproduce the above copyright notice,
|
|
* this list of conditions and the following disclaimer in the documentation
|
|
* and/or other materials provided with the distribution.
|
|
*
|
|
* Neither the name of the NLNET LABS nor the names of its contributors may
|
|
* be used to endorse or promote products derived from this software without
|
|
* specific prior written permission.
|
|
*
|
|
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
|
|
* "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
|
|
* LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
|
|
* A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
|
|
* HOLDER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
|
|
* SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED
|
|
* TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR
|
|
* PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF
|
|
* LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING
|
|
* NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
|
|
* SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
|
*
|
|
*/
|
|
/**
|
|
* \file
|
|
* Tests the infra functionality.
|
|
*/
|
|
|
|
#include "config.h"
|
|
#include "testcode/unitmain.h"
|
|
#include "iterator/iterator.h"
|
|
#include "services/cache/infra.h"
|
|
#include "util/config_file.h"
|
|
#include "util/net_help.h"
|
|
|
|
/* lookup and get key and data structs easily */
|
|
static struct infra_data* infra_lookup_host(struct infra_cache* infra,
|
|
struct sockaddr_storage* addr, socklen_t addrlen, uint8_t* zone,
|
|
size_t zonelen, int wr, time_t now, struct infra_key** k)
|
|
{
|
|
struct infra_data* d;
|
|
struct lruhash_entry* e = infra_lookup_nottl(infra, addr, addrlen,
|
|
zone, zonelen, wr);
|
|
if(!e) return NULL;
|
|
d = (struct infra_data*)e->data;
|
|
if(d->ttl < now) {
|
|
lock_rw_unlock(&e->lock);
|
|
return NULL;
|
|
}
|
|
*k = (struct infra_key*)e->key;
|
|
return d;
|
|
}
|
|
|
|
static void test_keep_probing(struct infra_cache* slab,
|
|
struct config_file* cfg, struct sockaddr_storage one, socklen_t onelen,
|
|
uint8_t* zone, size_t zonelen, time_t *now, int keep_probing,
|
|
int rtt_max_timeout)
|
|
{
|
|
uint8_t edns_lame;
|
|
int vs, to, lame, dnsseclame, reclame, probedelay;
|
|
struct infra_key* k;
|
|
struct infra_data* d;
|
|
|
|
/* configure */
|
|
cfg->infra_cache_max_rtt = rtt_max_timeout;
|
|
config_apply_max_rtt(rtt_max_timeout);
|
|
slab->infra_keep_probing = keep_probing;
|
|
|
|
/* expired previous entry */
|
|
*now += cfg->host_ttl + 10;
|
|
unit_assert( infra_host(slab, &one, onelen, zone, zonelen,
|
|
*now, &vs, &edns_lame, &to) );
|
|
|
|
/* simulate timeouts until the USEFUL_SERVER_TOP_TIMEOUT is reached */
|
|
while(to < USEFUL_SERVER_TOP_TIMEOUT) {
|
|
unit_assert( infra_rtt_update(slab, &one, onelen, zone, zonelen,
|
|
LDNS_RR_TYPE_A, -1, to, *now) );
|
|
unit_assert( infra_host(slab, &one, onelen, zone, zonelen,
|
|
*now, &vs, &edns_lame, &to) );
|
|
unit_assert( vs == 0 && to <= USEFUL_SERVER_TOP_TIMEOUT && edns_lame == 0 );
|
|
}
|
|
unit_assert( vs == 0 && to == USEFUL_SERVER_TOP_TIMEOUT && edns_lame == 0 );
|
|
|
|
/* don't let the record expire */
|
|
unit_assert( (d=infra_lookup_host(slab, &one, onelen, zone, zonelen, 0, *now, &k)) );
|
|
unit_assert( d->timeout_A >= TIMEOUT_COUNT_MAX );
|
|
unit_assert( d->probedelay > 0 );
|
|
probedelay = d->probedelay;
|
|
lock_rw_unlock(&k->entry.lock);
|
|
cfg->host_ttl = cfg->host_ttl + *now < probedelay
|
|
?cfg->host_ttl :probedelay + 10;
|
|
|
|
/* advance time and check that probing is as expected; we already had a
|
|
* lot of A timeouts (checked above). */
|
|
*now = probedelay;
|
|
unit_assert( infra_get_lame_rtt(slab, &one, onelen, zone, zonelen,
|
|
LDNS_RR_TYPE_A, &lame, &dnsseclame, &reclame, &to, *now) );
|
|
unit_assert( lame == 0 && dnsseclame == 0 && reclame == 0
|
|
&& to == keep_probing ?still_useful_timeout() :USEFUL_SERVER_TOP_TIMEOUT);
|
|
}
|
|
|
|
/** test host cache */
|
|
void infra_test(void)
|
|
{
|
|
struct sockaddr_storage one;
|
|
socklen_t onelen;
|
|
uint8_t* zone = (uint8_t*)"\007example\003com\000";
|
|
size_t zonelen = 13;
|
|
struct infra_cache* slab;
|
|
struct config_file* cfg = config_create();
|
|
time_t now = 0;
|
|
uint8_t edns_lame;
|
|
int vs, to;
|
|
struct infra_key* k;
|
|
struct infra_data* d;
|
|
int init = UNKNOWN_SERVER_NICENESS;
|
|
int default_max_rtt = USEFUL_SERVER_TOP_TIMEOUT;
|
|
|
|
unit_show_feature("infra cache");
|
|
unit_assert(ipstrtoaddr("127.0.0.1", 53, &one, &onelen));
|
|
|
|
slab = infra_create(cfg);
|
|
/* insert new record */
|
|
unit_assert( infra_host(slab, &one, onelen, zone, zonelen, now,
|
|
&vs, &edns_lame, &to) );
|
|
unit_assert( vs == 0 && to == init && edns_lame == 0 );
|
|
|
|
/* simulate no answer */
|
|
unit_assert( infra_rtt_update(slab, &one, onelen, zone, zonelen, LDNS_RR_TYPE_A, -1, init, now) );
|
|
unit_assert( infra_host(slab, &one, onelen, zone, zonelen,
|
|
now, &vs, &edns_lame, &to) );
|
|
unit_assert( vs == 0 && to == init*2 && edns_lame == 0 );
|
|
|
|
/* simulate EDNS lame */
|
|
unit_assert( infra_edns_update(slab, &one, onelen, zone, zonelen, -1, now) );
|
|
unit_assert( infra_host(slab, &one, onelen, zone, zonelen,
|
|
now, &vs, &edns_lame, &to) );
|
|
unit_assert( vs == -1 && to == init*2 && edns_lame == 1);
|
|
|
|
/* simulate cache expiry */
|
|
now += cfg->host_ttl + 10;
|
|
unit_assert( infra_host(slab, &one, onelen, zone, zonelen,
|
|
now, &vs, &edns_lame, &to) );
|
|
unit_assert( vs == 0 && to == init && edns_lame == 0 );
|
|
|
|
/* simulate no lame answer */
|
|
unit_assert( infra_set_lame(slab, &one, onelen,
|
|
zone, zonelen, now, 0, 0, LDNS_RR_TYPE_A) );
|
|
unit_assert( (d=infra_lookup_host(slab, &one, onelen, zone, zonelen, 0, now, &k)) );
|
|
unit_assert( d->ttl == now+cfg->host_ttl );
|
|
unit_assert( d->edns_version == 0 );
|
|
unit_assert(!d->isdnsseclame && !d->rec_lame && d->lame_type_A &&
|
|
!d->lame_other);
|
|
lock_rw_unlock(&k->entry.lock);
|
|
|
|
/* test merge of data */
|
|
unit_assert( infra_set_lame(slab, &one, onelen,
|
|
zone, zonelen, now, 0, 0, LDNS_RR_TYPE_AAAA) );
|
|
unit_assert( (d=infra_lookup_host(slab, &one, onelen, zone, zonelen, 0, now, &k)) );
|
|
unit_assert(!d->isdnsseclame && !d->rec_lame && d->lame_type_A &&
|
|
d->lame_other);
|
|
lock_rw_unlock(&k->entry.lock);
|
|
|
|
/* test that noEDNS cannot overwrite known-yesEDNS */
|
|
now += cfg->host_ttl + 10;
|
|
unit_assert( infra_host(slab, &one, onelen, zone, zonelen,
|
|
now, &vs, &edns_lame, &to) );
|
|
unit_assert( vs == 0 && to == init && edns_lame == 0 );
|
|
|
|
unit_assert( infra_edns_update(slab, &one, onelen, zone, zonelen, 0, now) );
|
|
unit_assert( infra_host(slab, &one, onelen, zone, zonelen,
|
|
now, &vs, &edns_lame, &to) );
|
|
unit_assert( vs == 0 && to == init && edns_lame == 1 );
|
|
|
|
unit_assert( infra_edns_update(slab, &one, onelen, zone, zonelen, -1, now) );
|
|
unit_assert( infra_host(slab, &one, onelen, zone, zonelen,
|
|
now, &vs, &edns_lame, &to) );
|
|
unit_assert( vs == 0 && to == init && edns_lame == 1 );
|
|
|
|
unit_show_feature("infra cache probing (keep-probing off, default infra-cache-max-rtt)");
|
|
test_keep_probing(slab, cfg, one, onelen, zone, zonelen, &now, 0, default_max_rtt);
|
|
|
|
unit_show_feature("infra cache probing (keep-probing on, default infra-cache-max-rtt)");
|
|
test_keep_probing(slab, cfg, one, onelen, zone, zonelen, &now, 1, default_max_rtt);
|
|
|
|
unit_show_feature("infra cache probing (keep-probing off, low infra-cache-max-rtt)");
|
|
test_keep_probing(slab, cfg, one, onelen, zone, zonelen, &now, 0, 3000);
|
|
|
|
unit_show_feature("infra cache probing (keep-probing on, low infra-cache-max-rtt)");
|
|
test_keep_probing(slab, cfg, one, onelen, zone, zonelen, &now, 1, 3000);
|
|
|
|
/* Re-apply defaults for other unit tests that follow */
|
|
config_apply_max_rtt(default_max_rtt);
|
|
|
|
infra_delete(slab);
|
|
config_delete(cfg);
|
|
}
|