Files
zonemaster.es/zonemaster-engine/t/recursor-A.t
Malin 8d4eaa1489 feat: add full Zonemaster stack with Docker and Spanish UI
- Clone all 5 Zonemaster component repos (LDNS, Engine, CLI, Backend, GUI)
- Dockerfile.backend: 8-stage multi-stage build LDNS→Engine→CLI→Backend
- Dockerfile.gui: Astro static build served via nginx
- docker-compose.yml: backend (internal) + frontend (port 5353)
- nginx.conf: root redirects to /es/, /api/ proxied to backend
- zonemaster-gui/config.ts: defaultLanguage set to 'es' (Spanish)

Co-Authored-By: Claude Sonnet 4.6 <noreply@anthropic.com>
2026-04-21 08:19:24 +02:00

263 lines
12 KiB
Perl

use Test::More;
use Test::Differences;
use File::Basename;
use 5.14.2;
use strict;
use warnings;
BEGIN {
use_ok( q{Zonemaster::Engine} );
use_ok( q{Zonemaster::Engine::Recursor} );
use_ok( q{Zonemaster::Engine::Util}, qw( name ) );
}
my $datafile = 't/' . basename ($0, '.t') . '.data';
if ( not $ENV{ZONEMASTER_RECORD} ) {
die "Stored data file missing" if not -r $datafile;
Zonemaster::Engine::Nameserver->restore( $datafile );
Zonemaster::Engine::Profile->effective->set( q{no_network}, 1 );
}
# Common hint file (test-zone-data/COMMON/hintfile)
Zonemaster::Engine::Recursor->remove_fake_addresses( '.' );
Zonemaster::Engine::Recursor->add_fake_addresses( '.',
{ 'ns1' => [ '127.1.0.1', 'fda1:b2:c3::127:1:0:1' ],
'ns2' => [ '127.1.0.2', 'fda1:b2:c3::127:1:0:2' ],
}
);
# Test zone scenarios
subtest 'GOOD-CNAME-1' => sub {
Zonemaster::Engine->logger->clear_history;
my $p = Zonemaster::Engine->recurse( 'good-cname-1.cname.recursor.engine.xa' );
isa_ok( $p, 'Zonemaster::Engine::Packet' );
is( scalar( $p->answer ), 2, 'two records in answer section' );
isa_ok( ($p->answer)[0], 'Zonemaster::LDNS::RR::CNAME' );
is( name( ($p->answer)[0]->owner ), 'good-cname-1.cname.recursor.engine.xa', 'RR name ok' );
is( name( ($p->answer)[0]->cname ), 'good-cname-1-target.cname.recursor.engine.xa', 'RR cname ok' );
isa_ok( ($p->answer)[1], 'Zonemaster::LDNS::RR::A' );
is( name( ($p->answer)[1]->owner ), 'good-cname-1-target.cname.recursor.engine.xa', 'RR name ok' );
is( ($p->answer)[1]->address, '127.0.0.1', 'RR address ok' );
my %res = map { $_->tag => $_ } @{ Zonemaster::Engine->logger->entries };
ok( $res{CNAME_START}, q{should emit CNAME_START} );
ok( $res{CNAME_FOLLOWED_IN_ZONE}, q{should emit CNAME_FOLLOWED_IN_ZONE} );
};
subtest 'GOOD-CNAME-2' => sub {
Zonemaster::Engine->logger->clear_history;
my $p = Zonemaster::Engine->recurse( 'good-cname-2.cname.recursor.engine.xa' );
isa_ok( $p, 'Zonemaster::Engine::Packet' );
is( scalar( $p->answer ), 3, 'three records in answer section' );
isa_ok( ($p->answer)[0], 'Zonemaster::LDNS::RR::CNAME' );
is( name( ($p->answer)[0]->owner ), 'good-cname-2.cname.recursor.engine.xa', 'RR name ok' );
is( name( ($p->answer)[0]->cname ), 'good-cname-2-target.cname.recursor.engine.xa', 'RR cname ok' );
isa_ok( ($p->answer)[1], 'Zonemaster::LDNS::RR::A' );
is( name( ($p->answer)[1]->owner ), 'good-cname-2-target.cname.recursor.engine.xa', 'RR name ok' );
is( ($p->answer)[1]->address, '127.0.0.1', 'RR address ok' );
isa_ok( ($p->answer)[2], 'Zonemaster::LDNS::RR::A' );
is( name( ($p->answer)[2]->owner ), 'good-cname-2-target.cname.recursor.engine.xa', 'RR name ok' );
is( ($p->answer)[2]->address, '127.0.0.2', 'RR address ok' );
my %res = map { $_->tag => $_ } @{ Zonemaster::Engine->logger->entries };
ok( $res{CNAME_START}, q{should emit CNAME_START} );
ok( $res{CNAME_FOLLOWED_IN_ZONE}, q{should emit CNAME_FOLLOWED_IN_ZONE} );
};
subtest 'GOOD-CNAME-CHAIN' => sub {
Zonemaster::Engine->logger->clear_history;
my $p = Zonemaster::Engine->recurse( 'good-cname-chain.cname.recursor.engine.xa' );
isa_ok( $p, 'Zonemaster::Engine::Packet' );
is( scalar( $p->answer ), 4, 'four records in answer section' );
isa_ok( ($p->answer)[0], 'Zonemaster::LDNS::RR::CNAME' );
is( name( ($p->answer)[0]->owner ), 'good-cname-chain.cname.recursor.engine.xa', 'RR name ok' );
is( name( ($p->answer)[0]->cname ), 'good-cname-chain-two.cname.recursor.engine.xa', 'RR cname ok' );
isa_ok( ($p->answer)[1], 'Zonemaster::LDNS::RR::CNAME' );
is( name( ($p->answer)[1]->owner ), 'good-cname-chain-two.cname.recursor.engine.xa', 'RR name ok' );
is( name( ($p->answer)[1]->cname ), 'good-cname-chain-three.cname.recursor.engine.xa', 'RR cname ok' );
isa_ok( ($p->answer)[2], 'Zonemaster::LDNS::RR::CNAME' );
is( name( ($p->answer)[2]->owner ), 'good-cname-chain-three.cname.recursor.engine.xa', 'RR name ok' );
is( name( ($p->answer)[2]->cname ), 'good-cname-chain-target.cname.recursor.engine.xa', 'RR cname ok' );
isa_ok( ($p->answer)[3], 'Zonemaster::LDNS::RR::A' );
is( name( ($p->answer)[3]->owner ), 'good-cname-chain-target.cname.recursor.engine.xa', 'RR name ok' );
is( ($p->answer)[3]->address, '127.0.0.1', 'RR address ok' );
my %res = map { $_->tag => $_ } @{ Zonemaster::Engine->logger->entries };
ok( $res{CNAME_START}, q{should emit CNAME_START} );
ok( $res{CNAME_FOLLOWED_IN_ZONE}, q{should emit CNAME_FOLLOWED_IN_ZONE} );
};
subtest 'GOOD-CNAME-OUT-OF-ZONE' => sub {
Zonemaster::Engine->logger->clear_history;
my $p = Zonemaster::Engine->recurse( 'good-cname-out-of-zone.cname.recursor.engine.xa' );
isa_ok( $p, 'Zonemaster::Engine::Packet' );
is( scalar( $p->answer ), 1, 'one record in answer section' );
isa_ok( ($p->answer)[0], 'Zonemaster::LDNS::RR::A' );
is( name( ($p->answer)[0]->owner ), 'target.goodsub.cname.recursor.engine.xa', 'RR name ok' );
is( ($p->answer)[0]->address, '127.0.0.1', 'RR address ok' );
my %res = map { $_->tag => $_ } @{ Zonemaster::Engine->logger->entries };
ok( $res{CNAME_START}, q{should emit CNAME_START} );
ok( $res{CNAME_FOLLOWED_OUT_OF_ZONE}, q{should emit CNAME_FOLLOWED_OUT_OF_ZONE} );
};
subtest 'NXDOMAIN-VIA-CNAME' => sub {
Zonemaster::Engine->logger->clear_history;
my $p = Zonemaster::Engine->recurse( 'nxdomain-via-cname.cname.recursor.engine.xa' );
isa_ok( $p, 'Zonemaster::Engine::Packet' );
is( $p->rcode, 'NXDOMAIN', 'NXDOMAIN in response' );
is( scalar( $p->answer ), 0, 'no records in answer section' );
is( scalar( $p->authority ), 1, 'one record in authority section' );
isa_ok( ($p->authority)[0], 'Zonemaster::LDNS::RR::SOA' );
is( name( ($p->authority)[0]->owner ), 'cname.recursor.engine.xa', 'RR name ok' );
my %res = map { $_->tag => $_ } @{ Zonemaster::Engine->logger->entries };
ok( $res{CNAME_START}, q{should emit CNAME_START} );
ok( $res{CNAME_FOLLOWED_OUT_OF_ZONE}, q{should emit CNAME_FOLLOWED_OUT_OF_ZONE} );
};
subtest 'NODATA-VIA-CNAME' => sub {
Zonemaster::Engine->logger->clear_history;
my $p = Zonemaster::Engine->recurse( 'nodata-via-cname.cname.recursor.engine.xa' );
isa_ok( $p, 'Zonemaster::Engine::Packet' );
is( $p->rcode, 'NOERROR', 'NOERROR in response' );
is( scalar( $p->answer ), 0, 'no records in answer section' );
is( scalar( $p->authority ), 1, 'one record in authority section' );
isa_ok( ($p->authority)[0], 'Zonemaster::LDNS::RR::SOA' );
is( name( ($p->authority)[0]->owner ), 'cname.recursor.engine.xa', 'RR name ok' );
my %res = map { $_->tag => $_ } @{ Zonemaster::Engine->logger->entries };
ok( $res{CNAME_START}, q{should emit CNAME_START} );
ok( $res{CNAME_FOLLOWED_OUT_OF_ZONE}, q{should emit CNAME_FOLLOWED_OUT_OF_ZONE} );
};
subtest 'MULT-CNAME' => sub {
Zonemaster::Engine->logger->clear_history;
my $p = Zonemaster::Engine->recurse( 'mult-cname.cname.recursor.engine.xa' );
is( $p, undef, "undefined as expected");
my %res = map { $_->tag => $_ } @{ Zonemaster::Engine->logger->entries };
ok( $res{CNAME_START}, q{should emit CNAME_START} );
ok( $res{CNAME_RECORDS_MULTIPLE_FOR_NAME}, q{should emit CNAME_RECORDS_MULTIPLE_FOR_NAME} );
};
subtest 'LOOPED-CNAME-IN-ZONE-1' => sub {
Zonemaster::Engine->logger->clear_history;
my $p = Zonemaster::Engine->recurse( 'looped-cname-in-zone-1.cname.recursor.engine.xa' );
is( $p, undef, "undefined as expected");
my %res = map { $_->tag => $_ } @{ Zonemaster::Engine->logger->entries };
ok( $res{CNAME_START}, q{should emit CNAME_START} );
ok( $res{CNAME_LOOP_INNER}, q{should emit CNAME_LOOP_INNER} );
};
subtest 'LOOPED-CNAME-IN-ZONE-2' => sub {
Zonemaster::Engine->logger->clear_history;
my $p = Zonemaster::Engine->recurse( 'looped-cname-in-zone-2.cname.recursor.engine.xa' );
is( $p, undef, "undefined as expected");
my %res = map { $_->tag => $_ } @{ Zonemaster::Engine->logger->entries };
ok( $res{CNAME_START}, q{should emit CNAME_START} );
ok( $res{CNAME_LOOP_INNER}, q{should emit CNAME_LOOP_INNER} );
};
subtest 'LOOPED-CNAME-IN-ZONE-3' => sub {
Zonemaster::Engine->logger->clear_history;
my $p = Zonemaster::Engine->recurse( 'looped-cname-in-zone-3.cname.recursor.engine.xa' );
is( $p, undef, "undefined as expected");
my %res = map { $_->tag => $_ } @{ Zonemaster::Engine->logger->entries };
ok( $res{CNAME_START}, q{should emit CNAME_START} );
ok( $res{CNAME_LOOP_INNER}, q{should emit CNAME_LOOP_INNER} );
};
subtest 'LOOPED-CNAME-OUT-OF-ZONE' => sub {
Zonemaster::Engine->logger->clear_history;
my $p = Zonemaster::Engine->recurse( 'looped-cname-out-of-zone.sub2.cname.recursor.engine.xa' );
is( $p, undef, "undefined as expected");
my %res = map { $_->tag => $_ } @{ Zonemaster::Engine->logger->entries };
ok( $res{CNAME_START}, q{should emit CNAME_START} );
ok( $res{CNAME_LOOP_OUTER}, q{should emit CNAME_LOOP_OUTER} );
};
subtest 'TOO-LONG-CNAME-CHAIN' => sub {
Zonemaster::Engine->logger->clear_history;
my $p = Zonemaster::Engine->recurse( 'too-long-cname-chain.cname.recursor.engine.xa' );
is( $p, undef, "undefined as expected");
my %res = map { $_->tag => $_ } @{ Zonemaster::Engine->logger->entries };
ok( $res{CNAME_START}, q{should emit CNAME_START} );
ok( $res{CNAME_RECORDS_TOO_MANY}, q{should emit CNAME_RECORDS_TOO_MANY} );
};
subtest 'TARGET-NO-MATCH-CNAME' => sub {
Zonemaster::Engine->logger->clear_history;
my $p = Zonemaster::Engine->recurse( 'target-no-match-cname.cname.recursor.engine.xa' );
is( $p, undef, "undefined as expected");
my %res = map { $_->tag => $_ } @{ Zonemaster::Engine->logger->entries };
ok( $res{CNAME_START}, q{should emit CNAME_START} );
ok( $res{CNAME_NO_MATCH}, q{should emit CNAME_NO_MATCH} );
};
subtest 'BROKEN-CNAME-CHAIN' => sub {
Zonemaster::Engine->logger->clear_history;
my $p = Zonemaster::Engine->recurse( 'broken-cname-chain.cname.recursor.engine.xa' );
is( $p, undef, "undefined as expected");
my %res = map { $_->tag => $_ } @{ Zonemaster::Engine->logger->entries };
ok( $res{CNAME_START}, q{should emit CNAME_START} );
ok( $res{CNAME_RECORDS_CHAIN_BROKEN}, q{should emit CNAME_RECORDS_CHAIN_BROKEN} );
};
subtest 'WRONG-CNAME-OWNER-NAME' => sub {
Zonemaster::Engine->logger->clear_history;
my $p = Zonemaster::Engine->recurse( 'wrong-cname-owner-name.cname.recursor.engine.xa' );
isa_ok( $p, 'Zonemaster::Engine::Packet' );
is( scalar( $p->answer ), 2, 'two records in answer section' );
ok( name( ($p->question)[0]->owner )->string ne name( ($p->answer)[0]->owner )->string, 'expected different owner name as QNAME on first RR' );
ok( name( ($p->question)[0]->owner )->string ne name( ($p->answer)[1]->owner )->string, 'expected different owner name as QNAME on second RR' );
my %res = map { $_->tag => $_ } @{ Zonemaster::Engine->logger->entries };
ok( !$res{CNAME_START}, q{should not emit CNAME_START} );
ok( scalar ( grep { index($_->tag, 'CNAME') != -1 } @{ Zonemaster::Engine->logger->entries } ) == 0, 'empty CNAME message tags' );
};
subtest 'EXTRA-CNAME-IN-ANSWER' => sub {
Zonemaster::Engine->logger->clear_history;
my $p = Zonemaster::Engine->recurse( 'extra-cname-in-answer.cname.recursor.engine.xa' );
isa_ok( $p, 'Zonemaster::Engine::Packet' );
is( scalar( $p->answer ), 2, 'two records in answer section' );
isa_ok( ($p->answer)[0], 'Zonemaster::LDNS::RR::A' );
ok( name( ($p->question)[0]->owner )->string eq name( ($p->answer)[0]->owner )->string, 'expected same owner name as QNAME on first RR' );
isa_ok( ($p->answer)[1], 'Zonemaster::LDNS::RR::CNAME' );
ok( name( ($p->question)[0]->owner )->string ne name( ($p->answer)[1]->owner )->string, 'expected different owner name as QNAME on second RR' );
my %res = map { $_->tag => $_ } @{ Zonemaster::Engine->logger->entries };
ok( !$res{CNAME_START}, q{should not emit CNAME_START} );
ok( scalar ( grep { index($_->tag, 'CNAME') != -1 } @{ Zonemaster::Engine->logger->entries } ) == 0, 'empty CNAME message tags' );
};
if ( $ENV{ZONEMASTER_RECORD} ) {
Zonemaster::Engine::Nameserver->save( $datafile );
}
done_testing;