defmodule Modbux.Model do
@moduledoc """
Model helper, functions to write and read the Slave/Server DB.
"""
require Logger
def apply(state, {:rc, slave, address, count}) do
reads(state, {slave, :c, address, count})
end
def apply(state, {:ri, slave, address, count}) do
reads(state, {slave, :i, address, count})
end
def apply(state, {:rhr, slave, address, count}) do
reads(state, {slave, :hr, address, count})
end
def apply(state, {:rir, slave, address, count}) do
reads(state, {slave, :ir, address, count})
end
def apply(state, {:fc, slave, address, value}) when is_integer(value) do
write(state, {slave, :c, address, value})
end
def apply(state, {:fc, slave, address, values}) when is_list(values) do
writes(state, {slave, :c, address, values})
end
def apply(state, {:phr, slave, address, value}) when is_integer(value) do
write(state, {slave, :hr, address, value})
end
def apply(state, {:phr, slave, address, values}) when is_list(values) do
writes(state, {slave, :hr, address, values})
end
# Error and exception clause
def apply(state, _cmd) do
# do nothing
{nil, state}
end
@spec reads(map, {any, any, any, any}) :: {nil | {:error, :eaddr} | {:ok, [any]}, map}
def reads(state, {slave, type, address, count}) do
# checks the slave_ids
if Map.has_key?(state, slave) do
try do
map = Map.fetch!(state, slave)
list =
for point <- address..(address + count - 1) do
Map.fetch!(map, {type, point})
end
{{:ok, list}, state}
rescue
_error ->
{{:error, :eaddr}, state}
end
else
# is a different slave id, do nothing.
{nil, state}
end
end
@spec write(map, {any, any, any, any}) :: {nil | {:error, :eaddr} | {:ok, nil}, map}
def write(state, {slave, type, address, value}) do
# checks the slave_ids
if Map.has_key?(state, slave) do
try do
cmap = Map.fetch!(state, slave)
nmap = Map.replace!(cmap, {type, address}, value)
{{:ok, nil}, Map.put(state, slave, nmap)}
rescue
_error ->
{{:error, :eaddr}, state}
end
else
# is a different slave id, do nothing.
{nil, state}
end
end
def writes(state, {slave, type, address, values}) do
# checks the slave_ids
if Map.has_key?(state, slave) do
try do
cmap = Map.fetch!(state, slave)
final = address + Enum.count(values)
{^final, nmap} =
Enum.reduce(values, {address, cmap}, fn value, {i, map} ->
{i + 1, Map.replace!(map, {type, i}, value)}
end)
{{:ok, nil}, Map.put(state, slave, nmap)}
rescue
_error ->
{{:error, :eaddr}, state}
end
else
# is a different slave id, do nothing.
{nil, state}
end
end
end