Custom Instrumentation

Learn how to capture performance data on any action in your app.

To capture transactions and spans customized to your organization's needs, you must first set up tracing.

To instrument certain regions of your code, you can create transactions to capture them.

The following example creates a transaction for a scope that contains an expensive operation (for example, process_item), and sends the result to Sentry:

Copied
# start a transaction
transaction = Sentry.start_transaction(op: "process_item")

# set the transaction on the scope so children span are attached to this transaction
Sentry.get_current_scope.set_span(transaction)

# perform the operation
process_item(args)


# finish the transaction, which will send it to Sentry automatically
transaction.finish

The next example contains the implementation of the hypothetical process_item function called from the code snippet in the previous section. Our SDK can determine if there is currently an open transaction and add all newly created spans as child operations to that transaction. You can choose the values of op and description.

Copied
class OrdersController < ApplicationController
  def create
    order = Order.new

    Sentry.with_child_span(op: :process_items, description: 'processing items') do |span|
      span.set_data(:key, 'value')

      order.process_items(params)
    end
  end # the child span ends with the block
end

Your new span will be nested under whichever span is currently running on the scope, otherwise it will be at the root of the transaction event.

Alternatively, you can manually grab the current transaction and use its with_child_span method to always create a top-level span.

Copied
class OrdersController < ApplicationController
  def create
    order = Order.new
    transaction = Sentry.get_current_scope.get_transaction

    transaction.with_child_span(op: :process_items, description: 'processing items') do |span|
      span.set_data(:key, 'value')
      order.process_items(params)
    end # the child span ends with the block
  end
end

Keep in mind that there may not be an active transaction, in which case get_transaction returns nil. This case needs to be handled manually and is missing from this example.

In cases where you want to attach Spans to an already ongoing Transaction you can use Sentry.get_current_scope.get_transaction. This property will return a Transaction in case there is a running Transaction otherwise it returns nil.

Copied
transaction = Sentry.get_current_scope.get_transaction || Sentry.start_transaction(name: 'task')

span = transaction.start_child(op: 'operation')
# perform the operation
span.finish

Started spans are stored in the scope, and can be fetched off the scope:

Copied
span = Sentry.get_current_scope.get_span

You can add data attributes to your transactions. This data is visible in the trace explorer in Sentry. Data attributes can be strings, numbers or booleans, as well as (non-mixed) arrays of these types:

Copied
Sentry.start_transaction(name: "my-transaction") do |transaction|
  transaction.set_data("my-data-attribute-1", "value1")
  transaction.set_data("my-data-attribute-2", 42)
  transaction.set_data("my-data-attribute-3", true)

  transaction.set_data("my-data-attribute-4", ["value1", "value2", "value3"])
  transaction.set_data("my-data-attribute-5", [42, 43, 44])
  transaction.set_data("my-data-attribute-6", [true, false, true])
end

You can add data attributes to your spans. This data is visible in the trace explorer in Sentry. Data attributes can be strings, numbers or booleans, as well as (non-mixed) arrays of these types:

Copied
Sentry.with_child_span(op: "my-span") do |span|
  span.set_data("my-data-attribute-1", "value1")
  span.set_data("my-data-attribute-2", 42)
  span.set_data("my-data-attribute-3", true)

  span.set_data("my-data-attribute-4", ["value1", "value2", "value3"])
  span.set_data("my-data-attribute-5", [42, 43, 44])
  span.set_data("my-data-attribute-6", [true, false, true])
end
Help improve this content
Our documentation is open source and available on GitHub. Your contributions are welcome, whether fixing a typo (drat!) or suggesting an update ("yeah, this would be better").