summaryrefslogtreecommitdiffstats
path: root/ksirc/dsirc
diff options
context:
space:
mode:
Diffstat (limited to 'ksirc/dsirc')
-rwxr-xr-xksirc/dsirc2721
1 files changed, 2721 insertions, 0 deletions
diff --git a/ksirc/dsirc b/ksirc/dsirc
new file mode 100755
index 00000000..c6e0b63c
--- /dev/null
+++ b/ksirc/dsirc
@@ -0,0 +1,2721 @@
+#!/usr/bin/perl
+
+# dsirc: dumb-mode small irc client in perl
+# by orabidoo <[email protected]>
+#
+# Copyright (C) 1995-1997 Roger Espel Llima
+#
+# for a full-screen termcap interface, use this with ssfe
+#
+# use: dsirc [options] [nick [server[:port[:password]]]]
+# options are:
+# -p = specify port number
+# -i = specify IRCNAME
+# -n = specify nickname (quite useless as an option)
+# -s = specify server (quite useless as an option)
+# -l = specify file to be loaded instead of ~/.sircrc.pl
+# -L = specify file to be loaded instead of ~/.sircrc
+# -H = specify virtual host to bind to
+# -q = don't load ~/.sircrc or ~/.sircrc.pl
+# -Q = don't load system sircrc or sircrc.pl
+# -R = run in restricted (secure) mode
+# -r = raw mode (no control-char filtering)
+# -8 = 8-bit mode
+# -S = connect using SSL
+
+# This program is free software; you can redistribute it and/or modify
+# it under the terms of the GNU General Public License as published by
+# the Free Software Foundation. See the file LICENSE for more details.
+#
+# If you make improvements to sirc, please send me the modifications
+# (context diffs appreciated) and they might make it to the next release.
+#
+# For bug reports, comments, questions, email [email protected]
+#
+# You can always find the latest version of sirc at the following URL:
+# http://www.eleves.ens.fr:8080/home/espel/sirc/sirc.html
+
+# Concerning the use in ksirc you'll find a mail from the author below:
+#
+# Subject: Re: dsirc in kde
+# Date: Thu, 7 Sep 2000 13:16:30 -0400
+# From: Roger Espel Llima <[email protected]>
+# To: Harri Porten <[email protected]>
+#
+# On Thu, Sep 07, 2000 at 07:12:33PM +0200, Harri Porten wrote:
+# [....]
+# > Ok. Your dsirc script is used in ksirc. I haven't checked how it is
+# > invoked and what legal ramifications that would have licensing wise but
+# > I would like to "officially" ask you anyway:
+# >
+# > Do you have oppose to your code being used this way in the past and in
+# > the future ? Do you "forgive" us [for use in prev. versions of KDE] ? :)
+#
+# I "officially" find it perfectly fine that dsirc is used in KDE. I knew
+# of ksirc when it started, and found it very flattering that someone
+# would write 200k of C++ to interface with my 62k of perl :=)
+
+$version='2.211';
+$date='10 Mar 1998';
+$add_ons='';
+
+$libdir=$ENV{"SIRCLIB"} || ".";
+push(@INC, $libdir, $ENV{"HOME"});
+@loadpath=($ENV{"HOME"}."/.sirc", $libdir, ".");
+$ENV{"SIRCWAIT"} or $ready=1;
+
+$|=1;
+
+$publicAway = 1;
+
+if (!eval "require 'getopts.pl';") {
+ print "\n\n\
+Your perl interpreter is *really* screwed up: the getopts.pl library is not
+even there! Have you even bothered to run 'install'?\n";
+ exit;
+}
+
+if ($] >= 5 && (eval "use Socket;", $@ eq '')) {
+ $sock6 = eval ("require Socket6;") and eval("use Socket6;");
+} elsif (-f "$libdir/sircsock.ph") {
+ do "$libdir/sircsock.ph";
+} elsif (-f $ENV{'HOME'}."/sircsock.ph") {
+ do $ENV{'HOME'}."/sircsock.ph";
+} elsif (!eval "require 'sys/socket.ph';") {
+ print "\n\n\
+Your perl installation is wrong somewhere, the sys/socket.ph include file
+couldn't be found. Have you even bothered to run 'install'?\n";
+ exit;
+}
+
+$hasPOSIX = 1;
+eval "use POSIX;";
+if($@) {
+ $hasPOSIX = 0;
+ print "*** No Posix library, falling back to blocking IO (dcc will suck)\n";
+}
+
+
+&Getopts('n:s:p:u:i:l:L:H:rqQR78S');
+
+%set=("LOGFILE", "", "LOG", "off", "PRINTUH", "none", "PRINTCHAN", "off",
+ "LOCALHOST", "", "CTCP", "noflood", "SENDAHEAD", 4096,
+ "USERINFO", "", "FINGER", "", "IRCNAME", "", "EIGHT_BIT", "on",
+ "LOADPATH", join(":", @loadpath), "CTRL_T", "/next");
+
+$raw_mode=$opt_r || (!-t STDOUT);
+$ansi=!$raw_mode && $ENV{"TERM"} =~ /^vt|^xterm|^ansi/i;
+$server=$opt_s || $ARGV[1] || $ENV{"SIRCSERVER"} || $ENV{"IRCSERVER"} ||
+ "irc.primenet.com";
+$port0=$opt_p || $ENV{"SIRCPORT"} || $ENV{"IRCPORT"} || 6667;
+$username=$opt_u || $ENV{"SIRCUSER"} || $ENV{"IRCUSER"} || (getpwuid($<))[0] ||
+ $ENV{"USER"} || "blah";
+$set{"IRCNAME"}=$opt_i || $ENV{"SIRCNAME"} || $ENV{"IRCNAME"} || "sirc user";
+$nick=$opt_n || $ARGV[0] || $ENV{"SIRCNICK"} || $ENV{"IRCNICK"} || $username;
+$set{"FINGER"}=$ENV{"IRCFINGER"} || "keep your fingers to yourself";
+$set{"USERINFO"}=$ENV{"USERINFO"} || "yep, I'm a user";
+if ($server =~ /^\[([^\]]+)\]:([0-9]*):?([^:]*)$/
+ or $server =~ /^([^:]+):([0-9]*):?([^:]*)$/)
+{
+ ($server, $port, $pass)=($1, $2, $3);
+}
+$port || ($port=$port0);
+$server0=$server1=$server;
+$port0=$port1=$port;
+$pass0=$pass1=$pass;
+$initfile=$opt_l || $ENV{"SIRCRCPL"} || $ENV{'HOME'}."/.sircrc.pl"
+ if $opt_l || !$opt_q;
+$sysinit=$libdir."/sircrc.pl" if $libdir ne '.' && !$opt_Q;
+$rcfile=$opt_L || $ENV{"SIRCRC"} || $ENV{'HOME'}."/.sircrc"
+ if $opt_L || !$opt_q;
+$sysrc=$libdir."/sircrc" if $libdir ne '.' && !$opt_Q;
+$set{"LOGFILE"}=$logfile=$ENV{'HOME'}."/sirc.log";
+$opt_8 || ($set{"EIGHT_BIT"}="off");
+$restrict=$opt_R;
+$set{"LOCALHOST"}=$opt_H || $ENV{"SIRCHOST"} || $ENV{"IRCHOST"} ||
+ $ENV{"LOCALHOST"} || "";
+$SSL=$opt_S;
+
+@ARGV=(); # ignore any more arguments
+
+if (open(H, "$libdir/sirc.help") || ((-f "$libdir/sirc.help.gz") &&
+ open(H, "gzip -cd $libdir/sirc.help.gz |"))) {
+ @help=<H>;
+ close H;
+ foreach (@help) {
+ chop;
+ s/\$version/$version/g;
+ s/\$date/$date/g;
+ }
+} else {
+ print "*** Warning: help file ($libdir/sirc.help) not found!\n";
+}
+$floodtimer=0;
+
+sub exit {
+ &dohooks("quit");
+ &sl("QUIT :using sirc version $version$add_ons") if $connected;
+ close LOG if $logging;
+ exit 0;
+}
+
+$SIG{'PIPE'}='IGNORE';
+$SIG{'QUIT'}='IGNORE';
+$SIG{'INT'}='exit';
+$SIG{'TERM'}='exit'; # KSIRC MOD
+
+sub eq {
+ local($a, $b)=@_;
+ $a =~ tr/A-Z/a-z/;
+ $b =~ tr/A-Z/a-z/;
+ return ($a eq $b);
+}
+
+sub tilde {
+ $_[0] =~ s|^\~(\w+)|(getpwnam($1))[7]|e;
+ $_[0] =~ s/^\~/$ENV{'HOME'}/;
+ $_[0]="." if $_[0] eq '';
+}
+
+sub sigquit {
+ # really ugly hack, but it works...
+ &dohooks("quit");
+ close($trysock);
+}
+
+sub resolve {
+ if ($sock6) {
+ my $addr = $_[0];
+ if ("$addr" =~ /^\d+$/)
+ {
+ $addr = pack("N", $addr);
+ my @i = unpack("C4", $addr);
+ $addr = "$i[0].$i[1].$i[2].$i[3]";
+ }
+ return getaddrinfo($addr, $_[1], $_[2] || &AF_UNSPEC, &SOCK_STREAM);
+ }
+ my $addr;
+ if ($_[0] =~ /^\d+$/) {
+ $addr = pack("N", $_[0]+0);
+ } elsif ($_[0] =~ /^(\d+)\.(\d+)\.(\d+)\.(\d+)$/) {
+ $addr = pack("c4", $1, $2, $3, $4);
+ } else {
+ $addr=(gethostbyname($_[0]))[4];
+ return -1 unless (defined($addr));
+ }
+ return (&AF_INET, &SOCK_STREAM, 0, pack_sockaddr_in($_[1], $addr), undef);
+}
+
+$nextfh="sircblah000";
+sub newfh {
+ return ++$nextfh;
+}
+
+sub connect {
+ $_[0]=&newfh;
+ local($fh, $host, $port)=@_;
+ my @res = resolve($host, $port);
+ &tell("*\cbE\cb* Hostname `$host' not found"), return -1 if scalar(@res) < 5;
+ $family = -1;
+ my $bindfailed;
+ while (scalar(@res) >= 5) {
+ ($family, my ($socktype, $proto, $addr), undef, @res) = @res;
+ &print("*\cbE\cb* Out of file descriptors: $!"), return -2
+ unless socket($fh, $family, $socktype, $proto);
+
+ $bindfailed = undef;
+ if ($set{"LOCALHOST"}) {
+ # once again, DCC only does ipv4
+ $bindaddr = (&resolve($set{"LOCALHOST"}, 0, &AF_INET))[3];
+ $bindfailed = 1 unless bind($fh, $bindaddr);
+ }
+
+ $trysock=$fh;
+ $SIG{'QUIT'}='sigquit';
+ $SIG{'QUIT'}='IGNORE', last if connect($fh, $addr);
+ $SIG{'QUIT'}='IGNORE';
+ $family = -1;
+ }
+ &print("*\cbE\cb* Can't connect to host: $!"), return -3 if $family == -1;
+ # Tried to just check for $family != &AF_INET where needed, but
+ # that segfaulted perl (!), guess it's a bug in Socket6.pm, but I won't try
+ # to debug that. (malte)
+ $ipv6 = 1 if ($sock6 && $family == &AF_INET6);
+ &tell("*\cbE\cb* Warning: can't bind to sirc host: ".$set{'LOCALHOST'})
+ if $bindfailed;
+
+ if ($ipv6 != 1)
+ {
+ $bindaddr=getsockname($fh) unless $bindaddr;
+ }
+ select($fh); $|=1; select(STDOUT);
+ return 1;
+}
+
+sub connectSSL {
+ eval "use IO::Socket::SSL;";
+
+ if($@){
+ &tell("Can't load SSL socket library, perl does not support SSL!");
+ &tell("To use SSL you must install the IO::Socket::SSL perl library");
+ &tell("Try as root: perl -MCPAN -e 'install IO::Socket::SSL'");
+ &tell("Giving up connect");
+ return 0;
+ }
+ local($fh, $host, $port)=@_;
+ &tell("*** Doing SSL server connect...");
+ $fh = new IO::Socket::SSL("$host:$port");
+ if(defined $fh){
+ $_[0] = $fh;
+ select($fh); $|=1; select(STDOUT);
+ return 1;
+ }
+ else {
+ warn "*** I encountered a problem: ($!) ",
+ &IO::Socket::SSL::errstr();
+ warn "*** Invalid hostname or port?\n";
+ return -1;
+ }
+}
+
+sub sel_nbconnecthandler {
+ local($fh) = $_[0];
+ &remwsel($fh);
+ $!="";
+ my $res = unpack("i", getsockopt("$fh", SOL_SOCKET(), SO_ERROR()) || die "Failed to get sockopt: $!");
+ select($fh); $|=1; select(STDOUT);
+ &{$nbconnectlist{$fh}{"callback"}}($fh, $res);
+ $nbconnectlist{$fh} = undef;
+}
+
+#
+# Non blocking connect
+# arguments are: filehandle(returned), host, port, callback function.
+#
+
+sub connectnb {
+ if($hasPOSIX == 0){
+ my $cb = $_[3];
+ $_[3] = undef;
+ my $ret = &connect(@_);
+ if($ret == 1){
+ &$cb($_[0], 0);
+ }
+ else {
+ &$cb($_[0], -1);
+ }
+ return $ret;
+
+ }
+ $_[0]=&newfh;
+ local($fh, $host, $port, $callback)=@_;
+ my @res = resolve($host, $port);
+ &tell("*\cbE\cb* Hostname `$host' not found"), return -1 if scalar(@res) < 5;
+
+ while (scalar(@res) >= 5) {
+ ($family, my ($socktype, $proto, $addr), undef, @res) = @res;
+ &print("*\cbE\cb* Out of file descriptors: $!"), return -2
+ unless socket($fh, $family, $socktype, $proto);
+
+ fcntl($fh, F_SETFL(), O_NONBLOCK());
+ &addwsel($fh, "nbconnecthandler", 0);
+ if(connect($fh, $addr)){
+ &$callback($fh, 0);
+ }
+ else {
+ if($! == EINPROGRESS()){
+ $nbconnectlist{$fh}{"callback"} = $callback;
+ }
+ else {
+ &print("*\cbI\cb* got other error $!");
+ return -1;
+ }
+ }
+ }
+ return 1;
+}
+
+sub listen {
+ $_[0]=&newfh;
+ local($fh, $port)=@_;
+ local($thisend);
+
+ &tell("\cbE\cb* first set your ipv4 hostname with /set LOCALHOST <hostname>"), return 0
+ unless (length $bindaddr);
+
+
+# XXX: don't use ipv6 for the time being as ipv6 and dcc don't mix
+# if ($ipv6) {
+ # XXX: substr() hack to avoid problems on some Linux systems
+# (undef, my $addr) = unpack_sockaddr_in6(substr($bindaddr, 0, 24));
+# $thisend = pack_sockaddr_in6($port, $addr);
+# } else {
+ (undef, my $addr) = unpack_sockaddr_in($bindaddr);
+ $thisend = pack_sockaddr_in($port, $addr);
+# }
+ &tell("*\cbE\cb* Out of file descriptors"), return 0
+ unless socket($fh, &AF_INET, &SOCK_STREAM, 0);
+ &tell("*\cbE\cb* Can't bind local socket!"), close $fh, return 0
+ unless bind($fh, $thisend);
+ &tell("*\cbE\cb* Can't listen to socket!"), close $fh, return
+ unless listen($fh, 5);
+ $ipv6=0;
+ return getsockname($fh);
+}
+
+sub accept {
+ $_[0]=&newfh;
+ return (accept($_[0], $_[1]), close($_[1]))[0];
+}
+
+sub bindtoserver {
+ @channels=(); $talkchannel='';
+ %mode=(); $umode=''; %limit=(); %haveops=(); %chankey=(); $away='';
+ $listmin=0; $listmax=100000; $listpat='';
+ @waituh=(); @douh=(); @erruh=(); $invited='';
+ &dostatus;
+ &tell("*** Connecting to $server, port $port...");
+ if($SSL == 1){
+ sleep 10, &bindtoserver if &connectSSL($S, $server, $port) < 0;
+ } else {
+ sleep 10, &bindtoserver if &connect($S, $server, $port) < 0;
+ }
+ $connected=1;
+ $server1=$server;
+ $port1=$port;
+ $pass1=$pass;
+ &sl("PASS $pass") if $pass;
+ &sl("USER $username blah blah :".$set{'IRCNAME'});
+ &sl("NICK $nick");
+ @channels=(); $talkchannel=''; %mode=(); $umode=''; %limit=();
+ %haveops=(); %chankey=();
+}
+
+sub gl {
+ if ($buffer{$_[0]} =~ /^([^\n\r]*)\r?\n\r?/) {
+ $buffer{$_[0]}=$';
+ $_=$1."\n";
+ return 1;
+ }
+ local($buf)='';
+ # &tell("About to sysread: $_[0]");
+ if (sysread($_[0], $buf, 4096)) {
+ $buffer{$_[0]}.=$buf;
+ if ($buffer{$_[0]} =~ /^([^\n\r]*)\r?\n\r?/) {
+ $buffer{$_[0]}=$';
+ $_=$1."\n";
+ return 1;
+ }
+ return '';
+ }
+ $_='';
+ return 1;
+}
+
+sub sl {
+ $logging && print LOG "<<".$_[0]."\n";
+ if(!print $S $_[0]."\n"){
+ &print("*\cbE\cb* Error writing to server: $!");
+ &tell("*\cbE\cb* Connection to server lost");
+ close($S);
+ delete $buffer{$S};
+ $connected=0;
+ &dohooks("disconnect");
+ &bindtoserver;
+ }
+ elsif (time-$floodtimer < 1){
+ select(undef, undef, undef, 0.5);
+ }
+ $floodtimer=time;
+}
+
+sub dostatus {
+ return unless $ssfe;
+ local($t, $s)=($talkchannel, " [sirc] ");
+ my($i);
+ for($i=0; $i<=$#channels; $i++){
+ $s = " [sirc] ";
+ $t = $channels[$i];
+ $t =~ tr/A-Z/a-z/;
+ $s.="*" if $umode =~ /o/;
+ $s.="\@" if $t && $haveops{$t};
+ $s.=$nick;
+ $s.=" (+$umode)" if $umode;
+ $s.=" [query: ${query}]" if $query;
+ $s.=" (away)" if $away;
+ if ($talkchannel ne '') {
+ $s.=" on $t (+$mode{$t})";
+ $s.=" <key: $chankey{$t}>" if $chankey{$t};
+ $s.=" <limit: $limit{$t}>" if $limit{$t};
+ }
+ &dohooks("status", $s);
+# $laststatus=$s, print "~${t}~`#ssfe#s$s\n" if $laststatus ne $s;
+ $laststatus=$s;
+ $logging && print LOG "** ~${t}~`#ssfe#s$s\n";
+ print "~${t}~`#ssfe#s$s\n";
+ }
+}
+
+$bold="\c[[1m";
+$underline="\c[[4m";
+$reverse="\c[[7m";
+$normal="\c[[m";
+$cls="\c[[H\c[[2J";
+
+sub enhance {
+ local($what)=@_;
+ $what =~ tr/\c@-\c^/@-^/;
+ return "\cv${what}\cv";
+}
+
+sub print {
+ local($skip, $what)=(0, @_);
+ &dohooks("print", $what);
+ return if $skip;
+ $what =~ s/\s+$//;
+ # thanks to Toy ([email protected]) for this translation
+ $what =~ tr/\x80-\xff/\x00-\x1f !cLxY|$_ca<\-\-R_o+23\'mp.,1o>123?AAAAAAACEEEEIIIIDNOOOOO*0UUUUYPBaaaaaaaceeeeiiiidnooooo:0uuuuypy/
+ if $set{"EIGHT_BIT"} ne 'on';
+ $logging && print LOG "-> " . $what."\n";
+ if ($raw_mode) {
+ print $what, "\n" || &exit;
+ } elsif ($ansi) {
+ # this is buggy if you combine effects
+ $what =~ s/([\ca\cc-\ch\cj-\cu\cw-\c^])/&enhance($1)/eg;
+ while ($what =~ /\cb/) {
+ ($what =~ s/\cb([^\cb]*)\cb/$bold$1$normal/) ||
+ $what =~ s/\cb/$bold/g;
+ }
+ while ($what =~ /\c_/) {
+ ($what =~ s/\c_([^\c_]*)\c_/$underline$1$normal/) ||
+ $what =~ s/\c_/$underline/g;
+ }
+ while ($what =~ /\cv/) {
+ ($what =~ s/\cv([^\cv]*)\cv/$reverse$1$normal/) ||
+ $what =~ s/\cv/$reverse/g;
+ }
+ print $what, $normal, "\n" || &exit;
+ } else {
+ $what =~ tr/\ca-\ch\cj-\c_//d;
+ print $what, "\n" || &exit;
+ }
+}
+
+sub tell {
+ $silent || &print;
+}
+
+sub dohooks {
+ $hooktype=shift;
+ local(@hl);
+ eval "\@hl=\@${hooktype}_hooks;";
+ foreach $h (@hl) {
+ eval { &$h(@_); };
+ $@ =~ s/\n$//, &tell("*\cbE\cb* error in $hooktype hook &$h: $@")
+ if $@ ne '';
+ }
+}
+
+sub dcerror {
+ local($fh, $n)=($_[0], $dcnick{$_[0]});
+ &dohooks("chat_disconnect", $n);
+ &tell("*\cbE\cb* DCC chat with $n lost");
+ &tell("~!dcc~Closing DCC CHAT with who: $n");
+ close($fh);
+ $n =~ tr/A-Z/a-z/;
+ delete $dcnick{$fh};
+ delete $dcvol{$n};
+ delete $dcfh{$n};
+ delete $buffer{$fh};
+}
+
+sub dgsclose {
+ local($sfh, $rfh, $type, $err)=@_;
+ &dohooks("dcc_disconnect", $dnick{$sfh}, $dfile{$rfh}, $dtransferred{$sfh},
+ time-$dstarttime{$rfh}, $rfh);
+ &tell("*\cbD\cb* DCC $type with $dnick{$sfh} ($dfile{$rfh}) terminated; $dtransferred{$sfh} bytes transferred in ".(time-$dstarttime{$rfh}). " seconds");
+ &tell("~!dcc~DCC $type terminated who: $dnick{$sfh} file: $dfile{$rfh} reason: $err");
+ close($sfh);
+ close($rfh);
+ delete $dgrfh{$sfh};
+ delete $dsrfh{$sfh};
+ delete $dfile{$rfh};
+ delete $dstarttime{$rfh};
+ delete $dtransferred{$sfh};
+ delete $dsoffset{$sfh};
+ delete $dsport{$sfh};
+ delete $dsresumedb{$sfh};
+ delete $dgxferadd{$sfh};
+ delete $dnick{$sfh};
+}
+
+sub msg {
+ local($towho, $what)=@_;
+ print "`#ssfe#t/m $towho \n" if $ssfe && !&eq($towho, $talkchannel);
+ if ($towho =~ s/^=//) {
+ local($n, $fh)=($towho);
+ $n =~ tr/A-Z/a-z/;
+ $fh=$dcfh{$n};
+ if ($fh) {
+ (print $fh $what."\n") || &dcerror($fh);
+ $dcvol{$n}+=length($what);
+ &dohooks("send_dcc_chat", $towho, $what);
+ &tell("~=${towho}~|\cb$towho\cb| $what"); #KSIRC MOD
+ } else {
+ &tell("*\cbE\cb* No active DCC chat with $towho");
+ }
+ } elsif ($connected>1) {
+ $what=substr($what, 0, 485);
+ &dohooks("send_text", $towho, $what);
+ if (&eq($towho, $talkchannel) && !$printchan) {
+ &tell("~${towho}~<${nick}> $what"); # KSIRC MOD
+ } elsif ($towho =~ /^[\&\#\+]/) {
+ &tell("~${towho}~<$nick> $what"); #KSIRC MOD
+ } else {
+ &tell("~${towho}~>${nick}< $what"); #KSIRC MOD
+ }
+ &sl("PRIVMSG $towho :$what");
+ } else {
+ &tell("*** You're not connected to a server");
+ }
+}
+
+sub say {
+ if ($query)
+ {
+ &msg($query, @_);
+ }
+ elsif ($talkchannel) {
+ &msg($talkchannel, @_);
+ } else {
+ &tell("*\cbE\cb* Not on a channel");
+ }
+}
+
+sub notice {
+ local($towho, $what)=@_;
+ $what=substr($what, 0, 485);
+ &dohooks("send_notice", $towho, $what);
+ &tell("~${towho}~-> -~n${towho}~n- $what");
+ &sl("NOTICE $towho :$what");
+}
+
+sub describe {
+ local($towho, $what)=@_;
+ $what=substr($what, 0, 480);
+ &dohooks("send_action", $towho, $what);
+ if (&eq($towho, $talkchannel) && !$printchan) {
+ &tell("~${towho}~* $nick $what"); # KSIRC MOD
+ } elsif ($towho =~ /^[\#\&\+]/) {
+ &tell("~${towho}~* $nick $what"); # KSIRC MOD
+ } else {
+ &tell("~${towho}~* $nick $what"); #KSIRC MOD
+# &tell("~${towho}~*-> \cb${towho}\cb: $nick $what"); #KSIRC MOD
+ }
+ &sl("PRIVMSG $towho :\caACTION".($what eq "" ? "" : " ").$what."\ca");
+}
+
+sub me {
+ if ($talkchannel) {
+ &describe($talkchannel, @_);
+ } else {
+ &tell("*\cbE\cb* Not on a channel");
+ }
+}
+
+sub yetonearg {
+ ($newarg, $args)=split(/ +/, $args, 2);
+ $args =~ s/^://;
+}
+
+sub getarg {
+ ($newarg, $args)=split(/ +/, $args, 2);
+}
+
+@weekdays=("Sun", "Mon", "Tue", "Wed", "Thu", "Fri", "Sat");
+@months=("Jan", "Feb", "Mar", "Apr", "May", "Jun", "Jul", "Aug", "Sep", "Oct",
+ "Nov", "Dec");
+
+sub date {
+ local($sec, $min, $hour, $mday, $mon, $year, $wday)=localtime($_[0]);
+ return sprintf("$weekdays[$wday] $months[$mon] $mday %.2d:%.2d:%.2d %d",
+ $hour, $min, $sec, $year+1900);
+}
+
+sub reply {
+ return if $set{"CTCP"} eq 'noreply';
+ if ($lastrep<time-10) {
+ $lastrep=time;
+ $nreps=1;
+ } else {
+ return if $nreps++>=2 && $set{"CTCP"} eq 'noflood';
+ }
+ &sl("NOTICE $who :\ca$_[0]\ca");
+}
+
+sub ctcp {
+ local($towho, $to, $what)=$_[0];
+ ($what, $args)=split(/ +/, $_[1], 2);
+ $what =~ tr/a-z/A-Z/;
+ &dohooks("ctcp", $towho, $what, $args);
+ return if $skip;
+ local($a)=$args;
+ $a && ($a=' '.$a);
+ $to = (&eq($towho, $nick) ? "you" : $towho);
+
+ &tell("~$to~*** $who$puh1 did a CTCP $what$a to $to")
+ unless $what =~ /^(ACTION|PING|DCC|VERSION)$/;
+ if ($what eq 'ACTION') {
+ &dohooks("action", $towho, $args);
+ if (&eq($towho, $nick)) {
+ &tell("~$who~* \cb${who}\cb$puh1 $args"); # KSIRC MOD
+ } elsif (&eq($towho, $talkchannel) && !$printchan) {
+ &tell("~$towho~* $who $args"); #KSIRC MOD
+ } else {
+ &tell("~$towho~* $who$puh2 $args"); #KSIRC MOD
+ }
+ } elsif ($what eq 'TIME') {
+ &reply("TIME ".&date(time));
+ } elsif ($what eq 'CLIENTINFO') {
+ &reply("CLIENTINFO ACTION, CLIENTINFO, DCC, ECHO, ERRMSG, FINGER, PING, TIME, USERINFO, VERSION");
+ } elsif ($what eq 'FINGER') {
+ &reply("FINGER ".$set{"FINGER"});
+ } elsif ($what eq 'USERINFO') {
+ &reply("USERINFO ".$set{"USERINFO"});
+ } elsif ($what eq 'VERSION') {
+ local($u)=$add_ons;
+ $u =~ s/^\+//;
+ $u =~ s/\+/ + /g;
+ $u=" -- using $u" if $u;
+ if($to eq 'you'){
+ &tell("~$who~*** $who$puh1 did a CTCP $what$a to $to")
+ }
+ else {
+ &tell("~$to~*** $who$puh1 did a CTCP $what$a to $to")
+ }
+ &reply("VERSION sirc $version, a \cbperl\cb client$u");
+ } elsif ($what eq 'PING') {
+ &reply("PING $args");
+ &tell("*** $who$puh1 did a CTCP PING to $to"); #KSIRC
+ } elsif ($what eq 'ECHO' || $what eq 'ERRMSG') {
+ &reply("$what $args");
+ } elsif ($what eq 'DCC') {
+ &getarg;
+ if ($newarg eq 'CHAT' || $newarg eq 'SEND' && !$restrict) {
+ local($dfile, $dhost, $dport, $dsize)=split(/ +/, $args, 4);
+ $dfile=$1 if $dfile =~ m|/([^/]*)$|;
+ $dfile =~ s/^\./_/;
+ if ($dhost==2130706433 || !$dport>1024 || $dhost !~ /^\d+$/ ||
+ $dport !~ /^\d+$/) {
+ &tell("*\cbE\cb* DCC $newarg ($dfile) from $who$puh1 rejected");
+ } elsif ($newarg eq 'CHAT' && grep (&eq($who, $dcwait{$_}),
+ keys(%dcwait))) {
+ &tell("*\cbD\cb* DCC chat already requested from $who, connecting...");
+ my ($wfh)=(grep(&eq($dcwait{$_}, $who), keys(%dcwait)));
+ my ($n, $fh)=$who;
+ delete $dcwait{$wfh};
+ close($wfh);
+ my $w = $who;
+ my $cb = sub {
+ my ($lfh, $lres) = @_;
+ if($lres != 0){
+ &tell("*\cbD\cb* DCC CHAT with $w failed: " . strerror($lres));
+ &tell("~!dcc~DCC CHAT failed who: $who reason: " . strerror($lres));
+ close($lfh);
+ return;
+ }
+ $dcnick{$lfh}=$w;
+ &tell("*\cbD\cb* DCC CHAT with $w established");
+ &tell("~!dcc~DCC CHAT established who: $w");
+ $n =~ tr/A-Z/a-z/;
+ $dcvol{$n}=0;
+ $dcfh{$n}=$lfh;
+ print "`#ssfe#t/m =$w \n" if $ssfe;
+ };
+ if(&connectnb($fh, $dhost, $dport, $cb) < 1) {
+ return;
+ }
+ } elsif ($newarg eq 'CHAT' && grep(&eq($who, $_), keys(%dcfh))) {
+ &tell("*\cbD\cb* DCC chat from $who$puh1 ignored (already established)");
+ } else {
+ #&tell("*\cbD\cb* DCC $newarg ($dfile) from $who$puh1 ".
+ # ($dsize ? "(size: $dsize) " : "")."[$dhost, $dport]");
+ my $ip = inet_ntoa(pack("N", $dhost));
+ if ($newarg eq 'CHAT') {
+ &tell("~!dcc~DCC CHAT OFFERED who: $who$puh1 ip: $ip port: $dport");
+ $dcoffered{$who}="$dhost $dport";
+ &dohooks("dcc_request", "CHAT", $dhost, $dport);
+ } else {
+ my $index = 1; # KSIRC MOD - Make the file name unique
+ UNIQ: {
+ foreach $i (keys(%dgoffered)) {
+ my($h, $p, $f) = split(/ /, $i);
+ if (&eq($f, $dfile)) {
+ $dfile =~ s/(.*)\.\d+$/$1/;
+ $dfile .= ".$index";
+ $index++;
+ redo UNIQ;
+ }
+ }
+ }
+ &tell("~!dcc~INBOUND DCC SEND who: $who$puh1 file: $dfile size: $dsize ip: $ip port: $dport");
+
+ $dgoffered{"$dhost $dport $dfile"}=$who;
+ &dohooks("dcc_request", "SEND", $dhost, $dport, $dfile, $dsize);
+ }
+ }
+ } else {
+ &tell("*** $who$puh1 did a CTCP ${what}$a to $to");
+ }
+ }
+}
+
+sub doset {
+ local($var, $val)=@_;
+ $var =~ tr/a-z/A-Z/;
+ $val="" unless defined($val);
+ if ($var eq 'PRINTUH') {
+ $set{$var}="all" if $val =~ /^(on|all)$/i;
+ $set{$var}="some" if $val =~ /^some$/i;
+ $set{$var}="none" if $val =~ /^(off|none)$/i;
+ } elsif ($var eq 'PRINTCHAN') {
+ $set{$var}="on", $printchan=1 if $val =~ /^on$/i;
+ $set{$var}="off", $printchan=0 if $val =~ /^off$/i;
+ } elsif ($var eq 'CTCP') {
+ $val =~ tr/A-Z/a-z/;
+ $set{$var}=$val if $val =~ /^(none|all)$/;
+ $set{$var}="noreply" if $val =~ /^(noreply|off)$/;
+ $set{$var}="noflood" if $val =~ /^(noflood|on)$/;
+ } elsif ($var eq 'SENDAHEAD') {
+ $set{$var}=$val if $val =~ /^\d+$/ && $val<=65536;
+ } elsif ($var eq 'USERINFO') {
+ $set{$var}=$val;
+ } elsif ($var eq 'FINGER') {
+ $set{$var}=$val;
+ } elsif ($var eq 'IRCNAME') {
+ $set{$var}=$val;
+ } elsif ($var eq 'EIGHT_BIT') {
+ $val =~ tr/A-Z/a-z/;
+ $set{$var}=$val if $val =~ /^(on|off)$/;
+ } elsif ($var eq 'LOCALHOST') {
+ &restrict || return;
+ # IPV6: DCC is always ipv4 :(
+ local($ad) = (&resolve($val, 0, &AF_INET))[3];
+ $set{$var}=$val, $bindaddr=$ad if $ad;
+ } elsif ($var eq 'LOADPATH') {
+ @loadpath=split(/:/, $val);
+ foreach (@loadpath) {
+ &tilde($_);
+ }
+ $set{$var}=join(":", @loadpath);
+ } elsif ($var eq 'CTRL_T') {
+ $set{$var}=$val;
+ print "`#ssfe#T$val\n" if $ssfe;
+ } elsif ($var eq 'LOGFILE') {
+ &restrict || return;
+ &tilde($val);
+ $logfile=$set{$var}=$val;
+ } elsif ($var eq 'LOG') {
+ &restrict || return;
+ if ($val =~ /^on$/i) {
+ $logging && close LOG;
+ if (open(LOG,
+ ($logfile =~ /\.gz$/ ? "| gzip >> $logfile" : ">> $logfile"))) {
+ $logging=1;
+ $set{$var}="on";
+ select(LOG); $|=1; select(STDOUT);
+ print LOG "*\cbL\cb* IRC log started on ".&date(time)."\n";
+ } else {
+ $logging='';
+ $set{$var}="off";
+ &tell("*\cbE\cb* Can't write to logfile $logfile");
+ }
+ } elsif ($val =~ /^off$/i) {
+ print LOG "*\cbL\cb* Log ended on ".&date(time)."\n", close LOG
+ if $logging;
+ $logging='';
+ $set{$var}="off";
+ }
+ } elsif (defined($sets{$var})) {
+ local($f)=$sets{$var};
+ eval { &$f($val); };
+ $@ =~ s/\n$//, &tell("*\cbE\cb* error in SET $var hook: $@") if $@ ne '';
+ }
+}
+
+sub ctcpreply {
+ local($ctcp, $rest)=split(/ +/, $_[1], 2);
+ $ctcp =~ tr/a-z/A-Z/;
+ &dohooks("ctcp_reply", $_[0], $ctcp, $rest);
+ $rest=(time-$rest)." seconds" if $ctcp eq 'PING';
+ if (&eq($_[0], $nick)) {
+ &tell("*** CTCP $ctcp reply from $who$puh1: $rest");
+ } else {
+ &tell("*** CTCP $ctcp reply to $_[0] from $who$puh2: $rest");
+ }
+}
+
+sub load {
+ local($f)=@_;
+ &tilde($f);
+ if ($f !~ /\//) {
+ foreach (@loadpath) {
+ $f="$_/$f", last if -f "$_/$f";
+ $f="$_/${f}.pl", last if $f !~ /\.pl$/ && -f "$_/${f}.pl";
+ }
+ } else {
+ $f.=".pl" if -f "${f}.pl" && !-f $f;
+ }
+ if ($f =~ /\// && -f $f) {
+ do $f;
+ $@ =~ s/\n$//, &tell("*\cbE\cb* Load error in $f: $@") if $@ ne '';
+ } else {
+ &tell("*\cbE\cb* $f: File not found");
+ }
+}
+
+sub restrict {
+ &tell("*\cbE\cb* Command not available"), return 0 if $restrict;
+ 1;
+}
+
+sub dosplat {
+ $args =~ s/^\s*\*($|\s)/${talkchannel}${1}/ if $talkchannel;
+}
+
+sub expand {
+ if ($_[0] eq '$') {
+ return '$';
+ } elsif ($_[0] =~ /^(\d+)$/) {
+ return (split(/ +/, $args))[$1];
+ } elsif ($_[0] =~ /^(\d+)-$/) {
+ return (split(/ +/, $args, 1+$1))[$1];
+ } else {
+ return eval "\$$_[0]";
+ }
+}
+
+$recdepth=0;
+$maxrecursion=20;
+
+sub docommand {
+ local($line)=@_;
+ local($recdepth)=$recdepth+1;
+ &print("*\cbE\cb* Max recursion exceeded!"), return
+ if $recdepth > $maxrecursion;
+ local($noalias)=($line =~ s/^\///);
+ local($silent)=1 if $line =~ s/^\^//;
+ local($cmd, $args)=split(/ +/, $line, 2);
+ $cmd =~ tr/a-z/A-Z/;
+ if (!$noalias && defined($aliases{$cmd})) {
+ $line=$aliases{$cmd};
+ $line.=($args ne '' ? " ".$args : "")
+ unless ($line =~ s/\$(\$|\d+-?|\w+)/&expand($1)/eg);
+ $line =~ s/^\///;
+ $noalias=1 if $line =~ s/^\///;
+ $silent=1 if $line =~ s/^\^//;
+ ($cmd, $args)=split(/ +/, $line, 2);
+ $cmd =~ tr/a-z/A-Z/;
+ }
+ if (!$noalias && defined($cmds{$cmd})) {
+ eval $cmds{$cmd};
+ $@ =~ s/\n$//, &tell("*\cbE\cb* error in command $cmd: $@") if $@ ne '';
+ } elsif ($cmd eq 'ALIAS') {
+ &getarg;
+ if ($newarg =~ /^-/) {
+ local($a)=$';
+ if ($a eq '') {
+ %aliases=();
+ &tell("*** All aliases removed");
+ } else {
+ $a =~ tr/a-z/A-Z/;
+ delete $aliases{$a};
+ &tell("*** Alias $a removed");
+ }
+ } elsif ($newarg ne '') {
+ $newarg =~ tr/a-z/A-Z/;
+ if ($args ne '') {
+ $aliases{$newarg}=$args;
+ &tell("*** $newarg aliased to $args");
+ } else {
+ if (defined($aliases{$newarg})) {
+ &tell("*** $newarg is aliased to: $aliases{$newarg}");
+ } else {
+ &tell("*** $newarg: no such alias");
+ }
+ }
+ } else {
+ foreach $a (sort(keys(%aliases))) {
+ &tell("*** $a is aliased to $aliases{$a}");
+ }
+ }
+ } elsif ($cmd eq 'SET') {
+ &getarg;
+ local($s)=$newarg;
+ $s =~ tr/a-z/A-Z/;
+ if ($s =~ s/^-//) {
+ &tell("*** No such variable $s"), return unless defined($set{$s});
+ &doset($s, "");
+ &tell("*** $s is ".($set{$s} ne '' ? "set to $set{$s}" : "unset"));
+ } elsif ($s ne '') {
+ &tell("*** No such variable $s"), return unless defined($set{$s});
+ &doset($s, $args) if $args ne '';
+ &tell("*** $s is ".($set{$s} ne '' ? "set to $set{$s}" : "unset"));
+ } else {
+ foreach $s (sort(keys (%set))) {
+ &tell("*** $s is ".($set{$s} ne '' ? "set to $set{$s}" : "unset"));
+ }
+ }
+ } elsif ($cmd eq 'NOTIFY' || $cmd eq 'N') {
+ if ($args eq '-') {
+ &tell("*** Notify list cleared");
+ my($value);
+ while(($_, $value) = each %notify){ # Remove all nicks
+ &tell("*\cb(\cb* Signoff by $_ detected!"); # KSIRC MOD
+ }
+ %notify=();
+ } elsif ($args eq '') {
+ local($l)='';
+ foreach (grep($notify{$_}, keys %notify)) {
+ &tell("*** Currently present: $l"), $l='' if length($l)>450;
+ &tell("*\cb)\cb* Signon by $_ detected!"); # KSIRC MOD
+ $l.=$_." ";
+ }
+ $l && &tell("*** Currently present: $l");
+ $l='';
+ foreach (grep(!$notify{$_}, keys %notify)) {
+ &tell("*** Currently absent: $l"), $l='' if length($l)>450;
+ &tell("*\cb(\cb* Signoff by $_ detected!"); # KSIRC MOD
+ $l.=$_." ";
+ }
+ $l && &tell("*** Currently absent: $l");
+ } else {
+ local($w, $n);
+ foreach $w (split(/ +/, $args)) {
+ if ($w =~ s/^-//) {
+ ($n)=(grep(&eq($_, $w), keys(%notify)), '');
+ $n ne '' && delete $notify{$n};
+ &tell("*** $w removed from notify list");
+ &tell("*\cb(\cb* Signoff by $w detected!"); # KSIRC MOD
+ } else {
+ $notify{$w}='0';
+ &tell("*** $w added to notify list");
+ $newisons=1;
+ }
+ }
+ }
+ } elsif ($cmd eq 'IGNORE' || $cmd eq 'IG') {
+ &getarg;
+ if ($newarg eq '-') {
+ @ignore=();
+ &tell("*** Ignore list cleared");
+ } elsif ($newarg eq '') {
+ local($p);
+ &tell("*** You're ignoring:");
+ foreach (@ignore) {
+ $p=$_;
+ $p =~ s/\\//g;
+ $p =~ s/\.\*/*/g;
+ &tell("*** $p");
+ }
+ } else {
+ local($d, $p)=('');
+ $d=1 if $newarg =~ s/^-//;
+ if ($newarg =~ /\!.*\@/) {
+ } elsif ($newarg !~ /[\@\!]/) {
+ $newarg.="!*";
+ } elsif ($newarg =~ /\@/) {
+ $newarg="*!".$newarg;
+ } else {
+ $newarg.="\@*";
+ }
+ $p=$newarg;
+ $newarg =~ s/([^\\])\./$1\\./g;
+ $newarg =~ s/\*/\.\*/g;
+ $newarg =~ s/([^\.\*\\\w])/\\$1/g;
+ if ($d) {
+ &tell("*** Removing $p from the ignore list");
+ @ignore=grep(!&eq($_, $newarg), @ignore);
+ } else {
+ &tell("*** Ignoring $p ... what a relief!");
+ push(@ignore, $newarg);
+ }
+ }
+ } elsif ($cmd eq 'ECHO') {
+ &print($args);
+ } elsif ($cmd eq 'CLEAR' || $cmd eq 'CL') {
+ print $cls if $ansi;
+ print "`#ssfe#l\n" if $ssfe;
+ } elsif ($cmd eq 'EVAL') {
+ &restrict || return;
+ eval ($args);
+ $@ =~ s/\n$//, &tell("*\cbE\cb* eval error: $@") if $@ ne '';
+ } elsif ($cmd eq 'HELP') {
+ &tell("*\cbH\cb* Help not available"), return unless @help;
+ $args='main' if $args =~ /^\s*$/;
+ $args =~ s/ *$//;
+ local($found)='';
+ foreach (@help) {
+ if (/^\@/) {
+ last if $found;
+ if (&eq($_, "\@$args")) {
+ $found=1;
+ &tell("*\cbH\cb* Help on $args") if $args ne 'main';
+ }
+ } else {
+ &tell("*\cbH\cb* $_") if $found;
+ }
+ }
+ &tell("*\cbH\cb* Unknown help topic; try /help") unless $found;
+ } elsif ($cmd eq 'LOAD') {
+ &restrict || return;
+ &getarg;
+ &tell("*\cbE\cb* Yeah, but what?"), return if $newarg eq '';
+ &load($newarg);
+ } elsif ($cmd eq 'VERSION') {
+ &tell("*** \cbsirc\cb version $version, written in \cbperl\cb by \cborabidoo\cb");
+ $_=$add_ons;
+ s/^\+//;
+ s/\+/, /g;
+ &tell("*** add-ons: $_") if $_;
+ $connected==2 && &sl("VERSION $args");
+ } elsif ($cmd eq 'CD') {
+ &restrict || return;
+ &getarg;
+ if ($newarg ne '') {
+ &tilde($newarg);
+ chdir($newarg) || &tell("*\cbE\cb* Can't chdir to $newarg");
+ }
+ local($cwd); chop($cwd=`pwd`);
+ &tell("*** Current directory is $cwd");
+ } elsif ($cmd eq 'SYSTEM') {
+ &restrict || return;
+ system($args);
+ } elsif ($cmd eq 'BYE' || $cmd eq 'QUIT' || $cmd eq 'EXIT' ||
+ $cmd eq 'SIGNOFF') {
+ $args || ($args="using sirc version $version$add_ons");
+ &dohooks("quit");
+ &sl("QUIT :$args") if $connected;
+ &exit;
+ } elsif ($cmd eq 'SERVER') {
+ $args=$1 if $args =~ /^\s*(.*)\s*$/;
+ $args="$server0:$port0:$pass0" if $args eq '0';
+ $args="$server1:$port1:$pass1" if $args eq '1';
+ if ($args eq '') {
+ &tell($connected ? "*** Your current server is $server" :
+ "*** You're not connected to a server");
+ } else {
+ ($server, $port, $pass)=split(/[\s:]+/, $args);
+ $server=$', $nick=$1 if $server =~ /^([^\@]+)\@/;
+ $port || ($port=$port0);
+ &sl("QUIT :changing servers"), close $S, delete $buffer{$S} if $connected;
+ $connected=0;
+ }
+ } elsif ($cmd eq 'MSG' || $cmd eq 'M') {
+ &dosplat;
+ if ($args) {
+ ($newarg, $args)=split(/ /, $args, 2);
+ &msg($newarg, $args);
+ } else {
+ &tell("*\cbE\cb* You must specify a nick or channel!");
+ }
+ } elsif ($cmd eq 'QUERY' || $cmd eq 'Q') {
+ if ($args) {
+ $args =~ s/\s+$//;
+ $query=$args;
+ &tell("*** Starting conversation with $query");
+ &dostatus;
+ } elsif ($query) {
+ &tell("*** Ending conversation with $query");
+ $query='';
+ &dostatus;
+ } else {
+ &tell("*** You aren't querying anyone :p");
+ }
+ } elsif ($cmd eq 'DCC') {
+ &getarg;
+ if ($newarg =~ /^chat$/i) {
+ &getarg;
+ local($n)=grep(&eq($newarg, $_), keys(%dcoffered));
+ if ($n) {
+ local($dcadr, $dcport)=split(/ +/, $dcoffered{$n});
+ local($fh);
+ delete $dcoffered{$n};
+ my $w = $n;
+ my $cb = sub {
+ my ($lfh, $lres) = @_;
+ if($lres != 0){
+ &tell("*\cbD\cb* DCC CHAT with $w failed: " . strerror($lres));
+ &tell("~!dcc~DCC CHAT failed who: $w reason: " . strerror($lres));
+ close($lfh);
+ return;
+ }
+ $dcnick{$lfh}=$w;
+ &tell("*\cbD\cb* DCC CHAT with $w established");
+ &tell("~!dcc~DCC CHAT established who: $w");
+ print "`#ssfe#t/m =$w \n" if $ssfe;
+ my $n = $w;
+ $n =~ tr/A-Z/a-z/;
+ $dcvol{$n}=0;
+ $dcfh{$n}=$fh;
+ };
+ if(&connectnb($fh, $dcadr, $dcport, $cb) < 1){
+ return;
+ }
+ } elsif (grep (&eq($newarg, $dcwait{$_}), keys(%dcwait))) {
+ &tell("*\cbE\cb* DCC CHAT request to $newarg already sent");
+ } elsif (grep(&eq($newarg, $dcnick{$_}), keys(%dcnick))) {
+ &tell("*\cbE\cb* DCC CHAT with $newarg already established");
+ } elsif ($newarg) {
+ &tell("*** You're not connected to a server"), return if $connected<2;
+ &tell("*** Don't be antisocial!"), return if &eq($newarg, $nick);
+ local($mynumber, $myport, $fh);
+ my $sockaddr = &listen($fh) or return;
+ if ($ipv6) {
+ # XXX: substr is used in order to avoid dying on Linux with older
+ # glibc that lacks the scope field from sockaddr_in6 but the kernel
+ # has it and returns it from getsockname()
+ ($myport, undef) = unpack_sockaddr_in6(substr($sockaddr, 0, 24));
+ $mynumber = '0';
+ } else {
+ ($myport, $mynumber) = unpack_sockaddr_in(&listen($fh)) or return;
+ $mynumber = unpack("N", $mynumber);
+ }
+ $dcwait{$fh}=$newarg;
+ &sl("PRIVMSG $newarg :\caDCC CHAT chat $mynumber $myport\ca");
+ &dohooks("send_ctcp", $newarg, "DCC CHAT chat $mynumber $myport");
+ &tell("*\cbD\cb* Sent DCC CHAT request to $newarg");
+ &tell("~!dcc~DCC CHAT SEND who: $newarg");
+ } else {
+ &tell("*** I need a nick");
+ }
+ } elsif ($newarg =~ /^rchat$/i) {
+ &getarg;
+ local($n)=$newarg;
+ &getarg;
+ if ($newarg) {
+ local($fh)=grep(&eq($dcnick{$_}, $n), keys(%dcnick));
+ if( ! $fh){
+ &tell("*\cbE\cb* No DCC CHAT established with $n");
+ &tell("~!dcc~No DCC CHAT established who: $n");
+ return;
+ }
+ &tell("*\cbE\cb* DCC CHAT already established with $newarg"), return
+ if grep(&eq($dcnick{$_}, $newarg), keys(%dcnick));
+ &tell("*\cbD\cb* DCC CHAT with $n renamed to $newarg");
+ &tell("~!dcc~DCC CHAT renamed who: $n to: $newarg");
+ $dcnick{$fh}=$newarg;
+ $n =~ tr/A-Z/a-z/;
+ $newarg =~ tr/A-Z/a-z/;
+ $dcfh{$newarg}=$dcfh{$n};
+ $dcvol{$newarg}=$dcvol{$n};
+ delete $dcfh{$n};
+ delete $dcvol{$n};
+ } else {
+ &tell("*** I need *two* nicks");
+ }
+ } elsif ($newarg =~ /^close$/i) {
+ &getarg;
+ if ($newarg =~ /^chat$/i) {
+ &getarg;
+ local($n)=$newarg;
+ $newarg =~ tr/A-Z/a-z/;
+ local($fh)=$dcfh{$newarg};
+ local($nn)=(grep(&eq($_, $newarg), keys(%dcoffered)));
+ if ($nn) {
+ &tell("*\cbD\cb* Forgetting offered DCC CHAT from $nn");
+ &tell("~!dcc~Closing DCC CHAT who: $nn");
+ delete $dcoffered{$nn};
+ if($no_reject == 0){
+ $who = $nn;
+ &reply("DCC REJECT CHAT chat");
+ }
+ $no_reject = 0;
+ } elsif ($fh) {
+ &dohooks("chat_disconnect", $n);
+ &tell("*\cbD\cb* Closing DCC CHAT connection with $n");
+ &tell("~!dcc~Closing DCC CHAT who: $n");
+ close($fh);
+ delete $dcnick{$fh};
+ delete $dcvol{$newarg};
+ delete $dcfh{$newarg};
+ delete $buffer{$fh};
+ if($no_reject == 0){
+ $who = $n;
+ &reply("DCC REJECT CHAT chat");
+ }
+ $no_reject = 0;
+
+ } elsif (($fh)=grep(&eq($dcwait{$_}, $n), keys (%dcwait)), $fh) {
+ close($fh);
+ delete $dcwait{$fh};
+ &tell("*\cbD\cb* Closing listening DCC CHAT with $n");
+ &tell("~!dcc~Closing DCC CHAT who: $n");
+ if($no_reject == 0){
+ $who = $n;
+ &reply("DCC REJECT CHAT chat");
+ }
+ $no_reject = 0;
+ } else {
+ if($n){
+ &tell("*\cbE\cb* No DCC CHAT connection with $n");
+ &tell("~!dcc~No DCC CHAT connection who: $n");
+ }
+ }
+ } elsif ($newarg =~ /^get$/i) {
+ &getarg;
+ my $arg = $newarg;
+ local($found)='';
+ foreach $i (keys(%dgoffered)) {
+ if (&eq($dgoffered{$i}, $newarg) && (!$args ||
+ &eq($args, (split(/ +/, $i))[2]))) {
+ &tell("*\cbE\cb* Forgetting pending DCC GET from $newarg");
+ my($host, $port, $file) = split(/ /, $i);
+ &tell("~!dcc~Closing DCC GET connection with who: $newarg file: $file"); # KSIRC MOD
+ delete $dgoffered{$i};
+ $found=1;
+ if($no_reject == 0){
+ $who = $newarg;
+ &reply("DCC REJECT GET $file");
+ }
+ $no_reject = 0;
+ }
+ }
+ foreach $sfh (grep(&eq($newarg, $dnick{$_}), keys(%dnick))) {
+ if (!$found && $dgrfh{$sfh}) {
+ local($fh)=$dgrfh{$sfh};
+ my($file)=$dfile{$fh};
+ next if $args && ($args ne $dfile{$fh});
+ &dohooks("dcc_disconnect", $dnick{$sfh}, $dfile{$fh},
+ $dtransferred{$sfh}, time-$dstarttime{$fh}, $fh);
+
+ &tell("*\cbE\cb* Closing DCC GET connection with: $newarg ($file)"); # KSIRC MOD
+ &tell("~!dcc~Closing DCC GET connection with who: $newarg file: $file"); # KSIRC MOD
+ $found=1;
+ close $sfh;
+ close $fh;
+ delete $dgrfh{$sfh};
+ delete $dfile{$fh};
+ delete $dstarttime{$fh};
+ delete $dtransferred{$sfh};
+ delete $dgxferadd{$sfh};
+ delete $dnick{$sfh};
+ if($no_reject == 0){
+ $who = $newarg;
+ &reply("DCC REJECT GET $file");
+ }
+ $no_reject = 0;
+ }
+ }
+ if( ! $found){
+ &tell("*\cbE\cb* No DCC GET connection with $newarg for $arg");
+ &tell("~!dcc~No DCC GET connection who: $newarg file: $arg");
+ }
+ } elsif ($newarg =~ /^send$/i) {
+ &getarg;
+ local($n, $found, $fh)=($newarg, '');
+ &getarg;
+ my $arg = $newarg;
+ $newarg =~ s/(\W)/\\$1/g;
+ foreach $sfh (keys(%dswait), keys(%dsrfh)) {
+ next unless &eq($dnick{$sfh}, $n);
+ $fh=$dswait{$sfh} || $dsrfh{$sfh} || next;
+ if ($newarg eq '' || $dfile{$fh} =~ /^${newarg}$/ ||
+ $dfile{$fh} =~ /\/${newarg}$/) {
+ #&tell("*\cbD\cb* DCC SEND connection with $n closed");
+ #my($file)=$dfile{$fh};
+ #&tell("~!dcc~Closing DCC SEND connection with who: $n file: $file"); # KSIRC MOD
+ #&dohooks("dcc_disconnect", $dnick{$sfh}, $dfile{$fh},
+ # $dtransferred{$sfh}, time-$dstarttime{$fh}, $fh);
+ #close($sfh);
+ #close($fh);
+ #delete $dswait{$sfh};
+ #delete $dsrfh{$sfh};
+ #delete $dfile{$fh};
+ #delete $dstarttime{$fh};
+ #delete $dtransferred{$sfh};
+ #delete $dsoffset{$sfh};
+ #delete $dsport{$sfh};
+ #delete $dsresumedb{$sfh};
+ #delete $dgxferadd{$sfh};
+ #delete $dnick{$sfh};
+ if($no_reject == 0){
+ $who = $n;
+ &reply("DCC REJECT SEND $dfile{$fh}");
+ }
+ $no_reject = 0;
+
+ if($dstarttime{$fh} == undef) {
+ $dstarttime{$fh} = time;
+ }
+ &dgsclose($sfh, $fh, "SEND", "CLOSE");
+
+ $found=1;
+ }
+ }
+ if(!$found){
+ &tell("*\cbE\cb* No DCC SEND connection with $n for $arg");
+ &tell("~!dcc~No DCC SEND connection with who: $n file: $arg");
+ }
+ } else {
+ &tell("*\cbE\cb* Unknown DCC type");
+ }
+ } elsif ($newarg =~ /^rename$/i) {
+ local($found, $n);
+ &getarg;
+ $n=$newarg;
+ &getarg;
+ $args=$newarg, $newarg='' if $args eq '';
+ &tell("*\cbE\cb* I need a filename :p"), return if $args eq '';
+ &tilde($args);
+ foreach $i (keys(%dgoffered)) {
+ if (&eq($dgoffered{$i}, $n) && (!$newarg ||
+ &eq($newarg, (split(/ +/, $i))[2]))) {
+ local($m, $p, $f)=split(/ +/, $i);
+ delete $dgoffered{$i};
+ $dgoffered{"$m $p $args"}=$n;
+ &tell("*\cbD\cb* Renaming \"$f\" (offered by $n) to \"$args\"");
+ $found=1;
+ last;
+ }
+ }
+ &tell("*\cbE\cb* No such file offered by $n") unless $found;
+ } elsif ($newarg =~ /^get$/i) {
+ &getarg;
+ local($n)=grep((&eq($newarg, $dgoffered{$_}) && (!$args ||
+ &eq($args, (split(/ +/, $_))[2]))),
+ keys(%dgoffered));
+ if ($n) {
+ my($dgadr, $dgport, $file)=split(/ +/, $n);
+ my($fh, $sfh);
+ my $offset = 0;
+ $n=(delete $dgoffered{$n});
+ $fh=&newfh;
+ if($dgresume{$dgport} && $dgresume{$dgport}{"GotReply"}){
+ &print("*\cbE\cb* Can't write to file $file"), return unless open($fh, ">> $file");
+ seek($fh, $dgresume{$dgport}{"pos"}, SEEK_SET);
+ $offset = $dgresume{$dgport}{"pos"};
+ delete $dgresume{$dgport};
+ }
+ else {
+ &print("*\cbE\cb* Can't write to file $file"), return unless open($fh, "> $file");
+ }
+ my $who = $n;
+ my $cb = sub {
+ my ($lfh, $lres) = @_;
+ if($lres != 0){
+ &tell("*\cbD\cb* DCC GET connection with $who ($file) failed: " . strerror($lres));
+ &tell("~!dcc~DCC GET failed who: $who file: $file reason: " . strerror($lres));
+ close($lfh);
+ return;
+ }
+ $dgrfh{$lfh}=$fh;
+ $dnick{$lfh}=$who;
+ $dfile{$fh}=$file;
+ $dstarttime{$fh}=time;
+ $dtransferred{$lfh}=0;
+ $dgxferadd{$lfh}=$offset;
+ &tell("*\cbD\cb* DCC GET connection with $who established");
+ &tell("~!dcc~DCC GET established who: $who file: $file");
+ &dohooks("dcc_get", $who, $file, $fh);
+ };
+ if(&connectnb($sfh, $dgadr, $dgport, $cb) < 1){
+ return;
+ }
+ } else {
+ if ($newarg) {
+ &tell("*\cbE\cb* No pending DCC GET from $newarg");
+ } else {
+ &tell("*\cbE\cb* Uhm, who from?");
+ }
+ }
+ } elsif ($newarg =~ /^list$/i || $newarg eq '') {
+ &tell("*\cbD\cb* List of DCC connections:");
+ foreach $n (keys(%dcfh)) {
+ &tell("*\cbD\cb* Established DCC CHAT with $n ($dcvol{$n} bytes)");
+ }
+ foreach $n (keys(%dcoffered)) {
+ my ($pip, $port) = split(/ /, $dcoffered{$n});
+ my $ip = inet_ntoa(pack("N", $pip));
+ &tell("*\cbD\cb* DCC CHAT offered by $n ($ip:$port)");
+ }
+ foreach $f (keys(%dcwait)) {
+ &tell("*\cbD\cb* DCC CHAT offered to $dcwait{$f}");
+ }
+ foreach $i (keys(%dgoffered)) {
+ my ($pip, $port, $file) = split(/ /, $i);
+ my $ip = inet_ntoa(pack("N", $pip));
+ &tell("*\cbD\cb* DCC GET \"$file\" ($ip:$port) offered by $dgoffered{$i}");
+ }
+ foreach $s (keys(%dgrfh)) {
+ local($f)=$dgrfh{$s};
+ &tell("*\cbD\cb* DCC GET \"$dfile{$f}\" established with $dnick{$s}, $dtransferred{$s} bytes read in ".(time-$dstarttime{$f})." seconds.");
+ }
+ foreach $s (keys(%dswait)) {
+ local($f)=$dswait{$s};
+ &tell("*\cbD\cb* DCC SEND \"$dfile{$f}\" offered to $dnick{$s}");
+ }
+ foreach $s (keys(%dsrfh)) {
+ local($f)=$dsrfh{$s};
+ &tell("*\cbD\cb* DCC SEND \"$dfile{$f}\" established with $dnick{$s}, $dtransferred{$s} bytes sent in ".(time-$dstarttime{$f})." seconds.");
+ }
+ } elsif ($newarg =~ /^send$/i) {
+ &tell("*** You're not connected to a server"), return if $connected<2;
+ &restrict || return;
+ local(($n),($f)) = $args =~ /^(.+?) (.+)/;
+ local($tf, $mynumber, $sz, $fh, $myport, $lfh)=($f);
+ &tilde($f);
+ while (my($fh, $ni) = each %dnick ) {
+ if(&eq($n, $ni)){
+ my $lfh = $dswait{$fh};
+ if(&eq($dfile{$lfh}, $f)){
+ &tell("*\cbE\cb* DCC Send already pending of $f to $n");
+ return;
+ }
+ if($dsrfh{$fh}){
+ &tell("*\cbE\cb* DCC Send already in progress $f to $n");
+ return;
+
+ }
+ }
+ }
+ $fh=&newfh;
+ &tell("*\cbE\cb* Can't open file $f"), return unless open($fh, "<$f");
+ my $sockaddr = &listen($lfh) or (close $fh, return);
+ if ($ipv6) {
+ # XXX: substr is used in order to avoid dying on Linux with older
+ # glibc that lacks the scope field from sockaddr_in6 but the kernel
+ # has it and returns it from getsockname()
+ ($myport, undef) = unpack_sockaddr_in6(substr($sockaddr, 0, 24));
+ $mynumber = 0;
+ } else {
+ ($myport, $mynumber) = unpack_sockaddr_in($sockaddr);
+ $mynumber = unpack("N", $mynumber);
+ }
+ $dswait{$lfh}=$fh;
+ $tf=$1 if $f =~ m|/([^/]*)$|;
+ $sz=(-s $f);
+ $tf =~ s/ /_/g; # we have to convert spaces in the filename to underscores
+ &sl("PRIVMSG $n :\caDCC SEND $tf $mynumber $myport $sz\ca");
+ &dohooks("send_ctcp", $n, "DCC SEND $tf $mynumber $myport $sz");
+ &dohooks("dcc_send", $n, $f, $sz, $fh);
+ #&tell("*\cbD\cb* Sent DCC SEND request to $n ($f,$sz)");
+ &tell("~!dcc~Sent DCC SEND request to who: $n file: $f size: $sz");
+ $dfile{$fh}=$f;
+ $dswait{$lfh}=$fh;
+ $dnick{$lfh}=$n;
+ $dsport{$lfh}=$myport;
+ $dsoffset{$lfh}=0;
+ } else {
+ &tell("*** I can \"only\" do DCC CHAT, RCHAT, GET, SEND, CLOSE, RENAME and LIST, *sheesh*");
+ }
+ } elsif ($cmd eq 'QUOTE') { #KSIRC MOD
+ $args ne '' && &sl($args); #Allow this even if not connected to talk to proxies
+ } elsif ($connected<2) {
+ &tell("*** You're not connected to a server");
+ } elsif ($cmd eq 'AWAY') {
+ &sl($args ? "AWAY :$args" : "AWAY");
+ my $oldchannel = $talkchannel;
+ if ( $publicAway == 1 ) {
+ foreach $talkchannel (@channels) {
+ &me($args ? "is away: $args" : "is back");
+ }
+ }
+ $talkchannel = $oldchannel;
+ } elsif ($cmd eq 'NEXT') {
+ if ($#channels>0) {
+ $talkchannel=shift(@channels);
+ push(@channels, $talkchannel);
+ !$ssfe && &tell("*** Talking to $talkchannel now");
+ &dostatus;
+ }
+ } elsif ($cmd eq 'SAY' || $cmd eq '') {
+ &say($args);
+ } elsif ($cmd eq 'NOTICE' || $cmd eq 'NO') {
+ &dosplat;
+ if ($args) {
+ ($newarg, $args)=split(/ /, $args, 2);
+ &notice($newarg, $args);
+ } else {
+ &tell("*\cbE\cb* You must specify a nick or channel!");
+ }
+ } elsif ($cmd eq 'DESCRIBE' || $cmd eq 'DE') {
+ &dosplat;
+ if ($args) {
+ ($newarg, $args)=split(/ /, $args, 2);
+ &describe($newarg, $args);
+ } else {
+ &tell("*\cbE\cb* You must specify a nick or channel!");
+ }
+ } elsif ($cmd eq 'KICK' || $cmd eq 'K') {
+ &dosplat;
+ &getarg;
+ local($c)=$talkchannel;
+ if ($newarg =~ /^[\#\&\+]/) {
+ $c=$newarg;
+ &getarg;
+ }
+ if ($newarg) {
+ $args || ($args=$nick);
+ &sl("KICK $c $newarg :$args");
+ } else {
+ &tell("*\cbE\cb* You must specify a nick!");
+ }
+ } elsif ($cmd eq 'DISCONNECT' || $cmd eq 'DIS') {
+ &tell("*** Disconnecting from $server");
+ close($S);
+ delete $buffer{$S};
+ $connected=0;
+ &dohooks("disconnect");
+ &bindtoserver;
+
+ } elsif ($cmd eq 'INVITE' || $cmd eq 'INV' || $cmd eq 'I') {
+ local(@ns)=split(/ +/, $args);
+ local($l, $c)=(pop(@ns), $talkchannel);
+ if ($l =~ /^[\#\&\+]/) {
+ $c=$l;
+ } else {
+ $l && push(@ns, $l);
+ }
+ foreach (@ns) {
+ &sl("INVITE $_ $c");
+ }
+ } elsif ($cmd eq 'CTCP') {
+ &dosplat;
+ if ($args) {
+ &getarg;
+ local($towho)=$newarg;
+ &getarg;
+ $newarg =~ tr/a-z/A-Z/;
+ $args=" ".$args if $args ne '';
+ &sl("PRIVMSG $towho :\ca$newarg$args\ca");
+ &dohooks("send_ctcp", $towho, $newarg.$args);
+ &tell("*** Sending a CTCP $newarg$args to $towho");
+ } else {
+ &tell("*\cbE\cb* You must specify a nick or channel!");
+ }
+ } elsif ($cmd eq 'PING' || $cmd eq 'P') {
+ &dosplat;
+ if ($args) {
+ &getarg;
+ local($t)=time;
+ &sl("PRIVMSG $newarg :\caPING $t\ca");
+ &dohooks("send_ctcp", $newarg, "PING $t");
+ &tell("*** Sending a CTCP PING to $newarg");
+ } else {
+ &tell("*\cbE\cb* You must specify a nick or channel!");
+ }
+ } elsif ($cmd eq 'ME') {
+ if ($talkchannel) {
+ &describe($talkchannel, $args);
+ } else {
+ &tell("*\cbE\cb* Not on a channel");
+ }
+ } elsif ($cmd eq 'TOPIC' || $cmd eq 'T') {
+ &dosplat;
+ local($c)=$talkchannel;
+ if ($args =~ /^[\#\&\+]/) {
+ &getarg;
+ $c=$newarg;
+ }
+ if ($args) {
+ &sl("TOPIC $c :$args");
+ } else {
+ &sl("TOPIC $c");
+ }
+ } elsif ($cmd eq 'LEAVE' || $cmd eq 'PART' || $cmd eq 'HOP') {
+ &dosplat;
+ $args=$talkchannel if $args eq '';
+ &sl("PART $args");
+ } elsif ($cmd eq 'LL') {
+ if ($talkchannel) {
+ &sl("WHO $talkchannel");
+ } else {
+ &tell("*\cbE\cb* Not on a channel");
+ }
+ } elsif ($cmd eq 'O' || $cmd eq 'OP') {
+ local($c, $n, $l)=($talkchannel, 0, '');
+ &getarg, $c=$newarg if ($args =~ /^[\#\&\+]/);
+ local(@ppl)=split(/ +/, $args);
+ foreach (@ppl) {
+ if ($n<4) {
+ $l .= " ".$_;
+ $n++;
+ } else {
+ &sl("MODE $c +oooo $l");
+ $l=$_;
+ $n=1;
+ }
+ }
+ $l && &sl("MODE $c +oooo $l");
+ } elsif ($cmd eq 'D' || $cmd eq 'DEOP') {
+ local($c, $n, $l)=($talkchannel, 0, '');
+ &getarg, $c=$newarg if ($args =~ /^[\#\&\+]/);
+ local(@ppl)=split(/ +/, $args);
+ foreach (@ppl) {
+ if ($n<4) {
+ $l .= " ".$_;
+ $n++;
+ } else {
+ &sl("MODE $c -oooo $l");
+ $l=$_;
+ $n=1;
+ }
+ }
+ $l && &sl("MODE $c -oooo $l");
+ } elsif ($cmd eq 'W' || $cmd eq 'WHOIS') {
+ &sl($args eq '' ? "WHOIS $nick" : "WHOIS $args");
+ } elsif ($cmd eq 'WI') {
+ &getarg;
+ $newarg=$nick if $newarg eq '';
+ &sl("WHOIS $newarg $newarg");
+ } elsif ($cmd eq 'WHO') {
+ &dosplat;
+ if ($args =~ /^[\s\*]*$/) {
+ &tell("*** Uhm, better not");
+ } else {
+ &sl("WHO $args");
+ }
+ } elsif ($cmd eq 'JOIN' || $cmd eq 'J') {
+ $args=$invited if $args eq '';
+ if ($args !~ /^[\#\&\+]/) {
+ $query = $args;
+ }
+ elsif (grep(&eq($_, $args), @channels)) {
+# &tell("*** Talking to $args now"); # KSIRC MOD
+ $talkchannel=$args;
+ $query = "";
+ &dostatus;
+ } else {
+ &sl("JOIN $args");
+ }
+ } elsif ($cmd eq 'UMODE') {
+ &sl("MODE $nick $args");
+ } elsif ($cmd eq 'MO') {
+ if ($talkchannel) {
+ &sl("MODE $talkchannel $args");
+ } else {
+ &tell("*\cbE\cb* You're not on any channel anyway");
+ }
+ } elsif ($cmd eq 'LIST') {
+ &dosplat;
+ $listmin=0;
+ $listmax=100000;
+ $listpat='';
+ if ($args =~ /\*/ || $args =~ /-m[ia][nx]\s/i) {
+ while (&getarg, $newarg ne '') {
+ if ($newarg =~ /^-min$/i) {
+ &getarg;
+ $listmin=$newarg if $newarg>0;
+ } elsif ($newarg =~ /^-max$/i) {
+ &getarg;
+ $listmax=$newarg if $newarg>0;
+ } else {
+ $newarg =~ s/([^\\])\./$1\\./g;
+ $newarg =~ s/\*/\.\*/g;
+ $newarg =~ s/([^\.\*\\\w])/\\$1/g;
+ $listpat=$newarg;
+ }
+ }
+ &sl("LIST");
+ } else {
+ &sl($line);
+ }
+ } elsif ($cmd eq 'RPING') {
+ &getarg;
+ &sl("RPING $newarg ".time);
+ } elsif ($cmd eq 'KILL') {
+ &getarg;
+ if ($newarg) {
+ $args || ($args=$nick);
+ &sl("KILL $newarg :$args");
+ } else {
+ &tell("*\cbE\cb* You must specify a nick!");
+ }
+ } elsif ($cmd eq 'MODE' || $cmd eq 'NAMES') {
+ &dosplat;
+ &sl("$cmd $args");
+ } elsif ($cmd eq 'OPER') {
+ &getarg;
+ $newarg=$nick unless $newarg;
+ &getuserpass("Oper password? ", "Passwd: "), $args=$_ unless $args;
+ &sl("OPER $newarg $args");
+ } elsif ($cmd eq 'CONNECT') {
+ &getarg;
+ local($srv)=$newarg;
+ &getarg;
+ if ($args) {
+ &sl("CONNECT $srv $newarg $args");
+ } else {
+ &sl("CONNECT $srv 6667 $newarg");
+ }
+ } elsif ($cmd eq 'SQUIT') {
+ &getarg;
+ &sl("SQUIT $newarg :$args");
+ } elsif ($cmd eq 'WHOWAS' || $cmd eq 'ADMIN' || $cmd eq 'STATS' ||
+ $cmd eq 'INFO' || $cmd eq 'LUSERS' || $cmd eq 'SQUIT' ||
+ $cmd eq 'REHASH' || $cmd eq 'DIE' || $cmd eq 'LINKS' ||
+ $cmd eq 'NOTE' || $cmd eq 'WALLOPS' || $cmd eq 'NICK' ||
+ $cmd eq 'MOTD' || $cmd eq 'TIME' || $cmd eq 'TRACE' ||
+ $cmd eq 'USERS' || $cmd eq 'SILENCE' || $cmd eq 'MAP' ||
+ $cmd eq 'UPING') {
+ &sl($line);
+ } else {
+ # Unknown command sucks. People want to use extensions like /nickserv, which works
+ # on some servers (Simon)
+ &sl($line);
+# &tell("*\cbE\cb* Unknown command: $cmd");
+ }
+}
+
+sub douserline {
+ local($skip, $line)=(0, @_);
+ if ($line =~ /^\@ssfe\@/) {
+ $ssfe=$raw_mode=1;
+ $add_ons.="+ssfe";
+ &dostatus;
+ } else {
+ &dohooks("command", $line);
+ return if $skip;
+ if ($line =~ s/^\///) {
+ &docommand($line);
+ } elsif ($query ne '') {
+ &msg($query, $line);
+ } else {
+ &say($line);
+ }
+ }
+}
+
+$ssfe_getline="`#ssfe#p";
+sub getuserline {
+ local($skip)='';
+ &dohooks("input", $_[0], $_[1]);
+ return if $skip;
+ print $_[0];
+ print "\n" if $raw_mode;
+ print $ssfe_getline.$_[1]."\n" if $ssfe;
+ while (($_=<STDIN>) ne '') {
+ if (/^\@ssfe\@/) {
+ $ssfe || ($add_ons.="+ssfe");
+ $ssfe=$raw_mode=1;
+ &dostatus;
+ } else {
+ &exit if $_ eq '';
+ chop;
+ return;
+ }
+ }
+ &exit;
+}
+
+sub getuserpass {
+ local($ssfe_getline)="`#ssfe#P";
+ &getuserline;
+}
+
+%cmds=();
+sub addcmd {
+ local($cmd)=$_[0];
+ $cmd =~ tr/a-z/A-Z/;
+ $cmds{$cmd}="&cmd_".$_[0].";";
+}
+
+sub addhelp {
+ local($cmd, $txt)=@_;
+ $cmd =~ tr/A-Z/a-z/;
+ foreach (reverse(split(/\n/, $txt))) {
+ s/\$v/$version/g;
+ s/\$d/$date/g;
+ unshift (@help, $_);
+ }
+ unshift(@help, "\@".$cmd);
+}
+
+sub addset {
+ local($var)=$_[0];
+ $var =~ tr/a-z/A-Z/;
+ $sets{$var}="set_".$_[0];
+}
+
+sub addsel {
+ $buf_fds{$_[0]}="sel_".$_[1] if $_[2];
+ $sel_fds{$_[0]}="sel_".$_[1] unless $_[2];
+}
+
+sub remsel {
+ delete $buf_fds{$_[0]};
+ delete $sel_fds{$_[0]};
+}
+
+sub addwsel {
+ $sel_w_fds{$_[0]}="sel_".$_[1];
+}
+
+sub remwsel {
+ delete $sel_w_fds{$_[0]};
+}
+
+@hooks=("action", "ctcp", "ctcp_reply", "dcc_chat", "dcc_request", "input",
+ "invite", "join", "kick", "leave", "mode", "msg", "nick", "notice",
+ "server_notice", "notify_signoff", "notify_signon", "public",
+ "raw_irc", "send_action", "send_dcc_chat", "send_text", "send_notice",
+ "signoff", "topic", "disconnect", "status", "print", "command",
+ "chat_disconnect", "dcc_disconnect", "send_ctcp",
+ "dcc_send", "dcc_send_status", "dcc_get", "dcc_get_status", "quit",
+ "pong"); # ksirc additions
+
+sub addhook {
+ local($type, $name)=@_;
+ $type =~ tr/A-Z/a-z/;
+ $name="hook_".$name;
+ if ($type =~ /^\d\d\d$/ || grep(($_ eq $type), @hooks)) {
+ ($type =~ /^\d\d\d$/) && ($type="num_".$type);
+ eval "*ugly_hack_hooks=*${type}_hooks;";
+ unless (grep(($_ eq $name), @ugly_hack_hooks)) {
+ push(@ugly_hack_hooks, $name);
+ }
+ } else {
+ &tell("*\cbE\cb* $type: no such hook");
+ }
+}
+
+sub remhook {
+ local($type, $name)=@_;
+ $type =~ tr/A-Z/a-z/;
+ $name="hook_".$name;
+ if ($type =~ /^\d\d\d$/ || grep(($_ eq $type), @hooks)) {
+ ($type =~ /^\d\d\d$/) && ($type="num_".$type);
+ eval "*ugly_hack_hooks=*${type}_hooks;";
+ @ugly_hack_hooks=grep(($_ ne $name), @ugly_hack_hooks);
+ } else {
+ &tell("*\cbE\cb* $type: no such hook");
+ }
+}
+
+sub userhost {
+ push (@waituh, $_[0]);
+ push (@douh, $_[1]);
+ push (@erruh, $_[2]);
+ &sl("USERHOST $_[0]");
+}
+
+sub deltimer {
+ local($ref)=$_[0];
+ local($i);
+ if ($#trefs>=0 && $ref!=0) {
+ # delete the timer if it exists
+ for ($i=0; $i<=$#trefs; $i++) {
+ if ($trefs[$i]==$ref) {
+ splice(@trefs,$i,1);
+ splice(@timers,$i,1);
+ splice(@timeactions,$i,1);
+ last;
+ }
+ }
+ }
+}
+
+sub timer {
+ local(@r, @t, @a)=();
+ local($t)=$_[0]+time;
+ local($ref)=$_[2] || 0;
+ &deltimer($ref) if $ref;
+ while ($#timers>=0 && $timers[0]<=$t) {
+ push (@r, shift(@trefs));
+ push (@t, shift(@timers));
+ push (@a, shift(@timeactions));
+ }
+ @trefs=(@r, $ref, @trefs);
+ @timers=(@t, $t, @timers);
+ @timeactions=(@a, $_[1], @timeactions);
+}
+
+sub disappeared {
+ local($n)=(grep(&eq($_, $_[0]), keys(%notify)));
+ if ($n ne '' && $notify{$n}>0) {
+ local($silent)=0;
+ &dohooks("notify_signoff", $_[0]);
+ &tell("*\cb(\cb* Signoff by $_[0] detected");
+ $notify{$n}=0;
+ }
+}
+
+sub appeared {
+ local($t, $n)=(time, grep(&eq($_, $_[0]), keys(%notify)));
+ if ($n ne '') {
+ if ($notify{$n}==0) {
+ local($silent)=0;
+ &dohooks("notify_signon", $_[0]);
+ &tell("*\cb)\cb* Signon by $_[0] detected!");
+ }
+ else {
+# &tell("*\cb(\cb* Signoff by $_[0] detected!");
+ }
+ $notify{$n}=$t;
+ }
+}
+
+$lastsendison=0;
+sub send_isons {
+ local($l)='';
+ foreach (keys %notify) {
+ &sl("ISON : $l"), $l='' if (length($l)>500);
+ $l.=$_." ";
+ }
+ &sl("ISON :$l") if $l;
+ $lastsendison=time;
+ $newisons='';
+ $checkisons=1;
+}
+
+sub signoffs {
+ foreach (keys %notify) {
+ if ($notify{$_}>0 && $notify{$_}<$lastsendison) {
+ $notify{$_}=0;
+ local($silent)=0;
+ &dohooks("notify_signoff", $_);
+ &tell("*\cb(\cb* Signoff by $_ detected");
+ }
+ }
+ $checkisons='';
+}
+
+sub modestripper {
+ local($chnl, $what)=@_;
+ $chnl =~ tr/A-Z/a-z/;
+ local($how, $modes, @args)=('+', split(/ +/, $what));
+ foreach $m (split(//, $modes)) {
+ if ($m =~ /[\-\+]/) {
+ $how=$m;
+ } elsif ($m =~ /[vb]/) {
+ shift(@args);
+ } elsif ($m eq 'k') {
+ $how eq '+' ? ($chankey{$chnl}=$args[0]) : delete $chankey{$chnl};
+ shift(@args);
+ } elsif ($m eq 'l') {
+ $how eq '+' ? ($limit{$chnl}=shift(@args)) : delete $limit{$chnl};
+ } elsif ($m eq 'o') {
+ $haveops{$chnl}=($how eq '+') if (&eq(shift(@args), $nick));
+ } else {
+ $mode{$chnl} =~ s/$m//g;
+ $mode{$chnl}.=$m if $how eq '+';
+ }
+ }
+}
+
+sub umodechange {
+ local($what)=@_;
+ local($how)='+';
+ foreach $m (split(//, $what)) {
+ if ($m =~ /[\-\+]/) {
+ $how=$m;
+ } else {
+ $umode =~ s/$m//g;
+ $umode.=$m if ($how eq '+' && $m !~ /\s/);
+ }
+ }
+}
+
+sub ignored {
+ foreach (@ignore) {
+ return 1 if $_[0] =~ /^${_}$/;
+ }
+ return '';
+}
+
+sub dorcfile {
+ return if !open(RCFILE, "<$_[0]");
+ while (<RCFILE>) {
+ chop;
+ s/^\///;
+ next if /^\#/;
+ &docommand($_) if $_;
+ $silent=$skip='';
+ }
+ close RCFILE;
+}
+
+sub loadrc {
+ $rcloaded=1;
+ $sysrc && &dorcfile($sysrc);
+ $rcfile && &dorcfile($rcfile);
+}
+
+sub selline {
+ $leftover=0;
+ $rin=$rout="\0" x 32;
+ $win=$wout="\0" x 32;
+ foreach ($S, 'STDIN', keys(%dcnick), keys(%buf_fds)) {
+ $leftover=1, return $_ if $buffer{$_} =~ /\n/;
+ }
+ foreach ('STDIN', keys(%dcnick), keys(%dcwait), keys(%dgrfh), keys(%dswait),
+ keys(%dsrfh), keys(%sel_fds), keys(%buf_fds)) {
+ vec($rin, fileno($_), 1)=1;
+ }
+ foreach (keys(%sel_w_fds)){
+ vec($win, fileno($_), 1)=1;
+ }
+ vec($rin, fileno($S), 1)=1 if $connected;
+ if ($#timers<0 || $timers[0]>time+30) {
+ select($rout=$rin, $wout=$win, undef, 30);
+ } elsif ($timers[0]<=time) {
+ select($rout=$rin, $wout=$win, undef, 0);
+ } else {
+ select($rout=$rin, $wout=$win, undef, $timers[0]-time);
+ }
+}
+
+sub getnick {
+ if ($ENV{'BACKUPNICK'} && !($nick eq $ENV{'BACKUPNICK'})) {
+ $nick=$ENV{'BACKUPNICK'};
+ } else {
+ &getuserline("Pick a nick: ", "Nick: ");
+ $nick=$_;
+ }
+ &sl("NICK $nick");
+ &dostatus;
+}
+
+sub donumeric {
+ local($from)=($who eq $myserver ? '' : " (from ${who})");
+ if ($cmd eq '401') {
+ &yetonearg;
+ &yetonearg;
+ &tell("*\cb?\cb* Cannot find $newarg on irc$from");
+ } elsif ($cmd eq '402') {
+ &yetonearg;
+ &yetonearg;
+ &tell("*\cb?\cb* $newarg: no such server$from");
+ } elsif ($cmd eq '403') {
+ &yetonearg;
+ &yetonearg;
+ &tell("*\cb?\cb* $newarg: no such channel$from");
+ } elsif ($cmd eq '406') {
+ &yetonearg;
+ &yetonearg;
+ &tell("*\cb?\cb* $newarg: there was no such nickname$from");
+ } elsif ($cmd eq '421') {
+ &yetonearg;
+ &yetonearg;
+ &tell("*\cb?\cb* $newarg: unknown command$from");
+ } elsif ($cmd =~ /^4[012]/) {
+ $args =~ s/^[^:]*://;
+ &tell("*** $args$from");
+ } elsif ($cmd eq '431') {
+ &tell("*** Was expecting a nickname somewhere...");
+ &getnick if $connected<2;
+ } elsif ($cmd eq '432') {
+ if ($connected==2) {
+ &tell("*\cbN\cb* Invalid nickname, you're still \"$nick\"");
+ } else {
+ &tell("*\cbN\cb* Invalid nickname!");
+ &getnick;
+ }
+ } elsif ($cmd eq '433') {
+ if ($connected==2) {
+ &tell("*\cbN\cb* Nick already taken, you're still \"$nick\"");
+ } else {
+ &tell("*\cbN\cb* Nick already taken!");
+ &getnick;
+ }
+ } elsif ($cmd eq '441') {
+ local($g, $w, $c)=split(/ +/, $args);
+ &tell("*\cbE\cb* $w is not on channel $c$from");
+ } elsif ($cmd eq '442') {
+ local($w, $c)=split(/ +/, $args);
+# &tell("*\cbE\cb* You're not on channel $c$from"); # KSIRC MOD
+ } elsif ($cmd eq '443') {
+ local($w, $o, $c)=split(/ +/, $args);
+ &tell("*\cbE\cb* $o is already on channel $c$from");
+ } elsif ($cmd eq '465') {
+ &tell("*\cbE\cb* You are banned from this server$from");
+ } elsif ($cmd eq '461') {
+ &yetonearg;
+ &yetonearg;
+ &tell("*\cbE\cb* The command $newarg needs more arguments than that$from");
+ } elsif ($cmd =~ /^47[1345]$/) {
+ &yetonearg;
+ &yetonearg;
+ local($r);
+ if ($cmd eq '471') {
+ $r="channel is full";
+ } elsif ($cmd eq '473') {
+ $r="channel is invite-only";
+ } elsif ($cmd eq '474') {
+ $r="banned from channel";
+ } else {
+ $r="bad channel key";
+ }
+ &tell("*\cbE\cb* Can't join $newarg: ${r}$from");
+ } elsif ($cmd eq '301') {
+ &yetonearg;
+ &yetonearg;
+ &tell("*** $newarg is away: $args");
+ } elsif ($cmd eq '302') {
+ &yetonearg;
+ &yetonearg;
+ local($n, $do, $err)=(shift(@waituh), shift(@douh), shift(@erruh));
+ if ($newarg =~ /^([^\s\*=]+)[\*]?=([\-+])/) {
+ $who=$1;
+ local($adr)=$';
+ if ($adr =~ /\@/) {
+ $user=$`;
+ $host=$';
+ } else {
+ $user=$host='';
+ }
+ if (&eq($who, $n)) {
+ eval $do;
+ $@ =~ s/\n$//, &tell("*\cbE\cb* error in userhost: $@") if $@ ne '';
+ } else {
+ &tell("*\cbE\cb* userhost returned for unexpected nick $who");
+ }
+ } else {
+ if (defined($err)) {
+ eval $err;
+ $@ =~ s/\n$//, &tell("*\cbE\cb* error in userhost: $@") if $@ ne '';
+ } else {
+ &tell("*\cb?\cb* Cannot find $n on irc");
+ }
+ }
+ } elsif ($cmd eq '303') {
+ &yetonearg;
+ local($n);
+ foreach $n (split(/ +/, $args)) {
+ &appeared($n);
+ }
+ } elsif ($cmd eq '305') {
+ &tell("*** You are no longer marked as away");
+ $away='';
+ &dostatus;
+ } elsif ($cmd eq '306') {
+ &tell("*** You are marked as being away");
+ $away=1;
+ &dostatus;
+ } elsif ($cmd eq '311') {
+ local($g, $n, $u, $m, $g, $r)=split(/ +/, $args, 6);
+ $r =~ s/^://;
+ &tell("*** $n is $u\@$m ($r)");
+ } elsif ($cmd eq '312') {
+ &yetonearg;
+ &yetonearg;
+ &yetonearg;
+ local($s)=$newarg;
+ &tell("*** on IRC via server $s ($args)");
+ } elsif ($cmd eq '313') {
+ &yetonearg;
+ &yetonearg;
+ &tell("*** $newarg $args");
+ } elsif ($cmd eq '314') {
+ local($g, $n, $u, $m, $g, $r)=split(/ +/, $args, 6);
+ $r =~ s/^://;
+ &tell("*** $n was $u\@$m ($r)");
+ } elsif ($cmd eq '317') {
+ &yetonearg;
+ &yetonearg;
+ local($n)=$newarg;
+ &yetonearg;
+ if ($newarg>=3600) {
+ &tell("*** $n has been idle for ".int($newarg/3600)." hours, ".
+ int(($newarg%3600)/60)." minutes and ".
+ ($newarg%60)." seconds");
+ } elsif ($newarg>=60) {
+ &tell("*** $n has been idle for ".int($newarg/60)." minutes and ".
+ ($newarg%60)." seconds");
+ } else {
+ &tell("*** $n has been idle for $newarg seconds");
+ }
+ } elsif ($cmd eq '319') {
+ local($g, $g, $c)=split(/ +/, $args, 3);
+ $c =~ s/^://;
+ &tell("*** on channels: $c");
+ } elsif ($cmd eq '322') {
+ local($g, $c, $n, $r)=split(/ +/, $args, 4);
+ $r =~ s/^://;
+ $n>=$listmin && $n <=$listmax && (!$listpat || $c =~ /^${listpat}$/i)
+ && &tell(sprintf("*** %-10s %-5s %s", $c, $n, $r));
+ } elsif ($cmd eq '323') {
+ $listmin=0;
+ $listmax=100000;
+ $listpat='';
+ } elsif ($cmd eq '324') {
+ local($g, $c, $m)=split(/ +/, $args, 3);
+ $m =~ s/^://;
+ $m =~ s/ $//;
+ $c =~ tr/A-Z/a-z/;
+ if (grep(&eq($_, $c), @channels)) {
+ if (defined($mode{$c})) {
+ &tell("*\cb+\cb* Mode for channel $c is \"$m\"");
+ } else {
+ $mode{$c}='';
+ }
+ &modestripper($c, $m);
+ &dostatus;
+ } else {
+ &tell("*\cb+\cb* Mode for channel $c is \"$m\"");
+ }
+ } elsif ($cmd eq '329') {
+ &yetonearg;
+ &yetonearg;
+ local($c)=$newarg;
+ &yetonearg;
+ local($t)=($newarg ? ("created " . &date($newarg)) : "0 TS");
+ &tell("*** $c : $t");
+ } elsif ($cmd eq '331') {
+ &yetonearg;
+ &yetonearg;
+ &tell("*\cbT\cb* No topic is set on channel $newarg");
+ } elsif ($cmd eq '332') {
+ &yetonearg;
+ &yetonearg;
+ &tell("*\cbT\cb* Topic for $newarg: $args");
+ } elsif ($cmd eq '333') {
+ local($g, $c, $n, $t)=split(/ +/, $args, 4);
+ local($d)=&date($t);
+ &tell("*\cbT\cb* Topic for $c set by $n on $d");
+ } elsif ($cmd eq '318' || $cmd eq '315' || $cmd eq '369' ||
+ $cmd eq '321' || $cmd eq '376' || # KSIRC MOD
+ $cmd eq '365' || $cmd eq '368' || $cmd eq '374' ||
+ $cmd eq '219' || $cmd eq '007') {
+ #nothing!
+ } elsif ($cmd eq '341') {
+ local($g, $n, $c)=split(/ +/, $args, 3);
+ &tell("*\cbI\cb* Inviting $n to channel $c");
+ } elsif ($cmd eq '352') {
+ local($g, $c, $u, $m, $s, $n, $st, $g, $i)=split(/ +/, $args, 9);
+ &tell(sprintf("%-10s %-9s %4s %s\@%s (%s)", $c, $n, $st, $u, $m, $i));
+ } elsif ($cmd eq '353') {
+ local($g, $m, $c, $r)=split(/ +/, $args, 4);
+ local($n)=$nick;
+ $n =~ s/(\W)/\\$1/g;
+ $r =~ s/^://;
+ if($DSIRC_NAMES eq ''){ #KSIRC MOD
+ &tell("*I* Users on $c: $r"); # KSIRC MOD
+ $DSIRC_NAMES = $c; # KSIRC MOD
+ } # KSIRC MOD
+ else { # KSIRC MOD
+ &tell("*\cbI\cb* Users on $c: $r"); # KSIRC MOD
+ } # KSIRC MOD
+ $c =~ tr/A-Z/a-z/;
+ $haveops{$c}=1 if ($r =~ /\@${n}( |$)/i);
+ &dostatus if &eq($c, $talkchannel);
+ } elsif ($cmd eq '366'){ # KSIRC MOD
+ #&tell("*I* Users on $DSIRC_NAMES:"); # KSIRC MOD
+ $DSIRC_NAMES = ''; # KSIRC MOD
+ } elsif ($cmd eq '221') {
+ &yetonearg;
+ &tell("*\cb+\cb* Your user mode is \"$args\"");
+ } elsif ($cmd eq '200') {
+ local($b, $l, $v, $n, $s)=split(/ +/, $args);
+ $s =~ s/^://;
+ &tell("*** $l $who ($v) ==> $n $s");
+ } elsif ($cmd eq '205') {
+ local($b, $u, $h, $n)=split(/ +/, $args);
+ $n =~ s/^://;
+ &tell("*** $u [$h] ==> $n");
+ } elsif ($cmd =~ /^20/) {
+ local($b, $t, $n, $r)=split(/ +/, $args, 4);
+ &tell("*** $t [$n] ==> $r");
+ } elsif ($cmd eq '375' || $cmd eq '372' || $cmd =~ /^25/) {
+ &yetonearg;
+ &tell("*** $args");
+ } elsif ($cmd eq '379' ) { # RPL_FORWARD (Simon)
+ &yetonearg;
+ local( $from_channel, $to_channel ) = split( / +/, $args );
+ &tell("~$from_channel~*\cb<\cb* You have left channel $from_channel");
+ } else {
+ &yetonearg;
+ #$args =~ s/ :/ /;
+ &tell("*** $args$from");
+ }
+}
+
+# main prog
+
+print "`#ssfe#i\n" unless (-t STDOUT);
+&tell("*** Welcome to \cbsirc\cb version $version; type /help for help");
+
+&load($sysinit) if $sysinit ne '' && -f $sysinit;
+&load($initfile) if !$restrict && $initfile ne '' && -f $initfile;
+
+while (1) {
+ &bindtoserver, undef $ready if $ready;
+ $silent=$skip='';
+ if ($connected==2) {
+ $time=time;
+ &loadrc unless $rcloaded;
+ &send_isons
+ if $time>=$lastsendison+90 || ($newisons && $time>=$lastsendison+10);
+ &signoffs if $checkisons && ($time>=$lastsendison+30);
+ }
+ $fh=&selline;
+ foreach $rfh (keys (%buf_fds)) {
+ if (vec($rout, fileno($rfh), 1) || ($leftover && $fh eq $rfh)) {
+ &gl($rfh) || next;
+ local($line, $h)=($_, $buf_fds{$rfh});
+ delete $buf_fds{$rfh}, delete $buffer{$rfh}, close($rfh) if $_ eq '';
+ eval { &$h($line); };
+ $@ =~ s/\n$//, &tell("*\cbE\cb* error in buffered fd hook &$h: $@")
+ if $@ ne '';
+ }
+ }
+ foreach $rfh (keys (%sel_fds)) {
+ if (vec($rout, fileno($rfh), 1)) {
+ local($h)=$sel_fds{$rfh};
+ eval { &$h($rfh); }; #KSIRC MOD
+ $@ =~ s/\n$//, &tell("*\cbE\cb* error in unbuffered fd hook &$h: $@")
+ if $@ ne '';
+ }
+ }
+ foreach $rfh (keys (%sel_w_fds)) {
+ if (vec($wout, fileno($rfh), 1)) {
+ local($h)=$sel_w_fds{$rfh};
+ eval { &$h($rfh); }; #KSIRC MOD
+ $@ =~ s/\n$//, &tell("*\cbE\cb* error in unbuffered fd hook &$h: $@")
+ if $@ ne '';
+ }
+ }
+ foreach $rfh (keys (%dcnick)) {
+ if (vec($rout, fileno($rfh), 1) || ($leftover && $fh eq $rfh)) {
+ &gl($rfh) || next;
+ &dcerror($rfh), next if $_ eq '';
+ chop;
+ local($who, $what)=($dcnick{$rfh}, $_);
+ $dcvol{$dcnick{$rfh}}+=length($what);
+ print "`#ssfe#t/m =$who \n" if $ssfe;
+ print "`#ssfe#o=${who}= $what\n" if $ssfe;
+ &dohooks("dcc_chat", $who, $what);
+ &tell("~=${who}~=\cb${who}\cb= $what"); # KSIRC MOD
+ $silent='';
+ }
+ }
+ foreach $rfh (keys (%dcwait)) {
+ if (vec($rout, fileno($rfh), 1)) {
+ local($n, $fh);
+ my $paddr;
+ if ($paddr = &accept($fh, $rfh)) {
+ select($fh); $|=1; select(STDOUT);
+ my($port,$iaddr) = sockaddr_in($paddr);
+ my $ip = inet_ntoa($iaddr);
+ $n=$dcwait{$rfh};
+ $dcnick{$fh}=$n;
+ $n =~ tr/A-Z/a-z/;
+ $dcvol{$n}=0;
+ $dcfh{$n}=$fh;
+ &tell("*\cbD\cb* DCC CHAT connection with $n established");
+ &tell("~!dcc~DCC CHAT inbound established who: $n ip: $ip");
+ print "`#ssfe#t/m =$n \n" if $ssfe;
+ }
+ delete $dcwait{$rfh};
+ }
+ }
+ foreach $sfh (keys (%dswait)) {
+ local($rfh, $fh)=$dswait{$sfh};
+ if (vec($rout, fileno($sfh), 1)) {
+ my $paddr;
+ if ($paddr = &accept($fh, $sfh)) {
+ my($port,$iaddr) = sockaddr_in($paddr);
+ my $ip = inet_ntoa($iaddr);
+ select($fh); $|=1; select(STDOUT);
+ $dsrfh{$fh}=$rfh;
+ $dstarttime{$rfh}=time;
+ $dtransferred{$fh}=0;
+ $dnick{$fh}=$dnick{$sfh};
+ $dsoffset{$fh}=$dsoffset{$sfh};
+ &tell("*\cbD\cb* DCC SEND connection with $dnick{$sfh}/$ip ($dfile{$rfh}) established");
+ &tell("~!dcc~DCC SEND established who: $dnick{$sfh} file: $dfile{$rfh} ip: $ip");
+ }
+ delete $dnick{$sfh};
+ delete $dswait{$sfh};
+ delete $dsoffset{$sfh};
+ delete $dsport{$sfh};
+ }
+ }
+ foreach $sfh (keys (%dgrfh)) {
+ local($rfh)=$dgrfh{$sfh};
+ if (vec($rout, fileno($sfh), 1)) {
+ local($a, $buf)=(0, '');
+ $a=sysread($sfh, $buf, 4096);
+ if ($a) {
+ $dtransferred{$sfh}+=$a;
+ &dohooks("dcc_get_status", $dfile{$rfh}, $dtransferred{$sfh}, $rfh);
+ # &tell("*\cbD\cb* DCC GET read: $dfile{$rfh} bytes: $dtransferred{$sfh}"); # KSIRC MOD FOR 971217
+ my $b = $dtransferred{$sfh}+$dgxferadd{$sfh};
+ &tell("~!dcc~DCC GET read: $dfile{$rfh} who: $dnick{$sfh} bytes: $b"); # KSIRC MOD FOR 971217
+ print $rfh $buf;
+ print $sfh pack("N", $b); # used to be just $dtransfered but most seem to want xfet + offset
+ } else {
+ &dgsclose($sfh, $rfh, "GET", "OK");
+ }
+ }
+ }
+ foreach $sfh (keys (%dsrfh)) {
+ local($rfh)=$dsrfh{$sfh};
+ if (vec($rout, fileno($sfh), 1) || !$dtransferred{$sfh}) {
+ local($ack, $csa, $buf, $b, $l, $w)=(0, '', '');
+ if ($dtransferred{$sfh}) {
+ &dgsclose($sfh, $rfh, "SEND", "Protocol Error"), next if sysread($sfh, $b, 4)!=4;
+ $ack=unpack("N", $b);
+ }
+ if($ack > ($dtransferred{$sfh} + $dsoffset{$sfh})){
+ my $v = $dtransferred{$sfh} + $dsoffset{$sfh};
+ &tell("*\cbD\cb* DCC transfer protocol failure! $ack $dtransferred{$sfh} $dsoffset{$sfh} $v");
+ &dgsclose($sfh, $rfh, "SEND", "Protocol Out of Sync");
+ next;
+ }
+ #
+ # When you do a dcc resume the ack value returned from the
+ # remote client is not well defined. Two different values
+ # are used, the current number of bytes transfered, or
+ # the current location in the file. We try to detech
+ # which type of ack we got and we adjust our math
+ # according so we keep up nice packet sizes.
+ # xchat can't seem to take > 4k packets after a resume
+ # and it causes the backoff to ack a little funny, but
+ # it's not our fault!
+ #
+ if($dsoffset{$sfh} && ($ack != 0) && ($dsresumedb{$sfh} == undef)) {
+ if($ack > $dsoffset{$sfh}){
+ $dsresumedb{$sfh} = 1;
+ }
+ else {
+ $dsresumedb{$sfh} = 2;
+ }
+ #&print("*** Resume style is: $dsresumedb{$sfh}");
+
+ }
+ if($dsoffset{$sfh} && ($dsresumedb{$sfh} == 1)){
+ $csa=$set{"SENDAHEAD"}-($dtransferred{$sfh}+$dsoffset{$sfh})+$ack;
+ }
+ else {
+ $csa=$set{"SENDAHEAD"}-$dtransferred{$sfh}+$ack;
+ }
+ #&print("*** CSA is: $csa ack: $ack dt: $dtransferred{$sfh} $dsoffset{$sfh}");
+ next if $csa<0;
+ $l=read($rfh, $buf, 512+$csa);
+ $w=syswrite($sfh, $buf, $l) if $l;
+ &dohooks("dcc_send_status", $dfile{$rfh}, $dtransferred{$sfh}, $rfh);
+ # &tell("*\cbD\cb* DCC SEND write: $dfile{$rfh} bytes: $dtransferred{$sfh}"); # KSIRC MOD FOR 971218
+ my $sz = $dtransferred{$sfh}+$dsoffset{$sfh};
+ &tell("~!dcc~DCC SEND write: $dfile{$rfh} who: $dnick{$sfh} bytes: $sz"); # KSIRC MOD FOR 971218
+ next if $l==0 && $ack<$dtransferred{$sfh};
+ $dtransferred{$sfh}+=$w;
+ &dgsclose($sfh, $rfh, "SEND", "OK"), next if ($w<$l || $l==0);
+ }
+ }
+ while ($#timers>=0 && $timers[0]<=time) {
+ shift (@timers);
+ shift (@trefs);
+ eval shift (@timeactions);
+ $@ =~ s/\n$//, &tell("*\cbE\cb* error in timer: $@") if $@ ne '';
+ }
+ if (vec($rout, fileno(STDIN), 1) || ($leftover && $fh eq 'STDIN')) {
+ &gl('STDIN') || next;
+ &exit if $_ eq '';
+ chop;
+ $logging && print LOG "<- " . $_ . "\n";
+ &douserline($_) if $_ ne '';
+ }
+ if ($connected && (($leftover && $fh eq $S) || vec($rout, fileno($S), 1))) {
+ &gl($S) || next;
+ if ($_ eq '') {
+ &tell("*\cbE\cb* Connection to server lost");
+ close($S);
+ delete $buffer{$S};
+ $connected=0;
+ &dohooks("disconnect");
+ &bindtoserver;
+ next;
+ }
+ chop;
+ $logging && print LOG ">> " . $_ . "\n";
+ $serverline=$_;
+ $_=$server." ".$_ unless /^:/;
+ ($who, $cmd, $args)=split(/ /, $_, 3);
+ $cmd =~ tr/a-z/A-Z/;
+ $who =~ s/^://;
+ $args =~ s/^://;
+ $user=$host=$puh1=$puh2='';
+ if ($who =~ /^([^!@ ]+)!([^@ ]+)@([^ ]+)$/) {
+ ($who, $user, $host) = ($1, $2, $3);
+ $puh1="!$user\@$host" if $set{"PRINTUH"} ne 'none';
+ $puh2=$puh1 if $set{"PRINTUH"} eq 'all';
+ }
+ &dohooks("raw_irc", $cmd, $args);
+ next if $skip;
+ next if (($cmd eq 'PRIVMSG' || $cmd eq 'NOTICE') &&
+ &ignored("$who!$user\@$host"));
+ if ($cmd eq '001') {
+ $connected=2;
+ $myserver=$who;
+ ($nick)=split(/ /, $args, 2);
+ }
+ if ($cmd =~ /^\d\d\d$/) {
+ &dohooks("num_".$cmd, $args);
+ next if $skip;
+ &donumeric;
+ } elsif ($cmd eq 'PING') {
+ &sl("PONG $args");
+ } elsif ($cmd eq 'PRIVMSG') {
+ &yetonearg;
+ if ($args =~ /^\001([^\001]*)\001$/ && $set{'CTCP'} ne 'none') {
+ &ctcp($newarg, $1);
+ } elsif (!$printchan && &eq($newarg, $talkchannel)) {
+ &dohooks("public", $newarg, $args);
+ &tell("~${newarg}~<${who}> $args"); # MOD FOR KSIRC
+ } elsif ($newarg =~ /^[\#\&\+]/) {
+ &dohooks("public", $newarg, $args);
+ &tell("~${newarg}~<${who}> $args"); # MOD FOR KSIRC
+ } elsif (&eq ($newarg, $nick)) {
+ print "`#ssfe#t/m $who \n" if $ssfe;
+ print "`#ssfe#o[$who$puh1] $args\n" if $ssfe;
+ &dohooks("msg", $args);
+ &tell("~${who}~[\cb${who}\cb${puh1}] $args"); # MOD FOR KSIRC
+ } else {
+ &tell("~${who}~[\cb${who}\cb${puh1}\cb] $args"); # MOD FOR KSIRC
+ }
+ } elsif ($cmd eq 'NOTICE') {
+ &yetonearg;
+ if ($args =~ /^\001([^\001]*)\001$/) {
+ &ctcpreply($newarg, $1);
+ } elsif ($newarg =~ /^[\#\&\+]/) {
+ &dohooks("notice", $newarg, $args);
+ &tell("~${newarg}~-${who}- $args"); # MOD FOR KSIRC
+ } elsif ($who =~ /\./) {
+ &dohooks("server_notice", $args);
+ $args="*** ".$args unless ($args =~ /^\*/);
+ &tell($args);
+ } elsif (&eq($newarg, $nick)) {
+ &dohooks("notice", $newarg, $args);
+ &tell("~${who}~-\cb${who}\cb${puh1}- $args"); # MOD FOR KSIRC
+ } else {
+ &dohooks("notice", $newarg, $args);
+ &tell("~${who}~-\cb$who$puh1\cb- $args"); # MOD FOR KSIRC
+ }
+ $newarg =~ s/\cg.*//; # ircnet kludge
+ } elsif ($cmd eq 'KICK') {
+ &yetonearg;
+ local($channel)=$newarg;
+ &yetonearg;
+ $args=$who unless $args;
+ if (&eq($nick, $newarg)) {
+ &tell("~${channel}~*\cb<\cb* You have been kicked off channel $channel by $who$puh2 ($args)"); # MOD FOR KSIRC
+ @channels=grep(!&eq($_, $channel), @channels);
+ if (@channels) {
+ $talkchannel=$channels[$#channels];
+ } else {
+ $talkchannel='';
+ }
+ $channel =~ tr/A-Z/a-z/;
+ &dohooks("kick", $newarg, $channel, $args);
+ delete $mode{$channel};
+ delete $limit{$channel};
+ delete $haveops{$channel};
+ delete $chankey{$channel};
+ $talkchannel && !$ssfe && &tell("*** Talking to $talkchannel now");
+ &dostatus;
+ } else {
+ &dohooks("kick", $newarg, $channel, $args);
+ &tell("~${channel}~*\cb<\cb* $newarg has been kicked off channel $channel by $who$puh2 ($args)"); # MOD FOR KSIRC
+ }
+ } elsif ($cmd eq 'PART') {
+ &yetonearg;
+ if (&eq($who, $nick)) {
+ #&tell("~!all~*\cb<\cb* You have left channel $newarg"); # MOD FOR KSIRC
+ @channels=grep(!&eq($_, $newarg), @channels);
+ if (@channels) {
+ $talkchannel=$channels[$#channels];
+ } else {
+ $talkchannel='';
+ }
+ $newarg =~ tr/A-Z/a-z/;
+ delete $mode{$newarg};
+ delete $limit{$newarg};
+ delete $haveops{$newarg};
+ delete $chankey{$newarg};
+ &dohooks("leave", $newarg);
+ $talkchannel && !$ssfe && &tell("*** Talking to $talkchannel now");
+ &dostatus;
+ } else {
+ &dohooks("leave", $newarg);
+ &tell("~${newarg}~*\cb<\cb* $who$puh2 has left channel $newarg"); # MOD FOR KSIRC
+ }
+ } elsif ($cmd eq 'JOIN') {
+ &yetonearg;
+ if (&eq($nick, $who)) {
+ $newarg =~ tr/A-Z/a-z/;
+ push(@channels, $newarg);
+ $talkchannel=$newarg;
+ &dohooks("join", $newarg);
+ &dostatus;
+ &tell("~${newarg}~*\cb>\cb* You have joined channel $newarg"); # MOD FOR KSIRC
+ &sl("MODE $newarg");
+ } else {
+ &dohooks("join", $newarg);
+ &tell("~${newarg}~*\cb>\cb* $who ($user\@$host) has joined channel $newarg"); # MOD FOR KSIRC
+ }
+ &appeared($who);
+ } elsif ($cmd eq 'NICK') {
+ &yetonearg;
+ if (&eq($nick, $who)) {
+ $oldnick = $nick;
+ $nick=$newarg;
+ &dohooks("nick", $newarg);
+ $who=$newarg;
+ &dostatus;
+ &tell("~!all~*\cbN\cb* $oldnick is now known as $newarg");
+ } else {
+ &dohooks("nick", $newarg);
+ &tell("~!all~*\cbN\cb* $who$puh2 is now known as $newarg"); # MOD FOR KSIRC
+ }
+ } elsif ($cmd eq 'MODE') {
+ &yetonearg;
+ $args =~ s/ $//;
+ if ($newarg =~ /^[\#\&\+]/) {
+ &modestripper($newarg, $args);
+ &dohooks("mode", $newarg, $args);
+ &dostatus;
+ &tell("~${newarg}~*\cb+\cb* Mode change \"$args\" on channel $newarg by $who$puh2"); # MOD FOR KSIRC
+ } else {
+ local($towho)=$newarg;
+ &yetonearg;
+ &umodechange($newarg), &dostatus if &eq($towho, $nick);
+ &dohooks("mode", $towho, $newarg);
+ &tell("*\cb+\cb* Mode change \"$newarg\" for user $towho by $who"); # MOD FOR KSIRC
+ }
+ } elsif ($cmd eq 'KILL') {
+ &yetonearg;
+ local($n)=$newarg;
+ $args || ($args=$who);
+ &tell("~${newarg}~*\cb<\cb* $n got killed by $who$puh1 ($args)"); # MOD FOR KSIRC
+ } elsif ($cmd eq 'INVITE') {
+ &yetonearg;
+ &yetonearg;
+ &dohooks("invite", $newarg);
+ $invited=$newarg;
+ &tell("~!default~*\cbI\cb* $who$puh1 invites you to channel $newarg"); # MOD FOR KSIRC
+ } elsif ($cmd eq 'TOPIC') {
+ &yetonearg;
+ &dohooks("topic", $newarg, $args);
+ &tell("~${newarg}~*\cbT\cb* $who$puh2 has changed the topic on channel $newarg to \"$args\""); # MOD FOR KSIRC
+ } elsif ($cmd eq 'SILENCE') {
+ &tell("*** Silence $args");
+ } elsif ($cmd eq 'PONG') {
+ &dohooks("pong", $args);
+ } elsif ($cmd eq 'QUIT') {
+ &dohooks("signoff", $args);
+ &tell("~!all~*\cb<\cb* Signoff: $who$puh2 ($args)"); # MOD FOR KSIRC
+ &disappeared($who);
+ } elsif ($cmd eq 'WALLOPS') {
+ &tell("!$who$puh2! ".$args);
+ } elsif ($cmd eq 'RPONG') {
+ local($n, $t, $ms, $ts)=split(/ +/, $args);
+ $ts =~ s/^://;
+ &tell("*** RPONG: $who - $t: $ms ms, ".time-$ts." sec");
+ } else {
+ &tell("*** The server says: $serverline");
+ }
+ }
+}
+