mirror of
https://github.com/pragma-/pbot.git
synced 2025-01-25 19:44:26 +01:00
3d97dc2c33
"Applet" is a much better name for the external command-line scripts and programs that can be loaded as PBot commands. They will no longer be confused with Perl modules. https://en.wikipedia.org/wiki/Applet
395 lines
13 KiB
Perl
Executable File
395 lines
13 KiB
Perl
Executable File
#!/usr/bin/env perl
|
|
|
|
# SPDX-FileCopyrightText: 2021 Pragmatic Software <pragma78@gmail.com>
|
|
# SPDX-License-Identifier: MIT
|
|
|
|
use warnings;
|
|
use strict;
|
|
|
|
use Text::Levenshtein qw(fastdistance);
|
|
use Time::HiRes qw(gettimeofday);
|
|
use Time::Duration qw(duration concise);
|
|
use Fcntl qw(:flock);
|
|
|
|
use lib ".";
|
|
|
|
use QStatskeeper;
|
|
use Scorekeeper;
|
|
use IRCColors;
|
|
|
|
my $CJEOPARDY_DATA = 'data/cjeopardy.dat';
|
|
my $CJEOPARDY_HINT = 'data/cjeopardy.hint';
|
|
my $CJEOPARDY_LAST_ANSWER = 'data/cjeopardy.last_ans';
|
|
|
|
my $hint_only_mode = 0;
|
|
my $concise_duration = 1;
|
|
|
|
my $nick = shift @ARGV;
|
|
my $channel = shift @ARGV;
|
|
my $text = join(' ', @ARGV);
|
|
|
|
sub encode { my $str = shift; $str =~ s/\\(.)/{sprintf "\\%03d", ord($1)}/ge; return $str; }
|
|
sub decode { my $str = shift; $str =~ s/\\(\d{3})/{"\\" . chr($1)}/ge; return $str }
|
|
|
|
if ($channel !~ /^#/) {
|
|
print "Sorry, C Jeopardy must be played in a channel. Feel free to join #cjeopardy.\n";
|
|
exit;
|
|
}
|
|
|
|
while ($text =~ s/^\s*(is|are|the|a|an)\s+//i){};
|
|
$text =~ s/\s*\?*$//;
|
|
$text =~ s/^\s+//;
|
|
$text =~ s/\s+$//;
|
|
my $lctext = lc $text;
|
|
|
|
if (not length $lctext) {
|
|
print "What?\n";
|
|
exit;
|
|
}
|
|
|
|
my @data;
|
|
|
|
open my $semaphore, ">", "$CJEOPARDY_DATA-$channel.lock" or die "Couldn't create semaphore lock: $!";
|
|
flock $semaphore, LOCK_EX;
|
|
|
|
my $ret = open my $fh, "<", "$CJEOPARDY_LAST_ANSWER-$channel";
|
|
if (defined $ret) {
|
|
my $last_nick = <$fh>;
|
|
my $last_answers = <$fh>;
|
|
my $last_timestamp = <$fh>;
|
|
close $fh;
|
|
|
|
chomp $last_nick;
|
|
|
|
if (scalar gettimeofday - $last_timestamp <= 15) {
|
|
$ret = open $fh, "<", "$CJEOPARDY_DATA-$channel";
|
|
if (defined $ret) {
|
|
@data = <$fh>;
|
|
close $fh;
|
|
}
|
|
|
|
my @current_answers = map { decode $_ } split /\|/, encode $data[1] if @data;
|
|
my @valid_answers = map { decode $_ } split /\|/, encode $last_answers;
|
|
|
|
foreach my $answer (@valid_answers) {
|
|
chomp $answer;
|
|
$answer =~ s/\\\|/|/g;
|
|
$answer =~ s/\s*{.*}\s*//;
|
|
|
|
my $skip_last;
|
|
if (@current_answers) {
|
|
foreach my $current_answer (@current_answers) {
|
|
chomp $current_answer;
|
|
$current_answer =~ s/\\\|/|/g;
|
|
$current_answer =~ s/\s*{.*}\s*//;
|
|
|
|
my $distance = fastdistance(lc $answer, lc $current_answer);
|
|
my $length = (length($answer) > length($current_answer)) ? length $answer : length $current_answer;
|
|
|
|
if ($distance / $length < 0.15) {
|
|
$skip_last = 1;
|
|
last;
|
|
}
|
|
}
|
|
}
|
|
|
|
last if $skip_last;
|
|
|
|
my $distance = fastdistance($lctext, lc $answer);
|
|
my $length = (length($lctext) > length($answer)) ? length $lctext : length $answer;
|
|
|
|
if ($distance / $length < 0.15) {
|
|
if ($last_nick eq $nick) {
|
|
print "$color{red}Er, you already correctly answered that question.$color{reset}\n";
|
|
} else {
|
|
my $elapsed = scalar gettimeofday - $last_timestamp;
|
|
my $duration;
|
|
if ($elapsed < 2) {
|
|
$elapsed = 0.01 if $elapsed <= 0.01;
|
|
$duration = sprintf("%.2f", $elapsed);
|
|
} else {
|
|
$duration = sprintf("%d", $elapsed);
|
|
}
|
|
print "$color{red}Too slow by $color{orange}$duration $color{red}second" . ($duration != 1 ? "s" : "") . "! $color{orange}$last_nick$color{red} got the correct answer.$color{reset}\n";
|
|
}
|
|
exit;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
if (not @data) {
|
|
open $fh, "<", "$CJEOPARDY_DATA-$channel" or print "There is no open C Jeopardy question. Use `cjeopardy` to get a question.\n" and exit;
|
|
@data = <$fh>;
|
|
close $fh;
|
|
}
|
|
|
|
my $scores = Scorekeeper->new;
|
|
$scores->begin;
|
|
my $player_id = $scores->get_player_id($nick, $channel);
|
|
my $player_data = $scores->get_player_data($player_id);
|
|
|
|
my ($id) = $data[0] =~ m/^(\d+)/;
|
|
my @valid_answers = map { decode $_ } split /\|/, encode $data[1];
|
|
|
|
my $qstats = QStatskeeper->new;
|
|
$qstats->begin;
|
|
my $qdata = $qstats->get_question_data($id);
|
|
|
|
$qdata->{last_touched} = gettimeofday;
|
|
|
|
my $incorrect_percentage = 100;
|
|
|
|
foreach my $answer (@valid_answers) {
|
|
chomp $answer;
|
|
$answer =~ s/\\\|/|/g;
|
|
|
|
my $supplemental_text;
|
|
if ($answer =~ s/\s*{(.*)}\s*$//) {
|
|
$supplemental_text = $1;
|
|
}
|
|
|
|
if ($answer =~ /^[+-]*[0-9]+$/ and $lctext =~ /^[+-]*[0-9]+$/) {
|
|
my $is_wrong = 0;
|
|
|
|
if ($lctext > $answer) {
|
|
print "$color{red}$lctext is too high!$color{reset}";
|
|
$is_wrong = 1;
|
|
} elsif ($lctext < $answer) {
|
|
print "$color{red}$lctext is too low!$color{reset}";
|
|
$is_wrong = 1;
|
|
}
|
|
|
|
goto WRONG_ANSWER if $is_wrong;
|
|
}
|
|
|
|
my $distance = fastdistance($lctext, lc $answer);
|
|
my $length = (length($lctext) > length($answer)) ? length $lctext : length $answer;
|
|
|
|
my $percentage = $distance / $length * 100;
|
|
|
|
if ($percentage < $incorrect_percentage) {
|
|
$incorrect_percentage = $percentage;
|
|
}
|
|
|
|
if ($percentage < 15) {
|
|
if ($distance == 0) {
|
|
print "'$color{green}$answer$color{reset}' is correct!";
|
|
} else {
|
|
print "'$color{green}$text$color{reset}' is close enough to '$color{green}$answer$color{reset}'. You are correct!"
|
|
}
|
|
|
|
if (defined $supplemental_text) {
|
|
print " $color{purple}$supplemental_text$color{reset}";
|
|
}
|
|
|
|
my $elapsed = scalar gettimeofday - $data[2];
|
|
|
|
if ($concise_duration) {
|
|
if ($elapsed < 60) {
|
|
printf " (%ds)\n", $elapsed;
|
|
} else {
|
|
my $duration = concise duration($elapsed);
|
|
print " ($duration)\n";
|
|
}
|
|
} else {
|
|
if ($elapsed < 60) {
|
|
printf " It took %.2f seconds to answer that question!\n", $elapsed;
|
|
} else {
|
|
my $duration = duration($elapsed);
|
|
print " It took $duration to answer that question.\n";
|
|
}
|
|
}
|
|
|
|
$qdata->{correct}++;
|
|
$qdata->{last_correct_time} = gettimeofday;
|
|
$qdata->{last_correct_nick} = $nick;
|
|
|
|
if (gettimeofday - $qdata->{last_asked} < 60 * 15) {
|
|
$qdata->{average_answer_time} *= $qdata->{correct} - 1;
|
|
$qdata->{average_answer_time} += $elapsed;
|
|
$qdata->{average_answer_time} /= $qdata->{correct};
|
|
}
|
|
|
|
if ($qdata->{quickest_answer_time} == 0 or $elapsed < $qdata->{quickest_answer_time}) {
|
|
$qdata->{quickest_answer_time} = $elapsed;
|
|
$qdata->{quickest_answer_date} = gettimeofday;
|
|
$qdata->{quickest_answer_nick} = $nick;
|
|
}
|
|
|
|
if ($elapsed > $qdata->{longest_answer_time}) {
|
|
$qdata->{longest_answer_time} = $elapsed;
|
|
$qdata->{longest_answer_date} = gettimeofday;
|
|
$qdata->{longest_answer_nick} = $nick;
|
|
}
|
|
|
|
my $streakers = $scores->get_all_correct_streaks($channel);
|
|
|
|
foreach my $streaker (@$streakers) {
|
|
next if lc $streaker->{nick} eq lc $nick;
|
|
|
|
if ($streaker->{correct_streak} >= 3) {
|
|
print "$color{orange}$nick$color{red} ended $color{orange}$streaker->{nick}$color{red}'s $color{orange}$streaker->{correct_streak}$color{red} correct answer streak!$color{reset}\n";
|
|
}
|
|
|
|
$streaker->{correct_streak} = 0;
|
|
$scores->update_player_data($streaker->{id}, $streaker);
|
|
}
|
|
|
|
$player_data->{correct_answers}++;
|
|
$player_data->{lifetime_correct_answers}++;
|
|
$player_data->{correct_streak}++;
|
|
$player_data->{last_correct_timestamp} = scalar gettimeofday;
|
|
$player_data->{wrong_streak} = 0;
|
|
|
|
if ($player_data->{quickest_correct} == 0 or $elapsed < $player_data->{quickest_correct}) {
|
|
$player_data->{quickest_correct} = $elapsed;
|
|
}
|
|
|
|
if ($player_data->{correct_streak} > $player_data->{highest_correct_streak}) {
|
|
$player_data->{highest_correct_streak} = $player_data->{correct_streak};
|
|
}
|
|
|
|
if ($player_data->{highest_correct_streak} > $player_data->{lifetime_highest_correct_streak}) {
|
|
$player_data->{lifetime_highest_correct_streak} = $player_data->{highest_correct_streak};
|
|
}
|
|
|
|
if ($player_data->{correct_streak} == 1) {
|
|
$player_data->{correct_streak_timestamp} = scalar gettimeofday;
|
|
}
|
|
|
|
my $dont_print_streak = 0;
|
|
|
|
my $t1 = $player_data->{lifetime_quickest_correct_streak} ? $player_data->{lifetime_quickest_correct_streak} : 32767;
|
|
my $t2 = gettimeofday - $player_data->{correct_streak_timestamp};
|
|
my $a1 = $player_data->{lifetime_highest_quick_correct_streak} ? $player_data->{lifetime_highest_quick_correct_streak} : 1;
|
|
my $a2 = $player_data->{correct_streak} ? $player_data->{correct_streak} : 1;
|
|
|
|
my $ratio1 = ($t1 + $t1) / $a1;
|
|
my $ratio2 = ($t2 + $t1) / $a2;
|
|
|
|
print STDERR "nick: $nick, t1 = $t1, t2 = $t2, a1 = $a1, a2 = $a2, ratio1 = $ratio1, ratio2 = $ratio2\n";
|
|
|
|
if ($ratio2 < $ratio1 and $player_data->{correct_streak} >= 3) {
|
|
$player_data->{highest_quick_correct_streak} = $player_data->{correct_streak};
|
|
$player_data->{quickest_correct_streak} = gettimeofday - $player_data->{correct_streak_timestamp};
|
|
|
|
$player_data->{lifetime_highest_quick_correct_streak} = $player_data->{highest_quick_correct_streak};
|
|
$player_data->{lifetime_quickest_correct_streak} = $player_data->{quickest_correct_streak};
|
|
|
|
print "$color{orange}$nick$color{cyan} just set a new personal quickest correct answer streak of $color{orange}$player_data->{highest_quick_correct_streak} $color{cyan}correct answers in $color{orange}", duration($player_data->{quickest_correct_streak}), "$color{cyan}!$color{reset}\n";
|
|
$dont_print_streak = 1;
|
|
}
|
|
|
|
unless ($dont_print_streak) {
|
|
my %streaks = (
|
|
3 => "$color{orange}$nick$color{cyan} is on a $color{orange}3$color{cyan} correct answer streak!",
|
|
4 => "$color{orange}$nick$color{cyan} is hot with a $color{orange}4$color{cyan} correct answer streak!",
|
|
5 => "$color{orange}$nick$color{cyan} is on fire with a $color{orange}5$color{cyan} correct answer streak!",
|
|
6 => "$color{orange}$nick$color{cyan} is ON FIRE with a $color{orange}6$color{cyan} correct answer streak!",
|
|
7 => "$color{orange}$nick$color{cyan} is DOMINATING with a $color{orange}7$color{cyan} correct answer streak!",
|
|
8 => "$color{orange}$nick$color{cyan} is DOMINATING with an $color{orange}8$color{cyan} correct answer streak!",
|
|
9 => "$color{orange}$nick$color{cyan} is DOMINATING with a $color{orange}9$color{cyan} correct answer streak!",
|
|
10 => "$color{orange}$nick$color{cyan} IS UNTOUCHABLE WITH A $color{orange}10$color{cyan} CORRECT ANSWER STREAK!"
|
|
);
|
|
|
|
if (exists $streaks{$player_data->{correct_streak}}) {
|
|
print "$streaks{$player_data->{correct_streak}}$color{reset}\n";
|
|
} elsif ($player_data->{correct_streak} > 10) {
|
|
print "$color{orange}$nick$color{cyan} IS UNTOUCHABLE WITH A $color{orange}$player_data->{correct_streak}$color{cyan} CORRECT ANSWER STREAK!$color{reset}\n";
|
|
}
|
|
}
|
|
|
|
$scores->update_player_data($player_id, $player_data);
|
|
$scores->end;
|
|
|
|
$qstats->update_question_data($id, $qdata);
|
|
$qstats->end;
|
|
|
|
unlink "$CJEOPARDY_DATA-$channel";
|
|
unlink "$CJEOPARDY_HINT-$channel";
|
|
|
|
open $fh, ">", "$CJEOPARDY_LAST_ANSWER-$channel" or die "Couldn't open $CJEOPARDY_LAST_ANSWER-$channel: $!";
|
|
my $time = scalar gettimeofday;
|
|
print $fh "$nick\n$data[1]$time\n";
|
|
close $fh;
|
|
|
|
close $semaphore;
|
|
|
|
if ($channel eq '#cjeopardy' or $channel eq '#c-jeopardy') {
|
|
my $question = `./cjeopardy.pl $channel`;
|
|
|
|
if ($hint_only_mode) {
|
|
my $hint = `./cjeopardy_hint.pl candide $channel`;
|
|
$hint =~ s/^Hint: //;
|
|
print "Next hint: $hint\n";
|
|
} else {
|
|
if ($question =~ s/^(\[Filter active[^\n]+\n)//) {
|
|
print $1;
|
|
}
|
|
print "$color{magneta}Next question$color{reset}: $question\n";
|
|
}
|
|
}
|
|
|
|
exit;
|
|
}
|
|
}
|
|
|
|
my $correct_percentage = 100 - $incorrect_percentage;
|
|
if ($correct_percentage >= 80) {
|
|
printf "Sorry, '$color{red}$text$color{reset}' is %.1f%% correct. So close!", $correct_percentage;
|
|
} elsif ($correct_percentage >= 70) {
|
|
printf "Sorry, '$color{red}$text$color{reset}' is %.1f%% correct. Almost.", $correct_percentage;
|
|
} elsif ($correct_percentage >= 50) {
|
|
printf "Sorry, '$color{red}$text$color{reset}' is only %.1f%% correct.", $correct_percentage;
|
|
} else {
|
|
print "Sorry, '$color{red}$text$color{reset}' is incorrect.";
|
|
}
|
|
|
|
WRONG_ANSWER:
|
|
$player_data->{wrong_answers}++;
|
|
$player_data->{lifetime_wrong_answers}++;
|
|
$player_data->{wrong_streak}++;
|
|
$player_data->{last_wrong_timestamp} = scalar gettimeofday;
|
|
|
|
if ($player_data->{correct_streak} >= 3) {
|
|
print " $color{red}You just ended your $color{orange}$player_data->{correct_streak} $color{red}correct answer streak!$color{reset}\n";
|
|
} else {
|
|
print "\n";
|
|
}
|
|
|
|
$player_data->{correct_streak} = 0;
|
|
|
|
if ($player_data->{wrong_streak} > $player_data->{highest_wrong_streak}) {
|
|
$player_data->{highest_wrong_streak} = $player_data->{wrong_streak};
|
|
}
|
|
|
|
if ($player_data->{highest_wrong_streak} > $player_data->{lifetime_highest_wrong_streak}) {
|
|
$player_data->{lifetime_highest_wrong_streak} = $player_data->{highest_wrong_streak};
|
|
}
|
|
|
|
$qdata->{wrong}++;
|
|
$qdata->{wrong_streak}++;
|
|
|
|
if ($qdata->{wrong_streak} > $qdata->{highest_wrong_streak}) {
|
|
$qdata->{highest_wrong_streak} = $qdata->{wrong_streak};
|
|
}
|
|
|
|
$qstats->add_wrong_answer($id, $lctext, $nick);
|
|
|
|
my %streaks = (
|
|
4 => "Maybe try asking for a hint, $nick?",
|
|
5 => "Guessing, are we, $nick?",
|
|
7 => "Think of your correct/incorrect ratio! Use a hint, $nick!",
|
|
);
|
|
|
|
if (exists $streaks{$player_data->{wrong_streak}}) {
|
|
print "$streaks{$player_data->{wrong_streak}}$color{reset}\n";
|
|
}
|
|
|
|
$scores->update_player_data($player_id, $player_data);
|
|
$scores->end;
|
|
|
|
$qstats->update_question_data($id, $qdata);
|
|
$qstats->end;
|