X-Git-Url: https://git.localhorst.tv/?a=blobdiff_plain;f=app%2FTwitchBot%2FTwitchChatBot.php;h=92894262f3b108d42826f76238dca6c909146a94;hb=8645b77ea2dc402f0265e1c8022ba18302506ca1;hp=7cdc704bedfcd4e6c25a40298929430405070690;hpb=1d3c8c6a96fc45d839f0e3719baca790059d189f;p=alttp.git diff --git a/app/TwitchBot/TwitchChatBot.php b/app/TwitchBot/TwitchChatBot.php index 7cdc704..9289426 100644 --- a/app/TwitchBot/TwitchChatBot.php +++ b/app/TwitchBot/TwitchChatBot.php @@ -3,6 +3,7 @@ namespace App\TwitchBot; use App\Models\Channel; +use App\Models\ChatBotLog; use App\Models\ChatLog; use Illuminate\Support\Arr; use Illuminate\Support\Str; @@ -74,17 +75,29 @@ class TwitchChatBot extends TwitchBot { $text = $this->contextualMsg($channel); if (!$text) $text = $this->randomChat($channel); if (!$text) return; + $actual_text = is_object($text) ? $text->text_content : $text; $this->tagChannelWrite($channel); - $this->sendIRCMessage(IRCMessage::privmsg($channel->twitch_chat, $text)); + $this->sendIRCMessage(IRCMessage::privmsg($channel->twitch_chat, $actual_text)); + $log = new ChatBotLog(); + $log->channel()->associate($channel); + if (is_object($text)) { + $log->origin()->associate($text); + $log->category = $text->classification; + } else { + $log->category = $this->getLastSpecialSent($channel); + } + $log->text = $actual_text; + $log->save(); } private function getNotes(Channel $channel) { if (!isset($this->notes[$channel->id])) { $this->notes[$channel->id] = [ 'last_read' => 0, - 'last_special' => '', + 'last_special' => [], 'last_write' => time(), 'latest_msgs' => [], + 'queued_special' => false, 'read_since_last_write' => 0, 'wait_msgs' => $this->randomWaitMsgs($channel), 'wait_time' => $this->randomWaitTime($channel), @@ -106,145 +119,86 @@ class TwitchChatBot extends TwitchBot { $this->notes[$channel->id][$name] = $value; } - private function checkForGG(Channel $channel) { - $notes = $this->getNotes($channel); - $ggs = 0; - foreach ($notes['latest_msgs'] as $text) { - if (ChatLog::classify($text) == 'gg') { - ++$ggs; - } - } - return $ggs > 2; - } - - private function checkForGLHF(Channel $channel) { - $notes = $this->getNotes($channel); - $gls = 0; - foreach ($notes['latest_msgs'] as $text) { - if (ChatLog::classify($text) == 'gl') { - ++$gls; - } - } - return $gls > 2; - } - - private function checkForGreeting(Channel $channel) { - $notes = $this->getNotes($channel); - $his = 0; - foreach ($notes['latest_msgs'] as $text) { - if (ChatLog::classify($text) == 'hi') { - ++$his; - } - } - return $his > 2; - } - - private function checkForHype(Channel $channel) { - $notes = $this->getNotes($channel); - $hypes = 0; - foreach ($notes['latest_msgs'] as $text) { - if (ChatLog::classify($text) == 'hype') { - ++$hypes; - } - } - return $hypes > 2; - } - - private function checkForLaughter(Channel $channel) { + private function collectClassifications(Channel $channel) { + $classifications = []; $notes = $this->getNotes($channel); - $lulz = 0; - foreach ($notes['latest_msgs'] as $text) { - if (ChatLog::classify($text) == 'lol') { - ++$lulz; + foreach ($notes['latest_msgs'] as $msg) { + $classification = $msg->classify(); + if ($classification == 'unclassified') continue; + if (isset($classifications[$classification])) { + ++$classifications[$classification]; + } else { + $classifications[$classification] = 1; } } - return $lulz > 2; - } - - private function checkForNumbers(Channel $channel) { - $notes = $this->getNotes($channel); - $numbers = 0; - foreach ($notes['latest_msgs'] as $text) { - if (is_numeric(trim($text))) { - ++$numbers; - } - } - return $numbers > 2; - } - - private function checkForPog(Channel $channel) { - $notes = $this->getNotes($channel); - $pogs = 0; - foreach ($notes['latest_msgs'] as $text) { - if (ChatLog::classify($text) == 'pog') { - ++$pogs; - } - } - return $pogs > 2; - } - - private function checkForSalute(Channel $channel) { - $notes = $this->getNotes($channel); - $o7s = 0; - foreach ($notes['latest_msgs'] as $text) { - if (ChatLog::classify($text) == 'o7') { - ++$o7s; - } - } - return $o7s > 2; + arsort($classifications); + return $classifications; } private function contextualMsg(Channel $channel) { - $last = $this->getNote($channel, 'last_special'); - if ($last != 'gg' && $this->checkForGG($channel)) { - $this->setNote($channel, 'last_special', 'gg'); - return $channel->randomOfClass('gg'); - } - if ($last != 'number' && $this->checkForNumbers($channel)) { - $this->setNote($channel, 'last_special', 'number'); - return $this->randomContextualNumber($channel); - } - if ($last != 'lol' && $this->checkForLaughter($channel)) { - $this->setNote($channel, 'last_special', 'lol'); - return $this->randomLaughter($channel); - } - if ($last != 'glhf' && $this->checkForGLHF($channel)) { - $this->setNote($channel, 'last_special', 'glhf'); - return $channel->randomOfClass('gl'); - } - if ($last != 'hi' && $this->checkForGreeting($channel)) { - $this->setNote($channel, 'last_special', 'hi'); - return $channel->randomOfClass('hi'); - } - if ($last != 'hype' && $this->checkForHype($channel)) { - $this->setNote($channel, 'last_special', 'hype'); - return $channel->randomOfClass('hype'); + if ($this->hasQueuedSpecial($channel)) { + $classification = $this->getQueuedSpecial($channel); + if (is_string($classification)) { + $this->tagChannelSpecialSent($channel, $classification); + } + $this->clearQueuedSpecial($channel); + return $this->getRandomOfClass($channel, $classification); } - if ($last != 'pog' && $this->checkForPog($channel)) { - $this->setNote($channel, 'last_special', 'pog'); - return $channel->randomOfClass('pog'); + $latest_msg = $this->getLatestMessage($channel); + if ($latest_msg->classify() == 'question') { + $response = $latest_msg->getResponseCategory(); + return $this->getRandomOfClass($channel, $response); } - if ($last != 'o7' && $this->checkForSalute($channel)) { - $this->setNote($channel, 'last_special', 'o7'); - return $channel->randomOfClass('o7'); + $last = $this->getLastSpecialSent($channel); + $classifications = $this->collectClassifications($channel); + $count_quotas = [ + 'gg' => 2, + 'gl' => 2, + 'hi' => 2, + 'hype' => 2, + 'lol' => 2, + 'love' => 2, + 'number' => 2, + 'pog' => 2, + 'o7' => 2, + 'wtf' => 2, + ]; + $time_quotas = [ + 'gg' => 600, + 'gl' => 900, + 'hi' => 60, + 'hype' => 60, + 'lol' => 60, + 'love' => 60, + 'number' => 300, + 'pog' => 60, + 'o7' => 300, + 'wtf' => 60, + ]; + foreach ($classifications as $classification => $count) { + if ($classification == $last) continue; + if (!isset($count_quotas[$classification]) || $count < $count_quotas[$classification]) continue; + if (!isset($time_quotas[$classification]) || $this->getTimeSinceSpecial($channel, $classification) < $time_quotas[$classification]) continue; + $this->tagChannelSpecialSent($channel, $classification); + $reaction = $this->getChimeInReaction($channel, $classification); + return $this->getRandomOfClass($channel, $reaction); } return false; } private function randomChat(Channel $channel) { - $line = $channel->queryChatlog() - ->whereIn('classification', ['hi', 'hype', 'lol', 'pog', 'unclassified']) + return $channel->queryChatlog() + ->whereNotIn('classification', ['gg', 'gl', 'number', 'o7']) ->first(); - return $line->text_content; } private function randomContextualNumber(Channel $channel) { $notes = $this->getNotes($channel); $min = 100000; $max = 0; - foreach ($notes['latest_msgs'] as $text) { - if (is_numeric(trim($text))) { - $number = intval(trim($text)); + foreach ($notes['latest_msgs'] as $msg) { + if ($msg->classify() == 'number') { + $number = $msg->getNumericValue(); $min = min($min, $number); $max = max($max, $number); } @@ -253,6 +207,9 @@ class TwitchChatBot extends TwitchBot { } private function randomLaughter(Channel $channel) { + if (!random_int(0, 2)) { + return $channel->randomOfClass('lol'); + } return Arr::random([ ':tf:', '4Head', @@ -263,7 +220,6 @@ class TwitchChatBot extends TwitchBot { 'GunRun', 'heh', 'Hhhehehe', - 'HypeLUL', 'Jebaited', 'Jebasted', 'KEKW', @@ -282,13 +238,11 @@ class TwitchChatBot extends TwitchBot { 'SUBprise', 'xD', 'YouDontSay', - $channel->randomOfClass('lol'), ]); } private function randomMsg(Channel $channel) { - $line = $channel->queryChatlog()->first(); - return $line->text_content; + return $channel->queryChatlog()->first(); } private function randomWaitMsgs(Channel $channel) { @@ -303,12 +257,33 @@ class TwitchChatBot extends TwitchBot { return random_int($min, $max); } + private function queueSpecial(Channel $channel, $classification) { + $this->getNotes($channel); + $this->notes[$channel->id]['queued_special'] = $classification; + } + + private function hasQueuedSpecial(Channel $channel) { + return !!$this->getQueuedSpecial($channel); + } + + private function getQueuedSpecial(Channel $channel) { + $notes = $this->getNotes($channel); + return $notes['queued_special']; + } + + private function clearQueuedSpecial(Channel $channel) { + $this->getNotes($channel); + $this->notes[$channel->id]['queued_special'] = false; + } + private function tagChannelRead(Channel $channel, IRCMessage $msg) { $this->getNotes($channel); $this->notes[$channel->id]['last_read'] = time(); ++$this->notes[$channel->id]['read_since_last_write']; - if (!ChatLog::isKnownBot($msg->nick) && !ChatLog::spammyText($msg->getText())) { - $this->notes[$channel->id]['latest_msgs'][] = $msg->getText(); + + $tokenized = $msg->tokenize(); + if (!ChatLog::isKnownBot($msg->nick) && !$tokenized->isSpammy()) { + $this->notes[$channel->id]['latest_msgs'][] = $tokenized; if (count($this->notes[$channel->id]['latest_msgs']) > 10) { array_shift($this->notes[$channel->id]['latest_msgs']); } @@ -316,6 +291,10 @@ class TwitchChatBot extends TwitchBot { if ($this->isDirectedAtMe($msg->getText()) && $this->shouldRespond($channel)) { $this->notes[$channel->id]['wait_msgs'] = 0; $this->notes[$channel->id]['wait_time'] = 0; + $response = $tokenized->getResponseCategory(); + if ($response) { + $this->queueSpecial($channel, $response); + } } } @@ -327,9 +306,43 @@ class TwitchChatBot extends TwitchBot { $this->notes[$channel->id]['wait_time'] = $this->randomWaitTime($channel); } + private function tagChannelSpecialSent(Channel $channel, $classification) { + $this->getNotes($channel); + $this->notes[$channel->id]['last_special'][$classification] = time(); + } + + private function getLatestMessage(Channel $channel) { + $this->getNotes($channel); + if (!empty($notes['latest_msgs'])) { + return $notes['latest_msgs'][count($notes['latest_msgs']) - 1]; + } + return TokenizedMessage::fromString(''); + } + + private function getLastSpecialSent(Channel $channel) { + $notes = $this->getNotes($channel); + $max_time = 0; + $max_classification = ''; + foreach ($notes['last_special'] as $classification => $time) { + if ($time > $max_time) { + $max_time = $time; + $max_classification = $classification; + } + } + return $max_classification; + } + + private function getTimeSinceSpecial(Channel $channel, $classification) { + $notes = $this->getNotes($channel); + if (isset($notes['last_special'][$classification])) { + return time() - $notes['last_special'][$classification]; + } + return 999999; + } + private function isDirectedAtMe($raw_text) { $text = strtolower($raw_text); - if (strpos($text, 'horstie') !== false) { + if (strpos($text, 'horsti') !== false) { return true; } return false; @@ -346,6 +359,32 @@ class TwitchChatBot extends TwitchBot { return false; } + private function getRandomOfClass(Channel $channel, $classification) { + if ($classification == 'number') { + return $this->randomContextualNumber($channel); + } + if ($classification == 'lol') { + return $this->randomLaughter($channel); + } + return $channel->randomOfClass($classification); + } + + private function getChimeInReaction(Channel $channel, $classification) { + switch ($classification) { + case 'hi': + return ['hi', 'love']; + case 'hype': + return ['hype', 'love', 'pog']; + case 'lol': + return ['kappa', 'lol']; + case 'pog': + return ['hype', 'pog']; + case 'wtf': + return ['lol', 'wtf']; + } + return $classification; + } + private $channels; private $notes = [];