rails-graphql 0.2.1 → 1.0.0.rc1

Sign up to get free protection for your applications and to get access to all the features.
Files changed (315) hide show
  1. checksums.yaml +4 -4
  2. data/ext/console.rb +18 -0
  3. data/ext/extconf.h +3 -0
  4. data/ext/extconf.rb +1 -54
  5. data/ext/gql_parser.c +631 -0
  6. data/ext/gql_parser.h +21 -0
  7. data/ext/shared.c +477 -0
  8. data/ext/shared.h +177 -0
  9. data/lib/generators/graphql/channel_generator.rb +27 -0
  10. data/lib/generators/graphql/controller_generator.rb +9 -4
  11. data/lib/generators/graphql/install_generator.rb +49 -0
  12. data/lib/generators/graphql/schema_generator.rb +9 -4
  13. data/lib/generators/graphql/templates/channel.erb +7 -0
  14. data/lib/generators/graphql/templates/config.rb +97 -0
  15. data/lib/generators/graphql/templates/controller.erb +2 -0
  16. data/lib/generators/graphql/templates/schema.erb +5 -3
  17. data/lib/gql_parser.so +0 -0
  18. data/lib/rails/graphql/adapters/mysql_adapter.rb +59 -0
  19. data/lib/rails/graphql/adapters/pg_adapter.rb +25 -22
  20. data/lib/rails/graphql/adapters/sqlite_adapter.rb +17 -14
  21. data/lib/rails/graphql/alternative/field_set.rb +48 -0
  22. data/lib/rails/graphql/alternative/mutation.rb +17 -0
  23. data/lib/rails/graphql/alternative/query.rb +98 -0
  24. data/lib/rails/graphql/alternative/subscription.rb +18 -0
  25. data/lib/rails/graphql/alternative.rb +20 -0
  26. data/lib/rails/graphql/argument.rb +25 -26
  27. data/lib/rails/graphql/callback.rb +30 -14
  28. data/lib/rails/graphql/collectors/hash_collector.rb +26 -7
  29. data/lib/rails/graphql/collectors/idented_collector.rb +10 -7
  30. data/lib/rails/graphql/collectors/json_collector.rb +43 -17
  31. data/lib/rails/graphql/collectors.rb +4 -4
  32. data/lib/rails/graphql/config.rb +154 -23
  33. data/lib/rails/graphql/directive/cached_directive.rb +33 -0
  34. data/lib/rails/graphql/directive/deprecated_directive.rb +10 -10
  35. data/lib/rails/graphql/directive/include_directive.rb +4 -4
  36. data/lib/rails/graphql/directive/skip_directive.rb +4 -4
  37. data/lib/rails/graphql/directive/specified_by_directive.rb +24 -0
  38. data/lib/rails/graphql/directive.rb +134 -73
  39. data/lib/rails/graphql/errors.rb +33 -4
  40. data/lib/rails/graphql/event.rb +21 -9
  41. data/lib/rails/graphql/field/authorized_field.rb +17 -6
  42. data/lib/rails/graphql/field/input_field.rb +8 -12
  43. data/lib/rails/graphql/field/mutation_field.rb +43 -9
  44. data/lib/rails/graphql/field/output_field.rb +112 -12
  45. data/lib/rails/graphql/field/proxied_field.rb +35 -26
  46. data/lib/rails/graphql/field/resolved_field.rb +27 -25
  47. data/lib/rails/graphql/field/scoped_config.rb +10 -4
  48. data/lib/rails/graphql/field/subscription_field.rb +123 -0
  49. data/lib/rails/graphql/field/typed_field.rb +69 -24
  50. data/lib/rails/graphql/field.rb +89 -74
  51. data/lib/rails/graphql/global_id.rb +89 -0
  52. data/lib/rails/graphql/helpers/attribute_delegator.rb +5 -5
  53. data/lib/rails/graphql/helpers/inherited_collection/array.rb +51 -0
  54. data/lib/rails/graphql/helpers/inherited_collection/base.rb +45 -0
  55. data/lib/rails/graphql/helpers/inherited_collection/hash.rb +88 -0
  56. data/lib/rails/graphql/helpers/inherited_collection.rb +25 -76
  57. data/lib/rails/graphql/helpers/instantiable.rb +15 -0
  58. data/lib/rails/graphql/helpers/leaf_from_ar.rb +7 -7
  59. data/lib/rails/graphql/helpers/registerable.rb +44 -62
  60. data/lib/rails/graphql/helpers/unregisterable.rb +16 -0
  61. data/lib/rails/graphql/helpers/with_arguments.rb +33 -28
  62. data/lib/rails/graphql/helpers/with_assignment.rb +6 -6
  63. data/lib/rails/graphql/helpers/with_callbacks.rb +28 -11
  64. data/lib/rails/graphql/helpers/with_description.rb +73 -0
  65. data/lib/rails/graphql/helpers/with_directives.rb +58 -30
  66. data/lib/rails/graphql/helpers/with_events.rb +22 -23
  67. data/lib/rails/graphql/helpers/with_fields.rb +86 -26
  68. data/lib/rails/graphql/helpers/with_global_id.rb +22 -0
  69. data/lib/rails/graphql/helpers/with_name.rb +44 -0
  70. data/lib/rails/graphql/helpers/with_namespace.rb +7 -4
  71. data/lib/rails/graphql/helpers/with_owner.rb +8 -7
  72. data/lib/rails/graphql/helpers/with_schema_fields.rb +162 -56
  73. data/lib/rails/graphql/helpers/with_validator.rb +9 -9
  74. data/lib/rails/graphql/helpers.rb +10 -3
  75. data/lib/rails/graphql/introspection.rb +43 -36
  76. data/lib/rails/graphql/railtie.rb +89 -33
  77. data/lib/rails/graphql/railties/app/base_channel.rb +10 -0
  78. data/lib/rails/graphql/railties/app/base_controller.rb +12 -0
  79. data/lib/rails/graphql/railties/app/views/_cable.js.erb +56 -0
  80. data/lib/rails/graphql/railties/app/views/_fetch.js.erb +20 -0
  81. data/lib/rails/graphql/railties/app/views/graphiql.html.erb +101 -0
  82. data/lib/rails/graphql/railties/base_generator.rb +5 -17
  83. data/lib/rails/graphql/railties/channel.rb +157 -0
  84. data/lib/rails/graphql/railties/controller.rb +91 -25
  85. data/lib/rails/graphql/railties/controller_runtime.rb +5 -5
  86. data/lib/rails/graphql/railties/log_subscriber.rb +81 -14
  87. data/lib/rails/graphql/request/arguments.rb +26 -50
  88. data/lib/rails/graphql/request/backtrace.rb +212 -0
  89. data/lib/rails/graphql/request/component/field.rb +98 -70
  90. data/lib/rails/graphql/request/component/fragment.rb +80 -26
  91. data/lib/rails/graphql/request/component/operation/subscription.rb +162 -4
  92. data/lib/rails/graphql/request/component/operation.rb +73 -34
  93. data/lib/rails/graphql/request/component/spread.rb +79 -27
  94. data/lib/rails/graphql/request/component/typename.rb +28 -13
  95. data/lib/rails/graphql/request/component.rb +77 -36
  96. data/lib/rails/graphql/request/context.rb +19 -9
  97. data/lib/rails/graphql/request/errors.rb +16 -6
  98. data/lib/rails/graphql/request/event.rb +23 -8
  99. data/lib/rails/graphql/request/helpers/directives.rb +69 -27
  100. data/lib/rails/graphql/request/helpers/selection_set.rb +57 -25
  101. data/lib/rails/graphql/request/helpers/value_writers.rb +24 -19
  102. data/lib/rails/graphql/request/prepared_data.rb +100 -0
  103. data/lib/rails/graphql/request/steps/authorizable.rb +24 -14
  104. data/lib/rails/graphql/request/steps/organizable.rb +111 -49
  105. data/lib/rails/graphql/request/steps/{prepareable.rb → preparable.rb} +21 -8
  106. data/lib/rails/graphql/request/steps/{resolveable.rb → resolvable.rb} +16 -7
  107. data/lib/rails/graphql/request/strategy/cached_strategy.rb +64 -0
  108. data/lib/rails/graphql/request/strategy/dynamic_instance.rb +6 -6
  109. data/lib/rails/graphql/request/strategy/multi_query_strategy.rb +6 -13
  110. data/lib/rails/graphql/request/strategy/sequenced_strategy.rb +9 -9
  111. data/lib/rails/graphql/request/strategy.rb +147 -77
  112. data/lib/rails/graphql/request/subscription.rb +82 -0
  113. data/lib/rails/graphql/request.rb +353 -104
  114. data/lib/rails/graphql/schema.rb +251 -106
  115. data/lib/rails/graphql/shortcuts.rb +33 -8
  116. data/lib/rails/graphql/source/active_record/builders.rb +64 -51
  117. data/lib/rails/graphql/source/active_record_source.rb +158 -82
  118. data/lib/rails/graphql/source/base.rb +83 -0
  119. data/lib/rails/graphql/source/builder.rb +115 -0
  120. data/lib/rails/graphql/source/scoped_arguments.rb +39 -21
  121. data/lib/rails/graphql/source.rb +90 -228
  122. data/lib/rails/graphql/subscription/provider/action_cable.rb +113 -0
  123. data/lib/rails/graphql/subscription/provider/base.rb +192 -0
  124. data/lib/rails/graphql/subscription/provider.rb +18 -0
  125. data/lib/rails/graphql/subscription/store/base.rb +141 -0
  126. data/lib/rails/graphql/subscription/store/memory.rb +136 -0
  127. data/lib/rails/graphql/subscription/store.rb +19 -0
  128. data/lib/rails/graphql/subscription.rb +17 -0
  129. data/lib/rails/graphql/to_gql.rb +29 -32
  130. data/lib/rails/graphql/type/creator.rb +196 -0
  131. data/lib/rails/graphql/type/enum/directive_location_enum.rb +11 -11
  132. data/lib/rails/graphql/type/enum/type_kind_enum.rb +3 -3
  133. data/lib/rails/graphql/type/enum.rb +44 -50
  134. data/lib/rails/graphql/type/input.rb +92 -25
  135. data/lib/rails/graphql/type/interface.rb +29 -28
  136. data/lib/rails/graphql/type/object/directive_object.rb +10 -9
  137. data/lib/rails/graphql/type/object/enum_value_object.rb +3 -3
  138. data/lib/rails/graphql/type/object/field_object.rb +24 -6
  139. data/lib/rails/graphql/type/object/input_value_object.rb +6 -7
  140. data/lib/rails/graphql/type/object/schema_object.rb +5 -8
  141. data/lib/rails/graphql/type/object/type_object.rb +62 -25
  142. data/lib/rails/graphql/type/object.rb +34 -26
  143. data/lib/rails/graphql/type/scalar/any_scalar.rb +30 -0
  144. data/lib/rails/graphql/type/scalar/bigint_scalar.rb +5 -5
  145. data/lib/rails/graphql/type/scalar/binary_scalar.rb +5 -3
  146. data/lib/rails/graphql/type/scalar/boolean_scalar.rb +8 -8
  147. data/lib/rails/graphql/type/scalar/date_scalar.rb +5 -3
  148. data/lib/rails/graphql/type/scalar/date_time_scalar.rb +5 -3
  149. data/lib/rails/graphql/type/scalar/decimal_scalar.rb +5 -3
  150. data/lib/rails/graphql/type/scalar/float_scalar.rb +5 -5
  151. data/lib/rails/graphql/type/scalar/id_scalar.rb +6 -5
  152. data/lib/rails/graphql/type/scalar/int_scalar.rb +6 -5
  153. data/lib/rails/graphql/type/scalar/json_scalar.rb +41 -0
  154. data/lib/rails/graphql/type/scalar/string_scalar.rb +18 -4
  155. data/lib/rails/graphql/type/scalar/time_scalar.rb +8 -6
  156. data/lib/rails/graphql/type/scalar.rb +26 -23
  157. data/lib/rails/graphql/type/union.rb +21 -18
  158. data/lib/rails/graphql/type.rb +43 -26
  159. data/lib/rails/graphql/type_map.rb +268 -165
  160. data/lib/rails/graphql/uri.rb +167 -0
  161. data/lib/rails/graphql/version.rb +19 -3
  162. data/lib/rails/graphql.rake +3 -0
  163. data/lib/rails/graphql.rb +91 -56
  164. data/lib/rails-graphql.rb +1 -1
  165. data/test/assets/en.yml +29 -0
  166. data/test/assets/introspection-mem.txt +1 -1
  167. data/test/assets/introspection.gql +2 -0
  168. data/test/assets/mem.gql +86 -99
  169. data/test/assets/mysql.gql +406 -0
  170. data/test/assets/sqlite.gql +96 -73
  171. data/test/assets/translate.gql +346 -0
  172. data/test/config.rb +19 -8
  173. data/test/graphql/schema_test.rb +14 -50
  174. data/test/graphql/source_test.rb +8 -85
  175. data/test/graphql/type/enum_test.rb +207 -203
  176. data/test/graphql/type/input_test.rb +14 -9
  177. data/test/graphql/type/interface_test.rb +12 -9
  178. data/test/graphql/type/object_test.rb +8 -2
  179. data/test/graphql/type/scalar/any_scalar_test.rb +38 -0
  180. data/test/graphql/type/scalar/boolean_scalar_test.rb +6 -3
  181. data/test/graphql/type/scalar/json_scalar_test.rb +23 -0
  182. data/test/graphql/type_map_test.rb +63 -81
  183. data/test/graphql/type_test.rb +0 -19
  184. data/test/graphql_test.rb +1 -1
  185. data/test/integration/{authorization/authorization_test.rb → authorization_test.rb} +40 -14
  186. data/test/integration/config.rb +36 -3
  187. data/test/integration/customization_test.rb +39 -0
  188. data/test/integration/global_id_test.rb +99 -0
  189. data/test/integration/memory/star_wars_introspection_test.rb +24 -16
  190. data/test/integration/memory/star_wars_query_test.rb +54 -3
  191. data/test/integration/memory/star_wars_validation_test.rb +3 -3
  192. data/test/integration/mysql/star_wars_introspection_test.rb +25 -0
  193. data/test/integration/persisted_query_test.rb +87 -0
  194. data/test/integration/resolver_precedence_test.rb +154 -0
  195. data/test/integration/schemas/memory.rb +24 -10
  196. data/test/integration/schemas/mysql.rb +62 -0
  197. data/test/integration/schemas/sqlite.rb +21 -12
  198. data/test/integration/sqlite/star_wars_global_id_test.rb +89 -0
  199. data/test/integration/sqlite/star_wars_introspection_test.rb +10 -0
  200. data/test/integration/sqlite/star_wars_query_test.rb +14 -1
  201. data/test/integration/translate_test.rb +73 -0
  202. data/test/test_ext.rb +16 -13
  203. metadata +125 -161
  204. data/ext/depend +0 -3
  205. data/ext/graphqlparser/Ast.cpp +0 -346
  206. data/ext/graphqlparser/Ast.h +0 -1214
  207. data/ext/graphqlparser/AstNode.h +0 -36
  208. data/ext/graphqlparser/AstVisitor.h +0 -137
  209. data/ext/graphqlparser/GraphQLParser.cpp +0 -76
  210. data/ext/graphqlparser/GraphQLParser.h +0 -55
  211. data/ext/graphqlparser/JsonVisitor.cpp +0 -161
  212. data/ext/graphqlparser/JsonVisitor.cpp.inc +0 -456
  213. data/ext/graphqlparser/JsonVisitor.h +0 -121
  214. data/ext/graphqlparser/JsonVisitor.h.inc +0 -110
  215. data/ext/graphqlparser/VERSION +0 -1
  216. data/ext/graphqlparser/c/GraphQLAst.cpp +0 -324
  217. data/ext/graphqlparser/c/GraphQLAst.h +0 -180
  218. data/ext/graphqlparser/c/GraphQLAstForEachConcreteType.h +0 -44
  219. data/ext/graphqlparser/c/GraphQLAstNode.cpp +0 -25
  220. data/ext/graphqlparser/c/GraphQLAstNode.h +0 -33
  221. data/ext/graphqlparser/c/GraphQLAstToJSON.cpp +0 -21
  222. data/ext/graphqlparser/c/GraphQLAstToJSON.h +0 -24
  223. data/ext/graphqlparser/c/GraphQLAstVisitor.cpp +0 -55
  224. data/ext/graphqlparser/c/GraphQLAstVisitor.h +0 -53
  225. data/ext/graphqlparser/c/GraphQLParser.cpp +0 -35
  226. data/ext/graphqlparser/c/GraphQLParser.h +0 -54
  227. data/ext/graphqlparser/dump_json_ast.cpp +0 -48
  228. data/ext/graphqlparser/lexer.lpp +0 -324
  229. data/ext/graphqlparser/parser.ypp +0 -693
  230. data/ext/graphqlparser/parsergen/lexer.cpp +0 -2633
  231. data/ext/graphqlparser/parsergen/lexer.h +0 -528
  232. data/ext/graphqlparser/parsergen/location.hh +0 -189
  233. data/ext/graphqlparser/parsergen/parser.tab.cpp +0 -3300
  234. data/ext/graphqlparser/parsergen/parser.tab.hpp +0 -646
  235. data/ext/graphqlparser/parsergen/position.hh +0 -179
  236. data/ext/graphqlparser/parsergen/stack.hh +0 -156
  237. data/ext/graphqlparser/syntaxdefs.h +0 -19
  238. data/ext/libgraphqlparser/AstNode.h +0 -36
  239. data/ext/libgraphqlparser/CMakeLists.txt +0 -148
  240. data/ext/libgraphqlparser/CONTRIBUTING.md +0 -23
  241. data/ext/libgraphqlparser/GraphQLParser.cpp +0 -76
  242. data/ext/libgraphqlparser/GraphQLParser.h +0 -55
  243. data/ext/libgraphqlparser/JsonVisitor.cpp +0 -161
  244. data/ext/libgraphqlparser/JsonVisitor.h +0 -121
  245. data/ext/libgraphqlparser/LICENSE +0 -22
  246. data/ext/libgraphqlparser/README.clang-tidy +0 -7
  247. data/ext/libgraphqlparser/README.md +0 -84
  248. data/ext/libgraphqlparser/ast/ast.ast +0 -203
  249. data/ext/libgraphqlparser/ast/ast.py +0 -61
  250. data/ext/libgraphqlparser/ast/c.py +0 -100
  251. data/ext/libgraphqlparser/ast/c.pyc +0 -0
  252. data/ext/libgraphqlparser/ast/c_impl.py +0 -61
  253. data/ext/libgraphqlparser/ast/c_impl.pyc +0 -0
  254. data/ext/libgraphqlparser/ast/c_visitor_impl.py +0 -39
  255. data/ext/libgraphqlparser/ast/c_visitor_impl.pyc +0 -0
  256. data/ext/libgraphqlparser/ast/casing.py +0 -26
  257. data/ext/libgraphqlparser/ast/casing.pyc +0 -0
  258. data/ext/libgraphqlparser/ast/cxx.py +0 -197
  259. data/ext/libgraphqlparser/ast/cxx.pyc +0 -0
  260. data/ext/libgraphqlparser/ast/cxx_impl.py +0 -61
  261. data/ext/libgraphqlparser/ast/cxx_impl.pyc +0 -0
  262. data/ext/libgraphqlparser/ast/cxx_json_visitor_header.py +0 -42
  263. data/ext/libgraphqlparser/ast/cxx_json_visitor_header.pyc +0 -0
  264. data/ext/libgraphqlparser/ast/cxx_json_visitor_impl.py +0 -80
  265. data/ext/libgraphqlparser/ast/cxx_json_visitor_impl.pyc +0 -0
  266. data/ext/libgraphqlparser/ast/cxx_visitor.py +0 -64
  267. data/ext/libgraphqlparser/ast/cxx_visitor.pyc +0 -0
  268. data/ext/libgraphqlparser/ast/js.py +0 -65
  269. data/ext/libgraphqlparser/ast/license.py +0 -10
  270. data/ext/libgraphqlparser/ast/license.pyc +0 -0
  271. data/ext/libgraphqlparser/c/GraphQLAstNode.cpp +0 -25
  272. data/ext/libgraphqlparser/c/GraphQLAstNode.h +0 -33
  273. data/ext/libgraphqlparser/c/GraphQLAstToJSON.cpp +0 -21
  274. data/ext/libgraphqlparser/c/GraphQLAstToJSON.h +0 -24
  275. data/ext/libgraphqlparser/c/GraphQLAstVisitor.cpp +0 -55
  276. data/ext/libgraphqlparser/c/GraphQLAstVisitor.h +0 -53
  277. data/ext/libgraphqlparser/c/GraphQLParser.cpp +0 -35
  278. data/ext/libgraphqlparser/c/GraphQLParser.h +0 -54
  279. data/ext/libgraphqlparser/clang-tidy-all.sh +0 -3
  280. data/ext/libgraphqlparser/cmake/version.cmake +0 -16
  281. data/ext/libgraphqlparser/dump_json_ast.cpp +0 -48
  282. data/ext/libgraphqlparser/go/README.md +0 -20
  283. data/ext/libgraphqlparser/go/callbacks.go +0 -18
  284. data/ext/libgraphqlparser/go/gotest.go +0 -64
  285. data/ext/libgraphqlparser/lexer.lpp +0 -324
  286. data/ext/libgraphqlparser/libgraphqlparser.pc.in +0 -11
  287. data/ext/libgraphqlparser/parser.ypp +0 -693
  288. data/ext/libgraphqlparser/parsergen/lexer.cpp +0 -2633
  289. data/ext/libgraphqlparser/parsergen/lexer.h +0 -528
  290. data/ext/libgraphqlparser/parsergen/location.hh +0 -189
  291. data/ext/libgraphqlparser/parsergen/parser.tab.cpp +0 -3300
  292. data/ext/libgraphqlparser/parsergen/parser.tab.hpp +0 -646
  293. data/ext/libgraphqlparser/parsergen/position.hh +0 -179
  294. data/ext/libgraphqlparser/parsergen/stack.hh +0 -156
  295. data/ext/libgraphqlparser/python/CMakeLists.txt +0 -14
  296. data/ext/libgraphqlparser/python/README.md +0 -5
  297. data/ext/libgraphqlparser/python/example.py +0 -31
  298. data/ext/libgraphqlparser/syntaxdefs.h +0 -19
  299. data/ext/libgraphqlparser/test/BuildCAPI.c +0 -5
  300. data/ext/libgraphqlparser/test/CMakeLists.txt +0 -25
  301. data/ext/libgraphqlparser/test/JsonVisitorTests.cpp +0 -28
  302. data/ext/libgraphqlparser/test/ParserTests.cpp +0 -352
  303. data/ext/libgraphqlparser/test/kitchen-sink.graphql +0 -59
  304. data/ext/libgraphqlparser/test/kitchen-sink.json +0 -1
  305. data/ext/libgraphqlparser/test/schema-kitchen-sink.graphql +0 -78
  306. data/ext/libgraphqlparser/test/schema-kitchen-sink.json +0 -1
  307. data/ext/libgraphqlparser/test/valgrind.supp +0 -33
  308. data/ext/version.cpp +0 -21
  309. data/lib/graphqlparser.so +0 -0
  310. data/lib/rails/graphql/native/functions.rb +0 -38
  311. data/lib/rails/graphql/native/location.rb +0 -41
  312. data/lib/rails/graphql/native/pointers.rb +0 -23
  313. data/lib/rails/graphql/native/visitor.rb +0 -349
  314. data/lib/rails/graphql/native.rb +0 -56
  315. data/test/integration/schemas/authorization.rb +0 -12
@@ -1,7 +1,7 @@
1
1
  # frozen_string_literal: true
2
2
 
3
- module Rails # :nodoc:
4
- module GraphQL # :nodoc:
3
+ module Rails
4
+ module GraphQL
5
5
  # = GraphQL Directive
6
6
  #
7
7
  # This is the base object for directives definition.
@@ -22,24 +22,38 @@ module Rails # :nodoc:
22
22
  # add :old_value, directives: DeprecatedDirective(reason: 'not used anymore')
23
23
  class Directive
24
24
  extend ActiveSupport::Autoload
25
- extend Helpers::InheritedCollection
26
25
  extend Helpers::WithEvents
27
26
  extend Helpers::WithCallbacks
28
27
  extend Helpers::WithArguments
28
+ extend Helpers::WithGlobalID
29
29
  extend Helpers::Registerable
30
30
 
31
- VALID_LOCATIONS = Rails::GraphQL::Type::Enum::DirectiveLocationEnum
32
- .values.to_a.map { |value| value.downcase.to_sym }.freeze
31
+ EXECUTION_LOCATIONS = %i[
32
+ query mutation subscription field fragment_definition fragment_spread inline_fragment
33
+ ].to_set.freeze
33
34
 
34
- EXECUTION_LOCATIONS = VALID_LOCATIONS[0..6].freeze
35
- DEFINITION_LOCATIONS = VALID_LOCATIONS[7..17].freeze
35
+ DEFINITION_LOCATIONS = %i[
36
+ schema scalar object field_definition argument_definition interface union
37
+ enum enum_value input_object input_field_definition
38
+ ].to_set.freeze
39
+
40
+ VALID_LOCATIONS = (EXECUTION_LOCATIONS + DEFINITION_LOCATIONS).freeze
36
41
 
37
42
  class << self
38
- def kind # :nodoc
43
+ def kind
39
44
  :directive
40
45
  end
41
46
 
42
- def gql_name # :nodoc:
47
+ # Ensure to return the directive class
48
+ def base_type
49
+ GraphQL::Directive
50
+ end
51
+
52
+ alias gid_base_class base_type
53
+
54
+ # Return the name of the object as a GraphQL name, ensure to use the
55
+ # first letter as lower case when being auto generated
56
+ def gql_name
43
57
  return @gql_name if defined?(@gql_name)
44
58
  @gql_name = super.camelize(:lower)
45
59
  end
@@ -64,33 +78,36 @@ module Rails # :nodoc:
64
78
  @locations = list.to_set
65
79
  end
66
80
 
67
- def eager_load! # :nodoc:
68
- super
81
+ # A helper method that allows directives to be initialized while
82
+ # correctly parsing the arguments
83
+ def build(**xargs)
84
+ xargs = xargs.stringify_keys
85
+ result = all_arguments&.each&.each_with_object({}) do |(name, argument), hash|
86
+ hash[name] = argument.deserialize(xargs[argument.gql_name] || xargs[name.to_s])
87
+ end
69
88
 
70
- TypeMap.loaded! :Directive
89
+ new(**result)
71
90
  end
72
91
 
73
- def inspect # :nodoc:
74
- return '#<GraphQL::Directive>' if eql?(GraphQL::Directive)
92
+ # Return the directive, instantiate if it has params
93
+ def find_by_gid(gid)
94
+ options = { namespaces: gid.namespace, base_class: :Directive }
95
+ klass = GraphQL.type_map.fetch!(gid.name, **options)
96
+ gid.instantiate? ? klass.build(**gid.params) : klass
97
+ end
75
98
 
76
- args = arguments.each_value.map(&:inspect)
99
+ def inspect
100
+ return super if eql?(GraphQL::Directive)
101
+
102
+ repeatable = ' [repeatable]' if repeatable?
103
+ args = all_arguments&.each_value&.map(&:inspect)
104
+ args = args.force if args.respond_to?(:force)
77
105
  args = args.presence && "(#{args.join(', ')})"
78
- "#<GraphQL::Directive @#{gql_name}#{args}>"
106
+ +"#<GraphQL::Directive @#{gql_name}#{repeatable}#{args}>"
79
107
  end
80
108
 
81
109
  private
82
110
 
83
- # Check if the given list the locations are valid
84
- def validate_locations!(list)
85
- list.flatten!
86
- list.map! { |item| item.to_s.underscore.to_sym }
87
-
88
- invalid = list - VALID_LOCATIONS
89
- raise ArgumentError, <<~MSG.squish unless invalid.empty?
90
- Invalid locations for @#{gql_name}: #{invalid.to_sentence}.
91
- MSG
92
- end
93
-
94
111
  # Provide a nice way to use a directive without calling
95
112
  # +Directive.new+, like the +DeprecatedDirective+ can be initialized
96
113
  # using +GraphQL::DeprecatedDirective(*args)+
@@ -105,54 +122,58 @@ module Rails # :nodoc:
105
122
  end
106
123
  end
107
124
 
108
- # Allows checking value existence
109
- def respond_to_missing?(method_name, *)
110
- (const_defined?(method_name) rescue nil) || autoload?(method_name) || super
125
+ # A helper method that allows the +on+ and +for+ event filters to be
126
+ # used with things from both the TypeMap and the GraphQL shortcut
127
+ # classes
128
+ def sanitize_objects(setting)
129
+ GraphQL.enumerate(setting).map do |item|
130
+ next item unless item.is_a?(String) || item.is_a?(Symbol)
131
+ GraphQL.type_map.fetch(item, namespaces: namespaces) ||
132
+ ::GraphQL.const_get(item)
133
+ end
111
134
  end
112
135
 
113
- # Allow fast creation of values
114
- def method_missing(method_name, *args, **xargs, &block)
115
- const_get(method_name)&.new(*args, **xargs, &block) || super
116
- rescue ::NameError
117
- super
136
+ # Check if the given list the locations are valid
137
+ def validate_locations!(list)
138
+ invalid = list.flatten.lazy.reject do |item|
139
+ item = item.to_s.underscore.to_sym unless item.is_a?(Symbol)
140
+ VALID_LOCATIONS.include?(item)
141
+ end
142
+
143
+ raise ArgumentError, (+<<~MSG).squish if invalid.any?
144
+ Invalid locations for @#{gql_name}: #{invalid.force.to_sentence}.
145
+ MSG
118
146
  end
119
147
  end
120
148
 
121
- self.abstract = true
149
+ # Marks if the directive may be used repeatedly at a single location
150
+ class_attribute :repeatable, instance_accessor: false, default: false
122
151
 
123
- eager_autoload do
124
- autoload :DeprecatedDirective
125
- autoload :IncludeDirective
126
- autoload :SkipDirective
127
- end
152
+ self.abstract = true
128
153
 
129
- delegate :locations, :gql_name, to: :class
154
+ autoload :DeprecatedDirective
155
+ autoload :IncludeDirective
156
+ autoload :SkipDirective
157
+ autoload :SpecifiedByDirective
130
158
 
131
- array_sanitizer = ->(setting) do
132
- Array.wrap(setting)
133
- end
159
+ autoload :CachedDirective
134
160
 
135
- object_sanitizer = ->(setting) do
136
- Array.wrap(setting).map! do |item|
137
- next item unless item.is_a?(String) || item.is_a?(Symbol)
138
- GraphQL.type_map.fetch(item, namespaces: namespaces) ||
139
- ::GraphQL.const_get(item)
140
- end
141
- end
161
+ delegate :locations, :gql_name, :gid_base_class, :repeatable?, to: :class
142
162
 
143
- event_filter(:for, object_sanitizer) do |options, event|
144
- options.any?(&event.source.method(:of_type?))
163
+ # TODO: This filters are a bit confusing now, but `for` is working for @deprecated
164
+ event_filter(:for) do |options, event|
165
+ sanitize_objects(options).any?(&event.source.method(:of_type?))
145
166
  end
146
167
 
147
- event_filter(:on, object_sanitizer) do |options, event|
148
- event.respond_to?(:on?) && options.any?(&event.method(:on?))
168
+ event_filter(:on) do |options, event|
169
+ event.respond_to?(:on?) && sanitize_objects(options).any?(&event.method(:on?))
149
170
  end
150
171
 
151
- event_filter(:during, array_sanitizer) do |options, event|
152
- event.key?(:phase) && options.include?(event[:phase])
172
+ event_filter(:during) do |options, event|
173
+ event.key?(:phase) && GraphQL.enumerate(options).include?(event[:phase])
153
174
  end
154
175
 
155
- attr_reader :args
176
+ attr_reader :args, :event
156
177
 
157
178
  def initialize(args = nil, **xargs)
158
179
  @args = args || OpenStruct.new(xargs.transform_keys { |key| key.to_s.underscore })
@@ -160,44 +181,84 @@ module Rails # :nodoc:
160
181
  end
161
182
 
162
183
  # Once the directive is correctly prepared, we need to assign the owner
163
- def assing_owner!(owner)
164
- raise ArgumentError, <<~MSG.squish if defined?(@owner)
184
+ def assign_owner!(owner)
185
+ raise ArgumentError, (+<<~MSG).squish if defined?(@owner)
165
186
  Owner already assigned for @#{gql_name} directive.
166
187
  MSG
167
188
 
168
189
  @owner = owner
169
190
  end
170
191
 
192
+ # Correctly turn all the arguments into their +as_json+ version and return
193
+ # a hash of them
194
+ def args_as_json
195
+ all_arguments&.each&.with_object({}) do |(name, argument), hash|
196
+ hash[argument.gql_name] = argument.as_json(@args[name])
197
+ end
198
+ end
199
+
200
+ # Correctly turn all the arguments into their +to_json+ version and return
201
+ # a hash of them
202
+ def args_to_json
203
+ all_arguments&.each&.with_object({}) do |(name, argument), hash|
204
+ hash[argument.gql_name] = argument.to_json(@args[name])
205
+ end
206
+ end
207
+
171
208
  # When fetching all the events, embed the actual instance as the context
172
209
  # of the callback
210
+ # TODO: Maybe add a soft cached, based on the total number of events
173
211
  def all_events
174
- @all_events ||= self.class.all_events.transform_values do |events|
212
+ return unless self.class.events?
213
+
214
+ self.class.all_events.transform_values do |events|
175
215
  events.map { |item| Callback.set_context(item, self) }
176
216
  end
177
217
  end
178
218
 
179
219
  # Checks if all the arguments provided to the directive instance are valid
180
220
  def validate!(*)
181
- invalid = all_arguments.reject { |name, arg| arg.valid?(@args[name]) }
182
- return if invalid.empty?
221
+ raise ArgumentError, (+<<~MSG).squish unless defined?(@owner)
222
+ The @#{gql_name} directive is unbounded.
223
+ MSG
224
+
225
+ invalid = all_arguments&.reject { |name, arg| arg.valid?(@args[name]) }
226
+ return if invalid.blank?
183
227
 
184
- invalid = invalid.map { |name, _| <<~MSG }
185
- Invalid value "#{@args[name].inspect}" for #{name} argument.
228
+ invalid = invalid.each_key.map { |name| (+<<~MSG).squish }
229
+ invalid value "#{@args[name].inspect}" for #{name} argument
186
230
  MSG
187
231
 
188
- raise ArgumentError, <<~MSG.squish
232
+ raise ArgumentError, (+<<~MSG).squish
189
233
  Invalid usage of @#{gql_name} directive: #{invalid.to_sentence}.
190
234
  MSG
191
235
  end
192
236
 
193
- def inspect # :nodoc:
194
- args = all_arguments.map do |name, arg|
195
- "#{arg.gql_name}: #{@args[name].inspect}" unless @args[name].nil?
196
- end.compact
237
+ # This allows combining directives
238
+ def +(other)
239
+ [self, other].flatten
240
+ end
241
+
242
+ alias_method :&, :+
243
+
244
+ def inspect
245
+ args = all_arguments&.filter_map do |name, arg|
246
+ +"#{arg.gql_name}: #{@args[name].inspect}" unless @args[name].nil?
247
+ end
197
248
 
198
- args = args.presence && "(#{args.join(', ')})"
199
- "@#{gql_name}#{args}"
249
+ args = args.presence && +"(#{args.join(', ')})"
250
+ repeatable = ' [repeatable]' if repeatable?
251
+ unbound = ' # unbound' unless defined?(@owner)
252
+ +"@#{gql_name}#{repeatable}#{args}#{unbound}"
200
253
  end
254
+
255
+ %i[to_global_id to_gid to_gid_param].each do |method_name|
256
+ define_method(method_name) do
257
+ # TODO: The option is kind of broken, because they should always be a Hash
258
+ self.class.public_send(method_name, args_as_json&.compact || '')
259
+ end
260
+ end
261
+
201
262
  end
202
263
  end
203
264
  end
@@ -1,8 +1,8 @@
1
1
  # frozen_string_literal: true
2
2
 
3
- module Rails # :nodoc:
4
- module GraphQL # :nodoc:
5
- # Error class tha wrappes all the other error classes
3
+ module Rails
4
+ module GraphQL
5
+ # Error class tha wraps all the other error classes
6
6
  StandardError = Class.new(::StandardError)
7
7
 
8
8
  # Error class related to problems during the definition process
@@ -14,10 +14,13 @@ module Rails # :nodoc:
14
14
  # Errors that can happen related to the arguments given to a method
15
15
  ArgumentError = Class.new(DefinitionError)
16
16
 
17
+ # Errors that can happen when locking for definition objects, like fields
18
+ NotFoundError = Class.new(DefinitionError)
19
+
17
20
  # Errors related to the name of the objects
18
21
  NameError = Class.new(DefinitionError)
19
22
 
20
- # Errors related to duplciated objects
23
+ # Errors related to duplicated objects
21
24
  DuplicatedError = Class.new(NameError)
22
25
 
23
26
  # Error class related to problems during the execution process
@@ -26,6 +29,9 @@ module Rails # :nodoc:
26
29
  # Error related to the parsing process
27
30
  ParseError = Class.new(ExecutionError)
28
31
 
32
+ # Error class related to parsing the arguments
33
+ ArgumentsError = Class.new(ParseError)
34
+
29
35
  # Error class related to problems that happened during execution of fields
30
36
  FieldError = Class.new(ExecutionError)
31
37
 
@@ -42,5 +48,28 @@ module Rails # :nodoc:
42
48
  # Error class related to when a field is unauthorized and can not be used,
43
49
  # similar to disabled fields
44
50
  UnauthorizedFieldError = Class.new(FieldError)
51
+
52
+ # Error class related to problems that happened while subscribing to a field
53
+ SubscriptionError = Class.new(FieldError)
54
+
55
+ # Error class related to execution responses that don't require processing
56
+ StaticResponse = Class.new(Interrupt)
57
+
58
+ # Error class related to cached responses, which doesn't need processing
59
+ CachedResponse = Class.new(StaticResponse)
60
+
61
+ # Error class related to a persisted query that has't been persisted yet
62
+ PersistedQueryNotFound = Class.new(StaticResponse)
63
+
64
+ # A simple module and way to extend errors with extra information
65
+ ExtendedError = Module.new do
66
+ delegate_missing_to :@extension
67
+
68
+ def self.extend(error, extension)
69
+ error.instance_variable_set(:@extension, extension)
70
+ error.extend(self)
71
+ error
72
+ end
73
+ end
45
74
  end
46
75
  end
@@ -1,13 +1,13 @@
1
1
  # frozen_string_literal: true
2
2
 
3
- module Rails # :nodoc:
4
- module GraphQL # :nodoc:
3
+ module Rails
4
+ module GraphQL
5
5
  # = GraphQL Event
6
6
  #
7
- # This class is responsible for trigerring events. It also contains the
7
+ # This class is responsible for triggering events. It also contains the
8
8
  # +data+ that can be used on the event handlers.
9
9
  class Event
10
- attr_reader :source, :data, :name, :object, :last_result
10
+ attr_reader :source, :data, :event_name, :object, :last_result
11
11
 
12
12
  alias event itself
13
13
 
@@ -34,12 +34,23 @@ module Rails # :nodoc:
34
34
  @collect = data.delete(:collect?)
35
35
  @reverse = data.delete(:reverse?)
36
36
 
37
- @name = name
37
+ @event_name = name
38
38
  @data = data
39
39
  @source = source
40
40
  @layers = []
41
41
  end
42
42
 
43
+ # Check if the provided +other+ is equal to the source of the event. If
44
+ # other is a directive, then check if the source is using that directive
45
+ # TODO: Other cannot be an instance
46
+ def same_source?(other)
47
+ if other.is_a?(Directive) || (other.is_a?(Module) && other < Directive)
48
+ event_name == :attach || source.using?(other)
49
+ else
50
+ source == other
51
+ end
52
+ end
53
+
43
54
  # Return a given +name+ information from the event
44
55
  def parameter(name)
45
56
  respond_to?(name) ? public_send(name) : data[name]
@@ -71,10 +82,13 @@ module Rails # :nodoc:
71
82
  end
72
83
  end
73
84
 
85
+ alias on_instance set_on
86
+
74
87
  # From the list of all given objects, run the +trigger_object+
75
88
  def trigger_all(*objects)
76
89
  catchable(:stack) do
77
90
  iterator = @collect ? :map : :each
91
+ objects = objects.first if objects.size == 1 && objects.first.is_a?(Enumerable)
78
92
  objects.flatten.send(iterator, &method(:trigger_object))
79
93
  end
80
94
  end
@@ -91,7 +105,7 @@ module Rails # :nodoc:
91
105
  old_items, old_object, old_result, @object = @items, @object, @last_result, object
92
106
 
93
107
  catchable(:object) do
94
- events ||= object.all_events[name]
108
+ events ||= object.all_events.try(:[], event_name)
95
109
  stop if events.blank?
96
110
 
97
111
  @items = @reverse ? events.reverse_each : events.each
@@ -125,8 +139,6 @@ module Rails # :nodoc:
125
139
  # Do not do anything when missing next/super
126
140
  end
127
141
 
128
- alias call_super call_next
129
-
130
142
  protected
131
143
 
132
144
  alias args_source itself
@@ -138,7 +150,7 @@ module Rails # :nodoc:
138
150
  @layers.unshift(layer)
139
151
  catch(layer) { yield }
140
152
  ensure
141
- @layers.pop
153
+ @layers.shift
142
154
  end
143
155
 
144
156
  # Check for data based readers
@@ -1,14 +1,19 @@
1
1
  # frozen_string_literal: true
2
2
 
3
- module Rails # :nodoc:
4
- module GraphQL # :nodoc:
3
+ module Rails
4
+ module GraphQL
5
5
  # This provides ways for fields to be authorized, giving a logical level for
6
6
  # enabling or disabling access to a field. It has a similar structure to
7
7
  # events, but has a different hierarchy of resolution
8
8
  module Field::AuthorizedField
9
- # Just add the callbacks setup to the field
10
- def self.included(other)
11
- other.event_types(:authorize, append: true)
9
+ module Proxied # :nodoc: all
10
+ def authorizer
11
+ super || field.authorizer
12
+ end
13
+
14
+ def authorizable?
15
+ super || field.authorizable?
16
+ end
12
17
  end
13
18
 
14
19
  # Add either settings for authorization or a block to be executed. It
@@ -27,7 +32,13 @@ module Rails # :nodoc:
27
32
  def authorizable?
28
33
  defined?(@authorizer)
29
34
  end
35
+
36
+ protected
37
+
38
+ def proxied
39
+ super if defined? super
40
+ extend Field::AuthorizedField::Proxied
41
+ end
30
42
  end
31
43
  end
32
44
  end
33
- #
@@ -1,7 +1,7 @@
1
1
  # frozen_string_literal: true
2
2
 
3
- module Rails # :nodoc:
4
- module GraphQL # :nodoc:
3
+ module Rails
4
+ module GraphQL
5
5
  # = GraphQL Input Field
6
6
  #
7
7
  # An input field works the same way as an argument and they are pretty much
@@ -13,8 +13,6 @@ module Rails # :nodoc:
13
13
  # ==== Options
14
14
  #
15
15
  # * <tt>:default</tt> - Sets a default value for the argument (defaults to nil).
16
- # * <tt>:directives</tt> - The list of directives associated with the value
17
- # (defaults to nil).
18
16
  class Field::InputField < Field
19
17
  include Field::TypedField
20
18
 
@@ -25,11 +23,9 @@ module Rails # :nodoc:
25
23
 
26
24
  def initialize(*args, default: nil, **xargs, &block)
27
25
  super(*args, **xargs, &block)
28
- @default = default
29
- end
30
26
 
31
- def configure # :nodoc:
32
- raise ArgumentError, 'Input fields can\'t be further configured using blocks'
27
+ @default = default
28
+ @default = deserialize(@default) if @default.is_a?(::GQLParser::Token)
33
29
  end
34
30
 
35
31
  # Allow change the default value for the input
@@ -72,11 +68,11 @@ module Rails # :nodoc:
72
68
  def validate!(*)
73
69
  super if defined? super
74
70
 
75
- raise ArgumentError, <<~MSG.squish unless type_klass.input_type?
71
+ raise ArgumentError, (+<<~MSG).squish unless type_klass.input_type?
76
72
  The "#{type_klass.gql_name}" is not a valid input type.
77
73
  MSG
78
74
 
79
- raise ArgumentError, <<~MSG.squish unless default.nil? || valid_input?(default)
75
+ raise ArgumentError, (+<<~MSG).squish unless default.nil? || valid_input?(default)
80
76
  The given default value "#{default.inspect}" is not valid for this field.
81
77
  MSG
82
78
  end
@@ -85,7 +81,7 @@ module Rails # :nodoc:
85
81
 
86
82
  # Check if the given +value+ is a valid array as input
87
83
  def valid_input_array?(value, deep)
88
- return false unless value.is_a?(Array)
84
+ return false unless value.is_a?(::Array)
89
85
 
90
86
  value.all? do |val|
91
87
  (val.nil? && nullable?) || (leaf_type? || !deep) ||
@@ -95,7 +91,7 @@ module Rails # :nodoc:
95
91
 
96
92
  # Display the default value when it is present for inspection
97
93
  def inspect_default_value
98
- " = #{to_hash.inspect}" if default?
94
+ +" = #{as_json.inspect}" if default_value?
99
95
  end
100
96
  end
101
97
  end
@@ -1,11 +1,16 @@
1
1
  # frozen_string_literal: true
2
2
 
3
- module Rails # :nodoc:
4
- module GraphQL # :nodoc:
5
- # = GraphQL Output Field
3
+ module Rails
4
+ module GraphQL
5
+ # = GraphQL Mutation Field
6
6
  #
7
7
  # This is an extension of a normal output field, which just add extra
8
- # validation and ensurance that the +perform+ step can be executed
8
+ # validation and insurance that the +perform+ step can be executed
9
+ #
10
+ # ==== Options
11
+ #
12
+ # * <tt>:call</tt> - The alternative method to call to actually perform the mutation.
13
+ # (defaults to nil).
9
14
  class Field::MutationField < Field::OutputField
10
15
  redefine_singleton_method(:mutation?) { true }
11
16
 
@@ -15,6 +20,34 @@ module Rails # :nodoc:
15
20
  end
16
21
  end
17
22
 
23
+ # Intercept the initializer to maybe set the +perform_method_name+
24
+ def initialize(*args, call: nil, **xargs, &block)
25
+ @perform_method_name = call.to_sym unless call.nil?
26
+ super(*args, **xargs, &block)
27
+ end
28
+
29
+ # Accept changes to the perform method name through the +apply_changes+
30
+ def apply_changes(**xargs, &block)
31
+ @perform_method_name = xargs.delete(:call) if xargs.key?(:call)
32
+ super
33
+ end
34
+
35
+ # Allows overrides for the default bang method
36
+ def perform_method_name
37
+ if defined?(@perform_method_name)
38
+ @perform_method_name
39
+ elsif from_alternative?
40
+ :perform
41
+ else
42
+ :"#{method_name}!"
43
+ end
44
+ end
45
+
46
+ # Change the schema type of the field
47
+ def schema_type
48
+ :mutation
49
+ end
50
+
18
51
  # Add a block or a callable method that is executed before the resolver
19
52
  # but after all the before resolve. It returns +self+ for chain purposes
20
53
  def perform(*args, **xargs, &block)
@@ -25,16 +58,17 @@ module Rails # :nodoc:
25
58
  # Get the performer that can be already defined or used through the
26
59
  # +method_name+ if that is callable
27
60
  def performer
28
- @performer ||= callable?(:"#{method_name}!") \
29
- ? Callback.new(self, :perform, :"#{method_name}!") \
30
- : false
61
+ return @performer if defined?(@performer)
62
+
63
+ @performer = callable?(perform_method_name)
64
+ @performer = Callback.new(self, :perform, perform_method_name) if @performer
31
65
  end
32
66
 
33
67
  # Ensures that the performer is defined
34
68
  def validate!(*)
35
69
  super if defined? super
36
70
 
37
- raise ValidationError, <<~MSG.squish unless performer.present?
71
+ raise ValidationError, (+<<~MSG).squish unless performer.present?
38
72
  The "#{gql_name}" mutation field must have a perform action through a given
39
73
  block or a method named #{method_name} on #{owner.class.name}.
40
74
  MSG
@@ -42,7 +76,7 @@ module Rails # :nodoc:
42
76
 
43
77
  protected
44
78
 
45
- def proxied # :nodoc:
79
+ def proxied
46
80
  super if defined? super
47
81
  extend Field::MutationField::Proxied
48
82
  end