Blame t/05-SIG.t

Packit e6c8bb
# $Id: 05-SIG.t 1528 2017-01-18 21:44:58Z willem $	-*-perl-*-
Packit e6c8bb
#
Packit e6c8bb
Packit e6c8bb
use strict;
Packit e6c8bb
use Test::More;
Packit e6c8bb
use Net::DNS;
Packit e6c8bb
Packit e6c8bb
my @prerequisite = qw(
Packit e6c8bb
		MIME::Base64
Packit e6c8bb
		Time::Local
Packit e6c8bb
		);
Packit e6c8bb
Packit e6c8bb
foreach my $package (@prerequisite) {
Packit e6c8bb
	next if eval "require $package";
Packit e6c8bb
	plan skip_all => "$package not installed";
Packit e6c8bb
	exit;
Packit e6c8bb
}
Packit e6c8bb
Packit e6c8bb
plan tests => 75;
Packit e6c8bb
Packit e6c8bb
Packit e6c8bb
my $name = '.';
Packit e6c8bb
my $type = 'SIG';
Packit e6c8bb
my $code = 24;
Packit e6c8bb
my @attr = qw( typecovered algorithm labels orgttl sigexpiration siginception keytag signame signature );
Packit e6c8bb
my @data = (
Packit e6c8bb
	qw( TYPE0 1 0 0 20150814181655 20150814181155 2871 rsamd5.example ),
Packit e6c8bb
	join '', qw(	GOjsIo2JXz2ASClRhdbD5W+IYkq+Eo5iF9l3R+LYS/14Q
Packit e6c8bb
			fxqX2M9YHPvuLfz5ORAdnqyuKJTi3/LsrHmF/cUzwY3UM
Packit e6c8bb
			ZJDeGce77WiUJlR93VRKZ4fTs/wPP7JHxgAIhhlYFB4xs
Packit e6c8bb
			vISZr/tgvblxwJSpa4pJIahUuitfaiijFwQw= )
Packit e6c8bb
			);
Packit e6c8bb
my @also = qw( sig sigex sigin vrfyerrstr _size );
Packit e6c8bb
Packit e6c8bb
my $wire =
Packit e6c8bb
'000001000000000055CE309755CE2F6B0B37067273616D6435076578616D706C650018E8EC228D895F3D8048295185D6C3E56F88624ABE128E6217D97747E2D84BFD7841FC6A5F633D6073EFB8B7F3E4E440767AB2B8A2538B7FCBB2B1E617F714CF063750C6490DE19C7BBED689426547DDD544A6787D3B3FC0F3FB247C60008861958141E31B2F21266BFED82F6E5C70252A5AE292486A152E8AD7DA8A28C5C10C';
Packit e6c8bb
Packit e6c8bb
Packit e6c8bb
{
Packit e6c8bb
	my $typecode = unpack 'xn', new Net::DNS::RR(". $type")->encode;
Packit e6c8bb
	is( $typecode, $code, "$type RR type code = $code" );
Packit e6c8bb
Packit e6c8bb
	my $hash = {};
Packit e6c8bb
	@{$hash}{@attr} = @data;
Packit e6c8bb
Packit e6c8bb
	my $rr = new Net::DNS::RR(
Packit e6c8bb
		name => $name,
Packit e6c8bb
		type => $type,
Packit e6c8bb
		%$hash
Packit e6c8bb
		);
Packit e6c8bb
Packit e6c8bb
	my $string = $rr->string;
Packit e6c8bb
	my $rr2	   = new Net::DNS::RR($string);
Packit e6c8bb
	is( $rr2->string, $string, 'new/string transparent' );
Packit e6c8bb
Packit e6c8bb
	is( $rr2->encode, $rr->encode, 'new($string) and new(%hash) equivalent' );
Packit e6c8bb
Packit e6c8bb
	foreach (@attr) {
Packit e6c8bb
		is( $rr->$_, $hash->{$_}, "expected result from rr->$_()" );
Packit e6c8bb
	}
Packit e6c8bb
Packit e6c8bb
	foreach (@also) {
Packit e6c8bb
		is( $rr2->$_, $rr->$_, "additional attribute rr->$_()" );
Packit e6c8bb
	}
Packit e6c8bb
Packit e6c8bb
Packit e6c8bb
	my $empty   = new Net::DNS::RR("$name $type");
Packit e6c8bb
	my $encoded = $rr->encode;
Packit e6c8bb
	my $decoded = decode Net::DNS::RR( \$encoded );
Packit e6c8bb
	my $hex1    = uc unpack 'H*', $decoded->encode;
Packit e6c8bb
	my $hex2    = uc unpack 'H*', $encoded;
Packit e6c8bb
	my $hex3    = uc unpack 'H*', substr( $encoded, length $empty->encode );
Packit e6c8bb
	is( $hex1, $hex2, 'encode/decode transparent' );
Packit e6c8bb
	is( $hex3, $wire, 'encoded RDATA matches example' );
Packit e6c8bb
Packit e6c8bb
	my @wire = unpack 'C*', $encoded;
Packit e6c8bb
	my $wireformat = pack 'C*', @wire, 0;
Packit e6c8bb
	eval { decode Net::DNS::RR( \$wireformat ); };
Packit e6c8bb
	my $exception = $1 if $@ =~ /^(.+)\n/;
Packit e6c8bb
	ok( $exception ||= '', "misplaced SIG RR\t[$exception]" );
Packit e6c8bb
}
Packit e6c8bb
Packit e6c8bb
Packit e6c8bb
{
Packit e6c8bb
	my @rdata	= @data;
Packit e6c8bb
	my $sig		= pop @rdata;
Packit e6c8bb
	my $lc		= new Net::DNS::RR( lc(". $type @rdata ") . $sig );
Packit e6c8bb
	my $rr		= new Net::DNS::RR( uc(". $type @rdata ") . $sig );
Packit e6c8bb
	my $hash	= {};
Packit e6c8bb
	my $predecessor = $rr->encode( 0, $hash );
Packit e6c8bb
	my $compressed	= $rr->encode( length $predecessor, $hash );
Packit e6c8bb
	ok( length $compressed == length $predecessor, 'encoded RDATA not compressible' );
Packit e6c8bb
	is( $rr->encode,    $lc->encode, 'encoded RDATA names downcased' );
Packit e6c8bb
	is( $rr->canonical, $lc->encode, 'canonical RDATA names downcased' );
Packit e6c8bb
}
Packit e6c8bb
Packit e6c8bb
Packit e6c8bb
{
Packit e6c8bb
	my $rr = new Net::DNS::RR(". $type");
Packit e6c8bb
	foreach ( @attr, 'rdstring' ) {
Packit e6c8bb
		ok( !$rr->$_(), "'$_' attribute of empty RR undefined" );
Packit e6c8bb
	}
Packit e6c8bb
}
Packit e6c8bb
Packit e6c8bb
Packit e6c8bb
{
Packit e6c8bb
	my $rr	  = new Net::DNS::RR(". $type @data");
Packit e6c8bb
	my $class = ref($rr);
Packit e6c8bb
Packit e6c8bb
	$rr->algorithm(255);
Packit e6c8bb
	is( $rr->algorithm(), 255, 'algorithm number accepted' );
Packit e6c8bb
	$rr->algorithm('RSASHA1');
Packit e6c8bb
	is( $rr->algorithm(),		5,	   'algorithm mnemonic accepted' );
Packit e6c8bb
	is( $rr->algorithm('MNEMONIC'), 'RSASHA1', 'rr->algorithm("MNEMONIC") returns mnemonic' );
Packit e6c8bb
	is( $rr->algorithm(),		5,	   'rr->algorithm("MNEMONIC") preserves value' );
Packit e6c8bb
Packit e6c8bb
	eval { $rr->algorithm('X'); };
Packit e6c8bb
	my $exception = $1 if $@ =~ /^(.+)\n/;
Packit e6c8bb
	ok( $exception ||= '', "unknown mnemonic\t[$exception]" );
Packit e6c8bb
Packit e6c8bb
	is( $class->algorithm('RSASHA256'), 8,		 'class method algorithm("RSASHA256")' );
Packit e6c8bb
	is( $class->algorithm(8),	    'RSASHA256', 'class method algorithm(8)' );
Packit e6c8bb
	is( $class->algorithm(255),	    255,	 'class method algorithm(255)' );
Packit e6c8bb
}
Packit e6c8bb
Packit e6c8bb
Packit e6c8bb
{
Packit e6c8bb
	my $object   = new Net::DNS::RR(". $type");
Packit e6c8bb
	my $class    = ref($object);
Packit e6c8bb
	my $scalar   = '';
Packit e6c8bb
	my %testcase = (		## test callable with invalid arguments
Packit e6c8bb
		'_CreateSig'	 => [$object, $scalar, $object],
Packit e6c8bb
		'_CreateSigData' => [$object, $object],
Packit e6c8bb
		'_string2time'	 => [undef],
Packit e6c8bb
		'_time2string'	 => [undef],
Packit e6c8bb
		'_VerifySig'	 => [$object, $object, $object],
Packit e6c8bb
		'create'	 => [$class,  $scalar, $object],
Packit e6c8bb
		'verify'	 => [$object, $object, $object],
Packit e6c8bb
		);
Packit e6c8bb
Packit e6c8bb
	foreach my $method ( sort keys %testcase ) {
Packit e6c8bb
		my $arglist = $testcase{$method};
Packit e6c8bb
		$object->{algorithm} = 0;			# induce exception
Packit e6c8bb
		no strict q/refs/;
Packit e6c8bb
		my $subroutine = join '::', $class, $method;
Packit e6c8bb
		eval { &$subroutine(@$arglist); };
Packit e6c8bb
		my $exception = $1 if $@ =~ /^(.*)\n*/;
Packit e6c8bb
		ok( defined $exception, "$method method callable\t[$exception]" );
Packit e6c8bb
	}
Packit e6c8bb
}
Packit e6c8bb
Packit e6c8bb
Packit e6c8bb
{
Packit e6c8bb
	my %testcase = (		## test time conversion edge cases
Packit e6c8bb
		-1	   => '21060207062815',
Packit e6c8bb
		0x00000000 => '19700101000000',
Packit e6c8bb
		0x7fffffff => '20380119031407',
Packit e6c8bb
		0x80000000 => '20380119031408',
Packit e6c8bb
		0xf4d41f7f => '21000228235959',
Packit e6c8bb
		0xf4d41f80 => '21000301000000',
Packit e6c8bb
		0xffffffff => '21060207062815',
Packit e6c8bb
		);
Packit e6c8bb
Packit e6c8bb
	foreach my $time ( sort keys %testcase ) {
Packit e6c8bb
		my $string = $testcase{$time};
Packit e6c8bb
		my $result = Net::DNS::RR::SIG::_time2string($time);
Packit e6c8bb
		is( $result, $string, "_time2string($time)" );
Packit e6c8bb
Packit e6c8bb
		# Test indirectly: $timeval can be 64-bit or negative 32-bit integer
Packit e6c8bb
		my $timeval = Net::DNS::RR::SIG::_string2time($string);
Packit e6c8bb
		my $timestr = Net::DNS::RR::SIG::_time2string($timeval);
Packit e6c8bb
		is( $timestr, $string, "_string2time($string)" );
Packit e6c8bb
	}
Packit e6c8bb
Packit e6c8bb
	my $timenow = time();
Packit e6c8bb
	my $timeval = Net::DNS::RR::SIG::_string2time($timenow);
Packit e6c8bb
	is( $timeval, $timenow, "_string2time( time() )\t$timeval" );
Packit e6c8bb
}
Packit e6c8bb
Packit e6c8bb
Packit e6c8bb
{
Packit e6c8bb
	ok( Net::DNS::RR::SIG::_ordered( undef,	     0 ),	   '_ordered( undef, 0 )' );
Packit e6c8bb
	ok( Net::DNS::RR::SIG::_ordered( 0,	     1 ),	   '_ordered( 0, 1 )' );
Packit e6c8bb
	ok( Net::DNS::RR::SIG::_ordered( 0x7fffffff, 0x80000000 ), '_ordered( 0x7fffffff, 0x80000000 )' );
Packit e6c8bb
	ok( Net::DNS::RR::SIG::_ordered( 0xffffffff, 0 ),	   '_ordered( 0xffffffff, 0 )' );
Packit e6c8bb
	ok( Net::DNS::RR::SIG::_ordered( -2,	     -1 ),	   '_ordered( -2, -1 )' );
Packit e6c8bb
	ok( Net::DNS::RR::SIG::_ordered( -1,	     0 ),	   '_ordered( -1, 0 )' );
Packit e6c8bb
	ok( !Net::DNS::RR::SIG::_ordered( undef,      undef ),	    '!_ordered( undef, undef )' );
Packit e6c8bb
	ok( !Net::DNS::RR::SIG::_ordered( 0,	      undef ),	    '!_ordered( 0, undef )' );
Packit e6c8bb
	ok( !Net::DNS::RR::SIG::_ordered( 0x80000000, 0x7fffffff ), '!_ordered( 0x80000000, 0x7fffffff )' );
Packit e6c8bb
	ok( !Net::DNS::RR::SIG::_ordered( 0,	      0xffffffff ), '!_ordered( 0, 0xffffffff )' );
Packit e6c8bb
	ok( !Net::DNS::RR::SIG::_ordered( -1,	      -2 ),	    '!_ordered( -1, -2 )' );
Packit e6c8bb
	ok( !Net::DNS::RR::SIG::_ordered( 0,	      -1 ),	    '!_ordered( 0, -1 )' );
Packit e6c8bb
}
Packit e6c8bb
Packit e6c8bb
Packit e6c8bb
{
Packit e6c8bb
	my $rr = new Net::DNS::RR("$name $type @data");
Packit e6c8bb
	$rr->print;
Packit e6c8bb
}
Packit e6c8bb
Packit e6c8bb
exit;
Packit e6c8bb
Packit e6c8bb