Add merge/1 and merge/2 to Markov.Chain
merge/1 takes a nonempty list of markov chains to merge merge/2 takes two markov chains of the same order and sums their weights Signed-off-by: Alek Ratzloff <alekratz@gmail.com>
This commit is contained in:
@@ -1,5 +1,6 @@
|
|||||||
defmodule Omnibot.Contrib.Markov.Chain do
|
defmodule Omnibot.Contrib.Markov.Chain do
|
||||||
alias Omnibot.{Contrib.Markov.Chain, Util}
|
alias Omnibot.{Contrib.Markov.Chain, Util}
|
||||||
|
require Logger
|
||||||
|
|
||||||
@enforce_keys [:order]
|
@enforce_keys [:order]
|
||||||
defstruct order: 2, chain: %{}
|
defstruct order: 2, chain: %{}
|
||||||
@@ -33,22 +34,6 @@ defmodule Omnibot.Contrib.Markov.Chain do
|
|||||||
%Chain{chain: chain, order: order}
|
%Chain{chain: chain, order: order}
|
||||||
end
|
end
|
||||||
|
|
||||||
def get(chain, key) when is_list(chain) do
|
|
||||||
item = Enum.find(chain, fn {listkey, _} -> listkey == key end)
|
|
||||||
case item do
|
|
||||||
nil -> nil
|
|
||||||
{_, weights} -> weights
|
|
||||||
end
|
|
||||||
end
|
|
||||||
|
|
||||||
def get(chain, key), do: get(chain.chain, key)
|
|
||||||
|
|
||||||
def find_index(chain, key) when is_list(chain) do
|
|
||||||
Enum.find_index(chain, fn {listkey, _} -> listkey == key end)
|
|
||||||
end
|
|
||||||
|
|
||||||
def find_index(chain = %Chain{}, key), do: find_index(chain.chain, key)
|
|
||||||
|
|
||||||
def generate(chain) do
|
def generate(chain) do
|
||||||
{seed, _} = Stream.filter(chain.chain, fn {key, _} -> length(key) == chain.order end)
|
{seed, _} = Stream.filter(chain.chain, fn {key, _} -> length(key) == chain.order end)
|
||||||
|> Enum.random()
|
|> Enum.random()
|
||||||
@@ -59,10 +44,44 @@ defmodule Omnibot.Contrib.Markov.Chain do
|
|||||||
do_generate(chain, key) |> Enum.join(" ")
|
do_generate(chain, key) |> Enum.join(" ")
|
||||||
end
|
end
|
||||||
|
|
||||||
|
def load!(path) do
|
||||||
|
{:ok, chain} = load(path)
|
||||||
|
chain
|
||||||
|
end
|
||||||
|
|
||||||
|
def load(path) do
|
||||||
|
Logger.debug("Loading markov chain #{path}")
|
||||||
|
with {:ok, contents} <- File.read(path),
|
||||||
|
do: {:ok, :erlang.binary_to_term(contents)}
|
||||||
|
end
|
||||||
|
|
||||||
|
def save!(chain, path) do
|
||||||
|
:ok = save(chain, path)
|
||||||
|
end
|
||||||
|
|
||||||
|
def save(chain, path) do
|
||||||
|
File.write!(path, :erlang.term_to_binary(chain))
|
||||||
|
end
|
||||||
|
|
||||||
|
def merge(lhs, rhs) do
|
||||||
|
if lhs.order != rhs.order do
|
||||||
|
raise(ArgumentError, message: "markov chain orders must match (#{lhs.order} vs #{rhs.order})")
|
||||||
|
end
|
||||||
|
|
||||||
|
merged = Map.merge(lhs.chain, rhs.chain,
|
||||||
|
fn _k, lhs, rhs -> Map.merge(lhs, rhs, fn _k, w1, w2-> w1 + w2 end) end
|
||||||
|
)
|
||||||
|
%Chain{order: lhs.order, chain: merged}
|
||||||
|
end
|
||||||
|
|
||||||
|
def merge([chain]), do: chain
|
||||||
|
|
||||||
|
def merge([chain | tail]), do: merge(tail) |> merge(chain)
|
||||||
|
|
||||||
defp do_generate(_chain, [nil | _]), do: []
|
defp do_generate(_chain, [nil | _]), do: []
|
||||||
|
|
||||||
defp do_generate(chain, key) do
|
defp do_generate(chain, key) do
|
||||||
weights = get(chain, key) || []
|
weights = chain.chain[key] || %{}
|
||||||
[next | key] = key ++ [Util.weighted_random(weights)]
|
[next | key] = key ++ [Util.weighted_random(weights)]
|
||||||
[next | do_generate(chain, key)]
|
[next | do_generate(chain, key)]
|
||||||
end
|
end
|
||||||
|
|||||||
@@ -67,4 +67,19 @@ defmodule MarkovChainTest do
|
|||||||
["foo", "bar"] => %{"baz" => 3, "qux" => 1},
|
["foo", "bar"] => %{"baz" => 3, "qux" => 1},
|
||||||
}
|
}
|
||||||
end
|
end
|
||||||
|
|
||||||
|
test "chain merge works correctly" do
|
||||||
|
chain1 = %Chain {order: 2}
|
||||||
|
|> Chain.add_weight(["foo", "bar"], "baz")
|
||||||
|
|
||||||
|
chain2 = %Chain {order: 2}
|
||||||
|
|> Chain.add_weight(["foo", "bar"], "baz")
|
||||||
|
|> Chain.add_weight(["bar", "baz"], "qux")
|
||||||
|
|
||||||
|
merged = Chain.merge(chain1, chain2)
|
||||||
|
assert merged.chain == %{
|
||||||
|
["foo", "bar"] => %{"baz" => 2},
|
||||||
|
["bar", "baz"] => %{"qux" => 1},
|
||||||
|
}
|
||||||
|
end
|
||||||
end
|
end
|
||||||
|
|||||||
Reference in New Issue
Block a user