2010-03-22 08:33:44 +01:00
# File: Factoids.pm
2010-03-24 07:47:40 +01:00
# Author: pragma_
2010-03-22 08:33:44 +01:00
#
# Purpose: Provides functionality for factoids and a type of external module execution.
2017-03-05 22:33:31 +01:00
# This Source Code Form is subject to the terms of the Mozilla Public
# License, v. 2.0. If a copy of the MPL was not distributed with this
# file, You can obtain one at http://mozilla.org/MPL/2.0/.
2010-03-22 08:33:44 +01:00
package PBot::Factoids ;
2020-02-08 20:04:13 +01:00
use parent 'PBot::Class' ;
2010-03-22 08:33:44 +01:00
2020-02-08 20:04:13 +01:00
use warnings ; use strict ;
2019-07-11 03:40:53 +02:00
use feature 'unicode_strings' ;
2015-09-14 19:22:55 +02:00
use feature 'switch' ;
no if $] >= 5.018 , warnings = > "experimental::smartmatch" ;
2010-03-22 08:33:44 +01:00
use HTML::Entities ;
use Time::HiRes qw( gettimeofday ) ;
2017-08-05 06:05:31 +02:00
use Time::Duration qw( duration ) ;
2013-10-12 15:35:57 +02:00
use POSIX qw( strftime ) ;
2015-06-26 07:56:10 +02:00
use Text::ParseWords ;
2017-09-12 14:50:49 +02:00
use JSON ;
2010-03-22 08:33:44 +01:00
2014-05-18 02:27:57 +02:00
use PBot::FactoidCommands ;
2010-03-22 08:33:44 +01:00
use PBot::FactoidModuleLauncher ;
2010-06-20 08:16:48 +02:00
use PBot::DualIndexHashObject ;
2010-03-22 08:33:44 +01:00
2017-08-26 08:36:11 +02:00
use PBot::Utils::Indefinite ;
2017-09-05 09:27:28 +02:00
use PBot::Utils::ValidateString ;
2017-08-26 08:36:11 +02:00
2010-03-22 08:33:44 +01:00
sub initialize {
my ( $ self , % conf ) = @ _ ;
2019-12-31 04:57:47 +01:00
my $ filename = $ conf { filename } ;
2020-01-15 03:10:53 +01:00
$ self - > { factoids } = PBot::DualIndexHashObject - > new ( name = > 'Factoids' , filename = > $ filename , pbot = > $ self - > { pbot } ) ;
2010-03-22 08:33:44 +01:00
2020-01-15 03:10:53 +01:00
$ self - > { pbot } = $ self - > { pbot } ;
$ self - > { commands } = PBot::FactoidCommands - > new ( pbot = > $ self - > { pbot } ) ;
$ self - > { factoidmodulelauncher } = PBot::FactoidModuleLauncher - > new ( pbot = > $ self - > { pbot } ) ;
2014-05-17 00:11:31 +02:00
2019-05-10 03:04:52 +02:00
$ self - > { pbot } - > { registry } - > add_default ( 'text' , 'factoids' , 'default_rate_limit' , 15 ) ;
$ self - > { pbot } - > { registry } - > add_default ( 'text' , 'factoids' , 'max_name_length' , 100 ) ;
$ self - > { pbot } - > { registry } - > add_default ( 'text' , 'factoids' , 'max_content_length' , 1024 * 8 ) ;
$ self - > { pbot } - > { registry } - > add_default ( 'text' , 'factoids' , 'max_channel_length' , 20 ) ;
2014-05-24 14:01:59 +02:00
2014-05-17 00:11:31 +02:00
$ self - > { pbot } - > { atexit } - > register ( sub { $ self - > save_factoids ; return ; } ) ;
2014-05-17 22:08:19 +02:00
$ self - > load_factoids ;
2010-03-22 08:33:44 +01:00
}
sub load_factoids {
my $ self = shift ;
2014-05-18 22:09:05 +02:00
$ self - > { factoids } - > load ;
2010-03-22 08:33:44 +01:00
2010-06-20 08:16:48 +02:00
my ( $ text , $ regex , $ modules ) ;
2020-02-13 22:31:36 +01:00
foreach my $ channel ( $ self - > { factoids } - > get_keys ) {
foreach my $ trigger ( $ self - > { factoids } - > get_keys ( $ channel ) ) {
2020-01-15 03:10:53 +01:00
next if $ trigger eq '_name' ;
2020-02-13 22:31:36 +01:00
$ self - > { pbot } - > { logger } - > log ( "Missing type for $channel->$trigger\n" ) if not $ self - > { factoids } - > get_data ( $ channel , $ trigger , 'type' ) ;
$ text + + if $ self - > { factoids } - > get_data ( $ channel , $ trigger , 'type' ) eq 'text' ;
$ regex + + if $ self - > { factoids } - > get_data ( $ channel , $ trigger , 'type' ) eq 'regex' ;
$ modules + + if $ self - > { factoids } - > get_data ( $ channel , $ trigger , 'type' ) eq 'module' ;
2010-03-22 08:33:44 +01:00
}
}
2014-05-18 22:09:05 +02:00
$ self - > { pbot } - > { logger } - > log ( " " . ( $ text + $ regex + $ modules ) . " factoids loaded ($text text, $regex regexs, $modules modules).\n" ) ;
2010-03-22 08:33:44 +01:00
}
sub save_factoids {
my $ self = shift ;
2014-05-18 22:09:05 +02:00
$ self - > { factoids } - > save ;
2010-06-20 08:16:48 +02:00
$ self - > export_factoids ;
}
2010-05-09 01:36:56 +02:00
2020-02-10 10:06:38 +01:00
sub get_meta {
my ( $ self , $ channel , $ trigger , $ key ) = @ _ ;
$ channel = lc $ channel ;
$ trigger = lc $ trigger ;
my ( $ chan , $ trig ) = $ self - > find_factoid ( $ channel , $ trigger , exact_channel = > 1 ) ;
return undef if not defined $ chan ;
2020-02-13 22:31:36 +01:00
return $ self - > { factoids } - > get_data ( $ chan , $ trig , $ key ) ;
2020-02-10 10:06:38 +01:00
}
2010-06-20 08:16:48 +02:00
sub add_factoid {
my $ self = shift ;
2014-05-17 22:08:19 +02:00
my ( $ type , $ channel , $ owner , $ trigger , $ action , $ dont_save ) = @ _ ;
2010-06-20 08:16:48 +02:00
$ type = lc $ type ;
2015-02-16 05:18:46 +01:00
$ channel = '.*' if $ channel !~ /^#/ ;
2010-05-09 01:36:56 +02:00
2020-01-15 03:10:53 +01:00
my $ data ;
2020-02-13 22:31:36 +01:00
if ( $ self - > { factoids } - > exists ( $ channel , $ trigger ) ) {
2020-01-15 03:10:53 +01:00
# only update action field if force-adding it through factadd -f
2020-02-13 22:31:36 +01:00
$ data = $ self - > { factoids } - > get_data ( $ channel , $ trigger ) ;
2020-01-15 03:10:53 +01:00
$ data - > { action } = $ action ;
$ data - > { type } = $ type ;
} else {
$ data = {
enabled = > 1 ,
type = > $ type ,
action = > $ action ,
owner = > $ owner ,
created_on = > scalar gettimeofday ,
ref_count = > 0 ,
ref_user = > "nobody" ,
rate_limit = > $ self - > { pbot } - > { registry } - > get_value ( 'factoids' , 'default_rate_limit' )
} ;
}
2010-03-23 19:24:02 +01:00
2020-02-13 22:31:36 +01:00
$ self - > { commands } - > log_factoid ( $ channel , $ trigger , $ owner , "created: $action" ) unless $ dont_save ;
2020-01-15 03:10:53 +01:00
$ self - > { factoids } - > add ( $ channel , $ trigger , $ data , $ dont_save ) ;
2010-03-22 08:33:44 +01:00
}
2010-06-20 08:16:48 +02:00
sub remove_factoid {
2010-03-22 08:33:44 +01:00
my $ self = shift ;
2010-06-20 08:16:48 +02:00
my ( $ channel , $ trigger ) = @ _ ;
2015-02-16 05:18:46 +01:00
$ channel = '.*' if $ channel !~ /^#/ ;
2020-02-13 22:31:36 +01:00
$ self - > { factoids } - > remove ( $ channel , $ trigger ) ;
2010-03-22 08:33:44 +01:00
}
sub export_factoids {
my $ self = shift ;
my $ filename ;
2019-12-31 04:57:47 +01:00
if ( @ _ ) { $ filename = shift ; } else { $ filename = $ self - > { pbot } - > { registry } - > get_value ( 'general' , 'data_dir' ) . '/factoids.html' ; }
2010-03-22 08:33:44 +01:00
return if not defined $ filename ;
open FILE , "> $filename" or return "Could not open export path." ;
2010-06-20 08:16:48 +02:00
2019-12-20 08:39:46 +01:00
my $ botnick = $ self - > { pbot } - > { registry } - > get_value ( 'irc' , 'botnick' ) ;
2010-03-22 08:33:44 +01:00
my $ time = localtime ;
2013-10-12 15:35:57 +02:00
print FILE "<html><head>\n<link href='css/blue.css' rel='stylesheet' type='text/css'>\n" ;
print FILE '<script type="text/javascript" src="js/jquery-latest.js"></script>' . "\n" ;
print FILE '<script type="text/javascript" src="js/jquery.tablesorter.js"></script>' . "\n" ;
2014-03-03 10:24:33 +01:00
print FILE '<script type="text/javascript" src="js/picnet.table.filter.min.js"></script>' . "\n" ;
2013-10-12 15:35:57 +02:00
print FILE "</head>\n<body><i>Last updated at $time</i>\n" ;
2019-12-20 08:39:46 +01:00
print FILE "<hr><h2>$botnick\'s factoids</h2>\n" ;
2019-06-26 18:34:19 +02:00
2010-03-22 08:33:44 +01:00
my $ i = 0 ;
2013-10-12 15:35:57 +02:00
my $ table_id = 1 ;
2010-06-20 08:16:48 +02:00
2020-02-13 22:31:36 +01:00
foreach my $ channel ( sort $ self - > { factoids } - > get_keys ) {
next if not $ self - > { factoids } - > get_keys ( $ channel ) ;
my $ chan = $ self - > { factoids } - > get_data ( $ channel , '_name' ) ;
2020-01-15 03:10:53 +01:00
$ chan = 'global' if $ chan eq '.*' ;
2013-10-12 15:52:12 +02:00
2017-11-19 23:37:02 +01:00
print FILE "<a href='#" . encode_entities ( $ chan ) . "'>" . encode_entities ( $ chan ) . "</a><br>\n" ;
2013-10-12 15:52:12 +02:00
}
2020-02-13 22:31:36 +01:00
foreach my $ channel ( sort $ self - > { factoids } - > get_keys ) {
next if not $ self - > { factoids } - > get_keys ( $ channel ) ;
my $ chan = $ self - > { factoids } - > get_data ( $ channel , '_name' ) ;
2020-01-15 03:10:53 +01:00
$ chan = 'global' if $ chan eq '.*' ;
2017-11-19 23:37:02 +01:00
print FILE "<a name='" . encode_entities ( $ chan ) . "'></a>\n" ;
print FILE "<hr>\n<h3>" . encode_entities ( $ chan ) . "</h3>\n<hr>\n" ;
2013-10-12 15:35:57 +02:00
print FILE "<table border=\"0\" id=\"table$table_id\" class=\"tablesorter\">\n" ;
print FILE "<thead>\n<tr>\n" ;
print FILE "<th>owner</th>\n" ;
print FILE "<th>created on</th>\n" ;
print FILE "<th>times referenced</th>\n" ;
print FILE "<th>factoid</th>\n" ;
print FILE "<th>last edited by</th>\n" ;
print FILE "<th>edited date</th>\n" ;
print FILE "<th>last referenced by</th>\n" ;
print FILE "<th>last referenced date</th>\n" ;
print FILE "</tr>\n</thead>\n<tbody>\n" ;
$ table_id + + ;
2020-02-13 22:31:36 +01:00
foreach my $ trigger ( sort $ self - > { factoids } - > get_keys ( $ channel ) ) {
my $ trigger_name = $ self - > { factoids } - > get_data ( $ channel , $ trigger , '_name' ) ;
if ( $ self - > { factoids } - > get_data ( $ channel , $ trigger , 'type' ) eq 'text' ) {
2010-06-20 08:16:48 +02:00
$ i + + ;
2019-05-28 18:19:42 +02:00
if ( $ i % 2 ) {
2010-06-20 08:16:48 +02:00
print FILE "<tr bgcolor=\"#dddddd\">\n" ;
} else {
print FILE "<tr>\n" ;
}
2019-06-26 18:34:19 +02:00
2020-02-13 22:31:36 +01:00
print FILE "<td>" . encode_entities ( $ self - > { factoids } - > get_data ( $ channel , $ trigger , 'owner' ) ) . "</td>\n" ;
print FILE "<td>" . encode_entities ( strftime "%Y/%m/%d %H:%M:%S" , localtime $ self - > { factoids } - > get_data ( $ channel , $ trigger , 'created_on' ) ) . "</td>\n" ;
2013-10-12 15:35:57 +02:00
2020-02-13 22:31:36 +01:00
print FILE "<td>" . $ self - > { factoids } - > get_data ( $ channel , $ trigger , 'ref_count' ) . "</td>\n" ;
2013-10-12 15:35:57 +02:00
2020-02-13 22:31:36 +01:00
my $ action = $ self - > { factoids } - > get_data ( $ channel , $ trigger , 'action' ) ;
2015-10-03 05:14:48 +02:00
if ( $ action =~ m/https?:\/\/[^ ]+/ ) {
$ action =~ s/(.*?)http(s?:\/\/[^ ]+)/encode_entities($1) . "<a href='http" . encode_entities($2) . "'>http" . encode_entities($2) . "<\/a>"/ge ;
$ action =~ s/(.*)<\/a>(.*$)/"$1<\/a>" . encode_entities($2)/e ;
} else {
$ action = encode_entities ( $ action ) ;
}
2014-04-07 06:50:00 +02:00
2020-02-13 22:31:36 +01:00
if ( $ self - > { factoids } - > exists ( $ channel , $ trigger , 'action_with_args' ) ) {
my $ with_args = $ self - > { factoids } - > get_data ( $ channel , $ trigger , 'action_with_args' ) ;
2014-04-07 06:50:00 +02:00
$ with_args =~ s/(.*?)http(s?:\/\/[^ ]+)/encode_entities($1) . "<a href='http" . encode_entities($2) . "'>http" . encode_entities($2) . "<\/a>"/ge ;
$ with_args =~ s/(.*)<\/a>(.*$)/"$1<\/a>" . encode_entities($2)/e ;
2020-01-15 03:10:53 +01:00
print FILE "<td width=100%><b>" . encode_entities ( $ trigger_name ) . "</b> is $action<br><br><b>with_args:</b> " . encode_entities ( $ with_args ) . "</td>\n" ;
2014-04-07 06:50:00 +02:00
} else {
2020-01-15 03:10:53 +01:00
print FILE "<td width=100%><b>" . encode_entities ( $ trigger_name ) . "</b> is $action</td>\n" ;
2014-04-07 06:50:00 +02:00
}
2013-10-12 15:35:57 +02:00
2020-02-13 22:31:36 +01:00
if ( $ self - > { factoids } - > exists ( $ channel , $ trigger , 'edited_by' ) ) {
print FILE "<td>" . $ self - > { factoids } - > get_data ( $ channel , $ trigger , 'edited_by' ) . "</td>\n" ;
print FILE "<td>" . encode_entities ( strftime "%Y/%m/%d %H:%M:%S" , localtime $ self - > { factoids } - > get_data ( $ channel , $ trigger , 'edited_on' ) ) . "</td>\n" ;
2013-10-12 15:35:57 +02:00
} else {
print FILE "<td></td>\n" ;
print FILE "<td></td>\n" ;
}
2020-02-13 22:31:36 +01:00
print FILE "<td>" . encode_entities ( $ self - > { factoids } - > get_data ( $ channel , $ trigger , 'ref_user' ) ) . "</td>\n" ;
2013-10-12 15:35:57 +02:00
2020-02-13 22:31:36 +01:00
if ( $ self - > { factoids } - > exists ( $ channel , $ trigger , 'last_referenced_on' ) ) {
print FILE "<td>" . encode_entities ( strftime "%Y/%m/%d %H:%M:%S" , localtime $ self - > { factoids } - > get_data ( $ channel , $ trigger , 'last_referenced_on' ) ) . "</td>\n" ;
2013-10-12 15:35:57 +02:00
} else {
print FILE "<td></td>\n" ;
}
2019-06-26 18:34:19 +02:00
2013-10-12 15:35:57 +02:00
print FILE "</tr>\n" ;
2010-03-22 08:33:44 +01:00
}
}
2013-10-12 15:35:57 +02:00
print FILE "</tbody>\n</table>\n" ;
2010-03-22 08:33:44 +01:00
}
2010-06-20 08:16:48 +02:00
2010-03-22 08:33:44 +01:00
print FILE "<hr>$i factoids memorized.<br>" ;
2010-06-29 08:12:52 +02:00
print FILE "<hr><i>Last updated at $time</i>\n" ;
2013-10-12 15:35:57 +02:00
print FILE "<script type='text/javascript'>\n" ;
$ table_id - - ;
print FILE '$(document).ready(function() {' . "\n" ;
2019-05-28 18:19:42 +02:00
while ( $ table_id > 0 ) {
2013-10-12 15:35:57 +02:00
print FILE '$("#table' . $ table_id . '").tablesorter();' . "\n" ;
2014-03-03 10:24:33 +01:00
print FILE '$("#table' . $ table_id . '").tableFilter();' . "\n" ;
2013-10-12 15:35:57 +02:00
$ table_id - - ;
}
print FILE "});\n" ;
print FILE "</script>\n" ;
print FILE "</body>\n</html>\n" ;
2019-06-26 18:34:19 +02:00
2010-03-22 08:33:44 +01:00
close ( FILE ) ;
2019-06-26 18:34:19 +02:00
2019-12-31 04:57:47 +01:00
return "/say $i factoids exported." ;
2010-03-22 08:33:44 +01:00
}
2010-04-02 19:33:18 +02:00
sub find_factoid {
2019-06-10 01:33:27 +02:00
my ( $ self , $ from , $ keyword , % opts ) = @ _ ;
2010-06-20 08:16:48 +02:00
2019-06-10 18:21:35 +02:00
my % default_opts = (
arguments = > '' ,
exact_channel = > 0 ,
exact_trigger = > 0 ,
find_alias = > 0
) ;
% opts = ( % default_opts , % opts ) ;
2012-07-22 21:22:30 +02:00
my $ debug = 0 ;
2019-06-10 01:33:27 +02:00
if ( $ debug ) {
use Data::Dumper ;
my $ dump = Dumper \ % opts ;
$ self - > { pbot } - > { logger } - > log ( "find_factiod: from: $from, kw: $keyword, opts: $dump\n" ) ;
}
2010-06-30 06:58:22 +02:00
$ from = '.*' if not defined $ from or $ from !~ /^#/ ;
2014-05-23 14:42:23 +02:00
$ from = lc $ from ;
2020-01-15 03:10:53 +01:00
$ keyword = lc $ keyword ;
2010-04-02 19:33:18 +02:00
2014-05-18 22:09:05 +02:00
$ self - > { pbot } - > { logger } - > log ( "from: $from\n" ) if $ debug ;
2012-07-22 21:22:30 +02:00
2019-06-10 18:21:35 +02:00
my $ arguments = $ opts { arguments } ;
2010-06-20 08:16:48 +02:00
my @ result = eval {
2015-07-22 00:07:56 +02:00
my @ results ;
2015-03-29 01:49:42 +01:00
for ( my $ depth = 0 ; $ depth < 5 ; $ depth + + ) {
2019-06-10 01:33:27 +02:00
my $ string = $ keyword . ( length $ arguments ? " $arguments" : "" ) ;
2017-10-10 04:35:11 +02:00
$ self - > { pbot } - > { logger } - > log ( "string: $string\n" ) if $ debug ;
2015-03-29 01:49:42 +01:00
# check factoids
2020-02-13 22:31:36 +01:00
foreach my $ channel ( sort $ self - > { factoids } - > get_keys ) {
2019-06-10 01:33:27 +02:00
if ( $ opts { exact_channel } ) {
if ( $ opts { exact_trigger } == 1 ) {
2015-04-10 23:59:17 +02:00
next unless $ from eq lc $ channel ;
} else {
next unless $ from eq lc $ channel or $ channel eq '.*' ;
}
2014-10-28 21:33:11 +01:00
}
2020-02-13 22:31:36 +01:00
foreach my $ trigger ( $ self - > { factoids } - > get_keys ( $ channel ) ) {
2020-01-15 03:10:53 +01:00
if ( $ keyword eq $ trigger ) {
2015-03-29 01:49:42 +01:00
$ self - > { pbot } - > { logger } - > log ( "return $channel: $trigger\n" ) if $ debug ;
2020-02-13 22:31:36 +01:00
if ( $ opts { find_alias } && $ self - > { factoids } - > get_data ( $ channel , $ trigger , 'action' ) =~ /^\/call\s+(.*)$/ms ) {
2015-03-29 01:49:42 +01:00
my $ command ;
2019-05-28 18:19:42 +02:00
if ( length $ arguments ) {
2015-03-29 01:49:42 +01:00
$ command = "$1 $arguments" ;
} else {
$ command = $ 1 ;
}
2019-06-09 02:57:31 +02:00
my $ arglist = $ self - > { pbot } - > { interpreter } - > make_args ( $ command ) ;
2020-02-02 07:19:53 +01:00
( $ keyword , $ arguments ) = $ self - > { pbot } - > { interpreter } - > split_args ( $ arglist , 2 , 0 , 1 ) ;
2015-03-29 01:49:42 +01:00
goto NEXT_DEPTH ;
2014-10-28 21:33:11 +01:00
}
2015-03-29 01:49:42 +01:00
2019-06-10 01:33:27 +02:00
if ( $ opts { exact_channel } == 1 ) {
2015-07-22 00:07:56 +02:00
return ( $ channel , $ trigger ) ;
} else {
push @ results , [ $ channel , $ trigger ] ;
}
2010-06-20 08:16:48 +02:00
}
2010-04-02 19:33:18 +02:00
}
}
2015-03-29 01:49:42 +01:00
# then check regex factoids
2019-06-10 01:33:27 +02:00
if ( not $ opts { exact_trigger } ) {
2020-02-13 22:31:36 +01:00
foreach my $ channel ( $ self - > { factoids } - > get_keys ) {
2019-06-10 01:33:27 +02:00
if ( $ opts { exact_channel } ) {
2015-03-29 01:49:42 +01:00
next unless $ from eq lc $ channel or $ channel eq '.*' ;
}
2020-02-13 22:31:36 +01:00
foreach my $ trigger ( sort $ self - > { factoids } - > get_keys ( $ channel ) ) {
2020-01-15 03:10:53 +01:00
next if $ trigger eq '_name' ;
2020-02-13 22:31:36 +01:00
if ( $ self - > { factoids } - > get_data ( $ channel , $ trigger , 'type' ) eq 'regex' ) {
2015-07-22 00:07:56 +02:00
$ self - > { pbot } - > { logger } - > log ( "checking regex $string =~ m/$trigger/i\n" ) if $ debug >= 2 ;
2019-05-28 18:19:42 +02:00
if ( $ string =~ m/$trigger/i ) {
2015-03-29 01:49:42 +01:00
$ self - > { pbot } - > { logger } - > log ( "return regex $channel: $trigger\n" ) if $ debug ;
2019-06-10 01:33:27 +02:00
if ( $ opts { find_alias } ) {
2020-02-13 22:31:36 +01:00
my $ command = $ self - > { factoids } - > get_data ( $ channel , $ trigger , 'action' ) ;
2019-06-09 02:57:31 +02:00
my $ arglist = $ self - > { pbot } - > { interpreter } - > make_args ( $ command ) ;
2020-02-02 07:19:53 +01:00
( $ keyword , $ arguments ) = $ self - > { pbot } - > { interpreter } - > split_args ( $ arglist , 2 , 0 , 1 ) ;
2015-03-29 01:49:42 +01:00
$ string = $ keyword . ( length $ arguments ? " $arguments" : "" ) ;
goto NEXT_DEPTH ;
}
2019-06-10 01:33:27 +02:00
if ( $ opts { exact_channel } == 1 ) {
2015-07-22 00:07:56 +02:00
return ( $ channel , $ trigger ) ;
} else {
push @ results , [ $ channel , $ trigger ] ;
}
2015-03-29 01:49:42 +01:00
}
}
}
}
}
NEXT_DEPTH:
2019-06-10 01:33:27 +02:00
last if not $ opts { find_alias } ;
2010-04-02 19:33:18 +02:00
}
2015-07-22 00:07:56 +02:00
if ( $ debug ) {
if ( not @ results ) {
$ self - > { pbot } - > { logger } - > log ( "find_factoid: no match\n" ) ;
} else {
$ self - > { pbot } - > { logger } - > log ( "find_factoid: got results: " . ( join ', ' , map { "$_->[0] -> $_->[1]" } @ results ) . "\n" ) ;
}
}
return @ results ;
2010-04-02 19:33:18 +02:00
} ;
2019-05-28 18:19:42 +02:00
if ( $@ ) {
2014-05-18 22:09:05 +02:00
$ self - > { pbot } - > { logger } - > log ( "find_factoid: bad regex: $@\n" ) ;
2010-04-02 19:33:18 +02:00
return undef ;
}
2010-06-20 08:16:48 +02:00
return @ result ;
2010-04-10 00:55:24 +02:00
}
2017-09-24 06:04:21 +02:00
sub escape_json {
my ( $ self , $ text ) = @ _ ;
my $ thing = { thing = > $ text } ;
my $ json = encode_json $ thing ;
$ json =~ s/^{".*":"// ;
$ json =~ s/"}$// ;
return $ json ;
}
2017-10-10 04:39:54 +02:00
sub expand_special_vars {
my ( $ self , $ from , $ nick , $ root_keyword , $ action ) = @ _ ;
2019-08-17 19:09:26 +02:00
$ action =~ s/\$nick:json|\$\{nick:json\}/$self->escape_json($nick)/ge ;
$ action =~ s/\$channel:json|\$\{channel:json\}/$self->escape_json($from)/ge ;
$ action =~ s/\$randomnick:json|\$\{randomnick:json\}/my $random = $self->{pbot}->{nicklist}->random_nick($from); $random ? $self->escape_json($random) : $self->escape_json($nick)/ge ;
$ action =~ s/\$0:json|\$\{0:json\}/$self->escape_json($root_keyword)/ge ;
2017-10-10 04:39:54 +02:00
2019-08-17 19:09:26 +02:00
$ action =~ s/\$nick|\$\{nick\}/$nick/g ;
$ action =~ s/\$channel|\$\{channel\}/$from/g ;
$ action =~ s/\$randomnick|\$\{randomnick\}/my $random = $self->{pbot}->{nicklist}->random_nick($from); $random ? $random : $nick/ge ;
$ action =~ s/\$0\b|\$\{0\}\b/$root_keyword/g ;
2017-10-10 04:39:54 +02:00
2019-05-10 03:04:52 +02:00
return validate_string ( $ action , $ self - > { pbot } - > { registry } - > get_value ( 'factoids' , 'max_content_length' ) ) ;
2017-10-10 04:39:54 +02:00
}
2015-07-07 04:39:33 +02:00
sub expand_factoid_vars {
2019-12-14 19:09:28 +01:00
my ( $ self , $ stuff , @ exclude ) = @ _ ;
2017-08-28 04:52:36 +02:00
2019-12-14 19:09:28 +01:00
my $ from = length $ stuff - > { ref_from } ? $ stuff - > { ref_from } : $ stuff - > { from } ;
my $ nick = $ stuff - > { nick } ;
my $ root_keyword = $ stuff - > { keyword_override } ? $ stuff - > { keyword_override } : $ stuff - > { root_keyword } ;
my $ action = $ stuff - > { action } ;
2015-07-07 04:39:33 +02:00
2019-08-17 19:11:34 +02:00
my $ debug = 0 ;
2017-08-23 09:21:46 +02:00
my $ depth = 0 ;
2019-12-14 19:09:28 +01:00
if ( $ debug ) {
$ self - > { pbot } - > { logger } - > log ( "enter expand_factoid_vars\n" ) ;
use Data::Dumper ;
$ self - > { pbot } - > { logger } - > log ( Dumper $ stuff ) ;
}
if ( $ action =~ m/^\/call --keyword-override=([^ ]+)/i ) {
$ root_keyword = $ 1 ;
}
2017-08-23 09:21:46 +02:00
while ( 1 ) {
2019-05-28 18:19:42 +02:00
last if + + $ depth >= 1000 ;
2018-04-02 00:03:04 +02:00
my $ offset = 0 ;
2017-08-23 09:21:46 +02:00
my $ matches = 0 ;
2019-05-29 18:49:24 +02:00
my $ expansions = 0 ;
2017-10-10 04:39:54 +02:00
$ action =~ s/\$0/$root_keyword/g ;
2017-08-23 09:21:46 +02:00
my $ const_action = $ action ;
2018-04-02 00:03:04 +02:00
2019-06-07 00:21:15 +02:00
$ self - > { pbot } - > { logger } - > log ( "action: $const_action\n" ) if $ debug ;
while ( $ const_action =~ /(\ba\s*|\ban\s*)?(?<!\\)\$(?:(\{[a-zA-Z0-9_:#]+\}|[a-zA-Z0-9_:#]+))/gi ) {
2017-08-26 08:36:11 +02:00
my ( $ a , $ v ) = ( $ 1 , $ 2 ) ;
2018-03-31 23:45:43 +02:00
$ a = '' if not defined $ a ;
2019-06-07 00:21:15 +02:00
next if not defined $ v ;
2019-06-07 22:16:54 +02:00
my $ original_v = $ v ;
my $ test_v = $ v ;
$ test_v =~ s/(.):$/$1/ ; # remove trailing : only if at least one character precedes it
next if $ test_v =~ m/^_/ ; # special character prefix skipped for shell/code-factoids/etc
next if $ test_v =~ m/^(?:nick|channel|randomnick|arglen|args|arg\[.+\]|[_0])(?:\:json)*$/i ; # don't override special variables
next if @ exclude && grep { $ test_v =~ m/^\Q$_\E$/i } @ exclude ;
2019-05-29 18:49:24 +02:00
last if + + $ depth >= 1000 ;
2017-08-23 09:21:46 +02:00
2019-06-07 22:16:54 +02:00
$ self - > { pbot } - > { logger } - > log ( "v: [$original_v], test v: [$test_v]\n" ) if $ debug ;
2018-04-02 00:03:04 +02:00
2017-08-23 09:21:46 +02:00
$ matches + + ;
2019-06-07 22:16:54 +02:00
$ test_v =~ s/\{(.+)\}/$1/ ;
2017-08-23 09:21:46 +02:00
my $ modifier = '' ;
2019-06-07 22:16:54 +02:00
if ( $ test_v =~ s/(:.*)$// ) {
2017-08-23 09:21:46 +02:00
$ modifier = $ 1 ;
}
2015-09-14 19:22:55 +02:00
2017-08-23 09:21:46 +02:00
if ( $ modifier =~ m/^:(#[^:]+|global)/i ) {
$ from = $ 1 ;
$ from = '.*' if lc $ from eq 'global' ;
}
2015-10-05 11:03:13 +02:00
2017-08-28 04:52:36 +02:00
my $ recurse = 0 ;
ALIAS:
2019-08-20 23:57:12 +02:00
my @ factoids = $ self - > find_factoid ( $ from , $ test_v , exact_channel = > 2 , exact_trigger = > 2 ) ;
2017-08-23 09:21:46 +02:00
next if not @ factoids or not $ factoids [ 0 ] ;
2015-09-14 19:22:55 +02:00
2017-08-23 09:21:46 +02:00
my ( $ var_chan , $ var ) = ( $ factoids [ 0 ] - > [ 0 ] , $ factoids [ 0 ] - > [ 1 ] ) ;
2015-07-07 04:39:33 +02:00
2020-02-13 22:31:36 +01:00
if ( $ self - > { factoids } - > get_data ( $ var_chan , $ var , 'action' ) =~ m {^/call (.*)}ms ) {
2017-08-28 04:52:36 +02:00
$ test_v = $ 1 ;
2019-05-29 18:49:24 +02:00
next if + + $ recurse > 100 ;
2017-08-28 04:52:36 +02:00
goto ALIAS ;
}
2020-02-13 22:31:36 +01:00
if ( $ self - > { factoids } - > get_data ( $ var_chan , $ var , 'type' ) eq 'text' ) {
my $ change = $ self - > { factoids } - > get_data ( $ var_chan , $ var , 'action' ) ;
2019-06-03 07:34:54 +02:00
my @ list = $ self - > { pbot } - > { interpreter } - > split_line ( $ change ) ;
2017-08-23 09:21:46 +02:00
my @ mylist ;
for ( my $ i = 0 ; $ i <= $# list ; $ i + + ) {
2017-09-02 10:44:57 +02:00
push @ mylist , $ list [ $ i ] if defined $ list [ $ i ] and length $ list [ $ i ] ;
2017-08-23 09:21:46 +02:00
}
my $ line = int ( rand ( $# mylist + 1 ) ) ;
2019-06-03 07:34:54 +02:00
if ( not $ mylist [ $ line ] =~ s/^"(.*)"$/$1/ ) {
$ mylist [ $ line ] =~ s/^'(.*)'$/$1/ ;
}
2015-09-14 19:22:55 +02:00
2017-08-23 09:21:46 +02:00
foreach my $ mod ( split /:/ , $ modifier ) {
2019-06-07 22:16:54 +02:00
next if not length $ mod ;
if ( $ mylist [ $ line ] =~ /^\$\{\$([a-zA-Z0-9_:#]+)\}(.*)$/ ) {
$ mylist [ $ line ] = "\${\$$1:$mod}$2" ;
next ;
} elsif ( $ mylist [ $ line ] =~ /^\$\{([a-zA-Z0-9_:#]+)\}(.*)$/ ) {
$ mylist [ $ line ] = "\${$1:$mod}$2" ;
next ;
} elsif ( $ mylist [ $ line ] =~ /^\$\$([a-zA-Z0-9_:#]+)(.*)$/ ) {
$ mylist [ $ line ] = "\${\$$1:$mod}$2" ;
next ;
} elsif ( $ mylist [ $ line ] =~ /^\$([a-zA-Z0-9_:#]+)(.*)$/ ) {
$ mylist [ $ line ] = "\${$1:$mod}$2" ;
next ;
}
2017-08-23 09:21:46 +02:00
given ( $ mod ) {
when ( 'uc' ) {
$ mylist [ $ line ] = uc $ mylist [ $ line ] ;
}
when ( 'lc' ) {
$ mylist [ $ line ] = lc $ mylist [ $ line ] ;
}
when ( 'ucfirst' ) {
$ mylist [ $ line ] = ucfirst $ mylist [ $ line ] ;
}
when ( 'title' ) {
$ mylist [ $ line ] = ucfirst lc $ mylist [ $ line ] ;
2019-12-14 19:09:28 +01:00
$ mylist [ $ line ] =~ s/ (\w)/' ' . uc $1/ge ;
2017-08-23 09:21:46 +02:00
}
2017-09-24 06:04:21 +02:00
when ( 'json' ) {
$ mylist [ $ line ] = $ self - > escape_json ( $ mylist [ $ line ] ) ;
}
2015-09-14 19:22:55 +02:00
}
}
2019-08-17 19:09:26 +02:00
my $ replacement = $ mylist [ $ line ] ;
2017-08-26 08:36:11 +02:00
if ( $ a ) {
my $ fixed_a = select_indefinite_article $ mylist [ $ line ] ;
2018-01-19 19:46:15 +01:00
$ fixed_a = ucfirst $ fixed_a if $ a =~ m/^A/ ;
2019-08-17 19:09:26 +02:00
$ replacement = "$fixed_a $mylist[$line]" ;
2018-03-23 21:34:28 +01:00
}
2019-06-07 00:21:15 +02:00
if ( $ debug and $ offset == 0 ) {
$ self - > { pbot } - > { logger } - > log ( ( "-" x 40 ) . "\n" ) ;
}
2019-06-07 22:16:54 +02:00
$ original_v = quotemeta $ original_v ;
$ original_v =~ s/\\:/:/g ;
2019-06-07 00:21:15 +02:00
2018-03-23 21:34:28 +01:00
if ( not length $ mylist [ $ line ] ) {
2018-04-02 00:03:04 +02:00
$ self - > { pbot } - > { logger } - > log ( "No length!\n" ) if $ debug ;
2018-04-02 01:34:29 +02:00
if ( $ debug ) {
2019-06-07 22:16:54 +02:00
$ self - > { pbot } - > { logger } - > log ( "before: v: $original_v, offset: $offset\n" ) ;
2018-04-02 01:34:29 +02:00
$ self - > { pbot } - > { logger } - > log ( "$action\n" ) ;
$ self - > { pbot } - > { logger } - > log ( ( " " x $ offset ) . "^\n" ) ;
}
2019-08-17 19:09:26 +02:00
substr ( $ action , $ offset ) =~ s/$a\$$original_v ?/$replacement/ ;
$ offset += $- [ 0 ] + length $ replacement ;
2018-04-02 01:34:29 +02:00
2018-04-02 00:03:04 +02:00
if ( $ debug ) {
2019-06-07 00:21:15 +02:00
$ self - > { pbot } - > { logger } - > log ( "after: r: EMPTY \$-[0]: $-[0], offset: $offset\n" ) ;
2018-04-02 00:03:04 +02:00
$ self - > { pbot } - > { logger } - > log ( "$action\n" ) ;
$ self - > { pbot } - > { logger } - > log ( ( " " x $ offset ) . "^\n" ) ;
}
2017-08-26 08:36:11 +02:00
} else {
2018-04-02 00:03:04 +02:00
if ( $ debug ) {
2019-06-07 22:16:54 +02:00
$ self - > { pbot } - > { logger } - > log ( "before: v: $original_v, offset: $offset\n" ) ;
2018-04-02 01:34:29 +02:00
$ self - > { pbot } - > { logger } - > log ( "$action\n" ) ;
$ self - > { pbot } - > { logger } - > log ( ( " " x $ offset ) . "^\n" ) ;
}
2019-08-17 19:09:26 +02:00
substr ( $ action , $ offset ) =~ s/$a\$$original_v/$replacement/ ;
$ offset += $- [ 0 ] + length $ replacement ;
2018-04-02 01:34:29 +02:00
if ( $ debug ) {
2019-08-17 19:09:26 +02:00
$ self - > { pbot } - > { logger } - > log ( "after: r: $replacement, \$-[0]: $-[0], offset: $offset\n" ) ;
2018-04-02 00:03:04 +02:00
$ self - > { pbot } - > { logger } - > log ( "$action\n" ) ;
$ self - > { pbot } - > { logger } - > log ( ( " " x $ offset ) . "^\n" ) ;
}
2017-08-26 08:36:11 +02:00
}
2019-05-29 18:49:24 +02:00
$ expansions + + ;
2017-08-23 09:21:46 +02:00
}
2015-07-07 04:39:33 +02:00
}
2019-05-29 18:49:24 +02:00
last if $ matches == 0 or $ expansions == 0 ;
2015-07-07 04:39:33 +02:00
}
$ action =~ s/\\\$/\$/g ;
2017-08-29 08:15:57 +02:00
2017-08-31 12:10:20 +02:00
unless ( @ exclude ) {
2017-10-10 04:39:54 +02:00
$ action = $ self - > expand_special_vars ( $ from , $ nick , $ root_keyword , $ action ) ;
2017-08-31 12:10:20 +02:00
}
2017-08-29 08:15:57 +02:00
2019-05-10 03:04:52 +02:00
return validate_string ( $ action , $ self - > { pbot } - > { registry } - > get_value ( 'factoids' , 'max_content_length' ) ) ;
2015-07-07 04:39:33 +02:00
}
2015-06-26 07:56:10 +02:00
sub expand_action_arguments {
my ( $ self , $ action , $ input , $ nick ) = @ _ ;
2019-05-10 03:04:52 +02:00
$ action = validate_string ( $ action , $ self - > { pbot } - > { registry } - > get_value ( 'factoids' , 'max_content_length' ) ) ;
$ input = validate_string ( $ input , $ self - > { pbot } - > { registry } - > get_value ( 'factoids' , 'max_content_length' ) ) ;
2017-09-05 09:27:28 +02:00
2017-09-12 14:50:49 +02:00
my % h ;
2015-06-26 07:56:10 +02:00
if ( not defined $ input or $ input eq '' ) {
2017-09-12 14:50:49 +02:00
% h = ( args = > $ nick ) ;
2015-06-26 07:56:10 +02:00
} else {
2017-09-12 14:50:49 +02:00
% h = ( args = > $ input ) ;
}
2017-12-01 03:46:14 +01:00
2017-09-12 14:50:49 +02:00
my $ jsonargs = encode_json \ % h ;
$ jsonargs =~ s/^{".*":"// ;
$ jsonargs =~ s/"}$// ;
if ( not defined $ input or $ input eq '' ) {
2017-11-19 23:37:02 +01:00
$ input = "" ;
2019-08-17 19:09:26 +02:00
$ action =~ s/\$args:json|\$\{args:json\}/$jsonargs/ge ;
$ action =~ s/\$args(?![[\w])|\$\{args(?![[\w])\}/$nick/g ;
2017-09-12 14:50:49 +02:00
} else {
2019-08-17 19:09:26 +02:00
$ action =~ s/\$args:json|\$\{args:json\}/$jsonargs/g ;
$ action =~ s/\$args(?![[\w])|\$\{args(?![[\w])\}/$input/g ;
2015-06-26 07:56:10 +02:00
}
2019-06-23 03:29:54 +02:00
my @ args = $ self - > { pbot } - > { interpreter } - > split_line ( $ input ) ;
2019-08-17 19:09:26 +02:00
$ action =~ s/\$arglen\b|\$\{arglen\}/scalar @args/eg ;
2015-07-02 00:21:08 +02:00
2017-11-12 17:04:42 +01:00
my $ depth = 0 ;
2017-08-02 06:31:58 +02:00
my $ const_action = $ action ;
2019-08-17 19:09:26 +02:00
while ( $ const_action =~ m/\$arg\[([^]]+)]|\$\{arg\[([^]]+)]\}/g ) {
my $ arg = defined $ 2 ? $ 2 : $ 1 ;
2015-06-26 07:56:10 +02:00
2017-11-16 18:23:58 +01:00
last if + + $ depth >= 100 ;
2017-11-12 17:04:42 +01:00
2015-06-26 07:56:10 +02:00
if ( $ arg eq '*' ) {
if ( not defined $ input or $ input eq '' ) {
2019-08-17 19:09:26 +02:00
$ action =~ s/\$arg\[\*\]|\$\{arg\[\*\]\}/$nick/ ;
2015-06-26 07:56:10 +02:00
} else {
2019-08-17 19:09:26 +02:00
$ action =~ s/\$arg\[\*\]|\$\{arg\[\*\]\}/$input/ ;
2015-06-26 07:56:10 +02:00
}
next ;
}
if ( $ arg =~ m/([^:]*):(.*)/ ) {
my $ arg1 = $ 1 ;
my $ arg2 = $ 2 ;
my $ arg1i = $ arg1 ;
my $ arg2i = $ arg2 ;
$ arg1i = 0 if $ arg1i eq '' ;
$ arg2i = $# args if $ arg2i eq '' ;
$ arg2i = $# args if $ arg2i > $# args ;
my @ values = eval {
local $ SIG { __WARN__ } = sub { } ;
return @ args [ $ arg1i .. $ arg2i ] ;
} ;
if ( $@ ) {
next ;
} else {
my $ string = join ( ' ' , @ values ) ;
if ( $ string eq '' ) {
2019-08-17 19:09:26 +02:00
$ action =~ s/\s*\$\{arg\[$arg1:$arg2\]\}// || $ action =~ s/\s*\$arg\[$arg1:$arg2\]// ;
2015-06-26 07:56:10 +02:00
} else {
2019-08-17 19:09:26 +02:00
$ action =~ s/\$\{arg\[$arg1:$arg2\]\}/$string/ || $ action =~ s/\$arg\[$arg1:$arg2\]/$string/ ;
2015-06-26 07:56:10 +02:00
}
}
next ;
}
my $ value = eval {
local $ SIG { __WARN__ } = sub { } ;
return $ args [ $ arg ] ;
} ;
if ( $@ ) {
next ;
} else {
if ( not defined $ value ) {
if ( $ arg == 0 ) {
2019-08-17 19:09:26 +02:00
$ action =~ s/\$\{arg\[$arg\]\}/$nick/ || $ action =~ s/\$arg\[$arg\]/$nick/ ;
2015-06-26 07:56:10 +02:00
} else {
2019-08-17 19:09:26 +02:00
$ action =~ s/\s*\$\{arg\[$arg\]\}// || $ action =~ s/\s*\$arg\[$arg\]// ;
2015-06-26 07:56:10 +02:00
}
} else {
2019-08-17 19:09:26 +02:00
$ action =~ s/\$arg\{\[$arg\]\}/$value/ || $ action =~ s/\$arg\[$arg\]/$value/ ;
2015-06-26 07:56:10 +02:00
}
}
}
return $ action ;
}
2017-09-11 04:53:29 +02:00
sub execute_code_factoid_using_vm {
2017-11-16 18:23:58 +01:00
my ( $ self , $ stuff ) = @ _ ;
2017-09-11 04:53:29 +02:00
2020-02-13 22:31:36 +01:00
unless ( $ self - > { factoids } - > exists ( $ stuff - > { channel } , $ stuff - > { keyword } , 'interpolate' ) and $ self - > { factoids } - > get_data ( $ stuff - > { channel } , $ stuff - > { keyword } , 'interpolate' ) eq '0' ) {
2019-08-17 19:09:26 +02:00
if ( $ stuff - > { code } =~ m/(?:\$\{?nick\b|\$\{?args\b|\$\{?arg\[)/ and length $ stuff - > { arguments } ) {
2018-01-23 22:58:03 +01:00
$ stuff - > { no_nickoverride } = 1 ;
} else {
$ stuff - > { no_nickoverride } = 0 ;
}
2019-07-06 10:29:14 +02:00
2019-12-14 19:09:28 +01:00
$ stuff - > { action } = $ stuff - > { code } ;
$ stuff - > { code } = $ self - > expand_factoid_vars ( $ stuff ) ;
2019-07-06 10:29:14 +02:00
2020-02-13 22:31:36 +01:00
if ( $ self - > { factoids } - > get_data ( $ stuff - > { channel } , $ stuff - > { keyword } , 'allow_empty_args' ) ) {
2019-06-03 01:28:55 +02:00
$ stuff - > { code } = $ self - > expand_action_arguments ( $ stuff - > { code } , $ stuff - > { arguments } , '' ) ;
} else {
$ stuff - > { code } = $ self - > expand_action_arguments ( $ stuff - > { code } , $ stuff - > { arguments } , $ stuff - > { nick } ) ;
}
2018-01-23 22:58:03 +01:00
} else {
$ stuff - > { no_nickoverride } = 0 ;
2017-09-11 04:53:29 +02:00
}
2017-11-16 18:23:58 +01:00
my % h = ( nick = > $ stuff - > { nick } , channel = > $ stuff - > { from } , lang = > $ stuff - > { lang } , code = > $ stuff - > { code } , arguments = > $ stuff - > { arguments } , factoid = > "$stuff->{channel}:$stuff->{keyword}" ) ;
2017-09-19 06:36:40 +02:00
2020-02-13 22:31:36 +01:00
if ( $ self - > { factoids } - > exists ( $ stuff - > { channel } , $ stuff - > { keyword } , 'persist-key' ) ) {
$ h { 'persist-key' } = $ self - > { factoids } - > get_data ( $ stuff - > { channel } , $ stuff - > { keyword } , 'persist-key' ) ;
2017-09-19 06:36:40 +02:00
}
2017-09-12 14:50:49 +02:00
my $ json = encode_json \ % h ;
2017-11-16 18:23:58 +01:00
2017-11-27 11:14:34 +01:00
$ stuff - > { special } = 'code-factoid' ;
2017-11-16 18:23:58 +01:00
$ stuff - > { root_channel } = $ stuff - > { channel } ;
$ stuff - > { keyword } = 'compiler' ;
$ stuff - > { arguments } = $ json ;
2019-07-11 03:40:53 +02:00
$ stuff - > { args_utf8 } = 1 ;
2017-11-16 18:23:58 +01:00
$ self - > { pbot } - > { factoids } - > { factoidmodulelauncher } - > execute_module ( $ stuff ) ;
2017-09-11 04:53:29 +02:00
return "" ;
}
sub execute_code_factoid {
2017-11-27 11:14:34 +01:00
my ( $ self , @ args ) = @ _ ;
return $ self - > execute_code_factoid_using_vm ( @ args ) ;
2017-09-11 04:53:29 +02:00
}
2010-03-22 08:33:44 +01:00
sub interpreter {
2017-11-16 18:23:58 +01:00
my ( $ self , $ stuff ) = @ _ ;
2010-03-22 08:33:44 +01:00
my $ pbot = $ self - > { pbot } ;
2017-11-21 01:10:48 +01:00
if ( $ self - > { pbot } - > { registry } - > get_value ( 'general' , 'debugcontext' ) ) {
use Data::Dumper ;
$ Data:: Dumper:: Sortkeys = 1 ;
$ self - > { pbot } - > { logger } - > log ( "Factoids::interpreter\n" ) ;
$ self - > { pbot } - > { logger } - > log ( Dumper $ stuff ) ;
}
2017-11-16 18:23:58 +01:00
return undef if not length $ stuff - > { keyword } or $ stuff - > { interpret_depth } > $ self - > { pbot } - > { registry } - > get_value ( 'interpreter' , 'max_recursion' ) ;
2010-06-30 13:36:45 +02:00
2017-11-16 18:23:58 +01:00
$ stuff - > { from } = lc $ stuff - > { from } ;
2012-07-22 21:22:30 +02:00
2018-05-12 11:52:52 +02:00
my $ strictnamespace = $ self - > { pbot } - > { registry } - > get_value ( $ stuff - > { from } , 'strictnamespace' ) ;
if ( not defined $ strictnamespace ) {
$ strictnamespace = $ self - > { pbot } - > { registry } - > get_value ( 'general' , 'strictnamespace' ) ;
}
2011-01-30 07:29:05 +01:00
# search for factoid against global channel and current channel (from unless ref_from is defined)
2017-11-16 18:23:58 +01:00
my $ original_keyword = $ stuff - > { keyword } ;
2019-06-10 01:33:27 +02:00
my ( $ channel , $ keyword ) = $ self - > find_factoid ( $ stuff - > { ref_from } ? $ stuff - > { ref_from } : $ stuff - > { from } , $ stuff - > { keyword } , arguments = > $ stuff - > { arguments } , exact_channel = > 1 ) ;
2011-01-30 07:29:05 +01:00
2018-08-10 22:12:24 +02:00
if ( not $ stuff - > { ref_from } or $ stuff - > { ref_from } eq '.*' or $ stuff - > { ref_from } eq $ stuff - > { from } ) {
2017-11-16 18:23:58 +01:00
$ stuff - > { ref_from } = "" ;
2012-07-22 21:22:30 +02:00
}
2020-01-15 03:10:53 +01:00
if ( defined $ channel and not $ channel eq '.*' and not $ channel eq lc $ stuff - > { from } ) {
2019-05-02 21:39:05 +02:00
$ stuff - > { ref_from } = $ channel ;
2011-01-30 07:29:05 +01:00
}
2017-11-16 18:23:58 +01:00
$ stuff - > { arguments } = "" if not defined $ stuff - > { arguments } ;
2010-06-20 08:16:48 +02:00
2011-01-30 04:55:09 +01:00
# if no match found, attempt to call factoid from another channel if it exists there
2017-11-16 18:23:58 +01:00
if ( not defined $ keyword ) {
my $ string = "$original_keyword $stuff->{arguments}" ;
2014-10-14 04:30:14 +02:00
my $ lc_keyword = lc $ original_keyword ;
2011-01-29 02:21:17 +01:00
my $ comma = "" ;
my $ found = 0 ;
2014-10-14 04:30:14 +02:00
my $ chans = "" ;
2011-01-29 02:21:17 +01:00
my ( $ fwd_chan , $ fwd_trig ) ;
2011-01-30 04:55:09 +01:00
# build string of which channels contain the keyword, keeping track of the last one and count
2020-02-13 22:31:36 +01:00
foreach my $ chan ( $ self - > { factoids } - > get_keys ) {
foreach my $ trig ( $ self - > { factoids } - > get_keys ( $ chan ) ) {
my $ type = $ self - > { factoids } - > get_data ( $ chan , $ trig , 'type' ) ;
2020-01-15 03:10:53 +01:00
if ( ( $ type eq 'text' or $ type eq 'module' ) and $ trig eq $ lc_keyword ) {
2020-02-13 22:31:36 +01:00
$ chans . = $ comma . $ self - > { factoids } - > get_data ( $ chan , '_name' ) ;
2011-01-29 02:21:17 +01:00
$ comma = ", " ;
$ found + + ;
$ fwd_chan = $ chan ;
$ fwd_trig = $ trig ;
last ;
}
}
}
2019-05-02 21:39:05 +02:00
my $ ref_from = $ stuff - > { ref_from } ? "[$stuff->{ref_from}] " : "" ;
2011-01-30 04:55:09 +01:00
# if multiple channels have this keyword, then ask user to disambiguate
2017-11-16 18:23:58 +01:00
if ( $ found > 1 ) {
return undef if $ stuff - > { referenced } ;
2020-01-15 03:10:53 +01:00
return $ ref_from . "Ambiguous keyword '$original_keyword' exists in multiple channels (use 'fact <channel> $original_keyword' to choose one): $chans" ;
2019-06-26 18:34:19 +02:00
}
2011-01-30 04:55:09 +01:00
# if there's just one other channel that has this keyword, trigger that instance
2017-11-16 18:23:58 +01:00
elsif ( $ found == 1 ) {
2014-05-18 22:09:05 +02:00
$ pbot - > { logger } - > log ( "Found '$original_keyword' as '$fwd_trig' in [$fwd_chan]\n" ) ;
2017-11-16 18:23:58 +01:00
$ stuff - > { keyword } = $ fwd_trig ;
$ stuff - > { interpret_depth } + + ;
$ stuff - > { ref_from } = $ fwd_chan ;
return $ pbot - > { factoids } - > interpreter ( $ stuff ) ;
2019-06-26 18:34:19 +02:00
}
2011-01-30 04:55:09 +01:00
# otherwise keyword hasn't been found, display similiar matches for all channels
else {
2011-01-30 03:44:56 +01:00
# if a non-nick argument was supplied, e.g., a sentence using the bot's nick, don't say anything
2017-11-16 18:23:58 +01:00
return undef if length $ stuff - > { arguments } and not $ self - > { pbot } - > { nicklist } - > is_present ( $ stuff - > { from } , $ stuff - > { arguments } ) ;
2019-06-26 18:34:19 +02:00
2018-05-12 11:52:52 +02:00
my $ namespace = $ strictnamespace ? $ stuff - > { from } : '.*' ;
$ namespace = '.*' if $ namespace !~ /^#/ ;
my $ namespace_regex = $ namespace ;
if ( $ strictnamespace ) {
$ namespace_regex = "(?:" . ( quotemeta $ namespace ) . '|\\.\\*)' ;
}
my $ matches = $ self - > { commands } - > factfind ( $ stuff - > { from } , $ stuff - > { nick } , $ stuff - > { user } , $ stuff - > { host } , quotemeta ( $ original_keyword ) . " -channel $namespace_regex" ) ;
2012-11-04 21:42:38 +01:00
# found factfind matches
2017-11-16 18:23:58 +01:00
if ( $ matches !~ m/^No factoids/ ) {
return undef if $ stuff - > { referenced } ;
2013-02-25 03:27:24 +01:00
return "No such factoid '$original_keyword'; $matches" ;
2012-11-04 21:42:38 +01:00
}
2018-05-12 11:52:52 +02:00
# otherwise find levenshtein closest matches
$ matches = $ self - > { factoids } - > levenshtein_matches ( $ namespace , lc $ original_keyword , 0.50 , $ strictnamespace ) ;
2010-03-24 07:47:40 +01:00
2011-01-29 02:21:17 +01:00
# don't say anything if nothing similiar was found
return undef if $ matches eq 'none' ;
2017-11-16 18:23:58 +01:00
return undef if $ stuff - > { referenced } ;
2010-06-20 08:16:48 +02:00
2019-05-02 21:39:05 +02:00
my $ ref_from = $ stuff - > { ref_from } ? "[$stuff->{ref_from}] " : "" ;
return $ ref_from . "No such factoid '$original_keyword'; did you mean $matches?" ;
2011-01-29 02:21:17 +01:00
}
2010-06-20 08:16:48 +02:00
}
2020-02-13 22:31:36 +01:00
my $ channel_name = $ self - > { factoids } - > get_data ( $ channel , '_name' ) ;
my $ trigger_name = $ self - > { factoids } - > get_data ( $ channel , $ keyword , '_name' ) ;
2020-01-17 06:51:03 +01:00
$ channel_name = 'global' if $ channel_name eq '.*' ;
$ trigger_name = "\"$trigger_name\"" if $ trigger_name =~ / / ;
2017-11-27 11:14:34 +01:00
$ stuff - > { keyword } = $ keyword ;
$ stuff - > { trigger } = $ keyword ;
$ stuff - > { channel } = $ channel ;
$ stuff - > { original_keyword } = $ original_keyword ;
2020-01-17 06:51:03 +01:00
$ stuff - > { channel_name } = $ channel_name ;
$ stuff - > { trigger_name } = $ trigger_name ;
2017-11-27 11:14:34 +01:00
2020-02-13 22:31:36 +01:00
return undef if $ stuff - > { referenced } and $ self - > { factoids } - > get_data ( $ channel , $ keyword , 'noembed' ) ;
2015-09-04 05:56:44 +02:00
2020-02-13 22:31:36 +01:00
if ( $ self - > { factoids } - > get_data ( $ channel , $ keyword , 'locked_to_channel' ) ) {
2020-01-15 03:10:53 +01:00
if ( $ stuff - > { ref_from } ne "" ) { # called from another channel
2020-01-17 06:51:03 +01:00
return "$trigger_name may be invoked only in $stuff->{ref_from}." ;
2019-05-13 09:24:08 +02:00
}
}
2020-02-13 22:31:36 +01:00
if ( $ self - > { factoids } - > exists ( $ channel , $ keyword , 'last_referenced_on' ) ) {
if ( $ self - > { factoids } - > exists ( $ channel , $ keyword , 'last_referenced_in' ) ) {
if ( $ self - > { factoids } - > get_data ( $ channel , $ keyword , 'last_referenced_in' ) eq $ stuff - > { from } ) {
2017-12-11 03:36:16 +01:00
my $ ratelimit = $ self - > { pbot } - > { registry } - > get_value ( $ stuff - > { from } , 'ratelimit_override' ) ;
2020-02-13 22:31:36 +01:00
$ ratelimit = $ self - > { factoids } - > get_data ( $ channel , $ keyword , 'rate_limit' ) if not defined $ ratelimit ;
if ( gettimeofday - $ self - > { factoids } - > get_data ( $ channel , $ keyword , 'last_referenced_on' ) < $ ratelimit ) {
2019-05-02 21:39:05 +02:00
my $ ref_from = $ stuff - > { ref_from } ? "[$stuff->{ref_from}] " : "" ;
2020-02-13 22:31:36 +01:00
return "/msg $stuff->{nick} $ref_from'$trigger_name' is rate-limited; try again in " . duration ( $ ratelimit - int ( gettimeofday - $ self - > { factoids } - > get_data ( $ channel , $ keyword , 'last_referenced_on' ) ) ) . "." unless $ self - > { pbot } - > { users } - > loggedin_admin ( $ channel , "$stuff->{nick}!$stuff->{user}\@$stuff->{host}" ) ;
2015-04-03 19:11:21 +02:00
}
}
}
}
2020-02-13 22:31:36 +01:00
my $ data = $ self - > { factoids } - > get_data ( $ channel , $ keyword ) ;
$ data - > { ref_count } + + ;
$ data - > { ref_user } = "$stuff->{nick}!$stuff->{user}\@$stuff->{host}" ;
$ data - > { last_referenced_on } = gettimeofday ;
$ data - > { last_referenced_in } = $ stuff - > { from } || "stdin" ;
$ self - > { factoids } - > add ( $ channel , $ keyword , $ data , 1 , 1 ) ;
2010-06-20 08:16:48 +02:00
2017-08-27 13:18:31 +02:00
my $ action ;
2020-02-13 22:31:36 +01:00
if ( $ self - > { factoids } - > exists ( $ channel , $ keyword , 'usage' ) and not length $ stuff - > { arguments } and $ self - > { factoids } - > get_data ( $ channel , $ keyword , 'requires_arguments' ) ) {
2019-06-03 04:30:35 +02:00
$ stuff - > { alldone } = 1 ;
2020-02-13 22:31:36 +01:00
my $ usage = $ self - > { factoids } - > get_data ( $ channel , $ keyword , 'usage' ) ;
2020-01-17 06:51:03 +01:00
$ usage =~ s/\$0|\$\{0\}/$trigger_name/g ;
2019-06-03 17:01:52 +02:00
return $ usage ;
2019-06-03 04:30:35 +02:00
}
2020-02-13 22:31:36 +01:00
if ( length $ stuff - > { arguments } and $ self - > { factoids } - > exists ( $ channel , $ keyword , 'action_with_args' ) ) {
$ action = $ self - > { factoids } - > get_data ( $ channel , $ keyword , 'action_with_args' ) ;
2017-08-27 13:18:31 +02:00
} else {
2020-02-13 22:31:36 +01:00
$ action = $ self - > { factoids } - > get_data ( $ channel , $ keyword , 'action' ) ;
2017-08-27 13:18:31 +02:00
}
2010-04-02 19:33:18 +02:00
2019-07-01 05:48:15 +02:00
if ( $ action =~ m {^/code\s+([^\s]+)\s+(.+)$}msi ) {
2017-09-11 04:53:29 +02:00
my ( $ lang , $ code ) = ( $ 1 , $ 2 ) ;
2019-06-03 01:59:27 +02:00
2020-02-13 22:31:36 +01:00
if ( $ self - > { factoids } - > exists ( $ channel , $ keyword , 'usage' ) and not length $ stuff - > { arguments } ) {
2019-06-03 04:30:35 +02:00
$ stuff - > { alldone } = 1 ;
2020-02-13 22:31:36 +01:00
my $ usage = $ self - > { factoids } - > get_data ( $ channel , $ keyword , 'usage' ) ;
2020-01-17 06:51:03 +01:00
$ usage =~ s/\$0|\$\{0\}/$trigger_name/g ;
2019-06-03 17:01:52 +02:00
return $ usage ;
2019-06-03 01:59:27 +02:00
}
2017-11-16 18:23:58 +01:00
$ stuff - > { lang } = $ lang ;
$ stuff - > { code } = $ code ;
$ self - > execute_code_factoid ( $ stuff ) ;
2017-09-11 04:53:29 +02:00
return "" ;
2017-08-24 04:25:43 +02:00
}
2017-11-16 18:23:58 +01:00
return $ self - > handle_action ( $ stuff , $ action ) ;
2017-09-11 04:53:29 +02:00
}
sub handle_action {
2017-11-16 18:23:58 +01:00
my ( $ self , $ stuff , $ action ) = @ _ ;
2017-11-21 01:10:48 +01:00
if ( $ self - > { pbot } - > { registry } - > get_value ( 'general' , 'debugcontext' ) ) {
use Data::Dumper ;
$ Data:: Dumper:: Sortkeys = 1 ;
$ self - > { pbot } - > { logger } - > log ( "Factoids::handle_action [$action]\n" ) ;
$ self - > { pbot } - > { logger } - > log ( Dumper $ stuff ) ;
}
2017-09-11 04:53:29 +02:00
2017-08-27 13:18:31 +02:00
return "" if not length $ action ;
2017-08-27 06:42:01 +02:00
2017-11-27 11:14:34 +01:00
my ( $ channel , $ keyword ) = ( $ stuff - > { channel } , $ stuff - > { trigger } ) ;
2020-01-17 06:51:03 +01:00
my ( $ channel_name , $ trigger_name ) = ( $ stuff - > { channel_name } , $ stuff - > { trigger_name } ) ;
2019-05-02 21:39:05 +02:00
my $ ref_from = $ stuff - > { ref_from } ? "[$stuff->{ref_from}] " : "" ;
2020-02-13 22:31:36 +01:00
unless ( $ self - > { factoids } - > exists ( $ channel , $ keyword , 'interpolate' ) and $ self - > { factoids } - > get_data ( $ channel , $ keyword , 'interpolate' ) eq '0' ) {
2019-06-10 01:33:27 +02:00
my ( $ root_channel , $ root_keyword ) = $ self - > find_factoid ( $ stuff - > { ref_from } ? $ stuff - > { ref_from } : $ stuff - > { from } , $ stuff - > { root_keyword } , arguments = > $ stuff - > { arguments } , exact_channel = > 1 ) ;
2019-05-03 00:04:04 +02:00
if ( not defined $ root_channel or not defined $ root_keyword ) {
2019-05-08 04:57:09 +02:00
$ root_channel = $ channel ;
$ root_keyword = $ keyword ;
2019-05-03 00:04:04 +02:00
}
2020-02-13 22:31:36 +01:00
if ( not length $ stuff - > { keyword_override } and length $ self - > { factoids } - > get_data ( $ root_channel , $ root_keyword , 'keyword_override' ) ) {
$ stuff - > { keyword_override } = $ self - > { factoids } - > get_data ( $ root_channel , $ root_keyword , 'keyword_override' ) ;
2019-12-14 19:09:28 +01:00
}
$ stuff - > { action } = $ action ;
$ action = $ self - > expand_factoid_vars ( $ stuff ) ;
2017-08-31 12:10:20 +02:00
}
2017-11-16 18:23:58 +01:00
if ( length $ stuff - > { arguments } ) {
2019-08-17 19:09:26 +02:00
if ( $ action =~ m/\$\{?args/ or $ action =~ m/\$\{?arg\[/ ) {
2020-02-13 22:31:36 +01:00
unless ( defined $ self - > { factoids } - > get_data ( $ channel , $ keyword , 'interpolate' ) and $ self - > { factoids } - > get_data ( $ channel , $ keyword , 'interpolate' ) eq '0' ) {
2017-11-16 18:23:58 +01:00
$ action = $ self - > expand_action_arguments ( $ action , $ stuff - > { arguments } , $ stuff - > { nick } ) ;
2018-01-23 22:58:03 +01:00
$ stuff - > { no_nickoverride } = 1 ;
} else {
$ stuff - > { no_nickoverride } = 0 ;
2017-08-31 12:10:20 +02:00
}
2017-11-16 18:23:58 +01:00
$ stuff - > { arguments } = "" ;
2018-05-22 04:27:57 +02:00
$ stuff - > { original_arguments } = "" ;
2015-06-26 07:56:10 +02:00
} else {
2020-02-13 22:31:36 +01:00
if ( $ self - > { factoids } - > get_data ( $ channel , $ keyword , 'type' ) eq 'text' ) {
2017-11-16 18:23:58 +01:00
my $ target = $ self - > { pbot } - > { nicklist } - > is_present_similar ( $ stuff - > { from } , $ stuff - > { arguments } ) ;
2016-10-14 14:56:54 +02:00
2019-08-17 19:09:26 +02:00
if ( $ target and $ action !~ /\$\{?(?:nick|args)\b/ ) {
2017-11-30 22:11:39 +01:00
$ stuff - > { nickoverride } = $ target unless $ stuff - > { force_nickoverride } ;
2018-01-23 22:58:03 +01:00
$ stuff - > { no_nickoverride } = 0 ;
} else {
$ stuff - > { no_nickoverride } = 1 ;
2015-04-03 19:11:21 +02:00
}
2010-08-14 11:45:58 +02:00
}
}
2015-04-03 19:11:21 +02:00
} else {
2017-08-27 13:18:31 +02:00
# no arguments supplied, replace $args with $nick/$tonick, etc
2020-02-13 22:31:36 +01:00
if ( $ self - > { factoids } - > exists ( $ channel , $ keyword , 'usage' ) ) {
$ action = "/say " . $ self - > { factoids } - > get_data ( $ channel , $ keyword , 'usage' ) ;
2020-01-17 06:51:03 +01:00
$ action =~ s/\$0|\$\{0\}/$trigger_name/g ;
2019-06-03 04:30:35 +02:00
$ stuff - > { alldone } = 1 ;
2019-06-03 01:28:55 +02:00
} else {
2020-02-13 22:31:36 +01:00
if ( $ self - > { factoids } - > get_data ( $ channel , $ keyword , 'allow_empty_args' ) ) {
2019-06-03 01:59:27 +02:00
$ action = $ self - > expand_action_arguments ( $ action , undef , '' ) ;
} else {
$ action = $ self - > expand_action_arguments ( $ action , undef , $ stuff - > { nick } ) ;
}
2019-06-03 01:28:55 +02:00
}
2018-01-23 22:58:03 +01:00
$ stuff - > { no_nickoverride } = 0 ;
2010-06-21 17:23:46 +02:00
}
2015-04-15 05:14:22 +02:00
# Check if it's an alias
2020-01-17 06:51:03 +01:00
if ( $ action =~ /^\/call\s+(.*)$/msi ) {
2017-12-01 03:53:40 +01:00
my $ command = $ 1 ;
2019-08-06 19:38:46 +02:00
$ command =~ s/\n$// ;
2020-02-13 22:31:36 +01:00
unless ( $ self - > { factoids } - > get_data ( $ channel , $ keyword , 'require_explicit_args' ) ) {
2019-05-29 19:36:09 +02:00
my $ args = $ stuff - > { arguments } ;
$ command . = " $args" if length $ args and not $ stuff - > { special } eq 'code-factoid' ;
$ stuff - > { arguments } = '' ;
2019-05-13 17:37:36 +02:00
}
2015-04-15 05:14:22 +02:00
2020-02-13 22:31:36 +01:00
unless ( $ self - > { factoids } - > get_data ( $ channel , $ keyword , 'no_keyword_override' ) ) {
2019-05-31 17:21:27 +02:00
if ( $ command =~ s/\s*--keyword-override=([^ ]+)\s*// ) {
$ stuff - > { keyword_override } = $ 1 ;
}
2019-05-02 23:09:33 +02:00
}
2017-11-16 18:23:58 +01:00
$ stuff - > { command } = $ command ;
2018-01-23 22:58:03 +01:00
$ stuff - > { aliased } = 1 ;
2017-11-16 18:23:58 +01:00
2020-01-17 06:51:03 +01:00
$ self - > { pbot } - > { logger } - > log ( "[" . ( defined $ stuff - > { from } ? $ stuff - > { from } : "stdin" ) . "] ($stuff->{nick}!$stuff->{user}\@$stuff->{host}) $trigger_name aliased to: $command\n" ) ;
2019-05-13 09:24:08 +02:00
2020-02-13 22:31:36 +01:00
if ( defined $ self - > { factoids } - > get_data ( $ channel , $ keyword , 'cap-override' ) ) {
if ( $ self - > { factoids } - > get_data ( $ channel , $ keyword , 'locked' ) ) {
$ self - > { pbot } - > { logger } - > log ( "Capability override set to " . $ self - > { factoids } - > get_data ( $ channel , $ keyword , 'cap-override' ) . "\n" ) ;
$ stuff - > { 'cap-override' } = $ self - > { factoids } - > get_data ( $ channel , $ keyword , 'cap-override' ) ;
2019-05-13 09:24:08 +02:00
} else {
2020-02-13 22:31:36 +01:00
$ self - > { pbot } - > { logger } - > log ( "Ignoring cap-override of " . $ self - > { factoids } - > get_data ( $ channel , $ keyword , 'cap-override' ) . " on unlocked factoid\n" ) ;
2019-05-13 09:24:08 +02:00
}
}
2017-11-16 18:23:58 +01:00
return $ self - > { pbot } - > { interpreter } - > interpret ( $ stuff ) ;
2015-04-15 05:14:22 +02:00
}
2020-01-17 06:51:03 +01:00
$ self - > { pbot } - > { logger } - > log ( "(" . ( defined $ stuff - > { from } ? $ stuff - > { from } : "(undef)" ) . "): $stuff->{nick}!$stuff->{user}\@$stuff->{host}: $trigger_name: action: \"$action\"\n" ) ;
2010-04-02 19:33:18 +02:00
2020-02-13 22:31:36 +01:00
if ( $ self - > { factoids } - > get_data ( $ channel , $ keyword , 'enabled' ) == 0 ) {
2020-01-17 06:51:03 +01:00
$ self - > { pbot } - > { logger } - > log ( "$trigger_name disabled.\n" ) ;
return "/msg $stuff->{nick} ${ref_from}$trigger_name is currently disabled." ;
2015-04-03 19:11:21 +02:00
}
2010-03-22 08:33:44 +01:00
2020-02-13 22:31:36 +01:00
unless ( $ self - > { factoids } - > exists ( $ channel , $ keyword , 'interpolate' ) and $ self - > { factoids } - > get_data ( $ channel , $ keyword , 'interpolate' ) eq '0' ) {
2019-06-10 01:33:27 +02:00
my ( $ root_channel , $ root_keyword ) = $ self - > find_factoid ( $ stuff - > { ref_from } ? $ stuff - > { ref_from } : $ stuff - > { from } , $ stuff - > { root_keyword } , arguments = > $ stuff - > { arguments } , exact_channel = > 1 ) ;
2019-05-03 00:04:04 +02:00
if ( not defined $ root_channel or not defined $ root_keyword ) {
2019-05-08 04:57:09 +02:00
$ root_channel = $ channel ;
$ root_keyword = $ keyword ;
2019-05-03 00:04:04 +02:00
}
2020-02-13 22:31:36 +01:00
if ( not length $ stuff - > { keyword_override } and length $ self - > { factoids } - > get_data ( $ root_channel , $ root_keyword , 'keyword_override' ) ) {
$ stuff - > { keyword_override } = $ self - > { factoids } - > get_data ( $ root_channel , $ root_keyword , 'keyword_override' ) ;
2019-12-14 19:09:28 +01:00
}
$ stuff - > { action } = $ action ;
$ action = $ self - > expand_factoid_vars ( $ stuff ) ;
2019-06-03 01:28:55 +02:00
2020-02-13 22:31:36 +01:00
if ( $ self - > { factoids } - > get_data ( $ channel , $ keyword , 'allow_empty_args' ) ) {
2019-06-03 01:28:55 +02:00
$ action = $ self - > expand_action_arguments ( $ action , $ stuff - > { arguments } , '' ) ;
} else {
$ action = $ self - > expand_action_arguments ( $ action , $ stuff - > { arguments } , $ stuff - > { nick } ) ;
}
2017-08-31 12:10:20 +02:00
}
2017-08-27 06:42:01 +02:00
2017-11-27 11:14:34 +01:00
return $ action if $ stuff - > { special } eq 'code-factoid' ;
2017-11-16 18:23:58 +01:00
2020-02-13 22:31:36 +01:00
if ( $ self - > { factoids } - > get_data ( $ channel , $ keyword , 'type' ) eq 'module' ) {
my $ preserve_whitespace = $ self - > { factoids } - > get_data ( $ channel , $ keyword , 'preserve_whitespace' ) ;
2015-04-03 19:11:21 +02:00
$ preserve_whitespace = 0 if not defined $ preserve_whitespace ;
2010-04-02 19:33:18 +02:00
2017-11-16 18:23:58 +01:00
$ stuff - > { preserve_whitespace } = $ preserve_whitespace ;
$ stuff - > { root_keyword } = $ keyword unless defined $ stuff - > { root_keyword } ;
$ stuff - > { root_channel } = $ channel ;
2010-06-20 08:16:48 +02:00
2017-11-21 01:12:13 +01:00
my $ result = $ self - > { factoidmodulelauncher } - > execute_module ( $ stuff ) ;
if ( length $ result ) {
2019-05-02 21:39:05 +02:00
return $ ref_from . $ result ;
2017-11-21 01:12:13 +01:00
} else {
return "" ;
}
2017-11-16 18:23:58 +01:00
}
2020-02-13 22:31:36 +01:00
elsif ( $ self - > { factoids } - > get_data ( $ channel , $ keyword , 'type' ) eq 'text' ) {
2015-04-03 19:11:21 +02:00
# Don't allow user-custom /msg factoids, unless factoid triggered by admin
2018-01-20 17:56:45 +01:00
if ( $ action =~ m/^\/msg/i ) {
2020-01-25 21:28:05 +01:00
my $ admin = $ self - > { pbot } - > { users } - > loggedin_admin ( $ stuff - > { from } , "$stuff->{nick}!$stuff->{user}\@$stuff->{host}" ) ;
2020-02-04 02:19:04 +01:00
if ( not $ admin ) {
2018-01-20 17:56:45 +01:00
$ self - > { pbot } - > { logger } - > log ( "[ABUSE] Bad factoid (contains /msg): $action\n" ) ;
2020-02-04 02:19:04 +01:00
return "You must be an admin to use /msg in a factoid." ;
2018-01-20 17:56:45 +01:00
}
2010-03-22 08:33:44 +01:00
}
2010-04-02 19:33:18 +02:00
2019-05-02 21:39:05 +02:00
if ( $ ref_from ) {
if ( $ action =~ s/^\/say\s+/$ref_from/i || $ action =~ s/^\/me\s+(.*)/\/me $1 $ref_from/i
|| $ action =~ s/^\/msg\s+([^ ]+)/\/msg $1 $ref_from/i ) {
2015-04-03 19:11:21 +02:00
return $ action ;
2013-10-12 17:06:27 +02:00
} else {
2020-01-17 06:51:03 +01:00
return $ ref_from . "$trigger_name is $action" ;
2013-10-12 17:06:27 +02:00
}
2013-10-14 19:22:06 +02:00
} else {
2016-11-17 04:07:01 +01:00
if ( $ action =~ m/^\/(?:say|me|msg)/i ) {
2015-04-03 19:11:21 +02:00
return $ action ;
2013-10-14 19:22:06 +02:00
} else {
2020-01-17 06:51:03 +01:00
return "/say $trigger_name is $action" ;
2013-10-14 19:22:06 +02:00
}
2010-04-02 19:33:18 +02:00
}
2020-02-13 22:31:36 +01:00
} elsif ( $ self - > { factoids } - > get_data ( $ channel , $ keyword , 'type' ) eq 'regex' ) {
2017-09-11 04:53:29 +02:00
my $ result = eval {
2017-11-16 18:23:58 +01:00
my $ string = "$stuff->{original_keyword}" . ( defined $ stuff - > { arguments } ? " $stuff->{arguments}" : "" ) ;
2010-06-30 05:48:13 +02:00
my $ cmd ;
2017-11-16 18:23:58 +01:00
if ( $ string =~ m/$keyword/i ) {
2015-04-03 19:11:21 +02:00
$ self - > { pbot } - > { logger } - > log ( "[$string] matches [$keyword] - calling [" . $ action . "$']\n" ) ;
$ cmd = $ action . $' ;
2010-06-20 08:16:48 +02:00
my ( $ a , $ b , $ c , $ d , $ e , $ f , $ g , $ h , $ i , $ before , $ after ) = ( $ 1 , $ 2 , $ 3 , $ 4 , $ 5 , $ 6 , $ 7 , $ 8 , $ 9 , $` , $' ) ;
2010-04-02 19:33:18 +02:00
$ cmd =~ s/\$1/$a/g ;
$ cmd =~ s/\$2/$b/g ;
$ cmd =~ s/\$3/$c/g ;
$ cmd =~ s/\$4/$d/g ;
$ cmd =~ s/\$5/$e/g ;
$ cmd =~ s/\$6/$f/g ;
$ cmd =~ s/\$7/$g/g ;
$ cmd =~ s/\$8/$h/g ;
$ cmd =~ s/\$9/$i/g ;
$ cmd =~ s/\$`/$before/g ;
$ cmd =~ s/\$'/$after/g ;
$ cmd =~ s/^\s+// ;
$ cmd =~ s/\s+$// ;
2010-06-30 05:48:13 +02:00
} else {
2015-04-03 19:11:21 +02:00
$ cmd = $ action ;
2010-04-02 19:33:18 +02:00
}
2010-06-30 05:48:13 +02:00
2017-11-16 18:23:58 +01:00
$ stuff - > { command } = $ cmd ;
return $ self - > { pbot } - > { interpreter } - > interpret ( $ stuff ) ;
2010-04-02 19:33:18 +02:00
} ;
2010-06-30 05:48:13 +02:00
2019-05-28 18:19:42 +02:00
if ( $@ ) {
2014-05-18 22:09:05 +02:00
$ self - > { pbot } - > { logger } - > log ( "Regex fail: $@\n" ) ;
2014-10-14 04:30:14 +02:00
return "" ;
2010-04-02 19:33:18 +02:00
}
2017-11-21 01:12:13 +01:00
if ( length $ result ) {
2019-05-02 21:39:05 +02:00
return $ ref_from . $ result ;
2017-11-21 01:12:13 +01:00
} else {
return "" ;
}
2010-04-02 19:33:18 +02:00
} else {
2020-01-17 06:51:03 +01:00
$ self - > { pbot } - > { logger } - > log ( "($stuff->{from}): $stuff->{nick}!$stuff->{user}\@$stuff->{host}): Unknown command type for '$trigger_name'\n" ) ;
2019-05-02 21:39:05 +02:00
return "/me blinks." . " $ref_from" ;
2010-03-22 08:33:44 +01:00
}
}
1 ;