lib/new_relic.ex

defmodule NewRelic do
  @moduledoc """
  New Relic Agent - Public API
  """

  @doc """
  Set the name of the current transaction.

  The first segment will be treated as the Transaction namespace,
  and commonly contains the name of the framework.

  ## Notes
  * At least 2 segments are required to light up the Transactions UI in APM

  In the following example, you will see `/custom/transaction/name`
  in the Transaction list.

  ```elixir
  NewRelic.set_transaction_name("/Plug/custom/transaction/name")
  ```
  """
  @spec set_transaction_name(String.t()) :: any()
  defdelegate set_transaction_name(name), to: NewRelic.Transaction.Reporter

  @doc """
  Report custom attributes on the current Transaction

  Reporting nested data structures is supported by auto-flattening them
  into a list of key-value pairs.

  ```elixir
  NewRelic.add_attributes(foo: "bar")
    # "foo" => "bar"

  NewRelic.add_attributes(map: %{foo: "bar", baz: "qux"})
    # "map.foo" => "bar"
    # "map.baz" => "qux"
    # "map.size" => 2

  NewRelic.add_attributes(list: ["a", "b", "c"])
    # "list.0" => "a"
    # "list.1" => "b"
    # "list.2" => "c"
    # "list.length" => 3
  ```

  ## Notes
  * Nested Lists and Maps are truncated at 10 items since there are a limited number
  of attributes that can be reported on Transaction events
  """
  @spec add_attributes(attributes :: Keyword.t()) :: any()
  defdelegate add_attributes(attributes), to: NewRelic.Transaction.Reporter

  @doc false
  @spec incr_attributes(attributes :: Keyword.t()) :: any()
  defdelegate incr_attributes(attributes), to: NewRelic.Transaction.Reporter

  @doc """
  Start an "Other" Transaction.

  This will begin monitoring the current process as an "Other" Transaction
  (ie: Not a "Web" Transaction).

  The first argument will be considered the "category", the second is the "name".

  The third argument is an optional map of headers that will connect this
  Transaction to an existing Distributed Trace. You can provide W3C "traceparent"
  and "tracestate" headers or another New Relic agent's "newrelic" header.

  The Transaction will end when the process exits, or when you call
  `NewRelic.stop_transaction()`

  ## Examples

  ```elixir
  NewRelic.start_transaction("GenStage", "MyConsumer/EventType")
  NewRelic.start_transaction("Task", "TaskName")
  NewRelic.start_transaction("WebSocket", "Handler", %{"newrelic" => "..."})
  ```

  > #### Warning {: .error}
  >
  > * You can't start a new transaction within an existing one. Any process
  > spawned inside a transaction belongs to that transaction.
  > * Do _not_ use this for processes that live a very long time, doing so
  > will risk increased memory growth tracking attributes in the transaction!

  ## Notes

  * Don't use this to track Web Transactions - Plug based HTTP servers
  are auto-instrumented based on `telemetry` events.
  * If multiple transactions are started in the same Process, you must
  call `NewRelic.stop_transaction/0` to mark the end of the Transaction.
  """
  @spec start_transaction(String.t(), String.t()) :: any()
  defdelegate start_transaction(category, name), to: NewRelic.OtherTransaction

  @spec start_transaction(String.t(), String.t(), headers :: map) :: any()
  defdelegate start_transaction(category, name, headers), to: NewRelic.OtherTransaction

  @doc """
  Stop an "Other" Transaction.

  If multiple Transactions are started in the same Process, you must
  call `NewRelic.stop_transaction/0` to mark the end of the Transaction.
  """
  @spec stop_transaction() :: any()
  defdelegate stop_transaction(), to: NewRelic.OtherTransaction

  @doc """
  Record an "Other" Transaction within the given block. The return value of
  the block is returned.

  See `start_transaction/2` and `stop_transaction/0` for more details about
  Transactions.

  ## Example

  ```elixir
  defmodule Worker do
    use NewRelic.Tracer

    def process_messages do
      NewRelic.other_transaction("Worker", "ProcessMessages") do
        # ...
      end
    end
  end
  ```
  """
  defmacro other_transaction(category, name, do: block) do
    quote do
      NewRelic.start_transaction(unquote(category), unquote(name))
      res = unquote(block)
      NewRelic.stop_transaction()
      res
    end
  end

  defmacro other_transaction(category, name, headers, do: block) do
    quote do
      NewRelic.start_transaction(unquote(category), unquote(name), unquote(headers))
      res = unquote(block)
      NewRelic.stop_transaction()
      res
    end
  end

  @doc """
  Call within a transaction to prevent it from reporting.

  ## Example

  ```elixir
  def index(conn, _) do
    NewRelic.ignore_transaction()
    send_resp(conn, 200, "Health check OK")
  end
  ```
  """
  @spec ignore_transaction() :: any()
  defdelegate ignore_transaction(), to: NewRelic.Transaction.Reporter

  @doc """
  Call to exclude the current process from being part of the Transaction.

  ## Example:

  ```elixir
  Task.async(fn ->
    NewRelic.exclude_from_transaction()
    Work.wont_be_included()
  end)
  ```
  """
  @spec exclude_from_transaction() :: any()
  defdelegate exclude_from_transaction(), to: NewRelic.Transaction.Reporter

  @doc """
  Advanced:
  Return a Transaction reference that can be used to manually connect a
  process to a Transaction with `NewRelic.connect_to_transaction/1`
  """
  @type tx_ref :: any()
  @spec get_transaction() :: tx_ref
  defdelegate get_transaction(), to: NewRelic.Transaction.Reporter

  @doc """
  Advanced:
  Call to manually connect the current process to a Transaction. Pass in a reference
  returned by `NewRelic.get_transaction/0`

  Only use this when there is no auto-discoverable connection (ex: the process was
  spawned without links or the logic is within a message handling callback).

  This connection will persist until the process exits or
  `NewRelic.disconnect_from_transaction/0` is called.

  ## Example:

  ```elixir
  tx = NewRelic.get_transaction()

  spawn(fn ->
    NewRelic.connect_to_transaction(tx)
    # ...
  end)
  ```
  """
  @spec connect_to_transaction(tx_ref) :: any()
  defdelegate connect_to_transaction(ref), to: NewRelic.Transaction.Reporter

  @doc """
  Advanced:
  Call to manually disconnect the current process from the current Transaction.
  """
  @spec disconnect_from_transaction() :: any()
  defdelegate disconnect_from_transaction(), to: NewRelic.Transaction.Reporter

  @doc """
  Store information about the type of work the current span is doing.

  ## Examples

  ```elixir
  NewRelic.set_span(:generic, some: "attribute")

  NewRelic.set_span(:http, url: "https://elixir-lang.org", method: "GET", component: "HttpClient")

  NewRelic.set_span(:datastore, statement: statement, instance: instance, address: address,
  hostname: hostname, component: component)
  ```
  """
  @spec set_span(:generic, attributes :: Keyword.t()) :: any()
  @spec set_span(:http, url: String.t(), method: String.t(), component: String.t()) :: any()
  @spec set_span(:datastore,
          statement: String.t(),
          instance: String.t(),
          address: String.t(),
          hostname: String.t(),
          component: String.t()
        ) :: any()
  defdelegate set_span(type, attributes), to: NewRelic.DistributedTrace

  @doc """
  Add additional attributes to the current Span (not the current Transaction).

  Useful for reporting additional information about work being done in, for example,
  a function being traced with `@trace`

  ## Example

  ```elixir
  NewRelic.add_span_attributes(some: "attribute")
  ```
  """
  @spec add_span_attributes(attributes :: Keyword.t()) :: any()
  defdelegate add_span_attributes(attributes), to: NewRelic.DistributedTrace

  @doc """
  You must manually instrument outgoing HTTP calls to connect them to a Distributed Trace.

  The agent will automatically read HTTP request headers and detect if the request is a part
  of an incoming Distributed Trace, but outgoing requests need an extra header:

  ```elixir
  Req.get(url, headers: ["x-api-key": "secret"] ++ NewRelic.distributed_trace_headers(:http))
  ```

  ## Notes

  * Call `distributed_trace_headers` immediately before making the
  request since calling the function marks the "start" time of the request.
  """
  @spec distributed_trace_headers(:http) :: [{key :: String.t(), value :: String.t()}]
  @spec distributed_trace_headers(:other) :: map()
  defdelegate distributed_trace_headers(type), to: NewRelic.DistributedTrace

  @type name :: String.t() | {primary_name :: String.t(), secondary_name :: String.t()}

  @doc """
  Record a "Span" within the given block. The return value of the block is returned.

  ```elixir
  NewRelic.span("do.some_work", user_id: "abc123") do
    # do some work
  end
  ```

  Note: You can also use `@trace` annotations to instrument functions without modifying code.
  """
  @spec span(name :: name, attributes :: Keyword.t()) :: term()
  defmacro span(name, attributes \\ [], do: block) do
    quote do
      id = make_ref()
      NewRelic.Tracer.Direct.start_span(id, unquote(name), attributes: unquote(attributes))
      res = unquote(block)
      NewRelic.Tracer.Direct.stop_span(id)
      res
    end
  end

  @doc """
  See: `NewRelic.distributed_trace_headers/1`
  """
  @deprecated "Use distributed_trace_headers instead"
  defdelegate create_distributed_trace_payload(type),
    to: NewRelic.DistributedTrace,
    as: :distributed_trace_headers

  @doc """
  To get detailed information about a particular process, you can install a Process sampler.
  You must tell the Agent about your process from within the process.

  For a `GenServer`, this function call should be made in the `init` function:

  ```elixir
  defmodule ImportantProcess do
    use GenServer
    def init(:ok) do
      NewRelic.sample_process()
      {:ok, %{}}
    end
  end
  ```

  Once installed, the agent will report `ElixirSample` events with:

  * `category = "Process"`
  * `message_queue_length`
  * `reductions`
  * `memory_kb`
  """
  @spec sample_process() :: any()
  defdelegate sample_process, to: NewRelic.Sampler.Process

  @doc """
  Report a Custom event to NRDB.

  ## Example

  ```elixir
  NewRelic.report_custom_event("EventType", %{"foo" => "bar"})
  ```
  """
  @spec report_custom_event(type :: String.t(), event :: map()) :: any()
  defdelegate report_custom_event(type, event),
    to: NewRelic.Harvest.Collector.CustomEvent.Harvester

  @doc """
  Report a Dimensional Metric.

  Valid types: `:count`, `:gauge`, and `:summary`.

  ## Example

  ```elixir
  NewRelic.report_dimensional_metric(:count, "my_metric_name", 1, %{some: "attributes"})
  ```
  """
  @spec report_dimensional_metric(
          type :: :count | :gauge | :summary,
          name :: String.t(),
          value :: number,
          attributes :: map()
        ) :: any()
  defdelegate report_dimensional_metric(type, name, value, attributes \\ %{}),
    to: NewRelic.Harvest.TelemetrySdk.DimensionalMetrics.Harvester

  @doc """
  Report a Custom metric.

  ## Example

  ```elixir
  NewRelic.report_custom_metric("My/Metric", 123)
  ```
  """
  @spec report_custom_event(name :: String.t(), value :: number()) :: any()
  defdelegate report_custom_metric(name, value),
    to: NewRelic.Harvest.Collector.Metric.Harvester

  @doc """
  Increment a Custom metric.

  ## Example

  ```elixir
  NewRelic.increment_custom_metric("My/Metric")
  ```
  """
  @spec increment_custom_metric(name :: String.t(), count :: integer()) :: any()
  defdelegate increment_custom_metric(name, count \\ 1),
    to: NewRelic.Harvest.Collector.Metric.Harvester

  @doc """
  Report an Exception inside a Transaction.

  This should only be used when you `rescue` an exception inside a Transaction,
  but still want to report it. All un-rescued exceptions are already reported as errors.

  ## Example

  ```elixir
  try do
    raise RuntimeError
  rescue
    exception -> NewRelic.notice_error(exception, __STACKTRACE__)
  end
  ```
  """
  @spec notice_error(Exception.t(), Exception.stacktrace()) :: any()
  defdelegate notice_error(exception, stacktrace), to: NewRelic.Transaction.Reporter

  @doc false
  defdelegate enable_erlang_trace, to: NewRelic.Transaction.ErlangTraceManager

  @doc false
  defdelegate disable_erlang_trace, to: NewRelic.Transaction.ErlangTraceManager

  @doc false
  defdelegate report_aggregate(meta, values), to: NewRelic.Aggregate.Reporter

  @doc false
  defdelegate report_sample(category, values), to: NewRelic.Sampler.Reporter

  @doc false
  defdelegate report_span(span), to: NewRelic.Span.Reporter

  @doc false
  defdelegate report_metric(identifier, values), to: NewRelic.Harvest.Collector.Metric.Harvester

  @doc false
  defdelegate log(level, message), to: NewRelic.Logger

  @doc false
  defdelegate manual_shutdown(), to: NewRelic.Harvest.Supervisor
end