graphql 1.13.12 → 2.0.21

Sign up to get free protection for your applications and to get access to all the features.
Files changed (273) hide show
  1. checksums.yaml +4 -4
  2. data/lib/generators/graphql/install_generator.rb +1 -1
  3. data/lib/generators/graphql/relay.rb +3 -17
  4. data/lib/generators/graphql/templates/schema.erb +3 -0
  5. data/lib/graphql/analysis/ast/field_usage.rb +3 -1
  6. data/lib/graphql/analysis/ast/max_query_complexity.rb +0 -1
  7. data/lib/graphql/analysis/ast/query_complexity.rb +1 -1
  8. data/lib/graphql/analysis/ast/query_depth.rb +0 -1
  9. data/lib/graphql/analysis/ast/visitor.rb +43 -36
  10. data/lib/graphql/analysis/ast.rb +2 -12
  11. data/lib/graphql/analysis.rb +0 -7
  12. data/lib/graphql/backtrace/table.rb +2 -20
  13. data/lib/graphql/backtrace/trace.rb +96 -0
  14. data/lib/graphql/backtrace/tracer.rb +2 -3
  15. data/lib/graphql/backtrace.rb +7 -8
  16. data/lib/graphql/dataloader/null_dataloader.rb +3 -1
  17. data/lib/graphql/dataloader/source.rb +9 -0
  18. data/lib/graphql/dataloader.rb +4 -1
  19. data/lib/graphql/dig.rb +1 -1
  20. data/lib/graphql/execution/errors.rb +12 -82
  21. data/lib/graphql/execution/interpreter/arguments.rb +1 -1
  22. data/lib/graphql/execution/interpreter/arguments_cache.rb +2 -3
  23. data/lib/graphql/execution/interpreter/resolve.rb +26 -0
  24. data/lib/graphql/execution/interpreter/runtime.rb +300 -222
  25. data/lib/graphql/execution/interpreter.rb +187 -78
  26. data/lib/graphql/execution/lazy.rb +7 -21
  27. data/lib/graphql/execution/lookahead.rb +44 -40
  28. data/lib/graphql/execution/multiplex.rb +3 -174
  29. data/lib/graphql/execution.rb +11 -4
  30. data/lib/graphql/filter.rb +7 -2
  31. data/lib/graphql/introspection/directive_type.rb +2 -2
  32. data/lib/graphql/introspection/dynamic_fields.rb +3 -8
  33. data/lib/graphql/introspection/entry_points.rb +2 -15
  34. data/lib/graphql/introspection/field_type.rb +1 -1
  35. data/lib/graphql/introspection/schema_type.rb +2 -2
  36. data/lib/graphql/introspection/type_type.rb +13 -6
  37. data/lib/graphql/introspection.rb +4 -3
  38. data/lib/graphql/language/document_from_schema_definition.rb +43 -44
  39. data/lib/graphql/language/lexer.rb +216 -1488
  40. data/lib/graphql/language/nodes.rb +66 -40
  41. data/lib/graphql/language/parser.rb +539 -510
  42. data/lib/graphql/language/parser.y +53 -44
  43. data/lib/graphql/language/printer.rb +37 -21
  44. data/lib/graphql/language/visitor.rb +191 -83
  45. data/lib/graphql/pagination/active_record_relation_connection.rb +0 -8
  46. data/lib/graphql/pagination/array_connection.rb +4 -2
  47. data/lib/graphql/pagination/connection.rb +33 -6
  48. data/lib/graphql/pagination/connections.rb +3 -28
  49. data/lib/graphql/pagination/relation_connection.rb +2 -0
  50. data/lib/graphql/query/context.rb +156 -196
  51. data/lib/graphql/query/input_validation_result.rb +10 -1
  52. data/lib/graphql/query/null_context.rb +1 -4
  53. data/lib/graphql/query/validation_pipeline.rb +12 -37
  54. data/lib/graphql/query/variable_validation_error.rb +2 -2
  55. data/lib/graphql/query/variables.rb +35 -21
  56. data/lib/graphql/query.rb +39 -46
  57. data/lib/graphql/railtie.rb +0 -104
  58. data/lib/graphql/rake_task/validate.rb +1 -1
  59. data/lib/graphql/rake_task.rb +29 -1
  60. data/lib/graphql/relay/range_add.rb +9 -20
  61. data/lib/graphql/relay.rb +0 -15
  62. data/lib/graphql/schema/addition.rb +7 -9
  63. data/lib/graphql/schema/argument.rb +38 -47
  64. data/lib/graphql/schema/build_from_definition.rb +47 -21
  65. data/lib/graphql/schema/directive/one_of.rb +12 -0
  66. data/lib/graphql/schema/directive/transform.rb +1 -1
  67. data/lib/graphql/schema/directive.rb +12 -23
  68. data/lib/graphql/schema/enum.rb +29 -41
  69. data/lib/graphql/schema/enum_value.rb +2 -25
  70. data/lib/graphql/schema/field/connection_extension.rb +4 -0
  71. data/lib/graphql/schema/field.rb +256 -349
  72. data/lib/graphql/schema/field_extension.rb +1 -4
  73. data/lib/graphql/schema/find_inherited_value.rb +2 -7
  74. data/lib/graphql/schema/input_object.rb +57 -69
  75. data/lib/graphql/schema/interface.rb +0 -35
  76. data/lib/graphql/schema/introspection_system.rb +3 -8
  77. data/lib/graphql/schema/late_bound_type.rb +8 -2
  78. data/lib/graphql/schema/list.rb +18 -9
  79. data/lib/graphql/schema/loader.rb +1 -2
  80. data/lib/graphql/schema/member/base_dsl_methods.rb +17 -19
  81. data/lib/graphql/schema/member/build_type.rb +5 -7
  82. data/lib/graphql/schema/member/has_arguments.rb +147 -56
  83. data/lib/graphql/schema/member/has_ast_node.rb +12 -0
  84. data/lib/graphql/schema/member/has_deprecation_reason.rb +3 -4
  85. data/lib/graphql/schema/member/has_directives.rb +81 -61
  86. data/lib/graphql/schema/member/has_fields.rb +97 -40
  87. data/lib/graphql/schema/member/has_interfaces.rb +49 -10
  88. data/lib/graphql/schema/member/has_validators.rb +32 -6
  89. data/lib/graphql/schema/member/relay_shortcuts.rb +47 -2
  90. data/lib/graphql/schema/member/type_system_helpers.rb +17 -0
  91. data/lib/graphql/schema/member/validates_input.rb +3 -3
  92. data/lib/graphql/schema/member.rb +0 -6
  93. data/lib/graphql/schema/mutation.rb +0 -9
  94. data/lib/graphql/schema/non_null.rb +3 -9
  95. data/lib/graphql/schema/object.rb +15 -52
  96. data/lib/graphql/schema/relay_classic_mutation.rb +53 -42
  97. data/lib/graphql/schema/resolver/has_payload_type.rb +20 -10
  98. data/lib/graphql/schema/resolver.rb +43 -44
  99. data/lib/graphql/schema/scalar.rb +8 -23
  100. data/lib/graphql/schema/subscription.rb +0 -7
  101. data/lib/graphql/schema/timeout.rb +24 -28
  102. data/lib/graphql/schema/type_membership.rb +3 -0
  103. data/lib/graphql/schema/union.rb +10 -17
  104. data/lib/graphql/schema/validator.rb +1 -1
  105. data/lib/graphql/schema/warden.rb +37 -9
  106. data/lib/graphql/schema/wrapper.rb +0 -5
  107. data/lib/graphql/schema.rb +265 -968
  108. data/lib/graphql/static_validation/all_rules.rb +1 -0
  109. data/lib/graphql/static_validation/base_visitor.rb +4 -21
  110. data/lib/graphql/static_validation/definition_dependencies.rb +7 -1
  111. data/lib/graphql/static_validation/error.rb +2 -2
  112. data/lib/graphql/static_validation/literal_validator.rb +19 -1
  113. data/lib/graphql/static_validation/rules/directives_are_defined.rb +11 -5
  114. data/lib/graphql/static_validation/rules/directives_are_in_valid_locations.rb +12 -12
  115. data/lib/graphql/static_validation/rules/fields_have_appropriate_selections.rb +12 -4
  116. data/lib/graphql/static_validation/rules/fields_will_merge.rb +2 -2
  117. data/lib/graphql/static_validation/rules/one_of_input_objects_are_valid.rb +66 -0
  118. data/lib/graphql/static_validation/rules/one_of_input_objects_are_valid_error.rb +29 -0
  119. data/lib/graphql/static_validation/rules/unique_directives_per_location.rb +12 -6
  120. data/lib/graphql/static_validation/rules/variable_default_values_are_correctly_typed.rb +1 -1
  121. data/lib/graphql/static_validation/validator.rb +3 -25
  122. data/lib/graphql/static_validation.rb +0 -2
  123. data/lib/graphql/subscriptions/action_cable_subscriptions.rb +7 -1
  124. data/lib/graphql/subscriptions/default_subscription_resolve_extension.rb +38 -1
  125. data/lib/graphql/subscriptions/event.rb +3 -8
  126. data/lib/graphql/subscriptions/instrumentation.rb +0 -51
  127. data/lib/graphql/subscriptions.rb +32 -20
  128. data/lib/graphql/tracing/active_support_notifications_trace.rb +16 -0
  129. data/lib/graphql/tracing/appoptics_trace.rb +231 -0
  130. data/lib/graphql/tracing/appsignal_trace.rb +77 -0
  131. data/lib/graphql/tracing/data_dog_trace.rb +148 -0
  132. data/lib/graphql/tracing/data_dog_tracing.rb +21 -2
  133. data/lib/graphql/tracing/legacy_trace.rb +65 -0
  134. data/lib/graphql/tracing/new_relic_trace.rb +75 -0
  135. data/lib/graphql/tracing/notifications_trace.rb +42 -0
  136. data/lib/graphql/tracing/platform_trace.rb +109 -0
  137. data/lib/graphql/tracing/platform_tracing.rb +33 -43
  138. data/lib/graphql/tracing/prometheus_trace.rb +89 -0
  139. data/lib/graphql/tracing/prometheus_tracing/graphql_collector.rb +1 -1
  140. data/lib/graphql/tracing/prometheus_tracing.rb +3 -3
  141. data/lib/graphql/tracing/scout_trace.rb +72 -0
  142. data/lib/graphql/tracing/statsd_trace.rb +56 -0
  143. data/lib/graphql/tracing/trace.rb +75 -0
  144. data/lib/graphql/tracing.rb +16 -40
  145. data/lib/graphql/type_kinds.rb +6 -3
  146. data/lib/graphql/types/iso_8601_date.rb +4 -1
  147. data/lib/graphql/types/iso_8601_date_time.rb +4 -0
  148. data/lib/graphql/types/relay/base_connection.rb +16 -6
  149. data/lib/graphql/types/relay/connection_behaviors.rb +29 -27
  150. data/lib/graphql/types/relay/edge_behaviors.rb +16 -5
  151. data/lib/graphql/types/relay/node_behaviors.rb +12 -2
  152. data/lib/graphql/types/relay/page_info_behaviors.rb +7 -2
  153. data/lib/graphql/types/relay.rb +0 -3
  154. data/lib/graphql/types/string.rb +1 -1
  155. data/lib/graphql/version.rb +1 -1
  156. data/lib/graphql.rb +17 -74
  157. metadata +33 -133
  158. data/lib/graphql/analysis/analyze_query.rb +0 -98
  159. data/lib/graphql/analysis/field_usage.rb +0 -45
  160. data/lib/graphql/analysis/max_query_complexity.rb +0 -26
  161. data/lib/graphql/analysis/max_query_depth.rb +0 -26
  162. data/lib/graphql/analysis/query_complexity.rb +0 -88
  163. data/lib/graphql/analysis/query_depth.rb +0 -43
  164. data/lib/graphql/analysis/reducer_state.rb +0 -48
  165. data/lib/graphql/argument.rb +0 -131
  166. data/lib/graphql/authorization.rb +0 -82
  167. data/lib/graphql/backtrace/legacy_tracer.rb +0 -56
  168. data/lib/graphql/backwards_compatibility.rb +0 -61
  169. data/lib/graphql/base_type.rb +0 -232
  170. data/lib/graphql/boolean_type.rb +0 -2
  171. data/lib/graphql/compatibility/execution_specification/counter_schema.rb +0 -53
  172. data/lib/graphql/compatibility/execution_specification/specification_schema.rb +0 -200
  173. data/lib/graphql/compatibility/execution_specification.rb +0 -436
  174. data/lib/graphql/compatibility/lazy_execution_specification/lazy_schema.rb +0 -111
  175. data/lib/graphql/compatibility/lazy_execution_specification.rb +0 -215
  176. data/lib/graphql/compatibility/query_parser_specification/parse_error_specification.rb +0 -87
  177. data/lib/graphql/compatibility/query_parser_specification/query_assertions.rb +0 -79
  178. data/lib/graphql/compatibility/query_parser_specification.rb +0 -266
  179. data/lib/graphql/compatibility/schema_parser_specification.rb +0 -682
  180. data/lib/graphql/compatibility.rb +0 -5
  181. data/lib/graphql/define/assign_argument.rb +0 -12
  182. data/lib/graphql/define/assign_connection.rb +0 -13
  183. data/lib/graphql/define/assign_enum_value.rb +0 -18
  184. data/lib/graphql/define/assign_global_id_field.rb +0 -11
  185. data/lib/graphql/define/assign_mutation_function.rb +0 -34
  186. data/lib/graphql/define/assign_object_field.rb +0 -42
  187. data/lib/graphql/define/defined_object_proxy.rb +0 -53
  188. data/lib/graphql/define/instance_definable.rb +0 -255
  189. data/lib/graphql/define/no_definition_error.rb +0 -7
  190. data/lib/graphql/define/non_null_with_bang.rb +0 -16
  191. data/lib/graphql/define/type_definer.rb +0 -31
  192. data/lib/graphql/define.rb +0 -31
  193. data/lib/graphql/deprecated_dsl.rb +0 -55
  194. data/lib/graphql/directive/deprecated_directive.rb +0 -2
  195. data/lib/graphql/directive/include_directive.rb +0 -2
  196. data/lib/graphql/directive/skip_directive.rb +0 -2
  197. data/lib/graphql/directive.rb +0 -107
  198. data/lib/graphql/enum_type.rb +0 -133
  199. data/lib/graphql/execution/execute.rb +0 -333
  200. data/lib/graphql/execution/flatten.rb +0 -40
  201. data/lib/graphql/execution/instrumentation.rb +0 -92
  202. data/lib/graphql/execution/lazy/resolve.rb +0 -91
  203. data/lib/graphql/execution/typecast.rb +0 -50
  204. data/lib/graphql/field/resolve.rb +0 -59
  205. data/lib/graphql/field.rb +0 -226
  206. data/lib/graphql/float_type.rb +0 -2
  207. data/lib/graphql/function.rb +0 -128
  208. data/lib/graphql/id_type.rb +0 -2
  209. data/lib/graphql/input_object_type.rb +0 -138
  210. data/lib/graphql/int_type.rb +0 -2
  211. data/lib/graphql/interface_type.rb +0 -72
  212. data/lib/graphql/internal_representation/document.rb +0 -27
  213. data/lib/graphql/internal_representation/node.rb +0 -206
  214. data/lib/graphql/internal_representation/print.rb +0 -51
  215. data/lib/graphql/internal_representation/rewrite.rb +0 -184
  216. data/lib/graphql/internal_representation/scope.rb +0 -88
  217. data/lib/graphql/internal_representation/visit.rb +0 -36
  218. data/lib/graphql/internal_representation.rb +0 -7
  219. data/lib/graphql/language/lexer.rl +0 -260
  220. data/lib/graphql/list_type.rb +0 -80
  221. data/lib/graphql/non_null_type.rb +0 -71
  222. data/lib/graphql/object_type.rb +0 -130
  223. data/lib/graphql/query/arguments.rb +0 -189
  224. data/lib/graphql/query/arguments_cache.rb +0 -24
  225. data/lib/graphql/query/executor.rb +0 -52
  226. data/lib/graphql/query/literal_input.rb +0 -136
  227. data/lib/graphql/query/serial_execution/field_resolution.rb +0 -92
  228. data/lib/graphql/query/serial_execution/operation_resolution.rb +0 -19
  229. data/lib/graphql/query/serial_execution/selection_resolution.rb +0 -23
  230. data/lib/graphql/query/serial_execution/value_resolution.rb +0 -87
  231. data/lib/graphql/query/serial_execution.rb +0 -40
  232. data/lib/graphql/relay/array_connection.rb +0 -83
  233. data/lib/graphql/relay/base_connection.rb +0 -189
  234. data/lib/graphql/relay/connection_instrumentation.rb +0 -54
  235. data/lib/graphql/relay/connection_resolve.rb +0 -43
  236. data/lib/graphql/relay/connection_type.rb +0 -54
  237. data/lib/graphql/relay/edge.rb +0 -27
  238. data/lib/graphql/relay/edge_type.rb +0 -19
  239. data/lib/graphql/relay/edges_instrumentation.rb +0 -39
  240. data/lib/graphql/relay/global_id_resolve.rb +0 -17
  241. data/lib/graphql/relay/mongo_relation_connection.rb +0 -50
  242. data/lib/graphql/relay/mutation/instrumentation.rb +0 -23
  243. data/lib/graphql/relay/mutation/resolve.rb +0 -56
  244. data/lib/graphql/relay/mutation/result.rb +0 -38
  245. data/lib/graphql/relay/mutation.rb +0 -106
  246. data/lib/graphql/relay/node.rb +0 -39
  247. data/lib/graphql/relay/page_info.rb +0 -7
  248. data/lib/graphql/relay/relation_connection.rb +0 -188
  249. data/lib/graphql/relay/type_extensions.rb +0 -32
  250. data/lib/graphql/scalar_type.rb +0 -91
  251. data/lib/graphql/schema/catchall_middleware.rb +0 -35
  252. data/lib/graphql/schema/default_parse_error.rb +0 -10
  253. data/lib/graphql/schema/default_type_error.rb +0 -17
  254. data/lib/graphql/schema/member/accepts_definition.rb +0 -164
  255. data/lib/graphql/schema/member/cached_graphql_definition.rb +0 -58
  256. data/lib/graphql/schema/member/instrumentation.rb +0 -131
  257. data/lib/graphql/schema/middleware_chain.rb +0 -82
  258. data/lib/graphql/schema/possible_types.rb +0 -44
  259. data/lib/graphql/schema/rescue_middleware.rb +0 -60
  260. data/lib/graphql/schema/timeout_middleware.rb +0 -88
  261. data/lib/graphql/schema/traversal.rb +0 -228
  262. data/lib/graphql/schema/validation.rb +0 -313
  263. data/lib/graphql/static_validation/default_visitor.rb +0 -15
  264. data/lib/graphql/static_validation/no_validate_visitor.rb +0 -10
  265. data/lib/graphql/string_type.rb +0 -2
  266. data/lib/graphql/subscriptions/subscription_root.rb +0 -76
  267. data/lib/graphql/tracing/skylight_tracing.rb +0 -70
  268. data/lib/graphql/types/relay/default_relay.rb +0 -31
  269. data/lib/graphql/types/relay/node_field.rb +0 -24
  270. data/lib/graphql/types/relay/nodes_field.rb +0 -43
  271. data/lib/graphql/union_type.rb +0 -115
  272. data/lib/graphql/upgrader/member.rb +0 -937
  273. data/lib/graphql/upgrader/schema.rb +0 -38
@@ -1,83 +0,0 @@
1
- # frozen_string_literal: true
2
- module GraphQL
3
- module Relay
4
- class ArrayConnection < BaseConnection
5
- def cursor_from_node(item)
6
- idx = (after ? index_from_cursor(after) : 0) + sliced_nodes.find_index(item) + 1
7
- encode(idx.to_s)
8
- end
9
-
10
- def has_next_page
11
- if first
12
- # There are more items after these items
13
- sliced_nodes.count > first
14
- elsif GraphQL::Relay::ConnectionType.bidirectional_pagination && before
15
- # The original array is longer than the `before` index
16
- index_from_cursor(before) < nodes.length + 1
17
- else
18
- false
19
- end
20
- end
21
-
22
- def has_previous_page
23
- if last
24
- # There are items preceding the ones in this result
25
- sliced_nodes.count > last
26
- elsif GraphQL::Relay::ConnectionType.bidirectional_pagination && after
27
- # We've paginated into the Array a bit, there are some behind us
28
- index_from_cursor(after) > 0
29
- else
30
- false
31
- end
32
- end
33
-
34
- def first
35
- @first ||= begin
36
- capped = limit_pagination_argument(arguments[:first], max_page_size)
37
- if capped.nil? && last.nil?
38
- capped = max_page_size
39
- end
40
- capped
41
- end
42
- end
43
-
44
- def last
45
- @last ||= limit_pagination_argument(arguments[:last], max_page_size)
46
- end
47
-
48
- private
49
-
50
- # apply first / last limit results
51
- def paged_nodes
52
- @paged_nodes ||= begin
53
- items = sliced_nodes
54
-
55
- items = items.first(first) if first
56
- items = items.last(last) if last
57
- items = items.first(max_page_size) if max_page_size && !first && !last
58
-
59
- items
60
- end
61
- end
62
-
63
- # Apply cursors to edges
64
- def sliced_nodes
65
- @sliced_nodes ||= if before && after
66
- nodes[index_from_cursor(after)..index_from_cursor(before)-1] || []
67
- elsif before
68
- nodes[0..index_from_cursor(before)-2] || []
69
- elsif after
70
- nodes[index_from_cursor(after)..-1] || []
71
- else
72
- nodes
73
- end
74
- end
75
-
76
- def index_from_cursor(cursor)
77
- decode(cursor).to_i
78
- end
79
- end
80
-
81
- BaseConnection.register_connection_implementation(Array, ArrayConnection)
82
- end
83
- end
@@ -1,189 +0,0 @@
1
- # frozen_string_literal: true
2
- module GraphQL
3
- module Relay
4
- # Subclasses must implement:
5
- # - {#cursor_from_node}, which returns an opaque cursor for the given item
6
- # - {#sliced_nodes}, which slices by `before` & `after`
7
- # - {#paged_nodes}, which applies `first` & `last` limits
8
- #
9
- # In a subclass, you have access to
10
- # - {#nodes}, the collection which the connection will wrap
11
- # - {#first}, {#after}, {#last}, {#before} (arguments passed to the field)
12
- # - {#max_page_size} (the specified maximum page size that can be returned from a connection)
13
- #
14
- class BaseConnection
15
- # Just to encode data in the cursor, use something that won't conflict
16
- CURSOR_SEPARATOR = "---"
17
-
18
- # Map of collection class names -> connection_classes
19
- # eg `{"Array" => ArrayConnection}`
20
- CONNECTION_IMPLEMENTATIONS = {}
21
-
22
- class << self
23
- # Find a connection implementation suitable for exposing `nodes`
24
- #
25
- # @param nodes [Object] A collection of nodes (eg, Array, AR::Relation)
26
- # @return [subclass of BaseConnection] a connection Class for wrapping `nodes`
27
- def connection_for_nodes(nodes)
28
- # If it's a new-style connection object, it's already ready to go
29
- if nodes.is_a?(GraphQL::Pagination::Connection)
30
- return nodes
31
- end
32
- # Check for class _names_ because classes can be redefined in Rails development
33
- nodes.class.ancestors.each do |ancestor|
34
- conn_impl = CONNECTION_IMPLEMENTATIONS[ancestor.name]
35
- if conn_impl
36
- return conn_impl
37
- end
38
- end
39
- # Should have found a connection during the loop:
40
- raise("No connection implementation to wrap #{nodes.class} (#{nodes})")
41
- end
42
-
43
- # Add `connection_class` as the connection wrapper for `nodes_class`
44
- # eg, `RelationConnection` is the implementation for `AR::Relation`
45
- # @param nodes_class [Class] A class representing a collection (eg, Array, AR::Relation)
46
- # @param connection_class [Class] A class implementing Connection methods
47
- def register_connection_implementation(nodes_class, connection_class)
48
- CONNECTION_IMPLEMENTATIONS[nodes_class.name] = connection_class
49
- end
50
- end
51
-
52
- attr_reader :nodes, :arguments, :max_page_size, :parent, :field, :context
53
-
54
- # Make a connection, wrapping `nodes`
55
- # @param nodes [Object] The collection of nodes
56
- # @param arguments [GraphQL::Query::Arguments] Query arguments
57
- # @param field [GraphQL::Field] The underlying field
58
- # @param max_page_size [Int] The maximum number of results to return
59
- # @param parent [Object] The object which this collection belongs to
60
- # @param context [GraphQL::Query::Context] The context from the field being resolved
61
- def initialize(nodes, arguments, field: nil, max_page_size: nil, parent: nil, context: nil)
62
- GraphQL::Deprecation.warn "GraphQL::Relay::BaseConnection (used for #{self.class}) will be removed from GraphQL-Ruby 2.0, use GraphQL::Pagination::Connections instead: https://graphql-ruby.org/pagination/overview.html"
63
-
64
- deprecated_caller = caller(0, 10).find { |c| !c.include?("lib/graphql") }
65
- if deprecated_caller
66
- GraphQL::Deprecation.warn " -> called from #{deprecated_caller}"
67
- end
68
-
69
- @context = context
70
- @nodes = nodes
71
- @arguments = arguments
72
- @field = field
73
- @parent = parent
74
- @encoder = context ? @context.schema.cursor_encoder : GraphQL::Schema::Base64Encoder
75
- @max_page_size = max_page_size.nil? && context ? @context.schema.default_max_page_size : max_page_size
76
- end
77
-
78
- def encode(data)
79
- @encoder.encode(data, nonce: true)
80
- end
81
-
82
- def decode(data)
83
- @encoder.decode(data, nonce: true)
84
- end
85
-
86
- # The value passed as `first:`, if there was one. Negative numbers become `0`.
87
- # @return [Integer, nil]
88
- def first
89
- @first ||= begin
90
- capped = limit_pagination_argument(arguments[:first], max_page_size)
91
- if capped.nil? && last.nil?
92
- capped = max_page_size
93
- end
94
- capped
95
- end
96
- end
97
-
98
- # The value passed as `after:`, if there was one
99
- # @return [String, nil]
100
- def after
101
- arguments[:after]
102
- end
103
-
104
- # The value passed as `last:`, if there was one. Negative numbers become `0`.
105
- # @return [Integer, nil]
106
- def last
107
- @last ||= limit_pagination_argument(arguments[:last], max_page_size)
108
- end
109
-
110
- # The value passed as `before:`, if there was one
111
- # @return [String, nil]
112
- def before
113
- arguments[:before]
114
- end
115
-
116
- # These are the nodes to render for this connection,
117
- # probably wrapped by {GraphQL::Relay::Edge}
118
- def edge_nodes
119
- @edge_nodes ||= paged_nodes
120
- end
121
-
122
- # Support the `pageInfo` field
123
- def page_info
124
- self
125
- end
126
-
127
- # Used by `pageInfo`
128
- def has_next_page
129
- !!(first && sliced_nodes.count > first)
130
- end
131
-
132
- # Used by `pageInfo`
133
- def has_previous_page
134
- !!(last && sliced_nodes.count > last)
135
- end
136
-
137
- # Used by `pageInfo`
138
- def start_cursor
139
- if start_node = (respond_to?(:paged_nodes_array, true) ? paged_nodes_array : paged_nodes).first
140
- return cursor_from_node(start_node)
141
- else
142
- return nil
143
- end
144
- end
145
-
146
- # Used by `pageInfo`
147
- def end_cursor
148
- if end_node = (respond_to?(:paged_nodes_array, true) ? paged_nodes_array : paged_nodes).last
149
- return cursor_from_node(end_node)
150
- else
151
- return nil
152
- end
153
- end
154
-
155
- # An opaque operation which returns a connection-specific cursor.
156
- def cursor_from_node(object)
157
- raise GraphQL::RequiredImplementationMissingError, "must return a cursor for this object/connection pair"
158
- end
159
-
160
- def inspect
161
- "#<GraphQL::Relay::Connection @parent=#{@parent.inspect} @arguments=#{@arguments.to_h.inspect}>"
162
- end
163
-
164
- private
165
-
166
- # @param argument [nil, Integer] `first` or `last`, as provided by the client
167
- # @param max_page_size [nil, Integer]
168
- # @return [nil, Integer] `nil` if the input was `nil`, otherwise a value between `0` and `max_page_size`
169
- def limit_pagination_argument(argument, max_page_size)
170
- if argument
171
- if argument < 0
172
- argument = 0
173
- elsif max_page_size && argument > max_page_size
174
- argument = max_page_size
175
- end
176
- end
177
- argument
178
- end
179
-
180
- def paged_nodes
181
- raise GraphQL::RequiredImplementationMissingError, "must return nodes for this connection after paging"
182
- end
183
-
184
- def sliced_nodes
185
- raise GraphQL::RequiredImplementationMissingError, "must return all nodes for this connection after chopping off first and last"
186
- end
187
- end
188
- end
189
- end
@@ -1,54 +0,0 @@
1
- # frozen_string_literal: true
2
- module GraphQL
3
- module Relay
4
- # Provided a GraphQL field which returns a collection of nodes,
5
- # wrap that field to expose those nodes as a connection.
6
- #
7
- # The original resolve proc is used to fetch nodes,
8
- # then a connection implementation is fetched with {BaseConnection.connection_for_nodes}.
9
- module ConnectionInstrumentation
10
- def self.default_arguments
11
- @default_arguments ||= begin
12
- argument_definitions = [
13
- ["first", GraphQL::DEPRECATED_INT_TYPE, "Returns the first _n_ elements from the list."],
14
- ["after", GraphQL::DEPRECATED_STRING_TYPE, "Returns the elements in the list that come after the specified cursor."],
15
- ["last", GraphQL::DEPRECATED_INT_TYPE, "Returns the last _n_ elements from the list."],
16
- ["before", GraphQL::DEPRECATED_STRING_TYPE, "Returns the elements in the list that come before the specified cursor."],
17
- ]
18
-
19
- argument_definitions.reduce({}) do |memo, arg_defn|
20
- argument = GraphQL::Argument.new
21
- name, type, description = arg_defn
22
- argument.name = name
23
- argument.type = type
24
- argument.description = description
25
- memo[argument.name.to_s] = argument
26
- memo
27
- end
28
- end
29
- end
30
-
31
- # Build a connection field from a {GraphQL::Field} by:
32
- # - Merging in the default arguments
33
- # - Transforming its resolve function to return a connection object
34
- def self.instrument(type, field)
35
- # Don't apply the wrapper to class-based fields, since they
36
- # use Schema::Field::ConnectionFilter
37
- if field.connection? && !field.metadata[:type_class]
38
- connection_arguments = default_arguments.merge(field.arguments)
39
- original_resolve = field.resolve_proc
40
- original_lazy_resolve = field.lazy_resolve_proc
41
- connection_resolve = GraphQL::Relay::ConnectionResolve.new(field, original_resolve)
42
- connection_lazy_resolve = GraphQL::Relay::ConnectionResolve.new(field, original_lazy_resolve)
43
- field.redefine(
44
- resolve: connection_resolve,
45
- lazy_resolve: connection_lazy_resolve,
46
- arguments: connection_arguments,
47
- )
48
- else
49
- field
50
- end
51
- end
52
- end
53
- end
54
- end
@@ -1,43 +0,0 @@
1
- # frozen_string_literal: true
2
- module GraphQL
3
- module Relay
4
- class ConnectionResolve
5
- def initialize(field, underlying_resolve)
6
- @field = field
7
- @underlying_resolve = underlying_resolve
8
- @max_page_size = field.connection_max_page_size
9
- end
10
-
11
- def call(obj, args, ctx)
12
- # in a lazy resolve hook, obj is the promise,
13
- # get the object that the promise was
14
- # originally derived from
15
- parent = ctx.object
16
-
17
- nodes = @underlying_resolve.call(obj, args, ctx)
18
-
19
- if nodes.nil? || ctx.schema.lazy?(nodes) || nodes.is_a?(GraphQL::Execution::Execute::Skip) || ctx.wrapped_connection
20
- nodes
21
- else
22
- ctx.wrapped_connection = true
23
- build_connection(nodes, args, parent, ctx)
24
- end
25
- end
26
-
27
- private
28
-
29
- def build_connection(nodes, args, parent, ctx)
30
- if nodes.is_a? GraphQL::ExecutionError
31
- ctx.add_error(nodes)
32
- nil
33
- else
34
- if parent.is_a?(GraphQL::Schema::Object)
35
- parent = parent.object
36
- end
37
- connection_class = GraphQL::Relay::BaseConnection.connection_for_nodes(nodes)
38
- connection_class.new(nodes, args, field: @field, max_page_size: @max_page_size, parent: parent, context: ctx)
39
- end
40
- end
41
- end
42
- end
43
- end
@@ -1,54 +0,0 @@
1
- # frozen_string_literal: true
2
- module GraphQL
3
- module Relay
4
- # @api deprecated
5
- module ConnectionType
6
- class << self
7
- # @return [Boolean] If true, connection types get a `nodes` shortcut field
8
- def default_nodes_field=(new_setting)
9
- if new_setting
10
- warn("GraphQL::Relay::ConnectionType will be removed in GraphQL 2.0.0; migrate to `GraphQL::Pagination::Connections` and remove this setting (`default_nodes_field = true`).")
11
- end
12
- @default_nodes_field = new_setting
13
- end
14
- attr_reader :default_nodes_field
15
-
16
- # @return [Boolean] If true, connections check for reverse-direction `has*Page` values
17
- def bidirectional_pagination=(new_setting)
18
- if new_setting
19
- warn("GraphQL::Relay::ConnectionType will be removed in GraphQL 2.0.0; migrate to `GraphQL::Pagination::Connections` and remove this setting (`bidirectional_pagination = true`).")
20
- end
21
- @bidirectional_pagination = new_setting
22
- end
23
- attr_reader :bidirectional_pagination
24
- end
25
-
26
- self.default_nodes_field = false
27
- self.bidirectional_pagination = false
28
-
29
- # @api deprecated
30
- def self.create_type(wrapped_type, edge_type: nil, edge_class: GraphQL::Relay::Edge, nodes_field: ConnectionType.default_nodes_field, &block)
31
- custom_edge_class = edge_class
32
-
33
- # Any call that would trigger `wrapped_type.ensure_defined`
34
- # must be inside this lazy block, otherwise we get weird
35
- # cyclical dependency errors :S
36
- ObjectType.deprecated_define do
37
- type_name = wrapped_type.is_a?(GraphQL::BaseType) ? wrapped_type.name : wrapped_type.graphql_name
38
- edge_type ||= wrapped_type.edge_type
39
- name("#{type_name}Connection")
40
- description("The connection type for #{type_name}.")
41
- field :edges, types[edge_type], "A list of edges.", edge_class: custom_edge_class, property: :edge_nodes
42
-
43
- if nodes_field
44
- field :nodes, types[wrapped_type], "A list of nodes.", property: :edge_nodes
45
- end
46
-
47
- field :pageInfo, !PageInfo, "Information to aid in pagination.", property: :page_info
48
- relay_node_type(wrapped_type)
49
- block && instance_eval(&block)
50
- end
51
- end
52
- end
53
- end
54
- end
@@ -1,27 +0,0 @@
1
- # frozen_string_literal: true
2
- module GraphQL
3
- module Relay
4
- # Mostly an internal concern.
5
- #
6
- # Wraps an object as a `node`, and exposes a connection-specific `cursor`.
7
- class Edge
8
- attr_reader :node, :connection
9
- def initialize(node, connection)
10
- @node = node
11
- @connection = connection
12
- end
13
-
14
- def cursor
15
- @cursor ||= connection.cursor_from_node(node)
16
- end
17
-
18
- def parent
19
- @parent ||= connection.parent
20
- end
21
-
22
- def inspect
23
- "#<GraphQL::Relay::Edge (#{parent.inspect} => #{node.inspect})>"
24
- end
25
- end
26
- end
27
- end
@@ -1,19 +0,0 @@
1
- # frozen_string_literal: true
2
- module GraphQL
3
- module Relay
4
- module EdgeType
5
- # @api deprecated
6
- def self.create_type(wrapped_type, name: nil, &block)
7
- GraphQL::ObjectType.define do
8
- type_name = wrapped_type.is_a?(GraphQL::BaseType) ? wrapped_type.name : wrapped_type.graphql_name
9
- name("#{type_name}Edge")
10
- description "An edge in a connection."
11
- field :node, wrapped_type, "The item at the end of the edge."
12
- field :cursor, !types.String, "A cursor for use in pagination."
13
- relay_node_type(wrapped_type)
14
- block && instance_eval(&block)
15
- end
16
- end
17
- end
18
- end
19
- end
@@ -1,39 +0,0 @@
1
- # frozen_string_literal: true
2
- module GraphQL
3
- module Relay
4
- module EdgesInstrumentation
5
- def self.instrument(type, field)
6
- if field.edges?
7
- edges_resolve = EdgesResolve.new(edge_class: field.edge_class, resolve: field.resolve_proc)
8
- edges_lazy_resolve = EdgesResolve.new(edge_class: field.edge_class, resolve: field.lazy_resolve_proc)
9
-
10
- field.redefine(
11
- resolve: edges_resolve,
12
- lazy_resolve: edges_lazy_resolve,
13
- )
14
- else
15
- field
16
- end
17
- end
18
-
19
- class EdgesResolve
20
- def initialize(edge_class:, resolve:)
21
- @edge_class = edge_class
22
- @resolve_proc = resolve
23
- end
24
-
25
- # A user's custom Connection may return a lazy object,
26
- # if so, handle it later.
27
- def call(obj, args, ctx)
28
- parent = ctx.object
29
- nodes = @resolve_proc.call(obj, args, ctx)
30
- if ctx.schema.lazy?(nodes)
31
- nodes
32
- else
33
- nodes.map { |item| item.is_a?(GraphQL::Pagination::Connection::Edge) ? item : @edge_class.new(item, parent) }
34
- end
35
- end
36
- end
37
- end
38
- end
39
- end
@@ -1,17 +0,0 @@
1
- # frozen_string_literal: true
2
- module GraphQL
3
- module Relay
4
- class GlobalIdResolve
5
- def initialize(type:)
6
- @type = type
7
- end
8
-
9
- def call(obj, args, ctx)
10
- if obj.is_a?(GraphQL::Schema::Object)
11
- obj = obj.object
12
- end
13
- ctx.query.schema.id_from_object(obj, @type, ctx)
14
- end
15
- end
16
- end
17
- end
@@ -1,50 +0,0 @@
1
- # frozen_string_literal: true
2
- module GraphQL
3
- module Relay
4
- # A connection implementation to expose MongoDB collection objects.
5
- # It works for:
6
- # - `Mongoid::Criteria`
7
- class MongoRelationConnection < RelationConnection
8
- private
9
-
10
- def relation_offset(relation)
11
- relation.options.skip
12
- end
13
-
14
- def relation_limit(relation)
15
- relation.options.limit
16
- end
17
-
18
- def relation_count(relation)
19
- # Must perform query (hence #to_a) to count results https://jira.mongodb.org/browse/MONGOID-2325
20
- relation.to_a.count
21
- end
22
-
23
- def limit_nodes(sliced_nodes, limit)
24
- if limit == 0
25
- if sliced_nodes.respond_to?(:none) # added in Mongoid 4.0
26
- sliced_nodes.without_options.none
27
- else
28
- sliced_nodes.where(id: nil) # trying to simulate #none for 3.1.7
29
- end
30
- else
31
- sliced_nodes.limit(limit)
32
- end
33
- end
34
- end
35
-
36
- if defined?(Mongoid::Criteria)
37
- BaseConnection.register_connection_implementation(Mongoid::Criteria, MongoRelationConnection)
38
- end
39
-
40
- # Mongoid 5 and 6
41
- if defined?(Mongoid::Relations::Targets::Enumerable)
42
- BaseConnection.register_connection_implementation(Mongoid::Relations::Targets::Enumerable, MongoRelationConnection)
43
- end
44
-
45
- # Mongoid 7
46
- if defined?(Mongoid::Association::Referenced::HasMany::Targets::Enumerable)
47
- BaseConnection.register_connection_implementation(Mongoid::Association::Referenced::HasMany::Targets::Enumerable, MongoRelationConnection)
48
- end
49
- end
50
- end
@@ -1,23 +0,0 @@
1
- # frozen_string_literal: true
2
- module GraphQL
3
- module Relay
4
- class Mutation
5
- # @api private
6
- module Instrumentation
7
- # Modify mutation `return_field` resolves by wrapping the returned object
8
- # in a {Mutation::Result}.
9
- #
10
- # By using an instrumention, we can apply our wrapper _last_,
11
- # giving users access to the original resolve function in earlier instrumentation.
12
- def self.instrument(type, field)
13
- if field.mutation.is_a?(GraphQL::Relay::Mutation) || (field.mutation.is_a?(Class) && field.mutation < GraphQL::Schema::RelayClassicMutation)
14
- new_resolve = Mutation::Resolve.new(field.mutation, field.resolve_proc)
15
- field.redefine(resolve: new_resolve)
16
- else
17
- field
18
- end
19
- end
20
- end
21
- end
22
- end
23
- end
@@ -1,56 +0,0 @@
1
- # frozen_string_literal: true
2
- module GraphQL
3
- module Relay
4
- class Mutation
5
- # Wrap a user-provided resolve function,
6
- # wrapping the returned value in a {Mutation::Result}.
7
- # Also, pass the `clientMutationId` to that result object.
8
- # @api private
9
- class Resolve
10
- def initialize(mutation, resolve)
11
- @mutation = mutation
12
- @resolve = resolve
13
- @wrap_result = mutation.is_a?(GraphQL::Relay::Mutation) && mutation.has_generated_return_type?
14
- @class_based = mutation.is_a?(Class)
15
- end
16
-
17
- def call(obj, args, ctx)
18
- mutation_result = begin
19
- @resolve.call(obj, args[:input], ctx)
20
- rescue GraphQL::ExecutionError => err
21
- err
22
- end
23
-
24
- ctx.schema.after_lazy(mutation_result) do |res|
25
- build_result(res, args, ctx)
26
- end
27
- end
28
-
29
- private
30
-
31
- def build_result(mutation_result, args, ctx)
32
- if mutation_result.is_a?(GraphQL::ExecutionError)
33
- ctx.add_error(mutation_result)
34
- mutation_result = nil
35
- end
36
-
37
- if mutation_result.nil?
38
- nil
39
- elsif @wrap_result
40
- if mutation_result && !mutation_result.is_a?(Hash)
41
- raise StandardError, "Expected `#{mutation_result}` to be a Hash."\
42
- " Return a hash when using `return_field` or specify a custom `return_type`."
43
- end
44
-
45
- @mutation.result_class.new(client_mutation_id: args[:input][:clientMutationId], result: mutation_result)
46
- elsif @class_based
47
- mutation_result[:client_mutation_id] = args[:input][:client_mutation_id]
48
- mutation_result
49
- else
50
- mutation_result
51
- end
52
- end
53
- end
54
- end
55
- end
56
- end