our
$AUTHORITY
=
'cpan:HINRIK'
;
$POE::Component::IRC::Qnet::State::VERSION
=
'6.93'
;
use
IRC::Utils
qw(uc_irc normalize_mask parse_user)
;
sub
_create {
my
$self
=
shift
;
$self
->SUPER::_create();
my
@qbot_commands
=
qw(
hello
whoami
challengeauth
showcommands
auth
challenge
help
unlock
requestpassword
reset
newpass
email
authhistory
banclear
op
invite
removeuser
banlist
recover
limit
unbanall
whois
version
autolimit
ban
clearchan
adduser
settopic
chanflags
deopall
requestowner
bandel
chanlev
key
welcome
voice
)
;
$self
->{OBJECT_STATES_HASHREF}->{
'qbot_'
.
$_
} =
'_qnet_bot_commands'
for
@qbot_commands
;
$self
->{OBJECT_STATES_HASHREF}->{
'resync_chan'
} =
'_resync_chan'
;
$self
->{OBJECT_STATES_HASHREF}->{
'resync_nick'
} =
'_resync_nick'
;
$self
->{server} =
'irc.quakenet.org'
;
$self
->{QBOT} =
'Q@Cserve.quakenet.org'
;
return
1;
}
sub
_resync_chan {
my
(
$kernel
,
$self
,
@channels
) =
@_
[KERNEL, OBJECT, ARG0 ..
$#_
];
my
$mapping
=
$self
->isupport(
'CASEMAPPING'
);
my
$nickname
=
$self
->nick_name();
my
$flags
=
'%cunharsft'
;
for
my
$channel
(
@channels
) {
next
if
!
$self
->is_channel_member(
$channel
,
$nickname
);
my
$uchan
= uc_irc
$channel
,
$mapping
;
delete
$self
->{STATE}->{Chans}->{
$uchan
};
$self
->{CHANNEL_SYNCH}->{
$uchan
} = {
MODE
=> 0,
WHO
=> 0,
BAN
=> 0,
_time
=>
time
() };
$self
->{STATE}->{Chans}->{
$uchan
} = {
Name
=>
$channel
,
Mode
=>
''
};
$self
->yield (
'sl'
=>
"WHO $channel $flags,101"
);
$self
->yield (
'mode'
=>
$channel
);
$self
->yield (
'mode'
=>
$channel
=>
'b'
);
}
return
;
}
sub
_resync_nick {
my
(
$kernel
,
$self
,
$nick
,
@channels
) =
@_
[KERNEL ,OBJECT, ARG0 ..
$#_
];
my
$info
=
$self
->nick_info(
$nick
);
return
if
!
$info
;
$nick
=
$info
->{Nick};
my
$user
=
$info
->{User};
my
$host
=
$info
->{Host};
my
$mapping
=
$self
->isupport(
'CASEMAPPING'
);
my
$unick
= uc_irc
$nick
,
$mapping
;
my
$flags
=
'%cunharsft'
;
for
my
$channel
(
@channels
) {
next
if
!
$self
->is_channel_member(
$channel
,
$nick
);
my
$uchan
= uc_irc
$channel
,
$mapping
;
$self
->yield (
'sl'
=>
"WHO $nick $flags,102"
);
$self
->{STATE}->{Nicks}->{
$unick
}->{Nick} =
$nick
;
$self
->{STATE}->{Nicks}->{
$unick
}->{User} =
$user
;
$self
->{STATE}->{Nicks}->{
$unick
}->{Host} =
$host
;
$self
->{STATE}->{Nicks}->{
$unick
}->{CHANS}->{
$uchan
} =
''
;
$self
->{STATE}->{Chans}->{
$uchan
}->{Nicks}->{
$unick
} =
''
;
push
@{
$self
->{NICK_SYNCH}->{
$unick
} },
$channel
;
}
return
;
}
sub
S_330 {
my
(
$self
,
$irc
) =
splice
@_
, 0, 2;
my
(
$nick
,
$account
) = (
split
/ /, ${
$_
[1] } )[0..1];
$self
->{WHOIS}->{
$nick
}->{account} =
$account
;
return
PCI_EAT_NONE;
}
sub
S_354 {
my
(
$self
,
$irc
) =
splice
@_
, 0, 2;
my
$mapping
=
$irc
->isupport(
'CASEMAPPING'
);
my
(
$query
,
$channel
,
$user
,
$host
,
$server
,
$nick
,
$status
,
$auth
,
$real
)
= @{ ${
$_
[2] } };
my
$unick
= uc_irc
$nick
,
$mapping
;
my
$uchan
= uc_irc
$channel
,
$mapping
;
$self
->{STATE}->{Nicks}->{
$unick
}->{Nick} =
$nick
;
$self
->{STATE}->{Nicks}->{
$unick
}->{User} =
$user
;
$self
->{STATE}->{Nicks}->{
$unick
}->{Host} =
$host
;
$self
->{STATE}->{Nicks}->{
$unick
}->{Real} =
$real
;
$self
->{STATE}->{Nicks}->{
$unick
}->{Server} =
$server
;
$self
->{STATE}->{Nicks}->{
$unick
}->{Auth} =
$auth
if
(
$auth
);
if
(
$auth
and
defined
(
$self
->{USER_AUTHED}->{
$unick
} ) ) {
$self
->{USER_AUTHED}->{
$unick
} =
$auth
;
}
if
(
$query
eq
'101'
) {
my
$whatever
=
''
;
$whatever
.=
'o'
if
$status
=~ /\@/;
$whatever
.=
'v'
if
$status
=~ /\+/;
$whatever
.=
'h'
if
$status
=~ /\%/;
$self
->{STATE}->{Nicks}->{
$unick
}->{CHANS}->{
$uchan
} =
$whatever
;
$self
->{STATE}->{Chans}->{
$uchan
}->{Name} =
$channel
;
$self
->{STATE}->{Chans}->{
$uchan
}->{Nicks}->{
$unick
} =
$whatever
;
}
if
(
$status
=~ /\*/ ) {
$self
->{STATE}->{Nicks}->{
$unick
}->{IRCop} = 1;
}
return
PCI_EAT_NONE;
}
sub
S_315 {
my
(
$self
,
$irc
) =
splice
@_
, 0, 2;
my
$mapping
=
$irc
->isupport(
'CASEMAPPING'
);
my
$channel
= ${
$_
[2] }->[0];
my
$uchan
= uc_irc
$channel
,
$mapping
;
if
(
exists
$self
->{STATE}->{Chans}->{
$uchan
} ) {
if
(
$self
->_channel_sync(
$channel
,
'WHO'
) ) {
my
$rec
=
delete
$self
->{CHANNEL_SYNCH}->{
$uchan
};
$self
->send_event_next(
'irc_chan_sync'
,
$channel
,
time
() -
$rec
->{_time} );
}
}
elsif
(
defined
$self
->{USER_AUTHED}->{
$uchan
} ) {
$self
->send_event_next(
'irc_nick_authed'
,
$channel
,
delete
$self
->{USER_AUTHED}->{
$uchan
} );
}
else
{
my
$chan
=
shift
@{
$self
->{NICK_SYNCH}->{
$uchan
} };
delete
$self
->{NICK_SYNCH}->{
$uchan
}
if
!@{
$self
->{NICK_SYNCH}->{
$uchan
} };
$self
->send_event_next(
'irc_nick_sync'
,
$channel
,
$chan
);
}
return
PCI_EAT_NONE;
}
sub
S_join {
my
(
$self
,
$irc
) =
splice
@_
, 0, 2;
my
(
$nick
,
$user
,
$host
) = parse_user(${
$_
[0] } );
my
$channel
= ${
$_
[1] };
my
$mapping
=
$irc
->isupport(
'CASEMAPPING'
);
my
$uchan
= uc_irc
$channel
,
$mapping
;
my
$unick
= uc_irc
$nick
,
$mapping
;
my
$flags
=
'%cunharsft'
;
if
(
$unick
eq uc_irc (
$self
->nick_name(),
$mapping
) ) {
delete
$self
->{STATE}->{Chans}->{
$uchan
};
$self
->{CHANNEL_SYNCH}->{
$uchan
} = {
MODE
=> 0,
WHO
=> 0,
BAN
=> 0,
_time
=>
time
()
};
$self
->{STATE}->{Chans}->{
$uchan
} = {
Name
=>
$channel
,
Mode
=>
''
};
$self
->yield (
'sl'
=>
"WHO $channel $flags,101"
);
$self
->yield (
'mode'
=>
$channel
);
$self
->yield (
'mode'
=>
$channel
=>
'b'
);
}
else
{
my
$netsplit
=
"$unick!$user\@$host"
;
if
(
exists
$self
->{NETSPLIT}->{Users}->{
$netsplit
} ) {
my
$nuser
=
delete
$self
->{NETSPLIT}->{Users}->{
$netsplit
};
if
( (
time
-
$nuser
->{stamp} ) < ( 60 * 60 ) ) {
$self
->{STATE}->{Nicks}->{
$unick
} =
$nuser
->{meta};
$self
->send_event_next(
irc_nick_sync
=>
$nick
,
$channel
);
}
return
PCI_EAT_NONE;
}
if
(
exists
$self
->{STATE}->{Nicks}->{
$unick
}->{Real} ) {
$self
->send_event_next(
irc_nick_sync
=>
$nick
,
$channel
);
return
PCI_EAT_NONE;
}
$self
->yield (
'sl'
=>
"WHO $nick $flags,102"
);
$self
->{STATE}->{Nicks}->{
$unick
}->{Nick} =
$nick
;
$self
->{STATE}->{Nicks}->{
$unick
}->{User} =
$user
;
$self
->{STATE}->{Nicks}->{
$unick
}->{Host} =
$host
;
$self
->{STATE}->{Nicks}->{
$unick
}->{CHANS}->{
$uchan
} =
''
;
$self
->{STATE}->{Chans}->{
$uchan
}->{Nicks}->{
$unick
} =
''
;
push
@{
$self
->{NICK_SYNCH}->{
$unick
} },
$channel
;
}
return
PCI_EAT_NONE;
}
sub
S_chan_mode {
my
(
$self
,
$irc
) =
splice
@_
, 0, 2;
my
$mapping
=
$irc
->isupport(
'CASEMAPPING'
);
pop
@_
;
my
$who
= ${
$_
[0] };
my
$source
= uc_irc ( (
split
/!/,
$who
)[0],
$mapping
);
my
$mode
= ${
$_
[2] };
my
$arg
=
defined
$_
[3] ? ${
$_
[3] } :
''
;
my
$uarg
= uc_irc
$arg
,
$mapping
;
return
PCI_EAT_NONE
if
$source
!~ /^[Q]$/ ||
$mode
!~ /[ov]/;
if
( !
$self
->is_nick_authed(
$arg
) && !
$self
->{USER_AUTHED}->{
$uarg
} ) {
$self
->{USER_AUTHED}->{
$uarg
} = 0;
$self
->yield (
'sl'
=>
"WHO $arg "
.
'%cunharsft,102'
);
}
return
PCI_EAT_NONE;
}
sub
S_part {
my
(
$self
,
$irc
) =
splice
@_
, 0, 2;
my
$mapping
=
$irc
->isupport(
'CASEMAPPING'
);
my
$nick
= uc_irc ( (
split
/!/, ${
$_
[0] } )[0],
$mapping
);
my
$channel
= uc_irc ${
$_
[1] },
$mapping
;
if
(
ref
$_
[2] eq
'ARRAY'
) {
push
@{
$_
[-1] },
''
,
$self
->is_nick_authed(
$nick
);
}
else
{
push
@{
$_
[-1] },
$self
->is_nick_authed(
$nick
);
}
if
(
$nick
eq uc_irc (
$self
->nick_name(),
$mapping
) ) {
delete
$self
->{STATE}->{Nicks}->{
$nick
}->{CHANS}->{
$channel
};
delete
$self
->{STATE}->{Chans}->{
$channel
}->{Nicks}->{
$nick
};
for
my
$member
(
keys
%{
$self
->{STATE}->{Chans}->{
$channel
}->{Nicks} } ) {
delete
$self
->{STATE}->{Nicks}->{
$member
}->{CHANS}->{
$channel
};
if
(
keys
%{
$self
->{STATE}->{Nicks}->{
$member
}->{CHANS} } <= 0 ) {
delete
$self
->{STATE}->{Nicks}->{
$member
};
}
}
delete
$self
->{STATE}->{Chans}->{
$channel
};
}
else
{
delete
$self
->{STATE}->{Nicks}->{
$nick
}->{CHANS}->{
$channel
};
delete
$self
->{STATE}->{Chans}->{
$channel
}->{Nicks}->{
$nick
};
if
(
keys
%{
$self
->{STATE}->{Nicks}->{
$nick
}->{CHANS} } <= 0 ) {
delete
$self
->{STATE}->{Nicks}->{
$nick
};
}
}
return
PCI_EAT_NONE;
}
sub
S_quit {
my
(
$self
,
$irc
) =
splice
@_
, 0, 2;
my
$mapping
=
$irc
->isupport(
'CASEMAPPING'
);
my
$nick
= (
split
/!/, ${
$_
[0] } )[0];
my
$msg
= ${
$_
[1] };
push
@{
$_
[2] }, [
$self
->nick_channels(
$nick
) ];
push
@{
$_
[2] },
$self
->is_nick_authed(
$nick
);
my
$unick
= uc_irc
$nick
,
$mapping
;
my
$netsplit
= 0;
$netsplit
= 1
if
_is_netsplit(
$msg
);
if
(
$unick
eq uc_irc (
$self
->nick_name(),
$mapping
) ) {
delete
$self
->{STATE};
}
else
{
for
my
$channel
(
keys
%{
$self
->{STATE}->{Nicks}->{
$unick
}->{CHANS} } ) {
delete
$self
->{STATE}->{Chans}->{
$channel
}->{Nicks}->{
$unick
};
}
my
$nickstate
=
delete
$self
->{STATE}->{Nicks}->{
$unick
};
if
(
$netsplit
) {
delete
$nickstate
->{CHANS};
$self
->{NETSPLIT}->{Users}->{
"$unick!"
.
join
'@'
, @{
$nickstate
}{
qw(User Host)
} } =
{
meta
=>
$nickstate
,
stamp
=>
time
};
}
}
return
PCI_EAT_NONE;
}
sub
_is_netsplit {
my
$msg
=
shift
||
return
;
return
1
if
$msg
=~ /^\s*\S+\.[a-z]{2,} \S+\.[a-z]{2,}$/i;
return
0;
}
sub
S_kick {
my
(
$self
,
$irc
) =
splice
@_
, 0, 2;
my
$mapping
=
$irc
->isupport(
'CASEMAPPING'
);
my
$channel
= ${
$_
[1] };
my
$nick
= ${
$_
[2] };
my
$unick
= uc_irc
$nick
,
$mapping
;
my
$uchan
= uc_irc
$channel
,
$mapping
;
push
@{
$_
[-1] },
$self
->nick_long_form(
$nick
);
push
@{
$_
[-1] },
$self
->is_nick_authed(
$nick
);
if
(
$unick
eq uc_irc (
$self
->nick_name(),
$mapping
) ) {
delete
$self
->{STATE}->{Nicks}->{
$unick
}->{CHANS}->{
$uchan
};
delete
$self
->{STATE}->{Chans}->{
$uchan
}->{Nicks}->{
$unick
};
for
my
$member
(
keys
%{
$self
->{STATE}->{Chans}->{
$uchan
}->{Nicks} } ) {
delete
$self
->{STATE}->{Nicks}->{
$member
}->{CHANS}->{
$uchan
};
if
(
keys
%{
$self
->{STATE}->{Nicks}->{
$member
}->{CHANS} } <= 0 ) {
delete
$self
->{STATE}->{Nicks}->{
$member
};
}
}
delete
$self
->{STATE}->{Chans}->{
$uchan
};
}
else
{
delete
$self
->{STATE}->{Nicks}->{
$unick
}->{CHANS}->{
$uchan
};
delete
$self
->{STATE}->{Chans}->{
$uchan
}->{Nicks}->{
$unick
};
if
(
keys
%{
$self
->{STATE}->{Nicks}->{
$unick
}->{CHANS} } <= 0 ) {
delete
$self
->{STATE}->{Nicks}->{
$unick
};
}
}
return
PCI_EAT_NONE;
}
sub
is_nick_authed {
my
(
$self
,
$nick
) =
@_
;
my
$mapping
=
$self
->isupport(
'CASEMAPPING'
);
my
$unick
= uc_irc
$nick
,
$mapping
;
return
if
!
$self
->_nick_exists(
$nick
);
if
(
defined
$self
->{STATE}->{Nicks}->{
$unick
}->{Auth}) {
return
$self
->{STATE}->{Nicks}->{
$unick
}->{Auth};
}
return
;
}
sub
find_auth_nicks {
my
(
$self
,
$auth
,
$channel
) =
@_
;
my
$mapping
=
$self
->isupport(
'CASEMAPPING'
);
my
$uchan
= uc_irc
$channel
,
$mapping
;
return
if
!
$self
->_channel_exists(
$channel
);
my
@results
;
for
my
$nick
(
keys
%{
$self
->{STATE}->{Chans}->{
$uchan
}->{Nicks} } ) {
if
(
defined
(
$self
->{STATE}->{Nicks}->{
$nick
}->{Auth} )
&&
$self
->{STATE}->{Nicks}->{
$nick
}->{Auth} eq
$auth
) {
push
@results
,
$self
->{STATE}->{Nicks}->{
$nick
}->{Nick};
}
}
return
@results
;
}
sub
ban_mask {
my
(
$self
,
$channel
,
$mask
) =
@_
;
$mask
= normalize_mask(
$mask
);
my
$mapping
=
$self
->isupport(
'CASEMAPPING'
);
my
@result
;
return
if
!
$self
->_channel_exists(
$channel
);
$mask
= u_irc (
$mask
,
$mapping
);
$mask
=
quotemeta
$mask
;
$mask
=~ s/\\\*/[\x01-\xFF]{0,}/g;
$mask
=~ s/\\\?/[\x01-\xFF]{1,1}/g;
for
my
$nick
(
$self
->channel_list(
$channel
) ) {
my
$long_form
=
$self
->nick_long_form(
$nick
);
if
( uc_irc (
$long_form
) =~ /^
$mask
$/ ) {
push
@result
,
$nick
;
next
;
}
if
(
my
$auth
=
$self
->is_nick_authed(
$nick
) ) {
$long_form
=~ s/\@(.+)$/\
@$auth
.users.quakenet.org/;
push
@result
,
$nick
if
uc_irc (
$long_form
) =~ /^
$mask
$/;
}
}
return
@result
;
}
1;