rails-graphql 0.1.0

Sign up to get free protection for your applications and to get access to all the features.
Files changed (266) hide show
  1. checksums.yaml +7 -0
  2. data/MIT-LICENSE +20 -0
  3. data/README.rdoc +19 -0
  4. data/Rakefile +31 -0
  5. data/ext/depend +3 -0
  6. data/ext/extconf.rb +57 -0
  7. data/ext/graphqlparser/Ast.cpp +346 -0
  8. data/ext/graphqlparser/Ast.h +1214 -0
  9. data/ext/graphqlparser/AstNode.h +36 -0
  10. data/ext/graphqlparser/AstVisitor.h +137 -0
  11. data/ext/graphqlparser/GraphQLParser.cpp +76 -0
  12. data/ext/graphqlparser/GraphQLParser.h +55 -0
  13. data/ext/graphqlparser/JsonVisitor.cpp +161 -0
  14. data/ext/graphqlparser/JsonVisitor.cpp.inc +456 -0
  15. data/ext/graphqlparser/JsonVisitor.h +121 -0
  16. data/ext/graphqlparser/JsonVisitor.h.inc +110 -0
  17. data/ext/graphqlparser/VERSION +1 -0
  18. data/ext/graphqlparser/c/GraphQLAst.cpp +324 -0
  19. data/ext/graphqlparser/c/GraphQLAst.h +180 -0
  20. data/ext/graphqlparser/c/GraphQLAstForEachConcreteType.h +44 -0
  21. data/ext/graphqlparser/c/GraphQLAstNode.cpp +25 -0
  22. data/ext/graphqlparser/c/GraphQLAstNode.h +33 -0
  23. data/ext/graphqlparser/c/GraphQLAstToJSON.cpp +21 -0
  24. data/ext/graphqlparser/c/GraphQLAstToJSON.h +24 -0
  25. data/ext/graphqlparser/c/GraphQLAstVisitor.cpp +55 -0
  26. data/ext/graphqlparser/c/GraphQLAstVisitor.h +53 -0
  27. data/ext/graphqlparser/c/GraphQLParser.cpp +35 -0
  28. data/ext/graphqlparser/c/GraphQLParser.h +54 -0
  29. data/ext/graphqlparser/dump_json_ast.cpp +48 -0
  30. data/ext/graphqlparser/lexer.lpp +324 -0
  31. data/ext/graphqlparser/parser.ypp +693 -0
  32. data/ext/graphqlparser/parsergen/lexer.cpp +2633 -0
  33. data/ext/graphqlparser/parsergen/lexer.h +528 -0
  34. data/ext/graphqlparser/parsergen/location.hh +189 -0
  35. data/ext/graphqlparser/parsergen/parser.tab.cpp +3300 -0
  36. data/ext/graphqlparser/parsergen/parser.tab.hpp +646 -0
  37. data/ext/graphqlparser/parsergen/position.hh +179 -0
  38. data/ext/graphqlparser/parsergen/stack.hh +156 -0
  39. data/ext/graphqlparser/syntaxdefs.h +19 -0
  40. data/ext/libgraphqlparser/AstNode.h +36 -0
  41. data/ext/libgraphqlparser/CMakeLists.txt +148 -0
  42. data/ext/libgraphqlparser/CONTRIBUTING.md +23 -0
  43. data/ext/libgraphqlparser/GraphQLParser.cpp +76 -0
  44. data/ext/libgraphqlparser/GraphQLParser.h +55 -0
  45. data/ext/libgraphqlparser/JsonVisitor.cpp +161 -0
  46. data/ext/libgraphqlparser/JsonVisitor.h +121 -0
  47. data/ext/libgraphqlparser/LICENSE +22 -0
  48. data/ext/libgraphqlparser/README.clang-tidy +7 -0
  49. data/ext/libgraphqlparser/README.md +84 -0
  50. data/ext/libgraphqlparser/ast/ast.ast +203 -0
  51. data/ext/libgraphqlparser/ast/ast.py +61 -0
  52. data/ext/libgraphqlparser/ast/c.py +100 -0
  53. data/ext/libgraphqlparser/ast/c.pyc +0 -0
  54. data/ext/libgraphqlparser/ast/c_impl.py +61 -0
  55. data/ext/libgraphqlparser/ast/c_impl.pyc +0 -0
  56. data/ext/libgraphqlparser/ast/c_visitor_impl.py +39 -0
  57. data/ext/libgraphqlparser/ast/c_visitor_impl.pyc +0 -0
  58. data/ext/libgraphqlparser/ast/casing.py +26 -0
  59. data/ext/libgraphqlparser/ast/casing.pyc +0 -0
  60. data/ext/libgraphqlparser/ast/cxx.py +197 -0
  61. data/ext/libgraphqlparser/ast/cxx.pyc +0 -0
  62. data/ext/libgraphqlparser/ast/cxx_impl.py +61 -0
  63. data/ext/libgraphqlparser/ast/cxx_impl.pyc +0 -0
  64. data/ext/libgraphqlparser/ast/cxx_json_visitor_header.py +42 -0
  65. data/ext/libgraphqlparser/ast/cxx_json_visitor_header.pyc +0 -0
  66. data/ext/libgraphqlparser/ast/cxx_json_visitor_impl.py +80 -0
  67. data/ext/libgraphqlparser/ast/cxx_json_visitor_impl.pyc +0 -0
  68. data/ext/libgraphqlparser/ast/cxx_visitor.py +64 -0
  69. data/ext/libgraphqlparser/ast/cxx_visitor.pyc +0 -0
  70. data/ext/libgraphqlparser/ast/js.py +65 -0
  71. data/ext/libgraphqlparser/ast/license.py +10 -0
  72. data/ext/libgraphqlparser/ast/license.pyc +0 -0
  73. data/ext/libgraphqlparser/c/GraphQLAstNode.cpp +25 -0
  74. data/ext/libgraphqlparser/c/GraphQLAstNode.h +33 -0
  75. data/ext/libgraphqlparser/c/GraphQLAstToJSON.cpp +21 -0
  76. data/ext/libgraphqlparser/c/GraphQLAstToJSON.h +24 -0
  77. data/ext/libgraphqlparser/c/GraphQLAstVisitor.cpp +55 -0
  78. data/ext/libgraphqlparser/c/GraphQLAstVisitor.h +53 -0
  79. data/ext/libgraphqlparser/c/GraphQLParser.cpp +35 -0
  80. data/ext/libgraphqlparser/c/GraphQLParser.h +54 -0
  81. data/ext/libgraphqlparser/clang-tidy-all.sh +3 -0
  82. data/ext/libgraphqlparser/cmake/version.cmake +16 -0
  83. data/ext/libgraphqlparser/dump_json_ast.cpp +48 -0
  84. data/ext/libgraphqlparser/go/README.md +20 -0
  85. data/ext/libgraphqlparser/go/callbacks.go +18 -0
  86. data/ext/libgraphqlparser/go/gotest.go +64 -0
  87. data/ext/libgraphqlparser/lexer.lpp +324 -0
  88. data/ext/libgraphqlparser/libgraphqlparser.pc.in +11 -0
  89. data/ext/libgraphqlparser/parser.ypp +693 -0
  90. data/ext/libgraphqlparser/parsergen/lexer.cpp +2633 -0
  91. data/ext/libgraphqlparser/parsergen/lexer.h +528 -0
  92. data/ext/libgraphqlparser/parsergen/location.hh +189 -0
  93. data/ext/libgraphqlparser/parsergen/parser.tab.cpp +3300 -0
  94. data/ext/libgraphqlparser/parsergen/parser.tab.hpp +646 -0
  95. data/ext/libgraphqlparser/parsergen/position.hh +179 -0
  96. data/ext/libgraphqlparser/parsergen/stack.hh +156 -0
  97. data/ext/libgraphqlparser/python/CMakeLists.txt +14 -0
  98. data/ext/libgraphqlparser/python/README.md +5 -0
  99. data/ext/libgraphqlparser/python/example.py +31 -0
  100. data/ext/libgraphqlparser/syntaxdefs.h +19 -0
  101. data/ext/libgraphqlparser/test/BuildCAPI.c +5 -0
  102. data/ext/libgraphqlparser/test/CMakeLists.txt +25 -0
  103. data/ext/libgraphqlparser/test/JsonVisitorTests.cpp +28 -0
  104. data/ext/libgraphqlparser/test/ParserTests.cpp +352 -0
  105. data/ext/libgraphqlparser/test/kitchen-sink.graphql +59 -0
  106. data/ext/libgraphqlparser/test/kitchen-sink.json +1 -0
  107. data/ext/libgraphqlparser/test/schema-kitchen-sink.graphql +78 -0
  108. data/ext/libgraphqlparser/test/schema-kitchen-sink.json +1 -0
  109. data/ext/libgraphqlparser/test/valgrind.supp +33 -0
  110. data/ext/version.cpp +21 -0
  111. data/lib/generators/graphql/controller_generator.rb +22 -0
  112. data/lib/generators/graphql/schema_generator.rb +22 -0
  113. data/lib/generators/graphql/templates/controller.erb +5 -0
  114. data/lib/generators/graphql/templates/schema.erb +6 -0
  115. data/lib/graphqlparser.so +0 -0
  116. data/lib/rails-graphql.rb +2 -0
  117. data/lib/rails/graphql.rake +1 -0
  118. data/lib/rails/graphql.rb +185 -0
  119. data/lib/rails/graphql/adapters/mysql_adapter.rb +0 -0
  120. data/lib/rails/graphql/adapters/pg_adapter.rb +50 -0
  121. data/lib/rails/graphql/adapters/sqlite_adapter.rb +39 -0
  122. data/lib/rails/graphql/argument.rb +220 -0
  123. data/lib/rails/graphql/callback.rb +124 -0
  124. data/lib/rails/graphql/collectors.rb +14 -0
  125. data/lib/rails/graphql/collectors/hash_collector.rb +83 -0
  126. data/lib/rails/graphql/collectors/idented_collector.rb +73 -0
  127. data/lib/rails/graphql/collectors/json_collector.rb +114 -0
  128. data/lib/rails/graphql/config.rb +61 -0
  129. data/lib/rails/graphql/directive.rb +203 -0
  130. data/lib/rails/graphql/directive/deprecated_directive.rb +59 -0
  131. data/lib/rails/graphql/directive/include_directive.rb +24 -0
  132. data/lib/rails/graphql/directive/skip_directive.rb +24 -0
  133. data/lib/rails/graphql/errors.rb +42 -0
  134. data/lib/rails/graphql/event.rb +141 -0
  135. data/lib/rails/graphql/field.rb +318 -0
  136. data/lib/rails/graphql/field/input_field.rb +92 -0
  137. data/lib/rails/graphql/field/mutation_field.rb +52 -0
  138. data/lib/rails/graphql/field/output_field.rb +96 -0
  139. data/lib/rails/graphql/field/proxied_field.rb +131 -0
  140. data/lib/rails/graphql/field/resolved_field.rb +96 -0
  141. data/lib/rails/graphql/field/scoped_config.rb +22 -0
  142. data/lib/rails/graphql/field/typed_field.rb +104 -0
  143. data/lib/rails/graphql/helpers.rb +40 -0
  144. data/lib/rails/graphql/helpers/attribute_delegator.rb +39 -0
  145. data/lib/rails/graphql/helpers/inherited_collection.rb +152 -0
  146. data/lib/rails/graphql/helpers/leaf_from_ar.rb +141 -0
  147. data/lib/rails/graphql/helpers/registerable.rb +103 -0
  148. data/lib/rails/graphql/helpers/with_arguments.rb +125 -0
  149. data/lib/rails/graphql/helpers/with_assignment.rb +113 -0
  150. data/lib/rails/graphql/helpers/with_callbacks.rb +55 -0
  151. data/lib/rails/graphql/helpers/with_directives.rb +126 -0
  152. data/lib/rails/graphql/helpers/with_events.rb +81 -0
  153. data/lib/rails/graphql/helpers/with_fields.rb +141 -0
  154. data/lib/rails/graphql/helpers/with_namespace.rb +40 -0
  155. data/lib/rails/graphql/helpers/with_owner.rb +35 -0
  156. data/lib/rails/graphql/helpers/with_schema_fields.rb +230 -0
  157. data/lib/rails/graphql/helpers/with_validator.rb +52 -0
  158. data/lib/rails/graphql/introspection.rb +53 -0
  159. data/lib/rails/graphql/native.rb +56 -0
  160. data/lib/rails/graphql/native/functions.rb +38 -0
  161. data/lib/rails/graphql/native/location.rb +41 -0
  162. data/lib/rails/graphql/native/pointers.rb +23 -0
  163. data/lib/rails/graphql/native/visitor.rb +349 -0
  164. data/lib/rails/graphql/railtie.rb +85 -0
  165. data/lib/rails/graphql/railties/base_generator.rb +35 -0
  166. data/lib/rails/graphql/railties/controller.rb +101 -0
  167. data/lib/rails/graphql/railties/controller_runtime.rb +40 -0
  168. data/lib/rails/graphql/railties/log_subscriber.rb +62 -0
  169. data/lib/rails/graphql/request.rb +343 -0
  170. data/lib/rails/graphql/request/arguments.rb +93 -0
  171. data/lib/rails/graphql/request/component.rb +100 -0
  172. data/lib/rails/graphql/request/component/field.rb +225 -0
  173. data/lib/rails/graphql/request/component/fragment.rb +118 -0
  174. data/lib/rails/graphql/request/component/operation.rb +178 -0
  175. data/lib/rails/graphql/request/component/operation/subscription.rb +16 -0
  176. data/lib/rails/graphql/request/component/spread.rb +119 -0
  177. data/lib/rails/graphql/request/component/typename.rb +82 -0
  178. data/lib/rails/graphql/request/context.rb +51 -0
  179. data/lib/rails/graphql/request/errors.rb +54 -0
  180. data/lib/rails/graphql/request/event.rb +112 -0
  181. data/lib/rails/graphql/request/helpers/directives.rb +64 -0
  182. data/lib/rails/graphql/request/helpers/selection_set.rb +87 -0
  183. data/lib/rails/graphql/request/helpers/value_writers.rb +115 -0
  184. data/lib/rails/graphql/request/steps/organizable.rb +146 -0
  185. data/lib/rails/graphql/request/steps/prepareable.rb +33 -0
  186. data/lib/rails/graphql/request/steps/resolveable.rb +32 -0
  187. data/lib/rails/graphql/request/strategy.rb +249 -0
  188. data/lib/rails/graphql/request/strategy/dynamic_instance.rb +41 -0
  189. data/lib/rails/graphql/request/strategy/multi_query_strategy.rb +36 -0
  190. data/lib/rails/graphql/request/strategy/sequenced_strategy.rb +28 -0
  191. data/lib/rails/graphql/schema.rb +272 -0
  192. data/lib/rails/graphql/shortcuts.rb +77 -0
  193. data/lib/rails/graphql/source.rb +371 -0
  194. data/lib/rails/graphql/source/active_record/builders.rb +154 -0
  195. data/lib/rails/graphql/source/active_record_source.rb +231 -0
  196. data/lib/rails/graphql/source/scoped_arguments.rb +87 -0
  197. data/lib/rails/graphql/to_gql.rb +368 -0
  198. data/lib/rails/graphql/type.rb +138 -0
  199. data/lib/rails/graphql/type/enum.rb +206 -0
  200. data/lib/rails/graphql/type/enum/directive_location_enum.rb +30 -0
  201. data/lib/rails/graphql/type/enum/type_kind_enum.rb +57 -0
  202. data/lib/rails/graphql/type/input.rb +134 -0
  203. data/lib/rails/graphql/type/interface.rb +82 -0
  204. data/lib/rails/graphql/type/object.rb +111 -0
  205. data/lib/rails/graphql/type/object/directive_object.rb +34 -0
  206. data/lib/rails/graphql/type/object/enum_value_object.rb +25 -0
  207. data/lib/rails/graphql/type/object/field_object.rb +54 -0
  208. data/lib/rails/graphql/type/object/input_value_object.rb +49 -0
  209. data/lib/rails/graphql/type/object/schema_object.rb +40 -0
  210. data/lib/rails/graphql/type/object/type_object.rb +136 -0
  211. data/lib/rails/graphql/type/scalar.rb +71 -0
  212. data/lib/rails/graphql/type/scalar/bigint_scalar.rb +34 -0
  213. data/lib/rails/graphql/type/scalar/binary_scalar.rb +30 -0
  214. data/lib/rails/graphql/type/scalar/boolean_scalar.rb +37 -0
  215. data/lib/rails/graphql/type/scalar/date_scalar.rb +34 -0
  216. data/lib/rails/graphql/type/scalar/date_time_scalar.rb +32 -0
  217. data/lib/rails/graphql/type/scalar/decimal_scalar.rb +35 -0
  218. data/lib/rails/graphql/type/scalar/float_scalar.rb +32 -0
  219. data/lib/rails/graphql/type/scalar/id_scalar.rb +39 -0
  220. data/lib/rails/graphql/type/scalar/int_scalar.rb +36 -0
  221. data/lib/rails/graphql/type/scalar/string_scalar.rb +28 -0
  222. data/lib/rails/graphql/type/scalar/time_scalar.rb +40 -0
  223. data/lib/rails/graphql/type/union.rb +87 -0
  224. data/lib/rails/graphql/type_map.rb +347 -0
  225. data/lib/rails/graphql/version.rb +7 -0
  226. data/test/assets/introspection-db.json +0 -0
  227. data/test/assets/introspection-mem.txt +1 -0
  228. data/test/assets/introspection.gql +91 -0
  229. data/test/assets/luke.jpg +0 -0
  230. data/test/assets/mem.gql +428 -0
  231. data/test/assets/sqlite.gql +423 -0
  232. data/test/config.rb +80 -0
  233. data/test/graphql/request/context_test.rb +70 -0
  234. data/test/graphql/schema_test.rb +190 -0
  235. data/test/graphql/source_test.rb +237 -0
  236. data/test/graphql/type/enum_test.rb +203 -0
  237. data/test/graphql/type/input_test.rb +138 -0
  238. data/test/graphql/type/interface_test.rb +72 -0
  239. data/test/graphql/type/object_test.rb +104 -0
  240. data/test/graphql/type/scalar/bigint_scalar_test.rb +42 -0
  241. data/test/graphql/type/scalar/binary_scalar_test.rb +17 -0
  242. data/test/graphql/type/scalar/boolean_scalar_test.rb +40 -0
  243. data/test/graphql/type/scalar/date_scalar_test.rb +29 -0
  244. data/test/graphql/type/scalar/date_time_scalar_test.rb +29 -0
  245. data/test/graphql/type/scalar/decimal_scalar_test.rb +28 -0
  246. data/test/graphql/type/scalar/float_scalar_test.rb +22 -0
  247. data/test/graphql/type/scalar/id_scalar_test.rb +26 -0
  248. data/test/graphql/type/scalar/int_scalar_test.rb +26 -0
  249. data/test/graphql/type/scalar/string_scalar_test.rb +17 -0
  250. data/test/graphql/type/scalar/time_scalar_test.rb +36 -0
  251. data/test/graphql/type/scalar_test.rb +45 -0
  252. data/test/graphql/type/union_test.rb +82 -0
  253. data/test/graphql/type_map_test.rb +362 -0
  254. data/test/graphql/type_test.rb +68 -0
  255. data/test/graphql_test.rb +55 -0
  256. data/test/integration/config.rb +56 -0
  257. data/test/integration/memory/star_wars_introspection_test.rb +144 -0
  258. data/test/integration/memory/star_wars_query_test.rb +184 -0
  259. data/test/integration/memory/star_wars_validation_test.rb +99 -0
  260. data/test/integration/schemas/memory.rb +232 -0
  261. data/test/integration/schemas/sqlite.rb +82 -0
  262. data/test/integration/sqlite/star_wars_introspection_test.rb +15 -0
  263. data/test/integration/sqlite/star_wars_mutation_test.rb +82 -0
  264. data/test/integration/sqlite/star_wars_query_test.rb +71 -0
  265. data/test/test_ext.rb +48 -0
  266. metadata +509 -0
@@ -0,0 +1,93 @@
1
+ # frozen_string_literal: true
2
+
3
+ module Rails # :nodoc:
4
+ module GraphQL # :nodoc:
5
+ class Request # :nodoc:
6
+ # = GraphQL Request Arguments
7
+ #
8
+ # This is an extension of an +OpenStruct+ since argument values can be
9
+ # assigned a Proc, which means that in order to collect their value, we
10
+ # need to rely on the current operation being processed.
11
+ #
12
+ # They lazy variable-based value is used for fragments, so that they can
13
+ # be organized only once and have their variables changed accordingly to
14
+ # the spread and operation.
15
+ class Arguments < OpenStruct
16
+ THREAD_KEY = :_rails_graphql_operation
17
+ LAZY_LOADER = ->(key, object) { object.variables[key] }.curry
18
+
19
+ delegate :key?, to: :@table
20
+
21
+ class << self
22
+ # Easy access to the easy loader method
23
+ def lazy
24
+ LAZY_LOADER
25
+ end
26
+
27
+ # Get the current operation thread safely
28
+ def operation
29
+ Thread.current[THREAD_KEY]
30
+ end
31
+
32
+ # Execute a block inside a scoped thread-safe arguments
33
+ def scoped(value)
34
+ old_value, Thread.current[THREAD_KEY] = operation, value
35
+
36
+ yield
37
+ ensure
38
+ Thread.current[THREAD_KEY] = old_value
39
+ end
40
+
41
+ # Check if it's performing inside a scoped value
42
+ def scoped?
43
+ operation.present?
44
+ end
45
+
46
+ # If it's running under a scope, transform proc based values
47
+ def transform(value)
48
+ return if value.nil?
49
+ scoped? && value.is_a?(Proc) ? value.call(operation) : value
50
+ end
51
+ end
52
+
53
+ # Transform any proc by its actual value before returning the hash
54
+ def to_h(*)
55
+ super.transform_values(&self.class.method(:transform))
56
+ end
57
+
58
+ # Before iterating, transform any needed value
59
+ def each_pair
60
+ enum = to_h.to_enum
61
+ return enum unless block_given?
62
+ enum.each { |v| yield v }
63
+ self
64
+ end
65
+
66
+ # rubocop:disable Style/MissingRespondToMissing
67
+ # Transform the value before returning
68
+ def method_missing(*)
69
+ self.class.transform(super)
70
+ end
71
+ # rubocop:enable Style/MissingRespondToMissing
72
+
73
+ # Transform the value before returning
74
+ def [](*)
75
+ self.class.transform(super)
76
+ end
77
+
78
+ # Transform the value before returning
79
+ def dig(name, *names)
80
+ result = self.class.transform(super(name))
81
+ names.empty? ? result : result&.dig(*names)
82
+ end
83
+
84
+ # Override the freeze method to just freeze the table and do not create
85
+ # the getters and setter methods
86
+ def freeze
87
+ @table.freeze
88
+ super
89
+ end
90
+ end
91
+ end
92
+ end
93
+ end
@@ -0,0 +1,100 @@
1
+ # frozen_string_literal: true
2
+
3
+ module Rails # :nodoc:
4
+ module GraphQL # :nodoc:
5
+ class Request # :nodoc:
6
+ # = GraphQL Request Component
7
+ #
8
+ # Component is an abstraction of any possible type of object represented
9
+ # by a not of the document of a request. This class helps building
10
+ # cross-component features, like holding event listeners, setting up
11
+ # commom initializer and providing helpers
12
+ class Component
13
+ extend ActiveSupport::Autoload
14
+
15
+ include Request::Organizable
16
+ include Request::Prepareable
17
+ include Request::Resolveable
18
+
19
+ class << self
20
+ # Return the kind of the component
21
+ def kind
22
+ @kind ||= name.demodulize.underscore.to_sym
23
+ end
24
+
25
+ # Helper to memoize results from parent delegation
26
+ def parent_memoize(*methods)
27
+ methods.each do |method_name|
28
+ define_method(method_name) do
29
+ result = parent.public_send(method_name)
30
+ define_singleton_method(method_name) { result }
31
+ result
32
+ end
33
+ end
34
+ end
35
+ end
36
+
37
+ attr_reader :data
38
+
39
+ delegate :schema, :visitor, :response, :strategy, to: :request
40
+ delegate :find_type!, :find_directive!, :trigger_event, to: :strategy
41
+ delegate :memo, to: :operation
42
+ delegate :kind, to: :class
43
+
44
+ alias of_type? is_a?
45
+
46
+ eager_autoload do
47
+ autoload :Field
48
+ autoload :Fragment
49
+ autoload :Operation
50
+ autoload :Spread
51
+ autoload :Typename
52
+ end
53
+
54
+ def initialize(node, data)
55
+ @node = node
56
+ @data = data
57
+ end
58
+
59
+ # Check if the component is in a invalid state
60
+ def invalid?
61
+ defined?(@invalid) && @invalid
62
+ end
63
+
64
+ # Mark the component as invalid
65
+ def invalidate!
66
+ @invalid = true
67
+ end
68
+
69
+ # Normaly, components are not assignable, only fields are
70
+ def assignable?
71
+ false
72
+ end
73
+
74
+ protected
75
+
76
+ # It's extremely important to have a way to access the current request
77
+ # since not all objects stores s direct pointer to it
78
+ def request
79
+ raise NotImplementedError
80
+ end
81
+
82
+ # Use the strategy to set the component into the stack
83
+ def stacked(value = self, &block)
84
+ strategy.stacked(value, &block)
85
+ end
86
+
87
+ # Run a given block and ensure to capture exceptions to set them as
88
+ # errors
89
+ def capture_exception(stage, invalidate = false)
90
+ yield
91
+ rescue StandardError => error
92
+ invalidate! if invalidate
93
+ stack_path = request.stack_to_path
94
+ stack_path << gql_name if respond_to?(:gql_name) && gql_name.present?
95
+ request.exception_to_error(error, @node, path: stack_path, stage: stage.to_s)
96
+ end
97
+ end
98
+ end
99
+ end
100
+ end
@@ -0,0 +1,225 @@
1
+ # frozen_string_literal: true
2
+
3
+ module Rails # :nodoc:
4
+ module GraphQL # :nodoc:
5
+ class Request # :nodoc:
6
+ # = GraphQL Request Component Field
7
+ #
8
+ # This class holds information about a given field that should be
9
+ # collected from the source of where it was requested.
10
+ class Component::Field < Component
11
+ include ValueWriters
12
+ include SelectionSet
13
+ include Directives
14
+
15
+ delegate :decorate, to: :type_klass
16
+ delegate :operation, :variables, to: :parent
17
+ delegate :method_name, :resolver, :performer, :type_klass, :leaf_type?,
18
+ :dynamic_resolver?, to: :field
19
+
20
+ parent_memoize :request
21
+
22
+ attr_reader :name, :alias_name, :parent, :field, :arguments, :current_object
23
+
24
+ alias args arguments
25
+
26
+ def initialize(parent, node, data)
27
+ @parent = parent
28
+
29
+ @name = data[:name]
30
+ @alias_name = data[:alias]
31
+
32
+ super(node, data)
33
+ end
34
+
35
+ # Return both the field directives and the request directives
36
+ def all_directives
37
+ field.all_directives + super
38
+ end
39
+
40
+ # Override that considers the requested field directives and also the
41
+ # definition field events, both from itself and its directives events
42
+ def all_listeners
43
+ field.all_listeners + super
44
+ end
45
+
46
+ # Override that considers the requested field directives and also the
47
+ # definition field events, both from itself and its directives events
48
+ def all_events
49
+ @all_events ||= Helpers.merge_hash_array(field.all_events, super)
50
+ end
51
+
52
+ # Get and cache all the arguments for the field
53
+ def all_arguments
54
+ request.cache(:arguments)[field] ||= begin
55
+ if (result = field.all_arguments).any?
56
+ result.each_value.map(&:gql_name).zip(result.each_value).to_h
57
+ else
58
+ {}
59
+ end
60
+ end
61
+ end
62
+
63
+ # Assign a given +field+ to this class. The field must be an output
64
+ # field, which means that +output_type?+ must be true. It also must be
65
+ # called exactly once per field.
66
+ def assing_to(field)
67
+ raise ArgumentError, <<~MSG.squish if defined?(@assigned)
68
+ The "#{gql_name}" field is already assigned to #{@field.inspect}.
69
+ MSG
70
+
71
+ @field = field
72
+ end
73
+
74
+ # Return the name of the field to be used on the response
75
+ def gql_name
76
+ alias_name || name
77
+ end
78
+
79
+ # A little helper for finding the correct parent type name
80
+ def typename
81
+ (try(:current_object) || try(:type_klass))&.gql_name
82
+ end
83
+
84
+ # Check if the field is an entry point, meaning that its parent is the
85
+ # operation and it is associated to a schema field
86
+ def entry_point?
87
+ parent.kind === :operation
88
+ end
89
+
90
+ # Fields are assignable because they are actually the selection, so they
91
+ # need to be assigned to a filed
92
+ def assignable?
93
+ true
94
+ end
95
+
96
+ # A little extension of the +is_a?+ method that allows checking it using
97
+ # the underlying +field+
98
+ def of_type?(klass)
99
+ super || field.of_type?(klass)
100
+ end
101
+
102
+ # When the field is invalid, there's no much to do
103
+ # TODO: Maybe add a invalid event trigger here
104
+ def resolve_invalid(error = nil)
105
+ request.exception_to_error(error, @node) if error.present?
106
+
107
+ validate_output!(nil)
108
+ response.safe_add(gql_name, nil)
109
+ rescue InvalidValueError
110
+ raise unless entry_point?
111
+ end
112
+
113
+ # When the +type_klass+ of an object is an interface or a union, the
114
+ # field needs to be redirected to the one from the actual resolved
115
+ # +object+ type
116
+ def resolve_with!(object)
117
+ return resolve! if invalid?
118
+
119
+ old_field, @field = @field, object[@field.name]
120
+ @current_object = object
121
+ resolve!
122
+ ensure
123
+ @field, @current_object = old_field, nil
124
+ end
125
+
126
+ protected
127
+
128
+ # Perform the organization step
129
+ def organize_then(&block)
130
+ super(block) do
131
+ check_assignment!
132
+
133
+ parse_arguments
134
+ parse_directives
135
+ parse_selection
136
+ end
137
+ end
138
+
139
+ # Perform the prepare step
140
+ def prepare_then(&block)
141
+ super { strategy.prepare(self, &block) }
142
+ end
143
+
144
+ # Perform the resolve step
145
+ def resolve_then(&block)
146
+ stacked do
147
+ strategy.perform(self) if field.mutation?
148
+ send(field.array? ? 'resolve_many' : 'resolve_one', &block)
149
+ rescue StandardError => error
150
+ resolve_invalid(error)
151
+ end
152
+ end
153
+
154
+ # Don't stack over response when it's processing as array
155
+ def stacked_selection?
156
+ !field.array?
157
+ end
158
+
159
+ private
160
+
161
+ # Resolve the value of the field for a single information
162
+ def resolve_one
163
+ strategy.resolve(self, decorate: true) do |value|
164
+ yield value if block_given?
165
+ trigger_event(:finalize)
166
+ end
167
+ end
168
+
169
+ # Resolve the field for a list of information
170
+ def resolve_many
171
+ strategy.resolve(self, array: true) do |item|
172
+ strategy.resolve(self, item, decorate: true) do |value|
173
+ yield value if block_given?
174
+ trigger_event(:finalize)
175
+ end
176
+ end
177
+ end
178
+
179
+ # This override allows reasigned fields to perform events. This
180
+ # happens when fields are originally organized from interfaces. If
181
+ # the event is stopped for the object, then it doesn't proceed to the
182
+ # strategy implementation, ensuring compatibility
183
+ def trigger_event(event_name, **xargs)
184
+ return super if !defined?(@current_object) || @current_object.nil?
185
+
186
+ listeners = request.cache(:dynamic_listeners)[field] ||= field.all_listeners
187
+ return super unless listeners.include?(event_name)
188
+
189
+ callbacks = request.cache(:dynamic_events)[field] ||= field.all_events
190
+ old_events, @all_events = @all_events, callbacks
191
+ super
192
+ ensure
193
+ @all_events = old_events
194
+ end
195
+
196
+ # Check if the field was assigned correctly to an output field
197
+ def check_assignment!
198
+ raise MissingFieldError, <<~MSG.squish if field.nil?
199
+ Unable to find a field named "#{gql_name}" on
200
+ #{entry_point? ? operation.kind : parent.type_klass.name}.
201
+ MSG
202
+
203
+ raise FieldError, <<~MSG.squish unless field.output_type?
204
+ The "#{gql_name}" was assigned to a non-output type of field: #{field.inspect}.
205
+ MSG
206
+
207
+ empty_selection = data[:selection].nil? || data[:selection].null?
208
+ raise FieldError, <<~MSG.squish if field.leaf_type? && !empty_selection
209
+ The "#{gql_name}" was assigned to the #{type_klass.gql_name} which
210
+ is a leaf type and does not have nested fields.
211
+ MSG
212
+
213
+ raise FieldError, <<~MSG.squish if !field.leaf_type? && empty_selection
214
+ The "#{gql_name}" was assigned to the #{type_klass.gql_name} which
215
+ is not a leaf type and requires a selection of fields.
216
+ MSG
217
+
218
+ raise DisabledFieldError, <<~MSG.squish if field.disabled?
219
+ The "#{gql_name}" was found but it is marked as disabled.
220
+ MSG
221
+ end
222
+ end
223
+ end
224
+ end
225
+ end
@@ -0,0 +1,118 @@
1
+ # frozen_string_literal: true
2
+
3
+ module Rails # :nodoc:
4
+ module GraphQL # :nodoc:
5
+ class Request # :nodoc:
6
+ # = GraphQL Request Component Fragment
7
+ #
8
+ # This class holds information about a given fragment defined using the
9
+ # +fragment+ statement during an execution. This will guide the validation
10
+ # and execution of it.
11
+ class Component::Fragment < Component
12
+ include SelectionSet
13
+ include Directives
14
+
15
+ attr_reader :name, :type_klass, :request
16
+
17
+ def initialize(request, node, data)
18
+ @name = data[:name]
19
+ @request = request
20
+
21
+ super(node, data)
22
+
23
+ check_duplicated_fragment!
24
+ end
25
+
26
+ # Return a lazy loaded variable proc
27
+ # TODO: Mark all the dependent variables
28
+ def variables
29
+ Request::Arguments.lazy
30
+ end
31
+
32
+ # Access the operation through the Request::Arguments
33
+ def operation
34
+ Request::Arguments.operation
35
+ end
36
+
37
+ # Spread should always be performed with a current object, thus the
38
+ # typename comes from it
39
+ def typename
40
+ @current_object.gql_name
41
+ end
42
+
43
+ # Only resolve if the +type_klass+ is equivalent to the given +object+
44
+ def resolve_with!(object)
45
+ return if invalid?
46
+
47
+ @current_object = object
48
+ resolve!
49
+ ensure
50
+ @current_object = nil
51
+ end
52
+
53
+ protected
54
+
55
+ # Fragments always resolve selection unstacked on response, meaning
56
+ # that its fields will be set in the same level as the parent
57
+ def stacked_selection?
58
+ false
59
+ end
60
+
61
+ # Perform the organization step
62
+ def organize_then(&block)
63
+ super(block) do
64
+ @type_klass = find_type!(data[:type])
65
+ parse_directives(:fragment_definition)
66
+
67
+ check_assignment!
68
+ parse_selection
69
+ end
70
+ end
71
+
72
+ # Resolve the spread operation
73
+ def resolve
74
+ return if invalid?
75
+
76
+ object = @current_object || type_klass
77
+ resolve_then if type_klass =~ object
78
+ end
79
+
80
+ # This will just trigger the selection resolver
81
+ def resolve_then(&block)
82
+ super(block) { resolve_fields(@current_object) }
83
+ end
84
+
85
+ private
86
+
87
+ # Check if the field was assigned correctly to an output field
88
+ def check_assignment!
89
+ raise ExecutionError, <<~MSG.squish unless type_klass.output_type?
90
+ Unable to assing #{type_klass.gql_name} to "#{name}" fragment because
91
+ it is not a output type.
92
+ MSG
93
+
94
+ raise ExecutionError, <<~MSG.squish if type_klass.leaf_type?
95
+ Unable to assing #{type_klass.gql_name} to "#{name}" fragment because
96
+ a "#{type_klass.kind}" type can not be the source of a fragmnet.
97
+ MSG
98
+ end
99
+
100
+ # If there is another fragment with the same name already defined,
101
+ # raise an error
102
+ def check_duplicated_fragment!
103
+ return unless request.fragments.key?(name)
104
+
105
+ invalidate!
106
+
107
+ other_node = request.fragments[name].instance_variable_get(:@node)
108
+ location = GraphQL::Native.get_location(other_node)
109
+
110
+ request.report_node_error(<<~MSG.squish, @node)
111
+ Duplicated fragment named "#{name}" defined on
112
+ line #{location.begin_line}:#{location.begin_column}
113
+ MSG
114
+ end
115
+ end
116
+ end
117
+ end
118
+ end