lib/tortoise311/events.ex

defmodule Tortoise311.Events do
  @moduledoc """
  A PubSub exposing various system events from a Tortoise311
  connection. This allows the user to integrate with custom metrics
  and logging solutions.

  Please read the documentation for `Tortoise311.Events.register/2` for
  information on how to subscribe to events, and
  `Tortoise311.Events.unregister/2` for how to unsubscribe.
  """

  @types [:connection, :status, :ping_response]

  @doc """
  Subscribe to messages on the client with the client id `client_id`
  of the type `type`.

  When a message of the subscribed type is dispatched it will end up
  in the mailbox of the process that placed the subscription. The
  received message will have the format:

      {{Tortoise311, client_id}, type, value}

  Making it possible to pattern match on multiple message types on
  multiple clients. The value depends on the message type.

  Possible message types are:

    - `:status` dispatched when the connection of a client changes
      status. The value will be `:up` when the client goes online, and
      `:down` when it goes offline.

    - `:ping_response` dispatched when the connection receive a
      response from a keep alive message. The value is the round trip
      time in milliseconds, and can be used to track the latency over
      time.

  Other message types exist, but unless they are mentioned in the
  possible message types above they should be considered for internal
  use only.

  It is possible to listen on all events for a given type by
  specifying `:_` as the `client_id`.
  """
  @spec register(Tortoise311.client_id(), atom()) :: {:ok, pid()} | no_return()
  def register(client_id, type) when type in @types do
    {:ok, _pid} = Registry.register(__MODULE__, type, client_id)
  end

  @doc """
  Unsubscribe from messages of `type` from `client_id`. This is the
  reverse of `Tortoise311.Events.register/2`.
  """
  @spec unregister(Tortoise311.client_id(), atom()) :: :ok | no_return()
  def unregister(client_id, type) when type in @types do
    :ok = Registry.unregister_match(__MODULE__, type, client_id)
  end

  @doc false
  @spec dispatch(Tortoise311.client_id(), type :: atom(), value :: term()) :: :ok
  def dispatch(client_id, type, value) when type in @types do
    :ok =
      Registry.dispatch(__MODULE__, type, fn subscribers ->
        for {pid, filter} <- subscribers,
            filter == client_id or filter == :_ do
          Kernel.send(pid, {{Tortoise311, client_id}, type, value})
        end
      end)
  end
end