2020-07-12 03:32:12 +02:00
|
|
|
{-# LANGUAGE TemplateHaskell #-}
|
|
|
|
module Main where
|
|
|
|
|
|
|
|
import Network.IRC.Client
|
|
|
|
import Control.Lens
|
|
|
|
import NLP.POS
|
|
|
|
import NLP.Types (POSTagger)
|
2020-07-31 17:17:34 +02:00
|
|
|
import qualified NLP.Types.Tags as Tags
|
2020-07-12 03:32:12 +02:00
|
|
|
import NLP.Types.Tree
|
|
|
|
import qualified NLP.Corpora.Conll as Conll
|
|
|
|
import NLP.Corpora.Conll (Tag)
|
|
|
|
import qualified Data.ByteString as BS
|
|
|
|
import System.Random
|
|
|
|
import System.Envy
|
2021-05-22 15:42:14 +02:00
|
|
|
import System.IO as S
|
2020-07-14 19:37:36 +02:00
|
|
|
import Data.Maybe
|
2021-05-22 15:42:14 +02:00
|
|
|
import Data.Foldable (traverse_)
|
2020-08-03 18:42:58 +02:00
|
|
|
import qualified Data.Text
|
2020-07-12 03:32:12 +02:00
|
|
|
--------------------------------------------------------------------------------
|
|
|
|
|
|
|
|
data Config = Config
|
2020-07-14 19:37:36 +02:00
|
|
|
{ _owoChance :: Int
|
|
|
|
, _ircServer :: ByteString
|
|
|
|
, _ircPort :: Int
|
|
|
|
, _ircServerPassword :: Maybe Text
|
|
|
|
, _nickservPassword :: Maybe Text
|
|
|
|
, _ircNick :: Maybe Text
|
2021-05-22 15:42:14 +02:00
|
|
|
, _ircIdent :: Maybe Text
|
|
|
|
, _ircChannels :: [Text]
|
2020-07-12 03:32:12 +02:00
|
|
|
}
|
|
|
|
deriving stock (Show, Eq, Generic)
|
|
|
|
makeLenses ''Config
|
|
|
|
|
2021-05-22 15:42:14 +02:00
|
|
|
instance Var [Text] where
|
|
|
|
toVar ts = show ts
|
|
|
|
fromVar s = readMaybe s >>= (pure . map Data.Text.pack)
|
|
|
|
|
2020-07-14 19:37:36 +02:00
|
|
|
instance FromEnv Config where
|
|
|
|
fromEnv _ =
|
|
|
|
Config <$> env "OWO_CHANCE"
|
|
|
|
<*> env "IRC_SERVER"
|
|
|
|
<*> env "IRC_PORT"
|
|
|
|
<*> envMaybe "IRC_SERVER_PASSWORD"
|
|
|
|
<*> envMaybe "NICKSERV_PASSWORD"
|
|
|
|
<*> envMaybe "IRC_NICK"
|
2021-05-22 15:42:14 +02:00
|
|
|
<*> envMaybe "IRC_IDENT"
|
|
|
|
<*> env "IRC_CHANNELS"
|
2020-07-14 19:37:36 +02:00
|
|
|
|
2020-07-13 21:09:41 +02:00
|
|
|
stopWord :: Text -> Bool
|
|
|
|
stopWord "'s" = True
|
2020-07-17 16:17:22 +02:00
|
|
|
stopWord "\"" = True
|
2020-07-13 21:09:41 +02:00
|
|
|
stopWord "is" = True
|
|
|
|
stopWord "are" = True
|
|
|
|
stopWord "am" = True
|
|
|
|
stopWord "were" = True
|
|
|
|
stopWord "was" = True
|
|
|
|
stopWord "be" = True
|
|
|
|
stopWord _ = False
|
|
|
|
|
2020-07-31 17:17:34 +02:00
|
|
|
pickVerb :: POS Tag -> Maybe Text
|
|
|
|
pickVerb (POS Conll.VB (Token verb)) = Just verb
|
|
|
|
pickVerb (POS Conll.VBD (Token verb)) = Just verb
|
|
|
|
pickVerb (POS Conll.VBG (Token verb)) = Just verb
|
|
|
|
pickVerb (POS Conll.VBN (Token verb)) = Just verb
|
|
|
|
pickVerb (POS Conll.VBZ (Token verb)) = Just verb
|
|
|
|
pickVerb _ = Nothing
|
|
|
|
|
|
|
|
pickNoun :: POS Tag -> Maybe Text
|
|
|
|
pickNoun (POS Conll.NN (Token noun)) = Just noun
|
|
|
|
pickNoun _ = Nothing
|
|
|
|
|
|
|
|
randomPOS
|
|
|
|
:: Tags.Tag tag
|
|
|
|
=> (POS tag -> Maybe Text)
|
|
|
|
-> POSTagger tag
|
|
|
|
-> Text
|
|
|
|
-> IO (Maybe Text)
|
|
|
|
randomPOS pickPOS tagger s = do
|
|
|
|
let candidates
|
|
|
|
= filter (not . stopWord)
|
|
|
|
. mapMaybe pickPOS
|
|
|
|
$ tag tagger s >>= \(TaggedSent ps) -> ps
|
|
|
|
i <- randomRIO (0, length candidates - 1)
|
|
|
|
pure $ candidates ^? ix i
|
2020-07-12 03:32:12 +02:00
|
|
|
|
2020-07-13 19:39:57 +02:00
|
|
|
doOwo :: MonadIO m => Config -> m Bool
|
|
|
|
doOwo conf = do
|
|
|
|
n <- liftIO (randomRIO @Int (0, conf ^. owoChance))
|
2020-07-12 03:32:12 +02:00
|
|
|
pure $ n == 0
|
|
|
|
|
2020-07-31 17:17:34 +02:00
|
|
|
data OwoType = Noun | Verb
|
|
|
|
deriving stock (Show, Eq)
|
|
|
|
|
|
|
|
instance Random OwoType where
|
|
|
|
random = over _1 (bool Noun Verb) . random
|
|
|
|
randomR = const random
|
|
|
|
|
2020-08-21 15:21:53 +02:00
|
|
|
vowels :: [Char]
|
|
|
|
vowels = "aeiou"
|
|
|
|
|
|
|
|
article :: Text -> Text
|
|
|
|
article (x :< _) | x `elem` vowels = "an"
|
|
|
|
article _ = "a"
|
|
|
|
|
2020-07-31 17:17:34 +02:00
|
|
|
owo :: OwoType -> Text -> Text
|
2020-08-21 15:21:53 +02:00
|
|
|
owo Noun n = mconcat
|
|
|
|
[ "I'm "
|
|
|
|
, article n
|
|
|
|
, " "
|
|
|
|
, n
|
|
|
|
, if "o" `Data.Text.isSuffixOf` n
|
|
|
|
then "wo"
|
|
|
|
else " owo"
|
|
|
|
]
|
2020-07-31 17:17:34 +02:00
|
|
|
owo Verb v = v <> " me owo"
|
|
|
|
|
|
|
|
pickOwo :: OwoType -> POS Tag -> Maybe Text
|
|
|
|
pickOwo Verb = pickVerb
|
|
|
|
pickOwo Noun = pickNoun
|
|
|
|
|
|
|
|
randomOwo :: OwoType -> POSTagger Tag -> Text -> IO (Maybe Text)
|
|
|
|
randomOwo = randomPOS . pickOwo
|
|
|
|
|
2020-07-14 19:37:36 +02:00
|
|
|
owothiaHandler :: Config -> Text -> IORef Bool -> POSTagger Tag -> EventHandler s
|
|
|
|
owothiaHandler conf nick state tagger = EventHandler Just $ \src ev -> do
|
2020-07-12 03:32:12 +02:00
|
|
|
hasIdentified <- readIORef state
|
|
|
|
when (not hasIdentified) $ do
|
|
|
|
nickservAuth
|
2021-05-22 15:42:14 +02:00
|
|
|
traverse_ (send . Join) (conf ^. ircChannels)
|
2020-07-12 03:32:12 +02:00
|
|
|
writeIORef state True
|
|
|
|
|
|
|
|
when ("You are now identified" `BS.isInfixOf` (ev ^. raw)) $
|
2021-05-22 15:42:14 +02:00
|
|
|
traverse_ (send . Join) (conf ^. ircChannels)
|
2020-07-12 03:32:12 +02:00
|
|
|
|
|
|
|
case (src, ev ^. message) of
|
2021-05-22 15:42:14 +02:00
|
|
|
(Channel chan nick, Privmsg _ (Right m)) -> do
|
2020-07-13 19:39:57 +02:00
|
|
|
willOwo <- doOwo conf
|
2021-05-22 15:42:14 +02:00
|
|
|
when willOwo $ owoMessage chan m
|
|
|
|
_ -> pure()
|
2020-07-12 03:32:12 +02:00
|
|
|
|
|
|
|
pure ()
|
|
|
|
|
|
|
|
where
|
2021-05-22 15:42:14 +02:00
|
|
|
owoMessage chan m = do
|
2020-07-31 17:17:34 +02:00
|
|
|
owoType <- liftIO randomIO
|
|
|
|
mWord <- liftIO $ randomOwo owoType tagger m
|
2021-05-22 15:42:14 +02:00
|
|
|
for_ mWord $ \word -> send $ Privmsg chan $ Right $ owo owoType word
|
2020-07-14 19:37:36 +02:00
|
|
|
nickservAuthMsg = "IDENTIFY " <> nick <> " " <> fromJust (conf ^. nickservPassword)
|
2020-07-12 03:32:12 +02:00
|
|
|
nickservAuth = send $ Privmsg "NickServ" $ Right nickservAuthMsg
|
|
|
|
|
2020-07-14 19:37:36 +02:00
|
|
|
main :: IO ()
|
|
|
|
main = do
|
|
|
|
conf <- either fail pure =<< decodeEnv
|
2020-07-12 03:32:12 +02:00
|
|
|
tagger <- defaultTagger
|
2020-07-14 19:37:36 +02:00
|
|
|
state <- newIORef $ not . isJust $ (conf ^. nickservPassword)
|
2021-05-22 15:42:14 +02:00
|
|
|
S.hSetBuffering stdout LineBuffering
|
2020-07-17 16:17:22 +02:00
|
|
|
let nick = fromMaybe "owothia" (conf ^. ircNick)
|
2020-07-14 19:37:36 +02:00
|
|
|
conn =
|
|
|
|
plainConnection (conf ^. ircServer) (conf ^. ircPort)
|
2020-07-13 21:11:11 +02:00
|
|
|
& realname .~ "Owothia Revströwö"
|
2020-07-14 19:37:36 +02:00
|
|
|
& password .~ (conf ^. ircServerPassword)
|
2021-05-22 15:42:14 +02:00
|
|
|
& username .~ fromMaybe "owothia" (conf ^. ircIdent)
|
2020-07-12 03:32:12 +02:00
|
|
|
& logfunc .~ stdoutLogger
|
|
|
|
cfg =
|
2020-07-14 19:37:36 +02:00
|
|
|
defaultInstanceConfig nick
|
2021-05-22 15:42:14 +02:00
|
|
|
& channels .~ (conf ^. ircChannels)
|
2020-07-14 19:37:36 +02:00
|
|
|
& handlers %~ (owothiaHandler conf nick state tagger : )
|
2020-07-12 03:32:12 +02:00
|
|
|
runClient conn cfg ()
|