graphql 1.9.18 → 1.13.24

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.
Files changed (353) hide show
  1. checksums.yaml +4 -4
  2. data/lib/generators/graphql/core.rb +21 -10
  3. data/lib/generators/graphql/enum_generator.rb +4 -10
  4. data/lib/generators/graphql/field_extractor.rb +31 -0
  5. data/lib/generators/graphql/input_generator.rb +50 -0
  6. data/lib/generators/graphql/install/mutation_root_generator.rb +34 -0
  7. data/lib/generators/graphql/{templates → install/templates}/base_mutation.erb +2 -0
  8. data/lib/generators/graphql/{templates → install/templates}/mutation_type.erb +2 -0
  9. data/lib/generators/graphql/install_generator.rb +44 -7
  10. data/lib/generators/graphql/interface_generator.rb +7 -7
  11. data/lib/generators/graphql/loader_generator.rb +1 -0
  12. data/lib/generators/graphql/mutation_create_generator.rb +22 -0
  13. data/lib/generators/graphql/mutation_delete_generator.rb +22 -0
  14. data/lib/generators/graphql/mutation_generator.rb +6 -30
  15. data/lib/generators/graphql/mutation_update_generator.rb +22 -0
  16. data/lib/generators/graphql/object_generator.rb +28 -12
  17. data/lib/generators/graphql/orm_mutations_base.rb +40 -0
  18. data/lib/generators/graphql/relay.rb +63 -0
  19. data/lib/generators/graphql/relay_generator.rb +21 -0
  20. data/lib/generators/graphql/scalar_generator.rb +4 -2
  21. data/lib/generators/graphql/templates/base_argument.erb +2 -0
  22. data/lib/generators/graphql/templates/base_connection.erb +8 -0
  23. data/lib/generators/graphql/templates/base_edge.erb +8 -0
  24. data/lib/generators/graphql/templates/base_enum.erb +2 -0
  25. data/lib/generators/graphql/templates/base_field.erb +2 -0
  26. data/lib/generators/graphql/templates/base_input_object.erb +2 -0
  27. data/lib/generators/graphql/templates/base_interface.erb +2 -0
  28. data/lib/generators/graphql/templates/base_object.erb +2 -0
  29. data/lib/generators/graphql/templates/base_scalar.erb +2 -0
  30. data/lib/generators/graphql/templates/base_union.erb +2 -0
  31. data/lib/generators/graphql/templates/enum.erb +7 -1
  32. data/lib/generators/graphql/templates/graphql_controller.erb +16 -12
  33. data/lib/generators/graphql/templates/input.erb +9 -0
  34. data/lib/generators/graphql/templates/interface.erb +6 -2
  35. data/lib/generators/graphql/templates/loader.erb +2 -0
  36. data/lib/generators/graphql/templates/mutation.erb +3 -1
  37. data/lib/generators/graphql/templates/mutation_create.erb +20 -0
  38. data/lib/generators/graphql/templates/mutation_delete.erb +20 -0
  39. data/lib/generators/graphql/templates/mutation_update.erb +21 -0
  40. data/lib/generators/graphql/templates/node_type.erb +9 -0
  41. data/lib/generators/graphql/templates/object.erb +7 -3
  42. data/lib/generators/graphql/templates/query_type.erb +3 -3
  43. data/lib/generators/graphql/templates/scalar.erb +5 -1
  44. data/lib/generators/graphql/templates/schema.erb +22 -27
  45. data/lib/generators/graphql/templates/union.erb +6 -2
  46. data/lib/generators/graphql/type_generator.rb +47 -10
  47. data/lib/generators/graphql/union_generator.rb +5 -5
  48. data/lib/graphql/analysis/analyze_query.rb +7 -0
  49. data/lib/graphql/analysis/ast/field_usage.rb +29 -2
  50. data/lib/graphql/analysis/ast/query_complexity.rb +174 -67
  51. data/lib/graphql/analysis/ast/visitor.rb +16 -7
  52. data/lib/graphql/analysis/ast.rb +21 -11
  53. data/lib/graphql/argument.rb +8 -36
  54. data/lib/graphql/backtrace/inspect_result.rb +0 -1
  55. data/lib/graphql/backtrace/legacy_tracer.rb +56 -0
  56. data/lib/graphql/backtrace/table.rb +44 -5
  57. data/lib/graphql/backtrace/traced_error.rb +0 -1
  58. data/lib/graphql/backtrace/tracer.rb +40 -9
  59. data/lib/graphql/backtrace.rb +28 -19
  60. data/lib/graphql/backwards_compatibility.rb +2 -1
  61. data/lib/graphql/base_type.rb +10 -4
  62. data/lib/graphql/boolean_type.rb +1 -1
  63. data/lib/graphql/compatibility/execution_specification/specification_schema.rb +2 -2
  64. data/lib/graphql/compatibility/execution_specification.rb +1 -0
  65. data/lib/graphql/compatibility/lazy_execution_specification.rb +2 -0
  66. data/lib/graphql/compatibility/query_parser_specification/parse_error_specification.rb +5 -9
  67. data/lib/graphql/compatibility/query_parser_specification.rb +2 -0
  68. data/lib/graphql/compatibility/schema_parser_specification.rb +2 -0
  69. data/lib/graphql/dataloader/null_dataloader.rb +22 -0
  70. data/lib/graphql/dataloader/request.rb +19 -0
  71. data/lib/graphql/dataloader/request_all.rb +19 -0
  72. data/lib/graphql/dataloader/source.rb +155 -0
  73. data/lib/graphql/dataloader.rb +308 -0
  74. data/lib/graphql/date_encoding_error.rb +16 -0
  75. data/lib/graphql/define/assign_enum_value.rb +1 -1
  76. data/lib/graphql/define/assign_global_id_field.rb +2 -2
  77. data/lib/graphql/define/assign_object_field.rb +1 -1
  78. data/lib/graphql/define/defined_object_proxy.rb +5 -8
  79. data/lib/graphql/define/instance_definable.rb +60 -110
  80. data/lib/graphql/define/type_definer.rb +5 -5
  81. data/lib/graphql/deprecated_dsl.rb +18 -5
  82. data/lib/graphql/deprecation.rb +9 -0
  83. data/lib/graphql/directive/deprecated_directive.rb +1 -12
  84. data/lib/graphql/directive/include_directive.rb +1 -1
  85. data/lib/graphql/directive/skip_directive.rb +1 -1
  86. data/lib/graphql/directive.rb +9 -6
  87. data/lib/graphql/enum_type.rb +14 -74
  88. data/lib/graphql/execution/directive_checks.rb +2 -2
  89. data/lib/graphql/execution/errors.rb +110 -8
  90. data/lib/graphql/execution/execute.rb +8 -1
  91. data/lib/graphql/execution/instrumentation.rb +1 -1
  92. data/lib/graphql/execution/interpreter/argument_value.rb +28 -0
  93. data/lib/graphql/execution/interpreter/arguments.rb +88 -0
  94. data/lib/graphql/execution/interpreter/arguments_cache.rb +105 -0
  95. data/lib/graphql/execution/interpreter/handles_raw_value.rb +18 -0
  96. data/lib/graphql/execution/interpreter/resolve.rb +37 -25
  97. data/lib/graphql/execution/interpreter/runtime.rb +721 -386
  98. data/lib/graphql/execution/interpreter.rb +42 -19
  99. data/lib/graphql/execution/lazy/lazy_method_map.rb +4 -0
  100. data/lib/graphql/execution/lazy.rb +5 -1
  101. data/lib/graphql/execution/lookahead.rb +39 -114
  102. data/lib/graphql/execution/multiplex.rb +50 -25
  103. data/lib/graphql/field.rb +15 -119
  104. data/lib/graphql/filter.rb +1 -1
  105. data/lib/graphql/float_type.rb +1 -1
  106. data/lib/graphql/function.rb +5 -30
  107. data/lib/graphql/id_type.rb +1 -1
  108. data/lib/graphql/input_object_type.rb +9 -25
  109. data/lib/graphql/int_type.rb +1 -1
  110. data/lib/graphql/integer_decoding_error.rb +17 -0
  111. data/lib/graphql/integer_encoding_error.rb +18 -2
  112. data/lib/graphql/interface_type.rb +10 -24
  113. data/lib/graphql/internal_representation/document.rb +2 -2
  114. data/lib/graphql/internal_representation/rewrite.rb +1 -1
  115. data/lib/graphql/internal_representation/scope.rb +2 -2
  116. data/lib/graphql/internal_representation/visit.rb +2 -2
  117. data/lib/graphql/introspection/base_object.rb +2 -5
  118. data/lib/graphql/introspection/directive_location_enum.rb +2 -2
  119. data/lib/graphql/introspection/directive_type.rb +12 -6
  120. data/lib/graphql/introspection/entry_points.rb +9 -9
  121. data/lib/graphql/introspection/enum_value_type.rb +2 -2
  122. data/lib/graphql/introspection/field_type.rb +9 -5
  123. data/lib/graphql/introspection/input_value_type.rb +41 -11
  124. data/lib/graphql/introspection/introspection_query.rb +6 -92
  125. data/lib/graphql/introspection/schema_type.rb +12 -12
  126. data/lib/graphql/introspection/type_type.rb +27 -17
  127. data/lib/graphql/introspection.rb +99 -0
  128. data/lib/graphql/invalid_null_error.rb +18 -0
  129. data/lib/graphql/language/block_string.rb +20 -5
  130. data/lib/graphql/language/cache.rb +37 -0
  131. data/lib/graphql/language/definition_slice.rb +21 -10
  132. data/lib/graphql/language/document_from_schema_definition.rb +116 -63
  133. data/lib/graphql/language/lexer.rb +53 -27
  134. data/lib/graphql/language/lexer.rl +5 -3
  135. data/lib/graphql/language/nodes.rb +67 -93
  136. data/lib/graphql/language/parser.rb +929 -896
  137. data/lib/graphql/language/parser.y +125 -102
  138. data/lib/graphql/language/printer.rb +11 -2
  139. data/lib/graphql/language/sanitized_printer.rb +222 -0
  140. data/lib/graphql/language/token.rb +0 -4
  141. data/lib/graphql/language/visitor.rb +2 -2
  142. data/lib/graphql/language.rb +3 -1
  143. data/lib/graphql/name_validator.rb +2 -7
  144. data/lib/graphql/non_null_type.rb +0 -10
  145. data/lib/graphql/object_type.rb +47 -58
  146. data/lib/graphql/pagination/active_record_relation_connection.rb +85 -0
  147. data/lib/graphql/pagination/array_connection.rb +77 -0
  148. data/lib/graphql/pagination/connection.rb +226 -0
  149. data/lib/graphql/pagination/connections.rb +160 -0
  150. data/lib/graphql/pagination/mongoid_relation_connection.rb +25 -0
  151. data/lib/graphql/pagination/relation_connection.rb +226 -0
  152. data/lib/graphql/pagination/sequel_dataset_connection.rb +28 -0
  153. data/lib/graphql/pagination.rb +6 -0
  154. data/lib/graphql/parse_error.rb +0 -1
  155. data/lib/graphql/query/arguments.rb +6 -4
  156. data/lib/graphql/query/arguments_cache.rb +1 -2
  157. data/lib/graphql/query/context.rb +52 -7
  158. data/lib/graphql/query/executor.rb +0 -1
  159. data/lib/graphql/query/fingerprint.rb +26 -0
  160. data/lib/graphql/query/input_validation_result.rb +32 -6
  161. data/lib/graphql/query/literal_input.rb +31 -11
  162. data/lib/graphql/query/null_context.rb +24 -8
  163. data/lib/graphql/query/serial_execution/field_resolution.rb +1 -1
  164. data/lib/graphql/query/serial_execution.rb +1 -0
  165. data/lib/graphql/query/validation_pipeline.rb +6 -4
  166. data/lib/graphql/query/variable_validation_error.rb +3 -3
  167. data/lib/graphql/query/variables.rb +50 -10
  168. data/lib/graphql/query.rb +77 -18
  169. data/lib/graphql/railtie.rb +9 -1
  170. data/lib/graphql/rake_task/validate.rb +3 -0
  171. data/lib/graphql/rake_task.rb +12 -9
  172. data/lib/graphql/relay/array_connection.rb +10 -12
  173. data/lib/graphql/relay/base_connection.rb +30 -13
  174. data/lib/graphql/relay/connection_instrumentation.rb +4 -4
  175. data/lib/graphql/relay/connection_type.rb +18 -4
  176. data/lib/graphql/relay/edge_type.rb +1 -0
  177. data/lib/graphql/relay/edges_instrumentation.rb +1 -2
  178. data/lib/graphql/relay/global_id_resolve.rb +1 -2
  179. data/lib/graphql/relay/mutation.rb +3 -87
  180. data/lib/graphql/relay/node.rb +3 -0
  181. data/lib/graphql/relay/page_info.rb +1 -1
  182. data/lib/graphql/relay/range_add.rb +27 -9
  183. data/lib/graphql/relay/relation_connection.rb +8 -10
  184. data/lib/graphql/relay/type_extensions.rb +2 -0
  185. data/lib/graphql/rubocop/graphql/base_cop.rb +36 -0
  186. data/lib/graphql/rubocop/graphql/default_null_true.rb +43 -0
  187. data/lib/graphql/rubocop/graphql/default_required_true.rb +43 -0
  188. data/lib/graphql/rubocop.rb +4 -0
  189. data/lib/graphql/scalar_type.rb +18 -60
  190. data/lib/graphql/schema/addition.rb +247 -0
  191. data/lib/graphql/schema/argument.rb +274 -18
  192. data/lib/graphql/schema/base_64_encoder.rb +2 -0
  193. data/lib/graphql/schema/build_from_definition/resolve_map/default_resolve.rb +1 -1
  194. data/lib/graphql/schema/build_from_definition/resolve_map.rb +13 -5
  195. data/lib/graphql/schema/build_from_definition.rb +320 -219
  196. data/lib/graphql/schema/built_in_types.rb +5 -5
  197. data/lib/graphql/schema/default_type_error.rb +2 -0
  198. data/lib/graphql/schema/directive/deprecated.rb +18 -0
  199. data/lib/graphql/schema/directive/feature.rb +1 -1
  200. data/lib/graphql/schema/directive/flagged.rb +57 -0
  201. data/lib/graphql/schema/directive/include.rb +2 -2
  202. data/lib/graphql/schema/directive/skip.rb +2 -2
  203. data/lib/graphql/schema/directive/transform.rb +14 -2
  204. data/lib/graphql/schema/directive.rb +130 -6
  205. data/lib/graphql/schema/enum.rb +121 -12
  206. data/lib/graphql/schema/enum_value.rb +24 -7
  207. data/lib/graphql/schema/field/connection_extension.rb +46 -20
  208. data/lib/graphql/schema/field/scope_extension.rb +1 -1
  209. data/lib/graphql/schema/field.rb +465 -181
  210. data/lib/graphql/schema/field_extension.rb +89 -2
  211. data/lib/graphql/schema/find_inherited_value.rb +17 -1
  212. data/lib/graphql/schema/finder.rb +16 -14
  213. data/lib/graphql/schema/input_object.rb +172 -37
  214. data/lib/graphql/schema/interface.rb +39 -25
  215. data/lib/graphql/schema/introspection_system.rb +106 -38
  216. data/lib/graphql/schema/late_bound_type.rb +3 -2
  217. data/lib/graphql/schema/list.rb +65 -1
  218. data/lib/graphql/schema/loader.rb +145 -102
  219. data/lib/graphql/schema/member/accepts_definition.rb +15 -3
  220. data/lib/graphql/schema/member/base_dsl_methods.rb +34 -28
  221. data/lib/graphql/schema/member/build_type.rb +19 -8
  222. data/lib/graphql/schema/member/cached_graphql_definition.rb +34 -2
  223. data/lib/graphql/schema/member/has_arguments.rb +206 -13
  224. data/lib/graphql/schema/member/has_ast_node.rb +20 -0
  225. data/lib/graphql/schema/member/has_deprecation_reason.rb +25 -0
  226. data/lib/graphql/schema/member/has_directives.rb +98 -0
  227. data/lib/graphql/schema/member/has_fields.rb +97 -32
  228. data/lib/graphql/schema/member/has_interfaces.rb +100 -0
  229. data/lib/graphql/schema/member/has_unresolved_type_error.rb +15 -0
  230. data/lib/graphql/schema/member/has_validators.rb +31 -0
  231. data/lib/graphql/schema/member/instrumentation.rb +0 -1
  232. data/lib/graphql/schema/member/type_system_helpers.rb +3 -3
  233. data/lib/graphql/schema/member/validates_input.rb +33 -0
  234. data/lib/graphql/schema/member.rb +11 -0
  235. data/lib/graphql/schema/middleware_chain.rb +1 -1
  236. data/lib/graphql/schema/mutation.rb +4 -0
  237. data/lib/graphql/schema/non_null.rb +37 -1
  238. data/lib/graphql/schema/object.rb +51 -38
  239. data/lib/graphql/schema/possible_types.rb +9 -4
  240. data/lib/graphql/schema/printer.rb +16 -35
  241. data/lib/graphql/schema/relay_classic_mutation.rb +40 -4
  242. data/lib/graphql/schema/resolver/has_payload_type.rb +34 -4
  243. data/lib/graphql/schema/resolver.rb +133 -79
  244. data/lib/graphql/schema/scalar.rb +43 -3
  245. data/lib/graphql/schema/subscription.rb +57 -21
  246. data/lib/graphql/schema/timeout.rb +29 -15
  247. data/lib/graphql/schema/timeout_middleware.rb +3 -1
  248. data/lib/graphql/schema/traversal.rb +2 -2
  249. data/lib/graphql/schema/type_expression.rb +21 -13
  250. data/lib/graphql/schema/type_membership.rb +19 -5
  251. data/lib/graphql/schema/union.rb +44 -3
  252. data/lib/graphql/schema/unique_within_type.rb +1 -2
  253. data/lib/graphql/schema/validation.rb +14 -4
  254. data/lib/graphql/schema/validator/allow_blank_validator.rb +29 -0
  255. data/lib/graphql/schema/validator/allow_null_validator.rb +26 -0
  256. data/lib/graphql/schema/validator/exclusion_validator.rb +33 -0
  257. data/lib/graphql/schema/validator/format_validator.rb +48 -0
  258. data/lib/graphql/schema/validator/inclusion_validator.rb +35 -0
  259. data/lib/graphql/schema/validator/length_validator.rb +59 -0
  260. data/lib/graphql/schema/validator/numericality_validator.rb +82 -0
  261. data/lib/graphql/schema/validator/required_validator.rb +82 -0
  262. data/lib/graphql/schema/validator.rb +171 -0
  263. data/lib/graphql/schema/warden.rb +193 -34
  264. data/lib/graphql/schema.rb +882 -247
  265. data/lib/graphql/static_validation/all_rules.rb +2 -0
  266. data/lib/graphql/static_validation/base_visitor.rb +17 -10
  267. data/lib/graphql/static_validation/definition_dependencies.rb +0 -1
  268. data/lib/graphql/static_validation/error.rb +3 -1
  269. data/lib/graphql/static_validation/literal_validator.rb +51 -26
  270. data/lib/graphql/static_validation/rules/argument_literals_are_compatible.rb +45 -83
  271. data/lib/graphql/static_validation/rules/argument_literals_are_compatible_error.rb +22 -6
  272. data/lib/graphql/static_validation/rules/arguments_are_defined.rb +35 -26
  273. data/lib/graphql/static_validation/rules/arguments_are_defined_error.rb +4 -2
  274. data/lib/graphql/static_validation/rules/directives_are_defined.rb +1 -1
  275. data/lib/graphql/static_validation/rules/directives_are_in_valid_locations.rb +2 -2
  276. data/lib/graphql/static_validation/rules/fields_are_defined_on_type.rb +4 -4
  277. data/lib/graphql/static_validation/rules/fields_have_appropriate_selections.rb +5 -5
  278. data/lib/graphql/static_validation/rules/fields_will_merge.rb +94 -51
  279. data/lib/graphql/static_validation/rules/fields_will_merge_error.rb +25 -4
  280. data/lib/graphql/static_validation/rules/fragment_spreads_are_possible.rb +3 -3
  281. data/lib/graphql/static_validation/rules/fragments_are_finite.rb +2 -2
  282. data/lib/graphql/static_validation/rules/input_object_names_are_unique.rb +30 -0
  283. data/lib/graphql/static_validation/rules/input_object_names_are_unique_error.rb +30 -0
  284. data/lib/graphql/static_validation/rules/query_root_exists.rb +17 -0
  285. data/lib/graphql/static_validation/rules/query_root_exists_error.rb +26 -0
  286. data/lib/graphql/static_validation/rules/required_arguments_are_present.rb +4 -2
  287. data/lib/graphql/static_validation/rules/required_input_object_attributes_are_present.rb +9 -10
  288. data/lib/graphql/static_validation/rules/unique_directives_per_location.rb +1 -1
  289. data/lib/graphql/static_validation/rules/variable_default_values_are_correctly_typed.rb +12 -13
  290. data/lib/graphql/static_validation/rules/variable_usages_are_allowed.rb +19 -14
  291. data/lib/graphql/static_validation/rules/variables_are_input_types.rb +1 -1
  292. data/lib/graphql/static_validation/rules/variables_are_used_and_defined.rb +5 -3
  293. data/lib/graphql/static_validation/type_stack.rb +2 -2
  294. data/lib/graphql/static_validation/validation_context.rb +13 -3
  295. data/lib/graphql/static_validation/validation_timeout_error.rb +25 -0
  296. data/lib/graphql/static_validation/validator.rb +43 -9
  297. data/lib/graphql/static_validation.rb +1 -0
  298. data/lib/graphql/string_encoding_error.rb +13 -3
  299. data/lib/graphql/string_type.rb +1 -1
  300. data/lib/graphql/subscriptions/action_cable_subscriptions.rb +123 -22
  301. data/lib/graphql/subscriptions/broadcast_analyzer.rb +81 -0
  302. data/lib/graphql/subscriptions/default_subscription_resolve_extension.rb +21 -0
  303. data/lib/graphql/subscriptions/event.rb +84 -30
  304. data/lib/graphql/subscriptions/instrumentation.rb +10 -6
  305. data/lib/graphql/subscriptions/serialize.rb +53 -6
  306. data/lib/graphql/subscriptions/subscription_root.rb +15 -5
  307. data/lib/graphql/subscriptions.rb +117 -49
  308. data/lib/graphql/tracing/active_support_notifications_tracing.rb +8 -17
  309. data/lib/graphql/tracing/appoptics_tracing.rb +173 -0
  310. data/lib/graphql/tracing/appsignal_tracing.rb +23 -0
  311. data/lib/graphql/tracing/data_dog_tracing.rb +32 -15
  312. data/lib/graphql/tracing/new_relic_tracing.rb +9 -12
  313. data/lib/graphql/tracing/notifications_tracing.rb +59 -0
  314. data/lib/graphql/tracing/platform_tracing.rb +66 -10
  315. data/lib/graphql/tracing/prometheus_tracing/graphql_collector.rb +4 -1
  316. data/lib/graphql/tracing/prometheus_tracing.rb +8 -0
  317. data/lib/graphql/tracing/scout_tracing.rb +19 -0
  318. data/lib/graphql/tracing/skylight_tracing.rb +9 -1
  319. data/lib/graphql/tracing/statsd_tracing.rb +42 -0
  320. data/lib/graphql/tracing.rb +15 -35
  321. data/lib/graphql/types/big_int.rb +5 -1
  322. data/lib/graphql/types/int.rb +10 -3
  323. data/lib/graphql/types/iso_8601_date.rb +16 -8
  324. data/lib/graphql/types/iso_8601_date_time.rb +32 -10
  325. data/lib/graphql/types/relay/base_connection.rb +6 -88
  326. data/lib/graphql/types/relay/base_edge.rb +2 -34
  327. data/lib/graphql/types/relay/connection_behaviors.rb +174 -0
  328. data/lib/graphql/types/relay/default_relay.rb +31 -0
  329. data/lib/graphql/types/relay/edge_behaviors.rb +64 -0
  330. data/lib/graphql/types/relay/has_node_field.rb +41 -0
  331. data/lib/graphql/types/relay/has_nodes_field.rb +41 -0
  332. data/lib/graphql/types/relay/node.rb +2 -4
  333. data/lib/graphql/types/relay/node_behaviors.rb +15 -0
  334. data/lib/graphql/types/relay/node_field.rb +3 -22
  335. data/lib/graphql/types/relay/nodes_field.rb +16 -18
  336. data/lib/graphql/types/relay/page_info.rb +2 -14
  337. data/lib/graphql/types/relay/page_info_behaviors.rb +25 -0
  338. data/lib/graphql/types/relay.rb +11 -3
  339. data/lib/graphql/types/string.rb +8 -2
  340. data/lib/graphql/unauthorized_error.rb +2 -2
  341. data/lib/graphql/union_type.rb +5 -25
  342. data/lib/graphql/unresolved_type_error.rb +2 -2
  343. data/lib/graphql/upgrader/member.rb +1 -0
  344. data/lib/graphql/upgrader/schema.rb +1 -0
  345. data/lib/graphql/version.rb +1 -1
  346. data/lib/graphql.rb +87 -31
  347. data/readme.md +3 -6
  348. metadata +126 -124
  349. data/lib/graphql/execution/interpreter/hash_response.rb +0 -46
  350. data/lib/graphql/literal_validation_error.rb +0 -6
  351. data/lib/graphql/types/relay/base_field.rb +0 -22
  352. data/lib/graphql/types/relay/base_interface.rb +0 -29
  353. data/lib/graphql/types/relay/base_object.rb +0 -26
@@ -8,6 +8,137 @@ module GraphQL
8
8
  #
9
9
  # @api private
10
10
  class Runtime
11
+
12
+ module GraphQLResult
13
+ def initialize(result_name, parent_result)
14
+ @graphql_parent = parent_result
15
+ if parent_result && parent_result.graphql_dead
16
+ @graphql_dead = true
17
+ end
18
+ @graphql_result_name = result_name
19
+ # Jump through some hoops to avoid creating this duplicate storage if at all possible.
20
+ @graphql_metadata = nil
21
+ end
22
+
23
+ attr_accessor :graphql_dead
24
+ attr_reader :graphql_parent, :graphql_result_name
25
+
26
+ # Although these are used by only one of the Result classes,
27
+ # it's handy to have the methods implemented on both (even though they just return `nil`)
28
+ # because it makes it easy to check if anything is assigned.
29
+ # @return [nil, Array<String>]
30
+ attr_accessor :graphql_non_null_field_names
31
+ # @return [nil, true]
32
+ attr_accessor :graphql_non_null_list_items
33
+
34
+ # @return [Hash] Plain-Ruby result data (`@graphql_metadata` contains Result wrapper objects)
35
+ attr_accessor :graphql_result_data
36
+ end
37
+
38
+ class GraphQLResultHash
39
+ def initialize(_result_name, _parent_result)
40
+ super
41
+ @graphql_result_data = {}
42
+ end
43
+
44
+ include GraphQLResult
45
+
46
+ attr_accessor :graphql_merged_into
47
+
48
+ def []=(key, value)
49
+ # This is a hack.
50
+ # Basically, this object is merged into the root-level result at some point.
51
+ # But the problem is, some lazies are created whose closures retain reference to _this_
52
+ # object. When those lazies are resolved, they cause an update to this object.
53
+ #
54
+ # In order to return a proper top-level result, we have to update that top-level result object.
55
+ # In order to return a proper partial result (eg, for a directive), we have to update this object, too.
56
+ # Yowza.
57
+ if (t = @graphql_merged_into)
58
+ t[key] = value
59
+ end
60
+
61
+ if value.respond_to?(:graphql_result_data)
62
+ @graphql_result_data[key] = value.graphql_result_data
63
+ # If we encounter some part of this response that requires metadata tracking,
64
+ # then create the metadata hash if necessary. It will be kept up-to-date after this.
65
+ (@graphql_metadata ||= @graphql_result_data.dup)[key] = value
66
+ else
67
+ @graphql_result_data[key] = value
68
+ # keep this up-to-date if it's been initialized
69
+ @graphql_metadata && @graphql_metadata[key] = value
70
+ end
71
+
72
+ value
73
+ end
74
+
75
+ def delete(key)
76
+ @graphql_metadata && @graphql_metadata.delete(key)
77
+ @graphql_result_data.delete(key)
78
+ end
79
+
80
+ def each
81
+ (@graphql_metadata || @graphql_result_data).each { |k, v| yield(k, v) }
82
+ end
83
+
84
+ def values
85
+ (@graphql_metadata || @graphql_result_data).values
86
+ end
87
+
88
+ def key?(k)
89
+ @graphql_result_data.key?(k)
90
+ end
91
+
92
+ def [](k)
93
+ (@graphql_metadata || @graphql_result_data)[k]
94
+ end
95
+ end
96
+
97
+ class GraphQLResultArray
98
+ include GraphQLResult
99
+
100
+ def initialize(_result_name, _parent_result)
101
+ super
102
+ @graphql_result_data = []
103
+ end
104
+
105
+ def graphql_skip_at(index)
106
+ # Mark this index as dead. It's tricky because some indices may already be storing
107
+ # `Lazy`s. So the runtime is still holding indexes _before_ skipping,
108
+ # this object has to coordinate incoming writes to account for any already-skipped indices.
109
+ @skip_indices ||= []
110
+ @skip_indices << index
111
+ offset_by = @skip_indices.count { |skipped_idx| skipped_idx < index}
112
+ delete_at_index = index - offset_by
113
+ @graphql_metadata && @graphql_metadata.delete_at(delete_at_index)
114
+ @graphql_result_data.delete_at(delete_at_index)
115
+ end
116
+
117
+ def []=(idx, value)
118
+ if @skip_indices
119
+ offset_by = @skip_indices.count { |skipped_idx| skipped_idx < idx }
120
+ idx -= offset_by
121
+ end
122
+ if value.respond_to?(:graphql_result_data)
123
+ @graphql_result_data[idx] = value.graphql_result_data
124
+ (@graphql_metadata ||= @graphql_result_data.dup)[idx] = value
125
+ else
126
+ @graphql_result_data[idx] = value
127
+ @graphql_metadata && @graphql_metadata[idx] = value
128
+ end
129
+
130
+ value
131
+ end
132
+
133
+ def values
134
+ (@graphql_metadata || @graphql_result_data)
135
+ end
136
+ end
137
+
138
+ class GraphQLSelectionSet < Hash
139
+ attr_accessor :graphql_directives
140
+ end
141
+
11
142
  # @return [GraphQL::Query]
12
143
  attr_reader :query
13
144
 
@@ -17,61 +148,139 @@ module GraphQL
17
148
  # @return [GraphQL::Query::Context]
18
149
  attr_reader :context
19
150
 
20
- def initialize(query:, response:)
151
+ def initialize(query:)
21
152
  @query = query
153
+ @dataloader = query.multiplex.dataloader
22
154
  @schema = query.schema
23
155
  @context = query.context
156
+ @multiplex_context = query.multiplex.context
24
157
  @interpreter_context = @context.namespace(:interpreter)
25
- @response = response
26
- @dead_paths = {}
27
- @types_at_paths = {}
158
+ @response = GraphQLResultHash.new(nil, nil)
159
+ # Identify runtime directives by checking which of this schema's directives have overridden `def self.resolve`
160
+ @runtime_directive_names = []
161
+ noop_resolve_owner = GraphQL::Schema::Directive.singleton_class
162
+ @schema_directives = schema.directives
163
+ @schema_directives.each do |name, dir_defn|
164
+ if dir_defn.method(:resolve).owner != noop_resolve_owner
165
+ @runtime_directive_names << name
166
+ end
167
+ end
28
168
  # A cache of { Class => { String => Schema::Field } }
29
169
  # Which assumes that MyObject.get_field("myField") will return the same field
30
170
  # during the lifetime of a query
31
171
  @fields_cache = Hash.new { |h, k| h[k] = {} }
172
+ # { Class => Boolean }
173
+ @lazy_cache = {}
32
174
  end
33
175
 
34
- def final_value
35
- @response.final_value
176
+ def final_result
177
+ @response && @response.graphql_result_data
36
178
  end
37
179
 
38
180
  def inspect
39
181
  "#<#{self.class.name} response=#{@response.inspect}>"
40
182
  end
41
183
 
184
+ def tap_or_each(obj_or_array)
185
+ if obj_or_array.is_a?(Array)
186
+ obj_or_array.each do |item|
187
+ yield(item, true)
188
+ end
189
+ else
190
+ yield(obj_or_array, false)
191
+ end
192
+ end
193
+
42
194
  # This _begins_ the execution. Some deferred work
43
195
  # might be stored up in lazies.
44
196
  # @return [void]
45
197
  def run_eager
46
-
47
198
  root_operation = query.selected_operation
48
199
  root_op_type = root_operation.operation_type || "query"
49
- legacy_root_type = schema.root_type_for_operation(root_op_type)
50
- root_type = legacy_root_type.metadata[:type_class] || raise("Invariant: type must be class-based: #{legacy_root_type}")
200
+ root_type = schema.root_type_for_operation(root_op_type)
51
201
  path = []
52
- @interpreter_context[:current_object] = query.root_value
53
- @interpreter_context[:current_path] = path
54
- object_proxy = root_type.authorized_new(query.root_value, context)
202
+ set_all_interpreter_context(query.root_value, nil, nil, path)
203
+ object_proxy = authorized_new(root_type, query.root_value, context)
55
204
  object_proxy = schema.sync_lazy(object_proxy)
205
+
56
206
  if object_proxy.nil?
57
207
  # Root .authorized? returned false.
58
- write_in_response(path, nil)
59
- nil
208
+ @response = nil
60
209
  else
61
- evaluate_selections(path, context.scoped_context, object_proxy, root_type, root_operation.selections, root_operation_type: root_op_type)
62
- nil
210
+ call_method_on_directives(:resolve, object_proxy, root_operation.directives) do # execute query level directives
211
+ gathered_selections = gather_selections(object_proxy, root_type, root_operation.selections)
212
+ # This is kind of a hack -- `gathered_selections` is an Array if any of the selections
213
+ # require isolation during execution (because of runtime directives). In that case,
214
+ # make a new, isolated result hash for writing the result into. (That isolated response
215
+ # is eventually merged back into the main response)
216
+ #
217
+ # Otherwise, `gathered_selections` is a hash of selections which can be
218
+ # directly evaluated and the results can be written right into the main response hash.
219
+ tap_or_each(gathered_selections) do |selections, is_selection_array|
220
+ if is_selection_array
221
+ selection_response = GraphQLResultHash.new(nil, nil)
222
+ final_response = @response
223
+ else
224
+ selection_response = @response
225
+ final_response = nil
226
+ end
227
+
228
+ @dataloader.append_job {
229
+ set_all_interpreter_context(query.root_value, nil, nil, path)
230
+ call_method_on_directives(:resolve, object_proxy, selections.graphql_directives) do
231
+ evaluate_selections(
232
+ path,
233
+ context.scoped_context,
234
+ object_proxy,
235
+ root_type,
236
+ root_op_type == "mutation",
237
+ selections,
238
+ selection_response,
239
+ final_response,
240
+ nil,
241
+ )
242
+ end
243
+ }
244
+ end
245
+ end
246
+ end
247
+ delete_interpreter_context(:current_path)
248
+ delete_interpreter_context(:current_field)
249
+ delete_interpreter_context(:current_object)
250
+ delete_interpreter_context(:current_arguments)
251
+ nil
252
+ end
253
+
254
+ # @return [void]
255
+ def deep_merge_selection_result(from_result, into_result)
256
+ from_result.each do |key, value|
257
+ if !into_result.key?(key)
258
+ into_result[key] = value
259
+ else
260
+ case value
261
+ when GraphQLResultHash
262
+ deep_merge_selection_result(value, into_result[key])
263
+ else
264
+ # We have to assume that, since this passed the `fields_will_merge` selection,
265
+ # that the old and new values are the same.
266
+ # There's no special handling of arrays because currently, there's no way to split the execution
267
+ # of a list over several concurrent flows.
268
+ into_result[key] = value
269
+ end
270
+ end
63
271
  end
272
+ from_result.graphql_merged_into = into_result
273
+ nil
64
274
  end
65
275
 
66
- def gather_selections(owner_object, owner_type, selections, selections_by_name)
276
+ def gather_selections(owner_object, owner_type, selections, selections_to_run = nil, selections_by_name = GraphQLSelectionSet.new)
67
277
  selections.each do |node|
68
278
  # Skip gathering this if the directive says so
69
279
  if !directives_include?(node, owner_object, owner_type)
70
280
  next
71
281
  end
72
282
 
73
- case node
74
- when GraphQL::Language::Nodes::Field
283
+ if node.is_a?(GraphQL::Language::Nodes::Field)
75
284
  response_key = node.alias || node.name
76
285
  selections = selections_by_name[response_key]
77
286
  # if there was already a selection of this field,
@@ -87,145 +296,228 @@ module GraphQL
87
296
  # No selection was found for this field yet
88
297
  selections_by_name[response_key] = node
89
298
  end
90
- when GraphQL::Language::Nodes::InlineFragment
91
- if node.type
92
- type_defn = schema.types[node.type.name]
93
- type_defn = type_defn.metadata[:type_class]
94
- # Faster than .map{}.include?()
95
- query.warden.possible_types(type_defn).each do |t|
96
- if t.metadata[:type_class] == owner_type
97
- gather_selections(owner_object, owner_type, node.selections, selections_by_name)
98
- break
99
- end
299
+ else
300
+ # This is an InlineFragment or a FragmentSpread
301
+ if @runtime_directive_names.any? && node.directives.any? { |d| @runtime_directive_names.include?(d.name) }
302
+ next_selections = GraphQLSelectionSet.new
303
+ next_selections.graphql_directives = node.directives
304
+ if selections_to_run
305
+ selections_to_run << next_selections
306
+ else
307
+ selections_to_run = []
308
+ selections_to_run << selections_by_name
309
+ selections_to_run << next_selections
100
310
  end
101
311
  else
102
- # it's an untyped fragment, definitely continue
103
- gather_selections(owner_object, owner_type, node.selections, selections_by_name)
312
+ next_selections = selections_by_name
104
313
  end
105
- when GraphQL::Language::Nodes::FragmentSpread
106
- fragment_def = query.fragments[node.name]
107
- type_defn = schema.types[fragment_def.type.name]
108
- type_defn = type_defn.metadata[:type_class]
109
- schema.possible_types(type_defn).each do |t|
110
- if t.metadata[:type_class] == owner_type
111
- gather_selections(owner_object, owner_type, fragment_def.selections, selections_by_name)
112
- break
314
+
315
+ case node
316
+ when GraphQL::Language::Nodes::InlineFragment
317
+ if node.type
318
+ type_defn = schema.get_type(node.type.name, context)
319
+
320
+ # Faster than .map{}.include?()
321
+ query.warden.possible_types(type_defn).each do |t|
322
+ if t == owner_type
323
+ gather_selections(owner_object, owner_type, node.selections, selections_to_run, next_selections)
324
+ break
325
+ end
326
+ end
327
+ else
328
+ # it's an untyped fragment, definitely continue
329
+ gather_selections(owner_object, owner_type, node.selections, selections_to_run, next_selections)
113
330
  end
331
+ when GraphQL::Language::Nodes::FragmentSpread
332
+ fragment_def = query.fragments[node.name]
333
+ type_defn = query.get_type(fragment_def.type.name)
334
+ possible_types = query.warden.possible_types(type_defn)
335
+ possible_types.each do |t|
336
+ if t == owner_type
337
+ gather_selections(owner_object, owner_type, fragment_def.selections, selections_to_run, next_selections)
338
+ break
339
+ end
340
+ end
341
+ else
342
+ raise "Invariant: unexpected selection class: #{node.class}"
114
343
  end
115
- else
116
- raise "Invariant: unexpected selection class: #{node.class}"
117
344
  end
118
345
  end
346
+ selections_to_run || selections_by_name
119
347
  end
120
348
 
121
- def evaluate_selections(path, scoped_context, owner_object, owner_type, selections, root_operation_type: nil)
122
- @interpreter_context[:current_object] = owner_object
123
- @interpreter_context[:current_path] = path
124
- selections_by_name = {}
125
- gather_selections(owner_object, owner_type, selections, selections_by_name)
126
- selections_by_name.each do |result_name, field_ast_nodes_or_ast_node|
127
- # As a performance optimization, the hash key will be a `Node` if
128
- # there's only one selection of the field. But if there are multiple
129
- # selections of the field, it will be an Array of nodes
130
- if field_ast_nodes_or_ast_node.is_a?(Array)
131
- field_ast_nodes = field_ast_nodes_or_ast_node
132
- ast_node = field_ast_nodes.first
133
- else
134
- field_ast_nodes = nil
135
- ast_node = field_ast_nodes_or_ast_node
136
- end
137
- field_name = ast_node.name
138
- field_defn = @fields_cache[owner_type][field_name] ||= owner_type.get_field(field_name)
139
- is_introspection = false
140
- if field_defn.nil?
141
- field_defn = if owner_type == schema.query.metadata[:type_class] && (entry_point_field = schema.introspection_system.entry_point(name: field_name))
142
- is_introspection = true
143
- entry_point_field.metadata[:type_class]
144
- elsif (dynamic_field = schema.introspection_system.dynamic_field(name: field_name))
145
- is_introspection = true
146
- dynamic_field.metadata[:type_class]
147
- else
148
- raise "Invariant: no field for #{owner_type}.#{field_name}"
349
+ NO_ARGS = {}.freeze
350
+
351
+ # @return [void]
352
+ def evaluate_selections(path, scoped_context, owner_object, owner_type, is_eager_selection, gathered_selections, selections_result, target_result, parent_object) # rubocop:disable Metrics/ParameterLists
353
+ set_all_interpreter_context(owner_object, nil, nil, path)
354
+
355
+ finished_jobs = 0
356
+ enqueued_jobs = gathered_selections.size
357
+ gathered_selections.each do |result_name, field_ast_nodes_or_ast_node|
358
+ @dataloader.append_job {
359
+ evaluate_selection(
360
+ path, result_name, field_ast_nodes_or_ast_node, scoped_context, owner_object, owner_type, is_eager_selection, selections_result, parent_object
361
+ )
362
+ finished_jobs += 1
363
+ if target_result && finished_jobs == enqueued_jobs
364
+ deep_merge_selection_result(selections_result, target_result)
149
365
  end
366
+ }
367
+ end
368
+
369
+ selections_result
370
+ end
371
+
372
+ attr_reader :progress_path
373
+
374
+ # @return [void]
375
+ def evaluate_selection(path, result_name, field_ast_nodes_or_ast_node, scoped_context, owner_object, owner_type, is_eager_field, selections_result, parent_object) # rubocop:disable Metrics/ParameterLists
376
+ return if dead_result?(selections_result)
377
+ # As a performance optimization, the hash key will be a `Node` if
378
+ # there's only one selection of the field. But if there are multiple
379
+ # selections of the field, it will be an Array of nodes
380
+ if field_ast_nodes_or_ast_node.is_a?(Array)
381
+ field_ast_nodes = field_ast_nodes_or_ast_node
382
+ ast_node = field_ast_nodes.first
383
+ else
384
+ field_ast_nodes = nil
385
+ ast_node = field_ast_nodes_or_ast_node
386
+ end
387
+ field_name = ast_node.name
388
+ # This can't use `query.get_field` because it gets confused on introspection below if `field_defn` isn't `nil`,
389
+ # because of how `is_introspection` is used to call `.authorized_new` later on.
390
+ field_defn = @fields_cache[owner_type][field_name] ||= owner_type.get_field(field_name, @context)
391
+ is_introspection = false
392
+ if field_defn.nil?
393
+ field_defn = if owner_type == schema.query && (entry_point_field = schema.introspection_system.entry_point(name: field_name))
394
+ is_introspection = true
395
+ entry_point_field
396
+ elsif (dynamic_field = schema.introspection_system.dynamic_field(name: field_name))
397
+ is_introspection = true
398
+ dynamic_field
399
+ else
400
+ raise "Invariant: no field for #{owner_type}.#{field_name}"
150
401
  end
402
+ end
403
+
404
+ return_type = field_defn.type
151
405
 
152
- return_type = resolve_if_late_bound_type(field_defn.type)
406
+ next_path = path.dup
407
+ next_path << result_name
408
+ next_path.freeze
153
409
 
154
- next_path = path.dup
155
- next_path << result_name
156
- next_path.freeze
410
+ # This seems janky, but we need to know
411
+ # the field's return type at this path in order
412
+ # to propagate `null`
413
+ if return_type.non_null?
414
+ (selections_result.graphql_non_null_field_names ||= []).push(result_name)
415
+ end
416
+ # Set this before calling `run_with_directives`, so that the directive can have the latest path
417
+ set_all_interpreter_context(nil, field_defn, nil, next_path)
157
418
 
158
- # This seems janky, but we need to know
159
- # the field's return type at this path in order
160
- # to propagate `null`
161
- set_type_at_path(next_path, return_type)
162
- # Set this before calling `run_with_directives`, so that the directive can have the latest path
163
- @interpreter_context[:current_path] = next_path
164
- @interpreter_context[:current_field] = field_defn
419
+ context.scoped_context = scoped_context
420
+ object = owner_object
165
421
 
166
- context.scoped_context = scoped_context
167
- object = owner_object
422
+ if is_introspection
423
+ object = authorized_new(field_defn.owner, object, context)
424
+ end
168
425
 
169
- if is_introspection
170
- object = field_defn.owner.authorized_new(object, context)
426
+ total_args_count = field_defn.arguments(context).size
427
+ if total_args_count == 0
428
+ resolved_arguments = GraphQL::Execution::Interpreter::Arguments::EMPTY
429
+ evaluate_selection_with_args(resolved_arguments, field_defn, next_path, ast_node, field_ast_nodes, scoped_context, owner_type, object, is_eager_field, result_name, selections_result, parent_object, return_type)
430
+ else
431
+ # TODO remove all arguments(...) usages?
432
+ @query.arguments_cache.dataload_for(ast_node, field_defn, object) do |resolved_arguments|
433
+ evaluate_selection_with_args(resolved_arguments, field_defn, next_path, ast_node, field_ast_nodes, scoped_context, owner_type, object, is_eager_field, result_name, selections_result, parent_object, return_type)
171
434
  end
435
+ end
436
+ end
172
437
 
173
- begin
174
- kwarg_arguments = arguments(object, field_defn, ast_node)
175
- rescue GraphQL::ExecutionError => e
176
- continue_value(next_path, e, field_defn, return_type.non_null?, ast_node)
438
+ def evaluate_selection_with_args(arguments, field_defn, next_path, ast_node, field_ast_nodes, scoped_context, owner_type, object, is_eager_field, result_name, selection_result, parent_object, return_type) # rubocop:disable Metrics/ParameterLists
439
+ context.scoped_context = scoped_context
440
+ after_lazy(arguments, owner: owner_type, field: field_defn, path: next_path, ast_node: ast_node, scoped_context: context.scoped_context, owner_object: object, arguments: arguments, result_name: result_name, result: selection_result) do |resolved_arguments|
441
+ if resolved_arguments.is_a?(GraphQL::ExecutionError) || resolved_arguments.is_a?(GraphQL::UnauthorizedError)
442
+ continue_value(next_path, resolved_arguments, owner_type, field_defn, return_type.non_null?, ast_node, result_name, selection_result)
177
443
  next
178
444
  end
179
445
 
180
- # It might turn out that making arguments for every field is slow.
181
- # If we have to cache them, we'll need a more subtle approach here.
182
- field_defn.extras.each do |extra|
183
- case extra
184
- when :ast_node
185
- kwarg_arguments[:ast_node] = ast_node
186
- when :execution_errors
187
- kwarg_arguments[:execution_errors] = ExecutionErrors.new(context, ast_node, next_path)
188
- when :path
189
- kwarg_arguments[:path] = next_path
190
- when :lookahead
191
- if !field_ast_nodes
192
- field_ast_nodes = [ast_node]
446
+ kwarg_arguments = if resolved_arguments.empty? && field_defn.extras.empty?
447
+ # We can avoid allocating the `{ Symbol => Object }` hash in this case
448
+ NO_ARGS
449
+ else
450
+ # Bundle up the extras, then make a new arguments instance
451
+ # that includes the extras, too.
452
+ extra_args = {}
453
+ field_defn.extras.each do |extra|
454
+ case extra
455
+ when :ast_node
456
+ extra_args[:ast_node] = ast_node
457
+ when :execution_errors
458
+ extra_args[:execution_errors] = ExecutionErrors.new(context, ast_node, next_path)
459
+ when :path
460
+ extra_args[:path] = next_path
461
+ when :lookahead
462
+ if !field_ast_nodes
463
+ field_ast_nodes = [ast_node]
464
+ end
465
+
466
+ extra_args[:lookahead] = Execution::Lookahead.new(
467
+ query: query,
468
+ ast_nodes: field_ast_nodes,
469
+ field: field_defn,
470
+ )
471
+ when :argument_details
472
+ # Use this flag to tell Interpreter::Arguments to add itself
473
+ # to the keyword args hash _before_ freezing everything.
474
+ extra_args[:argument_details] = :__arguments_add_self
475
+ when :irep_node
476
+ # This is used by `__typename` in order to support the legacy runtime,
477
+ # but it has no use here (and it's always `nil`).
478
+ # Stop adding it here to avoid the overhead of `.merge_extras` below.
479
+ when :parent
480
+ extra_args[:parent] = parent_object
481
+ else
482
+ extra_args[extra] = field_defn.fetch_extra(extra, context)
193
483
  end
194
- kwarg_arguments[:lookahead] = Execution::Lookahead.new(
195
- query: query,
196
- ast_nodes: field_ast_nodes,
197
- field: field_defn,
198
- )
199
- else
200
- kwarg_arguments[extra] = field_defn.fetch_extra(extra, context)
201
484
  end
485
+ if extra_args.any?
486
+ resolved_arguments = resolved_arguments.merge_extras(extra_args)
487
+ end
488
+ resolved_arguments.keyword_arguments
202
489
  end
203
490
 
204
- @interpreter_context[:current_arguments] = kwarg_arguments
491
+ set_all_interpreter_context(nil, nil, resolved_arguments, nil)
205
492
 
206
493
  # Optimize for the case that field is selected only once
207
494
  if field_ast_nodes.nil? || field_ast_nodes.size == 1
208
495
  next_selections = ast_node.selections
496
+ directives = ast_node.directives
209
497
  else
210
498
  next_selections = []
211
- field_ast_nodes.each { |f| next_selections.concat(f.selections) }
499
+ directives = []
500
+ field_ast_nodes.each { |f|
501
+ next_selections.concat(f.selections)
502
+ directives.concat(f.directives)
503
+ }
212
504
  end
213
505
 
214
- field_result = resolve_with_directives(object, ast_node) do
506
+ field_result = call_method_on_directives(:resolve, object, directives) do
215
507
  # Actually call the field resolver and capture the result
216
508
  app_result = begin
217
509
  query.with_error_handling do
218
- query.trace("execute_field", {owner: owner_type, field: field_defn, path: next_path, query: query, object: object, arguments: kwarg_arguments}) do
510
+ query.trace("execute_field", {owner: owner_type, field: field_defn, path: next_path, ast_node: ast_node, query: query, object: object, arguments: kwarg_arguments}) do
219
511
  field_defn.resolve(object, kwarg_arguments, context)
220
512
  end
221
513
  end
222
514
  rescue GraphQL::ExecutionError => err
223
515
  err
224
516
  end
225
- after_lazy(app_result, owner: owner_type, field: field_defn, path: next_path, scoped_context: context.scoped_context, owner_object: object, arguments: kwarg_arguments) do |inner_result|
226
- continue_value = continue_value(next_path, inner_result, field_defn, return_type.non_null?, ast_node)
517
+ after_lazy(app_result, owner: owner_type, field: field_defn, path: next_path, ast_node: ast_node, scoped_context: context.scoped_context, owner_object: object, arguments: resolved_arguments, result_name: result_name, result: selection_result) do |inner_result|
518
+ continue_value = continue_value(next_path, inner_result, owner_type, field_defn, return_type.non_null?, ast_node, result_name, selection_result)
227
519
  if HALT != continue_value
228
- continue_field(next_path, continue_value, field_defn, return_type, ast_node, next_selections, false, object, kwarg_arguments)
520
+ continue_field(next_path, continue_value, owner_type, field_defn, return_type, ast_node, next_selections, false, object, resolved_arguments, result_name, selection_result)
229
521
  end
230
522
  end
231
523
  end
@@ -233,47 +525,144 @@ module GraphQL
233
525
  # If this field is a root mutation field, immediately resolve
234
526
  # all of its child fields before moving on to the next root mutation field.
235
527
  # (Subselections of this mutation will still be resolved level-by-level.)
236
- if root_operation_type == "mutation"
237
- Interpreter::Resolve.resolve_all([field_result])
528
+ if is_eager_field
529
+ Interpreter::Resolve.resolve_all([field_result], @dataloader)
238
530
  else
531
+ # Return this from `after_lazy` because it might be another lazy that needs to be resolved
239
532
  field_result
240
533
  end
241
534
  end
242
535
  end
243
536
 
537
+ def dead_result?(selection_result)
538
+ selection_result.graphql_dead || ((parent = selection_result.graphql_parent) && parent.graphql_dead)
539
+ end
540
+
541
+ def set_result(selection_result, result_name, value)
542
+ if !dead_result?(selection_result)
543
+ if value.nil? &&
544
+ ( # there are two conditions under which `nil` is not allowed in the response:
545
+ (selection_result.graphql_non_null_list_items) || # this value would be written into a list that doesn't allow nils
546
+ ((nn = selection_result.graphql_non_null_field_names) && nn.include?(result_name)) # this value would be written into a field that doesn't allow nils
547
+ )
548
+ # This is an invalid nil that should be propagated
549
+ # One caller of this method passes a block,
550
+ # namely when application code returns a `nil` to GraphQL and it doesn't belong there.
551
+ # The other possibility for reaching here is when a field returns an ExecutionError, so we write
552
+ # `nil` to the response, not knowing whether it's an invalid `nil` or not.
553
+ # (And in that case, we don't have to call the schema's handler, since it's not a bug in the application.)
554
+ # TODO the code is trying to tell me something.
555
+ yield if block_given?
556
+ parent = selection_result.graphql_parent
557
+ name_in_parent = selection_result.graphql_result_name
558
+ if parent.nil? # This is a top-level result hash
559
+ @response = nil
560
+ else
561
+ set_result(parent, name_in_parent, nil)
562
+ set_graphql_dead(selection_result)
563
+ end
564
+ else
565
+ selection_result[result_name] = value
566
+ end
567
+ end
568
+ end
569
+
570
+ # Mark this node and any already-registered children as dead,
571
+ # so that it accepts no more writes.
572
+ def set_graphql_dead(selection_result)
573
+ case selection_result
574
+ when GraphQLResultArray
575
+ selection_result.graphql_dead = true
576
+ selection_result.values.each { |v| set_graphql_dead(v) }
577
+ when GraphQLResultHash
578
+ selection_result.graphql_dead = true
579
+ selection_result.each { |k, v| set_graphql_dead(v) }
580
+ else
581
+ # It's a scalar, no way to mark it dead.
582
+ end
583
+ end
584
+
244
585
  HALT = Object.new
245
- def continue_value(path, value, field, is_non_null, ast_node)
246
- if value.nil?
586
+ def continue_value(path, value, parent_type, field, is_non_null, ast_node, result_name, selection_result) # rubocop:disable Metrics/ParameterLists
587
+ case value
588
+ when nil
247
589
  if is_non_null
248
- err = GraphQL::InvalidNullError.new(field.owner, field, value)
249
- write_invalid_null_in_response(path, err)
590
+ set_result(selection_result, result_name, nil) do
591
+ # This block is called if `result_name` is not dead. (Maybe a previous invalid nil caused it be marked dead.)
592
+ err = parent_type::InvalidNullError.new(parent_type, field, value)
593
+ schema.type_error(err, context)
594
+ end
250
595
  else
251
- write_in_response(path, nil)
596
+ set_result(selection_result, result_name, nil)
252
597
  end
253
598
  HALT
254
- elsif value.is_a?(GraphQL::ExecutionError)
255
- value.path ||= path
256
- value.ast_node ||= ast_node
257
- write_execution_errors_in_response(path, [value])
258
- HALT
259
- elsif value.is_a?(Array) && value.any? && value.all? { |v| v.is_a?(GraphQL::ExecutionError) }
260
- value.each_with_index do |error, index|
261
- error.ast_node ||= ast_node
262
- error.path ||= path + (field.type.list? ? [index] : [])
599
+ when GraphQL::Error
600
+ # Handle these cases inside a single `when`
601
+ # to avoid the overhead of checking three different classes
602
+ # every time.
603
+ if value.is_a?(GraphQL::ExecutionError)
604
+ if selection_result.nil? || !dead_result?(selection_result)
605
+ value.path ||= path
606
+ value.ast_node ||= ast_node
607
+ context.errors << value
608
+ if selection_result
609
+ set_result(selection_result, result_name, nil)
610
+ end
611
+ end
612
+ HALT
613
+ elsif value.is_a?(GraphQL::UnauthorizedError)
614
+ # this hook might raise & crash, or it might return
615
+ # a replacement value
616
+ next_value = begin
617
+ schema.unauthorized_object(value)
618
+ rescue GraphQL::ExecutionError => err
619
+ err
620
+ end
621
+ continue_value(path, next_value, parent_type, field, is_non_null, ast_node, result_name, selection_result)
622
+ elsif GraphQL::Execution::Execute::SKIP == value
623
+ # It's possible a lazy was already written here
624
+ case selection_result
625
+ when GraphQLResultHash
626
+ selection_result.delete(result_name)
627
+ when GraphQLResultArray
628
+ selection_result.graphql_skip_at(result_name)
629
+ when nil
630
+ # this can happen with directives
631
+ else
632
+ raise "Invariant: unexpected result class #{selection_result.class} (#{selection_result.inspect})"
633
+ end
634
+ HALT
635
+ else
636
+ # What could this actually _be_? Anyhow,
637
+ # preserve the default behavior of doing nothing with it.
638
+ value
263
639
  end
264
- write_execution_errors_in_response(path, value)
265
- HALT
266
- elsif value.is_a?(GraphQL::UnauthorizedError)
267
- # this hook might raise & crash, or it might return
268
- # a replacement value
269
- next_value = begin
270
- schema.unauthorized_object(value)
271
- rescue GraphQL::ExecutionError => err
272
- err
640
+ when Array
641
+ # It's an array full of execution errors; add them all.
642
+ if value.any? && value.all? { |v| v.is_a?(GraphQL::ExecutionError) }
643
+ list_type_at_all = (field && (field.type.list?))
644
+ if selection_result.nil? || !dead_result?(selection_result)
645
+ value.each_with_index do |error, index|
646
+ error.ast_node ||= ast_node
647
+ error.path ||= path + (list_type_at_all ? [index] : [])
648
+ context.errors << error
649
+ end
650
+ if selection_result
651
+ if list_type_at_all
652
+ result_without_errors = value.map { |v| v.is_a?(GraphQL::ExecutionError) ? nil : v }
653
+ set_result(selection_result, result_name, result_without_errors)
654
+ else
655
+ set_result(selection_result, result_name, nil)
656
+ end
657
+ end
658
+ end
659
+ HALT
660
+ else
661
+ value
273
662
  end
274
-
275
- continue_value(path, next_value, field, is_non_null, ast_node)
276
- elsif GraphQL::Execution::Execute::SKIP == value
663
+ when GraphQL::Execution::Interpreter::RawValue
664
+ # Write raw value directly to the response without resolving nested objects
665
+ set_result(selection_result, result_name, value.resolve)
277
666
  HALT
278
667
  else
279
668
  value
@@ -288,93 +677,169 @@ module GraphQL
288
677
  # Location information from `path` and `ast_node`.
289
678
  #
290
679
  # @return [Lazy, Array, Hash, Object] Lazy, Array, and Hash are all traversed to resolve lazy values later
291
- def continue_field(path, value, field, type, ast_node, next_selections, is_non_null, owner_object, arguments) # rubocop:disable Metrics/ParameterLists
292
- case type.kind.name
680
+ def continue_field(path, value, owner_type, field, current_type, ast_node, next_selections, is_non_null, owner_object, arguments, result_name, selection_result) # rubocop:disable Metrics/ParameterLists
681
+ if current_type.non_null?
682
+ current_type = current_type.of_type
683
+ is_non_null = true
684
+ end
685
+
686
+ case current_type.kind.name
293
687
  when "SCALAR", "ENUM"
294
- r = type.coerce_result(value, context)
295
- write_in_response(path, r)
688
+ r = current_type.coerce_result(value, context)
689
+ set_result(selection_result, result_name, r)
296
690
  r
297
691
  when "UNION", "INTERFACE"
298
- resolved_type_or_lazy = query.resolve_type(type, value)
299
- after_lazy(resolved_type_or_lazy, owner: type, path: path, scoped_context: context.scoped_context, field: field, owner_object: owner_object, arguments: arguments) do |resolved_type|
300
- possible_types = query.possible_types(type)
692
+ resolved_type_or_lazy, resolved_value = resolve_type(current_type, value, path)
693
+ resolved_value ||= value
694
+
695
+ after_lazy(resolved_type_or_lazy, owner: current_type, path: path, ast_node: ast_node, scoped_context: context.scoped_context, field: field, owner_object: owner_object, arguments: arguments, trace: false, result_name: result_name, result: selection_result) do |resolved_type|
696
+ possible_types = query.possible_types(current_type)
301
697
 
302
698
  if !possible_types.include?(resolved_type)
303
- parent_type = field.owner
304
- type_error = GraphQL::UnresolvedTypeError.new(value, field, parent_type, resolved_type, possible_types)
699
+ parent_type = field.owner_type
700
+ err_class = current_type::UnresolvedTypeError
701
+ type_error = err_class.new(resolved_value, field, parent_type, resolved_type, possible_types)
305
702
  schema.type_error(type_error, context)
306
- write_in_response(path, nil)
703
+ set_result(selection_result, result_name, nil)
307
704
  nil
308
705
  else
309
- resolved_type = resolved_type.metadata[:type_class]
310
- continue_field(path, value, field, resolved_type, ast_node, next_selections, is_non_null, owner_object, arguments)
706
+ continue_field(path, resolved_value, owner_type, field, resolved_type, ast_node, next_selections, is_non_null, owner_object, arguments, result_name, selection_result)
311
707
  end
312
708
  end
313
709
  when "OBJECT"
314
710
  object_proxy = begin
315
- type.authorized_new(value, context)
711
+ authorized_new(current_type, value, context)
316
712
  rescue GraphQL::ExecutionError => err
317
713
  err
318
714
  end
319
- after_lazy(object_proxy, owner: type, path: path, scoped_context: context.scoped_context, field: field, owner_object: owner_object, arguments: arguments) do |inner_object|
320
- continue_value = continue_value(path, inner_object, field, is_non_null, ast_node)
715
+ after_lazy(object_proxy, owner: current_type, path: path, ast_node: ast_node, scoped_context: context.scoped_context, field: field, owner_object: owner_object, arguments: arguments, trace: false, result_name: result_name, result: selection_result) do |inner_object|
716
+ continue_value = continue_value(path, inner_object, owner_type, field, is_non_null, ast_node, result_name, selection_result)
321
717
  if HALT != continue_value
322
- response_hash = {}
323
- write_in_response(path, response_hash)
324
- evaluate_selections(path, context.scoped_context, continue_value, type, next_selections)
325
- response_hash
718
+ response_hash = GraphQLResultHash.new(result_name, selection_result)
719
+ set_result(selection_result, result_name, response_hash)
720
+ gathered_selections = gather_selections(continue_value, current_type, next_selections)
721
+ # There are two possibilities for `gathered_selections`:
722
+ # 1. All selections of this object should be evaluated together (there are no runtime directives modifying execution).
723
+ # This case is handled below, and the result can be written right into the main `response_hash` above.
724
+ # In this case, `gathered_selections` is a hash of selections.
725
+ # 2. Some selections of this object have runtime directives that may or may not modify execution.
726
+ # That part of the selection is evaluated in an isolated way, writing into a sub-response object which is
727
+ # eventually merged into the final response. In this case, `gathered_selections` is an array of things to run in isolation.
728
+ # (Technically, it's possible that one of those entries _doesn't_ require isolation.)
729
+ tap_or_each(gathered_selections) do |selections, is_selection_array|
730
+ if is_selection_array
731
+ this_result = GraphQLResultHash.new(result_name, selection_result)
732
+ final_result = response_hash
733
+ else
734
+ this_result = response_hash
735
+ final_result = nil
736
+ end
737
+ set_all_interpreter_context(continue_value, nil, nil, path) # reset this mutable state
738
+ call_method_on_directives(:resolve, continue_value, selections.graphql_directives) do
739
+ evaluate_selections(
740
+ path,
741
+ context.scoped_context,
742
+ continue_value,
743
+ current_type,
744
+ false,
745
+ selections,
746
+ this_result,
747
+ final_result,
748
+ owner_object.object,
749
+ )
750
+ this_result
751
+ end
752
+ end
326
753
  end
327
754
  end
328
755
  when "LIST"
329
- response_list = []
330
- write_in_response(path, response_list)
331
- inner_type = type.of_type
756
+ inner_type = current_type.of_type
757
+ # This is true for objects, unions, and interfaces
758
+ use_dataloader_job = !inner_type.unwrap.kind.input?
759
+ response_list = GraphQLResultArray.new(result_name, selection_result)
760
+ response_list.graphql_non_null_list_items = inner_type.non_null?
761
+ set_result(selection_result, result_name, response_list)
762
+
332
763
  idx = 0
333
764
  scoped_context = context.scoped_context
334
- value.each do |inner_value|
335
- next_path = path.dup
336
- next_path << idx
337
- next_path.freeze
338
- idx += 1
339
- set_type_at_path(next_path, inner_type)
340
- # This will update `response_list` with the lazy
341
- after_lazy(inner_value, owner: inner_type, path: next_path, scoped_context: scoped_context, field: field, owner_object: owner_object, arguments: arguments) do |inner_inner_value|
342
- # reset `is_non_null` here and below, because the inner type will have its own nullability constraint
343
- continue_value = continue_value(next_path, inner_inner_value, field, false, ast_node)
344
- if HALT != continue_value
345
- continue_field(next_path, continue_value, field, inner_type, ast_node, next_selections, false, owner_object, arguments)
765
+ begin
766
+ value.each do |inner_value|
767
+ break if dead_result?(response_list)
768
+ next_path = path.dup
769
+ next_path << idx
770
+ this_idx = idx
771
+ next_path.freeze
772
+ idx += 1
773
+ if use_dataloader_job
774
+ @dataloader.append_job do
775
+ resolve_list_item(inner_value, inner_type, next_path, ast_node, scoped_context, field, owner_object, arguments, this_idx, response_list, next_selections, owner_type)
776
+ end
777
+ else
778
+ resolve_list_item(inner_value, inner_type, next_path, ast_node, scoped_context, field, owner_object, arguments, this_idx, response_list, next_selections, owner_type)
346
779
  end
347
780
  end
781
+ rescue NoMethodError => err
782
+ # Ruby 2.2 doesn't have NoMethodError#receiver, can't check that one in this case. (It's been EOL since 2017.)
783
+ if err.name == :each && (err.respond_to?(:receiver) ? err.receiver == value : true)
784
+ # This happens when the GraphQL schema doesn't match the implementation. Help the dev debug.
785
+ raise ListResultFailedError.new(value: value, field: field, path: path)
786
+ else
787
+ # This was some other NoMethodError -- let it bubble to reveal the real error.
788
+ raise
789
+ end
348
790
  end
791
+
349
792
  response_list
350
- when "NON_NULL"
351
- inner_type = type.of_type
352
- # For fields like `__schema: __Schema!`
353
- inner_type = resolve_if_late_bound_type(inner_type)
354
- # Don't `set_type_at_path` because we want the static type,
355
- # we're going to use that to determine whether a `nil` should be propagated or not.
356
- continue_field(path, value, field, inner_type, ast_node, next_selections, true, owner_object, arguments)
357
793
  else
358
- raise "Invariant: Unhandled type kind #{type.kind} (#{type})"
794
+ raise "Invariant: Unhandled type kind #{current_type.kind} (#{current_type})"
359
795
  end
360
796
  end
361
797
 
362
- def resolve_with_directives(object, ast_node)
363
- run_directive(object, ast_node, 0) { yield }
798
+ def resolve_list_item(inner_value, inner_type, next_path, ast_node, scoped_context, field, owner_object, arguments, this_idx, response_list, next_selections, owner_type) # rubocop:disable Metrics/ParameterLists
799
+ set_all_interpreter_context(nil, nil, nil, next_path)
800
+ call_method_on_directives(:resolve_each, owner_object, ast_node.directives) do
801
+ # This will update `response_list` with the lazy
802
+ after_lazy(inner_value, owner: inner_type, path: next_path, ast_node: ast_node, scoped_context: scoped_context, field: field, owner_object: owner_object, arguments: arguments, result_name: this_idx, result: response_list) do |inner_inner_value|
803
+ continue_value = continue_value(next_path, inner_inner_value, owner_type, field, inner_type.non_null?, ast_node, this_idx, response_list)
804
+ if HALT != continue_value
805
+ continue_field(next_path, continue_value, owner_type, field, inner_type, ast_node, next_selections, false, owner_object, arguments, this_idx, response_list)
806
+ end
807
+ end
808
+ end
364
809
  end
365
810
 
366
- def run_directive(object, ast_node, idx)
367
- dir_node = ast_node.directives[idx]
811
+ def call_method_on_directives(method_name, object, directives, &block)
812
+ return yield if directives.nil? || directives.empty?
813
+ run_directive(method_name, object, directives, 0, &block)
814
+ end
815
+
816
+ def run_directive(method_name, object, directives, idx, &block)
817
+ dir_node = directives[idx]
368
818
  if !dir_node
369
819
  yield
370
820
  else
371
- dir_defn = schema.directives.fetch(dir_node.name)
821
+ dir_defn = @schema_directives.fetch(dir_node.name)
372
822
  if !dir_defn.is_a?(Class)
373
- dir_defn = dir_defn.metadata[:type_class] || raise("Only class-based directives are supported (not `@#{dir_node.name}`)")
374
- end
375
- dir_args = arguments(nil, dir_defn, dir_node)
376
- dir_defn.resolve(object, dir_args, context) do
377
- run_directive(object, ast_node, idx + 1) { yield }
823
+ dir_defn = dir_defn.type_class || raise("Only class-based directives are supported (not `@#{dir_node.name}`)")
824
+ end
825
+ raw_dir_args = arguments(nil, dir_defn, dir_node)
826
+ dir_args = continue_value(
827
+ @context[:current_path], # path
828
+ raw_dir_args, # value
829
+ dir_defn, # parent_type
830
+ nil, # field
831
+ false, # is_non_null
832
+ dir_node, # ast_node
833
+ nil, # result_name
834
+ nil, # selection_result
835
+ )
836
+
837
+ if dir_args == HALT
838
+ nil
839
+ else
840
+ dir_defn.public_send(method_name, object, dir_args, context) do
841
+ run_directive(method_name, object, directives, idx + 1, &block)
842
+ end
378
843
  end
379
844
  end
380
845
  end
@@ -382,7 +847,7 @@ module GraphQL
382
847
  # Check {Schema::Directive.include?} for each directive that's present
383
848
  def directives_include?(node, graphql_object, parent_type)
384
849
  node.directives.each do |dir_node|
385
- dir_defn = schema.directives.fetch(dir_node.name).metadata[:type_class] || raise("Only class-based directives are supported (not #{dir_node.name.inspect})")
850
+ dir_defn = @schema_directives.fetch(dir_node.name).type_class || raise("Only class-based directives are supported (not #{dir_node.name.inspect})")
386
851
  args = arguments(graphql_object, dir_defn, dir_node)
387
852
  if !dir_defn.include?(graphql_object, args, context)
388
853
  return false
@@ -391,11 +856,18 @@ module GraphQL
391
856
  true
392
857
  end
393
858
 
394
- def resolve_if_late_bound_type(type)
395
- if type.is_a?(GraphQL::Schema::LateBoundType)
396
- query.warden.get_type(type.name).metadata[:type_class]
397
- else
398
- type
859
+ def set_all_interpreter_context(object, field, arguments, path)
860
+ if object
861
+ @context[:current_object] = @interpreter_context[:current_object] = object
862
+ end
863
+ if field
864
+ @context[:current_field] = @interpreter_context[:current_field] = field
865
+ end
866
+ if arguments
867
+ @context[:current_arguments] = @interpreter_context[:current_arguments] = arguments
868
+ end
869
+ if path
870
+ @context[:current_path] = @interpreter_context[:current_path] = path
399
871
  end
400
872
  end
401
873
 
@@ -403,230 +875,93 @@ module GraphQL
403
875
  # @param path [Array<String>]
404
876
  # @param field [GraphQL::Schema::Field]
405
877
  # @param eager [Boolean] Set to `true` for mutation root fields only
878
+ # @param trace [Boolean] If `false`, don't wrap this with field tracing
406
879
  # @return [GraphQL::Execution::Lazy, Object] If loading `object` will be deferred, it's a wrapper over it.
407
- def after_lazy(lazy_obj, owner:, field:, path:, scoped_context:, owner_object:, arguments:, eager: false)
408
- @interpreter_context[:current_object] = owner_object
409
- @interpreter_context[:current_arguments] = arguments
410
- @interpreter_context[:current_path] = path
411
- @interpreter_context[:current_field] = field
412
- if schema.lazy?(lazy_obj)
880
+ def after_lazy(lazy_obj, owner:, field:, path:, scoped_context:, owner_object:, arguments:, ast_node:, result:, result_name:, eager: false, trace: true, &block)
881
+ if lazy?(lazy_obj)
413
882
  lazy = GraphQL::Execution::Lazy.new(path: path, field: field) do
414
- @interpreter_context[:current_path] = path
415
- @interpreter_context[:current_field] = field
416
- @interpreter_context[:current_object] = owner_object
417
- @interpreter_context[:current_arguments] = arguments
883
+ set_all_interpreter_context(owner_object, field, arguments, path)
418
884
  context.scoped_context = scoped_context
419
885
  # Wrap the execution of _this_ method with tracing,
420
886
  # but don't wrap the continuation below
421
887
  inner_obj = begin
422
888
  query.with_error_handling do
423
- query.trace("execute_field_lazy", {owner: owner, field: field, path: path, query: query, object: owner_object, arguments: arguments}) do
424
- schema.sync_lazy(lazy_obj)
889
+ begin
890
+ if trace
891
+ query.trace("execute_field_lazy", {owner: owner, field: field, path: path, query: query, object: owner_object, arguments: arguments, ast_node: ast_node}) do
892
+ schema.sync_lazy(lazy_obj)
893
+ end
894
+ else
895
+ schema.sync_lazy(lazy_obj)
896
+ end
897
+ rescue GraphQL::ExecutionError, GraphQL::UnauthorizedError => err
898
+ err
425
899
  end
426
900
  end
427
- rescue GraphQL::ExecutionError, GraphQL::UnauthorizedError => err
428
- yield(err)
429
- end
430
- after_lazy(inner_obj, owner: owner, field: field, path: path, scoped_context: context.scoped_context, owner_object: owner_object, arguments: arguments, eager: eager) do |really_inner_obj|
431
- yield(really_inner_obj)
901
+ rescue GraphQL::ExecutionError => ex_err
902
+ ex_err
432
903
  end
904
+ yield(inner_obj)
433
905
  end
434
906
 
435
907
  if eager
436
908
  lazy.value
437
909
  else
438
- write_in_response(path, lazy)
910
+ set_result(result, result_name, lazy)
439
911
  lazy
440
912
  end
441
913
  else
914
+ set_all_interpreter_context(owner_object, field, arguments, path)
442
915
  yield(lazy_obj)
443
916
  end
444
917
  end
445
918
 
446
- def each_argument_pair(ast_args_or_hash)
447
- case ast_args_or_hash
448
- when GraphQL::Language::Nodes::Field, GraphQL::Language::Nodes::InputObject, GraphQL::Language::Nodes::Directive
449
- ast_args_or_hash.arguments.each do |arg|
450
- yield(arg.name, arg.value)
451
- end
452
- when Hash
453
- ast_args_or_hash.each do |key, value|
454
- normalized_name = GraphQL::Schema::Member::BuildType.camelize(key.to_s)
455
- yield(normalized_name, value)
456
- end
919
+ def arguments(graphql_object, arg_owner, ast_node)
920
+ if arg_owner.arguments_statically_coercible?
921
+ query.arguments_for(ast_node, arg_owner)
457
922
  else
458
- raise "Invariant, unexpected #{ast_args_or_hash.inspect}"
923
+ # The arguments must be prepared in the context of the given object
924
+ query.arguments_for(ast_node, arg_owner, parent_object: graphql_object)
459
925
  end
460
926
  end
461
927
 
462
- def arguments(graphql_object, arg_owner, ast_node_or_hash)
463
- kwarg_arguments = {}
464
- arg_defns = arg_owner.arguments
465
- each_argument_pair(ast_node_or_hash) do |arg_name, arg_value|
466
- arg_defn = arg_defns[arg_name]
467
- # Need to distinguish between client-provided `nil`
468
- # and nothing-at-all
469
- is_present, value = arg_to_value(graphql_object, arg_defn.type, arg_value)
470
- if is_present
471
- # This doesn't apply to directives, which are legacy
472
- # Can remove this when Skip and Include use classes or something.
473
- if graphql_object
474
- value = arg_defn.prepare_value(graphql_object, value)
475
- end
476
- kwarg_arguments[arg_defn.keyword] = value
477
- end
478
- end
479
- arg_defns.each do |name, arg_defn|
480
- if arg_defn.default_value? && !kwarg_arguments.key?(arg_defn.keyword)
481
- _is_present, value = arg_to_value(graphql_object, arg_defn.type, arg_defn.default_value)
482
- kwarg_arguments[arg_defn.keyword] = value
483
- end
484
- end
485
- kwarg_arguments
928
+ # Set this pair in the Query context, but also in the interpeter namespace,
929
+ # for compatibility.
930
+ def set_interpreter_context(key, value)
931
+ @interpreter_context[key] = value
932
+ @context[key] = value
486
933
  end
487
934
 
488
- # Get a Ruby-ready value from a client query.
489
- # @param graphql_object [Object] The owner of the field whose argument this is
490
- # @param arg_type [Class, GraphQL::Schema::NonNull, GraphQL::Schema::List]
491
- # @param ast_value [GraphQL::Language::Nodes::VariableIdentifier, String, Integer, Float, Boolean]
492
- # @return [Array(is_present, value)]
493
- def arg_to_value(graphql_object, arg_type, ast_value)
494
- if ast_value.is_a?(GraphQL::Language::Nodes::VariableIdentifier)
495
- # If it's not here, it will get added later
496
- if query.variables.key?(ast_value.name)
497
- return true, query.variables[ast_value.name]
498
- else
499
- return false, nil
500
- end
501
- elsif ast_value.is_a?(GraphQL::Language::Nodes::NullValue)
502
- return true, nil
503
- elsif arg_type.is_a?(GraphQL::Schema::NonNull)
504
- arg_to_value(graphql_object, arg_type.of_type, ast_value)
505
- elsif arg_type.is_a?(GraphQL::Schema::List)
506
- # Treat a single value like a list
507
- arg_value = Array(ast_value)
508
- list = []
509
- arg_value.map do |inner_v|
510
- _present, value = arg_to_value(graphql_object, arg_type.of_type, inner_v)
511
- list << value
512
- end
513
- return true, list
514
- elsif arg_type.is_a?(Class) && arg_type < GraphQL::Schema::InputObject
515
- # For these, `prepare` is applied during `#initialize`.
516
- # Pass `nil` so it will be skipped in `#arguments`.
517
- # What a mess.
518
- args = arguments(nil, arg_type, ast_value)
519
- # We're not tracking defaults_used, but for our purposes
520
- # we compare the value to the default value.
521
-
522
- input_obj = query.with_error_handling do
523
- arg_type.new(ruby_kwargs: args, context: context, defaults_used: nil)
524
- end
525
- return true, input_obj
526
- else
527
- flat_value = flatten_ast_value(ast_value)
528
- return true, arg_type.coerce_input(flat_value, context)
529
- end
935
+ def delete_interpreter_context(key)
936
+ @interpreter_context.delete(key)
937
+ @context.delete(key)
530
938
  end
531
939
 
532
- def flatten_ast_value(v)
533
- case v
534
- when GraphQL::Language::Nodes::Enum
535
- v.name
536
- when GraphQL::Language::Nodes::InputObject
537
- h = {}
538
- v.arguments.each do |arg|
539
- h[arg.name] = flatten_ast_value(arg.value)
540
- end
541
- h
542
- when Array
543
- v.map { |v2| flatten_ast_value(v2) }
544
- when GraphQL::Language::Nodes::VariableIdentifier
545
- flatten_ast_value(query.variables[v.name])
546
- else
547
- v
940
+ def resolve_type(type, value, path)
941
+ trace_payload = { context: context, type: type, object: value, path: path }
942
+ resolved_type, resolved_value = query.trace("resolve_type", trace_payload) do
943
+ query.resolve_type(type, value)
548
944
  end
549
- end
550
-
551
- def write_invalid_null_in_response(path, invalid_null_error)
552
- if !dead_path?(path)
553
- schema.type_error(invalid_null_error, context)
554
- write_in_response(path, nil)
555
- add_dead_path(path)
556
- end
557
- end
558
945
 
559
- def write_execution_errors_in_response(path, errors)
560
- if !dead_path?(path)
561
- errors.each do |v|
562
- context.errors << v
946
+ if lazy?(resolved_type)
947
+ GraphQL::Execution::Lazy.new do
948
+ query.trace("resolve_type_lazy", trace_payload) do
949
+ schema.sync_lazy(resolved_type)
950
+ end
563
951
  end
564
- write_in_response(path, nil)
565
- add_dead_path(path)
566
- end
567
- end
568
-
569
- def write_in_response(path, value)
570
- if dead_path?(path)
571
- return
572
952
  else
573
- if value.nil? && path.any? && type_at(path).non_null?
574
- # This nil is invalid, try writing it at the previous spot
575
- propagate_path = path[0..-2]
576
- write_in_response(propagate_path, value)
577
- add_dead_path(propagate_path)
578
- else
579
- @response.write(path, value)
580
- end
953
+ [resolved_type, resolved_value]
581
954
  end
582
955
  end
583
956
 
584
- # To propagate nulls, we have to know what the field type was
585
- # at previous parts of the response.
586
- # This hash matches the response
587
- def type_at(path)
588
- t = @types_at_paths
589
- path.each do |part|
590
- t = t[part] || (raise("Invariant: #{part.inspect} not found in #{t}"))
591
- end
592
- t = t[:__type]
593
- t
594
- end
595
-
596
- def set_type_at_path(path, type)
597
- types = @types_at_paths
598
- path.each do |part|
599
- types = types[part] ||= {}
600
- end
601
- # Use this magic key so that the hash contains:
602
- # - string keys for nested fields
603
- # - :__type for the object type of a selection
604
- types[:__type] ||= type
605
- nil
606
- end
607
-
608
- # Mark `path` as having been permanently nulled out.
609
- # No values will be added beyond that path.
610
- def add_dead_path(path)
611
- dead = @dead_paths
612
- path.each do |part|
613
- dead = dead[part] ||= {}
614
- end
615
- dead[:__dead] = true
957
+ def authorized_new(type, value, context)
958
+ type.authorized_new(value, context)
616
959
  end
617
960
 
618
- def dead_path?(path)
619
- res = @dead_paths
620
- path.each do |part|
621
- if res
622
- if res[:__dead]
623
- break
624
- else
625
- res = res[part]
626
- end
627
- end
628
- end
629
- res && res[:__dead]
961
+ def lazy?(object)
962
+ @lazy_cache.fetch(object.class) {
963
+ @lazy_cache[object.class] = @schema.lazy?(object)
964
+ }
630
965
  end
631
966
  end
632
967
  end