graphql 1.12.23 → 1.13.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Potentially problematic release.
This version of graphql might be problematic. Click here for more details.
- checksums.yaml +4 -4
- data/lib/generators/graphql/mutation_generator.rb +1 -1
- data/lib/generators/graphql/type_generator.rb +0 -1
- data/lib/graphql/analysis/ast/field_usage.rb +4 -8
- data/lib/graphql/analysis/ast/query_complexity.rb +10 -14
- data/lib/graphql/analysis/ast/visitor.rb +4 -4
- data/lib/graphql/backtrace/table.rb +1 -1
- data/lib/graphql/dataloader.rb +55 -22
- data/lib/graphql/directive.rb +0 -4
- data/lib/graphql/enum_type.rb +5 -1
- data/lib/graphql/execution/errors.rb +1 -0
- data/lib/graphql/execution/interpreter/arguments.rb +1 -1
- data/lib/graphql/execution/interpreter/arguments_cache.rb +2 -2
- data/lib/graphql/execution/interpreter/runtime.rb +20 -12
- data/lib/graphql/execution/lookahead.rb +2 -2
- data/lib/graphql/execution/multiplex.rb +1 -1
- data/lib/graphql/introspection/directive_type.rb +1 -1
- data/lib/graphql/introspection/entry_points.rb +2 -2
- data/lib/graphql/introspection/enum_value_type.rb +2 -2
- data/lib/graphql/introspection/field_type.rb +2 -2
- data/lib/graphql/introspection/input_value_type.rb +4 -4
- data/lib/graphql/introspection/schema_type.rb +2 -2
- data/lib/graphql/introspection/type_type.rb +10 -10
- data/lib/graphql/language/block_string.rb +0 -4
- data/lib/graphql/language/document_from_schema_definition.rb +4 -2
- data/lib/graphql/language/lexer.rb +0 -3
- data/lib/graphql/language/lexer.rl +0 -4
- data/lib/graphql/language/nodes.rb +3 -11
- data/lib/graphql/language/parser.rb +442 -434
- data/lib/graphql/language/parser.y +5 -4
- data/lib/graphql/language/printer.rb +6 -1
- data/lib/graphql/language/sanitized_printer.rb +5 -5
- data/lib/graphql/language/token.rb +0 -4
- data/lib/graphql/name_validator.rb +0 -4
- data/lib/graphql/query/arguments.rb +1 -1
- data/lib/graphql/query/arguments_cache.rb +1 -1
- data/lib/graphql/query/context.rb +5 -2
- data/lib/graphql/query/literal_input.rb +1 -1
- data/lib/graphql/query/null_context.rb +12 -7
- data/lib/graphql/query/serial_execution/field_resolution.rb +1 -1
- data/lib/graphql/query/variables.rb +5 -1
- data/lib/graphql/relay/edges_instrumentation.rb +0 -1
- data/lib/graphql/rubocop/graphql/base_cop.rb +36 -0
- data/lib/graphql/rubocop/graphql/default_null_true.rb +43 -0
- data/lib/graphql/rubocop/graphql/default_required_true.rb +43 -0
- data/lib/graphql/rubocop.rb +4 -0
- data/lib/graphql/schema/addition.rb +37 -28
- data/lib/graphql/schema/argument.rb +6 -6
- data/lib/graphql/schema/build_from_definition.rb +5 -5
- data/lib/graphql/schema/directive/feature.rb +1 -1
- data/lib/graphql/schema/directive/flagged.rb +2 -2
- data/lib/graphql/schema/directive/include.rb +1 -1
- data/lib/graphql/schema/directive/skip.rb +1 -1
- data/lib/graphql/schema/directive/transform.rb +1 -1
- data/lib/graphql/schema/directive.rb +2 -2
- data/lib/graphql/schema/enum.rb +57 -9
- data/lib/graphql/schema/enum_value.rb +4 -0
- data/lib/graphql/schema/field/connection_extension.rb +1 -1
- data/lib/graphql/schema/field.rb +92 -17
- data/lib/graphql/schema/find_inherited_value.rb +1 -0
- data/lib/graphql/schema/finder.rb +5 -5
- data/lib/graphql/schema/input_object.rb +6 -5
- data/lib/graphql/schema/interface.rb +8 -19
- data/lib/graphql/schema/member/accepts_definition.rb +8 -1
- data/lib/graphql/schema/member/build_type.rb +0 -4
- data/lib/graphql/schema/member/has_arguments.rb +55 -13
- data/lib/graphql/schema/member/has_deprecation_reason.rb +1 -1
- data/lib/graphql/schema/member/has_fields.rb +76 -18
- data/lib/graphql/schema/member/has_interfaces.rb +90 -0
- data/lib/graphql/schema/member.rb +1 -0
- data/lib/graphql/schema/object.rb +7 -74
- data/lib/graphql/schema/printer.rb +1 -1
- data/lib/graphql/schema/relay_classic_mutation.rb +29 -3
- data/lib/graphql/schema/resolver/has_payload_type.rb +27 -2
- data/lib/graphql/schema/resolver.rb +19 -5
- data/lib/graphql/schema/subscription.rb +11 -1
- data/lib/graphql/schema/type_expression.rb +1 -1
- data/lib/graphql/schema/type_membership.rb +18 -4
- data/lib/graphql/schema/union.rb +6 -1
- data/lib/graphql/schema/validator/format_validator.rb +0 -4
- data/lib/graphql/schema/validator/numericality_validator.rb +1 -0
- data/lib/graphql/schema/warden.rb +116 -52
- data/lib/graphql/schema.rb +87 -15
- data/lib/graphql/static_validation/base_visitor.rb +5 -5
- data/lib/graphql/static_validation/definition_dependencies.rb +0 -1
- data/lib/graphql/static_validation/literal_validator.rb +1 -1
- data/lib/graphql/static_validation/rules/argument_literals_are_compatible.rb +1 -1
- data/lib/graphql/static_validation/rules/fields_will_merge.rb +8 -15
- data/lib/graphql/static_validation/rules/required_arguments_are_present.rb +1 -3
- data/lib/graphql/static_validation/rules/required_input_object_attributes_are_present.rb +4 -4
- data/lib/graphql/static_validation/rules/variable_usages_are_allowed.rb +7 -7
- data/lib/graphql/subscriptions/action_cable_subscriptions.rb +6 -4
- data/lib/graphql/subscriptions/event.rb +20 -12
- data/lib/graphql/subscriptions.rb +17 -19
- data/lib/graphql/types/relay/has_node_field.rb +1 -1
- data/lib/graphql/types/relay/has_nodes_field.rb +1 -1
- data/lib/graphql/version.rb +1 -1
- data/lib/graphql.rb +9 -31
- metadata +10 -5
| @@ -3,7 +3,7 @@ | |
| 3 3 | 
             
            module GraphQL
         | 
| 4 4 | 
             
              class Schema
         | 
| 5 5 | 
             
                class Member
         | 
| 6 | 
            -
                  # Shared code for  | 
| 6 | 
            +
                  # Shared code for Objects, Interfaces, Mutations, Subscriptions
         | 
| 7 7 | 
             
                  module HasFields
         | 
| 8 8 | 
             
                    # Add a field to this object or interface with the given definition
         | 
| 9 9 | 
             
                    # @see {GraphQL::Schema::Field#initialize} for method signature
         | 
| @@ -15,28 +15,39 @@ module GraphQL | |
| 15 15 | 
             
                    end
         | 
| 16 16 |  | 
| 17 17 | 
             
                    # @return [Hash<String => GraphQL::Schema::Field>] Fields on this object, keyed by name, including inherited fields
         | 
| 18 | 
            -
                    def fields
         | 
| 18 | 
            +
                    def fields(context = GraphQL::Query::NullContext)
         | 
| 19 | 
            +
                      warden = Warden.from_context(context)
         | 
| 20 | 
            +
                      is_object = self.respond_to?(:kind) && self.kind.object?
         | 
| 19 21 | 
             
                      # Local overrides take precedence over inherited fields
         | 
| 20 | 
            -
                       | 
| 21 | 
            -
                       | 
| 22 | 
            -
                        if ancestor.respond_to?(:own_fields)
         | 
| 23 | 
            -
             | 
| 22 | 
            +
                      visible_fields = {}
         | 
| 23 | 
            +
                      for ancestor in ancestors
         | 
| 24 | 
            +
                        if ancestor.respond_to?(:own_fields) &&
         | 
| 25 | 
            +
                            (is_object ? visible_interface_implementation?(ancestor, context, warden) : true)
         | 
| 26 | 
            +
             | 
| 27 | 
            +
                          ancestor.own_fields.each do |field_name, fields_entry|
         | 
| 28 | 
            +
                            # Choose the most local definition that passes `.visible?` --
         | 
| 29 | 
            +
                            # stop checking for fields by name once one has been found.
         | 
| 30 | 
            +
                            if !visible_fields.key?(field_name) && (f = Warden.visible_entry?(:visible_field?, fields_entry, context, warden))
         | 
| 31 | 
            +
                              visible_fields[field_name] = f
         | 
| 32 | 
            +
                            end
         | 
| 33 | 
            +
                          end
         | 
| 24 34 | 
             
                        end
         | 
| 25 35 | 
             
                      end
         | 
| 26 | 
            -
                       | 
| 36 | 
            +
                      visible_fields
         | 
| 27 37 | 
             
                    end
         | 
| 28 38 |  | 
| 29 | 
            -
                    def get_field(field_name)
         | 
| 30 | 
            -
                       | 
| 31 | 
            -
             | 
| 32 | 
            -
                       | 
| 33 | 
            -
                         | 
| 34 | 
            -
             | 
| 35 | 
            -
                             | 
| 36 | 
            -
             | 
| 39 | 
            +
                    def get_field(field_name, context = GraphQL::Query::NullContext)
         | 
| 40 | 
            +
                      warden = Warden.from_context(context)
         | 
| 41 | 
            +
                      is_object = self.respond_to?(:kind) && self.kind.object?
         | 
| 42 | 
            +
                      for ancestor in ancestors
         | 
| 43 | 
            +
                        if ancestor.respond_to?(:own_fields) &&
         | 
| 44 | 
            +
                            (is_object ? visible_interface_implementation?(ancestor, context, warden) : true) &&
         | 
| 45 | 
            +
                            (f_entry = ancestor.own_fields[field_name]) &&
         | 
| 46 | 
            +
                            (f = Warden.visible_entry?(:visible_field?, f_entry, context, warden))
         | 
| 47 | 
            +
                          return f
         | 
| 37 48 | 
             
                        end
         | 
| 38 | 
            -
                        nil
         | 
| 39 49 | 
             
                      end
         | 
| 50 | 
            +
                      nil
         | 
| 40 51 | 
             
                    end
         | 
| 41 52 |  | 
| 42 53 | 
             
                    # A list of Ruby keywords.
         | 
| @@ -64,7 +75,19 @@ module GraphQL | |
| 64 75 | 
             
                      if method_conflict_warning && CONFLICT_FIELD_NAMES.include?(field_defn.resolver_method) && field_defn.original_name == field_defn.resolver_method && field_defn.original_name == field_defn.method_sym
         | 
| 65 76 | 
             
                        warn(conflict_field_name_warning(field_defn))
         | 
| 66 77 | 
             
                      end
         | 
| 67 | 
            -
                      own_fields[field_defn.name] | 
| 78 | 
            +
                      prev_defn = own_fields[field_defn.name]
         | 
| 79 | 
            +
             | 
| 80 | 
            +
                      case prev_defn
         | 
| 81 | 
            +
                      when nil
         | 
| 82 | 
            +
                        own_fields[field_defn.name] = field_defn
         | 
| 83 | 
            +
                      when Array
         | 
| 84 | 
            +
                        prev_defn << field_defn
         | 
| 85 | 
            +
                      when GraphQL::Schema::Field
         | 
| 86 | 
            +
                        own_fields[field_defn.name] = [prev_defn, field_defn]
         | 
| 87 | 
            +
                      else
         | 
| 88 | 
            +
                        raise "Invariant: unexpected previous field definition for #{field_defn.name.inspect}: #{prev_defn.inspect}"
         | 
| 89 | 
            +
                      end
         | 
| 90 | 
            +
             | 
| 68 91 | 
             
                      nil
         | 
| 69 92 | 
             
                    end
         | 
| 70 93 |  | 
| @@ -87,13 +110,48 @@ module GraphQL | |
| 87 110 | 
             
                      end
         | 
| 88 111 | 
             
                    end
         | 
| 89 112 |  | 
| 90 | 
            -
                    # @return [Array<GraphQL::Schema::Field | 
| 113 | 
            +
                    # @return [Hash<String => GraphQL::Schema::Field, Array<GraphQL::Schema::Field>>] Fields defined on this class _specifically_, not parent classes
         | 
| 91 114 | 
             
                    def own_fields
         | 
| 92 115 | 
             
                      @own_fields ||= {}
         | 
| 93 116 | 
             
                    end
         | 
| 94 117 |  | 
| 118 | 
            +
                    def all_field_definitions
         | 
| 119 | 
            +
                      all_fields = {}
         | 
| 120 | 
            +
                      ancestors.reverse_each do |ancestor|
         | 
| 121 | 
            +
                        if ancestor.respond_to?(:own_fields)
         | 
| 122 | 
            +
                          all_fields.merge!(ancestor.own_fields)
         | 
| 123 | 
            +
                        end
         | 
| 124 | 
            +
                      end
         | 
| 125 | 
            +
                      all_fields = all_fields.values
         | 
| 126 | 
            +
                      all_fields.flatten!
         | 
| 127 | 
            +
                      all_fields
         | 
| 128 | 
            +
                    end
         | 
| 129 | 
            +
             | 
| 95 130 | 
             
                    private
         | 
| 96 131 |  | 
| 132 | 
            +
                    # If `type` is an interface, and `self` has a type membership for `type`, then make sure it's visible.
         | 
| 133 | 
            +
                    def visible_interface_implementation?(type, context, warden)
         | 
| 134 | 
            +
                      if type.respond_to?(:kind) && type.kind.interface?
         | 
| 135 | 
            +
                        implements_this_interface = false
         | 
| 136 | 
            +
                        implementation_is_visible = false
         | 
| 137 | 
            +
                        interface_type_memberships.each do |tm|
         | 
| 138 | 
            +
                          if tm.abstract_type == type
         | 
| 139 | 
            +
                            implements_this_interface ||= true
         | 
| 140 | 
            +
                            if warden.visible_type_membership?(tm, context)
         | 
| 141 | 
            +
                              implementation_is_visible = true
         | 
| 142 | 
            +
                              break
         | 
| 143 | 
            +
                            end
         | 
| 144 | 
            +
                          end
         | 
| 145 | 
            +
                        end
         | 
| 146 | 
            +
                        # It's possible this interface came by way of `include` in another interface which this
         | 
| 147 | 
            +
                        # object type _does_ implement, and that's ok
         | 
| 148 | 
            +
                        implements_this_interface ? implementation_is_visible : true
         | 
| 149 | 
            +
                      else
         | 
| 150 | 
            +
                        # If there's no implementation, then we're looking at Ruby-style inheritance instead
         | 
| 151 | 
            +
                        true
         | 
| 152 | 
            +
                      end
         | 
| 153 | 
            +
                    end
         | 
| 154 | 
            +
             | 
| 97 155 | 
             
                    # @param [GraphQL::Schema::Field]
         | 
| 98 156 | 
             
                    # @return [String] A warning to give when this field definition might conflict with a built-in method
         | 
| 99 157 | 
             
                    def conflict_field_name_warning(field_defn)
         | 
| @@ -0,0 +1,90 @@ | |
| 1 | 
            +
            # frozen_string_literal: true
         | 
| 2 | 
            +
             | 
| 3 | 
            +
            module GraphQL
         | 
| 4 | 
            +
              class Schema
         | 
| 5 | 
            +
                class Member
         | 
| 6 | 
            +
                  module HasInterfaces
         | 
| 7 | 
            +
                    def implements(*new_interfaces, **options)
         | 
| 8 | 
            +
                      new_memberships = []
         | 
| 9 | 
            +
                      new_interfaces.each do |int|
         | 
| 10 | 
            +
                        if int.is_a?(Module)
         | 
| 11 | 
            +
                          unless int.include?(GraphQL::Schema::Interface)
         | 
| 12 | 
            +
                            raise "#{int} cannot be implemented since it's not a GraphQL Interface. Use `include` for plain Ruby modules."
         | 
| 13 | 
            +
                          end
         | 
| 14 | 
            +
             | 
| 15 | 
            +
                          new_memberships << int.type_membership_class.new(int, self, **options)
         | 
| 16 | 
            +
             | 
| 17 | 
            +
                          # Include the methods here,
         | 
| 18 | 
            +
                          # `.fields` will use the inheritance chain
         | 
| 19 | 
            +
                          # to find inherited fields
         | 
| 20 | 
            +
                          include(int)
         | 
| 21 | 
            +
             | 
| 22 | 
            +
                          # If this interface has interfaces of its own, add those, too
         | 
| 23 | 
            +
                          int.interfaces.each do |next_interface|
         | 
| 24 | 
            +
                            implements(next_interface)
         | 
| 25 | 
            +
                          end
         | 
| 26 | 
            +
                        elsif int.is_a?(GraphQL::InterfaceType)
         | 
| 27 | 
            +
                          new_memberships << int.type_membership_class.new(int, self, **options)
         | 
| 28 | 
            +
                        elsif int.is_a?(String) || int.is_a?(GraphQL::Schema::LateBoundType)
         | 
| 29 | 
            +
                          if options.any?
         | 
| 30 | 
            +
                            raise ArgumentError, "`implements(...)` doesn't support options with late-loaded types yet. Remove #{options} and open an issue to request this feature."
         | 
| 31 | 
            +
                          end
         | 
| 32 | 
            +
                          new_memberships << int
         | 
| 33 | 
            +
                        else
         | 
| 34 | 
            +
                          raise ArgumentError, "Unexpected interface definition (expected module): #{int} (#{int.class})"
         | 
| 35 | 
            +
                        end
         | 
| 36 | 
            +
                      end
         | 
| 37 | 
            +
             | 
| 38 | 
            +
                      # Remove any String or late-bound interfaces which are being replaced
         | 
| 39 | 
            +
                      own_interface_type_memberships.reject! { |old_i_m|
         | 
| 40 | 
            +
                        if !(old_i_m.respond_to?(:abstract_type) && old_i_m.abstract_type.is_a?(Module))
         | 
| 41 | 
            +
                          old_int_type = old_i_m.respond_to?(:abstract_type) ? old_i_m.abstract_type : old_i_m
         | 
| 42 | 
            +
                          old_name = Schema::Member::BuildType.to_type_name(old_int_type)
         | 
| 43 | 
            +
             | 
| 44 | 
            +
                          new_memberships.any? { |new_i_m|
         | 
| 45 | 
            +
                            new_int_type = new_i_m.respond_to?(:abstract_type) ? new_i_m.abstract_type : new_i_m
         | 
| 46 | 
            +
                            new_name = Schema::Member::BuildType.to_type_name(new_int_type)
         | 
| 47 | 
            +
             | 
| 48 | 
            +
                            new_name == old_name
         | 
| 49 | 
            +
                          }
         | 
| 50 | 
            +
                        end
         | 
| 51 | 
            +
                      }
         | 
| 52 | 
            +
                      own_interface_type_memberships.concat(new_memberships)
         | 
| 53 | 
            +
                    end
         | 
| 54 | 
            +
             | 
| 55 | 
            +
                    def own_interface_type_memberships
         | 
| 56 | 
            +
                      @own_interface_type_memberships ||= []
         | 
| 57 | 
            +
                    end
         | 
| 58 | 
            +
             | 
| 59 | 
            +
                    def interface_type_memberships
         | 
| 60 | 
            +
                      own_interface_type_memberships + ((self.is_a?(Class) && superclass.respond_to?(:interface_type_memberships)) ? superclass.interface_type_memberships : [])
         | 
| 61 | 
            +
                    end
         | 
| 62 | 
            +
             | 
| 63 | 
            +
                    # param context [Query::Context] If omitted, skip filtering.
         | 
| 64 | 
            +
                    def interfaces(context = GraphQL::Query::NullContext)
         | 
| 65 | 
            +
                      warden = Warden.from_context(context)
         | 
| 66 | 
            +
                      visible_interfaces = []
         | 
| 67 | 
            +
                      own_interface_type_memberships.each do |type_membership|
         | 
| 68 | 
            +
                        # During initialization, `type_memberships` can hold late-bound types
         | 
| 69 | 
            +
                        case type_membership
         | 
| 70 | 
            +
                        when String, Schema::LateBoundType
         | 
| 71 | 
            +
                          visible_interfaces << type_membership
         | 
| 72 | 
            +
                        when Schema::TypeMembership
         | 
| 73 | 
            +
                          if warden.visible_type_membership?(type_membership, context)
         | 
| 74 | 
            +
                            visible_interfaces << type_membership.abstract_type
         | 
| 75 | 
            +
                          end
         | 
| 76 | 
            +
                        else
         | 
| 77 | 
            +
                          raise "Invariant: Unexpected type_membership #{type_membership.class}: #{type_membership.inspect}"
         | 
| 78 | 
            +
                        end
         | 
| 79 | 
            +
                      end
         | 
| 80 | 
            +
             | 
| 81 | 
            +
                      if self.is_a?(Class) && superclass <= GraphQL::Schema::Object
         | 
| 82 | 
            +
                        visible_interfaces.concat(superclass.interfaces(context))
         | 
| 83 | 
            +
                      end
         | 
| 84 | 
            +
             | 
| 85 | 
            +
                      visible_interfaces
         | 
| 86 | 
            +
                    end
         | 
| 87 | 
            +
                  end
         | 
| 88 | 
            +
                end
         | 
| 89 | 
            +
              end
         | 
| 90 | 
            +
            end
         | 
| @@ -6,6 +6,7 @@ require 'graphql/schema/member/graphql_type_names' | |
| 6 6 | 
             
            require 'graphql/schema/member/has_ast_node'
         | 
| 7 7 | 
             
            require 'graphql/schema/member/has_directives'
         | 
| 8 8 | 
             
            require 'graphql/schema/member/has_deprecation_reason'
         | 
| 9 | 
            +
            require 'graphql/schema/member/has_interfaces'
         | 
| 9 10 | 
             
            require 'graphql/schema/member/has_path'
         | 
| 10 11 | 
             
            require 'graphql/schema/member/has_unresolved_type_error'
         | 
| 11 12 | 
             
            require 'graphql/schema/member/has_validators'
         | 
| @@ -7,6 +7,7 @@ module GraphQL | |
| 7 7 | 
             
                class Object < GraphQL::Schema::Member
         | 
| 8 8 | 
             
                  extend GraphQL::Schema::Member::AcceptsDefinition
         | 
| 9 9 | 
             
                  extend GraphQL::Schema::Member::HasFields
         | 
| 10 | 
            +
                  extend GraphQL::Schema::Member::HasInterfaces
         | 
| 10 11 |  | 
| 11 12 | 
             
                  # @return [Object] the application object this type is wrapping
         | 
| 12 13 | 
             
                  attr_reader :object
         | 
| @@ -103,84 +104,16 @@ module GraphQL | |
| 103 104 | 
             
                      super
         | 
| 104 105 | 
             
                    end
         | 
| 105 106 |  | 
| 106 | 
            -
                    def implements(*new_interfaces, **options)
         | 
| 107 | 
            -
                      new_memberships = []
         | 
| 108 | 
            -
                      new_interfaces.each do |int|
         | 
| 109 | 
            -
                        if int.is_a?(Module)
         | 
| 110 | 
            -
                          unless int.include?(GraphQL::Schema::Interface)
         | 
| 111 | 
            -
                            raise "#{int} cannot be implemented since it's not a GraphQL Interface. Use `include` for plain Ruby modules."
         | 
| 112 | 
            -
                          end
         | 
| 113 | 
            -
             | 
| 114 | 
            -
                          new_memberships << int.type_membership_class.new(int, self, **options)
         | 
| 115 | 
            -
             | 
| 116 | 
            -
                          # Include the methods here,
         | 
| 117 | 
            -
                          # `.fields` will use the inheritance chain
         | 
| 118 | 
            -
                          # to find inherited fields
         | 
| 119 | 
            -
                          include(int)
         | 
| 120 | 
            -
                        elsif int.is_a?(GraphQL::InterfaceType)
         | 
| 121 | 
            -
                          new_memberships << int.type_membership_class.new(int, self, **options)
         | 
| 122 | 
            -
                        elsif int.is_a?(String) || int.is_a?(GraphQL::Schema::LateBoundType)
         | 
| 123 | 
            -
                          if options.any?
         | 
| 124 | 
            -
                            raise ArgumentError, "`implements(...)` doesn't support options with late-loaded types yet. Remove #{options} and open an issue to request this feature."
         | 
| 125 | 
            -
                          end
         | 
| 126 | 
            -
                          new_memberships << int
         | 
| 127 | 
            -
                        else
         | 
| 128 | 
            -
                          raise ArgumentError, "Unexpected interface definition (expected module): #{int} (#{int.class})"
         | 
| 129 | 
            -
                        end
         | 
| 130 | 
            -
                      end
         | 
| 131 | 
            -
             | 
| 132 | 
            -
                      # Remove any interfaces which are being replaced (late-bound types are updated in place this way)
         | 
| 133 | 
            -
                      own_interface_type_memberships.reject! { |old_i_m|
         | 
| 134 | 
            -
                        old_int_type = old_i_m.respond_to?(:abstract_type) ? old_i_m.abstract_type : old_i_m
         | 
| 135 | 
            -
                        old_name = Schema::Member::BuildType.to_type_name(old_int_type)
         | 
| 136 | 
            -
             | 
| 137 | 
            -
                        new_memberships.any? { |new_i_m|
         | 
| 138 | 
            -
                          new_int_type = new_i_m.respond_to?(:abstract_type) ? new_i_m.abstract_type : new_i_m
         | 
| 139 | 
            -
                          new_name = Schema::Member::BuildType.to_type_name(new_int_type)
         | 
| 140 | 
            -
             | 
| 141 | 
            -
                          new_name == old_name
         | 
| 142 | 
            -
                        }
         | 
| 143 | 
            -
                      }
         | 
| 144 | 
            -
                      own_interface_type_memberships.concat(new_memberships)
         | 
| 145 | 
            -
                    end
         | 
| 146 | 
            -
             | 
| 147 | 
            -
                    def own_interface_type_memberships
         | 
| 148 | 
            -
                      @own_interface_type_memberships ||= []
         | 
| 149 | 
            -
                    end
         | 
| 150 | 
            -
             | 
| 151 | 
            -
                    def interface_type_memberships
         | 
| 152 | 
            -
                      own_interface_type_memberships + (superclass.respond_to?(:interface_type_memberships) ? superclass.interface_type_memberships : [])
         | 
| 153 | 
            -
                    end
         | 
| 154 | 
            -
             | 
| 155 | 
            -
                    # param context [Query::Context] If omitted, skip filtering.
         | 
| 156 | 
            -
                    def interfaces(context = GraphQL::Query::NullContext)
         | 
| 157 | 
            -
                      visible_interfaces = []
         | 
| 158 | 
            -
                      unfiltered = context == GraphQL::Query::NullContext
         | 
| 159 | 
            -
                      own_interface_type_memberships.each do |type_membership|
         | 
| 160 | 
            -
                        # During initialization, `type_memberships` can hold late-bound types
         | 
| 161 | 
            -
                        case type_membership
         | 
| 162 | 
            -
                        when String, Schema::LateBoundType
         | 
| 163 | 
            -
                          visible_interfaces << type_membership
         | 
| 164 | 
            -
                        when Schema::TypeMembership
         | 
| 165 | 
            -
                          if unfiltered || type_membership.visible?(context)
         | 
| 166 | 
            -
                            visible_interfaces << type_membership.abstract_type
         | 
| 167 | 
            -
                          end
         | 
| 168 | 
            -
                        else
         | 
| 169 | 
            -
                          raise "Invariant: Unexpected type_membership #{type_membership.class}: #{type_membership.inspect}"
         | 
| 170 | 
            -
                        end
         | 
| 171 | 
            -
                      end
         | 
| 172 | 
            -
                      visible_interfaces + (superclass <= GraphQL::Schema::Object ? superclass.interfaces(context) : [])
         | 
| 173 | 
            -
                    end
         | 
| 174 | 
            -
             | 
| 175 107 | 
             
                    # @return [Hash<String => GraphQL::Schema::Field>] All of this object's fields, indexed by name
         | 
| 176 108 | 
             
                    # @see get_field A faster way to find one field by name ({#fields} merges hashes of inherited fields; {#get_field} just looks up one field.)
         | 
| 177 | 
            -
                    def fields
         | 
| 109 | 
            +
                    def fields(context = GraphQL::Query::NullContext)
         | 
| 178 110 | 
             
                      all_fields = super
         | 
| 111 | 
            +
                      # This adds fields from legacy-style interfaces only.
         | 
| 112 | 
            +
                      # Multi-fields are not supported here.
         | 
| 179 113 | 
             
                      interfaces.each do |int|
         | 
| 180 | 
            -
                        # Include legacy-style interfaces, too
         | 
| 181 114 | 
             
                        if int.is_a?(GraphQL::InterfaceType)
         | 
| 182 115 | 
             
                          int_f = {}
         | 
| 183 | 
            -
                          int.fields.each do |name, legacy_field|
         | 
| 116 | 
            +
                          int.fields.each do |name, legacy_field| # rubocop:disable Development/ContextIsPassedCop -- legacy-related
         | 
| 184 117 | 
             
                            int_f[name] = field_class.from_options(name, field: legacy_field)
         | 
| 185 118 | 
             
                          end
         | 
| 186 119 | 
             
                          all_fields = int_f.merge(all_fields)
         | 
| @@ -198,9 +131,9 @@ module GraphQL | |
| 198 131 | 
             
                      obj_type.introspection = introspection
         | 
| 199 132 | 
             
                      obj_type.mutation = mutation
         | 
| 200 133 | 
             
                      obj_type.ast_node = ast_node
         | 
| 201 | 
            -
                      fields.each do |field_name, field_inst|
         | 
| 134 | 
            +
                      fields.each do |field_name, field_inst| # rubocop:disable Development/ContextIsPassedCop -- legacy-related
         | 
| 202 135 | 
             
                        field_defn = field_inst.to_graphql
         | 
| 203 | 
            -
                        obj_type.fields[field_defn.name] = field_defn
         | 
| 136 | 
            +
                        obj_type.fields[field_defn.name] = field_defn # rubocop:disable Development/ContextIsPassedCop -- legacy-related
         | 
| 204 137 | 
             
                      end
         | 
| 205 138 |  | 
| 206 139 | 
             
                      obj_type.metadata[:type_class] = self
         | 
| @@ -56,7 +56,7 @@ module GraphQL | |
| 56 56 | 
             
                  def self.print_introspection_schema
         | 
| 57 57 | 
             
                    query_root = Class.new(GraphQL::Schema::Object) do
         | 
| 58 58 | 
             
                      graphql_name "Root"
         | 
| 59 | 
            -
                      field :throwaway_field, String | 
| 59 | 
            +
                      field :throwaway_field, String
         | 
| 60 60 | 
             
                    end
         | 
| 61 61 | 
             
                    schema = Class.new(GraphQL::Schema) { query(query_root) }
         | 
| 62 62 |  | 
| @@ -22,7 +22,7 @@ module GraphQL | |
| 22 22 | 
             
                #
         | 
| 23 23 | 
             
                class RelayClassicMutation < GraphQL::Schema::Mutation
         | 
| 24 24 | 
             
                  # The payload should always include this field
         | 
| 25 | 
            -
                  field(:client_mutation_id, String, "A unique identifier for the client performing the mutation." | 
| 25 | 
            +
                  field(:client_mutation_id, String, "A unique identifier for the client performing the mutation.")
         | 
| 26 26 | 
             
                  # Relay classic default:
         | 
| 27 27 | 
             
                  null(true)
         | 
| 28 28 |  | 
| @@ -81,6 +81,31 @@ module GraphQL | |
| 81 81 | 
             
                  end
         | 
| 82 82 |  | 
| 83 83 | 
             
                  class << self
         | 
| 84 | 
            +
             | 
| 85 | 
            +
                    # Also apply this argument to the input type:
         | 
| 86 | 
            +
                    def argument(*args, **kwargs, &block)
         | 
| 87 | 
            +
                      it = input_type # make sure any inherited arguments are already added to it
         | 
| 88 | 
            +
                      arg = super
         | 
| 89 | 
            +
             | 
| 90 | 
            +
                      # This definition might be overriding something inherited;
         | 
| 91 | 
            +
                      # if it is, remove the inherited definition so it's not confused at runtime as having multiple definitions
         | 
| 92 | 
            +
                      prev_args = it.own_arguments[arg.graphql_name]
         | 
| 93 | 
            +
                      case prev_args
         | 
| 94 | 
            +
                      when GraphQL::Schema::Argument
         | 
| 95 | 
            +
                        if prev_args.owner != self
         | 
| 96 | 
            +
                          it.own_arguments.delete(arg.graphql_name)
         | 
| 97 | 
            +
                        end
         | 
| 98 | 
            +
                      when Array
         | 
| 99 | 
            +
                        prev_args.reject! { |a| a.owner != self }
         | 
| 100 | 
            +
                        if prev_args.empty?
         | 
| 101 | 
            +
                          it.own_arguments.delete(arg.graphql_name)
         | 
| 102 | 
            +
                        end
         | 
| 103 | 
            +
                      end
         | 
| 104 | 
            +
             | 
| 105 | 
            +
                      it.add_argument(arg)
         | 
| 106 | 
            +
                      arg
         | 
| 107 | 
            +
                    end
         | 
| 108 | 
            +
             | 
| 84 109 | 
             
                    # The base class for generated input object types
         | 
| 85 110 | 
             
                    # @param new_class [Class] The base class to use for generating input object definitions
         | 
| 86 111 | 
             
                    # @return [Class] The base class for this mutation's generated input object (default is {GraphQL::Schema::InputObject})
         | 
| @@ -115,14 +140,15 @@ module GraphQL | |
| 115 140 | 
             
                    # To customize how input objects are generated, override this method
         | 
| 116 141 | 
             
                    # @return [Class] a subclass of {.input_object_class}
         | 
| 117 142 | 
             
                    def generate_input_type
         | 
| 118 | 
            -
                      mutation_args =  | 
| 143 | 
            +
                      mutation_args = all_argument_definitions
         | 
| 119 144 | 
             
                      mutation_name = graphql_name
         | 
| 120 145 | 
             
                      mutation_class = self
         | 
| 121 146 | 
             
                      Class.new(input_object_class) do
         | 
| 122 147 | 
             
                        graphql_name("#{mutation_name}Input")
         | 
| 123 148 | 
             
                        description("Autogenerated input type of #{mutation_name}")
         | 
| 124 149 | 
             
                        mutation(mutation_class)
         | 
| 125 | 
            -
                         | 
| 150 | 
            +
                        # these might be inherited:
         | 
| 151 | 
            +
                        mutation_args.each do |arg|
         | 
| 126 152 | 
             
                          add_argument(arg)
         | 
| 127 153 | 
             
                        end
         | 
| 128 154 | 
             
                        argument :client_mutation_id, String, "A unique identifier for the client performing the mutation.", required: false
         | 
| @@ -38,6 +38,9 @@ module GraphQL | |
| 38 38 | 
             
                    # @return [Class]
         | 
| 39 39 | 
             
                    def object_class(new_class = nil)
         | 
| 40 40 | 
             
                      if new_class
         | 
| 41 | 
            +
                        if defined?(@payload_type)
         | 
| 42 | 
            +
                          raise "Can't configure `object_class(...)` after the payload type has already been initialized. Move this configuration higher up the class definition."
         | 
| 43 | 
            +
                        end
         | 
| 41 44 | 
             
                        @object_class = new_class
         | 
| 42 45 | 
             
                      else
         | 
| 43 46 | 
             
                        @object_class || find_inherited_value(:object_class, GraphQL::Schema::Object)
         | 
| @@ -46,6 +49,28 @@ module GraphQL | |
| 46 49 |  | 
| 47 50 | 
             
                    NO_INTERFACES = [].freeze
         | 
| 48 51 |  | 
| 52 | 
            +
                    def field(*args, **kwargs, &block)
         | 
| 53 | 
            +
                      pt = payload_type # make sure it's initialized with any inherited fields
         | 
| 54 | 
            +
                      field_defn = super
         | 
| 55 | 
            +
             | 
| 56 | 
            +
                      # Remove any inherited fields to avoid false conflicts at runtime
         | 
| 57 | 
            +
                      prev_fields = pt.own_fields[field_defn.graphql_name]
         | 
| 58 | 
            +
                      case prev_fields
         | 
| 59 | 
            +
                      when GraphQL::Schema::Field
         | 
| 60 | 
            +
                        if prev_fields.owner != self
         | 
| 61 | 
            +
                          pt.own_fields.delete(field_defn.graphql_name)
         | 
| 62 | 
            +
                        end
         | 
| 63 | 
            +
                      when Array
         | 
| 64 | 
            +
                        prev_fields.reject! { |f| f.owner != self }
         | 
| 65 | 
            +
                        if prev_fields.empty?
         | 
| 66 | 
            +
                          pt.own_fields.delete(field_defn.graphql_name)
         | 
| 67 | 
            +
                        end
         | 
| 68 | 
            +
                      end
         | 
| 69 | 
            +
             | 
| 70 | 
            +
                      pt.add_field(field_defn, method_conflict_warning: false)
         | 
| 71 | 
            +
                      field_defn
         | 
| 72 | 
            +
                    end
         | 
| 73 | 
            +
             | 
| 49 74 | 
             
                    private
         | 
| 50 75 |  | 
| 51 76 | 
             
                    # Build a subclass of {.object_class} based on `self`.
         | 
| @@ -53,11 +78,11 @@ module GraphQL | |
| 53 78 | 
             
                    # Override this hook to customize return type generation.
         | 
| 54 79 | 
             
                    def generate_payload_type
         | 
| 55 80 | 
             
                      resolver_name = graphql_name
         | 
| 56 | 
            -
                      resolver_fields =  | 
| 81 | 
            +
                      resolver_fields = all_field_definitions
         | 
| 57 82 | 
             
                      Class.new(object_class) do
         | 
| 58 83 | 
             
                        graphql_name("#{resolver_name}Payload")
         | 
| 59 84 | 
             
                        description("Autogenerated return type of #{resolver_name}")
         | 
| 60 | 
            -
                        resolver_fields.each do | | 
| 85 | 
            +
                        resolver_fields.each do |f|
         | 
| 61 86 | 
             
                          # Reattach the already-defined field here
         | 
| 62 87 | 
             
                          # (The field's `.owner` will still point to the mutation, not the object type, I think)
         | 
| 63 88 | 
             
                          # Don't re-warn about a method conflict. Since this type is generated, it should be fixed in the resolver instead.
         | 
| @@ -37,7 +37,7 @@ module GraphQL | |
| 37 37 | 
             
                    @field = field
         | 
| 38 38 | 
             
                    # Since this hash is constantly rebuilt, cache it for this call
         | 
| 39 39 | 
             
                    @arguments_by_keyword = {}
         | 
| 40 | 
            -
                    self.class.arguments.each do |name, arg|
         | 
| 40 | 
            +
                    self.class.arguments(context).each do |name, arg|
         | 
| 41 41 | 
             
                      @arguments_by_keyword[arg.keyword] = arg
         | 
| 42 42 | 
             
                    end
         | 
| 43 43 | 
             
                    @prepared_arguments = nil
         | 
| @@ -145,7 +145,7 @@ module GraphQL | |
| 145 145 | 
             
                  # @raise [GraphQL::UnauthorizedError] To signal an authorization failure
         | 
| 146 146 | 
             
                  # @return [Boolean, early_return_data] If `false`, execution will stop (and `early_return_data` will be returned instead, if present.)
         | 
| 147 147 | 
             
                  def authorized?(**inputs)
         | 
| 148 | 
            -
                    self.class.arguments.each_value do |argument|
         | 
| 148 | 
            +
                    self.class.arguments(context).each_value do |argument|
         | 
| 149 149 | 
             
                      arg_keyword = argument.keyword
         | 
| 150 150 | 
             
                      if inputs.key?(arg_keyword) && !(arg_value = inputs[arg_keyword]).nil? && (arg_value != argument.default_value)
         | 
| 151 151 | 
             
                        arg_auth, err = argument.authorized?(self, arg_value, context)
         | 
| @@ -199,8 +199,8 @@ module GraphQL | |
| 199 199 | 
             
                    end
         | 
| 200 200 | 
             
                  end
         | 
| 201 201 |  | 
| 202 | 
            -
                  def get_argument(name)
         | 
| 203 | 
            -
                    self.class.get_argument(name)
         | 
| 202 | 
            +
                  def get_argument(name, context = GraphQL::Query::NullContext)
         | 
| 203 | 
            +
                    self.class.get_argument(name, context)
         | 
| 204 204 | 
             
                  end
         | 
| 205 205 |  | 
| 206 206 | 
             
                  class << self
         | 
| @@ -305,13 +305,27 @@ module GraphQL | |
| 305 305 | 
             
                    end
         | 
| 306 306 |  | 
| 307 307 | 
             
                    def field_options
         | 
| 308 | 
            +
             | 
| 309 | 
            +
                      all_args = {}
         | 
| 310 | 
            +
                      all_argument_definitions.each do |arg|
         | 
| 311 | 
            +
                        if (prev_entry = all_args[arg.graphql_name])
         | 
| 312 | 
            +
                          if prev_entry.is_a?(Array)
         | 
| 313 | 
            +
                            prev_entry << arg
         | 
| 314 | 
            +
                          else
         | 
| 315 | 
            +
                            all_args[arg.graphql_name] = [prev_entry, arg]
         | 
| 316 | 
            +
                          end
         | 
| 317 | 
            +
                        else
         | 
| 318 | 
            +
                          all_args[arg.graphql_name] = arg
         | 
| 319 | 
            +
                        end
         | 
| 320 | 
            +
                      end
         | 
| 321 | 
            +
             | 
| 308 322 | 
             
                      field_opts = {
         | 
| 309 323 | 
             
                        type: type_expr,
         | 
| 310 324 | 
             
                        description: description,
         | 
| 311 325 | 
             
                        extras: extras,
         | 
| 312 326 | 
             
                        resolver_method: :resolve_with_support,
         | 
| 313 327 | 
             
                        resolver_class: self,
         | 
| 314 | 
            -
                        arguments:  | 
| 328 | 
            +
                        arguments: all_args,
         | 
| 315 329 | 
             
                        null: null,
         | 
| 316 330 | 
             
                        complexity: complexity,
         | 
| 317 331 | 
             
                        broadcastable: broadcastable?,
         | 
| @@ -103,10 +103,12 @@ module GraphQL | |
| 103 103 | 
             
                  # Call this method to provide a new subscription_scope; OR
         | 
| 104 104 | 
             
                  # call it without an argument to get the subscription_scope
         | 
| 105 105 | 
             
                  # @param new_scope [Symbol]
         | 
| 106 | 
            +
                  # @param optional [Boolean] If true, then don't require `scope:` to be provided to updates to this subscription.
         | 
| 106 107 | 
             
                  # @return [Symbol]
         | 
| 107 | 
            -
                  def self.subscription_scope(new_scope = READING_SCOPE)
         | 
| 108 | 
            +
                  def self.subscription_scope(new_scope = READING_SCOPE, optional: false)
         | 
| 108 109 | 
             
                    if new_scope != READING_SCOPE
         | 
| 109 110 | 
             
                      @subscription_scope = new_scope
         | 
| 111 | 
            +
                      @subscription_scope_optional = optional
         | 
| 110 112 | 
             
                    elsif defined?(@subscription_scope)
         | 
| 111 113 | 
             
                      @subscription_scope
         | 
| 112 114 | 
             
                    else
         | 
| @@ -114,6 +116,14 @@ module GraphQL | |
| 114 116 | 
             
                    end
         | 
| 115 117 | 
             
                  end
         | 
| 116 118 |  | 
| 119 | 
            +
                  def self.subscription_scope_optional?
         | 
| 120 | 
            +
                    if defined?(@subscription_scope_optional)
         | 
| 121 | 
            +
                      @subscription_scope_optional
         | 
| 122 | 
            +
                    else
         | 
| 123 | 
            +
                      find_inherited_value(:subscription_scope_optional, false)
         | 
| 124 | 
            +
                    end
         | 
| 125 | 
            +
                  end
         | 
| 126 | 
            +
             | 
| 117 127 | 
             
                  # This is called during initial subscription to get a "name" for this subscription.
         | 
| 118 128 | 
             
                  # Later, when `.trigger` is called, this will be called again to build another "name".
         | 
| 119 129 | 
             
                  # Any subscribers with matching topic will begin the update flow.
         | 
| @@ -11,7 +11,7 @@ module GraphQL | |
| 11 11 | 
             
                  def self.build_type(type_owner, ast_node)
         | 
| 12 12 | 
             
                    case ast_node
         | 
| 13 13 | 
             
                    when GraphQL::Language::Nodes::TypeName
         | 
| 14 | 
            -
                      type_owner.get_type(ast_node.name)
         | 
| 14 | 
            +
                      type_owner.get_type(ast_node.name) # rubocop:disable Development/ContextIsPassedCop -- this is a `context` or `warden`, it's already query-aware
         | 
| 15 15 | 
             
                    when GraphQL::Language::Nodes::NonNullType
         | 
| 16 16 | 
             
                      ast_inner_type = ast_node.of_type
         | 
| 17 17 | 
             
                      inner_type = build_type(type_owner, ast_inner_type)
         | 
| @@ -4,8 +4,6 @@ module GraphQL | |
| 4 4 | 
             
              class Schema
         | 
| 5 5 | 
             
                # This class joins an object type to an abstract type (interface or union) of which
         | 
| 6 6 | 
             
                # it is a member.
         | 
| 7 | 
            -
                #
         | 
| 8 | 
            -
                # TODO: Not yet implemented for interfaces.
         | 
| 9 7 | 
             
                class TypeMembership
         | 
| 10 8 | 
             
                  # @return [Class<GraphQL::Schema::Object>]
         | 
| 11 9 | 
             
                  attr_accessor :object_type
         | 
| @@ -26,9 +24,25 @@ module GraphQL | |
| 26 24 | 
             
                  end
         | 
| 27 25 |  | 
| 28 26 | 
             
                  # @return [Boolean] if false, {#object_type} will be treated as _not_ a member of {#abstract_type}
         | 
| 29 | 
            -
                  def visible?( | 
| 30 | 
            -
                     | 
| 27 | 
            +
                  def visible?(ctx)
         | 
| 28 | 
            +
                    warden = Warden.from_context(ctx)
         | 
| 29 | 
            +
                    (@object_type.respond_to?(:visible?) ? warden.visible_type?(@object_type, ctx) : true) &&
         | 
| 30 | 
            +
                      (@abstract_type.respond_to?(:visible?) ? warden.visible_type?(@abstract_type, ctx) : true)
         | 
| 31 31 | 
             
                  end
         | 
| 32 | 
            +
             | 
| 33 | 
            +
                  def graphql_name
         | 
| 34 | 
            +
                    "#{@object_type.graphql_name}.#{@abstract_type.kind.interface? ? "implements" : "belongsTo" }.#{@abstract_type.graphql_name}"
         | 
| 35 | 
            +
                  end
         | 
| 36 | 
            +
             | 
| 37 | 
            +
                  def path
         | 
| 38 | 
            +
                    graphql_name
         | 
| 39 | 
            +
                  end
         | 
| 40 | 
            +
             | 
| 41 | 
            +
                  def inspect
         | 
| 42 | 
            +
                    "#<#{self.class} #{@object_type.inspect} => #{@abstract_type.inspect}>"
         | 
| 43 | 
            +
                  end
         | 
| 44 | 
            +
             | 
| 45 | 
            +
                  alias :type_class :itself
         | 
| 32 46 | 
             
                end
         | 
| 33 47 | 
             
              end
         | 
| 34 48 | 
             
            end
         | 
    
        data/lib/graphql/schema/union.rb
    CHANGED
    
    | @@ -19,8 +19,9 @@ module GraphQL | |
| 19 19 | 
             
                        end
         | 
| 20 20 | 
             
                      else
         | 
| 21 21 | 
             
                        visible_types = []
         | 
| 22 | 
            +
                        warden = Warden.from_context(context)
         | 
| 22 23 | 
             
                        type_memberships.each do |type_membership|
         | 
| 23 | 
            -
                          if  | 
| 24 | 
            +
                          if warden.visible_type_membership?(type_membership, context)
         | 
| 24 25 | 
             
                            visible_types << type_membership.object_type
         | 
| 25 26 | 
             
                          end
         | 
| 26 27 | 
             
                        end
         | 
| @@ -28,6 +29,10 @@ module GraphQL | |
| 28 29 | 
             
                      end
         | 
| 29 30 | 
             
                    end
         | 
| 30 31 |  | 
| 32 | 
            +
                    def all_possible_types
         | 
| 33 | 
            +
                      type_memberships.map(&:object_type)
         | 
| 34 | 
            +
                    end
         | 
| 35 | 
            +
             | 
| 31 36 | 
             
                    def to_graphql
         | 
| 32 37 | 
             
                      type_defn = GraphQL::UnionType.new
         | 
| 33 38 | 
             
                      type_defn.name = graphql_name
         | 
| @@ -18,10 +18,6 @@ module GraphQL | |
| 18 18 | 
             
                  #   # It's pretty hard to come up with a legitimate use case for `without:`
         | 
| 19 19 | 
             
                  #
         | 
| 20 20 | 
             
                  class FormatValidator < Validator
         | 
| 21 | 
            -
                    if !String.method_defined?(:match?)
         | 
| 22 | 
            -
                      using GraphQL::StringMatchBackport
         | 
| 23 | 
            -
                    end
         | 
| 24 | 
            -
             | 
| 25 21 | 
             
                    # @param with [RegExp, nil]
         | 
| 26 22 | 
             
                    # @param without [Regexp, nil]
         | 
| 27 23 | 
             
                    # @param message [String]
         |