| line |
stmt |
bran |
cond |
sub |
pod |
time |
code |
|
1
|
|
|
|
|
|
|
package POE::Component::Client::Rcon; |
|
2
|
|
|
|
|
|
|
|
|
3
|
1
|
|
|
1
|
|
742
|
use strict; |
|
|
1
|
|
|
|
|
3
|
|
|
|
1
|
|
|
|
|
35
|
|
|
4
|
|
|
|
|
|
|
|
|
5
|
1
|
|
|
1
|
|
6
|
use vars qw($VERSION $playerSN); |
|
|
1
|
|
|
|
|
1
|
|
|
|
1
|
|
|
|
|
75
|
|
|
6
|
|
|
|
|
|
|
$VERSION = '0.23'; |
|
7
|
|
|
|
|
|
|
$playerSN = 0; |
|
8
|
|
|
|
|
|
|
|
|
9
|
1
|
|
|
1
|
|
5
|
use Carp qw(croak); |
|
|
1
|
|
|
|
|
5
|
|
|
|
1
|
|
|
|
|
67
|
|
|
10
|
1
|
|
|
1
|
|
924
|
use Socket; |
|
|
1
|
|
|
|
|
4968
|
|
|
|
1
|
|
|
|
|
727
|
|
|
11
|
1
|
|
|
1
|
|
1243
|
use Time::HiRes qw(time); |
|
|
1
|
|
|
|
|
1911
|
|
|
|
1
|
|
|
|
|
5
|
|
|
12
|
1
|
|
|
1
|
|
1041
|
use POE qw(Session Wheel::SocketFactory); |
|
|
1
|
|
|
|
|
52124
|
|
|
|
1
|
|
|
|
|
8
|
|
|
13
|
|
|
|
|
|
|
|
|
14
|
|
|
|
|
|
|
sub DEBUG () { 0 }; |
|
15
|
|
|
|
|
|
|
|
|
16
|
|
|
|
|
|
|
sub new { |
|
17
|
0
|
|
|
0
|
0
|
|
my $type = shift; |
|
18
|
0
|
|
|
|
|
|
my $self = bless {}, $type; |
|
19
|
|
|
|
|
|
|
|
|
20
|
0
|
0
|
|
|
|
|
croak "$type requires an event number of parameters" if @_ % 2; |
|
21
|
|
|
|
|
|
|
|
|
22
|
0
|
|
|
|
|
|
my %params = @_; |
|
23
|
|
|
|
|
|
|
|
|
24
|
0
|
|
|
|
|
|
my $alias = delete $params{Alias}; |
|
25
|
0
|
0
|
|
|
|
|
$alias = 'rcon' unless defined $alias; |
|
26
|
|
|
|
|
|
|
|
|
27
|
0
|
|
|
|
|
|
my $timeout = delete $params{Timeout}; |
|
28
|
0
|
0
|
0
|
|
|
|
$timeout = 15 unless defined $timeout and $timeout >= 0; |
|
29
|
|
|
|
|
|
|
|
|
30
|
0
|
|
|
|
|
|
my $retry = delete $params{Retry}; |
|
31
|
0
|
0
|
0
|
|
|
|
$retry = 2 unless defined $retry and $retry >= 0; |
|
32
|
|
|
|
|
|
|
|
|
33
|
0
|
|
|
|
|
|
my $bytes = delete $params{Bytes}; |
|
34
|
0
|
0
|
0
|
|
|
|
$bytes = 8192 unless defined $bytes and $bytes > 0; |
|
35
|
|
|
|
|
|
|
|
|
36
|
0
|
0
|
|
|
|
|
croak "$type doesn't know these parameters: ", join(', ', sort(keys(%params))) if scalar(keys(%params)); |
|
37
|
|
|
|
|
|
|
|
|
38
|
0
|
|
|
|
|
|
POE::Session->create( |
|
39
|
|
|
|
|
|
|
inline_states => { |
|
40
|
|
|
|
|
|
|
_start => \&_start, |
|
41
|
|
|
|
|
|
|
rcon => \&rcon, |
|
42
|
|
|
|
|
|
|
got_socket => \&got_socket, |
|
43
|
|
|
|
|
|
|
got_message => \&got_message, |
|
44
|
|
|
|
|
|
|
got_error => \&got_error, |
|
45
|
|
|
|
|
|
|
got_challenge => \&got_challenge, |
|
46
|
|
|
|
|
|
|
got_rcon_response => \&got_rcon_response, |
|
47
|
|
|
|
|
|
|
challenge_timeout => \&challenge_timeout, |
|
48
|
|
|
|
|
|
|
rcon_timeout => \&rcon_timeout, |
|
49
|
|
|
|
|
|
|
players => \&players, |
|
50
|
|
|
|
|
|
|
player_response => \&player_response, |
|
51
|
|
|
|
|
|
|
player_parse_hl => \&player_parse_hl, |
|
52
|
|
|
|
|
|
|
player_parse_quake => \&player_parse_quake, |
|
53
|
|
|
|
|
|
|
}, |
|
54
|
|
|
|
|
|
|
args => [ $timeout, $retry, $alias, $bytes ], |
|
55
|
|
|
|
|
|
|
); |
|
56
|
|
|
|
|
|
|
|
|
57
|
0
|
|
|
|
|
|
return $self; |
|
58
|
|
|
|
|
|
|
} |
|
59
|
|
|
|
|
|
|
|
|
60
|
|
|
|
|
|
|
sub _start { |
|
61
|
0
|
|
|
0
|
|
|
my ($kernel, $heap, $timeout, $retry, $alias, $bytes) = @_[KERNEL, HEAP, ARG0..ARG3]; |
|
62
|
0
|
|
|
|
|
|
$heap->{timeout} = $timeout; |
|
63
|
0
|
|
|
|
|
|
$heap->{retry} = $retry; |
|
64
|
0
|
|
|
|
|
|
$heap->{bytes} = $bytes; |
|
65
|
0
|
|
|
|
|
|
$kernel->alias_set($alias); |
|
66
|
0
|
|
|
|
|
|
print STDERR "Rcon object started.\n" if DEBUG; |
|
67
|
|
|
|
|
|
|
} |
|
68
|
|
|
|
|
|
|
|
|
69
|
|
|
|
|
|
|
sub rcon { |
|
70
|
0
|
|
|
0
|
1
|
|
my ($kernel, $heap, $sender, $type, $ip, $port, $pw, $cmd, $postback) = @_[KERNEL, HEAP, SENDER, ARG0..ARG5]; |
|
71
|
0
|
0
|
|
|
|
|
my ($identifier) = defined($_[ARG6]) ? $_[ARG6] : undef; |
|
72
|
0
|
|
|
|
|
|
print STDERR "Got $ip:$port with password $pw running command $cmd with postback $postback\n" if DEBUG; |
|
73
|
0
|
0
|
|
|
|
|
croak "IP address required to execute an Rcon command" unless defined $ip; |
|
74
|
0
|
0
|
0
|
|
|
|
croak "Port requred to execute an Rcon command" if !defined $port || $port !~ /^\d+$/; |
|
75
|
0
|
0
|
0
|
|
|
|
croak "Password requires to execute an Rcon command" if !defined $pw || $pw eq ''; |
|
76
|
0
|
0
|
0
|
|
|
|
croak "Command required to execute an Rcon command" if !defined $cmd || $cmd eq ''; |
|
77
|
0
|
0
|
|
|
|
|
croak "Server type was not recognized" unless $type =~ /^(?:qw|q2|q3|oldhl|hl)$/; |
|
78
|
0
|
|
|
|
|
|
my $challenge = ''; |
|
79
|
0
|
|
|
|
|
|
my $wheel = POE::Wheel::SocketFactory->new( |
|
80
|
|
|
|
|
|
|
RemoteAddress => $ip, |
|
81
|
|
|
|
|
|
|
RemotePort => $port, |
|
82
|
|
|
|
|
|
|
SocketProtocol => 'udp', |
|
83
|
|
|
|
|
|
|
SuccessEvent => 'got_socket', |
|
84
|
|
|
|
|
|
|
FailureEvent => 'got_error', |
|
85
|
|
|
|
|
|
|
); |
|
86
|
0
|
|
|
|
|
|
$heap->{w_jobs}->{$wheel->ID()} = { |
|
87
|
|
|
|
|
|
|
ip => $ip, |
|
88
|
|
|
|
|
|
|
port => $port, |
|
89
|
|
|
|
|
|
|
pw => $pw, |
|
90
|
|
|
|
|
|
|
cmd => $cmd, |
|
91
|
|
|
|
|
|
|
postback => $postback, |
|
92
|
|
|
|
|
|
|
session => $sender->ID(), |
|
93
|
|
|
|
|
|
|
wheel => $wheel, |
|
94
|
|
|
|
|
|
|
identifier => $identifier, |
|
95
|
|
|
|
|
|
|
type => $type, |
|
96
|
|
|
|
|
|
|
try => 1, # number of tries... |
|
97
|
|
|
|
|
|
|
}; |
|
98
|
0
|
|
|
|
|
|
return undef; |
|
99
|
|
|
|
|
|
|
} |
|
100
|
|
|
|
|
|
|
|
|
101
|
|
|
|
|
|
|
sub got_error { |
|
102
|
0
|
|
|
0
|
0
|
|
my ($operation, $errnum, $errstr, $wheel_id, $heap) = @_[ARG0..ARG3,HEAP]; |
|
103
|
0
|
|
|
|
|
|
warn "Wheel $wheel_id generated $operation error $errnum: $errstr\n"; |
|
104
|
0
|
|
|
|
|
|
delete $heap->{w_jobs}->{$wheel_id}; # shut down that wheel |
|
105
|
|
|
|
|
|
|
} |
|
106
|
|
|
|
|
|
|
|
|
107
|
|
|
|
|
|
|
sub got_socket { |
|
108
|
0
|
|
|
0
|
0
|
|
my ($kernel, $heap, $socket, $wheelid) = @_[KERNEL, HEAP, ARG0, ARG3]; |
|
109
|
|
|
|
|
|
|
|
|
110
|
0
|
|
|
|
|
|
$heap->{jobs}->{$socket} = delete($heap->{w_jobs}->{$wheelid}); |
|
111
|
0
|
0
|
|
|
|
|
if($heap->{jobs}->{$socket}->{type} eq 'hl') { |
|
112
|
0
|
|
|
|
|
|
$kernel->select_read($socket, 'got_challenge'); |
|
113
|
0
|
|
|
|
|
|
send($socket, "\xFF\xFF\xFF\xFFchallenge rcon\n\0", 0); |
|
114
|
0
|
|
|
|
|
|
$heap->{jobs}->{$socket}->{timer} = $kernel->delay_set('challenge_timeout', $heap->{timeout}, $socket); |
|
115
|
0
|
|
|
|
|
|
print STDERR "Wheel $wheelid got socket and sent rcon challenge\n" if DEBUG; |
|
116
|
|
|
|
|
|
|
} else { |
|
117
|
0
|
|
|
|
|
|
$kernel->yield('got_challenge', $socket); |
|
118
|
|
|
|
|
|
|
} |
|
119
|
|
|
|
|
|
|
} |
|
120
|
|
|
|
|
|
|
|
|
121
|
|
|
|
|
|
|
sub got_challenge { |
|
122
|
0
|
|
|
0
|
0
|
|
my ($kernel, $heap, $socket) = @_[KERNEL, HEAP, ARG0]; |
|
123
|
|
|
|
|
|
|
|
|
124
|
0
|
0
|
|
|
|
|
if($heap->{jobs}->{$socket}->{type} eq 'hl') { |
|
125
|
0
|
0
|
|
|
|
|
$kernel->alarm_remove($heap->{jobs}->{$socket}->{timer}) if defined $heap->{jobs}->{$socket}->{timer}; |
|
126
|
0
|
|
|
|
|
|
delete($heap->{jobs}->{$socket}->{timer}); |
|
127
|
0
|
|
|
|
|
|
$kernel->select_read($socket); |
|
128
|
0
|
|
|
|
|
|
recv($socket, my $response = '', 8192, 0); |
|
129
|
|
|
|
|
|
|
|
|
130
|
0
|
|
|
|
|
|
print STDERR "got_challenge got the response \"$response\" for $socket\n" if DEBUG; |
|
131
|
0
|
0
|
|
|
|
|
if($response =~ /challenge +rcon +(\d+)/) { |
|
132
|
0
|
|
|
|
|
|
$heap->{jobs}->{$socket}->{challenge} = $1; |
|
133
|
0
|
|
|
|
|
|
$kernel->select_read($socket, 'got_rcon_response'); |
|
134
|
0
|
|
|
|
|
|
send($socket, "\xFF\xFF\xFF\xFFrcon $1 \"" . $heap->{jobs}->{$socket}->{pw} . "\" " . $heap->{jobs}->{$socket}->{cmd} . "\0", 0); |
|
135
|
0
|
|
|
|
|
|
$heap->{jobs}->{$socket}->{timer} = $kernel->delay_set('rcon_timeout', $heap->{timeout}, $socket); |
|
136
|
0
|
|
|
|
|
|
print STDERR "Got rcon response and sent rcon command\n" if DEBUG; |
|
137
|
|
|
|
|
|
|
} else { |
|
138
|
0
|
|
|
|
|
|
$kernel->post($heap->{jobs}->{$socket}->{session}, $heap->{jobs}->{$socket}->{postback}, |
|
139
|
|
|
|
|
|
|
$heap->{jobs}->{$socket}->{type}. |
|
140
|
|
|
|
|
|
|
$heap->{jobs}->{$socket}->{ip}, |
|
141
|
|
|
|
|
|
|
$heap->{jobs}->{$socket}->{port}, |
|
142
|
|
|
|
|
|
|
$heap->{jobs}->{$socket}->{cmd}, |
|
143
|
|
|
|
|
|
|
$heap->{jobs}->{$socket}->{identifier}, |
|
144
|
|
|
|
|
|
|
'ERROR: No challenge receieved from server.'); |
|
145
|
0
|
|
|
|
|
|
delete($heap->{jobs}->{$socket}); |
|
146
|
|
|
|
|
|
|
} |
|
147
|
|
|
|
|
|
|
} else { |
|
148
|
0
|
|
|
|
|
|
$kernel->select_read($socket, 'got_rcon_response'); |
|
149
|
0
|
|
|
|
|
|
send($socket, "\xFF\xFF\xFF\xFFrcon \"" . $heap->{jobs}->{$socket}->{pw} . "\" " . $heap->{jobs}->{$socket}->{cmd} . "\0", 0); |
|
150
|
0
|
|
|
|
|
|
$heap->{jobs}->{$socket}->{timer} = $kernel->delay_set('rcon_timeout', $heap->{timeout}, $socket); |
|
151
|
0
|
|
|
|
|
|
print STDERR "Got socket and sent rcon command\n" if DEBUG; |
|
152
|
|
|
|
|
|
|
} |
|
153
|
|
|
|
|
|
|
} |
|
154
|
|
|
|
|
|
|
|
|
155
|
|
|
|
|
|
|
sub got_rcon_response { |
|
156
|
0
|
|
|
0
|
0
|
|
my ($kernel, $heap, $socket) = @_[KERNEL, HEAP, ARG0]; |
|
157
|
|
|
|
|
|
|
|
|
158
|
0
|
|
|
|
|
|
$kernel->select_read($socket); |
|
159
|
0
|
0
|
|
|
|
|
$kernel->alarm_remove($heap->{jobs}->{$socket}->{timer}) if defined $heap->{jobs}->{$socket}->{timer}; |
|
160
|
0
|
|
|
|
|
|
delete $heap->{jobs}->{$socket}->{timer}; |
|
161
|
0
|
|
|
|
|
|
my $rsock = recv($socket, my $response = '', $heap->{bytes}, 0); |
|
162
|
|
|
|
|
|
|
|
|
163
|
0
|
0
|
|
|
|
|
if($response =~ /bad (?:rconpassword|rcon_password)/i) { |
|
164
|
0
|
|
|
|
|
|
$kernel->post($heap->{jobs}->{$socket}->{session}, $heap->{jobs}->{$socket}->{postback}, |
|
165
|
|
|
|
|
|
|
$heap->{jobs}->{$socket}->{type}, |
|
166
|
|
|
|
|
|
|
$heap->{jobs}->{$socket}->{ip}, |
|
167
|
|
|
|
|
|
|
$heap->{jobs}->{$socket}->{port}, |
|
168
|
|
|
|
|
|
|
$heap->{jobs}->{$socket}->{cmd}, |
|
169
|
|
|
|
|
|
|
$heap->{jobs}->{$socket}->{identifier}, |
|
170
|
|
|
|
|
|
|
'ERROR: Bad Rcon password.'); |
|
171
|
|
|
|
|
|
|
} else { |
|
172
|
|
|
|
|
|
|
# following regex's thanks to kkrcon |
|
173
|
0
|
|
|
|
|
|
$response =~ s/\x00+$//; # terminator |
|
174
|
0
|
|
|
|
|
|
$response =~ s/^\xff\xff\xff\xffl//; # new HL |
|
175
|
0
|
|
|
|
|
|
$response =~ s/^\xff\xff\xff\xffn//; # qw |
|
176
|
0
|
|
|
|
|
|
$response =~ s/^\xff\xff\xff\xff//; # q2/q3 |
|
177
|
0
|
|
|
|
|
|
$response =~ s/^\xfe\xff\xff\xff.....//; # old hl bug |
|
178
|
0
|
|
|
|
|
|
$kernel->post($heap->{jobs}->{$socket}->{session}, $heap->{jobs}->{$socket}->{postback}, |
|
179
|
|
|
|
|
|
|
$heap->{jobs}->{$socket}->{type}, |
|
180
|
|
|
|
|
|
|
$heap->{jobs}->{$socket}->{ip}, |
|
181
|
|
|
|
|
|
|
$heap->{jobs}->{$socket}->{port}, |
|
182
|
|
|
|
|
|
|
$heap->{jobs}->{$socket}->{cmd}, |
|
183
|
|
|
|
|
|
|
$heap->{jobs}->{$socket}->{identifier}, |
|
184
|
|
|
|
|
|
|
$response); |
|
185
|
0
|
|
|
|
|
|
print STDERR "Rcon Response was $response\n" if DEBUG; |
|
186
|
|
|
|
|
|
|
} |
|
187
|
0
|
|
|
|
|
|
delete($heap->{jobs}->{$socket}); |
|
188
|
|
|
|
|
|
|
} |
|
189
|
|
|
|
|
|
|
|
|
190
|
|
|
|
|
|
|
sub challenge_timeout { |
|
191
|
0
|
|
|
0
|
0
|
|
my ($kernel, $heap, $socket) = @_[KERNEL, HEAP, ARG0]; |
|
192
|
0
|
0
|
|
|
|
|
if($heap->{jobs}->{$socket}->{try} > ($heap->{retry} + 1)) { |
|
193
|
0
|
|
|
|
|
|
$kernel->post($heap->{jobs}->{$socket}->{session}, $heap->{jobs}->{$socket}->{postback}, |
|
194
|
|
|
|
|
|
|
$heap->{jobs}->{$socket}->{type}, |
|
195
|
|
|
|
|
|
|
$heap->{jobs}->{$socket}->{ip}, |
|
196
|
|
|
|
|
|
|
$heap->{jobs}->{$socket}->{port}, |
|
197
|
|
|
|
|
|
|
$heap->{jobs}->{$socket}->{cmd}, |
|
198
|
|
|
|
|
|
|
$heap->{jobs}->{$socket}->{identifier}, |
|
199
|
|
|
|
|
|
|
'ERROR: Timed out trying to obtain challenge.'); |
|
200
|
|
|
|
|
|
|
} else { |
|
201
|
0
|
|
|
|
|
|
print STDERR "Challenge request timed out for $socket. Retrying.\n" if DEBUG; |
|
202
|
0
|
|
|
|
|
|
send($socket, "\xFF\xFF\xFF\xFFchallenge rcon\n\0", 0); |
|
203
|
0
|
|
|
|
|
|
$heap->{jobs}->{$socket}->{timer} = $kernel->delay_set('challenge_timeout', $heap->{timeout}, $socket); |
|
204
|
0
|
|
|
|
|
|
$heap->{jobs}->{$socket}->{try}++; |
|
205
|
|
|
|
|
|
|
} |
|
206
|
|
|
|
|
|
|
} |
|
207
|
|
|
|
|
|
|
|
|
208
|
|
|
|
|
|
|
sub rcon_timeout { |
|
209
|
0
|
|
|
0
|
0
|
|
my ($kernel, $heap, $socket) = @_[KERNEL, HEAP, ARG0]; |
|
210
|
0
|
0
|
|
|
|
|
if($heap->{jobs}->{$socket}->{try} > ($heap->{retry} + 1)) { |
|
211
|
0
|
|
|
|
|
|
$kernel->post($heap->{jobs}->{$socket}->{session}, $heap->{jobs}->{$socket}->{postback}, |
|
212
|
|
|
|
|
|
|
$heap->{jobs}->{$socket}->{type}, |
|
213
|
|
|
|
|
|
|
$heap->{jobs}->{$socket}->{ip}, |
|
214
|
|
|
|
|
|
|
$heap->{jobs}->{$socket}->{port}, |
|
215
|
|
|
|
|
|
|
$heap->{jobs}->{$socket}->{cmd}, |
|
216
|
|
|
|
|
|
|
$heap->{jobs}->{$socket}->{identifier}, |
|
217
|
|
|
|
|
|
|
'ERROR: Timed out waiting for Rcon response.'); |
|
218
|
|
|
|
|
|
|
} else { |
|
219
|
0
|
|
|
|
|
|
print STDERR "Rcon timed out for $socket. Retrying.\n" if DEBUG; |
|
220
|
0
|
0
|
|
|
|
|
send($socket, "\xFF\xFF\xFF\xFFrcon " . $heap->{jobs}->{$socket}->{challenge} . " \"" . $heap->{jobs}->{$socket}->{pw} . "\" " . $heap->{jobs}->{$socket}->{cmd} . "\0", 0) if $heap->{jobs}->{$socket}->{type} =~ /hl$/; |
|
221
|
0
|
0
|
|
|
|
|
send($socket, "\xFF\xFF\xFF\xFFrcon \"" . $heap->{jobs}->{$socket}->{pw} . "\" " . $heap->{jobs}->{$socket}->{cmd} . "\0", 0) if $heap->{jobs}->{$socket}->{type} =~ /^(?:q2|q3|qw)$/; |
|
222
|
0
|
|
|
|
|
|
$heap->{jobs}->{$socket}->{timer} = $kernel->delay_set('rcon_timeout', $heap->{timeout}, $socket); |
|
223
|
0
|
|
|
|
|
|
$heap->{jobs}->{$socket}->{try}++; |
|
224
|
|
|
|
|
|
|
} |
|
225
|
|
|
|
|
|
|
} |
|
226
|
|
|
|
|
|
|
|
|
227
|
|
|
|
|
|
|
sub players { |
|
228
|
0
|
|
|
0
|
1
|
|
my ($kernel, $heap, $sender, $type, $ip, $port, $password, $postback) = @_[KERNEL, HEAP, SENDER, ARG0..ARG4]; |
|
229
|
0
|
0
|
|
|
|
|
my $identifier = defined($_[ARG5]) ? $_[ARG5] : undef; |
|
230
|
0
|
0
|
|
|
|
|
croak "IP address required to execute an Rcon command" unless defined $ip; |
|
231
|
0
|
0
|
0
|
|
|
|
croak "Port requred to execute an Rcon command" if !defined $port || $port !~ /^\d+$/; |
|
232
|
0
|
0
|
0
|
|
|
|
croak "Password requires to execute an Rcon command" if !defined $password || $password eq ''; |
|
233
|
0
|
0
|
|
|
|
|
croak "Server type was not recognized" unless $type =~ /^(?:qw|q2|q3|oldhl|hl)$/; |
|
234
|
0
|
|
|
|
|
|
my $jobid = $playerSN; |
|
235
|
0
|
|
|
|
|
|
$playerSN++; |
|
236
|
0
|
|
|
|
|
|
print STDERR "Got a request for players at $ip:$port with jobid $jobid\n" if DEBUG; |
|
237
|
0
|
|
|
|
|
|
$kernel->yield('rcon', $type, $ip, $port, $password, 'status', 'player_response', $jobid); |
|
238
|
0
|
|
|
|
|
|
$heap->{p_jobs}->{$jobid} = { |
|
239
|
|
|
|
|
|
|
ip => $ip, |
|
240
|
|
|
|
|
|
|
port => $port, |
|
241
|
|
|
|
|
|
|
pw => $password, |
|
242
|
|
|
|
|
|
|
identifier => $identifier, |
|
243
|
|
|
|
|
|
|
session => $sender->ID(), |
|
244
|
|
|
|
|
|
|
postback => $postback, |
|
245
|
|
|
|
|
|
|
type => $type, |
|
246
|
|
|
|
|
|
|
}; |
|
247
|
|
|
|
|
|
|
} |
|
248
|
|
|
|
|
|
|
|
|
249
|
|
|
|
|
|
|
sub player_response { |
|
250
|
0
|
|
|
0
|
0
|
|
my ($kernel, $heap, $jobid, $response) = @_[KERNEL, HEAP, ARG4, ARG5]; |
|
251
|
0
|
|
|
|
|
|
print STDERR "Got a player request response for job $jobid\n" if DEBUG; |
|
252
|
0
|
0
|
|
|
|
|
if($response =~ /^ERROR\: /) { |
|
253
|
0
|
|
|
|
|
|
$kernel->post($heap->{p_jobs}->{$jobid}->{session}, |
|
254
|
|
|
|
|
|
|
$heap->{p_jobs}->{$jobid}->{postback}, |
|
255
|
|
|
|
|
|
|
$heap->{p_jobs}->{$jobid}->{type}, |
|
256
|
|
|
|
|
|
|
$heap->{p_jobs}->{$jobid}->{ip}, |
|
257
|
|
|
|
|
|
|
$heap->{p_jobs}->{$jobid}->{port}, |
|
258
|
|
|
|
|
|
|
$heap->{p_jobs}->{$jobid}->{identifier}, |
|
259
|
|
|
|
|
|
|
$response |
|
260
|
|
|
|
|
|
|
); # One of the errors generated from the rcon command... |
|
261
|
|
|
|
|
|
|
} else { |
|
262
|
0
|
0
|
|
|
|
|
if($heap->{p_jobs}->{$jobid}->{type} =~ /hl$/) { |
|
|
|
0
|
|
|
|
|
|
|
263
|
0
|
|
|
|
|
|
$kernel->yield('player_parse_hl', $jobid, $response); |
|
264
|
|
|
|
|
|
|
} elsif($heap->{p_jobs}->{$jobid}->{type} =~ /^(?:q2|q3|qw)$/) { |
|
265
|
0
|
|
|
|
|
|
$kernel->yield('player_parse_quake', $jobid, $response); |
|
266
|
|
|
|
|
|
|
} |
|
267
|
|
|
|
|
|
|
} |
|
268
|
|
|
|
|
|
|
} |
|
269
|
|
|
|
|
|
|
|
|
270
|
|
|
|
|
|
|
sub player_parse_hl { |
|
271
|
0
|
|
|
0
|
0
|
|
my ($kernel, $heap, $jobid, $response) = @_[KERNEL, HEAP, ARG0, ARG1]; |
|
272
|
|
|
|
|
|
|
# This code is partially adapted from KKrcon |
|
273
|
0
|
|
|
|
|
|
my %players; |
|
274
|
0
|
|
|
|
|
|
foreach(split(/[\r\n]+/, $response)) { |
|
275
|
0
|
0
|
|
|
|
|
if(/^\#[\s\d]\d\s+ |
|
276
|
|
|
|
|
|
|
(?:\")(.+)(?:\")\s+ # Player name |
|
277
|
|
|
|
|
|
|
(\d+)\s+ # Player ID |
|
278
|
|
|
|
|
|
|
(\d+)\s+ # WonID |
|
279
|
|
|
|
|
|
|
([\d-]+)\s+ # Frag count |
|
280
|
|
|
|
|
|
|
([\d:]+)\s+ # time |
|
281
|
|
|
|
|
|
|
(\d+)\s+ # ping |
|
282
|
|
|
|
|
|
|
(\d+)\s+ # packetloss |
|
283
|
|
|
|
|
|
|
(\S+) # ip:port |
|
284
|
|
|
|
|
|
|
$/x) { |
|
285
|
0
|
|
|
|
|
|
$players{$2} = { |
|
286
|
|
|
|
|
|
|
"Name" => $1, |
|
287
|
|
|
|
|
|
|
"UserID" => $2, |
|
288
|
|
|
|
|
|
|
"WonID" => $3, |
|
289
|
|
|
|
|
|
|
"Frags" => $4, |
|
290
|
|
|
|
|
|
|
"Time" => $5, |
|
291
|
|
|
|
|
|
|
"Ping" => $6, |
|
292
|
|
|
|
|
|
|
"Loss" => $7, |
|
293
|
|
|
|
|
|
|
"Address" => $8, |
|
294
|
|
|
|
|
|
|
}; |
|
295
|
|
|
|
|
|
|
} |
|
296
|
|
|
|
|
|
|
} |
|
297
|
0
|
|
|
|
|
|
$kernel->post($heap->{p_jobs}->{$jobid}->{session}, |
|
298
|
|
|
|
|
|
|
$heap->{p_jobs}->{$jobid}->{postback}, |
|
299
|
|
|
|
|
|
|
$heap->{p_jobs}->{$jobid}->{type}, |
|
300
|
|
|
|
|
|
|
$heap->{p_jobs}->{$jobid}->{ip}, |
|
301
|
|
|
|
|
|
|
$heap->{p_jobs}->{$jobid}->{port}, |
|
302
|
|
|
|
|
|
|
$heap->{p_jobs}->{$jobid}->{identifier}, |
|
303
|
|
|
|
|
|
|
\%players |
|
304
|
|
|
|
|
|
|
); |
|
305
|
0
|
|
|
|
|
|
delete($heap->{p_jobs}->{$jobid}); |
|
306
|
|
|
|
|
|
|
} |
|
307
|
|
|
|
|
|
|
|
|
308
|
|
|
|
|
|
|
sub player_parse_quake { |
|
309
|
0
|
|
|
0
|
0
|
|
my ($kernel, $heap, $jobid, $response) = @_[KERNEL, HEAP, ARG0, ARG1]; |
|
310
|
0
|
|
|
|
|
|
my %players; |
|
311
|
0
|
|
|
|
|
|
foreach(split(/[\r\n]+/, $response)) { |
|
312
|
0
|
0
|
|
|
|
|
if(/^\s* |
|
313
|
|
|
|
|
|
|
(\d+)\s+ # num |
|
314
|
|
|
|
|
|
|
([\d-]+)\s+ # score |
|
315
|
|
|
|
|
|
|
(\d+)\s+ # ping |
|
316
|
|
|
|
|
|
|
(.+) # name |
|
317
|
|
|
|
|
|
|
\s+(\d+)\s+ # lastmsg |
|
318
|
|
|
|
|
|
|
(\S+)\s+ # address |
|
319
|
|
|
|
|
|
|
(\d+) # qport |
|
320
|
|
|
|
|
|
|
(?:\s+(\d+)|) # rate |
|
321
|
|
|
|
|
|
|
$/x) { |
|
322
|
0
|
|
|
|
|
|
$players{$1} = { |
|
323
|
|
|
|
|
|
|
"num" => $1, |
|
324
|
|
|
|
|
|
|
"score" => $2, |
|
325
|
|
|
|
|
|
|
"ping" => $3, |
|
326
|
|
|
|
|
|
|
"name" => $4, |
|
327
|
|
|
|
|
|
|
"lastmsg" => $5, |
|
328
|
|
|
|
|
|
|
"address" => $6, |
|
329
|
|
|
|
|
|
|
"qport" => $7, |
|
330
|
|
|
|
|
|
|
}; |
|
331
|
0
|
0
|
0
|
|
|
|
if(defined($8) && $8 ne '') { |
|
332
|
0
|
|
|
|
|
|
$players{$1}{"rate"} = $8; |
|
333
|
|
|
|
|
|
|
} |
|
334
|
|
|
|
|
|
|
} |
|
335
|
|
|
|
|
|
|
} |
|
336
|
0
|
|
|
|
|
|
$kernel->post($heap->{p_jobs}->{$jobid}->{session}, |
|
337
|
|
|
|
|
|
|
$heap->{p_jobs}->{$jobid}->{postback}, |
|
338
|
|
|
|
|
|
|
$heap->{p_jobs}->{$jobid}->{type}, |
|
339
|
|
|
|
|
|
|
$heap->{p_jobs}->{$jobid}->{ip}, |
|
340
|
|
|
|
|
|
|
$heap->{p_jobs}->{$jobid}->{port}, |
|
341
|
|
|
|
|
|
|
$heap->{p_jobs}->{$jobid}->{identifier}, |
|
342
|
|
|
|
|
|
|
\%players |
|
343
|
|
|
|
|
|
|
); |
|
344
|
0
|
|
|
|
|
|
delete($heap->{p_jobs}->{$jobid}); |
|
345
|
|
|
|
|
|
|
} |
|
346
|
|
|
|
|
|
|
1; |
|
347
|
|
|
|
|
|
|
|
|
348
|
|
|
|
|
|
|
__END__ |