# frozen_string_literal: true # Copyright The OpenTelemetry Authors # # SPDX-License-Identifier: Apache-2.0 require 'opentelemetry' module OpenTelemetry module Instrumentation module GraphQL module Tracers # GraphQLTrace contains the OpenTelemetry tracer implementation compatible with # the new GraphQL tracing API (>= 2.0.18) module GraphQLTrace # rubocop:disable Metrics/ModuleLength def initialize(trace_scalars: false, **_options) @trace_scalars = trace_scalars @platform_field_key_cache = Hash.new { |h, k| h[k] = platform_field_key(k) } @platform_authorized_key_cache = Hash.new { |h, k| h[k] = platform_authorized_key(k) } @platform_resolve_type_key_cache = Hash.new { |h, k| h[k] = platform_resolve_type_key(k) } super end def execute_multiplex(multiplex:, &block) tracer.in_span('graphql.execute_multiplex', &block) end def lex(query_string:, &block) tracer.in_span('graphql.lex', &block) end def parse(query_string:, &block) tracer.in_span('graphql.parse', &block) end def validate(query:, validate:, &block) tracer.in_span('graphql.validate') do |span| super.tap do |response| errors = response[:errors]&.compact&.map(&:to_h)&.to_json unless errors.nil? span.add_event( 'graphql.validation.error', attributes: { 'message' => errors } ) end end end end def analyze_multiplex(multiplex:, &block) tracer.in_span('graphql.analyze_query', &block) end def analyze_query(query:, &block) tracer.in_span('graphql.analyze_multiplex', &block) end def execute_query(query:, &block) attributes = {} attributes['graphql.operation.name'] = query.selected_operation_name if query.selected_operation_name attributes['graphql.operation.type'] = query.selected_operation.operation_type attributes['graphql.document'] = query.query_string tracer.in_span('graphql.execute_query', attributes: attributes, &block) end def execute_query_lazy(query:, multiplex:, &block) tracer.in_span('graphql.execute_query_lazy', &block) end def execute_field(field:, query:, ast_node:, arguments:, object:, &block) platform_key = platform_execute_field_key(field: field) return super unless platform_key attributes = {} attributes['graphql.field.parent'] = field.owner&.graphql_name attributes['graphql.field.name'] = field.graphql_name attributes['graphql.lazy'] = false tracer.in_span(platform_key, attributes: attributes, &block) end def execute_field_lazy(field:, query:, ast_node:, arguments:, object:, &block) platform_key = platform_execute_field_key(field: field) return super unless platform_key attributes = {} attributes['graphql.field.parent'] = field.owner&.graphql_name attributes['graphql.field.name'] = field.graphql_name attributes['graphql.lazy'] = true tracer.in_span(platform_key, attributes: attributes, &block) end def authorized(query:, type:, object:, &block) platform_key = @platform_authorized_key_cache[type] return super unless platform_key attributes = {} attributes['graphql.type.name'] = type.graphql_name attributes['graphql.lazy'] = false tracer.in_span(platform_key, attributes: attributes, &block) end def authorized_lazy(query:, type:, object:, &block) platform_key = @platform_authorized_key_cache[type] return super unless platform_key attributes = {} attributes['graphql.type.name'] = type.graphql_name attributes['graphql.lazy'] = true tracer.in_span(platform_key, attributes: attributes, &block) end def resolve_type(query:, type:, object:, &block) platform_key = @platform_resolve_type_key_cache[type] attributes = {} attributes['graphql.type.name'] = type.graphql_name attributes['graphql.lazy'] = false tracer.in_span(platform_key, attributes: attributes, &block) end def resolve_type_lazy(query:, type:, object:, &block) platform_key = @platform_resolve_type_key_cache[type] attributes = {} attributes['graphql.type.name'] = type.graphql_name attributes['graphql.lazy'] = true tracer.in_span(platform_key, attributes: attributes, &block) end private def platform_execute_field_key(field:, &block) trace_field = trace_field?(field) platform_key = @platform_field_key_cache[field] if trace_field platform_key if platform_key && trace_field end def trace_field?(field) return_type = field.type.unwrap if return_type.kind.scalar? || return_type.kind.enum? (field.trace.nil? && @trace_scalars) || field.trace else true end end def platform_field_key(field) return unless config[:enable_platform_field] if config[:legacy_platform_span_names] field.path else 'graphql.execute_field' end end def platform_authorized_key(type) return unless config[:enable_platform_authorized] if config[:legacy_platform_span_names] "#{type.graphql_name}.authorized" else 'graphql.authorized' end end def platform_resolve_type_key(type) return unless config[:enable_platform_resolve_type] if config[:legacy_platform_span_names] "#{type.graphql_name}.resolve_type" else 'graphql.resolve_type' end end def tracer GraphQL::Instrumentation.instance.tracer end def config GraphQL::Instrumentation.instance.config end end end end end end