2020-02-26 11:20:11 +01:00
|
|
|
# File: DualIndexSQLiteObject.pm
|
|
|
|
# Author: pragma_
|
|
|
|
#
|
|
|
|
# Purpose: Provides a dual-indexed SQLite object with an abstracted API that includes
|
2020-03-04 14:54:20 +01:00
|
|
|
# setting and deleting values, caching, displaying nearest matches, etc. Designed to
|
|
|
|
# be as compatible as possible with DualIndexHashObject; e.g. get_keys, get_data, etc.
|
2020-02-26 11:20:11 +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::DualIndexSQLiteObject;
|
|
|
|
|
|
|
|
use warnings; use strict;
|
|
|
|
use feature 'unicode_strings';
|
|
|
|
|
|
|
|
use DBI;
|
|
|
|
use Text::Levenshtein qw(fastdistance);
|
|
|
|
|
|
|
|
sub new {
|
|
|
|
my ($proto, %conf) = @_;
|
|
|
|
my $class = ref($proto) || $proto;
|
|
|
|
my $self = bless {}, $class;
|
|
|
|
Carp::croak("Missing pbot reference to " . __FILE__) unless exists $conf{pbot};
|
|
|
|
$self->{pbot} = $conf{pbot};
|
|
|
|
$self->initialize(%conf);
|
|
|
|
return $self;
|
|
|
|
}
|
|
|
|
|
|
|
|
sub initialize {
|
|
|
|
my ($self, %conf) = @_;
|
|
|
|
$self->{name} = $conf{name} // 'Dual Index SQLite object';
|
|
|
|
$self->{filename} = $conf{filename} // Carp::croak("Missing filename in " . __FILE__);
|
|
|
|
|
|
|
|
$self->{dbh} = undef;
|
|
|
|
$self->{cache} = {};
|
|
|
|
$self->{cache_timeouts} = {};
|
|
|
|
|
|
|
|
$self->{debug} = $conf{debug} // '';
|
|
|
|
|
|
|
|
$self->{pbot}->{registry}->add_default('text', 'dualindexsqliteobject', "debug_$self->{name}", 0);
|
|
|
|
$self->{pbot}->{registry}->add_trigger('dualindexsqliteobject', "debug_$self->{name}", sub { $self->sqlite_debug_trigger(@_) });
|
|
|
|
|
2020-05-15 01:55:23 +02:00
|
|
|
$self->{pbot}->{atexit}->register(sub { $self->end; return; });
|
2020-02-26 11:20:11 +01:00
|
|
|
$self->{pbot}->{timer}->register(sub {$self->trim_cache }, 60, "DualIndexSQLiteObject $self->{name} Timer");
|
|
|
|
|
|
|
|
$self->begin;
|
|
|
|
}
|
|
|
|
|
|
|
|
sub sqlite_debug_trigger {
|
|
|
|
my ($self, $section, $item, $newvalue) = @_;
|
|
|
|
$self->{dbh}->trace($self->{dbh}->parse_trace_flags("SQL|$newvalue")) if defined $self->{dbh};
|
|
|
|
}
|
|
|
|
|
|
|
|
sub begin {
|
|
|
|
my ($self) = @_;
|
|
|
|
|
|
|
|
$self->{pbot}->{logger}->log("Opening $self->{name} database ($self->{filename})\n");
|
|
|
|
|
|
|
|
$self->{dbh} = DBI->connect(
|
|
|
|
"dbi:SQLite:dbname=$self->{filename}", "", "",
|
|
|
|
{RaiseError => 1, PrintError => 0, AutoInactiveDestroy => 1, sqlite_unicode => 1}
|
|
|
|
) or die $DBI::errstr;
|
|
|
|
|
|
|
|
eval {
|
|
|
|
my $sqlite_debug = $self->{pbot}->{registry}->get_value('dualindexsqliteobject', "debug_$self->{name}");
|
|
|
|
use PBot::SQLiteLoggerLayer;
|
|
|
|
use PBot::SQLiteLogger;
|
|
|
|
open $self->{trace_layer}, '>:via(PBot::SQLiteLoggerLayer)', PBot::SQLiteLogger->new(pbot => $self->{pbot});
|
|
|
|
$self->{dbh}->trace($self->{dbh}->parse_trace_flags("SQL|$sqlite_debug"), $self->{trace_layer});
|
|
|
|
};
|
|
|
|
|
|
|
|
if ($@) { $self->{pbot}->{logger}->log("Error initializing $self->{name} database: $@\n"); }
|
|
|
|
}
|
|
|
|
|
|
|
|
sub end {
|
|
|
|
my ($self) = @_;
|
|
|
|
|
|
|
|
$self->{pbot}->{logger}->log("Closing $self->{name} database ($self->{filename})\n");
|
|
|
|
|
|
|
|
if (defined $self->{dbh}) {
|
|
|
|
$self->{dbh}->disconnect;
|
|
|
|
$self->{dbh} = undef;
|
|
|
|
}
|
|
|
|
|
|
|
|
$self->{pbot}->{timer}->unregister("DualIndexSQLiteObject $self->{name} Timer");
|
|
|
|
}
|
|
|
|
|
2020-02-29 10:45:17 +01:00
|
|
|
sub load {
|
|
|
|
my ($self) = @_;
|
|
|
|
$self->create_database;
|
|
|
|
$self->create_cache;
|
|
|
|
}
|
|
|
|
|
2020-02-26 11:20:11 +01:00
|
|
|
sub create_database {
|
|
|
|
my ($self) = @_;
|
|
|
|
|
|
|
|
eval {
|
|
|
|
$self->{dbh}->do(<<SQL);
|
|
|
|
CREATE TABLE IF NOT EXISTS Stuff (
|
|
|
|
index1 TEXT COLLATE NOCASE,
|
|
|
|
index2 TEXT COLLATE NOCASE
|
|
|
|
)
|
|
|
|
SQL
|
|
|
|
|
|
|
|
$self->{dbh}->do('CREATE INDEX IF NOT EXISTS idx1 ON Stuff (index1, index2)');
|
|
|
|
};
|
|
|
|
|
|
|
|
$self->{pbot}->{logger}->log($@) if $@;
|
|
|
|
}
|
|
|
|
|
|
|
|
sub create_cache {
|
|
|
|
my ($self) = @_;
|
|
|
|
$self->{cache} = {};
|
|
|
|
|
|
|
|
my ($index1_count, $index2_count) = (0, 0);
|
|
|
|
foreach my $index1 ($self->get_keys(undef, undef, 1)) {
|
|
|
|
$index1_count++;
|
|
|
|
my $lc_index1 = lc $index1;
|
|
|
|
foreach my $index2 ($self->get_keys($lc_index1, undef, 1)) {
|
|
|
|
my $lc_index2 = lc $index2;
|
|
|
|
$index2_count++;
|
|
|
|
$self->{cache}->{$lc_index1}->{lc $index2} = {};
|
|
|
|
$self->{cache}->{$lc_index1}->{_name} = $index1 if $index1 ne $lc_index1;
|
|
|
|
$self->{cache}->{$lc_index1}->{$lc_index2}->{_name} = $index2 if $index2 ne $lc_index2;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
$self->{pbot}->{logger}->log("Cached $index2_count $self->{name} objects in $index1_count groups.\n");
|
|
|
|
}
|
|
|
|
|
|
|
|
sub cache_remove {
|
|
|
|
my ($self, $index1, $index2) = @_;
|
|
|
|
|
|
|
|
if (not defined $index2) {
|
|
|
|
delete $self->{cache}->{$index1};
|
|
|
|
delete $self->{cache_timeouts}->{$index1};
|
|
|
|
} else {
|
|
|
|
delete $self->{cache}->{$index1}->{$index2};
|
|
|
|
delete $self->{cache}->{$index1} if not grep { $_ ne '_name' } keys %{$self->{cache}->{$index1}};
|
|
|
|
delete $self->{cache_timeouts}->{$index1}->{$index2};
|
|
|
|
delete $self->{cache_timeouts}->{$index1} if not keys %{$self->{cache_timeouts}->{$index1}};
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
sub trim_cache {
|
|
|
|
my ($self) = @_;
|
|
|
|
|
|
|
|
my $now = time;
|
|
|
|
foreach my $index1 (keys %{$self->{cache_timeouts}}) {
|
|
|
|
foreach my $index2 (keys %{$self->{cache_timeouts}->{$index1}}) {
|
|
|
|
if ($now >= $self->{cache_timeouts}->{$index1}->{$index2}) {
|
|
|
|
my $name = $self->{cache}->{$index1}->{$index2}->{_name};
|
|
|
|
$self->{cache}->{$index1}->{$index2} = {};
|
|
|
|
$self->{cache}->{$index1}->{$index2}->{_name} = $name if defined $name;
|
|
|
|
delete $self->{cache_timeouts}->{$index1}->{$index2};
|
|
|
|
delete $self->{cache_timeouts}->{$index1} if not keys %{$self->{cache_timeouts}->{$index1}};
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
sub create_metadata {
|
|
|
|
my ($self, $columns) = @_;
|
|
|
|
return if not $self->{dbh};
|
|
|
|
|
|
|
|
$self->{columns} = $columns;
|
|
|
|
|
|
|
|
eval {
|
|
|
|
my %existing = ();
|
|
|
|
foreach my $col (@{$self->{dbh}->selectall_arrayref("PRAGMA TABLE_INFO(Stuff)")}) { $existing{$col->[1]} = $col->[2]; }
|
|
|
|
|
|
|
|
$self->{dbh}->begin_work;
|
|
|
|
|
|
|
|
foreach my $col (sort keys %$columns) {
|
|
|
|
unless (exists $existing{$col}) { $self->{dbh}->do("ALTER TABLE Stuff ADD COLUMN \"$col\" $columns->{$col}"); }
|
|
|
|
}
|
|
|
|
|
|
|
|
$self->{dbh}->commit;
|
|
|
|
};
|
|
|
|
|
|
|
|
if ($@) {
|
|
|
|
$self->{pbot}->{logger}->log($@);
|
|
|
|
$self->{dbh}->rollback;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
sub levenshtein_matches {
|
|
|
|
my ($self, $index1, $index2, $distance, $strictnamespace) = @_;
|
|
|
|
my $comma = '';
|
|
|
|
my $result = '';
|
|
|
|
|
|
|
|
$distance = 0.60 if not defined $distance;
|
|
|
|
|
|
|
|
$index1 = '.*' if not defined $index1;
|
|
|
|
|
|
|
|
if (not $index2) {
|
|
|
|
foreach my $index (sort $self->get_keys) {
|
|
|
|
my $distance_result = fastdistance($index1, $index);
|
|
|
|
my $length = (length $index1 > length $index) ? length $index1 : length $index;
|
|
|
|
|
|
|
|
if ($distance_result / $length < $distance) {
|
|
|
|
my $name = $self->get_data($index, '_name');
|
|
|
|
if ($name =~ / /) { $result .= $comma . "\"$name\""; }
|
|
|
|
else { $result .= $comma . $name; }
|
|
|
|
$comma = ", ";
|
|
|
|
}
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
if (not $self->exists($index1)) { return 'none'; }
|
|
|
|
|
|
|
|
my $last_header = '';
|
|
|
|
my $header = '';
|
|
|
|
|
|
|
|
foreach my $i1 (sort $self->get_keys) {
|
|
|
|
$header = '[' . $self->get_data($i1, '_name') . '] ';
|
|
|
|
$header = '[global] ' if $header eq '[.*] ';
|
|
|
|
|
|
|
|
if ($strictnamespace) {
|
|
|
|
next unless $i1 eq '.*' or lc $i1 eq lc $index1;
|
|
|
|
$header = "" unless $header eq '[global] ';
|
|
|
|
}
|
|
|
|
|
|
|
|
foreach my $i2 (sort $self->get_keys($i1)) {
|
|
|
|
my $distance_result = fastdistance($index2, $i2);
|
|
|
|
my $length = (length $index2 > length $i2) ? length $index2 : length $i2;
|
|
|
|
|
|
|
|
if ($distance_result / $length < $distance) {
|
|
|
|
my $name = $self->get_data($i1, $i2, '_name');
|
|
|
|
$header = "" if $last_header eq $header;
|
|
|
|
$last_header = $header;
|
|
|
|
$comma = '; ' if $comma ne '' and $header ne '';
|
|
|
|
if ($name =~ / /) { $result .= $comma . $header . "\"$name\""; }
|
|
|
|
else { $result .= $comma . $header . $name; }
|
|
|
|
$comma = ", ";
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
$result =~ s/(.*), /$1 or /;
|
|
|
|
$result = 'none' if $comma eq '';
|
|
|
|
return $result;
|
|
|
|
}
|
|
|
|
|
|
|
|
sub exists {
|
|
|
|
my ($self, $index1, $index2, $data_index) = @_;
|
|
|
|
return 0 if not defined $index1;
|
|
|
|
$index1 = lc $index1;
|
|
|
|
return 0 if not grep { $_ eq $index1 } $self->get_keys;
|
|
|
|
return 1 if not defined $index2;
|
|
|
|
$index2 = lc $index2;
|
|
|
|
return 0 if not grep { $_ eq $index2 } $self->get_keys($index1);
|
|
|
|
return 1 if not defined $data_index;
|
|
|
|
return defined $self->get_data($index1, $index2, $data_index);
|
|
|
|
}
|
|
|
|
|
|
|
|
sub get_keys {
|
|
|
|
my ($self, $index1, $index2, $nocache) = @_;
|
|
|
|
my @keys;
|
|
|
|
|
|
|
|
if (not defined $index1) {
|
|
|
|
if (not $nocache) { return keys %{$self->{cache}}; }
|
|
|
|
|
|
|
|
@keys = eval {
|
2020-05-02 05:59:51 +02:00
|
|
|
my $context = $self->{dbh}->selectall_arrayref('SELECT DISTINCT index1 FROM Stuff');
|
|
|
|
if (@$context) {
|
|
|
|
return map { $_->[0] } @$context;
|
2020-02-26 11:20:11 +01:00
|
|
|
} else {
|
|
|
|
return ();
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
if ($@) {
|
|
|
|
$self->{pbot}->{logger}->log($@);
|
|
|
|
return undef;
|
|
|
|
}
|
|
|
|
|
|
|
|
return @keys;
|
|
|
|
}
|
|
|
|
|
|
|
|
$index1 = lc $index1;
|
|
|
|
|
|
|
|
if (not defined $index2) {
|
|
|
|
if (not $nocache) { return grep { $_ ne '_name' } keys %{$self->{cache}->{$index1}}; }
|
|
|
|
|
|
|
|
@keys = eval {
|
|
|
|
my $sth = $self->{dbh}->prepare('SELECT index2 FROM Stuff WHERE index1 = ?');
|
|
|
|
$sth->execute($index1);
|
2020-05-02 05:59:51 +02:00
|
|
|
my $context = $sth->fetchall_arrayref;
|
|
|
|
if (@$context) {
|
|
|
|
return map { $_->[0] } @$context;
|
2020-02-26 11:20:11 +01:00
|
|
|
} else {
|
|
|
|
return ();
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
if ($@) {
|
|
|
|
$self->{pbot}->{logger}->log($@);
|
|
|
|
return ();
|
|
|
|
}
|
|
|
|
|
|
|
|
return @keys;
|
|
|
|
}
|
|
|
|
|
|
|
|
$index2 = lc $index2;
|
|
|
|
|
|
|
|
if (not $nocache) {
|
|
|
|
@keys = grep { $_ ne '_name' } keys %{$self->{cache}->{$index1}->{$index2}};
|
|
|
|
return @keys if @keys;
|
|
|
|
}
|
|
|
|
|
|
|
|
@keys = eval {
|
|
|
|
my $sth = $self->{dbh}->prepare('SELECT * FROM Stuff WHERE index1 = ? AND index2 = ?');
|
|
|
|
$sth->execute($index1, $index2);
|
2020-05-02 05:59:51 +02:00
|
|
|
my $context = $sth->fetchall_arrayref({});
|
2020-02-26 11:20:11 +01:00
|
|
|
my @k = ();
|
2020-05-02 05:59:51 +02:00
|
|
|
return @k if not @{$context};
|
2020-02-26 11:20:11 +01:00
|
|
|
|
|
|
|
my ($lc_index1, $lc_index2) = (lc $index1, lc $index2);
|
2020-05-02 05:59:51 +02:00
|
|
|
foreach my $key (keys %{$context->[0]}) {
|
2020-02-26 11:20:11 +01:00
|
|
|
next if $key eq 'index1' or $key eq 'index2';
|
2020-05-02 05:59:51 +02:00
|
|
|
push @k, $key if defined $context->[0]->{$key};
|
|
|
|
$self->{cache}->{$lc_index1}->{$lc_index2}->{$key} = $context->[0]->{$key};
|
2020-02-26 11:20:11 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
my $timeout = $self->{pbot}->{registry}->get_value('dualindexsqliteobject', 'cache_timeout') // 60 * 30;
|
|
|
|
$self->{cache_timeouts}->{$lc_index1}->{$lc_index2} = time + $timeout;
|
|
|
|
|
|
|
|
return @k;
|
|
|
|
};
|
|
|
|
|
|
|
|
if ($@) {
|
|
|
|
$self->{pbot}->{logger}->log($@);
|
|
|
|
return ();
|
|
|
|
}
|
|
|
|
|
|
|
|
return @keys;
|
|
|
|
}
|
|
|
|
|
2020-02-29 10:45:17 +01:00
|
|
|
sub get_each {
|
2020-03-03 16:40:23 +01:00
|
|
|
my ($self, @opts) = @_;
|
2020-02-26 11:20:11 +01:00
|
|
|
|
2020-02-29 10:45:17 +01:00
|
|
|
my $sth = eval {
|
2020-02-26 11:20:11 +01:00
|
|
|
my $sql = 'SELECT ';
|
|
|
|
my @keys = ();
|
2020-03-03 16:40:23 +01:00
|
|
|
my @values = ();
|
2020-02-26 11:20:11 +01:00
|
|
|
my @where = ();
|
2020-03-03 16:40:23 +01:00
|
|
|
my @sort = ();
|
|
|
|
my $everything = 0;
|
2020-02-26 11:20:11 +01:00
|
|
|
|
2020-03-03 16:40:23 +01:00
|
|
|
foreach my $expr (@opts) {
|
|
|
|
my ($key, $op, $value) = split /\s*([!=<>]+)\s*/, $expr, 3;
|
2020-02-29 10:45:17 +01:00
|
|
|
|
|
|
|
if ($key eq '_everything') {
|
2020-03-03 16:40:23 +01:00
|
|
|
$everything = 1;
|
2020-02-29 10:45:17 +01:00
|
|
|
push @keys, '*';
|
|
|
|
next;
|
|
|
|
}
|
|
|
|
|
2020-03-03 16:40:23 +01:00
|
|
|
if ($key eq '_sort') {
|
2020-03-03 17:55:34 +01:00
|
|
|
if ($value =~ s/^\-//) {
|
2020-03-03 16:40:23 +01:00
|
|
|
push @sort, "$value DESC";
|
2020-02-27 20:14:43 +01:00
|
|
|
} else {
|
2020-03-03 17:55:34 +01:00
|
|
|
$value =~ s/^\+//; # optional
|
2020-03-03 16:40:23 +01:00
|
|
|
push @sort, "$value ASC";
|
2020-02-27 20:14:43 +01:00
|
|
|
}
|
2020-03-03 16:40:23 +01:00
|
|
|
next;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (defined $op) {
|
2020-03-03 19:10:20 +01:00
|
|
|
my $prefix = 'AND';
|
2020-03-03 16:40:23 +01:00
|
|
|
|
|
|
|
if ($op eq '=' or $op eq '==') {
|
|
|
|
$op = '=';
|
|
|
|
} elsif ($op eq '!=' or $op eq '<>') {
|
|
|
|
$op = '!=';
|
|
|
|
}
|
|
|
|
|
2020-03-03 19:10:20 +01:00
|
|
|
if ($key =~ s/^(OR|AND)\s+//) {
|
2020-03-03 16:40:23 +01:00
|
|
|
$prefix = $1;
|
|
|
|
}
|
|
|
|
|
|
|
|
$prefix = '' if not @where;
|
2020-03-03 19:10:20 +01:00
|
|
|
push @where, [ $prefix, $key, $op ];
|
2020-03-03 16:40:23 +01:00
|
|
|
push @values, $value;
|
2020-02-26 11:20:11 +01:00
|
|
|
}
|
2020-02-29 10:45:17 +01:00
|
|
|
|
2020-03-03 16:40:23 +01:00
|
|
|
push @keys, $key unless $everything or grep { $_ eq $key } @keys;
|
2020-02-26 11:20:11 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
$sql .= join ', ', @keys;
|
2020-03-03 19:10:20 +01:00
|
|
|
$sql .= ' FROM Stuff WHERE';
|
|
|
|
|
|
|
|
my $in_or = 0;
|
|
|
|
for (my $i = 0; $i < @where; $i++) {
|
|
|
|
my ($prefix, $key, $op) = @{$where[$i]};
|
|
|
|
my ($next_prefix, $next_key) = ('', '');
|
|
|
|
|
|
|
|
if ($i < @where - 1) {
|
|
|
|
($next_prefix, $next_key) = @{$where[$i + 1]};
|
|
|
|
}
|
|
|
|
|
|
|
|
if ($next_prefix eq 'OR' and $next_key eq $key) {
|
|
|
|
$sql .= "$prefix ";
|
|
|
|
$sql .= '(' if not $in_or;
|
|
|
|
$sql .= "\"$key\" $op ? ";
|
|
|
|
$in_or = 1;
|
|
|
|
} else {
|
|
|
|
$sql .= "$prefix \"$key\" $op ? ";
|
|
|
|
|
|
|
|
if ($in_or) {
|
|
|
|
$sql .= ') ';
|
|
|
|
$in_or = 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
$sql .= ')' if $in_or;
|
|
|
|
|
2020-03-03 16:40:23 +01:00
|
|
|
$sql .= ' ORDER BY ' . join(', ', @sort) if @sort;
|
2020-02-26 11:20:11 +01:00
|
|
|
|
|
|
|
my $sth = $self->{dbh}->prepare($sql);
|
|
|
|
|
|
|
|
my $param = 0;
|
2020-03-03 16:40:23 +01:00
|
|
|
foreach my $value (@values) {
|
|
|
|
$sth->bind_param(++$param, $value);
|
2020-02-26 11:20:11 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
$sth->execute;
|
2020-02-29 10:45:17 +01:00
|
|
|
return $sth;
|
|
|
|
};
|
|
|
|
|
|
|
|
if ($@) {
|
|
|
|
$self->{pbot}->{logger}->log("Error getting data: $@\n");
|
|
|
|
return undef;
|
|
|
|
}
|
|
|
|
|
|
|
|
return $sth;
|
|
|
|
}
|
|
|
|
|
|
|
|
sub get_next {
|
|
|
|
my ($self, $sth) = @_;
|
|
|
|
|
|
|
|
my $data = eval {
|
|
|
|
return $sth->fetchrow_hashref;
|
|
|
|
};
|
|
|
|
|
|
|
|
if ($@) {
|
|
|
|
$self->{pbot}->{logger}->log("Error getting next: $@\n");
|
|
|
|
return undef;
|
|
|
|
}
|
|
|
|
|
|
|
|
return $data;
|
|
|
|
}
|
|
|
|
|
|
|
|
sub get_all {
|
2020-03-03 16:40:23 +01:00
|
|
|
my ($self, @opts) = @_;
|
2020-02-29 10:45:17 +01:00
|
|
|
|
2020-03-03 16:40:23 +01:00
|
|
|
my $sth = $self->get_each(@opts);
|
2020-02-29 10:45:17 +01:00
|
|
|
|
|
|
|
my $data = eval {
|
2020-02-26 11:20:11 +01:00
|
|
|
return $sth->fetchall_arrayref({});
|
|
|
|
};
|
|
|
|
|
|
|
|
if ($@) {
|
|
|
|
$self->{pbot}->{logger}->log("Error getting data: $@\n");
|
|
|
|
return undef;
|
|
|
|
}
|
|
|
|
|
|
|
|
return @$data;
|
|
|
|
}
|
|
|
|
|
2020-05-01 18:54:50 +02:00
|
|
|
sub get_key_name {
|
|
|
|
my ($self, $index1, $index2) = @_;
|
|
|
|
|
|
|
|
my $lc_index1 = lc $index1;
|
|
|
|
|
|
|
|
return $lc_index1 if not exists $self->{cache}->{$lc_index1};
|
|
|
|
|
|
|
|
if (not defined $index2) {
|
|
|
|
if (exists $self->{cache}->{$lc_index1}->{_name}) {
|
|
|
|
return $self->{cache}->{$lc_index1}->{_name};
|
|
|
|
} else {
|
|
|
|
return $lc_index1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
my $lc_index2 = lc $index2;
|
|
|
|
|
|
|
|
return $lc_index2 if not exists $self->{cache}->{$lc_index1}->{$lc_index2};
|
|
|
|
|
|
|
|
if (exists $self->{cache}->{$lc_index1}->{$lc_index2}->{_name}) {
|
|
|
|
return $self->{cache}->{$lc_index1}->{$lc_index2}->{_name};
|
|
|
|
} else {
|
|
|
|
return $lc_index2;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-02-26 11:20:11 +01:00
|
|
|
sub get_data {
|
|
|
|
my ($self, $index1, $index2, $data_index) = @_;
|
|
|
|
|
|
|
|
my $lc_index1 = lc $index1;
|
|
|
|
my $lc_index2 = lc $index2;
|
|
|
|
|
|
|
|
return undef if not exists $self->{cache}->{$lc_index1};
|
|
|
|
return undef if not exists $self->{cache}->{$lc_index1}->{$lc_index2} and $lc_index2 ne '_name';
|
|
|
|
|
|
|
|
if (not defined $data_index) {
|
2020-02-27 02:49:59 +01:00
|
|
|
# special case for compatibility with DualIndexHashObject
|
2020-02-26 11:20:11 +01:00
|
|
|
if ($lc_index2 eq '_name') {
|
|
|
|
if (exists $self->{cache}->{$lc_index1}->{_name}) {
|
|
|
|
return $self->{cache}->{$lc_index1}->{_name};
|
|
|
|
} else {
|
|
|
|
return $lc_index1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
my $data = eval {
|
|
|
|
my $sth = $self->{dbh}->prepare('SELECT * FROM Stuff WHERE index1 = ? AND index2 = ?');
|
|
|
|
$sth->execute($index1, $index2);
|
2020-05-02 05:59:51 +02:00
|
|
|
my $context = $sth->fetchall_arrayref({});
|
2020-02-26 11:20:11 +01:00
|
|
|
|
|
|
|
my $d = {};
|
2020-05-02 05:59:51 +02:00
|
|
|
foreach my $key (keys %{$context->[0]}) {
|
2020-02-26 11:20:11 +01:00
|
|
|
next if $key eq 'index1' or $key eq 'index2';
|
2020-05-02 05:59:51 +02:00
|
|
|
if (defined $context->[0]->{$key}) {
|
|
|
|
$self->{cache}->{$lc_index1}->{$lc_index2}->{$key} = $context->[0]->{$key};
|
|
|
|
$d->{$key} = $context->[0]->{$key};
|
2020-03-20 02:58:24 +01:00
|
|
|
}
|
2020-02-26 11:20:11 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
my $timeout = $self->{pbot}->{registry}->get_value('dualindexsqliteobject', 'cache_timeout') // 60 * 30;
|
|
|
|
$self->{cache_timeouts}->{$lc_index1}->{$lc_index2} = time + $timeout;
|
|
|
|
|
|
|
|
return $d;
|
|
|
|
};
|
|
|
|
|
|
|
|
if ($@) {
|
|
|
|
$self->{pbot}->{logger}->log("Error getting data for ($index1, $index2): $@\n");
|
|
|
|
return undef;
|
|
|
|
}
|
|
|
|
|
|
|
|
return $data;
|
|
|
|
}
|
|
|
|
|
2020-02-27 02:49:59 +01:00
|
|
|
# special case for compatibility with DualIndexHashObject
|
2020-02-26 11:20:11 +01:00
|
|
|
if ($data_index eq '_name') {
|
|
|
|
if (exists $self->{cache}->{$lc_index1}->{$lc_index2}->{_name}) {
|
|
|
|
return $self->{cache}->{$lc_index1}->{$lc_index2}->{_name};
|
|
|
|
} else {
|
|
|
|
return $lc_index2;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (exists $self->{cache}->{$lc_index1}->{$lc_index2}->{$data_index}) {
|
|
|
|
return $self->{cache}->{$lc_index1}->{$lc_index2}->{$data_index};
|
|
|
|
}
|
|
|
|
|
|
|
|
my $value = eval {
|
|
|
|
my $sth = $self->{dbh}->prepare('SELECT * FROM Stuff WHERE index1 = ? AND index2 = ?');
|
|
|
|
$sth->execute($index1, $index2);
|
2020-05-02 05:59:51 +02:00
|
|
|
my $context = $sth->fetchall_arrayref({});
|
2020-02-26 11:20:11 +01:00
|
|
|
|
2020-05-02 05:59:51 +02:00
|
|
|
foreach my $key (keys %{$context->[0]}) {
|
2020-02-26 11:20:11 +01:00
|
|
|
next if $key eq 'index1' or $key eq 'index2';
|
2020-05-02 05:59:51 +02:00
|
|
|
$self->{cache}->{$lc_index1}->{$lc_index2}->{$key} = $context->[0]->{$key};
|
2020-02-26 11:20:11 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
my $timeout = $self->{pbot}->{registry}->get_value('dualindexsqliteobject', 'cache_timeout') // 60 * 30;
|
|
|
|
$self->{cache_timeouts}->{$lc_index1}->{$lc_index2} = time + $timeout;
|
|
|
|
|
2020-05-02 05:59:51 +02:00
|
|
|
return $context->[0]->{$data_index};
|
2020-02-26 11:20:11 +01:00
|
|
|
};
|
|
|
|
|
|
|
|
if ($@) {
|
|
|
|
$self->{pbot}->{logger}->log("Error getting data for ($index1, $index2, $data_index): $@\n");
|
|
|
|
return undef;
|
|
|
|
}
|
|
|
|
|
|
|
|
return $value;
|
|
|
|
}
|
|
|
|
|
|
|
|
sub add {
|
|
|
|
my ($self, $index1, $index2, $data, $quiet) = @_;
|
|
|
|
|
|
|
|
my $name1 = $self->get_data($index1, '_name') // $index1;
|
|
|
|
|
|
|
|
eval {
|
|
|
|
my $sth;
|
|
|
|
$self->{dbh}->begin_work;
|
|
|
|
|
|
|
|
if (not $self->exists($index1, $index2)) {
|
|
|
|
$sth = $self->{dbh}->prepare('INSERT INTO Stuff (index1, index2) VALUES (?, ?)');
|
|
|
|
$sth->execute($name1, $index2);
|
|
|
|
}
|
|
|
|
|
|
|
|
my $sql = 'UPDATE Stuff SET ';
|
|
|
|
|
|
|
|
my $comma = '';
|
|
|
|
foreach my $key (sort keys %$data) {
|
|
|
|
if (not exists $self->{columns}->{$key}) {
|
|
|
|
next;
|
|
|
|
}
|
|
|
|
$sql .= "$comma\"$key\" = ?";
|
|
|
|
$comma = ', ';
|
|
|
|
}
|
|
|
|
|
|
|
|
$sql .= ' WHERE index1 == ? AND index2 == ?';
|
|
|
|
|
|
|
|
$sth = $self->{dbh}->prepare($sql);
|
|
|
|
|
|
|
|
my $param = 1;
|
|
|
|
foreach my $key (sort keys %$data) {
|
|
|
|
next if not exists $self->{columns}->{$key};
|
|
|
|
$sth->bind_param($param++, $data->{$key});
|
|
|
|
}
|
|
|
|
|
|
|
|
$sth->bind_param($param++, $index1);
|
|
|
|
$sth->bind_param($param++, $index2);
|
|
|
|
$sth->execute();
|
|
|
|
|
|
|
|
$self->{dbh}->commit;
|
|
|
|
|
|
|
|
# no errors updating SQL -- now we update cache
|
|
|
|
my ($lc_index1, $lc_index2) = (lc $index1, lc $index2);
|
|
|
|
$self->{cache}->{$lc_index1}->{_name} = $index1 if $index1 ne $lc_index1 and not exists $self->{cache}->{$lc_index1}->{_name};
|
|
|
|
if (grep { $_ ne '_name' } keys %{$self->{cache}->{$lc_index1}->{$lc_index2}}) {
|
|
|
|
foreach my $key (sort keys %$data) {
|
|
|
|
next if not exists $self->{columns}->{$key};
|
|
|
|
$self->{cache}->{$lc_index1}->{$lc_index2}->{$key} = $data->{$key};
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
$self->{cache}->{$lc_index1}->{lc $index2} = {}
|
|
|
|
}
|
|
|
|
$self->{cache}->{$lc_index1}->{$lc_index2}->{_name} = $index2 if not exists $self->{cache}->{$lc_index1}->{$lc_index2}->{_name} and $index2 ne $lc_index2;
|
|
|
|
};
|
|
|
|
|
|
|
|
if ($@) {
|
|
|
|
$self->{pbot}->{logger}->log("Error adding $self->{name} $index2 to $index1: $@\n");
|
|
|
|
$self->{dbh}->rollback;
|
|
|
|
return "Error adding $index2 to $name1: $@\n";
|
|
|
|
}
|
|
|
|
|
|
|
|
$index1 = 'global' if $index1 eq '.*';
|
|
|
|
$index2 = "\"$index2\"" if $index2 =~ / /;
|
|
|
|
$self->{pbot}->{logger}->log("$self->{name}: [$index1]: $index2 added.\n") unless $quiet;
|
|
|
|
return "$index2 added to $name1.";
|
|
|
|
}
|
|
|
|
|
|
|
|
sub remove {
|
|
|
|
my ($self, $index1, $index2, $data_index, $dont_save) = @_;
|
|
|
|
|
|
|
|
if (not $self->exists($index1)) {
|
|
|
|
my $result = "$self->{name}: $index1 not found; similiar matches: ";
|
|
|
|
$result .= $self->levenshtein_matches($index1);
|
|
|
|
return $result;
|
|
|
|
}
|
|
|
|
|
|
|
|
my $name1 = $self->get_data($index1, '_name');
|
|
|
|
$name1 = 'global' if $name1 eq '.*';
|
|
|
|
my $lc_index1 = lc $index1;
|
|
|
|
|
|
|
|
if (not defined $index2) {
|
|
|
|
eval {
|
|
|
|
my $sth = $self->{dbh}->prepare("DELETE FROM Stuff WHERE index1 = ?");
|
|
|
|
$sth->execute($index1);
|
|
|
|
|
|
|
|
$self->cache_remove($index1);
|
|
|
|
};
|
|
|
|
|
|
|
|
if ($@) {
|
|
|
|
$self->{pbot}->{logger}->log("Error removing $index1 from $self->{name}: $@\n");
|
|
|
|
return "Error removing $name1: $@";
|
|
|
|
}
|
|
|
|
|
|
|
|
return "$name1 removed.";
|
|
|
|
}
|
|
|
|
|
|
|
|
if (not $self->exists($index1, $index2)) {
|
|
|
|
my $result = "$self->{name}: [$name1] $index2 not found; similiar matches: ";
|
|
|
|
$result .= $self->levenshtein_matches($index1, $index2);
|
|
|
|
return $result;
|
|
|
|
}
|
|
|
|
|
|
|
|
my $name2 = $self->get_data($index1, $index2, '_name');
|
|
|
|
$name2 = "\"$name2\"" if $name2 =~ / /;
|
|
|
|
my $lc_index2 = lc $index2;
|
|
|
|
|
|
|
|
if (not defined $data_index) {
|
|
|
|
eval {
|
|
|
|
my $sth = $self->{dbh}->prepare("DELETE FROM Stuff WHERE index1 = ? AND index2 = ?");
|
|
|
|
$sth->execute($index1, $index2);
|
|
|
|
|
|
|
|
$self->cache_remove($index1, $index2);
|
|
|
|
};
|
|
|
|
|
|
|
|
if ($@) {
|
|
|
|
$self->{pbot}->{logger}->log("Error removing $self->{name}: [$name1] $name2: $@\n");
|
|
|
|
return "Error removing $name2 from $name1: $@";
|
|
|
|
}
|
|
|
|
|
|
|
|
return "$name2 removed from $name1.";
|
|
|
|
}
|
|
|
|
|
|
|
|
if (not exists $self->{columns}->{$data_index}) {
|
|
|
|
return "$self->{name} have no such metadata $data_index.";
|
|
|
|
}
|
|
|
|
|
|
|
|
if (defined $self->get_data($lc_index1, $lc_index2, $data_index)) {
|
|
|
|
eval {
|
2020-04-30 05:01:49 +02:00
|
|
|
my $sth = $self->{dbh}->prepare("UPDATE Stuff SET '$data_index' = ? WHERE index1 = ? AND index2 = ?");
|
2020-02-26 11:20:11 +01:00
|
|
|
$sth->execute(undef, $index1, $index2);
|
|
|
|
|
|
|
|
$self->{cache}->{$index1}->{$index2}->{$data_index} = undef;
|
|
|
|
};
|
|
|
|
|
|
|
|
if ($@) {
|
|
|
|
$self->{pbot}->{logger}->log("Error unsetting $self->{name}: $name1.$name2: $@\n");
|
|
|
|
return "Error unsetting $data_index from $name2: $@";
|
|
|
|
}
|
|
|
|
|
|
|
|
return "$name2.$data_index unset.";
|
|
|
|
}
|
|
|
|
|
|
|
|
return "$name2.$data_index is not set.";
|
|
|
|
}
|
|
|
|
|
|
|
|
sub set {
|
|
|
|
my ($self, $index1, $index2, $key, $value) = @_;
|
|
|
|
|
|
|
|
if (not $self->exists($index1)) {
|
|
|
|
my $result = "$self->{name}: $index1 not found; similiar matches: ";
|
|
|
|
$result .= $self->levenshtein_matches($index1);
|
|
|
|
return $result;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (not $self->exists($index1, $index2)) {
|
|
|
|
my $secondary_text = $index2 =~ / / ? "\"$index2\"" : $index2;
|
|
|
|
my $result = "$self->{name}: [" . $self->get_data($index1, '_name') . "] $secondary_text not found; similiar matches: ";
|
|
|
|
$result .= $self->levenshtein_matches($index1, $index2);
|
|
|
|
return $result;
|
|
|
|
}
|
|
|
|
|
2020-02-27 02:49:59 +01:00
|
|
|
my $name1 = $self->get_data($index1, '_name');
|
2020-02-26 11:20:11 +01:00
|
|
|
my $name2 = $self->get_data($index1, $index2, '_name');
|
|
|
|
|
|
|
|
$name1 = 'global' if $name1 eq '.*';
|
|
|
|
$name2 = "\"$name2\"" if $name2 =~ / /;
|
|
|
|
|
|
|
|
if (not defined $key) {
|
|
|
|
my $result = "[$name1] $name2 keys:\n";
|
|
|
|
my @metadata = ();
|
|
|
|
foreach my $key (sort $self->get_keys($index1, $index2)) {
|
|
|
|
my $value = $self->get_data($index1, $index2, $key);
|
|
|
|
push @metadata, "$key => $value" if defined $value;
|
|
|
|
}
|
|
|
|
if (not @metadata) { $result .= "none"; }
|
2020-05-03 00:00:57 +02:00
|
|
|
else { $result .= join ";\n", @metadata; }
|
2020-02-26 11:20:11 +01:00
|
|
|
return $result;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (not exists $self->{columns}->{$key}) {
|
|
|
|
return "$self->{name} have no such metadata $key.";
|
|
|
|
}
|
|
|
|
|
|
|
|
if (not defined $value) { $value = $self->get_data($index1, $index2, $key); }
|
|
|
|
else {
|
|
|
|
eval {
|
2020-04-30 05:01:49 +02:00
|
|
|
my $sth = $self->{dbh}->prepare("UPDATE Stuff SET '$key' = ? WHERE index1 = ? AND index2 = ?");
|
2020-02-26 11:20:11 +01:00
|
|
|
$sth->execute($value, $index1, $index2);
|
|
|
|
|
|
|
|
my ($lc_index1, $lc_index2) = (lc $index1, lc $index2);
|
|
|
|
if (exists $self->{cache}->{$lc_index1}
|
|
|
|
and exists $self->{cache}->{$lc_index1}->{$lc_index2}
|
|
|
|
and exists $self->{cache}->{$lc_index1}->{$lc_index2}->{$key}) {
|
|
|
|
$self->{cache}->{$lc_index1}->{$lc_index2}->{$key} = $value;
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
if ($@) {
|
|
|
|
$self->{pbot}->{logger}->log("Error setting $self->{name} $index1 $index2.$key: $@\n");
|
|
|
|
return "Error setting $name2.$key: $@";
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return "[$name1] $name2.$key " . (defined $value ? "set to $value" : "is not set.");
|
|
|
|
}
|
|
|
|
|
|
|
|
sub unset {
|
|
|
|
my ($self, $index1, $index2, $key) = @_;
|
|
|
|
|
|
|
|
if (not $self->exists($index1)) {
|
|
|
|
my $result = "$self->{name}: $index1 not found; similiar matches: ";
|
|
|
|
$result .= $self->levenshtein_matches($index1);
|
|
|
|
return $result;
|
|
|
|
}
|
|
|
|
|
|
|
|
my $name1 = $self->get_data($index1, '_name');
|
|
|
|
$name1 = 'global' if $name1 eq '.*';
|
|
|
|
|
|
|
|
if (not $self->exists($index1, $index2)) {
|
|
|
|
my $result = "$self->{name}: [$name1] $index2 not found; similiar matches: ";
|
|
|
|
$result .= $self->levenshtein_matches($index1, $index2);
|
|
|
|
return $result;
|
|
|
|
}
|
|
|
|
|
|
|
|
my $name2 = $self->get_data($index1, $index2, '_name');
|
|
|
|
$name2 = "\"$name2\"" if $name2 =~ / /;
|
|
|
|
|
|
|
|
if (not exists $self->{columns}->{$key}) {
|
|
|
|
return "$self->{name} have no such metadata $key.";
|
|
|
|
}
|
|
|
|
|
|
|
|
eval {
|
2020-04-30 05:01:49 +02:00
|
|
|
my $sth = $self->{dbh}->prepare("UPDATE Stuff SET '$key' = ? WHERE index1 = ? AND index2 = ?");
|
2020-02-26 11:20:11 +01:00
|
|
|
$sth->execute(undef, $index1, $index2);
|
|
|
|
|
|
|
|
my ($lc_index1, $lc_index2) = (lc $index1, lc $index2);
|
|
|
|
if (exists $self->{cache}->{$lc_index1}
|
|
|
|
and exists $self->{cache}->{$lc_index1}->{$lc_index2}
|
|
|
|
and exists $self->{cache}->{$lc_index1}->{$lc_index2}->{$key}) {
|
|
|
|
$self->{cache}->{$lc_index1}->{$lc_index2}->{$key} = undef;
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
if ($@) {
|
|
|
|
$self->{pbot}->{logger}->log("Error unsetting key: $@\n");
|
|
|
|
return "Error unsetting key: $@";
|
|
|
|
}
|
|
|
|
|
|
|
|
return "[$name1] $name2.$key unset.";
|
|
|
|
}
|
|
|
|
|
|
|
|
# nothing to do here for SQLite
|
2020-02-27 02:49:59 +01:00
|
|
|
# kept for compatibility with DualIndexHashObject
|
2020-02-26 11:20:11 +01:00
|
|
|
sub save { }
|
|
|
|
sub clear { }
|
|
|
|
|
|
|
|
1;
|