mirror of
https://github.com/pragma-/pbot.git
synced 2024-11-01 01:29:31 +01:00
02552081d5
Misc clean-ups and refactors
223 lines
6.2 KiB
Perl
223 lines
6.2 KiB
Perl
# File: Registry.pm
|
|
# Author: pragma_
|
|
#
|
|
# Purpose: Provides a centralized registry of configuration settings that can
|
|
# easily be examined and updated via set/unset commands without restarting.
|
|
|
|
# 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 warnings;
|
|
use strict;
|
|
|
|
use feature 'unicode_strings';
|
|
|
|
use Time::HiRes qw(gettimeofday);
|
|
use Carp ();
|
|
|
|
use PBot::DualIndexHashObject;
|
|
use PBot::RegistryCommands;
|
|
|
|
sub new {
|
|
if (ref($_[1]) eq 'HASH') {
|
|
Carp::croak("Options to " . __FILE__ . " should be item/value pairs, not hash reference");
|
|
}
|
|
|
|
my ($class, %conf) = @_;
|
|
my $self = bless {}, $class;
|
|
$self->initialize(%conf);
|
|
return $self;
|
|
}
|
|
|
|
sub initialize {
|
|
my ($self, %conf) = @_;
|
|
|
|
$self->{pbot} = $conf{pbot} // Carp::croak("Missing pbot reference to " . __FILE__);
|
|
my $filename = $conf{filename} // Carp::croak("Missing filename reference in " . __FILE__);
|
|
|
|
$self->{registry} = PBot::DualIndexHashObject->new(name => 'Registry', filename => $filename, pbot => $self->{pbot});
|
|
$self->{triggers} = {};
|
|
|
|
$self->{pbot}->{atexit}->register(sub { $self->save; return; });
|
|
|
|
PBot::RegistryCommands->new(pbot => $self->{pbot});
|
|
}
|
|
|
|
sub load {
|
|
my $self = shift;
|
|
$self->{registry}->load;
|
|
foreach my $section (keys %{ $self->{registry}->{hash} }) {
|
|
foreach my $item (keys %{ $self->{registry}->{hash}->{$section} }) {
|
|
next if $item eq '_name';
|
|
$self->process_trigger($section, $item, $self->{registry}->{hash}->{$section}->{$item}->{value});
|
|
}
|
|
}
|
|
}
|
|
|
|
sub save {
|
|
my $self = shift;
|
|
$self->{registry}->save;
|
|
}
|
|
|
|
sub add_default {
|
|
my ($self, $type, $section, $item, $value) = @_;
|
|
$self->add($type, $section, $item, $value, 1);
|
|
}
|
|
|
|
sub add {
|
|
my $self = shift;
|
|
my ($type, $section, $item, $value, $is_default) = @_;
|
|
|
|
my $lc_section = lc $section;
|
|
my $lc_item = lc $item;
|
|
|
|
$type = lc $type;
|
|
|
|
if ($is_default) {
|
|
return if exists $self->{registry}->{hash}->{$lc_section} and exists $self->{registry}->{hash}->{$lc_section}->{$lc_item};
|
|
}
|
|
|
|
if (not exists $self->{registry}->{hash}->{$lc_section}) {
|
|
$self->{registry}->{hash}->{$lc_section}->{_name} = $section;
|
|
}
|
|
|
|
if (not exists $self->{registry}->{hash}->{$lc_section}->{$lc_item}) {
|
|
$self->{registry}->{hash}->{$lc_section}->{$lc_item}->{_name} = $item;
|
|
}
|
|
|
|
$self->{registry}->{hash}->{$lc_section}->{$lc_item}->{value} = $value;
|
|
$self->{registry}->{hash}->{$lc_section}->{$lc_item}->{type} = $type unless exists $self->{registry}->{hash}->{$lc_section}->{$lc_item}->{type};
|
|
|
|
$self->process_trigger($section, $item, $value) unless $is_default;
|
|
$self->save unless $is_default;
|
|
}
|
|
|
|
sub remove {
|
|
my $self = shift;
|
|
my ($section, $item) = @_;
|
|
|
|
$section = lc $section;
|
|
$item = lc $item;
|
|
|
|
delete $self->{registry}->{hash}->{$section}->{$item};
|
|
|
|
if (not scalar keys %{ $self->{registry}->{hash}->{$section} }) {
|
|
delete $self->{registry}->{hash}->{$section};
|
|
}
|
|
|
|
$self->save;
|
|
}
|
|
|
|
sub set_default {
|
|
my ($self, $section, $item, $key, $value) = @_;
|
|
$self->set($section, $item, $key, $value, 1);
|
|
}
|
|
|
|
sub set {
|
|
my ($self, $section, $item, $key, $value, $is_default, $dont_save) = @_;
|
|
|
|
$section = lc $section;
|
|
$item = lc $item;
|
|
$key = lc $key if defined $key;
|
|
|
|
if ($is_default) {
|
|
return if exists $self->{registry}->{hash}->{$section}
|
|
and exists $self->{registry}->{hash}->{$section}->{$item}
|
|
and exists $self->{registry}->{hash}->{$section}->{$item}->{$key};
|
|
}
|
|
|
|
my $oldvalue = $self->get_value($section, $item, 1) if defined $value;
|
|
$oldvalue = '' if not defined $oldvalue;
|
|
|
|
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);
|
|
}
|
|
|
|
$self->save if !$dont_save && $result =~ m/set to/ && not $is_default;
|
|
|
|
return $result;
|
|
}
|
|
|
|
sub unset {
|
|
my ($self, $section, $item, $key) = @_;
|
|
|
|
$section = lc $section;
|
|
$item = lc $item;
|
|
$key = lc $key;
|
|
|
|
my $result = $self->{registry}->unset($section, $item, $key);
|
|
$self->save;
|
|
return $result;
|
|
}
|
|
|
|
sub get_value {
|
|
my ($self, $section, $item, $as_text, $stuff) = @_;
|
|
$section = lc $section;
|
|
$item = lc $item;
|
|
my $key = $item;
|
|
|
|
if (defined $stuff and exists $stuff->{nick}) {
|
|
my $stuff_nick = lc $stuff->{nick};
|
|
if (exists $self->{registry}->{hash}->{$section} and exists $self->{registry}->{hash}->{$section}->{"$item.nick.$stuff_nick"}) {
|
|
$key = "$item.nick.$stuff_nick";
|
|
}
|
|
}
|
|
|
|
if (exists $self->{registry}->{hash}->{$section} and exists $self->{registry}->{hash}->{$section}->{$key}) {
|
|
if (not $as_text and $self->{registry}->{hash}->{$section}->{$key}->{type} eq 'array') {
|
|
return split /\s*,\s*/, $self->{registry}->{hash}->{$section}->{$key}->{value};
|
|
} else {
|
|
return $self->{registry}->{hash}->{$section}->{$key}->{value};
|
|
}
|
|
}
|
|
return undef;
|
|
}
|
|
|
|
sub get_array_value {
|
|
my ($self, $section, $item, $index, $stuff) = @_;
|
|
$section = lc $section;
|
|
$item = lc $item;
|
|
my $key = $item;
|
|
|
|
if (defined $stuff and exists $stuff->{nick}) {
|
|
my $stuff_nick = lc $stuff->{nick};
|
|
if (exists $self->{registry}->{hash}->{$section} and exists $self->{registry}->{hash}->{$section}->{"$item.nick.$stuff_nick"}) {
|
|
$key = "$item.nick.$stuff_nick";
|
|
}
|
|
}
|
|
|
|
if (exists $self->{registry}->{hash}->{$section} and exists $self->{registry}->{hash}->{$section}->{$key}) {
|
|
if ($self->{registry}->{hash}->{$section}->{$key}->{type} eq 'array') {
|
|
my @array = split /\s*,\s*/, $self->{registry}->{hash}->{$section}->{$key}->{value};
|
|
return $array[$index >= $#array ? $#array : $index];
|
|
} else {
|
|
return $self->{registry}->{hash}->{$section}->{$key}->{value};
|
|
}
|
|
}
|
|
return undef;
|
|
}
|
|
|
|
sub add_trigger {
|
|
my ($self, $section, $item, $subref) = @_;
|
|
$self->{triggers}->{lc $section}->{lc $item} = $subref;
|
|
}
|
|
|
|
sub process_trigger {
|
|
my $self = shift;
|
|
my ($section, $item) = @_;
|
|
$section = lc $section;
|
|
$item = lc $item;
|
|
|
|
if (exists $self->{triggers}->{$section} and exists $self->{triggers}->{$section}->{$item}) {
|
|
return &{ $self->{triggers}->{$section}->{$item} }(@_);
|
|
}
|
|
return undef;
|
|
}
|
|
|
|
1;
|