1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116
|
# frozen_string_literal: true
module GraphQL
module Tracing
module SentryTrace
include PlatformTrace
# @param set_transaction_name [Boolean] If true, the GraphQL operation name will be used as the transaction name.
# This is not advised if you run more than one query per HTTP request, for example, with `graphql-client` or multiplexing.
# It can also be specified per-query with `context[:set_sentry_transaction_name]`.
def initialize(set_transaction_name: false, **_rest)
@set_transaction_name = set_transaction_name
super
end
def execute_query(**data)
set_this_txn_name = data[:query].context[:set_sentry_transaction_name]
if set_this_txn_name == true || (set_this_txn_name.nil? && @set_transaction_name)
Sentry.configure_scope do |scope|
scope.set_transaction_name(transaction_name(data[:query]))
end
end
instrument_execution("graphql.execute", "execute_query", data) { super }
end
# rubocop:disable Development/NoEvalCop This eval takes static inputs at load-time
{
"lex" => "graphql.lex",
"parse" => "graphql.parse",
"validate" => "graphql.validate",
"analyze_query" => "graphql.analyze",
"analyze_multiplex" => "graphql.analyze_multiplex",
"execute_multiplex" => "graphql.execute_multiplex",
"execute_query_lazy" => "graphql.execute"
}.each do |trace_method, platform_key|
module_eval <<-RUBY, __FILE__, __LINE__
def #{trace_method}(**data)
instrument_execution("#{platform_key}", "#{trace_method}", data) { super }
end
RUBY
end
# rubocop:enable Development/NoEvalCop
def platform_execute_field(platform_key, &block)
instrument_execution(platform_key, "execute_field", &block)
end
def platform_execute_field_lazy(platform_key, &block)
instrument_execution(platform_key, "execute_field_lazy", &block)
end
def platform_authorized(platform_key, &block)
instrument_execution(platform_key, "authorized", &block)
end
def platform_authorized_lazy(platform_key, &block)
instrument_execution(platform_key, "authorized_lazy", &block)
end
def platform_resolve_type(platform_key, &block)
instrument_execution(platform_key, "resolve_type", &block)
end
def platform_resolve_type_lazy(platform_key, &block)
instrument_execution(platform_key, "resolve_type_lazy", &block)
end
def platform_field_key(field)
"graphql.field.#{field.path}"
end
def platform_authorized_key(type)
"graphql.authorized.#{type.graphql_name}"
end
def platform_resolve_type_key(type)
"graphql.resolve_type.#{type.graphql_name}"
end
private
def instrument_execution(platform_key, trace_method, data=nil, &block)
return yield unless Sentry.initialized?
Sentry.with_child_span(op: platform_key, start_timestamp: Sentry.utc_now.to_f) do |span|
result = yield
return result unless span
span.finish
if trace_method == "execute_multiplex" && data.key?(:multiplex)
operation_names = data[:multiplex].queries.map{|q| operation_name(q) }
span.set_description(operation_names.join(", "))
elsif trace_method == "execute_query" && data.key?(:query)
span.set_description(operation_name(data[:query]))
span.set_data('graphql.document', data[:query].query_string)
span.set_data('graphql.operation.name', data[:query].selected_operation_name) if data[:query].selected_operation_name
span.set_data('graphql.operation.type', data[:query].selected_operation.operation_type)
end
result
end
end
def operation_name(query)
selected_op = query.selected_operation
if selected_op
[selected_op.operation_type, selected_op.name].compact.join(' ')
else
'GraphQL Operation'
end
end
end
end
end
|