Blob Blame History Raw
# This script is no longer maintained, please use trigger.pl instead, it has all the features of this script and much more.
# There is a script for converting your replaces-file to a triggers-file on http://wouter.coekaerts.be/irssi/replace.html

use strict;
use Irssi 20020324 qw (command_bind command_runsub command signal_add_first signal_continue);
use Text::ParseWords;
use IO::File;
use Data::Dumper;
use vars qw($VERSION %IRSSI);

$VERSION = '0.1.4';
%IRSSI = (
	authors  	=> 'Wouter Coekaerts',
	contact  	=> 'wouter@coekaerts.be, coekie@#irssi',
	name    	=> 'replace',
	description 	=> '(replaces regexps in incoming events) Please use trigger.pl instead',
	license 	=> 'GPLv2',
	url     	=> 'http://wouter.coekaerts.be/irssi/',
	changed  	=> '20/09/03',
);

Irssi::print("%RThis script is no longer maintained, please use trigger.pl instead, it has all the features of this script and much more.");
Irssi::print("%RThere is a script for converting your replace-file to a trigger-file on http://wouter.coekaerts.be/irssi/replace.html");

my @replaces;

sub cmd_help {
	print ( <<EOF
	
REPLACE ADD [-[no]case] [-channels <channels>] [-masks <masks>] [-command]
            [-[no]<types>] <regexp> <replacement>
REPLACE DELETE <number>|<regexp>
REPLACE LIST
REPLACE SAVE 
REPLACE RELOAD

     -[no]case: regexp is [not] case sensitive
     -channels: only in <channels>. a space-delimited list. (use quotes)
     -masks: only for messages from someone mathing one of the masks
             (space seperated)
     -<[no]types>: [don't] replace these types. The different types are:
                    -publics,-privmsgs,-actions,-parts,-quits,-kicks,
                       -topics,-privactions,-all
                   -all is an alias for all of them, and is the default.
     -command: <replacement> isn't a replacement, but an irssi-command
                   \$S is expanded to the server tag, \$C to channelname
                   \$N to nickname, \$A to his address (foo\@bar.com), 
                   \$M to the message

Examples:
  replace every occurence of shit on #chan1 and #chan2 on ircnet with sh*t
     /REPLACE ADD -channels \"#chan1 ircnet/#chan2\" shit sh*t
  strip all colorcodes from *!lamer\@*
     /REPLACE ADD -masks *!lamer\@* '\\x03\\d?\\d?(,\\d\\d?)?|\\x02|\\x1f|\\x16|\\x06' ''
  strip backgroundcolors from quitmessages
     /REPLACE ADD -quits '\\003(\\d?\\d?),\\d\\d?' '\\003\\\$1'
  never let *!bot1\@foo.bar and *!bot2\@foo.bar hilight you
     /REPLACE ADD -masks '*!bot1\@foo.bar *!bot2\@foo.bar' mynick my\\x02\\x02nick
  avoid being hilighted by !top10 in eggdrops with stats.mod
     /REPLACE ADD -case '(Top.0\\(.*\\): 1.*)mynick' '\$1my\\x02\\x02nick'
  Convert a Windows-1252 Euro to an ISO-8859-15 Euro (same effect as euro.pl)
     /REPLACE ADD '\\x80' '\\xA4'
  Show tabs as spaces, not the inverted I (same effect as tab_stop.pl)
     /REPLACE ADD '\\t' '    '
  
EOF
    );
}

#switches in -all option
my @replace_all_switches = ('publics','privmsgs','actions','privactions','parts','quits','kicks','topics');
#list of all switches
my @replace_switches = ('case','command');
push @replace_switches,@replace_all_switches;
#list of all options (including switches)
my @replace_options = ('all','masks','channels');
push @replace_options, @replace_switches;



############################################
### catch the signals & do the replacing ###
############################################
 
# "message public", SERVER_REC, char *msg, char *nick, char *address, char *target
signal_add_first("message public" => sub {check_signal_message(\@_,1,4,2,3,'publics');});
# "message private", SERVER_REC, char *msg, char *nick, char *address
signal_add_first("message private" => sub {check_signal_message(\@_,1,-1,2,3,'privmsgs');});
# "message irc action", SERVER_REC, char *msg, char *nick, char *address, char *target
signal_add_first("message irc action" => sub {
	if ($_[4] eq $_[0]->{nick}) {
		check_signal_message(\@_,1,-1,2,3,'actions');
	} else {
		check_signal_message(\@_,1,4,2,3,'actions');
	}
});
# "message part", SERVER_REC, char *channel, char *nick, char *address, char *reason
signal_add_first("message part" => sub {check_signal_message(\@_,4,1,2,3,'parts');});
# "message quit", SERVER_REC, char *nick, char *address, char *reason
signal_add_first("message quit" => sub {check_signal_message(\@_,3,-1,1,2,'quits');});
# "message kick", SERVER_REC, char *channel, char *nick, char *kicker, char *address, char *reason
signal_add_first("message kick" => sub {check_signal_message(\@_,5,1,3,4,'kicks');});
# "message topic", SERVER_REC, char *channel, char *topic, char *nick, char *address
signal_add_first("message topic" => sub {check_signal_message(\@_,2,1,3,4,'topics');});

# do the replaces on $signal's $parammessage parameter, for replaces with $condition set
# in $paramchannel, for $paramnick!$paramaddress
#  set $param* to -1 if not present
sub check_signal_message {
	my ($signal,$parammessage,$paramchannel,$paramnick,$paramaddress,$condition) = @_;
	my ($replace, $channel, $matches);
	my $changed = 0;
	my $server = $signal->[0];
	foreach $replace (@replaces) {
		if (!$replace->{"$condition"}) {
			next; # wrong type of message
		}
		if ($replace->{'channels'} && $paramchannel != -1) { # check if the channel matches
			$matches = 0;
			foreach $channel (split(/ /,$replace->{'channels'})) {
				if (lc($signal->[$paramchannel]) eq $channel
				  || lc($server->{'tag'}.'/'.$signal->[$paramchannel]) eq $channel
				  || lc($server->{'tag'}.'/') eq $channel) {
					$matches = 1;
					last; # this channel matches, stop checking channels
				}
			}
			if (!$matches) {
				next; # this replace doesn't match, try next replace...
			}
		}
			# check the mask
		if ($replace->{'masks'} && !$server->masks_match($replace->{'masks'}, $signal->[$paramnick], $signal->[$paramaddress])) {
			next; # this replace doesn't match
			
		}
		# if were here, this replace matches
		if (!$replace->{'command'}) { # normal replace
			if ($replace->{'case'}) {
				eval('$changed = ($signal->[$parammessage] =~ s§'. $replace->{'regexp'} . '§' . $replace->{'replacement'} . '§g) || $changed;');
			} else {
				eval('$changed = ($signal->[$parammessage] =~ s§'. $replace->{'regexp'} . '§' . $replace->{'replacement'} . '§gi) || $changed;');
			}
		} else { # command
			if (($replace->{'case'} && $signal->[$parammessage] =~ /$replace->{'regexp'}/)
			     ||(!$replace->{'case'} && $signal->[$parammessage] =~ /$replace->{'regexp'}/i)) {
			     	my $command = $replace->{'replacement'};
				$command =~ s/\$M/$signal->[$parammessage]/g;
				$command =~ s/\$S/$server->{'tag'}/g;
				if ($paramchannel != -1) {$command =~ s/\$C/$signal->[$paramchannel]/g;}
				if ($paramnick != -1) {$command =~ s/\$N/$signal->[$paramnick]/g;}
				if ($paramaddress != -1) {$command =~ s/\$A/$signal->[$paramaddress]/g;}
				$server->command($command);
			}
		}
		#
	}
	if ($changed) {
		signal_continue(@$signal);
	}
}
# expand($display,"C",$name,"N",$number,"M",$mode,"H",$hilight,"S","}{sb_background}")
# stolen from chanact :)
sub expand {
	my ($string, %format) = @_;
	my ($exp, $repl);
	$string =~ s/\$$exp/$repl/g while (($exp, $repl) = each(%format));
	return $string;
}
	
# ugly function, if you know a better way (where /replace add 'a(.)a' '$1' works)
sub do_replace {
	my ($text,$replace,$changed) = @_;
	if (!$replace->{'command'}) { # normal replace
		if ($replace->{'case'}) {
			eval('$_[2] = $_[0] =~ s/'. $replace->{'regexp'} . '/' . $replace->{'replacement'} . '/g || $changed;');
		} else {
			eval('$_[2] = $_[0] =~ s/'. $replace->{'regexp'} . '/' . $replace->{'replacement'} . '/gi || $changed;');
		}
	} else { # command
		#if ($replace->{'case'} {
			if ($text =~ /$replace->{'regexp'}/) {
				command($replace->{'replacement'});
			}
		#}
	}
}

################################
### manage the replaces-list ###
################################

# REPLACE SAVE
sub cmd_save {
	my $filename = Irssi::settings_get_str('replace_file');
	my $io = new IO::File $filename, "w";
	if (defined $io) {
		my $dumper = Data::Dumper->new([\@replaces]);
		$dumper->Purity(1)->Deepcopy(1);
		$io->print($dumper->Dump);
		$io->close;
	}
	Irssi::print("Replaces saved to ".$filename);
}

sub sig_command_script_unload {
	my $script = shift;
	if ($script =~ /(.*\/)?$IRSSI{'name'}\.pl$/) {
		cmd_save();
	}
}

# REPLACE LOAD
sub cmd_load {
	my $filename = Irssi::settings_get_str('replace_file');
	my $io = new IO::File $filename, "r";
	if (not defined $io) {
		print ERROR "error opening replaces file";
		return;
	}
	if (defined $io) {
		no strict 'vars';
		my $text;
		$text .= $_ foreach ($io->getlines);
		my $rep = eval "$text";
		@replaces = @$rep if ref $rep;
	}
	Irssi::print("Replaces loaded from ".$filename);
}

# converts a replace back to '-options "foo" "bar"' form
sub to_string {
	my ($replace) = @_;
	my $string = "\'$replace->{'regexp'}\' \'$replace->{'replacement'}\'";

	# check if all @replace_all_switches are set
	my $all_set = 1;
	foreach my $switch (@replace_all_switches) {
		if (!$replace->{$switch}) {
			$all_set = 0;
			last;
		}
	}
	if ($all_set) {
		$string = '-all '.$string;
		if ($replace->{'case'}) {
			$string = '-case '.$string;
		}
	} else {
		foreach my $switch (@replace_switches) {
			if ($replace->{$switch}) {
				$string = '-'.$switch.' '.$string;
			}
		}
	}
	if ($replace->{'channels'}) {
		$string = "-channels \"$replace->{'channels'}\" ".$string;
	}
	if ($replace->{'masks'}) {
		$string = "-masks \"$replace->{'masks'}\" ".$string;
	}
	
	return $string;
}

# find a replace (for ADD and DELETE), returns index of replace, or -1 if not found
sub find_replace {
	my ($regexp) = @_;
	for (my $i=0;$i<scalar(@replaces);$i++) {
		if ($replaces[$i]->{'regexp'} eq $regexp) {
			return $i;
		}
	}
	return -1; # not found
}


# REPLACE ADD [-options(@replace_options)] <regexp> <replacement> (see help for more)
sub cmd_add {
	my ($data, $server, $item) = @_;
	my @args = &shellwords($data);

	# get regexp and replacement
	my $replacement = pop @args;
	my $regexp = pop @args;
	if (not defined $regexp) {
		print ERROR "not enough parameters";
		return;
	}
	
	# find existing, or make new replace
	my $replace;
	my $index = find_replace($regexp);
	if ($index != -1) { # change existing replace
		$replace = $replaces[$index];
	} else { # new replace
		$replace = {'regexp' => $regexp};
	}
	$replace->{'replacement'} = $replacement;
	
	# parse options
	my $arg = shift @args;
	while ($arg) {
		# -channels <channels>
		if ($arg eq '-channels') {
			$replace->{'channels'} = lc(shift @args);
		}
		# -masks <masks>
		if ($arg eq '-masks') {
			$replace->{'masks'} = shift @args;
		}
		# -all
		if ($arg eq '-all') {
			foreach my $switch (@replace_all_switches) {
				$replace->{$switch} = 1;
			}
		}
		# -<switch>
		foreach my $switch (@replace_switches) {
			# -<switch>
			if ($arg eq '-'.$switch) {
				$replace->{$switch} = 1;
				last;
			}
			# -no<switch>
			elsif ($arg eq '-no'.$switch) {
				$replace->{$switch} = undef;
				last;
			}
		}
		$arg = shift @args;
	}

	#check if some switch from replace_all_switches is set
	my $some_switch = 0;
	foreach my $switch (@replace_all_switches) {
		if ($replace->{$switch}) {
			$some_switch = 1;
			last;
		}
	}

	# if no switch set, default to all
	if (not $some_switch) {
		foreach my $switch (@replace_all_switches) {
			$replace->{$switch} = 1;
		}
	}

	if ($index == -1) { # new replace
		push @replaces, $replace;
		Irssi::print("Added replace " . scalar(@replaces) .": ". to_string($replace));
	} else { # change existing replace
		$replaces[$index] = $replace;
		Irssi::print("Replace " . ($index+1) ." changed to: ". to_string($replace));
	}
}

# REPLACE DELETE <num>|<regexp>
sub cmd_del {
	my ($data, $server, $item) = @_;
	my @args = &shellwords($data);
	my $index = $data-1;
	if ((not $data =~ /^[0-9]*$/) or not exists($replaces[$index])) {
		$index = find_replace($data);
		if ($index == -1) {
			Irssi::print ("Replace $data not found.");
			return;
		}
	}
	print("Deleted ". ($index+1) .": ". to_string($replaces[$index]));
	splice (@replaces,$index,1);
}

# REPLACE LIST
sub cmd_list {
	#my (@args) = @_;
	print ("Replace list:");
	my $i=1;
	foreach my $replace (@replaces) {
		print(" ". $i++ .": ". to_string($replace));
	}
}

######################
### initialisation ###
######################

command_bind('replace help',\&cmd_help);
command_bind('help replace',\&cmd_help);
command_bind('replace add',\&cmd_add);
command_bind('replace list',\&cmd_list);
command_bind('replace delete',\&cmd_del);
command_bind('replace save',\&cmd_save);
command_bind('replace reload',\&cmd_load);
command_bind 'replace' => sub {
    my ( $data, $server, $item ) = @_;
    $data =~ s/\s+$//g;
    command_runsub ( 'replace', $data, $server, $item ) ;
};
signal_add_first 'default command timer' => sub {
	# gets triggered if called with unknown subcommand
	cmd_help();
};

Irssi::signal_add_first('command script load', 'sig_command_script_unload');
Irssi::signal_add_first('command script unload', 'sig_command_script_unload');
Irssi::signal_add('setup saved', 'cmd_save');

# This makes tab completion work
Irssi::command_set_options('replace add',join(' ',@replace_options));

Irssi::settings_add_str($IRSSI{'name'}, 'replace_file', Irssi::get_irssi_dir()."/replaces");

cmd_load();