line |
stmt |
bran |
cond |
sub |
pod |
time |
code |
1
|
|
|
|
|
|
|
# vim:ts=4:sw=4:expandtab |
2
|
|
|
|
|
|
|
package App::AllKnowingDNS::Handler; |
3
|
|
|
|
|
|
|
|
4
|
2
|
|
|
2
|
|
6450
|
use strict; |
|
2
|
|
|
|
|
6
|
|
|
2
|
|
|
|
|
97
|
|
5
|
2
|
|
|
2
|
|
15
|
use warnings; |
|
2
|
|
|
|
|
52
|
|
|
2
|
|
|
|
|
115
|
|
6
|
2
|
|
|
2
|
|
13
|
use base 'Exporter'; |
|
2
|
|
|
|
|
4
|
|
|
2
|
|
|
|
|
312
|
|
7
|
2
|
|
|
2
|
|
2578
|
use Net::DNS; |
|
2
|
|
|
|
|
186051
|
|
|
2
|
|
|
|
|
281
|
|
8
|
2
|
|
|
2
|
|
22
|
use NetAddr::IP::Util qw(ipv6_aton); |
|
2
|
|
|
|
|
4
|
|
|
2
|
|
|
|
|
24
|
|
9
|
2
|
|
|
2
|
|
274
|
use App::AllKnowingDNS::Config; |
|
2
|
|
|
|
|
7
|
|
|
2
|
|
|
|
|
47
|
|
10
|
2
|
|
|
2
|
|
13
|
use App::AllKnowingDNS::Zone; |
|
2
|
|
|
|
|
4
|
|
|
2
|
|
|
|
|
48
|
|
11
|
2
|
|
|
2
|
|
1906
|
use POSIX qw(strftime); |
|
2
|
|
|
|
|
14368
|
|
|
2
|
|
|
|
|
15
|
|
12
|
2
|
|
|
2
|
|
2448
|
use v5.10; |
|
2
|
|
|
|
|
7
|
|
|
2
|
|
|
|
|
1863
|
|
13
|
|
|
|
|
|
|
|
14
|
|
|
|
|
|
|
=head1 NAME |
15
|
|
|
|
|
|
|
|
16
|
|
|
|
|
|
|
App::AllKnowingDNS::Handler - main code of AllKnowingDNS |
17
|
|
|
|
|
|
|
|
18
|
|
|
|
|
|
|
=head1 DESCRIPTION |
19
|
|
|
|
|
|
|
|
20
|
|
|
|
|
|
|
Note: User documentation is in L(1). |
21
|
|
|
|
|
|
|
|
22
|
|
|
|
|
|
|
This module contains the C handler function. |
23
|
|
|
|
|
|
|
|
24
|
|
|
|
|
|
|
=head1 FUNCTIONS |
25
|
|
|
|
|
|
|
|
26
|
|
|
|
|
|
|
=cut |
27
|
|
|
|
|
|
|
|
28
|
|
|
|
|
|
|
our @EXPORT = qw(reply_handler); |
29
|
|
|
|
|
|
|
|
30
|
|
|
|
|
|
|
sub handle_ptr_query { |
31
|
2
|
|
|
2
|
0
|
7
|
my ($querylog, $zone, $qname, $qclass, $qtype) = @_; |
32
|
|
|
|
|
|
|
|
33
|
|
|
|
|
|
|
# Forward this query to our upstream DNS first, if any. |
34
|
2
|
50
|
33
|
|
|
16
|
if (defined($zone->upstream_dns) && |
35
|
|
|
|
|
|
|
$zone->upstream_dns ne '') { |
36
|
0
|
|
|
|
|
0
|
my $resolver = Net::DNS::Resolver->new( |
37
|
|
|
|
|
|
|
nameservers => [ $zone->upstream_dns ], |
38
|
|
|
|
|
|
|
recurse => 0, |
39
|
|
|
|
|
|
|
); |
40
|
0
|
|
|
|
|
0
|
my $result = $resolver->query($qname . '.upstream', 'PTR'); |
41
|
|
|
|
|
|
|
|
42
|
|
|
|
|
|
|
# If the upstream query was successful, relay the response, otherwise |
43
|
|
|
|
|
|
|
# generate a reply. |
44
|
0
|
0
|
0
|
|
|
0
|
if (defined($result) && $result->header->rcode eq 'NOERROR') { |
45
|
0
|
0
|
|
|
|
0
|
if ($querylog) { |
46
|
0
|
|
|
|
|
0
|
say strftime('%x %X %z', localtime) . " - Relaying upstream answer for $qname"; |
47
|
|
|
|
|
|
|
} |
48
|
0
|
|
|
|
|
0
|
my @answer = $result->answer; |
49
|
0
|
|
|
|
|
0
|
for my $answer (@answer) { |
50
|
0
|
|
|
|
|
0
|
my $name = $answer->name; |
51
|
0
|
|
|
|
|
0
|
$name =~ s/\.upstream$//; |
52
|
0
|
|
|
|
|
0
|
$answer->name($name); |
53
|
|
|
|
|
|
|
} |
54
|
0
|
|
|
|
|
0
|
return ('NOERROR', [ $result->answer ], [], [], { aa => 1 }); |
55
|
|
|
|
|
|
|
} |
56
|
|
|
|
|
|
|
} |
57
|
|
|
|
|
|
|
|
58
|
2
|
|
|
|
|
3
|
my $ttl = 3600; |
59
|
2
|
|
|
|
|
5
|
my $fullname = $qname; |
60
|
2
|
|
|
|
|
9
|
substr($fullname, -1 * length($zone->ptrzone)) = ''; |
61
|
2
|
|
|
|
|
13
|
my $hostpart = join '', reverse split /\./, $fullname; |
62
|
2
|
|
|
|
|
10
|
my $rdata = $zone->resolves_to; |
63
|
2
|
|
|
|
|
10
|
$rdata =~ s/%DIGITS%/$hostpart/i; |
64
|
2
|
|
|
|
|
29
|
my $rr = Net::DNS::RR->new("$qname $ttl $qclass $qtype $rdata"); |
65
|
2
|
|
|
|
|
3990
|
return ('NOERROR', [ $rr ], [], [], { aa => 1 }); |
66
|
|
|
|
|
|
|
} |
67
|
|
|
|
|
|
|
|
68
|
|
|
|
|
|
|
sub handle_aaaa_query { |
69
|
4
|
|
|
4
|
0
|
8
|
my ($zone, $qname, $qclass, $qtype) = @_; |
70
|
|
|
|
|
|
|
|
71
|
4
|
|
|
|
|
11
|
my $ttl = 3600; |
72
|
4
|
|
|
|
|
7
|
my $block = '([a-z0-9]{4})'; |
73
|
4
|
|
|
|
|
18
|
my $regexp = quotemeta($zone->resolves_to); |
74
|
4
|
|
|
|
|
32
|
my ($address, $mask) = ($zone->network =~ m,^([^/]+)/([0-9]+),); |
75
|
4
|
|
|
|
|
141
|
my @components = unpack("n8", ipv6_aton($address)); |
76
|
|
|
|
|
|
|
|
77
|
4
|
|
|
|
|
256
|
my $numdigits = (128 - $mask) / 4; |
78
|
4
|
|
|
|
|
58
|
$regexp =~ s/\\%DIGITS\\%/([a-z0-9]{$numdigits})/i; |
79
|
4
|
|
|
|
|
84
|
my ($digits) = ($qname =~ /$regexp/); |
80
|
4
|
50
|
|
|
|
16
|
return ('NXDOMAIN', undef, undef, undef) unless defined($digits); |
81
|
|
|
|
|
|
|
|
82
|
4
|
100
|
|
|
|
12
|
if ($qtype ne 'AAAA') { |
83
|
1
|
|
|
|
|
9
|
return ('NOERROR', [ ], [], [], { aa => 1 }); |
84
|
|
|
|
|
|
|
} |
85
|
|
|
|
|
|
|
|
86
|
|
|
|
|
|
|
# Pad with zeros so that we can match 4 digits each. |
87
|
3
|
|
|
|
|
13
|
$digits = "0$digits" while (length($digits) % 4) != 0; |
88
|
|
|
|
|
|
|
|
89
|
|
|
|
|
|
|
# Collect blocks with 4 digits each |
90
|
3
|
|
|
|
|
8
|
my $numblocks = length($digits) / 4; |
91
|
3
|
|
|
|
|
13
|
for (my $c = 0; $c < $numblocks; $c++) { |
92
|
9
|
|
|
|
|
32
|
$components[8 - $numblocks + $c] |= hex(substr($digits, $c * 4, 4)); |
93
|
|
|
|
|
|
|
} |
94
|
|
|
|
|
|
|
|
95
|
3
|
|
|
|
|
18
|
my $rdata = sprintf("%04x:" x 7 . "%04x", @components); |
96
|
3
|
|
|
|
|
35
|
my $rr = Net::DNS::RR->new("$qname $ttl $qclass $qtype $rdata"); |
97
|
3
|
|
|
|
|
4346
|
return ('NOERROR', [ $rr ], [], [], { aa => 1 }); |
98
|
|
|
|
|
|
|
} |
99
|
|
|
|
|
|
|
|
100
|
|
|
|
|
|
|
=head2 reply_handler($config, $qname, $qclass, $qtype, $peerhost) |
101
|
|
|
|
|
|
|
|
102
|
|
|
|
|
|
|
Handler to be used for Net::DNS::Nameserver. |
103
|
|
|
|
|
|
|
|
104
|
|
|
|
|
|
|
Returns DNS RRs for PTR and AAAA queries of zones which are configured in |
105
|
|
|
|
|
|
|
C<$config>. |
106
|
|
|
|
|
|
|
|
107
|
|
|
|
|
|
|
=cut |
108
|
|
|
|
|
|
|
|
109
|
|
|
|
|
|
|
sub reply_handler { |
110
|
8
|
|
|
8
|
1
|
6861
|
my ($config, $querylog, $qname, $qclass, $qtype, $peerhost) = @_; |
111
|
|
|
|
|
|
|
|
112
|
8
|
50
|
|
|
|
30
|
if ($querylog) { |
113
|
0
|
|
|
|
|
0
|
say strftime('%x %X %z', localtime) . " - $peerhost - query for $qname ($qtype)"; |
114
|
|
|
|
|
|
|
} |
115
|
|
|
|
|
|
|
|
116
|
8
|
100
|
100
|
|
|
41
|
if ($qtype eq 'PTR' && |
117
|
|
|
|
|
|
|
defined(my $zone = $config->zone_for_ptr($qname))) { |
118
|
2
|
|
|
|
|
8
|
return handle_ptr_query($querylog, $zone, $qname, $qclass, $qtype); |
119
|
|
|
|
|
|
|
} |
120
|
|
|
|
|
|
|
|
121
|
6
|
100
|
|
|
|
29
|
if (defined(my $zone = $config->zone_for_aaaa($qname))) { |
122
|
4
|
|
|
|
|
31
|
return handle_aaaa_query($zone, $qname, $qclass, $qtype); |
123
|
|
|
|
|
|
|
} |
124
|
|
|
|
|
|
|
|
125
|
2
|
|
|
|
|
10
|
return ('NXDOMAIN', undef, undef, undef); |
126
|
|
|
|
|
|
|
} |
127
|
|
|
|
|
|
|
|
128
|
|
|
|
|
|
|
1 |
129
|
|
|
|
|
|
|
|
130
|
|
|
|
|
|
|
__END__ |