rails-graphql 0.1.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
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,85 @@
1
+ # frozen_string_literal: true
2
+
3
+ require 'rails/railtie'
4
+ require 'action_controller'
5
+ require 'action_controller/railtie'
6
+
7
+ module Rails # :nodoc:
8
+ module GraphQL # :nodoc:
9
+ # = Rails GraphQL Railtie
10
+ #
11
+ # Rails integration and configuration
12
+ class Railtie < Rails::Railtie
13
+ config.eager_load_namespaces << Rails::GraphQL
14
+ config.graphql = GraphQL.config
15
+
16
+ rake_tasks do
17
+ load 'rails/graphql.rake'
18
+ end
19
+
20
+ runner do
21
+ require_relative './schema'
22
+ end
23
+
24
+ # Ensure a valid logger
25
+ initializer 'graphql.logger' do
26
+ ActiveSupport.on_load(:graphql) do
27
+ return if config.logger.present?
28
+ if ::Rails.logger.respond_to?(:tagged)
29
+ config.logger = ::Rails.logger
30
+ else
31
+ config.logger = ActiveSupport::TaggedLogging.new(::Rails.logger)
32
+ end
33
+ end
34
+ end
35
+
36
+ # Expose database runtime to controller for logging.
37
+ initializer 'graphql.log_runtime' do
38
+ require_relative './railties/controller_runtime'
39
+ ActiveSupport.on_load(:action_controller) do
40
+ include GraphQL::ControllerRuntime
41
+ end
42
+ end
43
+
44
+ # Backtrace cleaner for removing gem paths
45
+ initializer 'graphql.backtrace_cleaner' do
46
+ require_relative './railties/log_subscriber'
47
+ ActiveSupport.on_load(:graphql) do
48
+ GraphQL::LogSubscriber.backtrace_cleaner = ::Rails.backtrace_cleaner
49
+ end
50
+ end
51
+
52
+ # Add reloader ability for files under 'app/graphql'
53
+ # TODO: Maybe improve to use rails auto loader
54
+ initializer 'graphql.reloader' do
55
+ Rails::GraphQL.eager_load!
56
+ ActiveSupport::Reloader.to_prepare do
57
+ Rails::GraphQL.type_map.use_checkpoint!
58
+ Rails::GraphQL.reload_ar_adapters!
59
+
60
+ Object.send(:remove_const, :GraphQL) if Object.const_defined?(:GraphQL)
61
+
62
+ load "#{__dir__}/shortcuts.rb"
63
+
64
+ $LOAD_PATH.each do |path|
65
+ next unless path =~ %r{\/app\/graphql$}
66
+ Dir.glob("#{path}/**/*.rb").sort.each(&method(:load))
67
+ end
68
+
69
+ GraphQL::Source.send(:build_pending!)
70
+ end
71
+ end
72
+
73
+ # initializer "active_record.set_reloader_hooks" do
74
+ # ActiveSupport.on_load(:active_record) do
75
+ # ActiveSupport::Reloader.before_class_unload do
76
+ # if ActiveRecord::Base.connected?
77
+ # ActiveRecord::Base.clear_cache!
78
+ # ActiveRecord::Base.clear_reloadable_connections!
79
+ # end
80
+ # end
81
+ # end
82
+ # end
83
+ end
84
+ end
85
+ end
@@ -0,0 +1,35 @@
1
+ # frozen_string_literal: true
2
+
3
+ module Rails # :nodoc:
4
+ module GraphQL # :nodoc:
5
+ # = GraphQL Base Generator
6
+ #
7
+ # A module to help generators to operate
8
+ module BaseGenerator
9
+ TEMPALTES_PATH = '../../../generators/graphql/templates'
10
+
11
+ def self.included(base)
12
+ base.send(:namespace, "graphql:#{base.name.demodulize.underscore[0..-11]}")
13
+ base.send(:source_root, File.expand_path(TEMPALTES_PATH, __dir__))
14
+ base.send(:class_option, :directory,
15
+ type: :string,
16
+ default: 'app/graphql',
17
+ desc: 'Directory where generated files should be saved',
18
+ )
19
+ end
20
+
21
+ protected
22
+
23
+ def app_module_name
24
+ require File.expand_path('config/application', destination_root)
25
+
26
+ app_class = Rails.application.class
27
+ source_name = app_class.respond_to?(:module_parent_name) \
28
+ ? :module_parent_name \
29
+ : :parent_name
30
+
31
+ app_class.send(source_name)
32
+ end
33
+ end
34
+ end
35
+ end
@@ -0,0 +1,101 @@
1
+ # frozen_string_literal: true
2
+
3
+ module Rails # :nodoc:
4
+ module GraphQL # :nodoc:
5
+ # = GraphQL Controller
6
+ #
7
+ # The controller helper methods that allow GraphQL to be performed on a
8
+ # Rails Controller class.
9
+ module Controller
10
+ extend ActiveSupport::Concern
11
+
12
+ REQUEST_XARGS = %i[operation_name variables context schema].freeze
13
+
14
+ included do
15
+ # Each controller is assigned to a GraphQL schema on which the requests
16
+ # will be performed from. It can be a string or the class
17
+ class_attribute :gql_schema, instance_accessor: false
18
+ end
19
+
20
+ # POST /execute
21
+ def execute
22
+ gql_request_response(gql_query)
23
+ end
24
+
25
+ # GET /describe
26
+ def describe
27
+ render plain: gql_schema.to_gql(
28
+ with_descriptions: !params.key?(:without_descriptions),
29
+ with_spec: !params.key?(:without_spec),
30
+ )
31
+ end
32
+
33
+ protected
34
+
35
+ # Render a response as a GraphQL request
36
+ def gql_request_response(*args, **xargs)
37
+ render json: gql_request(*args, **xargs)
38
+ end
39
+
40
+ # Execute a GraphQL request
41
+ def gql_request(query, **xargs)
42
+ request_xargs = REQUEST_XARGS.inject({}) do |result, setting|
43
+ result.merge(setting => (xargs[setting] || send("gql_#{setting}")))
44
+ end
45
+
46
+ ::Rails::GraphQL::Request.execute(query, **request_xargs)
47
+ end
48
+
49
+ # The schema on which the requests will be performed from
50
+ def gql_schema
51
+ schema = self.class.gql_schema
52
+ schema = schema.safe_constantize if schema.is_a?(String)
53
+ schema ||= application_default_schema
54
+ return schema if schema.is_a?(Module) && schema < ::Rails::GraphQL::Schema
55
+
56
+ raise ExecutionError, <<~MSG.squish
57
+ Unable to find a valid schema for #{self.class.name},
58
+ defined value: #{schema.inspect}.
59
+ MSG
60
+ end
61
+
62
+ # Get the GraphQL query to execute
63
+ def gql_query
64
+ params[:query]
65
+ end
66
+
67
+ # Get the GraphQL operation name
68
+ def gql_operation_name
69
+ params[:operationName]
70
+ end
71
+
72
+ # Get the GraphQL context for a requests
73
+ def gql_context
74
+ {}
75
+ end
76
+
77
+ # Get the GraphQL variables for a request
78
+ def gql_variables(variables = params[:variables])
79
+ case variables
80
+ when ::ActionController::Parameters then variables.permit!
81
+ when String then variables.present? ? JSON.parse(variables) : {}
82
+ when Hash then variables
83
+ else {}
84
+ end
85
+ end
86
+
87
+ private
88
+
89
+ # Find the default application schema
90
+ def application_default_schema
91
+ app_class = Rails.application.class
92
+ source_name = app_class.respond_to?(:module_parent_name) \
93
+ ? :module_parent_name \
94
+ : :parent_name
95
+
96
+ klass = "::GraphQL::#{app_class.send(source_name)}Schema".constantize
97
+ self.class.gql_schema = klass
98
+ end
99
+ end
100
+ end
101
+ end
@@ -0,0 +1,40 @@
1
+ # frozen_string_literal: true
2
+
3
+ require 'active_support/core_ext/module/attr_internal'
4
+
5
+ module Rails # :nodoc:
6
+ module GraphQL # :nodoc:
7
+ # = GraphQL Controller Runtime
8
+ #
9
+ # Tool that calculates the runtime of a GraphQL operation. This works
10
+ # similar to how Rails ActiveRecord calculate its execution time while
11
+ # performing a request.
12
+ module ControllerRuntime
13
+ extend ActiveSupport::Concern
14
+
15
+ module ClassMethods # :nodoc: all
16
+ def log_process_action(payload)
17
+ messages, gql_runtime = super, payload[:gql_runtime]
18
+ messages << format('GraphQL: %.1fms', gql_runtime.to_f) if gql_runtime
19
+ messages
20
+ end
21
+ end
22
+
23
+ private
24
+
25
+ attr_internal :gql_runtime
26
+
27
+ def process_action(*)
28
+ LogSubscriber.runtime = 0
29
+ super
30
+ end
31
+
32
+ def append_info_to_payload(payload)
33
+ super
34
+
35
+ payload[:gql_runtime] = LogSubscriber.runtime \
36
+ if (LogSubscriber.runtime || 0).positive?
37
+ end
38
+ end
39
+ end
40
+ end
@@ -0,0 +1,62 @@
1
+ # frozen_string_literal: true
2
+
3
+ module Rails # :nodoc:
4
+ module GraphQL # :nodoc:
5
+ # = GraphQL Log Subscriber
6
+ #
7
+ # This is the log tracker that workds the same way as ActiveRecord when it
8
+ # has to report on logs that a query was performed.
9
+ class LogSubscriber < ::ActiveSupport::LogSubscriber # :nodoc: all
10
+ class_attribute :backtrace_cleaner, default: ActiveSupport::BacktraceCleaner.new
11
+
12
+ def self.runtime
13
+ RuntimeRegistry.gql_runtime ||= 0
14
+ end
15
+
16
+ def self.runtime=(value)
17
+ RuntimeRegistry.gql_runtime = value
18
+ end
19
+
20
+ def request(event)
21
+ self.class.runtime += event.duration
22
+ return unless logger.debug?
23
+
24
+ payload = event.payload
25
+
26
+ name = ['GraphQL', payload[:name].presence]
27
+ name.unshift('CACHE') if payload[:cached]
28
+ name = "#{name.compact.join(' ')} (#{event.duration.round(1)}ms)"
29
+
30
+ document = payload[:document].squish
31
+ variables = payload[:variables].blank? ? nil : begin
32
+ " (#{JSON.pretty_generate(payload[:variables]).squish})"
33
+ end
34
+
35
+ debug " #{color(name, MAGENTA, true)} #{document}#{variables}"
36
+ end
37
+
38
+ private
39
+
40
+ def logger
41
+ GraphQL.config.logger
42
+ end
43
+
44
+ def debug(*)
45
+ return unless super
46
+
47
+ log_query_source if GraphQL.config.verbose_logs
48
+ end
49
+
50
+ def log_query_source
51
+ source = extract_query_source_location(caller)
52
+ logger.debug(" ↳ #{source}") if source
53
+ end
54
+
55
+ def extract_query_source_location(locations)
56
+ backtrace_cleaner.clean(locations.lazy).first
57
+ end
58
+ end
59
+ end
60
+
61
+ GraphQL::LogSubscriber.attach_to :graphql
62
+ end
@@ -0,0 +1,343 @@
1
+ # frozen_string_literal: true
2
+
3
+ module Rails # :nodoc:
4
+ module GraphQL # :nodoc:
5
+ # = GraphQL Request
6
+ #
7
+ # This class is responsible for processing a GraphQL response. It will
8
+ # handle queries, mutations, and subscription, as long as all of them are
9
+ # provided together. It also can be executed multiple times using the same
10
+ # context calling +execute+ multiple times.
11
+ #
12
+ # ==== Options
13
+ #
14
+ # * <tt>:namespace</tt> - Set what is the namespace used for the request
15
+ # (defaults to :base).
16
+ class Request
17
+ extend ActiveSupport::Autoload
18
+
19
+ RESPONSE_FORMATS = { string: :to_s, object: :to_h, hash: :to_h }.freeze
20
+
21
+ eager_autoload do
22
+ autoload_under :steps do
23
+ autoload :Organizable
24
+ autoload :Prepareable
25
+ autoload :Resolveable
26
+ end
27
+
28
+ autoload_under :helpers do
29
+ autoload :Directives
30
+ autoload :SelectionSet
31
+ autoload :ValueWriters
32
+ end
33
+
34
+ autoload :Arguments
35
+ autoload :Component
36
+ autoload :Context
37
+ autoload :Errors
38
+ autoload :Event
39
+ autoload :Strategy
40
+ end
41
+
42
+ attr_reader :schema, :visitor, :operations, :fragments, :errors,
43
+ :args, :response, :strategy, :stack
44
+
45
+ delegate :all_listeners, to: :schema
46
+
47
+ class << self
48
+ # Shortcut for initialize, set context, and execute
49
+ def execute(*args, schema: nil, namespace: :base, context: {}, **xargs)
50
+ result = new(schema, namespace: namespace)
51
+ result.context = context if context.present?
52
+ result.execute(*args, **xargs)
53
+ end
54
+
55
+ # Allow accessing component-based objects through the request
56
+ def const_defined?(name, *)
57
+ Component.const_defined?(name) || super
58
+ end
59
+
60
+ # Allow accessing component-based objects through the request
61
+ def const_missing(name)
62
+ Component.const_defined?(name) ? Component.const_get(name) : super
63
+ end
64
+
65
+ def eager_load! # :nodoc:
66
+ super
67
+
68
+ Request::Component.eager_load!
69
+ Request::Strategy.eager_load!
70
+ end
71
+ end
72
+
73
+ # Forces the schema to be registered on type map before moving forward
74
+ def initialize(schema = nil, namespace: :base)
75
+ @namespace = schema&.namespace || namespace
76
+ @schema = GraphQL::Schema.find!(@namespace)
77
+ @extensions = {}
78
+
79
+ ensure_schema!
80
+ end
81
+
82
+ # Cache all the schema events for this current request
83
+ def all_events
84
+ @all_events ||= schema.all_events
85
+ end
86
+
87
+ # Get the context of the request
88
+ def context
89
+ @context ||= OpenStruct.new.freeze
90
+ end
91
+
92
+ # Set the context of the request, it must be a +Hash+
93
+ def context=(data)
94
+ @context = build_ostruct(data).freeze
95
+ end
96
+
97
+ # Execute a given document with the given arguments
98
+ def execute(document, as: :string, **xargs)
99
+ reset!(**xargs)
100
+
101
+ to = RESPONSE_FORMATS[as]
102
+ @response = initialize_response(as, to)
103
+
104
+ execute!(document)
105
+ response.public_send(to)
106
+ end
107
+
108
+ alias perform execute
109
+
110
+ # Build a easy-to-access object representing the current information of
111
+ # the execution to be used on +rescue_with_handler+
112
+ def build_rescue_object(**extra)
113
+ OpenStruct.new(extra.reverse_merge(
114
+ args: @args,
115
+ source: stack.first,
116
+ request: self,
117
+ response: @response,
118
+ document: @document,
119
+ )).freeze
120
+ end
121
+
122
+ # Use schema handlers for exceptions caught during the execution process
123
+ def rescue_with_handler(exception, **extra)
124
+ schema.rescue_with_handler(exception, object: build_rescue_object(**extra))
125
+ end
126
+
127
+ # Add the given +exception+ to the errors using the +node+ location
128
+ def exception_to_error(exception, node, **xargs)
129
+ xargs[:exception] = exception.class.name
130
+ report_node_error(exception.message, node, **xargs)
131
+ end
132
+
133
+ # A little helper to report an error on a given node
134
+ def report_node_error(message, node, **xargs)
135
+ node = node.instance_variable_get(:@node) if node.is_a?(Request::Component)
136
+ location = GraphQL::Native.get_location(node)
137
+
138
+ xargs[:locations] ||= location.to_errors unless xargs.key?(:line)
139
+ report_error(message, **xargs)
140
+ end
141
+
142
+ # The final helper that facilitates how errors are reported
143
+ def report_error(message, **xargs)
144
+ xargs[:path] ||= stack_to_path
145
+ errors.add(message, **xargs)
146
+ end
147
+
148
+ # Add the given +object+ into the execution +stack+ and execute the given
149
+ # +block+ making sure to rescue exceptions using the +rescue_with_handler+
150
+ def stacked(object, &block)
151
+ stack.unshift(object)
152
+ block.call
153
+ rescue => exception
154
+ rescue_with_handler(exception) || raise
155
+ ensure
156
+ stack.shift
157
+ end
158
+
159
+ # Convert the current stack into a error path ignoring the schema
160
+ def stack_to_path
161
+ stack[0..-2].map do |item|
162
+ item.is_a?(Numeric) ? item : item.try(:gql_name)
163
+ end.compact.reverse
164
+ end
165
+
166
+ # Add extensions to the request, which ensures a bunch of extended
167
+ # behaviors for all the objects created through the request
168
+ def extend(*modules)
169
+ import_extensions(*modules)
170
+ request_ext = extensions[self.class]
171
+ super(request_ext) if request_ext && !is_a?(request_ext)
172
+ end
173
+
174
+ # This initiates a new object which is aware of extensions
175
+ def build(klass, *args, &block)
176
+ ext_module = extensions[klass]
177
+ obj = klass.new(*args, &block)
178
+ obj.extend(ext_module) if ext_module
179
+ obj
180
+ end
181
+
182
+ # Get and cache a sanitized version of the arguments
183
+ def sanitized_arguments
184
+ return {} unless args.each_pair.any?
185
+ cache(:sanitized_arguments) do
186
+ args.to_h.transform_keys { |key| key.to_s.camelize(:lower) }
187
+ end
188
+ end
189
+
190
+ # A shared way to cache information across the execution of an request
191
+ def cache(key, init_value = nil, &block)
192
+ @cache[key] ||= (init_value || block&.call || {})
193
+ end
194
+
195
+ private
196
+
197
+ attr_reader :extensions
198
+
199
+ # Reset principal variables and set the given +args+
200
+ def reset!(args: nil, variables: {}, operation_name: nil)
201
+ @args = build_ostruct(args || variables).freeze
202
+ @errors = Request::Errors.new(self)
203
+ @visitor = GraphQL::Native::Visitor.new
204
+ @operation_name = operation_name
205
+
206
+ @stack = [schema]
207
+ @cache = {}
208
+ @fragments = {}
209
+ @operations = {}
210
+ end
211
+
212
+ # This executes the whole process capturing any exceptions and handling
213
+ # them as defined by the schema
214
+ def execute!(document)
215
+ log_execution(document) do
216
+ @document = GraphQL::Native.parse(document)
217
+ collect_definitions!
218
+
219
+ @strategy = find_strategy!
220
+ @strategy.trigger_event(:request)
221
+ @strategy.resolve!
222
+ end
223
+ rescue ParseError => err
224
+ parts = err.message.match(/\A(\d+)\.(\d+)(?:-\d+)?: (.*)\z/)
225
+ errors.add(parts[3], line: parts[1], col: parts[2])
226
+ ensure
227
+ @cache.clear
228
+ @fragments.clear
229
+ @operations.clear
230
+
231
+ @visitor.terminate
232
+ @visitor = nil
233
+
234
+ GraphQL::Native.free_node(@document) if defined?(@document)
235
+ @response.try(:append_errors, errors)
236
+ end
237
+
238
+ # Use the visitor to collect the operations and fragments
239
+ def collect_definitions!
240
+ visitor.collect_definitions(@document) do |kind, node, data|
241
+ case kind
242
+ when :operation
243
+ operations[data[:name]] = Component::Operation.build(self, node, data)
244
+ when :fragment
245
+ fragments[data[:name]] = build(Component::Fragment, self, node, data)
246
+ end
247
+ end
248
+ end
249
+
250
+ # Find the best strategy to resolve the request
251
+ def find_strategy!
252
+ klass = schema.config.request_strategies.lazy.map(&:constantize).select do |k|
253
+ k.can_resolve?(self)
254
+ end.max_by(&:priority)
255
+ build(klass, self)
256
+ end
257
+
258
+ # Find all necessary extensions inside the given +modules+ and prepare
259
+ # the extension base module
260
+ def import_extensions(*modules)
261
+ modules.each do |mod|
262
+ mod.constants.each do |const_name|
263
+ const_name = const_name.to_s
264
+ const = mod.const_get(const_name)
265
+ next unless const.is_a?(Module)
266
+
267
+ # Find the related request class to extend
268
+ klass = const_name === 'Request' ? self.class : begin
269
+ const_name.split('_').inject(self.class) do |k, next_const|
270
+ k.const_defined?(next_const) ? k.const_get(next_const) : break
271
+ end
272
+ end
273
+
274
+ # Create the shared module and include the extension
275
+ next unless klass&.is_a?(Class)
276
+ extensions[klass] ||= Module.new
277
+ extensions[klass].include(const)
278
+ end
279
+ end
280
+ end
281
+
282
+ # Log the execution of a GraphQL document
283
+ def log_execution(document)
284
+ ActiveSupport::Notifications.instrument('request.graphql') do |payload|
285
+ yield.tap { log_payload(document, payload) }
286
+ end
287
+ end
288
+
289
+ # Build the payload to be sent to the log
290
+ def log_payload(document, data)
291
+ name = @operation_name.presence
292
+ name ||= operations.keys.first if operations.size.eql?(1)
293
+ map_variables = args.to_h if args.each_pair.any?
294
+
295
+ data.merge!(
296
+ name: name,
297
+ cached: false,
298
+ document: document,
299
+ variables: map_variables,
300
+ )
301
+ end
302
+
303
+ # Initialize the class that responsible for storaging the response
304
+ def initialize_response(as_format, to)
305
+ raise ::ArgumentError, <<~MSG.squish if to.nil?
306
+ The given format #{as_format.inspect} is not a valid reponse format.
307
+ MSG
308
+
309
+ klass = schema.config.enable_string_collector \
310
+ ? Collectors::JsonCollector \
311
+ : Collectors::HashCollector
312
+
313
+ obj = klass.new(self)
314
+ raise ::ArgumentError, <<~MSG.squish unless obj.respond_to?(to)
315
+ Unable to use "#{klass.name}" as response collector since it does
316
+ not implement a #{to.inspect} method.
317
+ MSG
318
+
319
+ obj
320
+ end
321
+
322
+ # Little helper to build an +OpenStruct+ ensure the given +value+ is a
323
+ # +Hash+. It can also +transform_keys+ with the given block
324
+ def build_ostruct(value, &block)
325
+ raise ::ArgumentError, <<~MSG.squish unless value.is_a?(Hash)
326
+ The "#{value.class.name}" is not a valid hash.
327
+ MSG
328
+
329
+ value = value.deep_transform_keys(&block) if block.present?
330
+ OpenStruct.new(value)
331
+ end
332
+
333
+ # Make sure that a schema was assigned by find the corresponding one for
334
+ # the namespace of the request
335
+ def ensure_schema!
336
+ raise ::ArgumentError, <<~MSG.squish if schema.nil?
337
+ Unable to perform a request under the #{@namespace.inspect} namespace,
338
+ because there are no schema assigned to it.
339
+ MSG
340
+ end
341
+ end
342
+ end
343
+ end