defmodule NewRelic.Telemetry.Phoenix do
use GenServer
@moduledoc """
Provides `Phoenix` instrumentation via `telemetry`.
This instrumentation adds extra Phoenix specific instrumentation
on top of the base `NewRelic.Telemetry.Plug` instrumentation.
"""
def start_link(_) do
config = %{
enabled?: NewRelic.Config.feature?(:phoenix_instrumentation),
handler_id: {:new_relic, :phoenix}
}
GenServer.start_link(__MODULE__, config, name: __MODULE__)
end
@phoenix_router_start [:phoenix, :router_dispatch, :start]
@phoenix_error [:phoenix, :error_rendered]
@phoenix_events [
@phoenix_router_start,
@phoenix_error
]
@doc false
def init(%{enabled?: false}), do: :ignore
def init(%{enabled?: true} = config) do
:telemetry.attach_many(
config.handler_id,
@phoenix_events,
&__MODULE__.handle_event/4,
config
)
{:ok, config}
end
@doc false
def terminate(_reason, %{handler_id: handler_id}) do
:telemetry.detach(handler_id)
end
def handle_event(
@phoenix_router_start,
_measurements,
%{conn: conn} = meta,
_config
) do
[
phoenix_name: phoenix_name(meta),
"phoenix.endpoint": conn.private[:phoenix_endpoint] |> inspect(),
"phoenix.router": conn.private[:phoenix_router] |> inspect(),
"phoenix.controller": meta.plug |> inspect(),
"phoenix.action":
case meta.plug_opts do
action when is_atom(action) -> to_string(action)
action -> inspect(action)
end,
"phoenix.format": conn.private[:phoenix_format],
"phoenix.template": conn.private[:phoenix_template],
"phoenix.view": conn.private[:phoenix_view] |> inspect()
]
|> NewRelic.add_attributes()
end
def handle_event(
@phoenix_error,
_measurements,
%{conn: conn, status: 404} = meta,
_config
) do
[
phoenix_name: phoenix_name(meta),
"phoenix.endpoint": conn.private[:phoenix_endpoint] |> inspect(),
"phoenix.router": conn.private[:phoenix_router] |> inspect()
]
|> NewRelic.add_attributes()
end
def handle_event(_event, _measurements, _meta, _config) do
:ignore
end
defp phoenix_name(%{phoenix_live_view: {module, action, _, _}}) when is_atom(action) do
"/Phoenix/#{inspect(module)}/#{action}"
end
defp phoenix_name(%{plug: controller, plug_opts: action}) when is_atom(action) do
"/Phoenix/#{inspect(controller)}/#{action}"
end
defp phoenix_name(%{conn: %{private: %{phoenix_endpoint: phoenix_endpoint}} = conn}) do
"/Phoenix/#{inspect(phoenix_endpoint)}/#{conn.method}"
end
defp phoenix_name(%{conn: conn}) do
"/Phoenix/#{conn.method}"
end
end