2020-07-08 11:29:13 -07:00
|
|
|
defmodule Omnibot.Contrib.Markov do
|
2020-07-12 12:19:14 -07:00
|
|
|
use Omnibot.Plugin
|
2020-07-11 15:43:07 -07:00
|
|
|
alias Omnibot.Contrib.Markov.Chain
|
|
|
|
|
require Logger
|
2020-07-08 11:29:13 -07:00
|
|
|
|
2020-07-09 14:51:33 -07:00
|
|
|
@default_config path: "markov", order: 2, save_every: 5 * 60
|
2020-07-08 11:29:13 -07:00
|
|
|
|
2020-07-11 15:43:07 -07:00
|
|
|
@impl true
|
2020-07-11 16:01:57 -07:00
|
|
|
def children(cfg, _state) do
|
2020-07-11 15:43:07 -07:00
|
|
|
[{Task, fn ->
|
2020-07-11 16:01:57 -07:00
|
|
|
Stream.timer(cfg[:save_every] * 1000)
|
2020-07-11 15:43:07 -07:00
|
|
|
|> Stream.cycle()
|
|
|
|
|
|> Stream.each(fn _ -> save_chains() end)
|
|
|
|
|
|> Stream.run()
|
|
|
|
|
end}]
|
2020-07-08 11:29:13 -07:00
|
|
|
end
|
|
|
|
|
|
|
|
|
|
@impl true
|
2020-07-11 15:43:07 -07:00
|
|
|
def on_init(_cfg) do
|
|
|
|
|
# Create the markov database
|
|
|
|
|
path = String.to_atom(cfg()[:path])
|
|
|
|
|
{:ok, db} = :dets.open_file(path, [:named_table])
|
|
|
|
|
chains = :ets.new(:markov_chains, [:public])
|
|
|
|
|
:dets.to_ets(db, chains)
|
|
|
|
|
:dets.close(db)
|
2020-07-08 17:25:26 -07:00
|
|
|
end
|
|
|
|
|
|
2020-07-11 15:43:07 -07:00
|
|
|
@impl true
|
|
|
|
|
def on_channel_msg(_irc, channel, nick, msg) do
|
|
|
|
|
train(channel, nick, msg)
|
2020-07-08 17:25:26 -07:00
|
|
|
end
|
|
|
|
|
|
2020-07-11 15:43:07 -07:00
|
|
|
def train(channel, user, msg) do
|
|
|
|
|
chain = (user_chain(channel, user) || create_user_chain(channel, user))
|
|
|
|
|
|> Chain.train(msg)
|
|
|
|
|
true = update_user_chain(channel, user, chain)
|
|
|
|
|
end
|
2020-07-08 17:25:26 -07:00
|
|
|
|
2020-07-11 15:43:07 -07:00
|
|
|
def user_chain(channel, user) do
|
|
|
|
|
db = state()
|
|
|
|
|
case :ets.lookup(db, {channel, user}) do
|
|
|
|
|
[] -> nil
|
|
|
|
|
[{{^channel, ^user}, chains}] -> chains
|
|
|
|
|
end
|
|
|
|
|
end
|
|
|
|
|
|
|
|
|
|
def update_user_chain(channel, user, chain) do
|
|
|
|
|
db = state()
|
|
|
|
|
case user_chain(channel, user) do
|
|
|
|
|
nil -> :ets.insert_new(db, {{channel, user}, chain})
|
|
|
|
|
chain -> :ets.insert(db, {{channel, user}, chain})
|
|
|
|
|
end
|
|
|
|
|
end
|
|
|
|
|
|
|
|
|
|
defp create_user_chain(channel, user) do
|
|
|
|
|
true = update_user_chain(channel, user, %Chain{order: cfg()[:order]})
|
|
|
|
|
user_chain(channel, user)
|
|
|
|
|
end
|
|
|
|
|
|
|
|
|
|
def save_chains() do
|
|
|
|
|
# TODO
|
|
|
|
|
Logger.info("Saved markov chains")
|
|
|
|
|
end
|
2020-07-08 11:29:13 -07:00
|
|
|
end
|