graphql 1.9.17 → 1.11.7
Sign up to get free protection for your applications and to get access to all the features.
- checksums.yaml +4 -4
- data/lib/generators/graphql/core.rb +18 -2
- data/lib/generators/graphql/install_generator.rb +27 -0
- data/lib/generators/graphql/object_generator.rb +52 -8
- data/lib/generators/graphql/templates/base_argument.erb +2 -0
- data/lib/generators/graphql/templates/base_enum.erb +2 -0
- data/lib/generators/graphql/templates/base_field.erb +2 -0
- data/lib/generators/graphql/templates/base_input_object.erb +2 -0
- data/lib/generators/graphql/templates/base_interface.erb +2 -0
- data/lib/generators/graphql/templates/base_mutation.erb +2 -0
- data/lib/generators/graphql/templates/base_object.erb +2 -0
- data/lib/generators/graphql/templates/base_scalar.erb +2 -0
- data/lib/generators/graphql/templates/base_union.erb +2 -0
- data/lib/generators/graphql/templates/enum.erb +2 -0
- data/lib/generators/graphql/templates/graphql_controller.erb +14 -10
- data/lib/generators/graphql/templates/interface.erb +2 -0
- data/lib/generators/graphql/templates/loader.erb +2 -0
- data/lib/generators/graphql/templates/mutation.erb +2 -0
- data/lib/generators/graphql/templates/mutation_type.erb +2 -0
- data/lib/generators/graphql/templates/object.erb +2 -0
- data/lib/generators/graphql/templates/query_type.erb +2 -0
- data/lib/generators/graphql/templates/scalar.erb +2 -0
- data/lib/generators/graphql/templates/schema.erb +10 -0
- data/lib/generators/graphql/templates/union.erb +3 -1
- data/lib/graphql/analysis/ast/field_usage.rb +1 -1
- data/lib/graphql/analysis/ast/query_complexity.rb +178 -67
- data/lib/graphql/analysis/ast/visitor.rb +3 -3
- data/lib/graphql/analysis/ast.rb +12 -11
- data/lib/graphql/argument.rb +10 -38
- data/lib/graphql/backtrace/table.rb +10 -2
- data/lib/graphql/backtrace/tracer.rb +2 -1
- data/lib/graphql/base_type.rb +4 -0
- data/lib/graphql/compatibility/execution_specification/specification_schema.rb +2 -2
- data/lib/graphql/compatibility/query_parser_specification/parse_error_specification.rb +5 -9
- data/lib/graphql/define/assign_enum_value.rb +1 -1
- data/lib/graphql/define/assign_global_id_field.rb +2 -2
- data/lib/graphql/define/assign_object_field.rb +3 -3
- data/lib/graphql/define/defined_object_proxy.rb +3 -0
- data/lib/graphql/define/instance_definable.rb +18 -108
- data/lib/graphql/directive/deprecated_directive.rb +1 -12
- data/lib/graphql/directive.rb +8 -1
- data/lib/graphql/enum_type.rb +5 -71
- data/lib/graphql/execution/directive_checks.rb +2 -2
- data/lib/graphql/execution/errors.rb +2 -3
- data/lib/graphql/execution/execute.rb +1 -1
- data/lib/graphql/execution/instrumentation.rb +1 -1
- data/lib/graphql/execution/interpreter/argument_value.rb +28 -0
- data/lib/graphql/execution/interpreter/arguments.rb +51 -0
- data/lib/graphql/execution/interpreter/arguments_cache.rb +79 -0
- data/lib/graphql/execution/interpreter/handles_raw_value.rb +25 -0
- data/lib/graphql/execution/interpreter/runtime.rb +227 -254
- data/lib/graphql/execution/interpreter.rb +34 -11
- data/lib/graphql/execution/lazy/lazy_method_map.rb +4 -0
- data/lib/graphql/execution/lookahead.rb +39 -114
- data/lib/graphql/execution/multiplex.rb +14 -5
- data/lib/graphql/field.rb +14 -118
- data/lib/graphql/filter.rb +1 -1
- data/lib/graphql/function.rb +1 -30
- data/lib/graphql/input_object_type.rb +6 -24
- data/lib/graphql/integer_decoding_error.rb +17 -0
- data/lib/graphql/interface_type.rb +7 -23
- data/lib/graphql/internal_representation/scope.rb +2 -2
- data/lib/graphql/internal_representation/visit.rb +2 -2
- data/lib/graphql/introspection/base_object.rb +2 -5
- data/lib/graphql/introspection/directive_type.rb +1 -1
- data/lib/graphql/introspection/entry_points.rb +7 -7
- data/lib/graphql/introspection/field_type.rb +7 -3
- data/lib/graphql/introspection/input_value_type.rb +33 -9
- data/lib/graphql/introspection/introspection_query.rb +6 -92
- data/lib/graphql/introspection/schema_type.rb +4 -9
- data/lib/graphql/introspection/type_type.rb +11 -7
- data/lib/graphql/introspection.rb +96 -0
- data/lib/graphql/invalid_null_error.rb +18 -0
- data/lib/graphql/language/block_string.rb +24 -5
- data/lib/graphql/language/definition_slice.rb +21 -10
- data/lib/graphql/language/document_from_schema_definition.rb +89 -64
- data/lib/graphql/language/lexer.rb +7 -3
- data/lib/graphql/language/lexer.rl +7 -3
- data/lib/graphql/language/nodes.rb +52 -91
- data/lib/graphql/language/parser.rb +719 -717
- data/lib/graphql/language/parser.y +104 -98
- data/lib/graphql/language/printer.rb +1 -1
- data/lib/graphql/language/sanitized_printer.rb +222 -0
- data/lib/graphql/language/visitor.rb +2 -2
- data/lib/graphql/language.rb +2 -1
- data/lib/graphql/name_validator.rb +6 -7
- data/lib/graphql/non_null_type.rb +0 -10
- data/lib/graphql/object_type.rb +45 -56
- data/lib/graphql/pagination/active_record_relation_connection.rb +41 -0
- data/lib/graphql/pagination/array_connection.rb +77 -0
- data/lib/graphql/pagination/connection.rb +208 -0
- data/lib/graphql/pagination/connections.rb +145 -0
- data/lib/graphql/pagination/mongoid_relation_connection.rb +25 -0
- data/lib/graphql/pagination/relation_connection.rb +185 -0
- data/lib/graphql/pagination/sequel_dataset_connection.rb +28 -0
- data/lib/graphql/pagination.rb +6 -0
- data/lib/graphql/query/arguments.rb +4 -2
- data/lib/graphql/query/context.rb +36 -9
- data/lib/graphql/query/fingerprint.rb +26 -0
- data/lib/graphql/query/input_validation_result.rb +23 -6
- data/lib/graphql/query/literal_input.rb +30 -10
- data/lib/graphql/query/null_context.rb +5 -1
- data/lib/graphql/query/validation_pipeline.rb +4 -1
- data/lib/graphql/query/variable_validation_error.rb +1 -1
- data/lib/graphql/query/variables.rb +16 -7
- data/lib/graphql/query.rb +64 -15
- data/lib/graphql/rake_task/validate.rb +3 -0
- data/lib/graphql/rake_task.rb +9 -9
- data/lib/graphql/relay/array_connection.rb +10 -12
- data/lib/graphql/relay/base_connection.rb +23 -13
- data/lib/graphql/relay/connection_type.rb +2 -1
- data/lib/graphql/relay/edge_type.rb +1 -0
- data/lib/graphql/relay/edges_instrumentation.rb +1 -1
- data/lib/graphql/relay/mutation.rb +1 -86
- data/lib/graphql/relay/node.rb +2 -2
- data/lib/graphql/relay/range_add.rb +14 -5
- data/lib/graphql/relay/relation_connection.rb +8 -10
- data/lib/graphql/scalar_type.rb +15 -59
- data/lib/graphql/schema/argument.rb +113 -11
- data/lib/graphql/schema/base_64_encoder.rb +2 -0
- data/lib/graphql/schema/build_from_definition/resolve_map/default_resolve.rb +1 -1
- data/lib/graphql/schema/build_from_definition/resolve_map.rb +13 -5
- data/lib/graphql/schema/build_from_definition.rb +212 -190
- data/lib/graphql/schema/built_in_types.rb +5 -5
- data/lib/graphql/schema/default_type_error.rb +2 -0
- data/lib/graphql/schema/directive/deprecated.rb +18 -0
- data/lib/graphql/schema/directive/include.rb +1 -1
- data/lib/graphql/schema/directive/skip.rb +1 -1
- data/lib/graphql/schema/directive.rb +34 -3
- data/lib/graphql/schema/enum.rb +52 -4
- data/lib/graphql/schema/enum_value.rb +6 -1
- data/lib/graphql/schema/field/connection_extension.rb +44 -20
- data/lib/graphql/schema/field/scope_extension.rb +1 -1
- data/lib/graphql/schema/field.rb +200 -129
- data/lib/graphql/schema/find_inherited_value.rb +13 -0
- data/lib/graphql/schema/finder.rb +13 -11
- data/lib/graphql/schema/input_object.rb +131 -22
- data/lib/graphql/schema/interface.rb +26 -8
- data/lib/graphql/schema/introspection_system.rb +108 -37
- data/lib/graphql/schema/late_bound_type.rb +3 -2
- data/lib/graphql/schema/list.rb +47 -0
- data/lib/graphql/schema/loader.rb +134 -96
- data/lib/graphql/schema/member/base_dsl_methods.rb +29 -12
- data/lib/graphql/schema/member/build_type.rb +19 -5
- data/lib/graphql/schema/member/cached_graphql_definition.rb +5 -0
- data/lib/graphql/schema/member/has_arguments.rb +105 -5
- data/lib/graphql/schema/member/has_ast_node.rb +20 -0
- data/lib/graphql/schema/member/has_fields.rb +20 -10
- data/lib/graphql/schema/member/has_unresolved_type_error.rb +15 -0
- data/lib/graphql/schema/member/type_system_helpers.rb +2 -2
- data/lib/graphql/schema/member/validates_input.rb +33 -0
- data/lib/graphql/schema/member.rb +6 -0
- data/lib/graphql/schema/mutation.rb +5 -1
- data/lib/graphql/schema/non_null.rb +30 -0
- data/lib/graphql/schema/object.rb +65 -12
- data/lib/graphql/schema/possible_types.rb +9 -4
- data/lib/graphql/schema/printer.rb +0 -15
- data/lib/graphql/schema/relay_classic_mutation.rb +5 -3
- data/lib/graphql/schema/resolver/has_payload_type.rb +5 -2
- data/lib/graphql/schema/resolver.rb +26 -18
- data/lib/graphql/schema/scalar.rb +27 -3
- data/lib/graphql/schema/subscription.rb +8 -18
- data/lib/graphql/schema/timeout.rb +29 -15
- data/lib/graphql/schema/traversal.rb +1 -1
- data/lib/graphql/schema/type_expression.rb +21 -13
- data/lib/graphql/schema/type_membership.rb +2 -2
- data/lib/graphql/schema/union.rb +37 -3
- data/lib/graphql/schema/unique_within_type.rb +1 -2
- data/lib/graphql/schema/validation.rb +10 -2
- data/lib/graphql/schema/warden.rb +115 -29
- data/lib/graphql/schema.rb +903 -195
- data/lib/graphql/static_validation/all_rules.rb +1 -0
- data/lib/graphql/static_validation/base_visitor.rb +10 -6
- data/lib/graphql/static_validation/literal_validator.rb +52 -27
- data/lib/graphql/static_validation/rules/argument_literals_are_compatible.rb +43 -83
- data/lib/graphql/static_validation/rules/argument_literals_are_compatible_error.rb +17 -5
- data/lib/graphql/static_validation/rules/arguments_are_defined.rb +33 -25
- data/lib/graphql/static_validation/rules/directives_are_in_valid_locations.rb +1 -1
- data/lib/graphql/static_validation/rules/fields_are_defined_on_type.rb +4 -4
- data/lib/graphql/static_validation/rules/fields_have_appropriate_selections.rb +5 -5
- data/lib/graphql/static_validation/rules/fields_will_merge.rb +29 -21
- data/lib/graphql/static_validation/rules/fragment_spreads_are_possible.rb +3 -3
- data/lib/graphql/static_validation/rules/input_object_names_are_unique.rb +30 -0
- data/lib/graphql/static_validation/rules/input_object_names_are_unique_error.rb +30 -0
- data/lib/graphql/static_validation/rules/required_arguments_are_present.rb +2 -2
- data/lib/graphql/static_validation/rules/required_input_object_attributes_are_present.rb +4 -5
- data/lib/graphql/static_validation/rules/variable_default_values_are_correctly_typed.rb +12 -13
- data/lib/graphql/static_validation/rules/variable_usages_are_allowed.rb +5 -6
- data/lib/graphql/static_validation/rules/variables_are_input_types.rb +1 -1
- data/lib/graphql/static_validation/rules/variables_are_used_and_defined.rb +5 -3
- data/lib/graphql/static_validation/type_stack.rb +2 -2
- data/lib/graphql/static_validation/validation_context.rb +1 -1
- data/lib/graphql/static_validation/validation_timeout_error.rb +25 -0
- data/lib/graphql/static_validation/validator.rb +30 -8
- data/lib/graphql/static_validation.rb +1 -0
- data/lib/graphql/subscriptions/action_cable_subscriptions.rb +89 -19
- data/lib/graphql/subscriptions/broadcast_analyzer.rb +84 -0
- data/lib/graphql/subscriptions/default_subscription_resolve_extension.rb +21 -0
- data/lib/graphql/subscriptions/event.rb +23 -5
- data/lib/graphql/subscriptions/instrumentation.rb +10 -5
- data/lib/graphql/subscriptions/serialize.rb +22 -4
- data/lib/graphql/subscriptions/subscription_root.rb +15 -5
- data/lib/graphql/subscriptions.rb +108 -35
- data/lib/graphql/tracing/active_support_notifications_tracing.rb +14 -10
- data/lib/graphql/tracing/appoptics_tracing.rb +171 -0
- data/lib/graphql/tracing/appsignal_tracing.rb +8 -0
- data/lib/graphql/tracing/data_dog_tracing.rb +8 -0
- data/lib/graphql/tracing/new_relic_tracing.rb +9 -12
- data/lib/graphql/tracing/platform_tracing.rb +53 -9
- data/lib/graphql/tracing/prometheus_tracing/graphql_collector.rb +4 -1
- data/lib/graphql/tracing/prometheus_tracing.rb +8 -0
- data/lib/graphql/tracing/scout_tracing.rb +19 -0
- data/lib/graphql/tracing/skylight_tracing.rb +8 -0
- data/lib/graphql/tracing/statsd_tracing.rb +42 -0
- data/lib/graphql/tracing.rb +14 -34
- data/lib/graphql/types/big_int.rb +1 -1
- data/lib/graphql/types/int.rb +9 -2
- data/lib/graphql/types/iso_8601_date.rb +3 -3
- data/lib/graphql/types/iso_8601_date_time.rb +25 -10
- data/lib/graphql/types/relay/base_connection.rb +11 -7
- data/lib/graphql/types/relay/base_edge.rb +2 -1
- data/lib/graphql/types/string.rb +7 -1
- data/lib/graphql/unauthorized_error.rb +1 -1
- data/lib/graphql/union_type.rb +13 -28
- data/lib/graphql/unresolved_type_error.rb +2 -2
- data/lib/graphql/version.rb +1 -1
- data/lib/graphql.rb +31 -6
- data/readme.md +1 -1
- metadata +34 -9
- data/lib/graphql/literal_validation_error.rb +0 -6
@@ -1,118 +1,19 @@
|
|
1
1
|
# frozen_string_literal: true
|
2
2
|
module GraphQL
|
3
3
|
module Define
|
4
|
-
#
|
5
|
-
# {GraphQL::BaseType}, {GraphQL::Field} and others.
|
6
|
-
#
|
7
|
-
# Calling `.accepts_definitions(...)` creates:
|
8
|
-
#
|
9
|
-
# - a keyword to the `.define` method
|
10
|
-
# - a helper method in the `.define { ... }` block
|
11
|
-
#
|
12
|
-
# The `.define { ... }` block will be called lazily. To be sure it has been
|
13
|
-
# called, use the private method `#ensure_defined`. That will call the
|
14
|
-
# definition block if it hasn't been called already.
|
15
|
-
#
|
16
|
-
# The goals are:
|
17
|
-
#
|
18
|
-
# - Minimal overhead in consuming classes
|
19
|
-
# - Independence between consuming classes
|
20
|
-
# - Extendable by third-party libraries without monkey-patching or other nastiness
|
21
|
-
#
|
22
|
-
# @example Make a class definable
|
23
|
-
# class Car
|
24
|
-
# include GraphQL::Define::InstanceDefinable
|
25
|
-
# attr_accessor :make, :model, :doors
|
26
|
-
# accepts_definitions(
|
27
|
-
# # These attrs will be defined with plain setters, `{attr}=`
|
28
|
-
# :make, :model,
|
29
|
-
# # This attr has a custom definition which applies the config to the target
|
30
|
-
# doors: ->(car, doors_count) { doors_count.times { car.doors << Door.new } }
|
31
|
-
# )
|
32
|
-
# ensure_defined(:make, :model, :doors)
|
33
|
-
#
|
34
|
-
# def initialize
|
35
|
-
# @doors = []
|
36
|
-
# end
|
37
|
-
# end
|
38
|
-
#
|
39
|
-
# class Door; end;
|
40
|
-
#
|
41
|
-
# # Create an instance with `.define`:
|
42
|
-
# subaru_baja = Car.define do
|
43
|
-
# make "Subaru"
|
44
|
-
# model "Baja"
|
45
|
-
# doors 4
|
46
|
-
# end
|
47
|
-
#
|
48
|
-
# # The custom proc was applied:
|
49
|
-
# subaru_baja.doors #=> [<Door>, <Door>, <Door>, <Door>]
|
50
|
-
#
|
51
|
-
# @example Extending the definition of a class
|
52
|
-
# # Add some definitions:
|
53
|
-
# Car.accepts_definitions(all_wheel_drive: GraphQL::Define.assign_metadata_key(:all_wheel_drive))
|
54
|
-
#
|
55
|
-
# # Use it in a definition
|
56
|
-
# subaru_baja = Car.define do
|
57
|
-
# # ...
|
58
|
-
# all_wheel_drive true
|
59
|
-
# end
|
60
|
-
#
|
61
|
-
# # Access it from metadata
|
62
|
-
# subaru_baja.metadata[:all_wheel_drive] # => true
|
63
|
-
#
|
64
|
-
# @example Extending the definition of a class via a plugin
|
65
|
-
# # A plugin is any object that responds to `.use(definition)`
|
66
|
-
# module SubaruCar
|
67
|
-
# extend self
|
68
|
-
#
|
69
|
-
# def use(defn)
|
70
|
-
# # `defn` has the same methods as within `.define { ... }` block
|
71
|
-
# defn.make "Subaru"
|
72
|
-
# defn.doors 4
|
73
|
-
# end
|
74
|
-
# end
|
75
|
-
#
|
76
|
-
# # Use the plugin within a `.define { ... }` block
|
77
|
-
# subaru_baja = Car.define do
|
78
|
-
# use SubaruCar
|
79
|
-
# model 'Baja'
|
80
|
-
# end
|
81
|
-
#
|
82
|
-
# subaru_baja.make # => "Subaru"
|
83
|
-
# subaru_baja.doors # => [<Door>, <Door>, <Door>, <Door>]
|
84
|
-
#
|
85
|
-
# @example Making a copy with an extended definition
|
86
|
-
# # Create an instance with `.define`:
|
87
|
-
# subaru_baja = Car.define do
|
88
|
-
# make "Subaru"
|
89
|
-
# model "Baja"
|
90
|
-
# doors 4
|
91
|
-
# end
|
92
|
-
#
|
93
|
-
# # Then extend it with `#redefine`
|
94
|
-
# two_door_baja = subaru_baja.redefine do
|
95
|
-
# doors 2
|
96
|
-
# end
|
4
|
+
# @api deprecated
|
97
5
|
module InstanceDefinable
|
98
6
|
def self.included(base)
|
99
7
|
base.extend(ClassMethods)
|
100
8
|
base.ensure_defined(:metadata)
|
101
9
|
end
|
102
10
|
|
103
|
-
#
|
104
|
-
#
|
105
|
-
# @return [Hash<Object, Object>] Hash for user-defined storage
|
11
|
+
# @api deprecated
|
106
12
|
def metadata
|
107
13
|
@metadata ||= {}
|
108
14
|
end
|
109
15
|
|
110
|
-
#
|
111
|
-
# Keywords or helpers in the block correspond to keys given to `accepts_definitions`.
|
112
|
-
#
|
113
|
-
# Note that the block is not called right away -- instead, it's deferred until
|
114
|
-
# one of the defined fields is needed.
|
115
|
-
# @return [void]
|
16
|
+
# @api deprecated
|
116
17
|
def define(**kwargs, &block)
|
117
18
|
# make sure the previous definition_proc was executed:
|
118
19
|
ensure_defined
|
@@ -121,9 +22,7 @@ module GraphQL
|
|
121
22
|
nil
|
122
23
|
end
|
123
24
|
|
124
|
-
#
|
125
|
-
# @see {#define} for arguments
|
126
|
-
# @return [InstanceDefinable] A new instance, with any extended definitions
|
25
|
+
# @api deprecated
|
127
26
|
def redefine(**kwargs, &block)
|
128
27
|
ensure_defined
|
129
28
|
new_inst = self.dup
|
@@ -154,7 +53,12 @@ module GraphQL
|
|
154
53
|
defn_proxy = DefinedObjectProxy.new(self)
|
155
54
|
# Apply definition from `define(...)` kwargs
|
156
55
|
defn.define_keywords.each do |keyword, value|
|
157
|
-
|
56
|
+
# Don't splat string hashes, which blows up on Rubies before 2.7
|
57
|
+
if value.is_a?(Hash) && value.each_key.all? { |k| k.is_a?(Symbol) }
|
58
|
+
defn_proxy.public_send(keyword, **value)
|
59
|
+
else
|
60
|
+
defn_proxy.public_send(keyword, value)
|
61
|
+
end
|
158
62
|
end
|
159
63
|
# and/or apply definition from `define { ... }` block
|
160
64
|
if defn.define_proc
|
@@ -287,13 +191,19 @@ module GraphQL
|
|
287
191
|
end
|
288
192
|
|
289
193
|
class AssignAttribute
|
194
|
+
extend GraphQL::Ruby2Keywords
|
195
|
+
|
290
196
|
def initialize(attr_name)
|
291
197
|
@attr_assign_method = :"#{attr_name}="
|
292
198
|
end
|
293
199
|
|
294
|
-
|
295
|
-
|
200
|
+
# Even though we're just using the first value here,
|
201
|
+
# We have to add a splat here to use `ruby2_keywords`,
|
202
|
+
# so that it will accept a `[{}]` input from the caller.
|
203
|
+
def call(defn, *value)
|
204
|
+
defn.public_send(@attr_assign_method, value.first)
|
296
205
|
end
|
206
|
+
ruby2_keywords :call
|
297
207
|
end
|
298
208
|
end
|
299
209
|
end
|
@@ -1,13 +1,2 @@
|
|
1
1
|
# frozen_string_literal: true
|
2
|
-
GraphQL::Directive::DeprecatedDirective = GraphQL::Directive.
|
3
|
-
name "deprecated"
|
4
|
-
description "Marks an element of a GraphQL schema as no longer supported."
|
5
|
-
locations([GraphQL::Directive::FIELD_DEFINITION, GraphQL::Directive::ENUM_VALUE])
|
6
|
-
|
7
|
-
reason_description = "Explains why this element was deprecated, usually also including a "\
|
8
|
-
"suggestion for how to access supported similar data. Formatted "\
|
9
|
-
"in [Markdown](https://daringfireball.net/projects/markdown/)."
|
10
|
-
|
11
|
-
argument :reason, GraphQL::STRING_TYPE, reason_description, default_value: GraphQL::Directive::DEFAULT_DEPRECATION_REASON
|
12
|
-
default_directive true
|
13
|
-
end
|
2
|
+
GraphQL::Directive::DeprecatedDirective = GraphQL::Schema::Directive::Deprecated.graphql_definition
|
data/lib/graphql/directive.rb
CHANGED
@@ -45,7 +45,6 @@ module GraphQL
|
|
45
45
|
INPUT_FIELD_DEFINITION = :INPUT_FIELD_DEFINITION,
|
46
46
|
]
|
47
47
|
|
48
|
-
DEFAULT_DEPRECATION_REASON = 'No longer supported'
|
49
48
|
LOCATION_DESCRIPTIONS = {
|
50
49
|
QUERY: 'Location adjacent to a query operation.',
|
51
50
|
MUTATION: 'Location adjacent to a mutation operation.',
|
@@ -96,6 +95,14 @@ module GraphQL
|
|
96
95
|
def inspect
|
97
96
|
"#<GraphQL::Directive #{name}>"
|
98
97
|
end
|
98
|
+
|
99
|
+
def type_class
|
100
|
+
metadata[:type_class]
|
101
|
+
end
|
102
|
+
|
103
|
+
def get_argument(argument_name)
|
104
|
+
arguments[argument_name]
|
105
|
+
end
|
99
106
|
end
|
100
107
|
end
|
101
108
|
|
data/lib/graphql/enum_type.rb
CHANGED
@@ -1,76 +1,6 @@
|
|
1
1
|
# frozen_string_literal: true
|
2
2
|
module GraphQL
|
3
|
-
#
|
4
|
-
# By convention, enum names are `SCREAMING_CASE_NAMES`,
|
5
|
-
# but other identifiers are supported too.
|
6
|
-
#
|
7
|
-
# You can use as return types _or_ as inputs.
|
8
|
-
#
|
9
|
-
# By default, enums are passed to `resolve` functions as
|
10
|
-
# the strings that identify them, but you can provide a
|
11
|
-
# custom Ruby value with the `value:` keyword.
|
12
|
-
#
|
13
|
-
# @example An enum of programming languages
|
14
|
-
# LanguageEnum = GraphQL::EnumType.define do
|
15
|
-
# name "Language"
|
16
|
-
# description "Programming language for Web projects"
|
17
|
-
# value("PYTHON", "A dynamic, function-oriented language")
|
18
|
-
# value("RUBY", "A very dynamic language aimed at programmer happiness")
|
19
|
-
# value("JAVASCRIPT", "Accidental lingua franca of the web")
|
20
|
-
# end
|
21
|
-
#
|
22
|
-
# @example Using an enum as a return type
|
23
|
-
# field :favoriteLanguage, LanguageEnum, "This person's favorite coding language"
|
24
|
-
# # ...
|
25
|
-
# # In a query:
|
26
|
-
# Schema.execute("{ coder(id: 1) { favoriteLanguage } }")
|
27
|
-
# # { "data" => { "coder" => { "favoriteLanguage" => "RUBY" } } }
|
28
|
-
#
|
29
|
-
# @example Defining an enum input
|
30
|
-
# field :coders, types[CoderType] do
|
31
|
-
# argument :knowing, types[LanguageEnum]
|
32
|
-
# resolve ->(obj, args, ctx) {
|
33
|
-
# Coder.where(language: args[:knowing])
|
34
|
-
# }
|
35
|
-
# end
|
36
|
-
#
|
37
|
-
# @example Using an enum as input
|
38
|
-
# {
|
39
|
-
# # find coders who know Python and Ruby
|
40
|
-
# coders(knowing: [PYTHON, RUBY]) {
|
41
|
-
# name
|
42
|
-
# hourlyRate
|
43
|
-
# }
|
44
|
-
# }
|
45
|
-
#
|
46
|
-
# @example Enum whose values are different in Ruby-land
|
47
|
-
# GraphQL::EnumType.define do
|
48
|
-
# # ...
|
49
|
-
# # use the `value:` keyword:
|
50
|
-
# value("RUBY", "Lisp? Smalltalk?", value: :rb)
|
51
|
-
# end
|
52
|
-
#
|
53
|
-
# # Now, resolve functions will receive `:rb` instead of `"RUBY"`
|
54
|
-
# field :favoriteLanguage, LanguageEnum
|
55
|
-
# resolve ->(obj, args, ctx) {
|
56
|
-
# args[:favoriteLanguage] # => :rb
|
57
|
-
# }
|
58
|
-
#
|
59
|
-
# @example Enum whose values are different in ActiveRecord-land
|
60
|
-
# class Language < ActiveRecord::Base
|
61
|
-
# enum language: {
|
62
|
-
# rb: 0
|
63
|
-
# }
|
64
|
-
# end
|
65
|
-
#
|
66
|
-
# # Now enum type should be defined as
|
67
|
-
# GraphQL::EnumType.define do
|
68
|
-
# # ...
|
69
|
-
# # use the `value:` keyword:
|
70
|
-
# value("RUBY", "Lisp? Smalltalk?", value: 'rb')
|
71
|
-
# end
|
72
|
-
#
|
73
|
-
|
3
|
+
# @api deprecated
|
74
4
|
class EnumType < GraphQL::BaseType
|
75
5
|
accepts_definitions :values, value: GraphQL::Define::AssignEnumValue
|
76
6
|
ensure_defined(:values, :validate_non_null_input, :coerce_non_null_input, :coerce_result)
|
@@ -151,6 +81,10 @@ module GraphQL
|
|
151
81
|
def graphql_name
|
152
82
|
name
|
153
83
|
end
|
84
|
+
|
85
|
+
def type_class
|
86
|
+
metadata[:type_class]
|
87
|
+
end
|
154
88
|
end
|
155
89
|
|
156
90
|
class UnresolvedValueError < GraphQL::Error
|
@@ -18,12 +18,12 @@ module GraphQL
|
|
18
18
|
case name
|
19
19
|
when SKIP
|
20
20
|
args = query.arguments_for(directive_ast_node, directive_defn)
|
21
|
-
if args[
|
21
|
+
if args[:if] == true
|
22
22
|
return false
|
23
23
|
end
|
24
24
|
when INCLUDE
|
25
25
|
args = query.arguments_for(directive_ast_node, directive_defn)
|
26
|
-
if args[
|
26
|
+
if args[:if] == false
|
27
27
|
return false
|
28
28
|
end
|
29
29
|
else
|
@@ -18,8 +18,7 @@ module GraphQL
|
|
18
18
|
#
|
19
19
|
class Errors
|
20
20
|
def self.use(schema)
|
21
|
-
|
22
|
-
schema_class.error_handler = self.new(schema_class)
|
21
|
+
schema.error_handler = self.new(schema)
|
23
22
|
end
|
24
23
|
|
25
24
|
def initialize(schema)
|
@@ -41,7 +40,7 @@ module GraphQL
|
|
41
40
|
def with_error_handling(ctx)
|
42
41
|
yield
|
43
42
|
rescue StandardError => err
|
44
|
-
rescues =
|
43
|
+
rescues = ctx.schema.rescues
|
45
44
|
_err_class, handler = rescues.find { |err_class, handler| err.is_a?(err_class) }
|
46
45
|
if handler
|
47
46
|
runtime_info = ctx.namespace(:interpreter) || {}
|
@@ -20,7 +20,7 @@ module GraphQL
|
|
20
20
|
|
21
21
|
def execute(ast_operation, root_type, query)
|
22
22
|
result = resolve_root_selection(query)
|
23
|
-
lazy_resolve_root_selection(result, {query: query})
|
23
|
+
lazy_resolve_root_selection(result, **{query: query})
|
24
24
|
GraphQL::Execution::Flatten.call(query.context)
|
25
25
|
end
|
26
26
|
|
@@ -77,7 +77,7 @@ module GraphQL
|
|
77
77
|
end
|
78
78
|
|
79
79
|
def call_after_hooks(instrumenters, object, after_hook_name, ex)
|
80
|
-
instrumenters.
|
80
|
+
instrumenters.reverse_each do |instrumenter|
|
81
81
|
begin
|
82
82
|
instrumenter.public_send(after_hook_name, object)
|
83
83
|
rescue => e
|
@@ -0,0 +1,28 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
|
3
|
+
module GraphQL
|
4
|
+
module Execution
|
5
|
+
class Interpreter
|
6
|
+
# A container for metadata regarding arguments present in a GraphQL query.
|
7
|
+
# @see Interpreter::Arguments#argument_values for a hash of these objects.
|
8
|
+
class ArgumentValue
|
9
|
+
def initialize(definition:, value:, default_used:)
|
10
|
+
@definition = definition
|
11
|
+
@value = value
|
12
|
+
@default_used = default_used
|
13
|
+
end
|
14
|
+
|
15
|
+
# @return [Object] The Ruby-ready value for this Argument
|
16
|
+
attr_reader :value
|
17
|
+
|
18
|
+
# @return [GraphQL::Schema::Argument] The definition instance for this argument
|
19
|
+
attr_reader :definition
|
20
|
+
|
21
|
+
# @return [Boolean] `true` if the schema-defined `default_value:` was applied in this case. (No client-provided value was present.)
|
22
|
+
def default_used?
|
23
|
+
@default_used
|
24
|
+
end
|
25
|
+
end
|
26
|
+
end
|
27
|
+
end
|
28
|
+
end
|
@@ -0,0 +1,51 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
|
3
|
+
module GraphQL
|
4
|
+
module Execution
|
5
|
+
class Interpreter
|
6
|
+
# A wrapper for argument hashes in GraphQL queries.
|
7
|
+
#
|
8
|
+
# @see GraphQL::Query#arguments_for to get access to these objects.
|
9
|
+
class Arguments
|
10
|
+
extend Forwardable
|
11
|
+
include GraphQL::Dig
|
12
|
+
|
13
|
+
# The Ruby-style arguments hash, ready for a resolver.
|
14
|
+
# This hash is the one used at runtime.
|
15
|
+
#
|
16
|
+
# @return [Hash<Symbol, Object>]
|
17
|
+
def keyword_arguments
|
18
|
+
@keyword_arguments ||= begin
|
19
|
+
kwargs = {}
|
20
|
+
argument_values.each do |name, arg_val|
|
21
|
+
kwargs[name] = arg_val.value
|
22
|
+
end
|
23
|
+
kwargs
|
24
|
+
end
|
25
|
+
end
|
26
|
+
|
27
|
+
# @param argument_values [nil, Hash{Symbol => ArgumentValue}]
|
28
|
+
def initialize(argument_values:)
|
29
|
+
@argument_values = argument_values
|
30
|
+
@empty = argument_values.nil? || argument_values.empty?
|
31
|
+
end
|
32
|
+
|
33
|
+
# @return [Hash{Symbol => ArgumentValue}]
|
34
|
+
def argument_values
|
35
|
+
@argument_values ||= {}
|
36
|
+
end
|
37
|
+
|
38
|
+
def empty?
|
39
|
+
@empty
|
40
|
+
end
|
41
|
+
|
42
|
+
def_delegators :keyword_arguments, :key?, :[], :fetch, :keys, :each, :values
|
43
|
+
def_delegators :argument_values, :each_value
|
44
|
+
|
45
|
+
def inspect
|
46
|
+
"#<#{self.class} @keyword_arguments=#{keyword_arguments.inspect}>"
|
47
|
+
end
|
48
|
+
end
|
49
|
+
end
|
50
|
+
end
|
51
|
+
end
|
@@ -0,0 +1,79 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
|
3
|
+
module GraphQL
|
4
|
+
module Execution
|
5
|
+
class Interpreter
|
6
|
+
class ArgumentsCache
|
7
|
+
def initialize(query)
|
8
|
+
@query = query
|
9
|
+
@storage = Hash.new do |h, ast_node|
|
10
|
+
h[ast_node] = Hash.new do |h2, arg_owner|
|
11
|
+
h2[arg_owner] = Hash.new do |h3, parent_object|
|
12
|
+
# First, normalize all AST or Ruby values to a plain Ruby hash
|
13
|
+
args_hash = prepare_args_hash(ast_node)
|
14
|
+
# Then call into the schema to coerce those incoming values
|
15
|
+
args = arg_owner.coerce_arguments(parent_object, args_hash, query.context)
|
16
|
+
|
17
|
+
h3[parent_object] = @query.schema.after_lazy(args) do |resolved_args|
|
18
|
+
# when this promise is resolved, update the cache with the resolved value
|
19
|
+
h3[parent_object] = resolved_args
|
20
|
+
end
|
21
|
+
end
|
22
|
+
end
|
23
|
+
end
|
24
|
+
end
|
25
|
+
|
26
|
+
def fetch(ast_node, argument_owner, parent_object)
|
27
|
+
@storage[ast_node][argument_owner][parent_object]
|
28
|
+
end
|
29
|
+
|
30
|
+
private
|
31
|
+
|
32
|
+
NO_ARGUMENTS = {}.freeze
|
33
|
+
|
34
|
+
NO_VALUE_GIVEN = Object.new
|
35
|
+
|
36
|
+
def prepare_args_hash(ast_arg_or_hash_or_value)
|
37
|
+
case ast_arg_or_hash_or_value
|
38
|
+
when Hash
|
39
|
+
if ast_arg_or_hash_or_value.empty?
|
40
|
+
return NO_ARGUMENTS
|
41
|
+
end
|
42
|
+
args_hash = {}
|
43
|
+
ast_arg_or_hash_or_value.each do |k, v|
|
44
|
+
args_hash[k] = prepare_args_hash(v)
|
45
|
+
end
|
46
|
+
args_hash
|
47
|
+
when Array
|
48
|
+
ast_arg_or_hash_or_value.map { |v| prepare_args_hash(v) }
|
49
|
+
when GraphQL::Language::Nodes::Field, GraphQL::Language::Nodes::InputObject, GraphQL::Language::Nodes::Directive
|
50
|
+
if ast_arg_or_hash_or_value.arguments.empty?
|
51
|
+
return NO_ARGUMENTS
|
52
|
+
end
|
53
|
+
args_hash = {}
|
54
|
+
ast_arg_or_hash_or_value.arguments.each do |arg|
|
55
|
+
v = prepare_args_hash(arg.value)
|
56
|
+
if v != NO_VALUE_GIVEN
|
57
|
+
args_hash[arg.name] = v
|
58
|
+
end
|
59
|
+
end
|
60
|
+
args_hash
|
61
|
+
when GraphQL::Language::Nodes::VariableIdentifier
|
62
|
+
if @query.variables.key?(ast_arg_or_hash_or_value.name)
|
63
|
+
variable_value = @query.variables[ast_arg_or_hash_or_value.name]
|
64
|
+
prepare_args_hash(variable_value)
|
65
|
+
else
|
66
|
+
NO_VALUE_GIVEN
|
67
|
+
end
|
68
|
+
when GraphQL::Language::Nodes::Enum
|
69
|
+
ast_arg_or_hash_or_value.name
|
70
|
+
when GraphQL::Language::Nodes::NullValue
|
71
|
+
nil
|
72
|
+
else
|
73
|
+
ast_arg_or_hash_or_value
|
74
|
+
end
|
75
|
+
end
|
76
|
+
end
|
77
|
+
end
|
78
|
+
end
|
79
|
+
end
|
@@ -0,0 +1,25 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
|
3
|
+
module GraphQL
|
4
|
+
module Execution
|
5
|
+
class Interpreter
|
6
|
+
# Wrapper for raw values
|
7
|
+
class RawValue
|
8
|
+
def initialize(obj = nil)
|
9
|
+
@object = obj
|
10
|
+
end
|
11
|
+
|
12
|
+
def resolve
|
13
|
+
@object
|
14
|
+
end
|
15
|
+
end
|
16
|
+
|
17
|
+
# Allows to return "raw" value from the resolver
|
18
|
+
module HandlesRawValue
|
19
|
+
def raw_value(obj)
|
20
|
+
RawValue.new(obj)
|
21
|
+
end
|
22
|
+
end
|
23
|
+
end
|
24
|
+
end
|
25
|
+
end
|