pbot/PBot/Registry.pm

309 lines
9.6 KiB
Perl
Raw Normal View History

# File: Registry.pm
#
# Purpose: Provides a centralized registry of configuration settings that can
# easily be examined and updated via getters and setters.
License project under MPL2 This patch adds the file LICENSE which is the verbatim copy of the Mozilla Public License Version 2.0 as retreived from https://www.mozilla.org/media/MPL/2.0/index.815ca599c9df.txt on 2017-03-05. This patch also places license headers for the MPL2 type A variant of the license header in the following files: PBot/AntiFlood.pm PBot/BanTracker.pm PBot/BlackList.pm PBot/BotAdminCommands.pm PBot/BotAdmins.pm PBot/ChanOpCommands.pm PBot/ChanOps.pm PBot/Channels.pm PBot/Commands.pm PBot/DualIndexHashObject.pm PBot/EventDispatcher.pm PBot/FactoidCommands.pm PBot/FactoidModuleLauncher.pm PBot/Factoids.pm PBot/HashObject.pm PBot/IRCHandlers.pm PBot/IgnoreList.pm PBot/IgnoreListCommands.pm PBot/Interpreter.pm PBot/LagChecker.pm PBot/Logger.pm PBot/MessageHistory.pm PBot/MessageHistory_SQLite.pm PBot/NickList.pm PBot/PBot.pm PBot/Plugins.pm PBot/Plugins/AntiAway.pm PBot/Plugins/AntiKickAutoRejoin.pm PBot/Plugins/AntiRepeat.pm PBot/Plugins/AntiTwitter.pm PBot/Plugins/AutoRejoin.pm PBot/Plugins/Counter.pm PBot/Plugins/Quotegrabs.pm PBot/Plugins/Quotegrabs/Quotegrabs_Hashtable.pm PBot/Plugins/Quotegrabs/Quotegrabs_SQLite.pm PBot/Plugins/UrlTitles.pm PBot/Plugins/_Example.pm PBot/Refresher.pm PBot/Registerable.pm PBot/Registry.pm PBot/RegistryCommands.pm PBot/SQLiteLogger.pm PBot/SQLiteLoggerLayer.pm PBot/SelectHandler.pm PBot/StdinReader.pm PBot/Timer.pm PBot/Utils/ParseDate.pm PBot/VERSION.pm build/update-version.pl modules/acronym.pl modules/ago.pl modules/c11std.pl modules/c2english.pl modules/c2english/CGrammar.pm modules/c2english/c2eng.pl modules/c99std.pl modules/cdecl.pl modules/cfaq.pl modules/cjeopardy/IRCColors.pm modules/cjeopardy/QStatskeeper.pm modules/cjeopardy/Scorekeeper.pm modules/cjeopardy/cjeopardy.pl modules/cjeopardy/cjeopardy_answer.pl modules/cjeopardy/cjeopardy_filter.pl modules/cjeopardy/cjeopardy_hint.pl modules/cjeopardy/cjeopardy_qstats.pl modules/cjeopardy/cjeopardy_scores.pl modules/cjeopardy/cjeopardy_show.pl modules/codepad.pl modules/compiler_block.pl modules/compiler_client.pl modules/compiler_vm/Diff.pm modules/compiler_vm/cc modules/compiler_vm/compiler_client.pl modules/compiler_vm/compiler_server.pl modules/compiler_vm/compiler_server_vbox_win32.pl modules/compiler_vm/compiler_server_watchdog.pl modules/compiler_vm/compiler_vm_client.pl modules/compiler_vm/compiler_vm_server.pl modules/compiler_vm/compiler_watchdog.pl modules/compiler_vm/languages/_c_base.pm modules/compiler_vm/languages/_default.pm modules/compiler_vm/languages/bash.pm modules/compiler_vm/languages/bc.pm modules/compiler_vm/languages/bf.pm modules/compiler_vm/languages/c11.pm modules/compiler_vm/languages/c89.pm modules/compiler_vm/languages/c99.pm modules/compiler_vm/languages/clang.pm modules/compiler_vm/languages/clang11.pm modules/compiler_vm/languages/clang89.pm modules/compiler_vm/languages/clang99.pm modules/compiler_vm/languages/clangpp.pm modules/compiler_vm/languages/clisp.pm modules/compiler_vm/languages/cpp.pm modules/compiler_vm/languages/freebasic.pm modules/compiler_vm/languages/go.pm modules/compiler_vm/languages/haskell.pm modules/compiler_vm/languages/java.pm modules/compiler_vm/languages/javascript.pm modules/compiler_vm/languages/ksh.pm modules/compiler_vm/languages/lua.pm modules/compiler_vm/languages/perl.pm modules/compiler_vm/languages/python.pm modules/compiler_vm/languages/python3.pm modules/compiler_vm/languages/qbasic.pm modules/compiler_vm/languages/scheme.pm modules/compiler_vm/languages/server/_c_base.pm modules/compiler_vm/languages/server/_default.pm modules/compiler_vm/languages/server/c11.pm modules/compiler_vm/languages/server/c89.pm modules/compiler_vm/languages/server/c99.pm modules/compiler_vm/languages/server/clang.pm modules/compiler_vm/languages/server/clang11.pm modules/compiler_vm/languages/server/clang89.pm modules/compiler_vm/languages/server/clang99.pm modules/compiler_vm/languages/server/cpp.pm modules/compiler_vm/languages/server/freebasic.pm modules/compiler_vm/languages/server/haskell.pm modules/compiler_vm/languages/server/java.pm modules/compiler_vm/languages/server/qbasic.pm modules/compiler_vm/languages/server/tendra.pm modules/compiler_vm/languages/sh.pm modules/compiler_vm/languages/tendra.pm modules/compliment modules/cstd.pl modules/define.pl modules/dice_roll.pl modules/excuse.sh modules/expand_macros.pl modules/fnord.pl modules/funnyish_quote.pl modules/g.pl modules/gdefine.pl modules/gen_cfacts.pl modules/gencstd.pl modules/get_title.pl modules/getcfact.pl modules/google.pl modules/gspy.pl modules/gtop10.pl modules/gtop15.pl modules/headlines.pl modules/horoscope modules/horrorscope modules/ideone.pl modules/insult.pl modules/love_quote.pl modules/man.pl modules/map.pl modules/math.pl modules/prototype.pl modules/qalc.pl modules/random_quote.pl modules/seen.pl modules/urban modules/weather.pl modules/wikipedia.pl pbot.pl pbot.sh It is highly recommended that this list of files is reviewed to ensure that all files are the copyright of the sole maintainer of the repository. If any files with license headers contain the intellectual property of anyone else, it is recommended that a request is made to revise this patch or that the explicit permission of the co-author is gained to allow for the license of the work to be changed. I (Tomasz Kramkowski), the contributor, take no responsibility for any legal action taken against the maintainer of this repository for incorrectly claiming copyright to any work not owned by the maintainer of this repository.
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/.
package PBot::Registry;
use parent 'PBot::Class';
2021-06-19 06:23:34 +02:00
use PBot::Imports;
2019-07-11 03:40:53 +02:00
use Time::HiRes qw(gettimeofday);
use PBot::RegistryCommands;
sub initialize {
2020-02-15 23:38:32 +01:00
my ($self, %conf) = @_;
# ensure we have a registry filename
my $filename = $conf{filename} // Carp::croak("Missing filename configuration item in " . __FILE__);
# registry is stored as a dual-index hash object
2020-02-15 23:38:32 +01:00
$self->{registry} = PBot::DualIndexHashObject->new(name => 'Registry', filename => $filename, pbot => $self->{pbot});
# registry triggers are processed when a registry entry is modified
2020-02-15 23:38:32 +01:00
$self->{triggers} = {};
# save registry data at bot exit
2020-02-15 23:38:32 +01:00
$self->{pbot}->{atexit}->register(sub { $self->save; return; });
# prepare registry-specific bot commands
2020-02-15 23:38:32 +01:00
PBot::RegistryCommands->new(pbot => $self->{pbot});
# load existing registry entries from file (if exists)
if (-e $filename) {
$self->load;
} else {
$self->{pbot}->{logger}->log("No registry found at $filename, using defaults.\n");
}
# add default registry items
$self->add_default('text', 'general', 'data_dir', $conf{data_dir});
$self->add_default('text', 'general', 'module_dir', $conf{module_dir});
$self->add_default('text', 'general', 'plugin_dir', $conf{plugin_dir});
$self->add_default('text', 'general', 'update_dir', $conf{update_dir});
# bot trigger
$self->add_default('text', 'general', 'trigger', $conf{trigger} // '!');
# irc
$self->add_default('text', 'irc', 'debug', $conf{irc_debug} // 0);
$self->add_default('text', 'irc', 'show_motd', $conf{show_motd} // 1);
$self->add_default('text', 'irc', 'max_msg_len', $conf{max_msg_len} // 425);
$self->add_default('text', 'irc', 'server', $conf{server} // "irc.libera.chat");
$self->add_default('text', 'irc', 'port', $conf{port} // 6667);
$self->add_default('text', 'irc', 'sasl', $conf{SASL} // 0);
$self->add_default('text', 'irc', 'ssl', $conf{SSL} // 0);
$self->add_default('text', 'irc', 'ssl_ca_file', $conf{SSL_ca_file} // 'none');
$self->add_default('text', 'irc', 'ssl_ca_path', $conf{SSL_ca_path} // 'none');
$self->add_default('text', 'irc', 'botnick', $conf{botnick} // "");
$self->add_default('text', 'irc', 'username', $conf{username} // "pbot3");
$self->add_default('text', 'irc', 'realname', $conf{realname} // "https://github.com/pragma-/pbot");
$self->add_default('text', 'irc', 'identify_password', $conf{identify_password} // '');
$self->add_default('text', 'irc', 'log_default_handler', 1);
# interpreter
$self->add_default('text', 'interpreter', 'max_embed', 3);
# make sensitive entries private
$self->set_default('irc', 'ssl_ca_file', 'private', 1);
$self->set_default('irc', 'ssl_ca_path', 'private', 1);
$self->set_default('irc', 'identify_password', 'private', 1);
# customizable regular expressions
$self->add_default('text', 'regex', 'nickname', '[_a-zA-Z0-9\[\]{}`\\-]+');
# update important paths
$self->set('general', 'data_dir', 'value', $conf{data_dir}, 0, 1);
$self->set('general', 'module_dir', 'value', $conf{module_dir}, 0, 1);
$self->set('general', 'plugin_dir', 'value', $conf{plugin_dir}, 0, 1);
$self->set('general', 'update_dir', 'value', $conf{update_dir}, 0, 1);
# override registry entries with command-line arguments, if any
foreach my $override (keys %{$self->{pbot}->{overrides}}) {
my $value = $self->{pbot}->{overrides}->{$override};
my ($section, $key) = split /\./, $override;
$self->{pbot}->{logger}->log("Overriding $section.$key to $value\n");
$self->set($section, $key, 'value', $value, 0, 1);
}
# add triggers
$self->add_trigger('irc', 'debug', sub { $self->trigger_irc_debug(@_) });
$self->add_trigger('irc', 'botnick', sub { $self->trigger_change_botnick(@_) });
}
# registry triggers fire when value changes
sub trigger_irc_debug {
my ($self, $section, $item, $newvalue) = @_;
$self->{pbot}->{irc}->debug($newvalue);
if ($self->{pbot}->{connected}) {
$self->{pbot}->{conn}->debug($newvalue);
}
}
sub trigger_change_botnick {
my ($self, $section, $item, $newvalue) = @_;
if ($self->{pbot}->{connected}) {
$self->{pbot}->{conn}->nick($newvalue)
}
}
# registry api
sub load {
2020-02-15 23:38:32 +01:00
my $self = shift;
# load registry from file
2020-02-15 23:38:32 +01:00
$self->{registry}->load;
# fire off all registered triggers
2020-02-15 23:38:32 +01:00
foreach my $section ($self->{registry}->get_keys) {
foreach my $item ($self->{registry}->get_keys($section)) {
$self->process_trigger($section, $item, $self->{registry}->get_data($section, $item, 'value'));
}
}
}
sub save {
2020-02-15 23:38:32 +01:00
my $self = shift;
2020-02-15 23:38:32 +01:00
$self->{registry}->save;
}
sub add_default {
2020-02-15 23:38:32 +01:00
my ($self, $type, $section, $item, $value) = @_;
2020-02-15 23:38:32 +01:00
$self->add($type, $section, $item, $value, 1);
}
sub add {
my ($self, $type, $section, $item, $value, $is_default) = @_;
2020-02-15 23:38:32 +01:00
$type = lc $type;
2020-02-15 23:38:32 +01:00
if (not $self->{registry}->exists($section, $item)) {
# registry entry does not exist
2020-02-15 23:38:32 +01:00
my $data = {
value => $value,
type => $type,
};
2020-02-15 23:38:32 +01:00
$self->{registry}->add($section, $item, $data, 1);
} else {
# registry entry already exists
if ($is_default) {
# don't replace existing registry values if we're just adding a default value
return;
}
# update value
2020-02-15 23:38:32 +01:00
$self->{registry}->set($section, $item, 'value', $value, 1);
# update type only if it doesn't exist
unless ($self->{registry}->exists($section, $item, 'type')) {
$self->{registry}->set($section, $item, 'type', $type, 1);
}
2020-02-15 23:38:32 +01:00
}
unless ($is_default) {
$self->process_trigger($section, $item, $value);
$self->save;
}
}
sub remove {
my ($self, $section, $item) = @_;
2020-02-15 23:38:32 +01:00
$self->{registry}->remove($section, $item);
}
sub set_default {
2020-02-15 23:38:32 +01:00
my ($self, $section, $item, $key, $value) = @_;
2020-02-15 23:38:32 +01:00
$self->set($section, $item, $key, $value, 1);
}
sub set {
2020-02-15 23:38:32 +01:00
my ($self, $section, $item, $key, $value, $is_default, $dont_save) = @_;
2020-02-15 23:38:32 +01:00
$key = lc $key if defined $key;
if ($is_default && $self->{registry}->exists($section, $item, $key)) {
return;
}
my $oldvalue;
if (defined $value) {
$oldvalue = $self->get_value($section, $item, 1);
}
$oldvalue //= '';
2020-02-15 23:38:32 +01:00
my $result = $self->{registry}->set($section, $item, $key, $value, 1);
if (defined $key and $key eq 'value' and defined $value and $oldvalue ne $value) {
$self->process_trigger($section, $item, $value);
}
2020-02-15 23:38:32 +01:00
$self->save if !$dont_save && $result =~ m/set to/ && not $is_default;
2020-02-15 23:38:32 +01:00
return $result;
}
sub unset {
2020-02-15 23:38:32 +01:00
my ($self, $section, $item, $key) = @_;
$key = lc $key if defined $key;
2020-02-15 23:38:32 +01:00
return $self->{registry}->unset($section, $item, $key);
}
sub get_value {
2020-05-02 05:59:51 +02:00
my ($self, $section, $item, $as_text, $context) = @_;
2020-02-15 23:38:32 +01:00
$section = lc $section;
$item = lc $item;
2020-02-15 23:38:32 +01:00
my $key = $item;
# TODO: use user-metadata for this
2020-05-02 05:59:51 +02:00
if (defined $context and exists $context->{nick}) {
my $context_nick = lc $context->{nick};
if ($self->{registry}->exists($section, "$item.nick.$context_nick")) {
$key = "$item.nick.$context_nick";
}
}
2020-02-15 23:38:32 +01:00
if ($self->{registry}->exists($section, $key)) {
if (not $as_text and $self->{registry}->get_data($section, $key, 'type') eq 'array') {
return split /\s*,\s*/, $self->{registry}->get_data($section, $key, 'value');
} else {
return $self->{registry}->get_data($section, $key, 'value');
}
}
2020-02-15 23:38:32 +01:00
return undef;
}
sub get_array_value {
2020-05-02 05:59:51 +02:00
my ($self, $section, $item, $index, $context) = @_;
2020-02-15 23:38:32 +01:00
$section = lc $section;
$item = lc $item;
2020-02-15 23:38:32 +01:00
my $key = $item;
# TODO: use user-metadata for this
2020-05-02 05:59:51 +02:00
if (defined $context and exists $context->{nick}) {
my $context_nick = lc $context->{nick};
if ($self->{registry}->exists($section, "$item.nick.$context_nick")) {
$key = "$item.nick.$context_nick";
}
}
2020-02-15 23:38:32 +01:00
if ($self->{registry}->exists($section, $key)) {
if ($self->{registry}->get_data($section, $key, 'type') eq 'array') {
my @array = split /\s*,\s*/, $self->{registry}->get_data($section, $key, 'value');
return $array[$index >= $#array ? $#array : $index];
} else {
return $self->{registry}->get_data($section, $key, 'value');
}
}
2020-02-15 23:38:32 +01:00
return undef;
}
sub add_trigger {
2020-02-15 23:38:32 +01:00
my ($self, $section, $item, $subref) = @_;
2020-02-15 23:38:32 +01:00
$self->{triggers}->{lc $section}->{lc $item} = $subref;
}
sub process_trigger {
my $self = shift; # shift $self off of the top of @_
my ($section, $item) = @_; # but leave $section, $item and anything else (i.e. $value) in @_
2020-02-15 23:38:32 +01:00
$section = lc $section;
$item = lc $item;
if (exists $self->{triggers}->{$section} and exists $self->{triggers}->{$section}->{$item}) {
return &{$self->{triggers}->{$section}->{$item}}(@_); # $section, $item, $value, etc in @_
}
2020-02-15 23:38:32 +01:00
return undef;
}
1;