lib/enum.ex

defmodule Moar.Enum do
  # @related [test](/test/enum_test.exs)

  @moduledoc "Enum-related functions."

  @doc "Like `Enum.at/3` but raises if `index` is out of bounds."
  @spec at!(Enum.t(), integer() | [integer()], any()) :: any()
  def at!(enum, index, default \\ nil) when is_integer(index) do
    if length(enum) < index + 1,
      do: raise("Out of range: index #{index} of enum with length #{length(enum)}: #{inspect(enum)}"),
      else: Enum.at(enum, index, default)
  end

  @doc "Removes nil elements from `enum`."
  @spec compact(Enum.t()) :: Enum.t()
  def compact(enum),
    do: enum |> Enum.reject(&is_nil(&1))

  @doc """
  Returns the indices of `elements` in `enum`, using `fun` for comparisons (defaulting to `Kernel.==/2`)

  ```elixir
  iex> Moar.Enum.find_indices(~w[apple banana cherry], ~w[cherry apple])
  [2, 0]

  iex> Moar.Enum.find_indices(~w[apple banana cherry], ~w[CHERRY APPLE], fn a, b ->
  ...>   String.downcase(a) == String.downcase(b)
  ...> end)
  [2, 0]
  ```
  """
  @spec find_indices(Enum.t(), [any()], (any(), any() -> boolean())) :: [integer()]
  def find_indices(enum, elements, fun \\ &Kernel.==/2),
    do: Enum.map(elements, &Enum.find_index(enum, fn element -> fun.(element, &1) end))

  @doc """
  Returns the indices of `elements` in `enum`, raising if any member of `elements` is not found.

  ```elixir
  iex> Moar.Enum.find_indices!(~w[apple banana cherry], ~w[cherry apple])
  [2, 0]

  iex> Moar.Enum.find_indices!(~w[apple banana], ~w[cherry apple])
  ** (RuntimeError) Element "cherry" not present in:
  ["apple", "banana"]
  ```
  """
  @spec find_indices!(Enum.t(), [any()], (any(), any() -> boolean())) :: [integer()]
  def find_indices!(enum, elements, fun \\ &Kernel.==/2) do
    Enum.map(elements, fn element ->
      Enum.find_index(enum, fn member ->
        fun.(member, element)
      end) || raise("Element #{inspect(element)} not present in:\n#{inspect(enum)}")
    end)
  end

  @doc "Returns the first item of `enum`, or raises if it is empty."
  @spec first!(Enum.t()) :: any()
  def first!(enum),
    do: Enum.at(enum, 0) || raise("Expected enumerable to have at least one item")

  @doc """
  Converts an enum into a map of maps indexed by the return value of `index_fun`.
  See also the similar map-specific `Moar.Map.index_by/2`.

  ```elixir
  iex> Moar.Enum.index_by([%{name: "Alice", tid: "alice"}, %{name: "Billy", tid: "billy"}], & &1.tid)
  %{"alice" => %{name: "Alice", tid: "alice"}, "billy" => %{name: "Billy", tid: "billy"}}
  ```
  """
  @spec index_by(Enum.t(), (any() -> any())) :: map()
  def index_by(enum, index_fun) when is_function(index_fun),
    do: Enum.reduce(enum, %{}, fn value, acc -> Moar.Map.put_new!(acc, index_fun.(value), value) end)

  @doc "Like `Enum.into` but accepts `nil` as the first argument"
  @spec into!(nil | Enum.t(), Enum.t()) :: Enum.t()
  def into!(nil, enumerable), do: enumerable
  def into!(%_{} = struct, enumerable), do: struct |> Map.from_struct() |> into!(enumerable)
  def into!(other, enumerable), do: Enum.into(other, enumerable)

  @doc """
  Deprecated in favor of `map_or_keyword?/1`.
  """
  @spec is_map_or_keyword(any()) :: boolean()
  @deprecated "Use map_or_keyword?/1"
  def is_map_or_keyword(value),
    do: map_or_keyword?(value)

  @doc """
  Returns true if the value is a map or a keyword list. This uses standard Elixir functions for determining
  if a term is a map or a keyword, and therefore counts an empty list as a keyword list. See also
  `map_or_nonempty_keyword?/1`.

  This cannot be used as a guard because it uses `Keyword.keyword?` under the hood. Also, because of that,
  it might scan an entire list to see if it's a keyword list, so it might be expensive.
  """
  @spec map_or_keyword?(any()) :: boolean()
  def map_or_keyword?(value),
    do: is_map(value) || (is_list(value) && Keyword.keyword?(value))

  @doc """
  Deprecated in favor of `map_or_nonempty_keyword?/1`.
  """
  @spec is_map_or_nonempty_keyword(any()) :: boolean()
  @deprecated "Use map_or_nonempty_keyword?/1"
  def is_map_or_nonempty_keyword(value),
    do: is_map(value) || (is_list(value) && !Enum.empty?(value) && Keyword.keyword?(value))

  @doc """
  Like `map_or_keyword?/1` but returns false if the term is an empty list.
  """
  @spec map_or_nonempty_keyword?(any()) :: boolean()
  def map_or_nonempty_keyword?(value),
    do: is_map(value) || (is_list(value) && !Enum.empty?(value) && Keyword.keyword?(value))

  @doc "Sorts `enum` case-insensitively. Uses `Enum.sort_by/3` under the hood."
  @spec isort(Enum.t()) :: Enum.t()
  def isort(enum),
    do: enum |> isort_by(&to_string/1)

  @doc "Sorts `enum` case-insensitively by `mapper` function. Uses `Enum.sort_by/3` under the hood."
  @spec isort_by(Enum.t(), (any() -> any())) :: Enum.t()
  def isort_by(enum, mapper),
    do: enum |> Enum.sort_by(&(mapper.(&1) |> String.downcase()))

  @doc """
  Converts a list of lists to a list of maps with the given keys. The keys can be a list, or can be `:first_list`
  which uses the first list in `list_of_lists` as the keys and the remaining lists in `list_of_lists` as the values.

  ```elixir
  iex> Moar.Enum.lists_to_maps([[1, 2], [3, 4]], ["a", "b"])
  [%{"a" => 1, "b" => 2}, %{"a" => 3, "b" => 4}]

  iex> csv = [["a", "b"], [1, 2], [3, 4]]
  iex> [headers | rows] = csv
  iex> Moar.Enum.lists_to_maps(rows, headers)
  [%{"a" => 1, "b" => 2}, %{"a" => 3, "b" => 4}]

  iex> csv = [["a", "b"], [1, 2], [3, 4]]
  iex> Moar.Enum.lists_to_maps(csv, :first_list)
  [%{"a" => 1, "b" => 2}, %{"a" => 3, "b" => 4}]
  ```
  """
  @spec lists_to_maps(list(list(any())), list(any()) | :first_list) :: list(map())
  def lists_to_maps([first_list | remaining_lists], :first_list),
    do: lists_to_maps(remaining_lists, first_list)

  def lists_to_maps(list_of_lists, keys),
    do: Enum.map(list_of_lists, &Map.new(Enum.zip([keys, &1])))

  @doc """
  Returns a list of elements at the given indices, in the given order. If `:all` is given instead of a list of indices,
  the entire enum is returned.

  ```elixir
  iex> Moar.Enum.take_at(["A", "B", "C"], [0, 2])
  ["A", "C"]

  iex> Moar.Enum.take_at(["A", "B", "C"], [2, 0])
  ["C", "A"]

  iex> Moar.Enum.take_at(["A", "B", "C"], :all)
  ["A", "B", "C"]
  ```
  """
  @spec take_at(Enum.t(), integer() | [integer()] | :all) :: any()
  def take_at(enum, :all), do: enum
  def take_at(enum, list) when is_list(list), do: Enum.map(list, &Enum.at(enum, &1))

  @doc """
  Returns `:test_id` fields from `enumerable`.

  This unusual function exists because the authors of Moar use test_ids extensively in tests.
  See also `tids/2`.
  """
  @type test_ids_opts() :: {:sorted, boolean()}
  @spec test_ids(Enum.t(), [test_ids_opts()]) :: list()
  def test_ids(enumerable, opts \\ []) do
    test_ids = enumerable |> Enum.map(& &1.test_id)

    if Keyword.get(opts, :sorted),
      do: test_ids |> Enum.sort(),
      else: test_ids
  end

  @doc """
  Returns `:tid` fields from `enumerable`.

  This unusual function exists because the authors of Moar use tids (test IDs) extensively in tests.
  See also `test_ids/2`.
  """
  @type tids_opts() :: {:sorted, boolean()}
  @spec tids(Enum.t(), [tids_opts()]) :: list()
  def tids(enumerable, opts \\ []) do
    tids = enumerable |> Enum.map(& &1.tid)

    if Keyword.get(opts, :sorted),
      do: tids |> Enum.sort(),
      else: tids
  end
end