Move markov to be a supervisor plugin

This is probably going to be necessary to avoid race conditions in the
ETS storage

Signed-off-by: Alek Ratzloff <alekratz@gmail.com>
This commit is contained in:
2020-07-09 14:51:33 -07:00
parent 010d60f4f9
commit 8cddd8e78e
3 changed files with 71 additions and 36 deletions

50
lib/contrib/markov/bot.ex Normal file
View File

@@ -0,0 +1,50 @@
defmodule Omnibot.Contrib.Markov.Bot do
use Omnibot.Plugin
alias Omnibot.Contrib.Markov.Chain
@impl true
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)
end
@impl true
def on_channel_msg(_irc, channel, nick, msg) do
train(channel, nick, msg)
end
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
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
end
end

View File

@@ -1,46 +1,32 @@
defmodule Omnibot.Contrib.Markov do
use Omnibot.Plugin
use Omnibot.Plugin.Base
alias Omnibot.Contrib.Markov
use Supervisor
alias Omnibot.Contrib.Markov.Chain
@default_config path: "markov", order: 2, save_every: 5 * 60
@default_config path: "markov", order: 2
@impl true
def on_init(cfg) do
# Create the markov database
path = String.to_atom(cfg[:path])
:ets.new(path, [:public])
def start_link(opts) do
Supervisor.start_link(__MODULE__, opts[:cfg], opts)
end
@impl true
def on_channel_msg(_irc, channel, nick, msg) do
train(channel, nick, msg)
def init(cfg) do
children = [
{Markov.Bot, cfg: cfg, name: Omnibot.Contrib.Markov.Bot},
{Task, fn -> save_loop(cfg) end}
]
Supervisor.init(children, strategy: :one_for_all)
end
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)
defp save_loop(cfg) do
save_every = cfg[:save_every]
Process.sleep(save_every * 1000)
Markov.Bot.save_chains()
end
def user_chain(channel, user) do
db = state()
case :ets.lookup(db, {channel, user}) do
[] -> nil
[{{^channel, ^user}, chains}] -> chains
end
end
@impl true
def on_msg(irc, msg), do: Markov.Bot.on_msg(irc, msg)
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
@impl true
def on_channel_msg(irc, channel, nick, msg), do: Markov.Bot.on_channel_msg(irc, channel, nick, msg)
end