mirror of
https://github.com/pragma-/pbot.git
synced 2025-01-11 20:42:38 +01:00
MessageHistory: use all aka ids when recalling messages
This commit is contained in:
parent
d4e47c0eb1
commit
1dd4180f1f
@ -34,7 +34,6 @@ sub initialize {
|
|||||||
$self->{MSG_NICKCHANGE} = 3; # CHANGED NICK
|
$self->{MSG_NICKCHANGE} = 3; # CHANGED NICK
|
||||||
|
|
||||||
$self->{pbot}->{registry}->add_default('text', 'messagehistory', 'max_recall_time', $conf{max_recall_time} // 0);
|
$self->{pbot}->{registry}->add_default('text', 'messagehistory', 'max_recall_time', $conf{max_recall_time} // 0);
|
||||||
$self->{pbot}->{registry}->add_default('text', 'messagehistory', 'max_messages', 32);
|
|
||||||
|
|
||||||
$self->{pbot}->{commands}->register(sub { $self->cmd_recall_message(@_) }, "recall", 0);
|
$self->{pbot}->{commands}->register(sub { $self->cmd_recall_message(@_) }, "recall", 0);
|
||||||
$self->{pbot}->{commands}->register(sub { $self->cmd_rebuild_aliases(@_) }, "rebuildaliases", 1);
|
$self->{pbot}->{commands}->register(sub { $self->cmd_rebuild_aliases(@_) }, "rebuildaliases", 1);
|
||||||
@ -220,11 +219,6 @@ sub cmd_list_also_known_as {
|
|||||||
sub cmd_recall_message {
|
sub cmd_recall_message {
|
||||||
my ($self, $context) = @_;
|
my ($self, $context) = @_;
|
||||||
|
|
||||||
if (not defined $context->{from}) {
|
|
||||||
$self->{pbot}->{logger}->log("Command missing ~from parameter!\n");
|
|
||||||
return "";
|
|
||||||
}
|
|
||||||
|
|
||||||
my $usage = 'Usage: recall [nick [history [channel]]] [-c <channel>] [-t <text>] [-b <context before>] [-a <context after>] [-x <filter to nick>] [-n <count>] [-r raw mode] [+ ...]';
|
my $usage = 'Usage: recall [nick [history [channel]]] [-c <channel>] [-t <text>] [-b <context before>] [-a <context after>] [-x <filter to nick>] [-n <count>] [-r raw mode] [+ ...]';
|
||||||
|
|
||||||
my $arguments = $context->{arguments};
|
my $arguments = $context->{arguments};
|
||||||
@ -338,7 +332,9 @@ sub cmd_recall_message {
|
|||||||
}
|
}
|
||||||
|
|
||||||
# skip recall command if recalling self without arguments
|
# skip recall command if recalling self without arguments
|
||||||
$recall_history = $context->{nick} eq $recall_nick ? 2 : 1 if defined $recall_nick and not defined $recall_history;
|
if (defined $recall_nick and not defined $recall_history) {
|
||||||
|
$recall_history = $context->{nick} eq $recall_nick ? 2 : 1;
|
||||||
|
}
|
||||||
|
|
||||||
# set history to most recent message if not specified
|
# set history to most recent message if not specified
|
||||||
$recall_history = '1' if not defined $recall_history;
|
$recall_history = '1' if not defined $recall_history;
|
||||||
@ -352,26 +348,39 @@ sub cmd_recall_message {
|
|||||||
$recall_channel = $context->{from};
|
$recall_channel = $context->{from};
|
||||||
}
|
}
|
||||||
|
|
||||||
if (not defined $recall_nick and defined $recall_context) { $recall_nick = $recall_context; }
|
# set nick argument to -x argument if no nick was provided but -x was
|
||||||
|
if (not defined $recall_nick and defined $recall_context) {
|
||||||
|
$recall_nick = $recall_context;
|
||||||
|
}
|
||||||
|
|
||||||
|
# message account and stored nickname with proper typographical casing
|
||||||
my ($account, $found_nick);
|
my ($account, $found_nick);
|
||||||
|
|
||||||
|
# get message account and found nick if a nick was provided
|
||||||
if (defined $recall_nick) {
|
if (defined $recall_nick) {
|
||||||
|
# account and hostmask
|
||||||
($account, $found_nick) = $self->{database}->find_message_account_by_nick($recall_nick);
|
($account, $found_nick) = $self->{database}->find_message_account_by_nick($recall_nick);
|
||||||
|
|
||||||
if (not defined $account) { return "I don't know anybody named $recall_nick."; }
|
if (not defined $account) {
|
||||||
|
return "I don't know anybody named $recall_nick.";
|
||||||
|
}
|
||||||
|
|
||||||
|
# keep only nick portion of hostmask
|
||||||
$found_nick =~ s/!.*$//;
|
$found_nick =~ s/!.*$//;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
# matching message found in database, if any
|
||||||
my $message;
|
my $message;
|
||||||
|
|
||||||
if ($random) {
|
if ($random) {
|
||||||
$message = $self->{database}->get_random_message($account, $recall_channel);
|
# get a random message
|
||||||
|
$message = $self->{database}->get_random_message($account, $recall_channel, $recall_nick);
|
||||||
} elsif ($recall_history =~ /^\d+$/ and not defined $recall_text) {
|
} elsif ($recall_history =~ /^\d+$/ and not defined $recall_text) {
|
||||||
# integral history
|
# integral history
|
||||||
|
|
||||||
|
# if a nick was given, ensure requested history is within range of nick's history count
|
||||||
if (defined $account) {
|
if (defined $account) {
|
||||||
my $max_messages = $self->{database}->get_max_messages($account, $recall_channel);
|
my $max_messages = $self->{database}->get_max_messages($account, $recall_channel, $recall_nick);
|
||||||
if ($recall_history < 1 || $recall_history > $max_messages) {
|
if ($recall_history < 1 || $recall_history > $max_messages) {
|
||||||
if ($max_messages == 0) {
|
if ($max_messages == 0) {
|
||||||
return "No messages for $recall_nick in $recall_channel yet.";
|
return "No messages for $recall_nick in $recall_channel yet.";
|
||||||
@ -382,57 +391,81 @@ sub cmd_recall_message {
|
|||||||
}
|
}
|
||||||
|
|
||||||
$recall_history--;
|
$recall_history--;
|
||||||
$message = $self->{database}->recall_message_by_count($account, $recall_channel, $recall_history, '(?:recall|mock|ftfy|fix|clapper)');
|
$message = $self->{database}->recall_message_by_count($account, $recall_channel, $recall_history, '(?:recall|mock|ftfy|fix|clapper)', $recall_nick);
|
||||||
|
|
||||||
if (not defined $message) {
|
|
||||||
return "No message found at index $recall_history in channel $recall_channel.";
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
# regex history
|
|
||||||
$message = $self->{database}->recall_message_by_text($account, $recall_channel, $recall_history, '(?:recall|mock|ftfy|fix|clapper)');
|
|
||||||
|
|
||||||
if (not defined $message) {
|
if (not defined $message) {
|
||||||
if (defined $account) {
|
if (defined $account) {
|
||||||
return "No message for nick $found_nick in channel $recall_channel containing \"$recall_history\"";
|
return "No message found at index $recall_history for $found_nick in $recall_channel.";
|
||||||
} else {
|
} else {
|
||||||
return "No message in channel $recall_channel containing \"$recall_history\".";
|
return "No message found at index $recall_history in $recall_channel.";
|
||||||
|
}
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
# regex history
|
||||||
|
$message = $self->{database}->recall_message_by_text($account, $recall_channel, $recall_history, '(?:recall|mock|ftfy|fix|clapper)', $recall_nick);
|
||||||
|
|
||||||
|
if (not defined $message) {
|
||||||
|
if (defined $account) {
|
||||||
|
return "No message for $found_nick in $recall_channel containing \"$recall_history\"";
|
||||||
|
} else {
|
||||||
|
return "No message in $recall_channel containing \"$recall_history\".";
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
my $context_account;
|
my ($context_account, $context_nick);
|
||||||
|
|
||||||
if (defined $recall_context) {
|
if (defined $recall_context) {
|
||||||
($context_account) = $self->{database}->find_message_account_by_nick($recall_context);
|
($context_account, $context_nick) = $self->{database}->find_message_account_by_nick($recall_context);
|
||||||
|
|
||||||
if (not defined $context_account) { return "I don't know anybody named $recall_context."; }
|
if (not defined $context_account) {
|
||||||
|
return "I don't know anybody named $recall_context.";
|
||||||
|
}
|
||||||
|
|
||||||
|
# keep only nick portion of hostmask
|
||||||
|
$context_nick =~ s/!.*$//;
|
||||||
}
|
}
|
||||||
|
|
||||||
my $messages = $self->{database}->get_message_context($message, $recall_before, $recall_after, $recall_count, $recall_history, $context_account);
|
my $messages = $self->{database}->get_message_context($message, $recall_before, $recall_after, $recall_count, $recall_history, $context_account, $context_nick);
|
||||||
|
|
||||||
my $max_recall_time = $self->{pbot}->{registry}->get_value('messagehistory', 'max_recall_time');
|
my $max_recall_time = $self->{pbot}->{registry}->get_value('messagehistory', 'max_recall_time');
|
||||||
|
|
||||||
foreach my $msg (@$messages) {
|
foreach my $msg (@$messages) {
|
||||||
if ($max_recall_time && time - $msg->{timestamp} > $max_recall_time && not $self->{pbot}->{users}->loggedin_admin($context->{from}, $context->{hostmask})) {
|
# optionally limit messages by by a maximum recall duration from the current time, for privacy
|
||||||
$max_recall_time = duration($max_recall_time);
|
if ($max_recall_time && time - $msg->{timestamp} > $max_recall_time
|
||||||
|
&& not $self->{pbot}->{users}->loggedin_admin($context->{from}, $context->{hostmask}))
|
||||||
|
{
|
||||||
|
$max_recall_time = duration $max_recall_time;
|
||||||
$result .= "Sorry, you can not recall messages older than $max_recall_time.";
|
$result .= "Sorry, you can not recall messages older than $max_recall_time.";
|
||||||
return $result;
|
return $result;
|
||||||
}
|
}
|
||||||
|
|
||||||
my $text = $msg->{msg};
|
my $text = $msg->{msg};
|
||||||
my $ago = concise ago(time - $msg->{timestamp});
|
my $ago = concise ago (time - $msg->{timestamp});
|
||||||
|
my $nick;
|
||||||
|
|
||||||
|
if (not $raw) {
|
||||||
|
if ($msg->{hostmask}) {
|
||||||
|
($nick) = $msg->{hostmask} =~ /^([^!]+)!/;
|
||||||
|
} else {
|
||||||
|
$nick = $self->{database}->find_most_recent_hostmask($msg->{id});
|
||||||
|
($nick) = $nick =~ m/^([^!]+)/;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
if ( $text =~ s/^(NICKCHANGE)\b/changed nick to/
|
if ( $text =~ s/^(NICKCHANGE)\b/changed nick to/
|
||||||
or $text =~ s/^(KICKED|QUIT)\b/lc "$1"/e
|
or $text =~ s/^(KICKED|QUIT)\b/lc "$1"/e
|
||||||
or $text =~ s/^MODE ([^ ]+) (.*)/set mode $1 on $2/
|
or $text =~ s/^MODE ([^ ]+) (.*)/set mode $1 on $2/
|
||||||
or $text =~ s/^(JOIN|PART)\b/lc "$1ed"/e)
|
or $text =~ s/^(JOIN|PART)\b/lc "$1ed"/e)
|
||||||
{
|
{
|
||||||
$text =~ s/^(quit) (.*)/$1 ($2)/; # fix ugly "[nick] quit Quit: Leaving."
|
$text =~ s/^(quit) (.*)/$1 ($2)/; # fix ugly "[nick] quit Quit: Leaving."
|
||||||
$result .= $raw ? "$text\n" : "[$ago] $msg->{nick} $text\n";
|
$result .= $raw ? "$text\n" : "[$ago] $nick $text\n";
|
||||||
} elsif ($text =~ s/^\/me\s+//) {
|
}
|
||||||
$result .= $raw ? "$text\n" : "[$ago] * $msg->{nick} $text\n";
|
elsif ($text =~ s/^\/me\s+//) {
|
||||||
} else {
|
$result .= $raw ? "$text\n" : "[$ago] * $nick $text\n";
|
||||||
$result .= $raw ? "$text\n" : "[$ago] <$msg->{nick}> $text\n";
|
}
|
||||||
|
else {
|
||||||
|
$result .= $raw ? "$text\n" : "[$ago] <$nick> $text\n";
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -139,7 +139,8 @@ CREATE TABLE IF NOT EXISTS Messages (
|
|||||||
channel TEXT COLLATE NOCASE,
|
channel TEXT COLLATE NOCASE,
|
||||||
msg TEXT COLLATE NOCASE,
|
msg TEXT COLLATE NOCASE,
|
||||||
timestamp NUMERIC,
|
timestamp NUMERIC,
|
||||||
mode INTEGER
|
mode INTEGER,
|
||||||
|
hostmask TEXT COLLATE NOCASE
|
||||||
)
|
)
|
||||||
SQL
|
SQL
|
||||||
|
|
||||||
@ -863,23 +864,24 @@ sub get_hostmasks_for_nickserv {
|
|||||||
}
|
}
|
||||||
|
|
||||||
sub add_message {
|
sub add_message {
|
||||||
my ($self, $id, $mask, $channel, $message) = @_;
|
my ($self, $id, $hostmask, $channel, $message) = @_;
|
||||||
|
|
||||||
#$self->{pbot}->{logger}->log("Adding message [$id][$mask][$channel][$message->{msg}][$message->{timestamp}][$message->{mode}]\n");
|
|
||||||
|
|
||||||
eval {
|
eval {
|
||||||
my $sth = $self->{dbh}->prepare('INSERT INTO Messages VALUES (?, ?, ?, ?, ?)');
|
my $sth = $self->{dbh}->prepare('INSERT INTO Messages VALUES (?, ?, ?, ?, ?, ?)');
|
||||||
$sth->execute($id, $channel, $message->{msg}, $message->{timestamp}, $message->{mode});
|
$sth->execute($id, $channel, $message->{msg}, $message->{timestamp}, $message->{mode}, $hostmask);
|
||||||
$self->{new_entries}++;
|
$self->{new_entries}++;
|
||||||
};
|
};
|
||||||
|
|
||||||
$self->{pbot}->{logger}->log($@) if $@;
|
$self->{pbot}->{logger}->log($@) if $@;
|
||||||
$self->update_channel_data($id, $channel, {last_seen => $message->{timestamp}});
|
|
||||||
$self->update_hostmask_data($mask, {last_seen => $message->{timestamp}});
|
$self->update_channel_data($id, $channel, { last_seen => $message->{timestamp }});
|
||||||
|
$self->update_hostmask_data($hostmask, { last_seen => $message->{timestamp }});
|
||||||
}
|
}
|
||||||
|
|
||||||
sub get_recent_messages {
|
sub get_recent_messages {
|
||||||
my ($self, $id, $channel, $limit, $mode, $nick) = @_;
|
my ($self, $id, $channel, $limit, $mode, $nick) = @_;
|
||||||
$limit = 25 if not defined $limit;
|
|
||||||
|
$limit //= 25;
|
||||||
|
|
||||||
$channel = lc $channel;
|
$channel = lc $channel;
|
||||||
|
|
||||||
@ -887,65 +889,55 @@ sub get_recent_messages {
|
|||||||
$mode_query = "AND mode = $mode" if defined $mode;
|
$mode_query = "AND mode = $mode" if defined $mode;
|
||||||
|
|
||||||
my $messages = eval {
|
my $messages = eval {
|
||||||
my $sql = "SELECT msg, mode, timestamp FROM Messages WHERE ";
|
my $sql = "SELECT * FROM Messages WHERE ";
|
||||||
|
|
||||||
|
my %seen_id;
|
||||||
|
|
||||||
my %akas;
|
my %akas;
|
||||||
if (defined $mode and $mode == $self->{pbot}->{messagehistory}->{MSG_NICKCHANGE}) { %akas = $self->get_also_known_as($nick); }
|
if (defined $mode and $mode == $self->{pbot}->{messagehistory}->{MSG_NICKCHANGE}) {
|
||||||
else { $akas{'this'} = {id => $id, type => $self->{alias_type}->{STRONG}, nickchange => 0}; }
|
%akas = $self->get_also_known_as($nick);
|
||||||
|
} else {
|
||||||
|
$akas{$id} = {
|
||||||
|
id => $id,
|
||||||
|
type => $self->{alias_type}->{STRONG},
|
||||||
|
nickchange => 0,
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
my $ids;
|
|
||||||
my %seen_id;
|
|
||||||
my $or = '';
|
|
||||||
foreach my $aka (keys %akas) {
|
foreach my $aka (keys %akas) {
|
||||||
next if $akas{$aka}->{type} == $self->{alias_type}->{WEAK};
|
next if $akas{$aka}->{type} == $self->{alias_type}->{WEAK};
|
||||||
next if $akas{$aka}->{nickchange} == 1;
|
next if $akas{$aka}->{nickchange} == 1;
|
||||||
next if exists $seen_id{$akas{$aka}->{id}};
|
next if exists $seen_id{$akas{$aka}->{id}};
|
||||||
$seen_id{$akas{$aka}->{id}} = 1;
|
|
||||||
|
|
||||||
$ids .= "${or}id = ?";
|
$seen_id{$akas{$aka}->{id}} = 1;
|
||||||
$or = ' OR ';
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
my $ids = join " OR ", map { "id = ?" } keys %seen_id;
|
||||||
|
|
||||||
$sql .= "($ids) AND channel = ? $mode_query ORDER BY timestamp ASC LIMIT ? OFFSET (SELECT COUNT(*) FROM Messages WHERE ($ids) AND channel = ? $mode_query) - ?";
|
$sql .= "($ids) AND channel = ? $mode_query ORDER BY timestamp ASC LIMIT ? OFFSET (SELECT COUNT(*) FROM Messages WHERE ($ids) AND channel = ? $mode_query) - ?";
|
||||||
|
|
||||||
my $sth = $self->{dbh}->prepare($sql);
|
my $sth = $self->{dbh}->prepare($sql);
|
||||||
|
|
||||||
my $param = 1;
|
my $param = 1;
|
||||||
%seen_id = ();
|
map { $sth->bind_param($param++, $_) } keys %seen_id;
|
||||||
foreach my $aka (keys %akas) {
|
|
||||||
next if $akas{$aka}->{type} == $self->{alias_type}->{WEAK};
|
|
||||||
next if $akas{$aka}->{nickchange} == 1;
|
|
||||||
next if exists $seen_id{$akas{$aka}->{id}};
|
|
||||||
$seen_id{$akas{$aka}->{id}} = 1;
|
|
||||||
|
|
||||||
$sth->bind_param($param++, $akas{$aka}->{id});
|
|
||||||
}
|
|
||||||
|
|
||||||
$sth->bind_param($param++, $channel);
|
$sth->bind_param($param++, $channel);
|
||||||
$sth->bind_param($param++, $limit);
|
$sth->bind_param($param++, $limit);
|
||||||
|
map { $sth->bind_param($param++, $_) } keys %seen_id;
|
||||||
%seen_id = ();
|
|
||||||
foreach my $aka (keys %akas) {
|
|
||||||
next if $akas{$aka}->{type} == $self->{alias_type}->{WEAK};
|
|
||||||
next if $akas{$aka}->{nickchange} == 1;
|
|
||||||
next if exists $seen_id{$akas{$aka}->{id}};
|
|
||||||
$seen_id{$akas{$aka}->{id}} = 1;
|
|
||||||
|
|
||||||
$sth->bind_param($param++, $akas{$aka}->{id});
|
|
||||||
}
|
|
||||||
|
|
||||||
$sth->bind_param($param++, $channel);
|
$sth->bind_param($param++, $channel);
|
||||||
$sth->bind_param($param, $limit);
|
$sth->bind_param($param, $limit);
|
||||||
$sth->execute();
|
$sth->execute;
|
||||||
return $sth->fetchall_arrayref({});
|
return $sth->fetchall_arrayref({});
|
||||||
};
|
};
|
||||||
|
|
||||||
$self->{pbot}->{logger}->log($@) if $@;
|
$self->{pbot}->{logger}->log($@) if $@;
|
||||||
return $messages;
|
return $messages;
|
||||||
}
|
}
|
||||||
|
|
||||||
sub get_recent_messages_from_channel {
|
sub get_recent_messages_from_channel {
|
||||||
my ($self, $channel, $limit, $mode, $direction) = @_;
|
my ($self, $channel, $limit, $mode, $direction) = @_;
|
||||||
$limit = 25 if not defined $limit;
|
|
||||||
$direction = 'ASC' if not defined $direction;
|
$limit //= 25;
|
||||||
|
$direction //= 'ASC';
|
||||||
|
|
||||||
$channel = lc $channel;
|
$channel = lc $channel;
|
||||||
|
|
||||||
@ -953,7 +945,7 @@ sub get_recent_messages_from_channel {
|
|||||||
$mode_query = "AND mode = $mode" if defined $mode;
|
$mode_query = "AND mode = $mode" if defined $mode;
|
||||||
|
|
||||||
my $messages = eval {
|
my $messages = eval {
|
||||||
my $sql = "SELECT id, msg, mode, timestamp FROM Messages WHERE channel = ? $mode_query ORDER BY timestamp $direction LIMIT ?";
|
my $sql = "SELECT * FROM Messages WHERE channel = ? $mode_query ORDER BY timestamp $direction LIMIT ?";
|
||||||
my $sth = $self->{dbh}->prepare($sql);
|
my $sth = $self->{dbh}->prepare($sql);
|
||||||
$sth->execute($channel, $limit);
|
$sth->execute($channel, $limit);
|
||||||
return $sth->fetchall_arrayref({});
|
return $sth->fetchall_arrayref({});
|
||||||
@ -963,7 +955,39 @@ sub get_recent_messages_from_channel {
|
|||||||
}
|
}
|
||||||
|
|
||||||
sub get_message_context {
|
sub get_message_context {
|
||||||
my ($self, $message, $before, $after, $count, $text, $context_id) = @_;
|
my ($self, $message, $before, $after, $count, $text, $context_id, $context_nick) = @_;
|
||||||
|
|
||||||
|
my %seen_id;
|
||||||
|
my $ids = '';
|
||||||
|
|
||||||
|
my $sql = 'SELECT * FROM Messages WHERE channel = ? ';
|
||||||
|
|
||||||
|
if (defined $context_id) {
|
||||||
|
my %akas;
|
||||||
|
|
||||||
|
if (defined $context_nick) {
|
||||||
|
%akas = $self->get_also_known_as($context_nick);
|
||||||
|
} else {
|
||||||
|
$akas{$context_id} = {
|
||||||
|
id => $context_id,
|
||||||
|
type => $self->{alias_type}->{STRONG},
|
||||||
|
nickchange => 0,
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
foreach my $aka (keys %akas) {
|
||||||
|
next if $akas{$aka}->{type} == $self->{alias_type}->{WEAK};
|
||||||
|
next if $akas{$aka}->{nickchange} == 1;
|
||||||
|
next if exists $seen_id{$akas{$aka}->{id}};
|
||||||
|
|
||||||
|
$seen_id{$akas{$aka}->{id}} = 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
$ids = join " OR ", map { "id = ?" } keys %seen_id;
|
||||||
|
$ids = "AND ($ids) ";
|
||||||
|
}
|
||||||
|
|
||||||
|
$sql .= $ids;
|
||||||
|
|
||||||
my ($messages_before, $messages_after, $messages_count);
|
my ($messages_before, $messages_after, $messages_count);
|
||||||
|
|
||||||
@ -973,70 +997,50 @@ sub get_message_context {
|
|||||||
$search =~ s/\?/_/g;
|
$search =~ s/\?/_/g;
|
||||||
|
|
||||||
$messages_count = eval {
|
$messages_count = eval {
|
||||||
my $sth;
|
$sql .= 'AND msg LIKE ? ESCAPE "\" AND timestamp < ? AND mode = 0 ORDER BY timestamp DESC LIMIT ?';
|
||||||
if (defined $context_id) {
|
my $sth = $self->{dbh}->prepare($sql);
|
||||||
$sth = $self->{dbh}->prepare(
|
my $param = 1;
|
||||||
'SELECT id, msg, mode, timestamp, channel FROM Messages WHERE id = ? AND channel = ? AND msg LIKE ? ESCAPE "\" AND timestamp < ? AND mode = 0 ORDER BY timestamp DESC LIMIT ?');
|
$sth->bind_param($param++, $message->{channel});
|
||||||
$sth->bind_param(1, $context_id);
|
map { $sth->bind_param($param++, $_) } keys %seen_id;
|
||||||
$sth->bind_param(2, $message->{channel});
|
$sth->bind_param($param++, $search);
|
||||||
$sth->bind_param(3, $search);
|
$sth->bind_param($param++, $message->{timestamp});
|
||||||
$sth->bind_param(4, $message->{timestamp});
|
$sth->bind_param($param++, $count - 1);
|
||||||
$sth->bind_param(5, $count - 1);
|
$sth->execute;
|
||||||
} else {
|
|
||||||
$sth = $self->{dbh}
|
|
||||||
->prepare('SELECT id, msg, mode, timestamp, channel FROM Messages WHERE channel = ? AND msg LIKE ? ESCAPE "\" AND timestamp < ? AND mode = 0 ORDER BY timestamp DESC LIMIT ?');
|
|
||||||
$sth->bind_param(1, $message->{channel});
|
|
||||||
$sth->bind_param(2, $search);
|
|
||||||
$sth->bind_param(3, $message->{timestamp});
|
|
||||||
$sth->bind_param(4, $count - 1);
|
|
||||||
}
|
|
||||||
$sth->execute();
|
|
||||||
return [reverse @{$sth->fetchall_arrayref({})}];
|
return [reverse @{$sth->fetchall_arrayref({})}];
|
||||||
};
|
};
|
||||||
|
|
||||||
$self->{pbot}->{logger}->log($@) if $@;
|
$self->{pbot}->{logger}->log($@) if $@;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (defined $before and $before > 0) {
|
if (defined $before and $before > 0) {
|
||||||
$messages_before = eval {
|
$messages_before = eval {
|
||||||
my $sth;
|
$sql .= ' AND timestamp < ? AND mode = 0 ORDER BY timestamp DESC LIMIT ?';
|
||||||
if (defined $context_id) {
|
my $sth = $self->{dbh}->prepare($sql);
|
||||||
$sth = $self->{dbh}
|
my $param = 1;
|
||||||
->prepare('SELECT id, msg, mode, timestamp, channel FROM Messages WHERE id = ? AND channel = ? AND timestamp < ? AND mode = 0 ORDER BY timestamp DESC LIMIT ?');
|
$sth->bind_param($param++, $message->{channel});
|
||||||
$sth->bind_param(1, $context_id);
|
map { $sth->bind_param($param++, $_) } keys %seen_id;
|
||||||
$sth->bind_param(2, $message->{channel});
|
$sth->bind_param($param++, $message->{timestamp});
|
||||||
$sth->bind_param(3, $message->{timestamp});
|
$sth->bind_param($param++, $before);
|
||||||
$sth->bind_param(4, $before);
|
$sth->execute;
|
||||||
} else {
|
|
||||||
$sth = $self->{dbh}->prepare('SELECT id, msg, mode, timestamp, channel FROM Messages WHERE channel = ? AND timestamp < ? AND mode = 0 ORDER BY timestamp DESC LIMIT ?');
|
|
||||||
$sth->bind_param(1, $message->{channel});
|
|
||||||
$sth->bind_param(2, $message->{timestamp});
|
|
||||||
$sth->bind_param(3, $before);
|
|
||||||
}
|
|
||||||
$sth->execute();
|
|
||||||
return [reverse @{$sth->fetchall_arrayref({})}];
|
return [reverse @{$sth->fetchall_arrayref({})}];
|
||||||
};
|
};
|
||||||
|
|
||||||
$self->{pbot}->{logger}->log($@) if $@;
|
$self->{pbot}->{logger}->log($@) if $@;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (defined $after and $after > 0) {
|
if (defined $after and $after > 0) {
|
||||||
$messages_after = eval {
|
$messages_after = eval {
|
||||||
my $sth;
|
$sql .= ' AND timestamp > ? AND mode = 0 ORDER BY timestamp ASC LIMIT ?';
|
||||||
if (defined $context_id) {
|
my $sth = $self->{dbh}->prepare($sql);
|
||||||
$sth = $self->{dbh}
|
my $param = 1;
|
||||||
->prepare('SELECT id, msg, mode, timestamp, channel FROM Messages WHERE id = ? AND channel = ? AND timestamp > ? AND mode = 0 ORDER BY timestamp ASC LIMIT ?');
|
$sth->bind_param($param++, $message->{channel});
|
||||||
$sth->bind_param(1, $context_id);
|
map { $sth->bind_param($param++, $_) } keys %seen_id;
|
||||||
$sth->bind_param(2, $message->{channel});
|
$sth->bind_param($param++, $message->{timestamp});
|
||||||
$sth->bind_param(3, $message->{timestamp});
|
$sth->bind_param($param++, $after);
|
||||||
$sth->bind_param(4, $after);
|
$sth->execute;
|
||||||
} else {
|
|
||||||
$sth = $self->{dbh}->prepare('SELECT id, msg, mode, timestamp, channel FROM Messages WHERE channel = ? AND timestamp > ? AND mode = 0 ORDER BY timestamp ASC LIMIT ?');
|
|
||||||
$sth->bind_param(1, $message->{channel});
|
|
||||||
$sth->bind_param(2, $message->{timestamp});
|
|
||||||
$sth->bind_param(3, $after);
|
|
||||||
}
|
|
||||||
$sth->execute();
|
|
||||||
return $sth->fetchall_arrayref({});
|
return $sth->fetchall_arrayref({});
|
||||||
};
|
};
|
||||||
|
|
||||||
$self->{pbot}->{logger}->log($@) if $@;
|
$self->{pbot}->{logger}->log($@) if $@;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -1046,75 +1050,54 @@ sub get_message_context {
|
|||||||
push(@messages, $message);
|
push(@messages, $message);
|
||||||
push(@messages, @$messages_after) if defined $messages_after;
|
push(@messages, @$messages_after) if defined $messages_after;
|
||||||
|
|
||||||
my %nicks;
|
|
||||||
foreach my $msg (@messages) {
|
|
||||||
if (not exists $nicks{$msg->{id}}) {
|
|
||||||
my $hostmask = $self->find_most_recent_hostmask($msg->{id});
|
|
||||||
my ($nick) = $hostmask =~ m/^([^!]+)/;
|
|
||||||
$nicks{$msg->{id}} = $nick;
|
|
||||||
}
|
|
||||||
$msg->{nick} = $nicks{$msg->{id}};
|
|
||||||
}
|
|
||||||
|
|
||||||
return \@messages;
|
return \@messages;
|
||||||
}
|
}
|
||||||
|
|
||||||
sub recall_message_by_count {
|
sub recall_message_by_count {
|
||||||
my ($self, $id, $channel, $count, $ignore_command, $use_aliases) = @_;
|
my ($self, $id, $channel, $count, $ignore_command, $use_aliases) = @_;
|
||||||
|
|
||||||
my $messages;
|
my $messages = eval {
|
||||||
|
my $sql = 'SELECT * FROM Messages WHERE ';
|
||||||
|
|
||||||
|
my %seen_id;
|
||||||
|
|
||||||
|
if (defined $id) {
|
||||||
|
my %akas;
|
||||||
|
|
||||||
if (defined $id) {
|
|
||||||
$messages = eval {
|
|
||||||
if (defined $use_aliases) {
|
if (defined $use_aliases) {
|
||||||
my %akas = $self->get_also_known_as($use_aliases);
|
%akas = $self->get_also_known_as($use_aliases);
|
||||||
my %seen_id;
|
|
||||||
my $ids;
|
|
||||||
my $or = '';
|
|
||||||
foreach my $aka (keys %akas) {
|
|
||||||
next if $akas{$aka}->{type} == $self->{alias_type}->{WEAK};
|
|
||||||
next if $akas{$aka}->{nickchange} == 1;
|
|
||||||
next if exists $seen_id{$akas{$aka}->{id}};
|
|
||||||
$seen_id{$akas{$aka}->{id}} = 1;
|
|
||||||
|
|
||||||
$ids .= "${or}id = ?";
|
|
||||||
$or = ' OR ';
|
|
||||||
}
|
|
||||||
|
|
||||||
my $sql = "SELECT id, msg, mode, timestamp, channel FROM Messages WHERE ($ids) AND channel = ? ORDER BY timestamp DESC LIMIT 10 OFFSET ?";
|
|
||||||
my $sth = $self->{dbh}->prepare($sql);
|
|
||||||
|
|
||||||
my $param = 1;
|
|
||||||
%seen_id = ();
|
|
||||||
foreach my $aka (keys %akas) {
|
|
||||||
next if $akas{$aka}->{type} == $self->{alias_type}->{WEAK};
|
|
||||||
next if $akas{$aka}->{nickchange} == 1;
|
|
||||||
next if exists $seen_id{$akas{$aka}->{id}};
|
|
||||||
$seen_id{$akas{$aka}->{id}} = 1;
|
|
||||||
|
|
||||||
$sth->bind_param($param++, $akas{$aka}->{id});
|
|
||||||
}
|
|
||||||
|
|
||||||
$sth->bind_param($param++, $channel);
|
|
||||||
$sth->bind_param($param++, $count);
|
|
||||||
$sth->execute();
|
|
||||||
return $sth->fetchall_arrayref({});
|
|
||||||
} else {
|
} else {
|
||||||
my $sth = $self->{dbh}->prepare('SELECT id, msg, mode, timestamp, channel FROM Messages WHERE id = ? AND channel = ? ORDER BY timestamp DESC LIMIT 10 OFFSET ?');
|
$akas{$id} = {
|
||||||
$sth->bind_param(1, $id);
|
id => $id,
|
||||||
$sth->bind_param(2, $channel);
|
type => $self->{alias_type}->{STRONG},
|
||||||
$sth->bind_param(3, $count);
|
nickchange => 0,
|
||||||
$sth->execute();
|
};
|
||||||
return $sth->fetchall_arrayref({});
|
|
||||||
}
|
}
|
||||||
};
|
|
||||||
} else {
|
foreach my $aka (keys %akas) {
|
||||||
$messages = eval {
|
next if $akas{$aka}->{type} == $self->{alias_type}->{WEAK};
|
||||||
my $sth = $self->{dbh}->prepare('SELECT id, msg, mode, timestamp, channel FROM Messages WHERE channel = ? ORDER BY timestamp DESC LIMIT 10 OFFSET ?');
|
next if $akas{$aka}->{nickchange} == 1;
|
||||||
$sth->execute($channel, $count);
|
next if exists $seen_id{$akas{$aka}->{id}};
|
||||||
return $sth->fetchall_arrayref({});
|
|
||||||
};
|
$seen_id{$akas{$aka}->{id}} = 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
my $ids = join " OR ", map { "id = ?" } keys %seen_id;
|
||||||
|
|
||||||
|
$sql .= "($ids) AND ";
|
||||||
|
}
|
||||||
|
|
||||||
|
$sql .= 'channel = ? ORDER BY timestamp DESC LIMIT 10 OFFSET ?';
|
||||||
|
|
||||||
|
my $sth = $self->{dbh}->prepare($sql);
|
||||||
|
|
||||||
|
my $param = 1;
|
||||||
|
map { $sth->bind_param($param++, $_) } keys %seen_id;
|
||||||
|
$sth->bind_param($param++, $channel);
|
||||||
|
$sth->bind_param($param++, $count);
|
||||||
|
$sth->execute;
|
||||||
|
return $sth->fetchall_arrayref({});
|
||||||
|
};
|
||||||
|
|
||||||
$self->{pbot}->{logger}->log($@) if $@;
|
$self->{pbot}->{logger}->log($@) if $@;
|
||||||
|
|
||||||
@ -1125,33 +1108,64 @@ sub recall_message_by_count {
|
|||||||
next if $message->{msg} =~ m/^$botnick.? $ignore_command/ or $message->{msg} =~ m/^$bot_trigger$ignore_command/;
|
next if $message->{msg} =~ m/^$botnick.? $ignore_command/ or $message->{msg} =~ m/^$bot_trigger$ignore_command/;
|
||||||
return $message;
|
return $message;
|
||||||
}
|
}
|
||||||
|
|
||||||
return undef;
|
return undef;
|
||||||
}
|
}
|
||||||
|
|
||||||
return $messages->[0];
|
return $messages->[0];
|
||||||
}
|
}
|
||||||
|
|
||||||
sub recall_message_by_text {
|
sub recall_message_by_text {
|
||||||
my ($self, $id, $channel, $text, $ignore_command) = @_;
|
my ($self, $id, $channel, $text, $ignore_command, $use_aliases) = @_;
|
||||||
|
|
||||||
my $search = "%$text%";
|
my $search = "%$text%";
|
||||||
$search =~ s/\*/%/g;
|
$search =~ s/\*/%/g;
|
||||||
$search =~ s/\?/_/g;
|
$search =~ s/\?/_/g;
|
||||||
|
|
||||||
my $messages;
|
my $messages = eval {
|
||||||
|
my $sql = 'SELECT * FROM Messages WHERE channel = ? AND msg LIKE ? ESCAPE "\" ';
|
||||||
|
|
||||||
if (defined $id) {
|
my %seen_id;
|
||||||
$messages = eval {
|
|
||||||
my $sth = $self->{dbh}->prepare('SELECT id, msg, mode, timestamp, channel FROM Messages WHERE id = ? AND channel = ? AND msg LIKE ? ESCAPE "\" ORDER BY timestamp DESC LIMIT 10');
|
if (defined $id) {
|
||||||
$sth->execute($id, $channel, $search);
|
my %akas;
|
||||||
return $sth->fetchall_arrayref({});
|
|
||||||
};
|
if (defined $use_aliases) {
|
||||||
} else {
|
%akas = $self->get_also_known_as($use_aliases);
|
||||||
$messages = eval {
|
} else {
|
||||||
my $sth = $self->{dbh}->prepare('SELECT id, msg, mode, timestamp, channel FROM Messages WHERE channel = ? AND msg LIKE ? ESCAPE "\" ORDER BY timestamp DESC LIMIT 10');
|
$akas{$id} = {
|
||||||
$sth->execute($channel, $search);
|
id => $id,
|
||||||
return $sth->fetchall_arrayref({});
|
type => $self->{alias_type}->{STRONG},
|
||||||
};
|
nickchange => 0,
|
||||||
}
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
foreach my $aka (keys %akas) {
|
||||||
|
next if $akas{$aka}->{type} == $self->{alias_type}->{WEAK};
|
||||||
|
next if $akas{$aka}->{nickchange} == 1;
|
||||||
|
next if exists $seen_id{$akas{$aka}->{id}};
|
||||||
|
|
||||||
|
$seen_id{$akas{$aka}->{id}} = 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
my $ids = join " OR ", map { "id = ?" } keys %seen_id;
|
||||||
|
|
||||||
|
$sql .= "AND ($ids) ";
|
||||||
|
}
|
||||||
|
|
||||||
|
$sql .= 'ORDER BY timestamp DESC LIMIT 10';
|
||||||
|
|
||||||
|
my $sth = $self->{dbh}->prepare($sql);
|
||||||
|
|
||||||
|
my $param = 1;
|
||||||
|
$sth->bind_param($param++, $channel);
|
||||||
|
$sth->bind_param($param++, $search);
|
||||||
|
|
||||||
|
map { $sth->bind_param($param++, $_) } keys %seen_id;
|
||||||
|
|
||||||
|
$sth->execute;
|
||||||
|
return $sth->fetchall_arrayref({});
|
||||||
|
};
|
||||||
|
|
||||||
$self->{pbot}->{logger}->log($@) if $@;
|
$self->{pbot}->{logger}->log($@) if $@;
|
||||||
|
|
||||||
@ -1165,29 +1179,61 @@ sub recall_message_by_text {
|
|||||||
or $message->{msg} =~ m/^\s*$ignore_command.? $botnick$/i;
|
or $message->{msg} =~ m/^\s*$ignore_command.? $botnick$/i;
|
||||||
return $message;
|
return $message;
|
||||||
}
|
}
|
||||||
|
|
||||||
return undef;
|
return undef;
|
||||||
}
|
}
|
||||||
|
|
||||||
return $messages->[0];
|
return $messages->[0];
|
||||||
}
|
}
|
||||||
|
|
||||||
sub get_random_message {
|
sub get_random_message {
|
||||||
my ($self, $id, $channel) = @_;
|
my ($self, $id, $channel, $use_aliases) = @_;
|
||||||
|
|
||||||
my $message;
|
my $message = eval {
|
||||||
|
my $sql = 'SELECT * FROM Messages WHERE channel = ? AND mode = ? ';
|
||||||
|
|
||||||
if (defined $id) {
|
my %seen_id;
|
||||||
$message = eval {
|
|
||||||
my $sth = $self->{dbh}->prepare('SELECT id, msg, mode, timestamp, channel FROM Messages WHERE id = ? AND channel = ? AND mode = ? ORDER BY RANDOM() LIMIT 1');
|
if (defined $id) {
|
||||||
$sth->execute($id, $channel, $self->{pbot}->{messagehistory}->{MSG_CHAT});
|
my %akas;
|
||||||
return $sth->fetchrow_hashref;
|
|
||||||
};
|
if (defined $use_aliases) {
|
||||||
} else {
|
%akas = $self->get_also_known_as($use_aliases);
|
||||||
$message = eval {
|
} else {
|
||||||
my $sth = $self->{dbh}->prepare('SELECT id, msg, mode, timestamp, channel FROM Messages WHERE channel = ? AND mode = ? ORDER BY RANDOM() LIMIT 1');
|
$akas{$id} = {
|
||||||
$sth->execute($channel, $self->{pbot}->{messagehistory}->{MSG_CHAT});
|
id => $id,
|
||||||
return $sth->fetchrow_hashref;
|
type => $self->{alias_type}->{STRONG},
|
||||||
};
|
nickchange => 0,
|
||||||
}
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
foreach my $aka (keys %akas) {
|
||||||
|
next if $akas{$aka}->{type} == $self->{alias_type}->{WEAK};
|
||||||
|
next if $akas{$aka}->{nickchange} == 1;
|
||||||
|
next if exists $seen_id{$akas{$aka}->{id}};
|
||||||
|
|
||||||
|
$seen_id{$akas{$aka}->{id}} = 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
my $ids = join " OR ", map { "id = ?" } keys %seen_id;
|
||||||
|
|
||||||
|
$sql .= "AND ($ids) ";
|
||||||
|
}
|
||||||
|
|
||||||
|
$sql .= 'ORDER BY RANDOM() LIMIT 1';
|
||||||
|
|
||||||
|
my $sth = $self->{dbh}->prepare($sql);
|
||||||
|
|
||||||
|
my $param = 1;
|
||||||
|
$sth->bind_param($param++, $channel);
|
||||||
|
$sth->bind_param($param++, $self->{pbot}->{messagehistory}->{MSG_CHAT});
|
||||||
|
|
||||||
|
map { $sth->bind_param($param++, $_) } keys %seen_id;
|
||||||
|
|
||||||
|
$sth->execute;
|
||||||
|
|
||||||
|
return $sth->fetchrow_hashref;
|
||||||
|
};
|
||||||
|
|
||||||
$self->{pbot}->{logger}->log($@) if $@;
|
$self->{pbot}->{logger}->log($@) if $@;
|
||||||
|
|
||||||
@ -1198,46 +1244,44 @@ sub get_max_messages {
|
|||||||
my ($self, $id, $channel, $use_aliases) = @_;
|
my ($self, $id, $channel, $use_aliases) = @_;
|
||||||
|
|
||||||
my $count = eval {
|
my $count = eval {
|
||||||
my $sql = "SELECT COUNT(*) FROM Messages WHERE channel = ? AND ";
|
my $sql = 'SELECT COUNT(*) FROM Messages WHERE channel = ? AND ';
|
||||||
|
|
||||||
my %akas;
|
my %akas;
|
||||||
if (defined $use_aliases) { %akas = $self->get_also_known_as($use_aliases); }
|
|
||||||
else { $akas{'this'} = {id => $id, type => $self->{alias_type}->{STRONG}, nickchange => 0}; }
|
|
||||||
|
|
||||||
my $ids;
|
if (defined $use_aliases) {
|
||||||
|
%akas = $self->get_also_known_as($use_aliases);
|
||||||
|
} else {
|
||||||
|
$akas{$id} = {
|
||||||
|
id => $id,
|
||||||
|
type => $self->{alias_type}->{STRONG},
|
||||||
|
nickchange => 0,
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
my %seen_id;
|
my %seen_id;
|
||||||
my $or = '';
|
|
||||||
foreach my $aka (keys %akas) {
|
foreach my $aka (keys %akas) {
|
||||||
next if $akas{$aka}->{type} == $self->{alias_type}->{WEAK};
|
next if $akas{$aka}->{type} == $self->{alias_type}->{WEAK};
|
||||||
next if $akas{$aka}->{nickchange} == 1;
|
next if $akas{$aka}->{nickchange} == 1;
|
||||||
next if exists $seen_id{$akas{$aka}->{id}};
|
next if exists $seen_id{$akas{$aka}->{id}};
|
||||||
$seen_id{$akas{$aka}->{id}} = 1;
|
|
||||||
|
|
||||||
$ids .= "${or}id = ?";
|
$seen_id{$akas{$aka}->{id}} = 1;
|
||||||
$or = ' OR ';
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
my $ids = join " OR ", map { "id = ?" } keys %seen_id;
|
||||||
|
|
||||||
$sql .= "($ids)";
|
$sql .= "($ids)";
|
||||||
|
|
||||||
my $sth = $self->{dbh}->prepare($sql);
|
my $sth = $self->{dbh}->prepare($sql);
|
||||||
my $param = 1;
|
my $param = 1;
|
||||||
$sth->bind_param($param++, $channel);
|
$sth->bind_param($param++, $channel);
|
||||||
|
|
||||||
%seen_id = ();
|
map { $sth->bind_param($param++, $_) } keys %seen_id;
|
||||||
foreach my $aka (keys %akas) {
|
|
||||||
next if $akas{$aka}->{type} == $self->{alias_type}->{WEAK};
|
|
||||||
next if $akas{$aka}->{nickchange} == 1;
|
|
||||||
next if exists $seen_id{$akas{$aka}->{id}};
|
|
||||||
$seen_id{$akas{$aka}->{id}} = 1;
|
|
||||||
|
|
||||||
$sth->bind_param($param++, $akas{$aka}->{id});
|
$sth->execute;
|
||||||
}
|
return $sth->fetchrow_hashref->{'COUNT(*)'};
|
||||||
|
|
||||||
$sth->execute();
|
|
||||||
my $row = $sth->fetchrow_hashref();
|
|
||||||
$sth->finish();
|
|
||||||
return $row->{'COUNT(*)'};
|
|
||||||
};
|
};
|
||||||
|
|
||||||
$self->{pbot}->{logger}->log($@) if $@;
|
$self->{pbot}->{logger}->log($@) if $@;
|
||||||
$count = 0 if not defined $count;
|
$count = 0 if not defined $count;
|
||||||
return $count;
|
return $count;
|
||||||
|
Loading…
Reference in New Issue
Block a user