line |
stmt |
bran |
cond |
sub |
pod |
time |
code |
1
|
|
|
|
|
|
|
package POE::Component::IRC::Plugin::BotCommand; |
2
|
|
|
|
|
|
|
our $AUTHORITY = 'cpan:HINRIK'; |
3
|
|
|
|
|
|
|
# vim: set expandtab ts=4 sw=4 ai: |
4
|
|
|
|
|
|
|
$POE::Component::IRC::Plugin::BotCommand::VERSION = '6.92'; |
5
|
9
|
|
|
9
|
|
8895
|
use strict; |
|
9
|
|
|
|
|
30
|
|
|
9
|
|
|
|
|
361
|
|
6
|
9
|
|
|
9
|
|
58
|
use warnings FATAL => 'all'; |
|
9
|
|
|
|
|
24
|
|
|
9
|
|
|
|
|
572
|
|
7
|
9
|
|
|
9
|
|
66
|
use Carp; |
|
9
|
|
|
|
|
23
|
|
|
9
|
|
|
|
|
821
|
|
8
|
9
|
|
|
9
|
|
70
|
use IRC::Utils qw( parse_user strip_color strip_formatting ); |
|
9
|
|
|
|
|
28
|
|
|
9
|
|
|
|
|
1093
|
|
9
|
9
|
|
|
9
|
|
282
|
use POE::Component::IRC::Plugin qw( :ALL ); |
|
9
|
|
|
|
|
39
|
|
|
9
|
|
|
|
|
14050
|
|
10
|
|
|
|
|
|
|
|
11
|
|
|
|
|
|
|
sub new { |
12
|
8
|
|
|
8
|
1
|
5843
|
my ($package) = shift; |
13
|
8
|
50
|
|
|
|
45
|
croak "$package requires an even number of arguments" if @_ & 1; |
14
|
8
|
|
|
|
|
39
|
my %args = @_; |
15
|
|
|
|
|
|
|
|
16
|
8
|
50
|
|
|
|
48
|
$args{Method} = 'notice' if !defined $args{Method}; |
17
|
|
|
|
|
|
|
|
18
|
8
|
|
|
|
|
20
|
for my $cmd (keys %{ $args{Commands} }) { |
|
8
|
|
|
|
|
44
|
|
19
|
11
|
100
|
|
|
|
54
|
if (ref $args{Commands}->{$cmd} eq 'HASH') { |
20
|
|
|
|
|
|
|
croak "$cmd: no info provided" |
21
|
1
|
50
|
|
|
|
6
|
if !exists $args{Commands}->{$cmd}->{info} ; |
22
|
|
|
|
|
|
|
$args{Commands}->{lc $cmd}->{handler} = |
23
|
|
|
|
|
|
|
sprintf("irc_botcmd_%s", lc($cmd)) |
24
|
1
|
50
|
|
|
|
12
|
if !$args{Commands}->{lc $cmd}->{handler}; |
25
|
|
|
|
|
|
|
} |
26
|
11
|
|
|
|
|
42
|
$args{Commands}->{lc $cmd} = delete $args{Commands}->{$cmd}; |
27
|
|
|
|
|
|
|
} |
28
|
8
|
|
|
|
|
52
|
return bless \%args, $package; |
29
|
|
|
|
|
|
|
} |
30
|
|
|
|
|
|
|
|
31
|
|
|
|
|
|
|
sub PCI_register { |
32
|
8
|
|
|
8
|
0
|
1291
|
my ($self, $irc) = splice @_, 0, 2; |
33
|
|
|
|
|
|
|
|
34
|
8
|
100
|
|
|
|
62
|
$self->{Addressed} = 1 if !defined $self->{Addressed}; |
35
|
8
|
100
|
|
|
|
66
|
$self->{Prefix} = '!' if !defined $self->{Prefix}; |
36
|
8
|
50
|
|
|
|
49
|
$self->{In_channels} = 1 if !defined $self->{In_channels}; |
37
|
8
|
50
|
|
|
|
37
|
$self->{In_private} = 1 if !defined $self->{In_private}; |
38
|
8
|
|
|
|
|
57
|
$self->{rx_cmd_args} = qr/^(\S+)(?:\s+(.+))?$/; |
39
|
8
|
|
|
|
|
31
|
$self->{irc} = $irc; |
40
|
|
|
|
|
|
|
|
41
|
8
|
|
|
|
|
96
|
$irc->plugin_register( $self, 'SERVER', qw(msg public) ); |
42
|
8
|
|
|
|
|
392
|
return 1; |
43
|
|
|
|
|
|
|
} |
44
|
|
|
|
|
|
|
|
45
|
|
|
|
|
|
|
sub PCI_unregister { |
46
|
8
|
|
|
8
|
0
|
2075
|
return 1; |
47
|
|
|
|
|
|
|
} |
48
|
|
|
|
|
|
|
|
49
|
|
|
|
|
|
|
sub S_msg { |
50
|
2
|
|
|
2
|
0
|
165
|
my ($self, $irc) = splice @_, 0, 2; |
51
|
2
|
|
|
|
|
9
|
my $who = ${ $_[0] }; |
|
2
|
|
|
|
|
9
|
|
52
|
2
|
|
|
|
|
15
|
my $where = parse_user($who); |
53
|
2
|
|
|
|
|
72
|
my $what = ${ $_[2] }; |
|
2
|
|
|
|
|
10
|
|
54
|
|
|
|
|
|
|
|
55
|
2
|
50
|
|
|
|
13
|
return PCI_EAT_NONE if !$self->{In_private}; |
56
|
2
|
|
|
|
|
11
|
$what = $self->_normalize($what); |
57
|
|
|
|
|
|
|
|
58
|
2
|
50
|
|
|
|
12
|
if (!$self->{Bare_private}) { |
59
|
0
|
0
|
|
|
|
0
|
return PCI_EAT_NONE if $what !~ s/^\Q$self->{Prefix}\E//; |
60
|
|
|
|
|
|
|
} |
61
|
|
|
|
|
|
|
|
62
|
2
|
|
|
|
|
9
|
my ($cmd, $args); |
63
|
2
|
50
|
|
|
|
35
|
if (!(($cmd, $args) = $what =~ $self->{rx_cmd_args})) { |
64
|
0
|
|
|
|
|
0
|
return PCI_EAT_NONE; |
65
|
|
|
|
|
|
|
} |
66
|
|
|
|
|
|
|
|
67
|
2
|
|
|
|
|
16
|
$self->_handle_cmd($who, $where, $cmd, $args); |
68
|
2
|
50
|
|
|
|
16
|
return $self->{Eat} ? PCI_EAT_PLUGIN : PCI_EAT_NONE; |
69
|
|
|
|
|
|
|
} |
70
|
|
|
|
|
|
|
|
71
|
|
|
|
|
|
|
sub S_public { |
72
|
16
|
|
|
16
|
0
|
924
|
my ($self, $irc) = splice @_, 0, 2; |
73
|
16
|
|
|
|
|
42
|
my $who = ${ $_[0] }; |
|
16
|
|
|
|
|
45
|
|
74
|
16
|
|
|
|
|
30
|
my $where = ${ $_[1] }->[0]; |
|
16
|
|
|
|
|
57
|
|
75
|
16
|
|
|
|
|
33
|
my $what = ${ $_[2] }; |
|
16
|
|
|
|
|
39
|
|
76
|
16
|
|
|
|
|
81
|
my $me = $irc->nick_name(); |
77
|
|
|
|
|
|
|
|
78
|
16
|
50
|
|
|
|
80
|
return PCI_EAT_NONE if !$self->{In_channels}; |
79
|
16
|
|
|
|
|
64
|
$what = $self->_normalize($what); |
80
|
|
|
|
|
|
|
|
81
|
16
|
100
|
|
|
|
63
|
if ($self->{Addressed}) { |
82
|
10
|
50
|
|
|
|
249
|
return PCI_EAT_NONE if !(($what) = $what =~ m/^\s*\Q$me\E[:,;.!?~]?\s*(.*)$/); |
83
|
|
|
|
|
|
|
} |
84
|
|
|
|
|
|
|
else { |
85
|
6
|
50
|
|
|
|
101
|
return PCI_EAT_NONE if $what !~ s/^\Q$self->{Prefix}\E//; |
86
|
|
|
|
|
|
|
} |
87
|
|
|
|
|
|
|
|
88
|
16
|
|
|
|
|
53
|
my ($cmd, $args); |
89
|
16
|
50
|
|
|
|
159
|
if (!(($cmd, $args) = $what =~ $self->{rx_cmd_args})) { |
90
|
0
|
|
|
|
|
0
|
return PCI_EAT_NONE; |
91
|
|
|
|
|
|
|
} |
92
|
|
|
|
|
|
|
|
93
|
16
|
|
|
|
|
125
|
$self->_handle_cmd($who, $where, $cmd, $args); |
94
|
16
|
100
|
|
|
|
85
|
return $self->{Eat} ? PCI_EAT_PLUGIN : PCI_EAT_NONE; |
95
|
|
|
|
|
|
|
} |
96
|
|
|
|
|
|
|
|
97
|
|
|
|
|
|
|
sub _normalize { |
98
|
18
|
|
|
18
|
|
106
|
my ($self, $line) = @_; |
99
|
18
|
|
|
|
|
89
|
$line = strip_color($line); |
100
|
18
|
|
|
|
|
643
|
$line = strip_formatting($line); |
101
|
18
|
|
|
|
|
397
|
return $line; |
102
|
|
|
|
|
|
|
} |
103
|
|
|
|
|
|
|
|
104
|
|
|
|
|
|
|
sub _handle_cmd { |
105
|
18
|
|
|
18
|
|
80
|
my ($self, $who, $where, $cmd, $args) = @_; |
106
|
18
|
|
|
|
|
52
|
my $irc = $self->{irc}; |
107
|
18
|
50
|
|
|
|
45
|
my $chantypes = join('', @{ $irc->isupport('CHANTYPES') || ['#', '&']}); |
|
18
|
|
|
|
|
126
|
|
108
|
18
|
100
|
|
|
|
228
|
my $public = $where =~ /^[$chantypes]/ ? 1 : 0; |
109
|
18
|
|
|
|
|
65
|
$cmd = lc $cmd; |
110
|
|
|
|
|
|
|
|
111
|
18
|
|
|
|
|
42
|
my $cmd_unresolved = $cmd; |
112
|
|
|
|
|
|
|
|
113
|
18
|
50
|
|
|
|
92
|
if((my $cmd_resolved = $self->resolve_alias($cmd))) |
114
|
|
|
|
|
|
|
{ |
115
|
0
|
|
|
|
|
0
|
$cmd = $cmd_resolved; |
116
|
|
|
|
|
|
|
} |
117
|
|
|
|
|
|
|
|
118
|
|
|
|
|
|
|
|
119
|
18
|
100
|
|
|
|
153
|
if (defined $self->{Commands}->{$cmd}) { |
120
|
11
|
100
|
|
|
|
69
|
if (ref $self->{Commands}->{$cmd} eq 'HASH') { |
121
|
1
|
50
|
|
|
|
10
|
my @args_array = defined $args ? split /\s+/, $args : (); |
122
|
1
|
50
|
33
|
|
|
12
|
if (defined($self->{Commands}->{$cmd}->{args}) && |
|
|
|
33
|
|
|
|
|
|
|
|
33
|
|
|
|
|
|
|
|
33
|
|
|
|
|
123
|
|
|
|
|
|
|
ref($self->{Commands}->{$cmd}->{args}) eq 'ARRAY' && |
124
|
1
|
|
|
|
|
9
|
@{ $self->{Commands}->{$cmd}->{args} } && |
125
|
|
|
|
|
|
|
(@args_array < @{ $self->{Commands}->{$cmd}->{args} } || |
126
|
|
|
|
|
|
|
(!defined $self->{Commands}->{$cmd}->{variable} && |
127
|
|
|
|
|
|
|
@args_array > @{ $self->{Commands}->{$cmd}->{args} })) |
128
|
|
|
|
|
|
|
) { |
129
|
0
|
|
|
|
|
0
|
$irc->yield($self->{Method}, $where, |
130
|
|
|
|
|
|
|
"Not enough or too many arguments. See help for $cmd"); |
131
|
0
|
|
|
|
|
0
|
return; |
132
|
|
|
|
|
|
|
} |
133
|
|
|
|
|
|
|
|
134
|
1
|
50
|
0
|
|
|
6
|
if(defined $self->{Commands}->{$cmd}->{variable} || |
|
|
|
0
|
|
|
|
|
|
|
|
33
|
|
|
|
|
135
|
|
|
|
|
|
|
(defined($self->{Commands}->{$cmd}->{args}) && |
136
|
|
|
|
|
|
|
ref($self->{Commands}->{$cmd}->{args}) eq 'ARRAY' && |
137
|
0
|
|
|
|
|
0
|
@{ $self->{Commands}->{$cmd}->{args} })) |
138
|
|
|
|
|
|
|
{ |
139
|
1
|
|
|
|
|
3
|
$args = {}; |
140
|
1
|
50
|
33
|
|
|
9
|
if( defined($self->{Commands}->{$cmd}->{args}) && |
|
|
|
33
|
|
|
|
|
141
|
|
|
|
|
|
|
ref($self->{Commands}->{$cmd}->{args}) eq 'ARRAY' && |
142
|
1
|
|
|
|
|
5
|
@{ $self->{Commands}->{$cmd}->{args} }) |
143
|
|
|
|
|
|
|
{ |
144
|
1
|
|
|
|
|
2
|
for (@{ $self->{Commands}->{$cmd}->{args} }) { |
|
1
|
|
|
|
|
6
|
|
145
|
2
|
|
|
|
|
5
|
my $in_arg = shift @args_array; |
146
|
2
|
100
|
|
|
|
8
|
if (ref $self->{Commands}->{$cmd}->{$_} eq 'ARRAY') { |
147
|
1
|
|
|
|
|
2
|
my @values = @{ $self->{Commands}->{$cmd}->{$_} }; |
|
1
|
|
|
|
|
4
|
|
148
|
1
|
|
|
|
|
3
|
shift @values; |
149
|
|
|
|
|
|
|
|
150
|
9
|
|
|
9
|
|
99
|
use List::Util qw(none); |
|
9
|
|
|
|
|
30
|
|
|
9
|
|
|
|
|
17798
|
|
151
|
|
|
|
|
|
|
# Check if argument has one of possible values |
152
|
1
|
50
|
|
1
|
|
10
|
if (none { $_ eq $in_arg} @values) { |
|
1
|
|
|
|
|
5
|
|
153
|
0
|
|
|
|
|
0
|
$irc->yield($self->{Method}, $where, |
154
|
|
|
|
|
|
|
"$_ can be one of ".join '|', @values); |
155
|
0
|
|
|
|
|
0
|
return; |
156
|
|
|
|
|
|
|
} |
157
|
|
|
|
|
|
|
|
158
|
|
|
|
|
|
|
} |
159
|
2
|
|
|
|
|
8
|
$args->{$_} = $in_arg; |
160
|
|
|
|
|
|
|
} |
161
|
|
|
|
|
|
|
} |
162
|
|
|
|
|
|
|
|
163
|
|
|
|
|
|
|
# Process remaining arguments if variable is set |
164
|
1
|
|
|
|
|
3
|
my $arg_cnt = 0; |
165
|
1
|
50
|
|
|
|
5
|
if (defined $self->{Commands}->{$cmd}->{variable}) { |
166
|
1
|
|
|
|
|
3
|
for (@args_array) { |
167
|
1
|
|
|
|
|
6
|
$args->{"opt".$arg_cnt++} = $_; |
168
|
|
|
|
|
|
|
} |
169
|
|
|
|
|
|
|
} |
170
|
|
|
|
|
|
|
} |
171
|
|
|
|
|
|
|
} |
172
|
|
|
|
|
|
|
} |
173
|
|
|
|
|
|
|
|
174
|
18
|
100
|
|
|
|
78
|
if (ref $self->{Auth_sub} eq 'CODE') { |
175
|
3
|
|
|
|
|
17
|
my ($authed, $errors) = $self->{Auth_sub}->($self->{irc}, $who, $where, $cmd, $args, $cmd_unresolved); |
176
|
|
|
|
|
|
|
|
177
|
3
|
100
|
|
|
|
37
|
if (!$authed) { |
178
|
1
|
50
|
|
|
|
6
|
my @errors = ref $errors eq 'ARRAY' |
179
|
|
|
|
|
|
|
? @$errors |
180
|
|
|
|
|
|
|
: 'You are not authorized to use this command.'; |
181
|
1
|
50
|
|
|
|
3
|
if (!$self->{Ignore_unauthorized}) { |
182
|
1
|
|
|
|
|
3
|
for my $error (@errors) { |
183
|
1
|
|
|
|
|
4
|
$irc->yield($self->{Method}, $where, $error); |
184
|
|
|
|
|
|
|
} |
185
|
|
|
|
|
|
|
} |
186
|
1
|
|
|
|
|
107
|
return; |
187
|
|
|
|
|
|
|
} |
188
|
|
|
|
|
|
|
} |
189
|
|
|
|
|
|
|
|
190
|
17
|
100
|
|
|
|
86
|
if (defined $self->{Commands}->{$cmd}) { |
|
|
50
|
|
|
|
|
|
|
|
0
|
|
|
|
|
|
191
|
11
|
100
|
|
|
|
72
|
my $handler = (ref($self->{Commands}->{$cmd}) eq 'HASH' ? $self->{Commands}->{$cmd}->{handler} : "irc_botcmd_$cmd"); |
192
|
11
|
|
|
|
|
126
|
$irc->send_event_next($handler => $who, $where, $args, $cmd, $cmd_unresolved); |
193
|
|
|
|
|
|
|
} |
194
|
|
|
|
|
|
|
elsif ($cmd =~ /^help$/i) { |
195
|
6
|
|
|
|
|
28
|
my @help = $self->_get_help($args, $public); |
196
|
6
|
|
|
|
|
38
|
$irc->yield($self->{Method} => $where => $_) for @help; |
197
|
|
|
|
|
|
|
} |
198
|
|
|
|
|
|
|
elsif (!$self->{Ignore_unknown}) { |
199
|
0
|
|
|
|
|
0
|
my @help = $self->_get_help($cmd, $public); |
200
|
0
|
|
|
|
|
0
|
$irc->yield($self->{Method} => $where => $_) for @help; |
201
|
|
|
|
|
|
|
} |
202
|
|
|
|
|
|
|
|
203
|
17
|
|
|
|
|
2096
|
return; |
204
|
|
|
|
|
|
|
} |
205
|
|
|
|
|
|
|
|
206
|
|
|
|
|
|
|
sub _get_help { |
207
|
6
|
|
|
6
|
|
16
|
my ($self, $args, $public) = @_; |
208
|
6
|
|
|
|
|
13
|
my $irc = $self->{irc}; |
209
|
|
|
|
|
|
|
my $p = $self->{Addressed} && $public |
210
|
|
|
|
|
|
|
? $irc->nick_name().': ' |
211
|
6
|
50
|
33
|
|
|
102
|
: $self->{Prefix}; |
212
|
|
|
|
|
|
|
|
213
|
6
|
|
|
|
|
16
|
my @help; |
214
|
6
|
100
|
|
|
|
19
|
if (defined $args) { |
215
|
3
|
|
|
|
|
18
|
my $cmd = (split /\s+/, $args, 2)[0]; |
216
|
|
|
|
|
|
|
|
217
|
3
|
|
|
|
|
9
|
$cmd = lc $cmd; |
218
|
|
|
|
|
|
|
|
219
|
3
|
|
33
|
|
|
12
|
my $cmd_resolved = $self->resolve_alias($cmd) || $cmd; |
220
|
|
|
|
|
|
|
|
221
|
3
|
100
|
|
|
|
14
|
if (exists $self->{Commands}->{$cmd_resolved}) { |
222
|
1
|
50
|
|
|
|
6
|
if (ref $self->{Commands}->{$cmd_resolved} eq 'HASH') { |
223
|
|
|
|
|
|
|
push @help, "Syntax: $p$cmd". |
224
|
|
|
|
|
|
|
( defined($self->{Commands}->{$cmd_resolved}->{args}) && |
225
|
|
|
|
|
|
|
ref($self->{Commands}->{$cmd_resolved}->{args}) eq 'ARRAY' ? |
226
|
1
|
|
|
|
|
7
|
" ".join ' ', @{ $self->{Commands}->{$cmd_resolved}->{args} } : |
227
|
|
|
|
|
|
|
"" ). |
228
|
|
|
|
|
|
|
(defined $self->{Commands}->{$cmd_resolved}->{variable} ? |
229
|
1
|
50
|
33
|
|
|
11
|
" ..." : ""); |
|
|
50
|
|
|
|
|
|
230
|
|
|
|
|
|
|
push @help, split /\015?\012/, |
231
|
1
|
|
|
|
|
6
|
"Description: ".$self->{Commands}->{$cmd_resolved}->{info}; |
232
|
1
|
50
|
33
|
|
|
14
|
if( defined($self->{Commands}->{$cmd_resolved}->{args}) && |
|
|
|
33
|
|
|
|
|
233
|
|
|
|
|
|
|
ref($self->{Commands}->{$cmd_resolved}->{args}) eq 'ARRAY' && |
234
|
1
|
|
|
|
|
6
|
@{ $self->{Commands}->{$cmd_resolved}->{args} }) |
235
|
|
|
|
|
|
|
{ |
236
|
1
|
|
|
|
|
4
|
push @help, "Arguments:"; |
237
|
|
|
|
|
|
|
|
238
|
1
|
|
|
|
|
2
|
for my $arg (@{ $self->{Commands}->{$cmd_resolved}->{args} }) { |
|
1
|
|
|
|
|
5
|
|
239
|
2
|
50
|
|
|
|
7
|
next if not defined $self->{Commands}->{$cmd_resolved}->{$arg}; |
240
|
2
|
100
|
|
|
|
23
|
if (ref $self->{Commands}->{$cmd_resolved}->{$arg} eq 'ARRAY') { |
241
|
1
|
|
|
|
|
4
|
my @arg_usage = @{$self->{Commands}->{$cmd_resolved}->{$arg}}; |
|
1
|
|
|
|
|
5
|
|
242
|
1
|
|
|
|
|
11
|
push @help, " $arg: ".$arg_usage[0]. |
243
|
|
|
|
|
|
|
" (".(join '|', @arg_usage[1..$#arg_usage]).")" |
244
|
|
|
|
|
|
|
} |
245
|
|
|
|
|
|
|
else { |
246
|
|
|
|
|
|
|
push @help, " $arg: ". |
247
|
1
|
|
|
|
|
6
|
$self->{Commands}->{$cmd_resolved}->{$arg}; |
248
|
|
|
|
|
|
|
} |
249
|
|
|
|
|
|
|
} |
250
|
|
|
|
|
|
|
} |
251
|
|
|
|
|
|
|
|
252
|
|
|
|
|
|
|
push @help, "Alias of: ${p}${cmd_resolved}" . |
253
|
|
|
|
|
|
|
(ref($self->{Commands}->{$cmd_resolved}->{args}) eq 'ARRAY' ? |
254
|
0
|
|
|
|
|
0
|
" ".join ' ', @{ $self->{Commands}->{$cmd_resolved}->{args} } : |
255
|
|
|
|
|
|
|
"" ). |
256
|
|
|
|
|
|
|
(defined $self->{Commands}->{$cmd_resolved}->{variable} ? |
257
|
1
|
0
|
|
|
|
15
|
" ..." : "") |
|
|
0
|
|
|
|
|
|
|
|
50
|
|
|
|
|
|
258
|
|
|
|
|
|
|
if $cmd_resolved ne $cmd; |
259
|
|
|
|
|
|
|
|
260
|
1
|
|
|
|
|
6
|
my @aliases = grep { $_ ne $cmd } $self->list_aliases($cmd_resolved); |
|
0
|
|
|
|
|
0
|
|
261
|
|
|
|
|
|
|
|
262
|
1
|
50
|
|
|
|
4
|
if($cmd_resolved ne $cmd) |
263
|
|
|
|
|
|
|
{ |
264
|
0
|
|
|
|
|
0
|
push @aliases, $cmd_resolved; |
265
|
|
|
|
|
|
|
} |
266
|
|
|
|
|
|
|
|
267
|
1
|
50
|
|
|
|
5
|
push @help, "Aliases: ".join( " ", @aliases) if scalar(@aliases); |
268
|
|
|
|
|
|
|
} |
269
|
|
|
|
|
|
|
else { |
270
|
0
|
|
|
|
|
0
|
@help = split /\015?\012/, $self->{Commands}->{$cmd}; |
271
|
|
|
|
|
|
|
} |
272
|
|
|
|
|
|
|
} |
273
|
|
|
|
|
|
|
else { |
274
|
2
|
|
|
|
|
11
|
push @help, "Unknown command: $cmd"; |
275
|
2
|
|
|
|
|
10
|
push @help, "To get a list of commands, use: ${p}help"; |
276
|
|
|
|
|
|
|
} |
277
|
|
|
|
|
|
|
} |
278
|
|
|
|
|
|
|
else { |
279
|
3
|
100
|
|
|
|
8
|
if (keys %{ $self->{Commands} }) { |
|
3
|
|
|
|
|
13
|
|
280
|
1
|
|
|
|
|
3
|
push @help, 'Commands: ' . join ', ', sort keys %{ $self->{Commands} }; |
|
1
|
|
|
|
|
11
|
|
281
|
1
|
|
|
|
|
4
|
push @help, "For more details, use: ${p}help "; |
282
|
|
|
|
|
|
|
} |
283
|
|
|
|
|
|
|
else { |
284
|
2
|
|
|
|
|
8
|
push @help, 'No commands are defined'; |
285
|
|
|
|
|
|
|
} |
286
|
|
|
|
|
|
|
} |
287
|
|
|
|
|
|
|
|
288
|
6
|
50
|
|
|
|
25
|
if(ref($self->{'Help_sub'}) eq 'CODE') |
289
|
|
|
|
|
|
|
{ |
290
|
0
|
0
|
|
|
|
0
|
my ($cmd, $args) = (defined $args ? split /\s+/, $args, 2 : ('', '')); |
291
|
|
|
|
|
|
|
|
292
|
0
|
|
0
|
|
|
0
|
my $cmd_resolved = $self->resolve_alias($cmd) || $cmd; |
293
|
|
|
|
|
|
|
|
294
|
0
|
|
|
|
|
0
|
return $self->{'Help_sub'}->($self->{irc}, $cmd, $cmd_resolved, $args, @help); |
295
|
|
|
|
|
|
|
} |
296
|
|
|
|
|
|
|
else |
297
|
|
|
|
|
|
|
{ |
298
|
6
|
|
|
|
|
25
|
return @help; |
299
|
|
|
|
|
|
|
} |
300
|
|
|
|
|
|
|
} |
301
|
|
|
|
|
|
|
|
302
|
|
|
|
|
|
|
sub add { |
303
|
8
|
|
|
8
|
1
|
3175
|
my ($self, $cmd, $usage) = @_; |
304
|
8
|
|
|
|
|
29
|
$cmd = lc $cmd; |
305
|
8
|
50
|
|
|
|
41
|
return if exists $self->{Commands}->{$cmd}; |
306
|
|
|
|
|
|
|
|
307
|
8
|
100
|
|
|
|
48
|
if (ref $usage eq 'HASH') { |
308
|
1
|
50
|
33
|
|
|
8
|
return if !exists $usage->{info} || !@{ $usage->{args} }; |
|
1
|
|
|
|
|
6
|
|
309
|
|
|
|
|
|
|
} |
310
|
|
|
|
|
|
|
|
311
|
8
|
|
|
|
|
29
|
$self->{Commands}->{$cmd} = $usage; |
312
|
8
|
|
|
|
|
54
|
return 1; |
313
|
|
|
|
|
|
|
} |
314
|
|
|
|
|
|
|
|
315
|
|
|
|
|
|
|
sub remove { |
316
|
5
|
|
|
5
|
1
|
21
|
my ($self, $cmd) = @_; |
317
|
5
|
|
|
|
|
17
|
$cmd = lc $cmd; |
318
|
5
|
50
|
|
|
|
24
|
return if !exists $self->{Commands}->{$cmd}; |
319
|
5
|
|
|
|
|
25
|
delete $self->{Commands}->{$cmd}; |
320
|
5
|
|
|
|
|
23
|
return 1; |
321
|
|
|
|
|
|
|
} |
322
|
|
|
|
|
|
|
|
323
|
|
|
|
|
|
|
sub list { |
324
|
26
|
|
|
26
|
1
|
66
|
my ($self) = @_; |
325
|
26
|
|
|
|
|
52
|
return %{ $self->{Commands} }; |
|
26
|
|
|
|
|
170
|
|
326
|
|
|
|
|
|
|
} |
327
|
|
|
|
|
|
|
|
328
|
|
|
|
|
|
|
sub resolve_alias { |
329
|
21
|
|
|
21
|
1
|
81
|
my ($self, $alias) = @_; |
330
|
|
|
|
|
|
|
|
331
|
21
|
|
|
|
|
76
|
my %cmds = $self->list(); |
332
|
|
|
|
|
|
|
|
333
|
|
|
|
|
|
|
#TODO: refactor using smartmatch/Perl6::Junction if feasible |
334
|
21
|
|
|
|
|
130
|
while(my ($cmd, $info) = each(%cmds)) |
335
|
|
|
|
|
|
|
{ |
336
|
32
|
100
|
|
|
|
168
|
next unless ref($info) eq 'HASH'; |
337
|
6
|
50
|
33
|
|
|
33
|
next unless $info->{aliases} && ref($info->{aliases}) eq 'ARRAY'; |
338
|
0
|
|
|
|
|
0
|
my @aliases = @{$info->{aliases}}; |
|
0
|
|
|
|
|
0
|
|
339
|
|
|
|
|
|
|
|
340
|
0
|
|
|
|
|
0
|
foreach my $cmdalias (@aliases) |
341
|
|
|
|
|
|
|
{ |
342
|
0
|
0
|
|
|
|
0
|
return $cmd if $alias eq $cmdalias; |
343
|
|
|
|
|
|
|
} |
344
|
|
|
|
|
|
|
} |
345
|
|
|
|
|
|
|
|
346
|
21
|
|
|
|
|
115
|
return undef; |
347
|
|
|
|
|
|
|
} |
348
|
|
|
|
|
|
|
|
349
|
|
|
|
|
|
|
sub list_aliases |
350
|
|
|
|
|
|
|
{ |
351
|
1
|
|
|
1
|
0
|
4
|
my ($self, $cmd) = @_; |
352
|
1
|
|
|
|
|
9
|
$cmd = lc $cmd; |
353
|
1
|
50
|
|
|
|
8
|
return if !exists $self->{Commands}->{$cmd}; |
354
|
1
|
50
|
|
|
|
4
|
return unless ref($self->{Commands}->{$cmd}) eq 'HASH'; |
355
|
1
|
50
|
33
|
|
|
7
|
return unless exists $self->{Commands}->{$cmd}->{aliases} && ref($self->{Commands}->{$cmd}->{aliases}) eq 'ARRAY'; |
356
|
0
|
|
|
|
|
|
return @{$self->{Commands}->{$cmd}->{aliases}}; |
|
0
|
|
|
|
|
|
|
357
|
|
|
|
|
|
|
|
358
|
|
|
|
|
|
|
} |
359
|
|
|
|
|
|
|
|
360
|
|
|
|
|
|
|
1; |
361
|
|
|
|
|
|
|
|
362
|
|
|
|
|
|
|
=encoding utf8 |
363
|
|
|
|
|
|
|
|
364
|
|
|
|
|
|
|
=head1 NAME |
365
|
|
|
|
|
|
|
|
366
|
|
|
|
|
|
|
POE::Component::IRC::Plugin::BotCommand - A PoCo-IRC plugin which handles |
367
|
|
|
|
|
|
|
commands issued to your bot |
368
|
|
|
|
|
|
|
|
369
|
|
|
|
|
|
|
=head1 SYNOPSIS |
370
|
|
|
|
|
|
|
|
371
|
|
|
|
|
|
|
use POE; |
372
|
|
|
|
|
|
|
use POE::Component::Client::DNS; |
373
|
|
|
|
|
|
|
use POE::Component::IRC; |
374
|
|
|
|
|
|
|
use POE::Component::IRC::Plugin::BotCommand; |
375
|
|
|
|
|
|
|
|
376
|
|
|
|
|
|
|
my @channels = ('#channel1', '#channel2'); |
377
|
|
|
|
|
|
|
my $dns = POE::Component::Client::DNS->spawn(); |
378
|
|
|
|
|
|
|
my $irc = POE::Component::IRC->spawn( |
379
|
|
|
|
|
|
|
nick => 'YourBot', |
380
|
|
|
|
|
|
|
server => 'some.irc.server', |
381
|
|
|
|
|
|
|
); |
382
|
|
|
|
|
|
|
|
383
|
|
|
|
|
|
|
POE::Session->create( |
384
|
|
|
|
|
|
|
package_states => [ |
385
|
|
|
|
|
|
|
main => [ qw(_start irc_001 irc_botcmd_slap irc_botcmd_lookup dns_response) ], |
386
|
|
|
|
|
|
|
], |
387
|
|
|
|
|
|
|
); |
388
|
|
|
|
|
|
|
|
389
|
|
|
|
|
|
|
$poe_kernel->run(); |
390
|
|
|
|
|
|
|
|
391
|
|
|
|
|
|
|
sub _start { |
392
|
|
|
|
|
|
|
$irc->plugin_add('BotCommand', POE::Component::IRC::Plugin::BotCommand->new( |
393
|
|
|
|
|
|
|
Commands => { |
394
|
|
|
|
|
|
|
slap => 'Takes one argument: a nickname to slap.', |
395
|
|
|
|
|
|
|
lookup => 'Takes two arguments: a record type (optional), and a host.', |
396
|
|
|
|
|
|
|
} |
397
|
|
|
|
|
|
|
)); |
398
|
|
|
|
|
|
|
$irc->yield(register => qw(001 botcmd_slap botcmd_lookup)); |
399
|
|
|
|
|
|
|
$irc->yield(connect => { }); |
400
|
|
|
|
|
|
|
} |
401
|
|
|
|
|
|
|
|
402
|
|
|
|
|
|
|
# join some channels |
403
|
|
|
|
|
|
|
sub irc_001 { |
404
|
|
|
|
|
|
|
$irc->yield(join => $_) for @channels; |
405
|
|
|
|
|
|
|
return; |
406
|
|
|
|
|
|
|
} |
407
|
|
|
|
|
|
|
|
408
|
|
|
|
|
|
|
# the good old slap |
409
|
|
|
|
|
|
|
sub irc_botcmd_slap { |
410
|
|
|
|
|
|
|
my $nick = (split /!/, $_[ARG0])[0]; |
411
|
|
|
|
|
|
|
my ($where, $arg) = @_[ARG1, ARG2]; |
412
|
|
|
|
|
|
|
$irc->yield(ctcp => $where, "ACTION slaps $arg"); |
413
|
|
|
|
|
|
|
return; |
414
|
|
|
|
|
|
|
} |
415
|
|
|
|
|
|
|
|
416
|
|
|
|
|
|
|
# non-blocking dns lookup |
417
|
|
|
|
|
|
|
sub irc_botcmd_lookup { |
418
|
|
|
|
|
|
|
my $nick = (split /!/, $_[ARG0])[0]; |
419
|
|
|
|
|
|
|
my ($where, $arg) = @_[ARG1, ARG2]; |
420
|
|
|
|
|
|
|
my ($type, $host) = $arg =~ /^(?:(\w+) )?(\S+)/; |
421
|
|
|
|
|
|
|
|
422
|
|
|
|
|
|
|
my $res = $dns->resolve( |
423
|
|
|
|
|
|
|
event => 'dns_response', |
424
|
|
|
|
|
|
|
host => $host, |
425
|
|
|
|
|
|
|
type => $type, |
426
|
|
|
|
|
|
|
context => { |
427
|
|
|
|
|
|
|
where => $where, |
428
|
|
|
|
|
|
|
nick => $nick, |
429
|
|
|
|
|
|
|
}, |
430
|
|
|
|
|
|
|
); |
431
|
|
|
|
|
|
|
$poe_kernel->yield(dns_response => $res) if $res; |
432
|
|
|
|
|
|
|
return; |
433
|
|
|
|
|
|
|
} |
434
|
|
|
|
|
|
|
|
435
|
|
|
|
|
|
|
sub dns_response { |
436
|
|
|
|
|
|
|
my $res = $_[ARG0]; |
437
|
|
|
|
|
|
|
my @answers = map { $_->rdatastr } $res->{response}->answer() if $res->{response}; |
438
|
|
|
|
|
|
|
|
439
|
|
|
|
|
|
|
$irc->yield( |
440
|
|
|
|
|
|
|
'notice', |
441
|
|
|
|
|
|
|
$res->{context}->{where}, |
442
|
|
|
|
|
|
|
$res->{context}->{nick} . (@answers |
443
|
|
|
|
|
|
|
? ": @answers" |
444
|
|
|
|
|
|
|
: ': no answers for "' . $res->{host} . '"') |
445
|
|
|
|
|
|
|
); |
446
|
|
|
|
|
|
|
|
447
|
|
|
|
|
|
|
return; |
448
|
|
|
|
|
|
|
} |
449
|
|
|
|
|
|
|
|
450
|
|
|
|
|
|
|
=head1 DESCRIPTION |
451
|
|
|
|
|
|
|
|
452
|
|
|
|
|
|
|
POE::Component::IRC::Plugin::BotCommand is a |
453
|
|
|
|
|
|
|
L plugin. It provides you with a |
454
|
|
|
|
|
|
|
standard interface to define bot commands and lets you know when they are |
455
|
|
|
|
|
|
|
issued. Commands are accepted as channel or private messages. |
456
|
|
|
|
|
|
|
|
457
|
|
|
|
|
|
|
The plugin will respond to the 'help' command by default, listing available |
458
|
|
|
|
|
|
|
commands and information on how to use them. However, if you add a help |
459
|
|
|
|
|
|
|
command yourself, that one will be used instead. |
460
|
|
|
|
|
|
|
|
461
|
|
|
|
|
|
|
=head1 METHODS |
462
|
|
|
|
|
|
|
|
463
|
|
|
|
|
|
|
=head2 C |
464
|
|
|
|
|
|
|
|
465
|
|
|
|
|
|
|
B<'Commands'>, a hash reference, with your commands as keys, and usage |
466
|
|
|
|
|
|
|
information as values. If the usage string contains newlines, the plugin |
467
|
|
|
|
|
|
|
will send one message for each line. |
468
|
|
|
|
|
|
|
|
469
|
|
|
|
|
|
|
If a command's value is a HASH ref like this: |
470
|
|
|
|
|
|
|
|
471
|
|
|
|
|
|
|
$irc->plugin_add('BotCommand', POE::Component::IRC::Plugin::BotCommand->new( |
472
|
|
|
|
|
|
|
Commands => { |
473
|
|
|
|
|
|
|
slap => { |
474
|
|
|
|
|
|
|
info => 'Slap someone', |
475
|
|
|
|
|
|
|
args => [qw(nickname)], |
476
|
|
|
|
|
|
|
nickname => 'nickname to slap' |
477
|
|
|
|
|
|
|
} |
478
|
|
|
|
|
|
|
} |
479
|
|
|
|
|
|
|
)); |
480
|
|
|
|
|
|
|
|
481
|
|
|
|
|
|
|
The args array reference is than used to validate number of arguments required |
482
|
|
|
|
|
|
|
and to name arguments passed to event handler. Help is than generated from |
483
|
|
|
|
|
|
|
C and other hash keys which represent arguments (they are optional). |
484
|
|
|
|
|
|
|
|
485
|
|
|
|
|
|
|
An optional C key can be specified inside the HASH ref to override the event handler. |
486
|
|
|
|
|
|
|
The irc_botcmd_ prefix is not automatically prepended to the handler name when overriding it. |
487
|
|
|
|
|
|
|
|
488
|
|
|
|
|
|
|
An optional C key can be specified inside the HASH ref containing a array ref with alias names. |
489
|
|
|
|
|
|
|
The aliases can be specified for help and to run the command. |
490
|
|
|
|
|
|
|
|
491
|
|
|
|
|
|
|
=head3 Accepting commands |
492
|
|
|
|
|
|
|
|
493
|
|
|
|
|
|
|
B<'In_channels'>, a boolean value indicating whether to accept commands in |
494
|
|
|
|
|
|
|
channels. Default is true. |
495
|
|
|
|
|
|
|
|
496
|
|
|
|
|
|
|
B<'In_private'>, a boolean value indicating whether to accept commands in |
497
|
|
|
|
|
|
|
private. Default is true. |
498
|
|
|
|
|
|
|
|
499
|
|
|
|
|
|
|
B<'Addressed'>, requires users to address the bot by name in order |
500
|
|
|
|
|
|
|
to issue commands. Default is true. |
501
|
|
|
|
|
|
|
|
502
|
|
|
|
|
|
|
B<'Prefix'>, a string which all commands must be prefixed with (except in |
503
|
|
|
|
|
|
|
channels when B<'Addressed'> is true). Default is '!'. You can set it to '' |
504
|
|
|
|
|
|
|
to allow bare commands. |
505
|
|
|
|
|
|
|
|
506
|
|
|
|
|
|
|
B<'Bare_private'>, a boolean value indicating whether bare commands (without |
507
|
|
|
|
|
|
|
the prefix) are allowed in private messages. Default is false. |
508
|
|
|
|
|
|
|
|
509
|
|
|
|
|
|
|
=head3 Authorization |
510
|
|
|
|
|
|
|
|
511
|
|
|
|
|
|
|
B<'Auth_sub'>, a subroutine reference which, if provided, will be called |
512
|
|
|
|
|
|
|
for every command. The subroutine will be called in list context. If the |
513
|
|
|
|
|
|
|
first value returned is true, the command will be processed as normal. If |
514
|
|
|
|
|
|
|
the value is false, then no events will be generated, and an error message |
515
|
|
|
|
|
|
|
will possibly be sent back to the user. |
516
|
|
|
|
|
|
|
|
517
|
|
|
|
|
|
|
You can override the default error message by returning a second value, an |
518
|
|
|
|
|
|
|
array reference of (zero or more) strings. Each string will be sent as a |
519
|
|
|
|
|
|
|
message to the user. |
520
|
|
|
|
|
|
|
|
521
|
|
|
|
|
|
|
Your subroutine will be called with the following arguments: |
522
|
|
|
|
|
|
|
|
523
|
|
|
|
|
|
|
=over 4 |
524
|
|
|
|
|
|
|
|
525
|
|
|
|
|
|
|
=item 1. The IRC component object |
526
|
|
|
|
|
|
|
|
527
|
|
|
|
|
|
|
=item 2. The nick!user@host of the user |
528
|
|
|
|
|
|
|
|
529
|
|
|
|
|
|
|
=item 3. The place where the command was issued (the nickname of the user if |
530
|
|
|
|
|
|
|
it was in private) |
531
|
|
|
|
|
|
|
|
532
|
|
|
|
|
|
|
=item 4. The name of the command |
533
|
|
|
|
|
|
|
|
534
|
|
|
|
|
|
|
=item 5. The command argument string |
535
|
|
|
|
|
|
|
|
536
|
|
|
|
|
|
|
=back |
537
|
|
|
|
|
|
|
|
538
|
|
|
|
|
|
|
B<'Ignore_unauthorized'>, if true, the plugin will ignore unauthorized |
539
|
|
|
|
|
|
|
commands, rather than printing an error message upon receiving them. This is |
540
|
|
|
|
|
|
|
only relevant if B<'Auth_sub'> is also supplied. Default is false. |
541
|
|
|
|
|
|
|
|
542
|
|
|
|
|
|
|
=head3 Help Command |
543
|
|
|
|
|
|
|
|
544
|
|
|
|
|
|
|
B<'Help_sub'>, a subroutine reference which, if provided, will be called upon |
545
|
|
|
|
|
|
|
the end of the predefined help command. The subroutine will be called in list context. |
546
|
|
|
|
|
|
|
|
547
|
|
|
|
|
|
|
Your subroutine will be called with the following arguments: |
548
|
|
|
|
|
|
|
|
549
|
|
|
|
|
|
|
=over 4 |
550
|
|
|
|
|
|
|
|
551
|
|
|
|
|
|
|
=item 1. The IRC component object |
552
|
|
|
|
|
|
|
|
553
|
|
|
|
|
|
|
=item 2. The command. |
554
|
|
|
|
|
|
|
|
555
|
|
|
|
|
|
|
=item 3. The resolved command(after alias processing). |
556
|
|
|
|
|
|
|
|
557
|
|
|
|
|
|
|
=item 4. The arguments. |
558
|
|
|
|
|
|
|
|
559
|
|
|
|
|
|
|
=item 5. The generated help text as array. |
560
|
|
|
|
|
|
|
|
561
|
|
|
|
|
|
|
|
562
|
|
|
|
|
|
|
=back |
563
|
|
|
|
|
|
|
|
564
|
|
|
|
|
|
|
|
565
|
|
|
|
|
|
|
=head3 Miscellaneous |
566
|
|
|
|
|
|
|
|
567
|
|
|
|
|
|
|
B<'Ignore_unknown'>, if true, the plugin will ignore undefined commands, |
568
|
|
|
|
|
|
|
rather than printing a help message upon receiving them. Default is false. |
569
|
|
|
|
|
|
|
|
570
|
|
|
|
|
|
|
B<'Method'>, how you want help messages to be delivered. Valid options are |
571
|
|
|
|
|
|
|
'notice' (the default) and 'privmsg'. |
572
|
|
|
|
|
|
|
|
573
|
|
|
|
|
|
|
B<'Eat'>, set to true to make the plugin hide |
574
|
|
|
|
|
|
|
L|POE::Component::IRC/irc_public> events from other plugins |
575
|
|
|
|
|
|
|
when they look like commands. Probably only useful when a B<'Prefix'> is |
576
|
|
|
|
|
|
|
defined. Default is false. |
577
|
|
|
|
|
|
|
|
578
|
|
|
|
|
|
|
Returns a plugin object suitable for feeding to |
579
|
|
|
|
|
|
|
L's C method. |
580
|
|
|
|
|
|
|
|
581
|
|
|
|
|
|
|
=head2 C |
582
|
|
|
|
|
|
|
|
583
|
|
|
|
|
|
|
Adds a new command. Takes two arguments, the name of the command, and a string |
584
|
|
|
|
|
|
|
or hash reference containing its usage information (see C). Returns false |
585
|
|
|
|
|
|
|
if the command has already been defined or no info or arguments are provided, |
586
|
|
|
|
|
|
|
true otherwise. |
587
|
|
|
|
|
|
|
|
588
|
|
|
|
|
|
|
=head2 C |
589
|
|
|
|
|
|
|
|
590
|
|
|
|
|
|
|
Removes a command. Takes one argument, the name of the command. Returns false |
591
|
|
|
|
|
|
|
if the command wasn't defined to begin with, true otherwise. |
592
|
|
|
|
|
|
|
|
593
|
|
|
|
|
|
|
=head2 C |
594
|
|
|
|
|
|
|
|
595
|
|
|
|
|
|
|
Takes no arguments. Returns a list of key/value pairs, the keys being the |
596
|
|
|
|
|
|
|
command names and the values being the usage strings or hash references. |
597
|
|
|
|
|
|
|
|
598
|
|
|
|
|
|
|
=head2 C |
599
|
|
|
|
|
|
|
|
600
|
|
|
|
|
|
|
Takes one argument, a string to match against command aliases, if no matching |
601
|
|
|
|
|
|
|
command can be found undef is returned. |
602
|
|
|
|
|
|
|
|
603
|
|
|
|
|
|
|
=head1 OUTPUT EVENTS |
604
|
|
|
|
|
|
|
|
605
|
|
|
|
|
|
|
=head2 C |
606
|
|
|
|
|
|
|
|
607
|
|
|
|
|
|
|
You will receive an event like this for every valid command issued. E.g. if |
608
|
|
|
|
|
|
|
'slap' were a valid command, you would receive an C event |
609
|
|
|
|
|
|
|
every time someone issued that command. It receives the following arguments: |
610
|
|
|
|
|
|
|
|
611
|
|
|
|
|
|
|
=over 4 |
612
|
|
|
|
|
|
|
|
613
|
|
|
|
|
|
|
=item * C: the nick!hostmask of the user who issued the command. |
614
|
|
|
|
|
|
|
|
615
|
|
|
|
|
|
|
=item * C is the name of the channel in which the command was issued, |
616
|
|
|
|
|
|
|
or the sender's nickname if this was a private message. |
617
|
|
|
|
|
|
|
|
618
|
|
|
|
|
|
|
=item * C: a string of arguments to the command, or hash reference with |
619
|
|
|
|
|
|
|
arguments in case you defined command along with arguments, or undef if there |
620
|
|
|
|
|
|
|
were no arguments |
621
|
|
|
|
|
|
|
|
622
|
|
|
|
|
|
|
=back |
623
|
|
|
|
|
|
|
|
624
|
|
|
|
|
|
|
=head1 AUTHOR |
625
|
|
|
|
|
|
|
|
626
|
|
|
|
|
|
|
Hinrik Ern SigurEsson, hinrik.sig@gmail.com |
627
|
|
|
|
|
|
|
|
628
|
|
|
|
|
|
|
=cut |