lib/indexed.ex

defimpl Inspect, for: Indexed do
  def inspect(_state, _opts) do
    "#Indexed<>"
  end
end

defmodule Indexed do
  @moduledoc """
  Tools for creating an index.
  """
  alias Indexed.View
  alias __MODULE__

  # Baseline opts. Others such as :named_table may be added.
  @ets_opts [read_concurrency: true]

  defstruct entities: %{}, index_ref: nil

  @typedoc """
  * `:entities` - Map of entity name keys to `t:Indexed.Entity.t/0`
  * `:index_ref` - ETS table reference for all indexed data.
  """
  @type t :: %Indexed{
          entities: %{atom => Indexed.Entity.t()},
          index_ref: table_ref
        }

  @typedoc "A record map being cached & indexed. `:id` key is required."
  @type record :: map

  @typedoc "The value of a record's `:id` field - usually a UUID or integer."
  @type id :: any

  @typedoc "A value to prefix ETS table names."
  @type namespace :: atom

  @typedoc """
  Specifies a discrete data set of an entity, pre-partitioned into a group.
  A tuple indicates a field name and value which must match, a string
  indicates a view fingerprint, and `nil` means the full data set.
  """
  @type prefilter :: {atom, any} | String.t() | nil

  @typedoc """
  A function which takes a record and returns a value which will be evaluated
  for truthiness. If true, the value will be included in the result set.
  """
  @type filter :: (record -> any)

  @typedoc "Map held in ETS - tracks all views and their created timestamps."
  @type views :: %{String.t() => DateTime.t()}

  @typedoc "A parameter to indicate a sort field and optionally direction."
  @type order_hint ::
          atom | {direction :: :asc | :desc, field_name :: atom} | [{:asc | :desc, atom}]

  @typedoc "Reference to a table. If an atom, then a namespace is in use."
  @type table_ref :: atom | :ets.tid()

  @typedoc """
  A lookup index allows quickly finding record ids by their field values.
  These are leveraged with with the "get_by" functions.
  """
  @type lookup :: %{any => [id]}

  defdelegate prewarm(args), to: Indexed.Actions.Warm, as: :pre
  defdelegate warm(args), to: Indexed.Actions.Warm, as: :run
  defdelegate warm(index, args), to: Indexed.Actions.Warm, as: :run
  defdelegate put(index, entity_name, record), to: Indexed.Actions.Put, as: :run
  defdelegate drop(index, entity_name, id), to: Indexed.Actions.Drop, as: :run
  defdelegate create_view(index, entity_name, fp, opts), to: Indexed.Actions.CreateView, as: :run
  defdelegate destroy_view(index, entity_name, fp), to: Indexed.Actions.DestroyView, as: :run

  if Code.ensure_loaded?(Paginator) do
    defdelegate paginate(index, entity_name, params), to: Indexed.Actions.Paginate, as: :run
  end

  @doc "Get the ETS options to be used for any and all tables."
  @spec ets_opts(namespace | nil) :: keyword
  def ets_opts(nil), do: @ets_opts
  def ets_opts(_), do: [:named_table | @ets_opts]

  @doc "Get a record by id from the index."
  @spec get(t, atom, id, any) :: any
  def get(index, entity_name, id, default \\ nil) do
    case :ets.lookup(Map.fetch!(index.entities, entity_name).ref, id) do
      [{_, val}] -> val
      [] -> default
    end
  end

  @doc "Gets the list of `name` records having `value` under `field`."
  @spec get_by(t, atom, atom, any) :: [record]
  def get_by(index, name, field, value),
    do: index |> get_ids_by(name, field, value) |> Enum.map(&get(index, name, &1))

  @doc "Gets the list of `name` record ids having `value` under `field`."
  @spec get_ids_by(t, atom, atom, any) :: [id]
  def get_ids_by(index, name, field, value),
    do: index |> get_lookup(name, field) |> Map.get(value, [])

  @doc "Build an ETS table name from its namespace and entity_name."
  @spec table_name(namespace | nil, atom) :: atom
  def table_name(namespace \\ nil, entity_name \\ nil)
  def table_name(nil, nil), do: :indexed
  def table_name(nil, entity_name), do: entity_name
  def table_name(namespace, nil), do: :"#{namespace}"
  def table_name(namespace, entity_name), do: :"#{namespace}:#{entity_name}"

  @doc "Gets a list of record ids."
  @spec get_index(t, atom, prefilter, order_hint | nil) :: list
  def get_index(index, entity_name, prefilter, order_hint) when is_atom(entity_name) do
    order_hint = order_hint || default_order_hint(index, entity_name)
    get_index(index, index_key(entity_name, prefilter, order_hint), [])
  end

  @doc """
  Gets an index data structure by key. Returns default if cache doesn't exist.
  """
  @spec get_index(t, String.t(), any) :: any
  def get_index(index, index_key, default \\ nil) do
    case :ets.lookup(ref(index), index_key) do
      [{^index_key, val}] -> val
      [] -> default
    end
  end

  @doc "Gets an index data structure."
  @spec get_lookup(t, atom, atom) :: lookup | nil
  def get_lookup(index, entity_name, field_name) do
    get_index(index, lookup_key(entity_name, field_name))
  end

  @doc """
  Gets an ETS table reference or name for an entity name.
  Default with no `entity_name` is indexes table ref.
  """
  @spec ref(t, atom) :: table_ref
  def ref(index, entity_name \\ nil)
  def ref(%{index_ref: ref}, nil), do: ref

  def ref(%{entities: entities}, entity_name) do
    %{^entity_name => %{ref: ref}} = entities
    ref
  end

  @doc """
  For the given data set, get a list (sorted ascending) of unique values for
  `field_name` under `entity_name`. Returns `nil` if no data is found.
  """
  @spec get_uniques_list(t, atom, prefilter, atom) :: list
  def get_uniques_list(index, entity_name, prefilter, field_name) do
    get_index(index, uniques_list_key(entity_name, prefilter, field_name), [])
  end

  @doc """
  For the given `prefilter`, get a map where keys are unique values for
  `field_name` under `entity_name` and vals are occurrence counts. Returns
  `nil` if no data is found.
  """
  @spec get_uniques_map(t, atom, prefilter, atom) :: Indexed.UniquesBundle.counts_map()
  def get_uniques_map(index, entity_name, prefilter, field_name) do
    get_index(index, uniques_map_key(entity_name, prefilter, field_name)) || %{}
  end

  @doc """
  Get a list of all cached records of a certain type.

  `prefilter` - 2-element tuple (`t:prefilter/0`) indicating which
  sub-section of the data should be queried. Default is `nil` - no prefilter.
  """
  @spec get_records(t, atom, prefilter, order_hint | nil) :: [record]
  def get_records(index, entity_name, prefilter, order_hint \\ nil) do
    default_order_hint = fn ->
      path = [Access.key(:entities), entity_name, Access.key(:fields)]
      index |> get_in(path) |> hd() |> elem(0)
    end

    order_hint = order_hint || default_order_hint.()

    index
    |> get_index(entity_name, prefilter, order_hint)
    |> Enum.map(&get(index, entity_name, &1))
  end

  @doc "Cache key for a given entity, field and direction."
  @spec index_key(atom, prefilter, order_hint) :: String.t()
  def index_key(entity_name, prefilter, order_hint) do
    sort_str =
      order_hint
      |> Indexed.Helpers.normalize_order_hint()
      |> Enum.map_join(",", fn {d, n} -> "#{d}_#{n}" end)

    "idx_#{entity_name}#{prefilter_id(prefilter)}#{sort_str}"
  end

  @doc """
  Cache key for a lookup map of the given entity's records:
  `%{"Some Field Value" => [123, 456]}`
  """
  @spec lookup_key(atom, atom) :: String.t()
  def lookup_key(entity_name, field) do
    "lookup_#{entity_name}#{field}"
  end

  @doc """
  Cache key holding unique values & counts for a given entity and field.
  """
  @spec uniques_map_key(atom, prefilter, atom) :: String.t()
  def uniques_map_key(entity_name, prefilter, field_name) do
    "uniques_map_#{entity_name}#{prefilter_id(prefilter)}#{field_name}"
  end

  @doc "Cache key holding unique values for a given entity and field."
  @spec uniques_list_key(atom, prefilter, atom) :: String.t()
  def uniques_list_key(entity_name, prefilter, field_name) do
    "uniques_list_#{entity_name}#{prefilter_id(prefilter)}#{field_name}"
  end

  @doc "Cache key holding `t:views/0` for a certain entity."
  @spec views_key(atom) :: String.t()
  def views_key(entity_name), do: "views_#{entity_name}"

  @doc "Get a map of fingerprints to view structs (view metadata)."
  @spec get_views(t, atom) :: %{View.fingerprint() => View.t()}
  def get_views(index, entity_name) do
    get_index(index, views_key(entity_name), %{})
  end

  @doc "Get a particular view struct (view metadata) by its fingerprint."
  @spec get_view(t, atom, View.fingerprint()) :: View.t() | nil
  def get_view(index, entity_name, fingerprint) do
    with %{} = views <- get_views(index, entity_name) do
      Map.get(views, fingerprint)
    end
  end

  # Create a piece of an ETS table key to identify the set being stored.
  @spec prefilter_id(prefilter) :: String.t()
  defp prefilter_id({k, v}), do: "[#{k}=#{v}]"
  defp prefilter_id(fp) when is_binary(fp), do: "<#{fp}>"
  defp prefilter_id(_), do: "[]"

  @doc """
  Get the name of the first indexed field for an entity.
  Good order_hint default.
  """
  @spec default_order_hint(t, atom) :: atom
  def default_order_hint(index, entity_name) do
    k = &Access.key(&1)
    index |> get_in([k.(:entities), entity_name, k.(:fields)]) |> hd() |> elem(0)
  end

  @doc "Delete all ETS tables associated with the given index."
  @spec delete_tables(t) :: :ok
  def delete_tables(index) do
    :ets.delete(index.index_ref)
    Enum.each(index.entities, fn {_name, e} -> :ets.delete(e.ref) end)
  end
end