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,82 @@
1
+ # frozen_string_literal: true
2
+
3
+ module Rails # :nodoc:
4
+ module GraphQL # :nodoc:
5
+ class Type # :nodoc:
6
+ # = GraphQL InterfaceType
7
+ #
8
+ # Interfaces represent a list of named fields and their types.
9
+ # See http://spec.graphql.org/June2018/#InterfaceTypeDefinition
10
+ #
11
+ # This class doesn't implements +valid_output?+ nor any of the output like
12
+ # methods because the Object class that uses interface already fetches
13
+ # all the fields for its composition and does the validating and
14
+ # serializing process.
15
+ class Interface < Type
16
+ extend Helpers::WithAssignment
17
+ extend Helpers::WithFields
18
+
19
+ setup! output: true
20
+
21
+ self.field_type = Field::OutputField
22
+
23
+ # The purpose of instantiating an interface is to have access to its
24
+ # public methods. It then runs from the strategy perspective, pointing
25
+ # out any other methods to the manually set event
26
+ delegate_missing_to :event
27
+ attr_reader :event
28
+
29
+ class << self
30
+ # TODO: Use inherited attribute for types
31
+
32
+ # Stores the list of types associated with the interface so it can
33
+ # be used during the execution step to find the right object type
34
+ def types
35
+ @types ||= Set.new
36
+ end
37
+
38
+ # Get the list of all inherited-aware associated types
39
+ def all_types
40
+ (superclass.try(:all_types) || Set.new) + (defined?(@types) ? @types : Set.new)
41
+ end
42
+
43
+ # Check if the other type is equivalent, by checking if the other is
44
+ # an object and the object implements this interface
45
+ def =~(other)
46
+ super || (other.object? && other.implements?(self))
47
+ end
48
+
49
+ # When attaching an interface to an object, copy the fields and add to
50
+ # the list of types. Pre-existing same-named fields with are not
51
+ # equivalent produces an exception.
52
+ def implemented(object)
53
+ fields.each do |name, field|
54
+ defined = object.field?(name)
55
+ invalid = defined && object.fields[name] !~ field
56
+ raise ArgumentError, <<~MSG.squish if invalid
57
+ The "#{object.gql_name}" object already has a "#{field.gql_name}" field and it
58
+ is not equivalent to the one defined on the "#{gql_name}" interface.
59
+ MSG
60
+
61
+ object.proxy_field(field) unless defined
62
+ end
63
+
64
+ types << object
65
+ end
66
+
67
+ def inspect # :nodoc:
68
+ fields = @fields.values.map(&:inspect)
69
+ fields = fields.presence && " {#{fields.join(', ')}}"
70
+ "#<GraphQL::Interface #{gql_name}#{fields}>"
71
+ end
72
+
73
+ # Check if the given object is properly implementing this interface
74
+ def validate(*)
75
+ # Don't validate interfaces since the fields are copied and
76
+ # the interface might have broken field types due to namespaces
77
+ end
78
+ end
79
+ end
80
+ end
81
+ end
82
+ end
@@ -0,0 +1,111 @@
1
+ # frozen_string_literal: true
2
+
3
+ module Rails # :nodoc:
4
+ module GraphQL # :nodoc:
5
+ class Type # :nodoc:
6
+ # = GraphQL ObjectType
7
+ #
8
+ # Objects represent a list of named fields, each of which yield a value of
9
+ # a specific type.
10
+ # See http://spec.graphql.org/June2018/#ObjectTypeDefinition
11
+ class Object < Type
12
+ extend ActiveSupport::Autoload
13
+ extend Helpers::WithAssignment
14
+ extend Helpers::WithFields
15
+
16
+ setup! output: true
17
+
18
+ self.field_type = Field::OutputField
19
+ self.valid_field_types = [
20
+ Type::Enum,
21
+ Type::Interface,
22
+ Type::Object,
23
+ Type::Scalar,
24
+ Type::Union,
25
+ ].freeze
26
+
27
+ eager_autoload do
28
+ autoload :DirectiveObject
29
+ autoload :EnumValueObject
30
+ autoload :FieldObject
31
+ autoload :InputValueObject
32
+ autoload :SchemaObject
33
+ autoload :TypeObject
34
+ end
35
+
36
+ # Define the methods for accessing the interfaces of the object
37
+ inherited_collection :interfaces, instance_reader: false
38
+
39
+ # The purpose of instantiating an object is to have access to its
40
+ # public methods. It then runs from the strategy perspective, pointing
41
+ # out any other methods to the manually set event
42
+ delegate_missing_to :event
43
+ attr_reader :event
44
+
45
+ class << self
46
+ # Plain objects can check if a given value is a valid member
47
+ def valid_member?(value)
48
+ return true if valid_assignment?(value)
49
+ checker = value.is_a?(Hash) ? :key? : :respond_to?
50
+ value = value.with_indifferent_access if value.is_a?(Hash)
51
+ fields.values.all? { |field| value.public_send(checker, field.method_name) }
52
+ end
53
+
54
+ # Check if the other type is equivalent, by checking if the other is
55
+ # an interface that the current object implements
56
+ def =~(other)
57
+ super || (other.interface? && implements?(other))
58
+ end
59
+
60
+ # Use this method to assign interfaces to the object
61
+ def implements(*others)
62
+ return if others.blank?
63
+
64
+ cache = all_interfaces.dup
65
+ others.flat_map do |item|
66
+ item = find_interface!(item)
67
+ next if cache.include?(item)
68
+
69
+ item.implemented(self)
70
+ interfaces << item
71
+ cache << item
72
+ end
73
+ end
74
+
75
+ # Check if the object implements the given +interface+
76
+ def implements?(interface)
77
+ (object = find_interface(interface)).present? && all_interfaces.include?(object)
78
+ end
79
+
80
+ private
81
+
82
+ # Soft find an object as an +interface+
83
+ def find_interface(object)
84
+ find_interface!(object)
85
+ rescue ArgumentError
86
+ # The object was not found as an actual interface
87
+ end
88
+
89
+ # Find a given +object+ and ensures it is an interface
90
+ def find_interface!(object)
91
+ object = GraphQL.type_map.fetch!(object,
92
+ namespaces: namespaces,
93
+ prevent_register: self,
94
+ ) unless object.is_a?(Module) && object < Type::Interface
95
+
96
+ return object if object.try(:interface?)
97
+ raise ArgumentError, <<~MSG.squish
98
+ The given "#{object}" is not a valid interface.
99
+ MSG
100
+ end
101
+ end
102
+
103
+ protected
104
+
105
+ def deprecated_directive
106
+ GraphQL::Directive::DeprecatedDirective
107
+ end
108
+ end
109
+ end
110
+ end
111
+ end
@@ -0,0 +1,34 @@
1
+ # frozen_string_literal: true
2
+
3
+ module Rails # :nodoc:
4
+ module GraphQL # :nodoc:
5
+ class Type # :nodoc:
6
+ # The introspection object for directives
7
+ class Object::DirectiveObject < Object
8
+ self.assigned_to = 'Rails::GraphQL::Directive'
9
+ self.spec_object = true
10
+
11
+ rename! '__Directive'
12
+
13
+ desc <<~DESC
14
+ Directives provide a way to describe alternate runtime execution
15
+ and type validation behavior in a GraphQL document.
16
+
17
+ In some cases, you need to provide options to alter GraphQL’s execution
18
+ behavior in ways field arguments will not suffice, such as conditionally
19
+ including or skipping a field. Directives provide this by describing
20
+ additional information to the executor.
21
+ DESC
22
+
23
+ field :name, :string, null: false, method_name: :gql_name
24
+ field :description, :string
25
+ field :locations, '__DirectiveLocation', full: true
26
+ field :args, '__InputValue', full: true
27
+
28
+ def args
29
+ all_arguments.values
30
+ end
31
+ end
32
+ end
33
+ end
34
+ end
@@ -0,0 +1,25 @@
1
+ # frozen_string_literal: true
2
+
3
+ module Rails # :nodoc:
4
+ module GraphQL # :nodoc:
5
+ class Type # :nodoc:
6
+ # The introspection object for an enum value
7
+ class Object::EnumValueObject < Object
8
+ self.spec_object = true
9
+
10
+ rename! '__EnumValue'
11
+
12
+ desc <<~DESC
13
+ One of the values of an Enum object. It is unique within the Enum set
14
+ of values. It's a string representation, not a numeric representation,
15
+ of a value kept as all caps (ie. ONE_VALUE).
16
+ DESC
17
+
18
+ field :name, :string, null: false
19
+ field :description, :string
20
+ field :is_deprecated, :boolean, null: false
21
+ field :deprecation_reason, :string
22
+ end
23
+ end
24
+ end
25
+ end
@@ -0,0 +1,54 @@
1
+ # frozen_string_literal: true
2
+
3
+ module Rails # :nodoc:
4
+ module GraphQL # :nodoc:
5
+ class Type # :nodoc:
6
+ # The introspection object for a field on objects and interfaces
7
+ class Object::FieldObject < Object
8
+ self.assigned_to = 'Rails::GraphQL::Field'
9
+ self.spec_object = true
10
+
11
+ delegate :fake_type_object, to: 'Object::TypeObject'
12
+
13
+ rename! '__Field'
14
+
15
+ desc <<~DESC
16
+ Fields are the elements that compose both Objects and Interfaces. Each
17
+ field in these other objects may contain arguments and always yields
18
+ a value of a specific type.
19
+ DESC
20
+
21
+ field :name, :string, null: false, method_name: :gql_name
22
+ field :description, :string
23
+ field :args, '__InputValue', full: true
24
+ field :type, '__Type', null: false, method_name: :build_type
25
+ field :is_deprecated, :boolean, null: false, method_name: :deprecated?
26
+ field :deprecation_reason, :string
27
+
28
+ def build_type
29
+ result = current.type_klass
30
+
31
+ if current.array?
32
+ result = fake_type_object(:non_null, result) unless current.nullable?
33
+ result = fake_type_object(:list, result)
34
+ end
35
+
36
+ result = fake_type_object(:non_null, result) unless current.null?
37
+ result
38
+ end
39
+
40
+ def args
41
+ all_arguments.values
42
+ end
43
+
44
+ def deprecated?
45
+ current.using?(deprecated_directive)
46
+ end
47
+
48
+ def deprecation_reason
49
+ current.all_directives.find { |item| item.is_a?(deprecated_directive) }&.args&.reason
50
+ end
51
+ end
52
+ end
53
+ end
54
+ end
@@ -0,0 +1,49 @@
1
+ # frozen_string_literal: true
2
+
3
+ module Rails # :nodoc:
4
+ module GraphQL # :nodoc:
5
+ class Type # :nodoc:
6
+ # The introspection object for a input object
7
+ class Object::InputValueObject < Object
8
+ self.assigned_to = 'Rails::GraphQL::Field::InputField'
9
+ self.spec_object = true
10
+
11
+ def self.valid_member?(value)
12
+ value.is_a?(GraphQL::Argument) || super
13
+ end
14
+
15
+ delegate :fake_type_object, to: 'Object::TypeObject'
16
+
17
+ rename! '__InputValue'
18
+
19
+ desc <<~DESC
20
+ Alongside with scalars and enums, input value objects allow the user
21
+ to provide values to arguments on fields and directives. Different
22
+ from those, input values accepts a list of keyed values, instead of
23
+ a single value.
24
+ DESC
25
+
26
+ field :name, :string, null: false, method_name: :gql_name
27
+ field :description, :string
28
+ field :type, '__Type', null: false, method_name: :build_type
29
+ field :default_value, :string
30
+
31
+ def default_value
32
+ current.to_json if current.default_value?
33
+ end
34
+
35
+ def build_type
36
+ result = current.type_klass
37
+
38
+ if current.array?
39
+ result = fake_type_object(:non_null, result) unless current.nullable?
40
+ result = fake_type_object(:list, result)
41
+ end
42
+
43
+ result = fake_type_object(:non_null, result) unless current.null?
44
+ result
45
+ end
46
+ end
47
+ end
48
+ end
49
+ end
@@ -0,0 +1,40 @@
1
+ # frozen_string_literal: true
2
+
3
+ module Rails # :nodoc:
4
+ module GraphQL # :nodoc:
5
+ class Type # :nodoc:
6
+ # The introspection object for a schema object
7
+ class Object::SchemaObject < Object
8
+ self.assigned_to = 'Rails::GraphQL::Schema'
9
+ self.spec_object = true
10
+
11
+ rename! '__Schema'
12
+
13
+ desc <<~DESC
14
+ A GraphQL service’s collective type system capabilities are referred
15
+ to as that service’s "schema". A schema is defined in terms of the
16
+ types and directives it supports as well as the root operation types
17
+ for each kind of operation: query, mutation, and subscription; this
18
+ determines the place in the type system where those operations begin.
19
+ DESC
20
+
21
+ field :types, '__Type', full: true, method_name: :read_types
22
+ field :query_type, '__Type', null: false
23
+ field :mutation_type, '__Type'
24
+ field :subscription_type, '__Type'
25
+ field :directives, '__Directive', full: true, method_name: :read_directives
26
+
27
+ # TODO: make it work for lazy enumerator
28
+ def read_types
29
+ event.schema.types(base_class: :Type).force
30
+ end
31
+
32
+ # TODO: it only works after eager_load!
33
+ def read_directives
34
+ Directive.eager_load!
35
+ event.schema.types(base_class: :Directive).force
36
+ end
37
+ end
38
+ end
39
+ end
40
+ end
@@ -0,0 +1,136 @@
1
+ # frozen_string_literal: true
2
+
3
+ module Rails # :nodoc:
4
+ module GraphQL # :nodoc:
5
+ class Type # :nodoc:
6
+ # The introspection object for any kind of type
7
+ class Object::TypeObject < Object
8
+ FAKE_TYPES = {
9
+ list: {
10
+ kind: :list,
11
+ kind_enum: 'LIST',
12
+ name: 'List',
13
+ object?: true,
14
+ description: nil,
15
+ },
16
+ non_null: {
17
+ kind: :non_null,
18
+ kind_enum: 'NON_NULL',
19
+ name: 'Non-Null',
20
+ object?: true,
21
+ description: nil,
22
+ },
23
+ }.freeze
24
+
25
+ self.assigned_to = 'Rails::GraphQL::Type'
26
+ self.spec_object = true
27
+
28
+ def self.valid_member?(value)
29
+ value.is_a?(OpenStruct) ? value.try(:object?) : super
30
+ end
31
+
32
+ def self.fake_type_object(type, subtype)
33
+ OpenStruct.new(**FAKE_TYPES[type].merge(of_type: subtype))
34
+ end
35
+
36
+ rename! '__Type'
37
+
38
+ desc <<~DESC
39
+ The fundamental unit of any GraphQL Schema is the type. There are six
40
+ kinds of named type definitions in GraphQL, and two wrapping types.
41
+
42
+ The most basic type is a +Scalar+. A scalar represents a primitive value,
43
+ like a string or an integer.
44
+
45
+ +Scalars+ and +Enums+ form the leaves in response trees; the intermediate
46
+ levels are +Object+ types, which define a set of fields.
47
+
48
+ An +Interface+ defines a list of fields; +Object+ types that implement
49
+ that interface are guaranteed to implement those fields.
50
+
51
+ A +Union+ defines a list of possible types; similar to interfaces,
52
+ whenever the type system claims a union will be returned, one of the
53
+ possible types will be returned.
54
+
55
+ Finally, oftentimes it is useful to provide complex structs as inputs
56
+ to GraphQL field arguments or variables; the +Input Object+ type allows
57
+ the schema to define exactly what data is expected.
58
+ DESC
59
+
60
+ field :kind, '__TypeKind', null: false,
61
+ method_name: :kind_enum
62
+
63
+ field :name, :string,
64
+ method_name: :gql_name
65
+
66
+ field :description, :string
67
+
68
+ field :fields, '__Field', array: true, nullable: false do
69
+ desc 'OBJECT and INTERFACE only'
70
+ argument :include_deprecated, :boolean, default: false
71
+ end
72
+
73
+ field :interfaces, '__Type', array: true, nullable: false,
74
+ desc: 'OBJECT only'
75
+
76
+ field :possible_types, '__Type', array: true, nullable: false,
77
+ desc: 'INTERFACE and UNION only'
78
+
79
+ field :enum_values, '__EnumValue', array: true, nullable: false do
80
+ desc 'ENUM only'
81
+ argument :include_deprecated, :boolean, default: false
82
+ end
83
+
84
+ field :input_fields, '__InputValue', array: true, nullable: false,
85
+ desc: 'INPUT_OBJECT only'
86
+
87
+ field :of_type, '__Type',
88
+ desc: 'NON_NULL and LIST only'
89
+
90
+ def fields(include_deprecated:)
91
+ return [] unless current.object? || current.interface?
92
+
93
+ list = current.fields.enum_for(:each_value)
94
+ list = list.reject { |field| field.using?(deprecated_directive) } \
95
+ unless include_deprecated
96
+
97
+ list
98
+ end
99
+
100
+ def enum_values(include_deprecated:)
101
+ return [] unless current.enum?
102
+
103
+ descs = all_value_description
104
+ deprecated = all_deprecated_values
105
+
106
+ list = all_values.lazy
107
+ list = list.reject { |value| deprecated.key?(value) } \
108
+ unless include_deprecated
109
+
110
+ list.map do |value|
111
+ OpenStruct.new(
112
+ name: value,
113
+ description: descs[value],
114
+ is_deprecated: deprecated.key?(value),
115
+ deprecation_reason: deprecated[value],
116
+ )
117
+ end
118
+ end
119
+
120
+ def interfaces
121
+ return [] unless current.object?
122
+ current.all_interfaces || []
123
+ end
124
+
125
+ def possible_types
126
+ return all_types if current.interface?
127
+ current.union? ? all_members : []
128
+ end
129
+
130
+ def input_fields
131
+ current.input? ? current.fields.enum_for(:each_value) : []
132
+ end
133
+ end
134
+ end
135
+ end
136
+ end