module GraphQL module StaticValidation # - Ride along with `GraphQL::Language::Visitor` # - Track type info, expose it to validators class TypeStack # These are jumping-off points for infering types down the tree TYPE_INFERRENCE_ROOTS = [ GraphQL::Language::Nodes::OperationDefinition, GraphQL::Language::Nodes::FragmentDefinition, ] # @return [GraphQL::Schema] the schema whose types are present in this document attr_reader :schema # When it enters an object (starting with query or mutation root), it's pushed on this stack. # When it exits, it's popped off. # @return [Array] attr_reader :object_types # When it enters a field, it's pushed on this stack (useful for nested fields, args). # When it exits, it's popped off. # @return [Array] fields which have been entered attr_reader :field_definitions # Directives are pushed on, then popped off while traversing the tree # @return [Array] directives which have been entered attr_reader :directive_definitions # @return [Array] arguments which have been entered attr_reader :argument_definitions # @param schema [GraphQL::Schema] the schema whose types to use when climbing this document # @param visitor [GraphQL::Language::Visitor] a visitor to follow & watch the types def initialize(schema, visitor) @schema = schema @object_types = [] @field_definitions = [] @directive_definitions = [] @argument_definitions = [] visitor.enter << -> (node, parent) { PUSH_STRATEGIES[node.class].push(self, node) } visitor.leave << -> (node, parent) { PUSH_STRATEGIES[node.class].pop(self, node) } end private # Look up strategies by name and use singleton instance to push and pop PUSH_STRATEGIES = Hash.new { |hash, key| hash[key] = get_strategy_for_node_class(key) } def self.get_strategy_for_node_class(node_class) node_class_name = node_class.name.split("::").last strategy_key = "#{node_class_name}Strategy" const_defined?(strategy_key) ? const_get(strategy_key).new : NullStrategy.new end class FragmentWithTypeStrategy def push(stack, node) object_type = if node.type stack.schema.types.fetch(node.type, nil) else stack.object_types.last end if !object_type.nil? object_type = object_type.unwrap end stack.object_types.push(object_type) end def pop(stack, node) stack.object_types.pop end end class FragmentDefinitionStrategy < FragmentWithTypeStrategy; end class InlineFragmentStrategy < FragmentWithTypeStrategy; end class OperationDefinitionStrategy def push(stack, node) # eg, QueryType, MutationType object_type = stack.schema.root_type_for_operation(node.operation_type) stack.object_types.push(object_type) end def pop(stack, node) stack.object_types.pop end end class FieldStrategy def push(stack, node) parent_type = stack.object_types.last parent_type = parent_type.unwrap if parent_type.kind.fields? field_class = stack.schema.get_field(parent_type, node.name) stack.field_definitions.push(field_class) if !field_class.nil? next_object_type = field_class.type stack.object_types.push(next_object_type) else stack.object_types.push(nil) end else stack.field_definitions.push(nil) stack.object_types.push(parent_type) end end def pop(stack, node) stack.field_definitions.pop stack.object_types.pop end end class DirectiveStrategy def push(stack, node) directive_defn = stack.schema.directives[node.name] stack.directive_definitions.push(directive_defn) end def pop(stack, node) stack.directive_definitions.pop end end class ArgumentStrategy # Push `argument_defn` onto the stack. # It's possible that `argument_defn` will be nil. # Push it anyways so `pop` has something to pop. def push(stack, node) if stack.argument_definitions.last arg_type = stack.argument_definitions.last.type.unwrap if arg_type.kind.input_object? argument_defn = arg_type.input_fields[node.name] else argument_defn = nil end elsif stack.directive_definitions.last argument_defn = stack.directive_definitions.last.arguments[node.name] elsif stack.field_definitions.last argument_defn = stack.field_definitions.last.arguments[node.name] else argument_defn = nil end stack.argument_definitions.push(argument_defn) end def pop(stack, node) stack.argument_definitions.pop end end # A no-op strategy (don't handle this node) class NullStrategy def self.new; self; end def self.push(stack, node); end def self.pop(stack, node); end end end end end