graphql 1.13.25 → 2.0.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/graphql/analysis/ast/query_complexity.rb +1 -1
- data/lib/graphql/analysis/ast/query_depth.rb +0 -1
- data/lib/graphql/analysis/ast/visitor.rb +1 -1
- data/lib/graphql/analysis/ast.rb +0 -10
- data/lib/graphql/analysis.rb +0 -7
- data/lib/graphql/backtrace/table.rb +0 -18
- data/lib/graphql/backtrace/tracer.rb +1 -2
- data/lib/graphql/backtrace.rb +2 -8
- data/lib/graphql/dig.rb +1 -1
- data/lib/graphql/execution/errors.rb +1 -9
- data/lib/graphql/execution/interpreter/runtime.rb +6 -13
- data/lib/graphql/execution/interpreter.rb +0 -22
- data/lib/graphql/execution/lazy.rb +1 -1
- data/lib/graphql/execution/lookahead.rb +6 -13
- data/lib/graphql/execution/multiplex.rb +50 -107
- data/lib/graphql/execution.rb +11 -3
- data/lib/graphql/introspection/directive_type.rb +2 -2
- data/lib/graphql/introspection/dynamic_fields.rb +3 -8
- data/lib/graphql/introspection/entry_points.rb +2 -15
- data/lib/graphql/introspection/field_type.rb +1 -1
- data/lib/graphql/introspection/schema_type.rb +2 -2
- data/lib/graphql/introspection/type_type.rb +5 -5
- data/lib/graphql/language/document_from_schema_definition.rb +0 -17
- data/lib/graphql/language/nodes.rb +0 -3
- data/lib/graphql/pagination/connections.rb +2 -28
- data/lib/graphql/pagination/relation_connection.rb +0 -2
- data/lib/graphql/query/context.rb +1 -185
- data/lib/graphql/query/input_validation_result.rb +0 -9
- data/lib/graphql/query/literal_input.rb +8 -13
- data/lib/graphql/query/validation_pipeline.rb +6 -34
- data/lib/graphql/query/variable_validation_error.rb +2 -2
- data/lib/graphql/query/variables.rb +8 -31
- data/lib/graphql/query.rb +5 -34
- data/lib/graphql/railtie.rb +0 -104
- data/lib/graphql/relay/range_add.rb +0 -4
- data/lib/graphql/relay.rb +0 -15
- data/lib/graphql/schema/addition.rb +1 -8
- data/lib/graphql/schema/argument.rb +6 -28
- data/lib/graphql/schema/build_from_definition.rb +7 -9
- data/lib/graphql/schema/directive.rb +1 -22
- data/lib/graphql/schema/enum.rb +3 -19
- data/lib/graphql/schema/enum_value.rb +1 -23
- data/lib/graphql/schema/field.rb +22 -221
- data/lib/graphql/schema/input_object.rb +17 -65
- data/lib/graphql/schema/interface.rb +1 -30
- data/lib/graphql/schema/introspection_system.rb +3 -8
- data/lib/graphql/schema/late_bound_type.rb +2 -2
- data/lib/graphql/schema/list.rb +3 -24
- data/lib/graphql/schema/loader.rb +0 -1
- data/lib/graphql/schema/member/base_dsl_methods.rb +1 -6
- data/lib/graphql/schema/member/build_type.rb +4 -6
- data/lib/graphql/schema/member/has_arguments.rb +16 -20
- data/lib/graphql/schema/member/has_fields.rb +3 -3
- data/lib/graphql/schema/member/has_interfaces.rb +1 -13
- data/lib/graphql/schema/member/validates_input.rb +2 -2
- data/lib/graphql/schema/member.rb +0 -6
- data/lib/graphql/schema/mutation.rb +0 -9
- data/lib/graphql/schema/non_null.rb +3 -9
- data/lib/graphql/schema/object.rb +0 -40
- data/lib/graphql/schema/relay_classic_mutation.rb +17 -28
- data/lib/graphql/schema/scalar.rb +1 -16
- data/lib/graphql/schema/union.rb +0 -16
- data/lib/graphql/schema/warden.rb +3 -12
- data/lib/graphql/schema/wrapper.rb +0 -5
- data/lib/graphql/schema.rb +106 -945
- data/lib/graphql/static_validation/base_visitor.rb +4 -21
- data/lib/graphql/static_validation/rules/directives_are_in_valid_locations.rb +12 -12
- data/lib/graphql/static_validation/validator.rb +2 -24
- data/lib/graphql/static_validation.rb +0 -2
- data/lib/graphql/subscriptions/default_subscription_resolve_extension.rb +38 -1
- data/lib/graphql/subscriptions/event.rb +1 -1
- data/lib/graphql/subscriptions/instrumentation.rb +0 -51
- data/lib/graphql/subscriptions.rb +4 -13
- data/lib/graphql/tracing/data_dog_tracing.rb +16 -20
- data/lib/graphql/tracing/platform_tracing.rb +4 -32
- data/lib/graphql/tracing.rb +0 -1
- data/lib/graphql/types/relay/connection_behaviors.rb +2 -6
- data/lib/graphql/types/relay/default_relay.rb +0 -10
- data/lib/graphql/types/relay/node_behaviors.rb +5 -1
- data/lib/graphql/types/relay.rb +0 -2
- data/lib/graphql/types/string.rb +1 -1
- data/lib/graphql/version.rb +1 -1
- data/lib/graphql.rb +1 -66
- metadata +28 -164
- data/lib/graphql/analysis/analyze_query.rb +0 -98
- data/lib/graphql/analysis/field_usage.rb +0 -45
- data/lib/graphql/analysis/max_query_complexity.rb +0 -26
- data/lib/graphql/analysis/max_query_depth.rb +0 -26
- data/lib/graphql/analysis/query_complexity.rb +0 -88
- data/lib/graphql/analysis/query_depth.rb +0 -43
- data/lib/graphql/analysis/reducer_state.rb +0 -48
- data/lib/graphql/argument.rb +0 -131
- data/lib/graphql/authorization.rb +0 -82
- data/lib/graphql/backtrace/legacy_tracer.rb +0 -56
- data/lib/graphql/backwards_compatibility.rb +0 -61
- data/lib/graphql/base_type.rb +0 -232
- data/lib/graphql/boolean_type.rb +0 -2
- data/lib/graphql/compatibility/execution_specification/counter_schema.rb +0 -53
- data/lib/graphql/compatibility/execution_specification/specification_schema.rb +0 -200
- data/lib/graphql/compatibility/execution_specification.rb +0 -436
- data/lib/graphql/compatibility/lazy_execution_specification/lazy_schema.rb +0 -111
- data/lib/graphql/compatibility/lazy_execution_specification.rb +0 -215
- data/lib/graphql/compatibility/query_parser_specification/parse_error_specification.rb +0 -87
- data/lib/graphql/compatibility/query_parser_specification/query_assertions.rb +0 -79
- data/lib/graphql/compatibility/query_parser_specification.rb +0 -266
- data/lib/graphql/compatibility/schema_parser_specification.rb +0 -682
- data/lib/graphql/compatibility.rb +0 -5
- data/lib/graphql/define/assign_argument.rb +0 -12
- data/lib/graphql/define/assign_connection.rb +0 -13
- data/lib/graphql/define/assign_enum_value.rb +0 -18
- data/lib/graphql/define/assign_global_id_field.rb +0 -11
- data/lib/graphql/define/assign_mutation_function.rb +0 -34
- data/lib/graphql/define/assign_object_field.rb +0 -42
- data/lib/graphql/define/defined_object_proxy.rb +0 -53
- data/lib/graphql/define/instance_definable.rb +0 -255
- data/lib/graphql/define/no_definition_error.rb +0 -7
- data/lib/graphql/define/non_null_with_bang.rb +0 -16
- data/lib/graphql/define/type_definer.rb +0 -31
- data/lib/graphql/define.rb +0 -31
- data/lib/graphql/deprecated_dsl.rb +0 -55
- data/lib/graphql/directive/deprecated_directive.rb +0 -2
- data/lib/graphql/directive/include_directive.rb +0 -2
- data/lib/graphql/directive/skip_directive.rb +0 -2
- data/lib/graphql/directive.rb +0 -107
- data/lib/graphql/enum_type.rb +0 -133
- data/lib/graphql/execution/execute.rb +0 -333
- data/lib/graphql/execution/flatten.rb +0 -40
- data/lib/graphql/execution/typecast.rb +0 -50
- data/lib/graphql/field/resolve.rb +0 -59
- data/lib/graphql/field.rb +0 -226
- data/lib/graphql/float_type.rb +0 -2
- data/lib/graphql/function.rb +0 -128
- data/lib/graphql/id_type.rb +0 -2
- data/lib/graphql/input_object_type.rb +0 -138
- data/lib/graphql/int_type.rb +0 -2
- data/lib/graphql/interface_type.rb +0 -72
- data/lib/graphql/internal_representation/document.rb +0 -27
- data/lib/graphql/internal_representation/node.rb +0 -206
- data/lib/graphql/internal_representation/print.rb +0 -51
- data/lib/graphql/internal_representation/rewrite.rb +0 -184
- data/lib/graphql/internal_representation/scope.rb +0 -88
- data/lib/graphql/internal_representation/visit.rb +0 -36
- data/lib/graphql/internal_representation.rb +0 -7
- data/lib/graphql/list_type.rb +0 -80
- data/lib/graphql/non_null_type.rb +0 -71
- data/lib/graphql/object_type.rb +0 -130
- data/lib/graphql/query/arguments.rb +0 -189
- data/lib/graphql/query/arguments_cache.rb +0 -24
- data/lib/graphql/query/executor.rb +0 -52
- data/lib/graphql/query/serial_execution/field_resolution.rb +0 -92
- data/lib/graphql/query/serial_execution/operation_resolution.rb +0 -19
- data/lib/graphql/query/serial_execution/selection_resolution.rb +0 -23
- data/lib/graphql/query/serial_execution/value_resolution.rb +0 -87
- data/lib/graphql/query/serial_execution.rb +0 -40
- data/lib/graphql/relay/array_connection.rb +0 -83
- data/lib/graphql/relay/base_connection.rb +0 -189
- data/lib/graphql/relay/connection_instrumentation.rb +0 -54
- data/lib/graphql/relay/connection_resolve.rb +0 -43
- data/lib/graphql/relay/connection_type.rb +0 -54
- data/lib/graphql/relay/edge.rb +0 -27
- data/lib/graphql/relay/edge_type.rb +0 -19
- data/lib/graphql/relay/edges_instrumentation.rb +0 -39
- data/lib/graphql/relay/global_id_resolve.rb +0 -17
- data/lib/graphql/relay/mongo_relation_connection.rb +0 -50
- data/lib/graphql/relay/mutation/instrumentation.rb +0 -23
- data/lib/graphql/relay/mutation/resolve.rb +0 -56
- data/lib/graphql/relay/mutation/result.rb +0 -38
- data/lib/graphql/relay/mutation.rb +0 -106
- data/lib/graphql/relay/node.rb +0 -39
- data/lib/graphql/relay/page_info.rb +0 -7
- data/lib/graphql/relay/relation_connection.rb +0 -188
- data/lib/graphql/relay/type_extensions.rb +0 -32
- data/lib/graphql/scalar_type.rb +0 -91
- data/lib/graphql/schema/catchall_middleware.rb +0 -35
- data/lib/graphql/schema/default_parse_error.rb +0 -10
- data/lib/graphql/schema/default_type_error.rb +0 -17
- data/lib/graphql/schema/member/accepts_definition.rb +0 -164
- data/lib/graphql/schema/member/cached_graphql_definition.rb +0 -58
- data/lib/graphql/schema/member/instrumentation.rb +0 -131
- data/lib/graphql/schema/middleware_chain.rb +0 -82
- data/lib/graphql/schema/possible_types.rb +0 -44
- data/lib/graphql/schema/rescue_middleware.rb +0 -60
- data/lib/graphql/schema/timeout_middleware.rb +0 -88
- data/lib/graphql/schema/traversal.rb +0 -228
- data/lib/graphql/schema/validation.rb +0 -313
- data/lib/graphql/static_validation/default_visitor.rb +0 -15
- data/lib/graphql/static_validation/no_validate_visitor.rb +0 -10
- data/lib/graphql/string_type.rb +0 -2
- data/lib/graphql/subscriptions/subscription_root.rb +0 -76
- data/lib/graphql/tracing/skylight_tracing.rb +0 -70
- data/lib/graphql/types/relay/node_field.rb +0 -24
- data/lib/graphql/types/relay/nodes_field.rb +0 -43
- data/lib/graphql/union_type.rb +0 -115
- data/lib/graphql/upgrader/member.rb +0 -937
- data/lib/graphql/upgrader/schema.rb +0 -38
@@ -1,206 +0,0 @@
|
|
1
|
-
# frozen_string_literal: true
|
2
|
-
module GraphQL
|
3
|
-
module InternalRepresentation
|
4
|
-
class Node
|
5
|
-
# @api private
|
6
|
-
DEFAULT_TYPED_CHILDREN = Proc.new { |h, k| h[k] = {} }
|
7
|
-
|
8
|
-
# A specialized, reusable object for leaf nodes.
|
9
|
-
NO_TYPED_CHILDREN = Hash.new({}.freeze)
|
10
|
-
def NO_TYPED_CHILDREN.dup; self; end;
|
11
|
-
NO_TYPED_CHILDREN.freeze
|
12
|
-
|
13
|
-
# @return [String] the name this node has in the response
|
14
|
-
attr_reader :name
|
15
|
-
|
16
|
-
# @return [GraphQL::ObjectType]
|
17
|
-
attr_reader :owner_type
|
18
|
-
|
19
|
-
# Each key is a {GraphQL::ObjectType} which this selection _may_ be made on.
|
20
|
-
# The values for that key are selections which apply to that type.
|
21
|
-
#
|
22
|
-
# This value is derived from {#scoped_children} after the rewrite is finished.
|
23
|
-
# @return [Hash<GraphQL::ObjectType, Hash<String => Node>>]
|
24
|
-
def typed_children
|
25
|
-
@typed_children ||= begin
|
26
|
-
if @scoped_children.any?
|
27
|
-
new_tc = Hash.new(&DEFAULT_TYPED_CHILDREN)
|
28
|
-
all_object_types = Set.new
|
29
|
-
scoped_children.each_key { |t| all_object_types.merge(@query.possible_types(t)) }
|
30
|
-
# Remove any scoped children which don't follow this return type
|
31
|
-
# (This can happen with fragment merging where lexical scope is lost)
|
32
|
-
all_object_types &= @query.possible_types(@return_type.unwrap)
|
33
|
-
all_object_types.each do |t|
|
34
|
-
new_tc[t] = get_typed_children(t)
|
35
|
-
end
|
36
|
-
new_tc
|
37
|
-
else
|
38
|
-
NO_TYPED_CHILDREN
|
39
|
-
end
|
40
|
-
|
41
|
-
end
|
42
|
-
end
|
43
|
-
|
44
|
-
# These children correspond closely to scopes in the AST.
|
45
|
-
# Keys _may_ be abstract types. They're assumed to be read-only after rewrite is finished
|
46
|
-
# because {#typed_children} is derived from them.
|
47
|
-
#
|
48
|
-
# Using {#scoped_children} during the rewrite step reduces the overhead of reifying
|
49
|
-
# abstract types because they're only reified _after_ the rewrite.
|
50
|
-
# @return [Hash<GraphQL::BaseType, Hash<String => Node>>]
|
51
|
-
attr_reader :scoped_children
|
52
|
-
|
53
|
-
# @return [Array<Language::Nodes::AbstractNode>] AST nodes which are represented by this node
|
54
|
-
attr_reader :ast_nodes
|
55
|
-
|
56
|
-
# @return [Array<GraphQL::Field>] Field definitions for this node (there should only be one!)
|
57
|
-
attr_reader :definitions
|
58
|
-
|
59
|
-
# @return [GraphQL::BaseType] The expected wrapped type this node must return.
|
60
|
-
attr_reader :return_type
|
61
|
-
|
62
|
-
# @return [InternalRepresentation::Node, nil]
|
63
|
-
attr_reader :parent
|
64
|
-
|
65
|
-
def initialize(
|
66
|
-
name:, owner_type:, query:, return_type:, parent:,
|
67
|
-
ast_nodes: [],
|
68
|
-
definitions: []
|
69
|
-
)
|
70
|
-
@name = name
|
71
|
-
@query = query
|
72
|
-
@owner_type = owner_type
|
73
|
-
@parent = parent
|
74
|
-
@typed_children = nil
|
75
|
-
@scoped_children = Hash.new { |h1, k1| h1[k1] = {} }
|
76
|
-
@ast_nodes = ast_nodes
|
77
|
-
@definitions = definitions
|
78
|
-
@return_type = return_type
|
79
|
-
end
|
80
|
-
|
81
|
-
def initialize_copy(other_node)
|
82
|
-
super
|
83
|
-
# Bust some caches:
|
84
|
-
@typed_children = nil
|
85
|
-
@definition = nil
|
86
|
-
@definition_name = nil
|
87
|
-
@ast_node = nil
|
88
|
-
# Shallow-copy some state:
|
89
|
-
@scoped_children = other_node.scoped_children.dup
|
90
|
-
@ast_nodes = other_node.ast_nodes.dup
|
91
|
-
@definitions = other_node.definitions.dup
|
92
|
-
end
|
93
|
-
|
94
|
-
def ==(other)
|
95
|
-
other.is_a?(self.class) &&
|
96
|
-
other.name == name &&
|
97
|
-
other.parent == parent &&
|
98
|
-
other.return_type == return_type &&
|
99
|
-
other.owner_type == owner_type &&
|
100
|
-
other.scoped_children == scoped_children &&
|
101
|
-
other.definitions == definitions &&
|
102
|
-
other.ast_nodes == ast_nodes
|
103
|
-
end
|
104
|
-
|
105
|
-
def definition_name
|
106
|
-
definition && definition.name
|
107
|
-
end
|
108
|
-
|
109
|
-
def arguments
|
110
|
-
@query.arguments_for(self, definition)
|
111
|
-
end
|
112
|
-
|
113
|
-
def definition
|
114
|
-
@definition ||= begin
|
115
|
-
first_def = @definitions.first
|
116
|
-
first_def && @query.get_field(@owner_type, first_def.name)
|
117
|
-
end
|
118
|
-
end
|
119
|
-
|
120
|
-
def ast_node
|
121
|
-
@ast_node ||= ast_nodes.first
|
122
|
-
end
|
123
|
-
|
124
|
-
def inspect
|
125
|
-
all_children_names = scoped_children.values.map(&:keys).flatten.uniq.join(", ")
|
126
|
-
all_locations = ast_nodes.map {|n| "#{n.line}:#{n.col}" }.join(", ")
|
127
|
-
"#<Node #{@owner_type}.#{@name} -> #{@return_type} {#{all_children_names}} @ [#{all_locations}] #{object_id}>"
|
128
|
-
end
|
129
|
-
|
130
|
-
# Merge selections from `new_parent` into `self`.
|
131
|
-
# Selections are merged in place, not copied.
|
132
|
-
def deep_merge_node(new_parent, scope: nil, merge_self: true)
|
133
|
-
if merge_self
|
134
|
-
@ast_nodes |= new_parent.ast_nodes
|
135
|
-
@definitions |= new_parent.definitions
|
136
|
-
end
|
137
|
-
new_sc = new_parent.scoped_children
|
138
|
-
if new_sc.any?
|
139
|
-
scope ||= Scope.new(@query, @return_type.unwrap)
|
140
|
-
new_sc.each do |obj_type, new_fields|
|
141
|
-
inner_scope = scope.enter(obj_type)
|
142
|
-
inner_scope.each do |scoped_type|
|
143
|
-
prev_fields = @scoped_children[scoped_type]
|
144
|
-
new_fields.each do |name, new_node|
|
145
|
-
prev_node = prev_fields[name]
|
146
|
-
if prev_node
|
147
|
-
prev_node.deep_merge_node(new_node)
|
148
|
-
else
|
149
|
-
prev_fields[name] = new_node
|
150
|
-
end
|
151
|
-
end
|
152
|
-
end
|
153
|
-
end
|
154
|
-
end
|
155
|
-
end
|
156
|
-
|
157
|
-
# @return [GraphQL::Query]
|
158
|
-
attr_reader :query
|
159
|
-
|
160
|
-
def subscription_topic
|
161
|
-
@subscription_topic ||= begin
|
162
|
-
scope = if definition.subscription_scope
|
163
|
-
@query.context[definition.subscription_scope]
|
164
|
-
else
|
165
|
-
nil
|
166
|
-
end
|
167
|
-
Subscriptions::Event.serialize(
|
168
|
-
definition_name,
|
169
|
-
@query.arguments_for(self, definition),
|
170
|
-
definition,
|
171
|
-
scope: scope
|
172
|
-
)
|
173
|
-
end
|
174
|
-
end
|
175
|
-
|
176
|
-
protected
|
177
|
-
|
178
|
-
attr_writer :owner_type, :parent
|
179
|
-
|
180
|
-
private
|
181
|
-
|
182
|
-
# Get applicable children from {#scoped_children}
|
183
|
-
# @param obj_type [GraphQL::ObjectType]
|
184
|
-
# @return [Hash<String => Node>]
|
185
|
-
def get_typed_children(obj_type)
|
186
|
-
new_tc = {}
|
187
|
-
@scoped_children.each do |scope_type, scope_nodes|
|
188
|
-
if GraphQL::Execution::Typecast.subtype?(scope_type, obj_type)
|
189
|
-
scope_nodes.each do |name, new_node|
|
190
|
-
prev_node = new_tc[name]
|
191
|
-
if prev_node
|
192
|
-
prev_node.deep_merge_node(new_node)
|
193
|
-
else
|
194
|
-
copied_node = new_node.dup
|
195
|
-
copied_node.owner_type = obj_type
|
196
|
-
copied_node.parent = self
|
197
|
-
new_tc[name] = copied_node
|
198
|
-
end
|
199
|
-
end
|
200
|
-
end
|
201
|
-
end
|
202
|
-
new_tc
|
203
|
-
end
|
204
|
-
end
|
205
|
-
end
|
206
|
-
end
|
@@ -1,51 +0,0 @@
|
|
1
|
-
# frozen_string_literal: true
|
2
|
-
module GraphQL
|
3
|
-
module InternalRepresentation
|
4
|
-
module Print
|
5
|
-
module_function
|
6
|
-
|
7
|
-
def print(schema, query_string)
|
8
|
-
query = GraphQL::Query.new(schema, query_string)
|
9
|
-
print_node(query.irep_selection)
|
10
|
-
end
|
11
|
-
|
12
|
-
def print_node(node, indent: 0)
|
13
|
-
padding = " " * indent
|
14
|
-
typed_children_padding = " " * (indent + 2)
|
15
|
-
query_str = "".dup
|
16
|
-
|
17
|
-
if !node.definition
|
18
|
-
op_node = node.ast_node
|
19
|
-
name = op_node.name ? " " + op_node.name : ""
|
20
|
-
op_type = op_node.operation_type
|
21
|
-
query_str << "#{op_type}#{name}"
|
22
|
-
else
|
23
|
-
if node.name == node.definition_name
|
24
|
-
query_str << "#{padding}#{node.name}"
|
25
|
-
else
|
26
|
-
query_str << "#{padding}#{node.name}: #{node.definition_name}"
|
27
|
-
end
|
28
|
-
|
29
|
-
args = node.ast_nodes.map { |n| n.arguments.map(&:to_query_string).join(",") }.uniq
|
30
|
-
query_str << args.map { |a| "(#{a})"}.join("|")
|
31
|
-
end
|
32
|
-
|
33
|
-
if node.typed_children.any?
|
34
|
-
query_str << " {\n"
|
35
|
-
node.typed_children.each do |type, children|
|
36
|
-
query_str << "#{typed_children_padding}... on #{type.name} {\n"
|
37
|
-
children.each do |name, child|
|
38
|
-
query_str << print_node(child, indent: indent + 4)
|
39
|
-
end
|
40
|
-
query_str << "#{typed_children_padding}}\n"
|
41
|
-
end
|
42
|
-
query_str << "#{padding}}\n"
|
43
|
-
else
|
44
|
-
query_str << "\n"
|
45
|
-
end
|
46
|
-
|
47
|
-
query_str
|
48
|
-
end
|
49
|
-
end
|
50
|
-
end
|
51
|
-
end
|
@@ -1,184 +0,0 @@
|
|
1
|
-
# frozen_string_literal: true
|
2
|
-
module GraphQL
|
3
|
-
module InternalRepresentation
|
4
|
-
# While visiting an AST, build a normalized, flattened tree of {InternalRepresentation::Node}s.
|
5
|
-
#
|
6
|
-
# No unions or interfaces are present in this tree, only object types.
|
7
|
-
#
|
8
|
-
# Selections from the AST are attached to the object types they apply to.
|
9
|
-
#
|
10
|
-
# Inline fragments and fragment spreads are preserved in {InternalRepresentation::Node#ast_spreads},
|
11
|
-
# where they can be used to check for the presence of directives. This might not be sufficient
|
12
|
-
# for future directives, since the selections' grouping is lost.
|
13
|
-
#
|
14
|
-
# The rewritten query tree serves as the basis for the `FieldsWillMerge` validation.
|
15
|
-
#
|
16
|
-
module Rewrite
|
17
|
-
include GraphQL::Language
|
18
|
-
|
19
|
-
NO_DIRECTIVES = [].freeze
|
20
|
-
|
21
|
-
# @return InternalRepresentation::Document
|
22
|
-
attr_reader :rewrite_document
|
23
|
-
|
24
|
-
def initialize(*)
|
25
|
-
super
|
26
|
-
@query = context.query
|
27
|
-
@rewrite_document = InternalRepresentation::Document.new
|
28
|
-
# Hash<Nodes::FragmentSpread => Set<InternalRepresentation::Node>>
|
29
|
-
# A record of fragment spreads and the irep nodes that used them
|
30
|
-
@rewrite_spread_parents = Hash.new { |h, k| h[k] = Set.new }
|
31
|
-
# Hash<Nodes::FragmentSpread => Scope>
|
32
|
-
@rewrite_spread_scopes = {}
|
33
|
-
# Array<Set<InternalRepresentation::Node>>
|
34
|
-
# The current point of the irep_tree during visitation
|
35
|
-
@rewrite_nodes_stack = []
|
36
|
-
# Array<Scope>
|
37
|
-
@rewrite_scopes_stack = []
|
38
|
-
@rewrite_skip_nodes = Set.new
|
39
|
-
|
40
|
-
# Resolve fragment spreads.
|
41
|
-
# Fragment definitions got their own irep trees during visitation.
|
42
|
-
# Those nodes are spliced in verbatim (not copied), but this is OK
|
43
|
-
# because fragments are resolved from the "bottom up", each fragment
|
44
|
-
# can be shared between its usages.
|
45
|
-
context.on_dependency_resolve do |defn_ast_node, spread_ast_nodes, frag_ast_node|
|
46
|
-
frag_name = frag_ast_node.name
|
47
|
-
fragment_node = @rewrite_document.fragment_definitions[frag_name]
|
48
|
-
|
49
|
-
if fragment_node
|
50
|
-
spread_ast_nodes.each do |spread_ast_node|
|
51
|
-
parent_nodes = @rewrite_spread_parents[spread_ast_node]
|
52
|
-
parent_scope = @rewrite_spread_scopes[spread_ast_node]
|
53
|
-
parent_nodes.each do |parent_node|
|
54
|
-
parent_node.deep_merge_node(fragment_node, scope: parent_scope, merge_self: false)
|
55
|
-
end
|
56
|
-
end
|
57
|
-
end
|
58
|
-
end
|
59
|
-
end
|
60
|
-
|
61
|
-
# @return [Hash<String, Node>] Roots of this query
|
62
|
-
def operations
|
63
|
-
GraphQL::Deprecation.warn "#{self.class}#operations is deprecated; use `document.operation_definitions` instead"
|
64
|
-
@document.operation_definitions
|
65
|
-
end
|
66
|
-
|
67
|
-
def on_operation_definition(ast_node, parent)
|
68
|
-
push_root_node(ast_node, @rewrite_document.operation_definitions) { super }
|
69
|
-
end
|
70
|
-
|
71
|
-
def on_fragment_definition(ast_node, parent)
|
72
|
-
push_root_node(ast_node, @rewrite_document.fragment_definitions) { super }
|
73
|
-
end
|
74
|
-
|
75
|
-
def push_root_node(ast_node, definitions)
|
76
|
-
# Either QueryType or the fragment type condition
|
77
|
-
owner_type = context.type_definition
|
78
|
-
defn_name = ast_node.name
|
79
|
-
|
80
|
-
node = Node.new(
|
81
|
-
parent: nil,
|
82
|
-
name: defn_name,
|
83
|
-
owner_type: owner_type,
|
84
|
-
query: @query,
|
85
|
-
ast_nodes: [ast_node],
|
86
|
-
return_type: owner_type,
|
87
|
-
)
|
88
|
-
|
89
|
-
definitions[defn_name] = node
|
90
|
-
@rewrite_scopes_stack.push(Scope.new(@query, owner_type))
|
91
|
-
@rewrite_nodes_stack.push([node])
|
92
|
-
yield
|
93
|
-
@rewrite_nodes_stack.pop
|
94
|
-
@rewrite_scopes_stack.pop
|
95
|
-
end
|
96
|
-
|
97
|
-
def on_inline_fragment(node, parent)
|
98
|
-
# Inline fragments provide two things to the rewritten tree:
|
99
|
-
# - They _may_ narrow the scope by their type condition
|
100
|
-
# - They _may_ apply their directives to their children
|
101
|
-
if skip?(node)
|
102
|
-
@rewrite_skip_nodes.add(node)
|
103
|
-
end
|
104
|
-
|
105
|
-
if @rewrite_skip_nodes.empty?
|
106
|
-
@rewrite_scopes_stack.push(@rewrite_scopes_stack.last.enter(context.type_definition))
|
107
|
-
end
|
108
|
-
|
109
|
-
super
|
110
|
-
|
111
|
-
if @rewrite_skip_nodes.empty?
|
112
|
-
@rewrite_scopes_stack.pop
|
113
|
-
end
|
114
|
-
|
115
|
-
if @rewrite_skip_nodes.include?(node)
|
116
|
-
@rewrite_skip_nodes.delete(node)
|
117
|
-
end
|
118
|
-
end
|
119
|
-
|
120
|
-
def on_field(ast_node, ast_parent)
|
121
|
-
if skip?(ast_node)
|
122
|
-
@rewrite_skip_nodes.add(ast_node)
|
123
|
-
end
|
124
|
-
|
125
|
-
if @rewrite_skip_nodes.empty?
|
126
|
-
node_name = ast_node.alias || ast_node.name
|
127
|
-
parent_nodes = @rewrite_nodes_stack.last
|
128
|
-
next_nodes = []
|
129
|
-
|
130
|
-
field_defn = context.field_definition
|
131
|
-
if field_defn.nil?
|
132
|
-
# It's a non-existent field
|
133
|
-
new_scope = nil
|
134
|
-
else
|
135
|
-
field_return_type = field_defn.type
|
136
|
-
@rewrite_scopes_stack.last.each do |scope_type|
|
137
|
-
parent_nodes.each do |parent_node|
|
138
|
-
node = parent_node.scoped_children[scope_type][node_name] ||= Node.new(
|
139
|
-
parent: parent_node,
|
140
|
-
name: node_name,
|
141
|
-
owner_type: scope_type,
|
142
|
-
query: @query,
|
143
|
-
return_type: field_return_type,
|
144
|
-
)
|
145
|
-
node.ast_nodes << ast_node
|
146
|
-
node.definitions << field_defn
|
147
|
-
next_nodes << node
|
148
|
-
end
|
149
|
-
end
|
150
|
-
new_scope = Scope.new(@query, field_return_type.unwrap)
|
151
|
-
end
|
152
|
-
|
153
|
-
@rewrite_nodes_stack.push(next_nodes)
|
154
|
-
@rewrite_scopes_stack.push(new_scope)
|
155
|
-
end
|
156
|
-
|
157
|
-
super
|
158
|
-
|
159
|
-
if @rewrite_skip_nodes.empty?
|
160
|
-
@rewrite_nodes_stack.pop
|
161
|
-
@rewrite_scopes_stack.pop
|
162
|
-
end
|
163
|
-
|
164
|
-
if @rewrite_skip_nodes.include?(ast_node)
|
165
|
-
@rewrite_skip_nodes.delete(ast_node)
|
166
|
-
end
|
167
|
-
end
|
168
|
-
|
169
|
-
def on_fragment_spread(ast_node, ast_parent)
|
170
|
-
if @rewrite_skip_nodes.empty? && !skip?(ast_node)
|
171
|
-
# Register the irep nodes that depend on this AST node:
|
172
|
-
@rewrite_spread_parents[ast_node].merge(@rewrite_nodes_stack.last)
|
173
|
-
@rewrite_spread_scopes[ast_node] = @rewrite_scopes_stack.last
|
174
|
-
end
|
175
|
-
super
|
176
|
-
end
|
177
|
-
|
178
|
-
def skip?(ast_node)
|
179
|
-
dir = ast_node.directives
|
180
|
-
dir.any? && !GraphQL::Execution::DirectiveChecks.include?(dir, @query)
|
181
|
-
end
|
182
|
-
end
|
183
|
-
end
|
184
|
-
end
|
@@ -1,88 +0,0 @@
|
|
1
|
-
# frozen_string_literal: true
|
2
|
-
module GraphQL
|
3
|
-
module InternalRepresentation
|
4
|
-
# At a point in the AST, selections may apply to one or more types.
|
5
|
-
# {Scope} represents those types which selections may apply to.
|
6
|
-
#
|
7
|
-
# Scopes can be defined by:
|
8
|
-
#
|
9
|
-
# - A single concrete or abstract type
|
10
|
-
# - An array of types
|
11
|
-
# - `nil`
|
12
|
-
#
|
13
|
-
# The AST may be scoped to an array of types when two abstractly-typed
|
14
|
-
# fragments occur in inside one another.
|
15
|
-
class Scope
|
16
|
-
NO_TYPES = [].freeze
|
17
|
-
|
18
|
-
# @param query [GraphQL::Query]
|
19
|
-
# @param type_defn [GraphQL::BaseType, Array<GraphQL::BaseType>, nil]
|
20
|
-
def initialize(query, type_defn)
|
21
|
-
@query = query
|
22
|
-
@type = type_defn
|
23
|
-
@abstract_type = false
|
24
|
-
@types = case type_defn
|
25
|
-
when Array
|
26
|
-
type_defn
|
27
|
-
when GraphQL::BaseType
|
28
|
-
@abstract_type = true
|
29
|
-
nil
|
30
|
-
when nil
|
31
|
-
NO_TYPES
|
32
|
-
else
|
33
|
-
raise "Unexpected scope type: #{type_defn}"
|
34
|
-
end
|
35
|
-
end
|
36
|
-
|
37
|
-
# From a starting point of `self`, create a new scope by condition `other_type_defn`.
|
38
|
-
# @param other_type_defn [GraphQL::BaseType, nil]
|
39
|
-
# @return [Scope]
|
40
|
-
def enter(other_type_defn)
|
41
|
-
case other_type_defn
|
42
|
-
when nil
|
43
|
-
# The type wasn't found, who cares
|
44
|
-
Scope.new(@query, nil)
|
45
|
-
when @type
|
46
|
-
# The condition is the same as current, so reuse self
|
47
|
-
self
|
48
|
-
when GraphQL::UnionType, GraphQL::InterfaceType
|
49
|
-
# Make a new scope of the intersection between the previous & next conditions
|
50
|
-
new_types = @query.possible_types(other_type_defn) & concrete_types
|
51
|
-
Scope.new(@query, new_types)
|
52
|
-
when GraphQL::BaseType
|
53
|
-
# If this type is valid within the current scope,
|
54
|
-
# return a new scope of _exactly_ this type.
|
55
|
-
# Otherwise, this type is out-of-scope so the scope is null.
|
56
|
-
if concrete_types.include?(other_type_defn)
|
57
|
-
Scope.new(@query, other_type_defn)
|
58
|
-
else
|
59
|
-
Scope.new(@query, nil)
|
60
|
-
end
|
61
|
-
else
|
62
|
-
raise "Unexpected scope: #{other_type_defn.inspect}"
|
63
|
-
end
|
64
|
-
end
|
65
|
-
|
66
|
-
# Call the block for each type in `self`.
|
67
|
-
# This uses the simplest possible expression of `self`,
|
68
|
-
# so if this scope is defined by an abstract type, it gets yielded.
|
69
|
-
def each(&block)
|
70
|
-
if @abstract_type
|
71
|
-
yield(@type)
|
72
|
-
else
|
73
|
-
@types.each(&block)
|
74
|
-
end
|
75
|
-
end
|
76
|
-
|
77
|
-
private
|
78
|
-
|
79
|
-
def concrete_types
|
80
|
-
@concrete_types ||= if @abstract_type
|
81
|
-
@query.possible_types(@type)
|
82
|
-
else
|
83
|
-
@types
|
84
|
-
end
|
85
|
-
end
|
86
|
-
end
|
87
|
-
end
|
88
|
-
end
|
@@ -1,36 +0,0 @@
|
|
1
|
-
# frozen_string_literal: true
|
2
|
-
module GraphQL
|
3
|
-
module InternalRepresentation
|
4
|
-
# Traverse a re-written query tree, calling handlers for each node
|
5
|
-
module Visit
|
6
|
-
module_function
|
7
|
-
def visit_each_node(operations, handlers)
|
8
|
-
return if handlers.empty?
|
9
|
-
# Post-validation: make some assertions about the rewritten query tree
|
10
|
-
operations.each do |op_name, op_node|
|
11
|
-
# Yield each node to listeners which were attached by validators
|
12
|
-
op_node.typed_children.each do |obj_type, children|
|
13
|
-
children.each do |name, op_child_node|
|
14
|
-
each_node(op_child_node) do |node|
|
15
|
-
for h in handlers
|
16
|
-
h.call(node)
|
17
|
-
end
|
18
|
-
end
|
19
|
-
end
|
20
|
-
end
|
21
|
-
end
|
22
|
-
end
|
23
|
-
|
24
|
-
# Traverse a node in a rewritten query tree,
|
25
|
-
# visiting the node itself and each of its typed children.
|
26
|
-
def each_node(node, &block)
|
27
|
-
yield(node)
|
28
|
-
node.typed_children.each do |obj_type, children|
|
29
|
-
children.each do |name, node|
|
30
|
-
each_node(node, &block)
|
31
|
-
end
|
32
|
-
end
|
33
|
-
end
|
34
|
-
end
|
35
|
-
end
|
36
|
-
end
|
@@ -1,7 +0,0 @@
|
|
1
|
-
# frozen_string_literal: true
|
2
|
-
require "graphql/internal_representation/document"
|
3
|
-
require "graphql/internal_representation/node"
|
4
|
-
require "graphql/internal_representation/print"
|
5
|
-
require "graphql/internal_representation/rewrite"
|
6
|
-
require "graphql/internal_representation/scope"
|
7
|
-
require "graphql/internal_representation/visit"
|
data/lib/graphql/list_type.rb
DELETED
@@ -1,80 +0,0 @@
|
|
1
|
-
# frozen_string_literal: true
|
2
|
-
module GraphQL
|
3
|
-
# A list type modifies another type.
|
4
|
-
#
|
5
|
-
# List types can be created with the type helper (`types[InnerType]`)
|
6
|
-
# or {BaseType#to_list_type} (`InnerType.to_list_type`)
|
7
|
-
#
|
8
|
-
# For return types, it says that the returned value will be a list of the modified.
|
9
|
-
#
|
10
|
-
# @example A field which returns a list of items
|
11
|
-
# field :items, types[ItemType]
|
12
|
-
# # or
|
13
|
-
# field :items, ItemType.to_list_type
|
14
|
-
#
|
15
|
-
# For input types, it says that the incoming value will be a list of the modified type.
|
16
|
-
#
|
17
|
-
# @example A field which accepts a list of strings
|
18
|
-
# field :newNames do
|
19
|
-
# # ...
|
20
|
-
# argument :values, types[types.String]
|
21
|
-
# # or
|
22
|
-
# argument :values, types.String.to_list_type
|
23
|
-
# end
|
24
|
-
#
|
25
|
-
# Given a list type, you can always get the underlying type with {#unwrap}.
|
26
|
-
#
|
27
|
-
class ListType < GraphQL::BaseType
|
28
|
-
include GraphQL::BaseType::ModifiesAnotherType
|
29
|
-
attr_reader :of_type
|
30
|
-
def initialize(of_type:)
|
31
|
-
super()
|
32
|
-
@of_type = of_type
|
33
|
-
end
|
34
|
-
|
35
|
-
def kind
|
36
|
-
GraphQL::TypeKinds::LIST
|
37
|
-
end
|
38
|
-
|
39
|
-
def to_s
|
40
|
-
"[#{of_type.to_s}]"
|
41
|
-
end
|
42
|
-
alias_method :inspect, :to_s
|
43
|
-
alias :to_type_signature :to_s
|
44
|
-
|
45
|
-
def coerce_result(value, ctx = nil)
|
46
|
-
if ctx.nil?
|
47
|
-
warn_deprecated_coerce("coerce_isolated_result")
|
48
|
-
ctx = GraphQL::Query::NullContext
|
49
|
-
end
|
50
|
-
ensure_array(value).map { |item| item.nil? ? nil : of_type.coerce_result(item, ctx) }
|
51
|
-
end
|
52
|
-
|
53
|
-
def list?
|
54
|
-
true
|
55
|
-
end
|
56
|
-
|
57
|
-
private
|
58
|
-
|
59
|
-
def coerce_non_null_input(value, ctx)
|
60
|
-
ensure_array(value).map { |item| of_type.coerce_input(item, ctx) }
|
61
|
-
end
|
62
|
-
|
63
|
-
def validate_non_null_input(value, ctx)
|
64
|
-
result = GraphQL::Query::InputValidationResult.new
|
65
|
-
|
66
|
-
ensure_array(value).each_with_index do |item, index|
|
67
|
-
item_result = of_type.validate_input(item, ctx)
|
68
|
-
if !item_result.valid?
|
69
|
-
result.merge_result!(index, item_result)
|
70
|
-
end
|
71
|
-
end
|
72
|
-
|
73
|
-
result
|
74
|
-
end
|
75
|
-
|
76
|
-
def ensure_array(value)
|
77
|
-
value.is_a?(Array) ? value : [value]
|
78
|
-
end
|
79
|
-
end
|
80
|
-
end
|