lib/formula.ex

defmodule Formula do
  @moduledoc """
  Documentation for `Formula`.
  """
  alias Formula.Errors.InvalidFormulaError
  alias Formula.Evaluator
  alias Formula.Lexer

  @doc """
  Evaluate the formula given the data
  """
  @spec evaluate(map(), String.t()) :: term()
  def evaluate(data, formula) when is_binary(formula) do
    data
    |> Map.merge(%{"result" => "=" <> formula})
    |> evaluate()
    |> Map.get("result")
  end

  def evaluate(_data, _formula) do
    {:error, InvalidFormulaError.exception(message: "formula must be a string")}
  end

  @doc """
  Evaluate a given map that contains function
  """
  @spec evaluate(map()) :: map()
  def evaluate(data) do
    case parse(data) do
      {:error, error} -> {:error, error}
      parsed_data -> Evaluator.evaluate(parsed_data)
    end
  end

  @doc """
  Parse a given map into functions and symbols
  """
  @spec parse(map()) :: map() | {:error, term()}
  def parse(data) do
    Enum.reduce_while(data, %{}, fn kv, acc ->
      case parse_data(kv) do
        {_key, {:error, error}} -> {:halt, {:error, error}}
        {key, value} -> {:cont, Map.put(acc, key, value)}
      end
    end)
  end

  defp parse_data({key, "=" <> value}) do
    {key, parse_formula(value, key)}
  end

  defp parse_data({key, value}) when is_binary(value) do
    case Decimal.parse(value) do
      {decimal, _} -> {key, decimal}
      :error -> {key, value}
    end
  end

  defp parse_data({key, value}) when is_float(value) do
    {key, Decimal.from_float(value)}
  end

  defp parse_data({key, value}), do: {key, value}

  defp parse_formula(string, key) do
    result =
      string
      |> Lexer.tokenize!()
      |> :formula_parser.parse()

    case result do
      {:ok, function} ->
        function

      {:error, _} ->
        {:error, InvalidFormulaError.exception(message: "Please check your formula", key: key)}
    end
  end
end