| line |
stmt |
bran |
cond |
sub |
pod |
time |
code |
|
1
|
|
|
|
|
|
|
package POE::Component::Client::RADIUS; |
|
2
|
|
|
|
|
|
|
$POE::Component::Client::RADIUS::VERSION = '1.04'; |
|
3
|
|
|
|
|
|
|
#ABSTRACT: a flexible POE-based RADIUS client |
|
4
|
|
|
|
|
|
|
|
|
5
|
11
|
|
|
11
|
|
361539
|
use strict; |
|
|
11
|
|
|
|
|
388
|
|
|
|
11
|
|
|
|
|
468
|
|
|
6
|
11
|
|
|
11
|
|
62
|
use warnings; |
|
|
11
|
|
|
|
|
26
|
|
|
|
11
|
|
|
|
|
413
|
|
|
7
|
11
|
|
|
11
|
|
63
|
use Carp; |
|
|
11
|
|
|
|
|
30
|
|
|
|
11
|
|
|
|
|
1060
|
|
|
8
|
11
|
|
|
11
|
|
6002
|
use POE; |
|
|
0
|
|
|
|
|
|
|
|
|
0
|
|
|
|
|
|
|
|
9
|
|
|
|
|
|
|
use IO::Socket::INET; |
|
10
|
|
|
|
|
|
|
use Net::Radius::Dictionary; |
|
11
|
|
|
|
|
|
|
use Net::Radius::Packet; |
|
12
|
|
|
|
|
|
|
use Math::Random; |
|
13
|
|
|
|
|
|
|
|
|
14
|
|
|
|
|
|
|
use constant DATAGRAM_MAXLEN => 4096; |
|
15
|
|
|
|
|
|
|
use constant RADIUS_PORT => 1812; |
|
16
|
|
|
|
|
|
|
use constant ACCOUNTING_PORT => 1813; |
|
17
|
|
|
|
|
|
|
|
|
18
|
|
|
|
|
|
|
my $ERROR; |
|
19
|
|
|
|
|
|
|
my $ERRNO; |
|
20
|
|
|
|
|
|
|
|
|
21
|
|
|
|
|
|
|
# Stolen from POE::Wheel. This is static data, shared by all |
|
22
|
|
|
|
|
|
|
my $current_id = 0; |
|
23
|
|
|
|
|
|
|
my %active_identifiers; |
|
24
|
|
|
|
|
|
|
|
|
25
|
|
|
|
|
|
|
sub spawn { |
|
26
|
|
|
|
|
|
|
my $package = shift; |
|
27
|
|
|
|
|
|
|
return $package->_create( 'spawn', @_ ); |
|
28
|
|
|
|
|
|
|
} |
|
29
|
|
|
|
|
|
|
|
|
30
|
|
|
|
|
|
|
sub authenticate { |
|
31
|
|
|
|
|
|
|
my $self; |
|
32
|
|
|
|
|
|
|
eval { |
|
33
|
|
|
|
|
|
|
if ( (ref $_[0]) && $_[0]->isa(__PACKAGE__) ) { |
|
34
|
|
|
|
|
|
|
$self = shift; |
|
35
|
|
|
|
|
|
|
} |
|
36
|
|
|
|
|
|
|
}; |
|
37
|
|
|
|
|
|
|
if ( $self ) { |
|
38
|
|
|
|
|
|
|
$poe_kernel->post( $self->{session_id}, 'authenticate', @_ ); |
|
39
|
|
|
|
|
|
|
return 1; |
|
40
|
|
|
|
|
|
|
} |
|
41
|
|
|
|
|
|
|
my $package = shift; |
|
42
|
|
|
|
|
|
|
return $package->_create( 'authenticate', @_ ); |
|
43
|
|
|
|
|
|
|
} |
|
44
|
|
|
|
|
|
|
|
|
45
|
|
|
|
|
|
|
sub accounting { |
|
46
|
|
|
|
|
|
|
my $self; |
|
47
|
|
|
|
|
|
|
eval { |
|
48
|
|
|
|
|
|
|
if ( (ref $_[0]) && $_[0]->isa(__PACKAGE__) ) { |
|
49
|
|
|
|
|
|
|
$self = shift; |
|
50
|
|
|
|
|
|
|
} |
|
51
|
|
|
|
|
|
|
}; |
|
52
|
|
|
|
|
|
|
if ( $self ) { |
|
53
|
|
|
|
|
|
|
$poe_kernel->post( $self->{session_id}, 'accounting', @_ ); |
|
54
|
|
|
|
|
|
|
return 1; |
|
55
|
|
|
|
|
|
|
} |
|
56
|
|
|
|
|
|
|
my $package = shift; |
|
57
|
|
|
|
|
|
|
return $package->_create( 'accounting', @_ ); |
|
58
|
|
|
|
|
|
|
} |
|
59
|
|
|
|
|
|
|
|
|
60
|
|
|
|
|
|
|
sub _create { |
|
61
|
|
|
|
|
|
|
my $package = shift; |
|
62
|
|
|
|
|
|
|
my $command = shift; |
|
63
|
|
|
|
|
|
|
my %opts = @_; |
|
64
|
|
|
|
|
|
|
$opts{lc $_} = delete $opts{$_} for grep { !/^_/ } keys %opts; |
|
65
|
|
|
|
|
|
|
unless ( ref $opts{dict} and $opts{dict}->isa('Net::Radius::Dictionary') ) { |
|
66
|
|
|
|
|
|
|
warn "No 'dict' object provided, bailing out\n"; |
|
67
|
|
|
|
|
|
|
return; |
|
68
|
|
|
|
|
|
|
} |
|
69
|
|
|
|
|
|
|
my $options = delete $opts{options}; |
|
70
|
|
|
|
|
|
|
my $self = bless { }, $package; |
|
71
|
|
|
|
|
|
|
if ( $command =~ /^a/ ) { |
|
72
|
|
|
|
|
|
|
unless ( $opts{event} ) { |
|
73
|
|
|
|
|
|
|
warn "You must specify 'event' for '$command'\n"; |
|
74
|
|
|
|
|
|
|
return; |
|
75
|
|
|
|
|
|
|
} |
|
76
|
|
|
|
|
|
|
unless ( $opts{server} and _ip_is_v4( $opts{server} ) ) { |
|
77
|
|
|
|
|
|
|
warn "You must specify 'server' as a valid IPv4 address\n"; |
|
78
|
|
|
|
|
|
|
return; |
|
79
|
|
|
|
|
|
|
} |
|
80
|
|
|
|
|
|
|
unless ( $opts{secret} ) { |
|
81
|
|
|
|
|
|
|
warn "You must specify a 'secret'\n"; |
|
82
|
|
|
|
|
|
|
return; |
|
83
|
|
|
|
|
|
|
} |
|
84
|
|
|
|
|
|
|
unless ( $opts{attributes} and ref $opts{attributes} eq 'HASH' ) { |
|
85
|
|
|
|
|
|
|
warn "You must specify 'attributes' as a hashref of RADIUS attributes\n"; |
|
86
|
|
|
|
|
|
|
return; |
|
87
|
|
|
|
|
|
|
} |
|
88
|
|
|
|
|
|
|
if ( $command eq 'authenticate' and !( $opts{username} and $opts{password} ) ) { |
|
89
|
|
|
|
|
|
|
warn "You must specify 'username' and 'password' for 'authenticate'\n"; |
|
90
|
|
|
|
|
|
|
return; |
|
91
|
|
|
|
|
|
|
} |
|
92
|
|
|
|
|
|
|
if ( $command eq 'accounting' and !$opts{type} ) { |
|
93
|
|
|
|
|
|
|
warn "You must specify 'type' for an accounting request\n"; |
|
94
|
|
|
|
|
|
|
return; |
|
95
|
|
|
|
|
|
|
} |
|
96
|
|
|
|
|
|
|
$opts{port} = RADIUS_PORT if $command eq 'authenticate' and !$opts{port}; |
|
97
|
|
|
|
|
|
|
$opts{port} = ACCOUNTING_PORT if $command eq 'accounting' and !$opts{port}; |
|
98
|
|
|
|
|
|
|
} |
|
99
|
|
|
|
|
|
|
$self->{session_id} = POE::Session->create( |
|
100
|
|
|
|
|
|
|
object_states => [ |
|
101
|
|
|
|
|
|
|
$self => { shutdown => '_shutdown', |
|
102
|
|
|
|
|
|
|
authenticate => '_command', |
|
103
|
|
|
|
|
|
|
accounting => '_command', }, |
|
104
|
|
|
|
|
|
|
$self => [qw(_start _create_socket _dispatch _get_datagram _sock_timeout)], |
|
105
|
|
|
|
|
|
|
], |
|
106
|
|
|
|
|
|
|
heap => $self, |
|
107
|
|
|
|
|
|
|
args => [ $command, %opts ], |
|
108
|
|
|
|
|
|
|
( ref($options) eq 'HASH' ? ( options => $options ) : () ), |
|
109
|
|
|
|
|
|
|
)->ID(); |
|
110
|
|
|
|
|
|
|
return $self; |
|
111
|
|
|
|
|
|
|
} |
|
112
|
|
|
|
|
|
|
|
|
113
|
|
|
|
|
|
|
sub _allocate_identifier { |
|
114
|
|
|
|
|
|
|
while (1) { |
|
115
|
|
|
|
|
|
|
++$current_id; |
|
116
|
|
|
|
|
|
|
$current_id = 1 if $current_id > 255; |
|
117
|
|
|
|
|
|
|
last unless exists $active_identifiers{ $current_id }; |
|
118
|
|
|
|
|
|
|
} |
|
119
|
|
|
|
|
|
|
return $active_identifiers{$current_id} = $current_id; |
|
120
|
|
|
|
|
|
|
} |
|
121
|
|
|
|
|
|
|
|
|
122
|
|
|
|
|
|
|
sub _free_identifier { |
|
123
|
|
|
|
|
|
|
my $id = shift; |
|
124
|
|
|
|
|
|
|
delete $active_identifiers{$id}; |
|
125
|
|
|
|
|
|
|
} |
|
126
|
|
|
|
|
|
|
|
|
127
|
|
|
|
|
|
|
sub session_id { |
|
128
|
|
|
|
|
|
|
return $_[0]->{session_id}; |
|
129
|
|
|
|
|
|
|
} |
|
130
|
|
|
|
|
|
|
|
|
131
|
|
|
|
|
|
|
sub shutdown { |
|
132
|
|
|
|
|
|
|
my $self = shift; |
|
133
|
|
|
|
|
|
|
$poe_kernel->post( $self->{session_id}, 'shutdown' ); |
|
134
|
|
|
|
|
|
|
} |
|
135
|
|
|
|
|
|
|
|
|
136
|
|
|
|
|
|
|
sub _start { |
|
137
|
|
|
|
|
|
|
my ($kernel,$self,$sender,$command,@args) = @_[KERNEL,OBJECT,SENDER,ARG0..$#_]; |
|
138
|
|
|
|
|
|
|
$self->{session_id} = $_[SESSION]->ID(); |
|
139
|
|
|
|
|
|
|
if ( $command eq 'spawn' ) { |
|
140
|
|
|
|
|
|
|
my $opts = { @args }; |
|
141
|
|
|
|
|
|
|
$self->{$_} = $opts->{$_} for keys %{ $opts }; |
|
142
|
|
|
|
|
|
|
$kernel->alias_set($self->{alias}) if $self->{alias}; |
|
143
|
|
|
|
|
|
|
$kernel->refcount_increment($self->{session_id}, __PACKAGE__) unless $self->{alias}; |
|
144
|
|
|
|
|
|
|
return; |
|
145
|
|
|
|
|
|
|
} |
|
146
|
|
|
|
|
|
|
if ( $kernel == $sender ) { |
|
147
|
|
|
|
|
|
|
croak "'authenticate' and 'accounting' should be called from another POE Session\n"; |
|
148
|
|
|
|
|
|
|
} |
|
149
|
|
|
|
|
|
|
$self->{sender_id} = $sender->ID(); |
|
150
|
|
|
|
|
|
|
$kernel->refcount_increment( $self->{sender_id}, __PACKAGE__ ); |
|
151
|
|
|
|
|
|
|
$kernel->yield( $command, @args ); |
|
152
|
|
|
|
|
|
|
return; |
|
153
|
|
|
|
|
|
|
} |
|
154
|
|
|
|
|
|
|
|
|
155
|
|
|
|
|
|
|
sub _shutdown { |
|
156
|
|
|
|
|
|
|
my ($kernel,$self) = @_[KERNEL,OBJECT]; |
|
157
|
|
|
|
|
|
|
$kernel->alarm_remove_all(); |
|
158
|
|
|
|
|
|
|
$kernel->alias_remove( $_ ) for $kernel->alias_list(); |
|
159
|
|
|
|
|
|
|
$kernel->refcount_decrement( $self->{session_id} => __PACKAGE__ ) unless $self->{alias}; |
|
160
|
|
|
|
|
|
|
return; |
|
161
|
|
|
|
|
|
|
} |
|
162
|
|
|
|
|
|
|
|
|
163
|
|
|
|
|
|
|
sub _command { |
|
164
|
|
|
|
|
|
|
my ($kernel,$self,$state,$session,$sender) = @_[KERNEL,OBJECT,STATE,SESSION,SENDER]; |
|
165
|
|
|
|
|
|
|
my $args; |
|
166
|
|
|
|
|
|
|
if ( ref $_[ARG0] eq 'HASH' ) { |
|
167
|
|
|
|
|
|
|
$args = $_[ARG0]; |
|
168
|
|
|
|
|
|
|
} |
|
169
|
|
|
|
|
|
|
else { |
|
170
|
|
|
|
|
|
|
$args = { @_[ARG0..$#_] }; |
|
171
|
|
|
|
|
|
|
} |
|
172
|
|
|
|
|
|
|
$args->{cmd} = $state; |
|
173
|
|
|
|
|
|
|
if ( $session == $sender ) { |
|
174
|
|
|
|
|
|
|
$args->{sender_id} = $self->{sender_id}; |
|
175
|
|
|
|
|
|
|
$self->{dict} = delete $args->{dict}; |
|
176
|
|
|
|
|
|
|
} |
|
177
|
|
|
|
|
|
|
else { |
|
178
|
|
|
|
|
|
|
$args->{lc $_} = delete $args->{$_} for grep { !/^_/ } keys %{ $args }; |
|
179
|
|
|
|
|
|
|
$args->{sender_id} = $sender->ID(); |
|
180
|
|
|
|
|
|
|
unless ( $args->{event} ) { |
|
181
|
|
|
|
|
|
|
warn "You must specify 'SuccessEvent' and 'FailureEvent' for '$state'\n"; |
|
182
|
|
|
|
|
|
|
return; |
|
183
|
|
|
|
|
|
|
} |
|
184
|
|
|
|
|
|
|
unless ( $args->{server} and _ip_is_v4( $args->{server} ) ) { |
|
185
|
|
|
|
|
|
|
warn "You must specify 'server' as a valid IPv4 address\n"; |
|
186
|
|
|
|
|
|
|
return; |
|
187
|
|
|
|
|
|
|
} |
|
188
|
|
|
|
|
|
|
unless ( $args->{secret} ) { |
|
189
|
|
|
|
|
|
|
warn "You must specify a 'secret'\n"; |
|
190
|
|
|
|
|
|
|
return; |
|
191
|
|
|
|
|
|
|
} |
|
192
|
|
|
|
|
|
|
unless ( $args->{attributes} and ref $args->{attributes} eq 'HASH' ) { |
|
193
|
|
|
|
|
|
|
warn "You must specify 'attributes' as a hashref of RADIUS attributes\n"; |
|
194
|
|
|
|
|
|
|
return; |
|
195
|
|
|
|
|
|
|
} |
|
196
|
|
|
|
|
|
|
if ( $state eq 'authenticate' and !( $args->{username} and $args->{password} ) ) { |
|
197
|
|
|
|
|
|
|
warn "You must specify 'username' and 'password' for 'authenticate'\n"; |
|
198
|
|
|
|
|
|
|
return; |
|
199
|
|
|
|
|
|
|
} |
|
200
|
|
|
|
|
|
|
if ( $state eq 'accounting' and !$args->{type} ) { |
|
201
|
|
|
|
|
|
|
warn "You must specify 'type' for an accounting request\n"; |
|
202
|
|
|
|
|
|
|
return; |
|
203
|
|
|
|
|
|
|
} |
|
204
|
|
|
|
|
|
|
$args->{port} = RADIUS_PORT if $state eq 'authenticate' and !$args->{port}; |
|
205
|
|
|
|
|
|
|
$args->{port} = ACCOUNTING_PORT if $state eq 'accounting' and !$args->{port}; |
|
206
|
|
|
|
|
|
|
$kernel->refcount_increment( $args->{sender_id}, __PACKAGE__ ); |
|
207
|
|
|
|
|
|
|
} |
|
208
|
|
|
|
|
|
|
my $req = Net::Radius::Packet->new( $self->{dict} ); |
|
209
|
|
|
|
|
|
|
my $packet; |
|
210
|
|
|
|
|
|
|
if ( $state eq 'authenticate' ) { |
|
211
|
|
|
|
|
|
|
$args->{identifier} = _allocate_identifier(); |
|
212
|
|
|
|
|
|
|
$args->{authenticator} = _bigrand(); |
|
213
|
|
|
|
|
|
|
$req->set_code('Access-Request'); |
|
214
|
|
|
|
|
|
|
$req->set_attr('User-Name' => $args->{username}); |
|
215
|
|
|
|
|
|
|
$req->set_attr('Service-Type' => '2'); |
|
216
|
|
|
|
|
|
|
$req->set_attr('Framed-Protocol' => 'PPP'); |
|
217
|
|
|
|
|
|
|
$req->set_attr('NAS-Port' => 1234); |
|
218
|
|
|
|
|
|
|
$req->set_attr('NAS-Identifier' => 'PoCoClientRADIUS'); |
|
219
|
|
|
|
|
|
|
$req->set_attr('NAS-IP-Address' => _my_address( $args->{server} ) ); |
|
220
|
|
|
|
|
|
|
$req->set_attr('Called-Station-Id' => '0000'); |
|
221
|
|
|
|
|
|
|
$req->set_attr('Calling-Station-Id' => '01234567890'); |
|
222
|
|
|
|
|
|
|
delete $args->{attributes}->{'User-Name'}; |
|
223
|
|
|
|
|
|
|
$req->set_attr( $_ => $args->{attributes}->{$_} ) for keys %{ $args->{attributes} }; |
|
224
|
|
|
|
|
|
|
$req->set_identifier( $args->{identifier} ); |
|
225
|
|
|
|
|
|
|
$req->set_authenticator( $args->{authenticator} ); |
|
226
|
|
|
|
|
|
|
$req->set_password( $args->{password}, $args->{secret} ); |
|
227
|
|
|
|
|
|
|
$packet = $req->pack; |
|
228
|
|
|
|
|
|
|
} |
|
229
|
|
|
|
|
|
|
if ( $state eq 'accounting' ) { |
|
230
|
|
|
|
|
|
|
$args->{identifier} = _allocate_identifier(); |
|
231
|
|
|
|
|
|
|
$args->{authenticator} = ''; |
|
232
|
|
|
|
|
|
|
$req->set_code('Accounting-Request'); |
|
233
|
|
|
|
|
|
|
$req->set_attr('Acct-Status-Type', ucfirst lc $args->{type}); |
|
234
|
|
|
|
|
|
|
delete $args->{attributes}->{'Acct-Status-Type'}; |
|
235
|
|
|
|
|
|
|
$req->set_attr( $_ => $args->{attributes}->{$_} ) for keys %{ $args->{attributes} }; |
|
236
|
|
|
|
|
|
|
$req->set_identifier( $args->{identifier} ); |
|
237
|
|
|
|
|
|
|
$req->set_authenticator( $args->{authenticator} ); |
|
238
|
|
|
|
|
|
|
$packet = auth_resp($req->pack,$args->{secret}); |
|
239
|
|
|
|
|
|
|
} |
|
240
|
|
|
|
|
|
|
$kernel->yield( '_create_socket', $packet, $args ); |
|
241
|
|
|
|
|
|
|
return; |
|
242
|
|
|
|
|
|
|
} |
|
243
|
|
|
|
|
|
|
|
|
244
|
|
|
|
|
|
|
sub _create_socket { |
|
245
|
|
|
|
|
|
|
my ($kernel,$self,$packet,$data) = @_[KERNEL,OBJECT,ARG0,ARG1]; |
|
246
|
|
|
|
|
|
|
my $socket = IO::Socket::INET->new( Proto => 'udp' ); |
|
247
|
|
|
|
|
|
|
$kernel->select_read( $socket, '_get_datagram', $data ); |
|
248
|
|
|
|
|
|
|
unless ( $socket ) { |
|
249
|
|
|
|
|
|
|
$data->{error} = $!; |
|
250
|
|
|
|
|
|
|
$kernel->yield( '_dispatch', $data ); |
|
251
|
|
|
|
|
|
|
return; |
|
252
|
|
|
|
|
|
|
} |
|
253
|
|
|
|
|
|
|
my $server_address = pack_sockaddr_in( $data->{port}, inet_aton($data->{server}) ); |
|
254
|
|
|
|
|
|
|
unless ( $server_address ) { |
|
255
|
|
|
|
|
|
|
$data->{error} = 'Couldn\'t create packed server address and socket'; |
|
256
|
|
|
|
|
|
|
$kernel->yield( '_dispatch', $data ); |
|
257
|
|
|
|
|
|
|
return; |
|
258
|
|
|
|
|
|
|
} |
|
259
|
|
|
|
|
|
|
unless ( send( $socket, $packet, 0, $server_address ) == length($packet) ) { |
|
260
|
|
|
|
|
|
|
$data->{error} = $!; |
|
261
|
|
|
|
|
|
|
$kernel->yield( '_dispatch', $data ); |
|
262
|
|
|
|
|
|
|
return; |
|
263
|
|
|
|
|
|
|
} |
|
264
|
|
|
|
|
|
|
$data->{alarm_id} = $kernel->delay_set( '_sock_timeout', $self->{timeout} || 10, $socket, $data ); |
|
265
|
|
|
|
|
|
|
return; |
|
266
|
|
|
|
|
|
|
} |
|
267
|
|
|
|
|
|
|
|
|
268
|
|
|
|
|
|
|
sub _sock_timeout { |
|
269
|
|
|
|
|
|
|
my ($kernel,$self,$socket,$data) = @_[KERNEL,OBJECT,ARG0,ARG1]; |
|
270
|
|
|
|
|
|
|
$kernel->select_read( $socket ); |
|
271
|
|
|
|
|
|
|
$data->{timeout} = 'Timeout waiting for a response'; |
|
272
|
|
|
|
|
|
|
$kernel->yield( '_dispatch', $data ); |
|
273
|
|
|
|
|
|
|
return; |
|
274
|
|
|
|
|
|
|
} |
|
275
|
|
|
|
|
|
|
|
|
276
|
|
|
|
|
|
|
sub _get_datagram { |
|
277
|
|
|
|
|
|
|
my ($kernel,$self,$socket,$data) = @_[KERNEL,OBJECT,ARG0,ARG2]; |
|
278
|
|
|
|
|
|
|
$kernel->alarm_remove( delete $data->{alarm_id} ); |
|
279
|
|
|
|
|
|
|
$kernel->select_read( $socket ); |
|
280
|
|
|
|
|
|
|
my $remote_address = recv( $socket, my $message = '', 4096, 0 ); |
|
281
|
|
|
|
|
|
|
unless ( defined $remote_address ) { |
|
282
|
|
|
|
|
|
|
$data->{error} = $!; |
|
283
|
|
|
|
|
|
|
$kernel->yield( '_dispatch', $data ); |
|
284
|
|
|
|
|
|
|
return; |
|
285
|
|
|
|
|
|
|
} |
|
286
|
|
|
|
|
|
|
my $resp = Net::Radius::Packet->new( $self->{dict}, $message ); |
|
287
|
|
|
|
|
|
|
my ($port, $iaddr) = unpack_sockaddr_in( $remote_address ); |
|
288
|
|
|
|
|
|
|
$iaddr = inet_ntoa( $iaddr ); |
|
289
|
|
|
|
|
|
|
if ( $data->{identifier} ne $resp->identifier or $iaddr ne $data->{server} ) { |
|
290
|
|
|
|
|
|
|
$data->{error} = 'Unexpected response to request.'; |
|
291
|
|
|
|
|
|
|
$kernel->yield( '_dispatch', $data ); |
|
292
|
|
|
|
|
|
|
return; |
|
293
|
|
|
|
|
|
|
} |
|
294
|
|
|
|
|
|
|
if ( $data->{cmd} eq 'authenticate' and !auth_req_verify( $message, $data->{secret}, $data->{authenticator} ) ) { |
|
295
|
|
|
|
|
|
|
$data->{error} = 'Couldn\'t authenticate the response from the server.'; |
|
296
|
|
|
|
|
|
|
$kernel->yield( '_dispatch', $data ); |
|
297
|
|
|
|
|
|
|
return; |
|
298
|
|
|
|
|
|
|
} |
|
299
|
|
|
|
|
|
|
my $reply = { |
|
300
|
|
|
|
|
|
|
map { ( $_, $resp->attr($_) ) } $resp->attributes() |
|
301
|
|
|
|
|
|
|
}; |
|
302
|
|
|
|
|
|
|
$reply->{Code} = $resp->code; |
|
303
|
|
|
|
|
|
|
$data->{response} = $reply; |
|
304
|
|
|
|
|
|
|
$kernel->yield( '_dispatch', $data ); |
|
305
|
|
|
|
|
|
|
return; |
|
306
|
|
|
|
|
|
|
} |
|
307
|
|
|
|
|
|
|
|
|
308
|
|
|
|
|
|
|
sub _dispatch { |
|
309
|
|
|
|
|
|
|
my ($kernel,$self,$data) = @_[KERNEL,OBJECT,ARG0]; |
|
310
|
|
|
|
|
|
|
delete $data->{authenticator}; |
|
311
|
|
|
|
|
|
|
my $ident = delete $data->{identifier}; |
|
312
|
|
|
|
|
|
|
_free_identifier( $ident ) if $ident; |
|
313
|
|
|
|
|
|
|
$kernel->post( $data->{sender_id}, $data->{event}, $data ); |
|
314
|
|
|
|
|
|
|
$kernel->refcount_decrement( delete $data->{sender_id}, __PACKAGE__ ); |
|
315
|
|
|
|
|
|
|
return; |
|
316
|
|
|
|
|
|
|
} |
|
317
|
|
|
|
|
|
|
|
|
318
|
|
|
|
|
|
|
#------------------------------------------------------------------------------ |
|
319
|
|
|
|
|
|
|
# Subroutine _ip_is_ipv4 |
|
320
|
|
|
|
|
|
|
# Purpose : Check if an IP address is version 4 |
|
321
|
|
|
|
|
|
|
# Params : IP address |
|
322
|
|
|
|
|
|
|
# Returns : 1 (yes) or 0 (no) |
|
323
|
|
|
|
|
|
|
sub _ip_is_v4 { |
|
324
|
|
|
|
|
|
|
my $ip = shift; |
|
325
|
|
|
|
|
|
|
|
|
326
|
|
|
|
|
|
|
# Check for invalid chars |
|
327
|
|
|
|
|
|
|
unless ($ip =~ m/^[\d\.]+$/) { |
|
328
|
|
|
|
|
|
|
$ERROR = "Invalid chars in IP $ip"; |
|
329
|
|
|
|
|
|
|
$ERRNO = 107; |
|
330
|
|
|
|
|
|
|
return 0; |
|
331
|
|
|
|
|
|
|
} |
|
332
|
|
|
|
|
|
|
|
|
333
|
|
|
|
|
|
|
if ($ip =~ m/^\./) { |
|
334
|
|
|
|
|
|
|
$ERROR = "Invalid IP $ip - starts with a dot"; |
|
335
|
|
|
|
|
|
|
$ERRNO = 103; |
|
336
|
|
|
|
|
|
|
return 0; |
|
337
|
|
|
|
|
|
|
} |
|
338
|
|
|
|
|
|
|
|
|
339
|
|
|
|
|
|
|
if ($ip =~ m/\.$/) { |
|
340
|
|
|
|
|
|
|
$ERROR = "Invalid IP $ip - ends with a dot"; |
|
341
|
|
|
|
|
|
|
$ERRNO = 104; |
|
342
|
|
|
|
|
|
|
return 0; |
|
343
|
|
|
|
|
|
|
} |
|
344
|
|
|
|
|
|
|
|
|
345
|
|
|
|
|
|
|
# Single Numbers are considered to be IPv4 |
|
346
|
|
|
|
|
|
|
if ($ip =~ m/^(\d+)$/ and $1 < 256) { return 1 } |
|
347
|
|
|
|
|
|
|
|
|
348
|
|
|
|
|
|
|
# Count quads |
|
349
|
|
|
|
|
|
|
my $n = ($ip =~ tr/\./\./); |
|
350
|
|
|
|
|
|
|
|
|
351
|
|
|
|
|
|
|
# IPv4 must have from 1 to 4 quads |
|
352
|
|
|
|
|
|
|
unless ($n >= 0 and $n < 4) { |
|
353
|
|
|
|
|
|
|
$ERROR = "Invalid IP address $ip"; |
|
354
|
|
|
|
|
|
|
$ERRNO = 105; |
|
355
|
|
|
|
|
|
|
return 0; |
|
356
|
|
|
|
|
|
|
} |
|
357
|
|
|
|
|
|
|
|
|
358
|
|
|
|
|
|
|
# Check for empty quads |
|
359
|
|
|
|
|
|
|
if ($ip =~ m/\.\./) { |
|
360
|
|
|
|
|
|
|
$ERROR = "Empty quad in IP address $ip"; |
|
361
|
|
|
|
|
|
|
$ERRNO = 106; |
|
362
|
|
|
|
|
|
|
return 0; |
|
363
|
|
|
|
|
|
|
} |
|
364
|
|
|
|
|
|
|
|
|
365
|
|
|
|
|
|
|
foreach (split /\./, $ip) { |
|
366
|
|
|
|
|
|
|
|
|
367
|
|
|
|
|
|
|
# Check for invalid quads |
|
368
|
|
|
|
|
|
|
unless ($_ >= 0 and $_ < 256) { |
|
369
|
|
|
|
|
|
|
$ERROR = "Invalid quad in IP address $ip - $_"; |
|
370
|
|
|
|
|
|
|
$ERRNO = 107; |
|
371
|
|
|
|
|
|
|
return 0; |
|
372
|
|
|
|
|
|
|
} |
|
373
|
|
|
|
|
|
|
} |
|
374
|
|
|
|
|
|
|
return 1; |
|
375
|
|
|
|
|
|
|
} |
|
376
|
|
|
|
|
|
|
|
|
377
|
|
|
|
|
|
|
sub _bigrand { |
|
378
|
|
|
|
|
|
|
my @numbers; |
|
379
|
|
|
|
|
|
|
push @numbers, scalar random_uniform_integer(1,0,65536) for 0 .. 7; |
|
380
|
|
|
|
|
|
|
pack "n8", @numbers; |
|
381
|
|
|
|
|
|
|
} |
|
382
|
|
|
|
|
|
|
|
|
383
|
|
|
|
|
|
|
sub _my_address { |
|
384
|
|
|
|
|
|
|
my $remote = shift || '198.41.0.4'; |
|
385
|
|
|
|
|
|
|
my $socket = IO::Socket::INET->new( |
|
386
|
|
|
|
|
|
|
Proto => 'udp', |
|
387
|
|
|
|
|
|
|
PeerAddr => $remote, |
|
388
|
|
|
|
|
|
|
PeerPort => 53, |
|
389
|
|
|
|
|
|
|
); |
|
390
|
|
|
|
|
|
|
return unless $socket; |
|
391
|
|
|
|
|
|
|
return $socket->sockhost; |
|
392
|
|
|
|
|
|
|
} |
|
393
|
|
|
|
|
|
|
|
|
394
|
|
|
|
|
|
|
qq[Sound of crickets]; |
|
395
|
|
|
|
|
|
|
|
|
396
|
|
|
|
|
|
|
__END__ |