graphql 1.9.11 → 1.9.12

Sign up to get free protection for your applications and to get access to all the features.
Files changed (286) hide show
  1. checksums.yaml +4 -4
  2. data/lib/graphql/analysis/ast/query_complexity.rb +0 -8
  3. data/lib/graphql/analysis/ast/query_depth.rb +0 -8
  4. data/lib/graphql/analysis/ast/visitor.rb +26 -24
  5. data/lib/graphql/execution.rb +1 -0
  6. data/lib/graphql/execution/errors.rb +60 -0
  7. data/lib/graphql/execution/interpreter/runtime.rb +21 -17
  8. data/lib/graphql/static_validation/rules/fields_will_merge.rb +15 -8
  9. data/lib/graphql/version.rb +1 -1
  10. metadata +5 -556
  11. data/spec/dummy/Gemfile +0 -12
  12. data/spec/dummy/README.md +0 -24
  13. data/spec/dummy/Rakefile +0 -7
  14. data/spec/dummy/app/assets/config/manifest.js +0 -1
  15. data/spec/dummy/app/assets/javascripts/application.js +0 -66
  16. data/spec/dummy/app/channels/application_cable/channel.rb +0 -5
  17. data/spec/dummy/app/channels/application_cable/connection.rb +0 -5
  18. data/spec/dummy/app/channels/graphql_channel.rb +0 -116
  19. data/spec/dummy/app/controllers/application_controller.rb +0 -4
  20. data/spec/dummy/app/controllers/pages_controller.rb +0 -5
  21. data/spec/dummy/app/helpers/application_helper.rb +0 -3
  22. data/spec/dummy/app/jobs/application_job.rb +0 -3
  23. data/spec/dummy/app/views/layouts/application.html.erb +0 -12
  24. data/spec/dummy/app/views/pages/show.html +0 -16
  25. data/spec/dummy/bin/bundle +0 -4
  26. data/spec/dummy/bin/rails +0 -5
  27. data/spec/dummy/bin/rake +0 -5
  28. data/spec/dummy/bin/setup +0 -31
  29. data/spec/dummy/bin/update +0 -27
  30. data/spec/dummy/bin/yarn +0 -12
  31. data/spec/dummy/config.ru +0 -6
  32. data/spec/dummy/config/application.rb +0 -30
  33. data/spec/dummy/config/boot.rb +0 -4
  34. data/spec/dummy/config/cable.yml +0 -10
  35. data/spec/dummy/config/environment.rb +0 -6
  36. data/spec/dummy/config/environments/development.rb +0 -40
  37. data/spec/dummy/config/environments/production.rb +0 -76
  38. data/spec/dummy/config/environments/test.rb +0 -37
  39. data/spec/dummy/config/initializers/application_controller_renderer.rb +0 -9
  40. data/spec/dummy/config/initializers/backtrace_silencers.rb +0 -8
  41. data/spec/dummy/config/initializers/cookies_serializer.rb +0 -6
  42. data/spec/dummy/config/initializers/filter_parameter_logging.rb +0 -5
  43. data/spec/dummy/config/initializers/inflections.rb +0 -17
  44. data/spec/dummy/config/initializers/mime_types.rb +0 -5
  45. data/spec/dummy/config/initializers/wrap_parameters.rb +0 -10
  46. data/spec/dummy/config/locales/en.yml +0 -33
  47. data/spec/dummy/config/puma.rb +0 -57
  48. data/spec/dummy/config/routes.rb +0 -4
  49. data/spec/dummy/config/secrets.yml +0 -32
  50. data/spec/dummy/package.json +0 -5
  51. data/spec/dummy/public/404.html +0 -67
  52. data/spec/dummy/public/422.html +0 -67
  53. data/spec/dummy/public/500.html +0 -66
  54. data/spec/dummy/public/apple-touch-icon-precomposed.png +0 -0
  55. data/spec/dummy/public/apple-touch-icon.png +0 -0
  56. data/spec/dummy/public/favicon.ico +0 -0
  57. data/spec/dummy/public/robots.txt +0 -1
  58. data/spec/dummy/test/application_system_test_case.rb +0 -6
  59. data/spec/dummy/test/system/action_cable_subscription_test.rb +0 -45
  60. data/spec/dummy/test/test_helper.rb +0 -4
  61. data/spec/fixtures/upgrader/account.original.rb +0 -19
  62. data/spec/fixtures/upgrader/account.transformed.rb +0 -20
  63. data/spec/fixtures/upgrader/blame_range.original.rb +0 -43
  64. data/spec/fixtures/upgrader/blame_range.transformed.rb +0 -30
  65. data/spec/fixtures/upgrader/date_time.original.rb +0 -24
  66. data/spec/fixtures/upgrader/date_time.transformed.rb +0 -23
  67. data/spec/fixtures/upgrader/delete_project.original.rb +0 -28
  68. data/spec/fixtures/upgrader/delete_project.transformed.rb +0 -27
  69. data/spec/fixtures/upgrader/gist_order_field.original.rb +0 -14
  70. data/spec/fixtures/upgrader/gist_order_field.transformed.rb +0 -13
  71. data/spec/fixtures/upgrader/increment_count.original.rb +0 -59
  72. data/spec/fixtures/upgrader/increment_count.transformed.rb +0 -50
  73. data/spec/fixtures/upgrader/mutation.original.rb +0 -28
  74. data/spec/fixtures/upgrader/mutation.transformed.rb +0 -28
  75. data/spec/fixtures/upgrader/photo.original.rb +0 -10
  76. data/spec/fixtures/upgrader/photo.transformed.rb +0 -12
  77. data/spec/fixtures/upgrader/release_order.original.rb +0 -15
  78. data/spec/fixtures/upgrader/release_order.transformed.rb +0 -14
  79. data/spec/fixtures/upgrader/starrable.original.rb +0 -49
  80. data/spec/fixtures/upgrader/starrable.transformed.rb +0 -46
  81. data/spec/fixtures/upgrader/subscribable.original.rb +0 -55
  82. data/spec/fixtures/upgrader/subscribable.transformed.rb +0 -51
  83. data/spec/fixtures/upgrader/type_x.original.rb +0 -65
  84. data/spec/fixtures/upgrader/type_x.transformed.rb +0 -56
  85. data/spec/graphql/analysis/analyze_query_spec.rb +0 -261
  86. data/spec/graphql/analysis/ast/field_usage_spec.rb +0 -51
  87. data/spec/graphql/analysis/ast/max_query_complexity_spec.rb +0 -120
  88. data/spec/graphql/analysis/ast/max_query_depth_spec.rb +0 -134
  89. data/spec/graphql/analysis/ast/query_complexity_spec.rb +0 -299
  90. data/spec/graphql/analysis/ast/query_depth_spec.rb +0 -108
  91. data/spec/graphql/analysis/ast_spec.rb +0 -296
  92. data/spec/graphql/analysis/field_usage_spec.rb +0 -62
  93. data/spec/graphql/analysis/max_query_complexity_spec.rb +0 -102
  94. data/spec/graphql/analysis/max_query_depth_spec.rb +0 -103
  95. data/spec/graphql/analysis/query_complexity_spec.rb +0 -301
  96. data/spec/graphql/analysis/query_depth_spec.rb +0 -81
  97. data/spec/graphql/argument_spec.rb +0 -159
  98. data/spec/graphql/authorization_spec.rb +0 -974
  99. data/spec/graphql/backtrace_spec.rb +0 -206
  100. data/spec/graphql/base_type_spec.rb +0 -171
  101. data/spec/graphql/boolean_type_spec.rb +0 -21
  102. data/spec/graphql/compatibility/execution_specification_spec.rb +0 -4
  103. data/spec/graphql/compatibility/lazy_execution_specification_spec.rb +0 -4
  104. data/spec/graphql/compatibility/query_parser_specification_spec.rb +0 -6
  105. data/spec/graphql/compatibility/schema_parser_specification_spec.rb +0 -6
  106. data/spec/graphql/define/assign_argument_spec.rb +0 -61
  107. data/spec/graphql/define/instance_definable_spec.rb +0 -203
  108. data/spec/graphql/directive/skip_directive_spec.rb +0 -9
  109. data/spec/graphql/directive_spec.rb +0 -295
  110. data/spec/graphql/enum_type_spec.rb +0 -158
  111. data/spec/graphql/execution/execute_spec.rb +0 -303
  112. data/spec/graphql/execution/instrumentation_spec.rb +0 -212
  113. data/spec/graphql/execution/interpreter_spec.rb +0 -485
  114. data/spec/graphql/execution/lazy/lazy_method_map_spec.rb +0 -57
  115. data/spec/graphql/execution/lazy_spec.rb +0 -247
  116. data/spec/graphql/execution/lookahead_spec.rb +0 -390
  117. data/spec/graphql/execution/multiplex_spec.rb +0 -222
  118. data/spec/graphql/execution/typecast_spec.rb +0 -47
  119. data/spec/graphql/execution_error_spec.rb +0 -329
  120. data/spec/graphql/field_spec.rb +0 -246
  121. data/spec/graphql/float_type_spec.rb +0 -16
  122. data/spec/graphql/function_spec.rb +0 -152
  123. data/spec/graphql/id_type_spec.rb +0 -33
  124. data/spec/graphql/input_object_type_spec.rb +0 -25
  125. data/spec/graphql/int_type_spec.rb +0 -35
  126. data/spec/graphql/interface_type_spec.rb +0 -196
  127. data/spec/graphql/internal_representation/print_spec.rb +0 -41
  128. data/spec/graphql/internal_representation/rewrite_spec.rb +0 -381
  129. data/spec/graphql/introspection/directive_type_spec.rb +0 -66
  130. data/spec/graphql/introspection/input_value_type_spec.rb +0 -144
  131. data/spec/graphql/introspection/introspection_query_spec.rb +0 -64
  132. data/spec/graphql/introspection/schema_type_spec.rb +0 -57
  133. data/spec/graphql/introspection/type_type_spec.rb +0 -155
  134. data/spec/graphql/language/block_string_spec.rb +0 -70
  135. data/spec/graphql/language/definition_slice_spec.rb +0 -226
  136. data/spec/graphql/language/document_from_schema_definition_spec.rb +0 -770
  137. data/spec/graphql/language/equality_spec.rb +0 -84
  138. data/spec/graphql/language/generation_spec.rb +0 -38
  139. data/spec/graphql/language/lexer_spec.rb +0 -153
  140. data/spec/graphql/language/nodes_spec.rb +0 -67
  141. data/spec/graphql/language/parser_spec.rb +0 -183
  142. data/spec/graphql/language/printer_spec.rb +0 -215
  143. data/spec/graphql/language/visitor_spec.rb +0 -419
  144. data/spec/graphql/list_type_spec.rb +0 -57
  145. data/spec/graphql/non_null_type_spec.rb +0 -48
  146. data/spec/graphql/object_type_spec.rb +0 -197
  147. data/spec/graphql/query/arguments_spec.rb +0 -346
  148. data/spec/graphql/query/context_spec.rb +0 -292
  149. data/spec/graphql/query/executor_spec.rb +0 -341
  150. data/spec/graphql/query/literal_input_spec.rb +0 -91
  151. data/spec/graphql/query/result_spec.rb +0 -29
  152. data/spec/graphql/query/serial_execution/value_resolution_spec.rb +0 -109
  153. data/spec/graphql/query_spec.rb +0 -803
  154. data/spec/graphql/rake_task_spec.rb +0 -59
  155. data/spec/graphql/scalar_type_spec.rb +0 -66
  156. data/spec/graphql/schema/argument_spec.rb +0 -186
  157. data/spec/graphql/schema/build_from_definition_spec.rb +0 -1197
  158. data/spec/graphql/schema/catchall_middleware_spec.rb +0 -32
  159. data/spec/graphql/schema/directive/feature_spec.rb +0 -81
  160. data/spec/graphql/schema/directive/transform_spec.rb +0 -39
  161. data/spec/graphql/schema/enum_spec.rb +0 -83
  162. data/spec/graphql/schema/enum_value_spec.rb +0 -24
  163. data/spec/graphql/schema/field_extension_spec.rb +0 -159
  164. data/spec/graphql/schema/field_spec.rb +0 -319
  165. data/spec/graphql/schema/finder_spec.rb +0 -135
  166. data/spec/graphql/schema/input_object_spec.rb +0 -421
  167. data/spec/graphql/schema/instrumentation_spec.rb +0 -43
  168. data/spec/graphql/schema/interface_spec.rb +0 -215
  169. data/spec/graphql/schema/introspection_system_spec.rb +0 -80
  170. data/spec/graphql/schema/list_spec.rb +0 -73
  171. data/spec/graphql/schema/loader_spec.rb +0 -350
  172. data/spec/graphql/schema/member/accepts_definition_spec.rb +0 -115
  173. data/spec/graphql/schema/member/build_type_spec.rb +0 -63
  174. data/spec/graphql/schema/member/scoped_spec.rb +0 -217
  175. data/spec/graphql/schema/member/type_system_helpers_spec.rb +0 -63
  176. data/spec/graphql/schema/middleware_chain_spec.rb +0 -57
  177. data/spec/graphql/schema/mutation_spec.rb +0 -150
  178. data/spec/graphql/schema/non_null_spec.rb +0 -46
  179. data/spec/graphql/schema/object_spec.rb +0 -355
  180. data/spec/graphql/schema/printer_spec.rb +0 -883
  181. data/spec/graphql/schema/relay_classic_mutation_spec.rb +0 -252
  182. data/spec/graphql/schema/rescue_middleware_spec.rb +0 -88
  183. data/spec/graphql/schema/resolver_spec.rb +0 -743
  184. data/spec/graphql/schema/scalar_spec.rb +0 -101
  185. data/spec/graphql/schema/subscription_spec.rb +0 -505
  186. data/spec/graphql/schema/timeout_middleware_spec.rb +0 -188
  187. data/spec/graphql/schema/timeout_spec.rb +0 -206
  188. data/spec/graphql/schema/traversal_spec.rb +0 -222
  189. data/spec/graphql/schema/type_expression_spec.rb +0 -39
  190. data/spec/graphql/schema/union_spec.rb +0 -72
  191. data/spec/graphql/schema/unique_within_type_spec.rb +0 -44
  192. data/spec/graphql/schema/validation_spec.rb +0 -355
  193. data/spec/graphql/schema/warden_spec.rb +0 -926
  194. data/spec/graphql/schema_spec.rb +0 -169
  195. data/spec/graphql/static_validation/rules/argument_literals_are_compatible_spec.rb +0 -466
  196. data/spec/graphql/static_validation/rules/argument_names_are_unique_spec.rb +0 -44
  197. data/spec/graphql/static_validation/rules/arguments_are_defined_spec.rb +0 -112
  198. data/spec/graphql/static_validation/rules/directives_are_defined_spec.rb +0 -35
  199. data/spec/graphql/static_validation/rules/directives_are_in_valid_locations_spec.rb +0 -42
  200. data/spec/graphql/static_validation/rules/fields_are_defined_on_type_spec.rb +0 -167
  201. data/spec/graphql/static_validation/rules/fields_have_appropriate_selections_spec.rb +0 -66
  202. data/spec/graphql/static_validation/rules/fields_will_merge_spec.rb +0 -740
  203. data/spec/graphql/static_validation/rules/fragment_names_are_unique_spec.rb +0 -28
  204. data/spec/graphql/static_validation/rules/fragment_spreads_are_possible_spec.rb +0 -52
  205. data/spec/graphql/static_validation/rules/fragment_types_exist_spec.rb +0 -42
  206. data/spec/graphql/static_validation/rules/fragments_are_finite_spec.rb +0 -123
  207. data/spec/graphql/static_validation/rules/fragments_are_named_spec.rb +0 -24
  208. data/spec/graphql/static_validation/rules/fragments_are_on_composite_types_spec.rb +0 -56
  209. data/spec/graphql/static_validation/rules/fragments_are_used_spec.rb +0 -61
  210. data/spec/graphql/static_validation/rules/mutation_root_exists_spec.rb +0 -39
  211. data/spec/graphql/static_validation/rules/no_definitions_are_present_spec.rb +0 -62
  212. data/spec/graphql/static_validation/rules/operation_names_are_valid_spec.rb +0 -82
  213. data/spec/graphql/static_validation/rules/required_arguments_are_present_spec.rb +0 -98
  214. data/spec/graphql/static_validation/rules/required_input_object_attributes_are_present_spec.rb +0 -86
  215. data/spec/graphql/static_validation/rules/subscription_root_exists_spec.rb +0 -34
  216. data/spec/graphql/static_validation/rules/unique_directives_per_location_spec.rb +0 -188
  217. data/spec/graphql/static_validation/rules/variable_default_values_are_correctly_typed_spec.rb +0 -196
  218. data/spec/graphql/static_validation/rules/variable_names_are_unique_spec.rb +0 -23
  219. data/spec/graphql/static_validation/rules/variable_usages_are_allowed_spec.rb +0 -236
  220. data/spec/graphql/static_validation/rules/variables_are_input_types_spec.rb +0 -78
  221. data/spec/graphql/static_validation/rules/variables_are_used_and_defined_spec.rb +0 -81
  222. data/spec/graphql/static_validation/type_stack_spec.rb +0 -29
  223. data/spec/graphql/static_validation/validator_spec.rb +0 -204
  224. data/spec/graphql/string_type_spec.rb +0 -80
  225. data/spec/graphql/subscriptions/serialize_spec.rb +0 -49
  226. data/spec/graphql/subscriptions_spec.rb +0 -540
  227. data/spec/graphql/tracing/new_relic_tracing_spec.rb +0 -84
  228. data/spec/graphql/tracing/platform_tracing_spec.rb +0 -141
  229. data/spec/graphql/tracing/prometheus_tracing_spec.rb +0 -42
  230. data/spec/graphql/tracing/scout_tracing_spec.rb +0 -17
  231. data/spec/graphql/tracing/skylight_tracing_spec.rb +0 -63
  232. data/spec/graphql/tracing_spec.rb +0 -52
  233. data/spec/graphql/types/big_int_spec.rb +0 -24
  234. data/spec/graphql/types/iso_8601_date_time_spec.rb +0 -137
  235. data/spec/graphql/types/relay/base_edge_spec.rb +0 -33
  236. data/spec/graphql/union_type_spec.rb +0 -211
  237. data/spec/graphql/upgrader/member_spec.rb +0 -583
  238. data/spec/graphql/upgrader/schema_spec.rb +0 -82
  239. data/spec/integration/mongoid/graphql/relay/mongo_relation_connection_spec.rb +0 -528
  240. data/spec/integration/mongoid/spec_helper.rb +0 -2
  241. data/spec/integration/mongoid/star_trek/data.rb +0 -126
  242. data/spec/integration/mongoid/star_trek/schema.rb +0 -424
  243. data/spec/integration/rails/data.rb +0 -110
  244. data/spec/integration/rails/generators/base_generator_test.rb +0 -7
  245. data/spec/integration/rails/generators/graphql/enum_generator_spec.rb +0 -30
  246. data/spec/integration/rails/generators/graphql/install_generator_spec.rb +0 -238
  247. data/spec/integration/rails/generators/graphql/interface_generator_spec.rb +0 -34
  248. data/spec/integration/rails/generators/graphql/loader_generator_spec.rb +0 -59
  249. data/spec/integration/rails/generators/graphql/mutation_generator_spec.rb +0 -71
  250. data/spec/integration/rails/generators/graphql/object_generator_spec.rb +0 -54
  251. data/spec/integration/rails/generators/graphql/scalar_generator_spec.rb +0 -28
  252. data/spec/integration/rails/generators/graphql/union_generator_spec.rb +0 -67
  253. data/spec/integration/rails/graphql/input_object_spec.rb +0 -19
  254. data/spec/integration/rails/graphql/input_object_type_spec.rb +0 -364
  255. data/spec/integration/rails/graphql/query/variables_spec.rb +0 -375
  256. data/spec/integration/rails/graphql/relay/array_connection_spec.rb +0 -309
  257. data/spec/integration/rails/graphql/relay/base_connection_spec.rb +0 -101
  258. data/spec/integration/rails/graphql/relay/connection_instrumentation_spec.rb +0 -80
  259. data/spec/integration/rails/graphql/relay/connection_resolve_spec.rb +0 -79
  260. data/spec/integration/rails/graphql/relay/connection_type_spec.rb +0 -106
  261. data/spec/integration/rails/graphql/relay/edge_spec.rb +0 -10
  262. data/spec/integration/rails/graphql/relay/mutation_spec.rb +0 -387
  263. data/spec/integration/rails/graphql/relay/node_spec.rb +0 -263
  264. data/spec/integration/rails/graphql/relay/page_info_spec.rb +0 -111
  265. data/spec/integration/rails/graphql/relay/range_add_spec.rb +0 -117
  266. data/spec/integration/rails/graphql/relay/relation_connection_spec.rb +0 -837
  267. data/spec/integration/rails/graphql/schema_spec.rb +0 -507
  268. data/spec/integration/rails/graphql/tracing/active_support_notifications_tracing_spec.rb +0 -62
  269. data/spec/integration/rails/spec_helper.rb +0 -25
  270. data/spec/spec_helper.rb +0 -116
  271. data/spec/support/dummy/data.rb +0 -45
  272. data/spec/support/dummy/schema.rb +0 -519
  273. data/spec/support/error_bubbling_helpers.rb +0 -23
  274. data/spec/support/global_id.rb +0 -23
  275. data/spec/support/jazz.rb +0 -778
  276. data/spec/support/lazy_helpers.rb +0 -192
  277. data/spec/support/magic_cards/schema.graphql +0 -33
  278. data/spec/support/minimum_input_object.rb +0 -21
  279. data/spec/support/new_relic.rb +0 -27
  280. data/spec/support/parser/filename_example.graphql +0 -5
  281. data/spec/support/parser/filename_example_error_1.graphql +0 -4
  282. data/spec/support/parser/filename_example_error_2.graphql +0 -5
  283. data/spec/support/parser/filename_example_invalid_utf8.graphql +0 -1
  284. data/spec/support/skylight.rb +0 -39
  285. data/spec/support/star_wars/schema.rb +0 -464
  286. data/spec/support/static_validation_helpers.rb +0 -32
@@ -1,81 +0,0 @@
1
- # frozen_string_literal: true
2
- require "spec_helper"
3
-
4
- describe GraphQL::Analysis::QueryDepth do
5
- let(:depths) { [] }
6
- let(:query_depth) { GraphQL::Analysis::QueryDepth.new { |query, max_depth| depths << query << max_depth } }
7
- let(:reduce_result) { GraphQL::Analysis.analyze_query(query, [query_depth]) }
8
- let(:query) { GraphQL::Query.new(Dummy::Schema, query_string, variables: variables) }
9
- let(:variables) { {} }
10
-
11
- describe "simple queries" do
12
- let(:query_string) {%|
13
- query cheeses($isIncluded: Boolean = true){
14
- # depth of 2
15
- cheese1: cheese(id: 1) {
16
- id
17
- flavor
18
- }
19
-
20
- # depth of 4
21
- cheese2: cheese(id: 2) @include(if: $isIncluded) {
22
- similarCheese(source: SHEEP) {
23
- ... on Cheese {
24
- similarCheese(source: SHEEP) {
25
- id
26
- }
27
- }
28
- }
29
- }
30
- }
31
- |}
32
-
33
- it "finds the max depth" do
34
- reduce_result
35
- assert_equal depths, [query, 4]
36
- end
37
-
38
- describe "with directives" do
39
- let(:variables) { { "isIncluded" => false } }
40
- it "doesn't count skipped fields" do
41
- reduce_result
42
- assert_equal depths.last, 2
43
- end
44
- end
45
- end
46
-
47
- describe "query with fragments" do
48
- let(:query_string) {%|
49
- {
50
- # depth of 2
51
- cheese1: cheese(id: 1) {
52
- id
53
- flavor
54
- }
55
-
56
- # depth of 4
57
- cheese2: cheese(id: 2) {
58
- ... cheeseFields1
59
- }
60
- }
61
-
62
- fragment cheeseFields1 on Cheese {
63
- similarCheese(source: COW) {
64
- id
65
- ... cheeseFields2
66
- }
67
- }
68
-
69
- fragment cheeseFields2 on Cheese {
70
- similarCheese(source: SHEEP) {
71
- id
72
- }
73
- }
74
- |}
75
-
76
- it "finds the max depth" do
77
- reduce_result
78
- assert_equal depths, [query, 4]
79
- end
80
- end
81
- end
@@ -1,159 +0,0 @@
1
- # frozen_string_literal: true
2
- require "spec_helper"
3
-
4
- describe GraphQL::Argument do
5
- it "is validated at schema build-time" do
6
- query_type = GraphQL::ObjectType.define do
7
- name "Query"
8
- field :invalid, types.Boolean do
9
- argument :invalid, types.Float, default_value: ["123"]
10
- end
11
- end
12
-
13
- err = assert_raises(GraphQL::Schema::InvalidTypeError) {
14
- schema = GraphQL::Schema.define(query: query_type)
15
- schema.types
16
- }
17
-
18
- expected_error = %|Query is invalid: field "invalid" argument "invalid" default value ["123"] is not valid for type Float|
19
- assert_includes err.message, expected_error
20
- end
21
-
22
- describe ".from_dsl" do
23
- it "accepts an existing argument" do
24
- existing = GraphQL::Argument.define do
25
- name "bar"
26
- type GraphQL::STRING_TYPE
27
- end
28
-
29
- arg = GraphQL::Argument.from_dsl(:foo, existing)
30
-
31
- assert_equal "foo", arg.name
32
- assert_equal GraphQL::STRING_TYPE, arg.type
33
- end
34
-
35
- it "accepts a definition block after defining kwargs" do
36
- arg = GraphQL::Argument.from_dsl(:foo, GraphQL::STRING_TYPE) do
37
- description "my type is #{target.type}"
38
- end
39
-
40
- assert_equal "my type is String", arg.description
41
- end
42
-
43
- it "accepts a definition block and yields the argument if the block has an arity of one" do
44
- arg = GraphQL::Argument.from_dsl(:foo, GraphQL::STRING_TYPE) do |argument|
45
- argument.description "my type is #{target.type}"
46
- end
47
-
48
- assert_equal "my type is String", arg.description
49
- end
50
-
51
- it "accepts a definition block with existing arg" do
52
- existing = GraphQL::Argument.define do
53
- name "bar"
54
- type GraphQL::STRING_TYPE
55
- end
56
-
57
- arg = GraphQL::Argument.from_dsl(:foo, existing) do
58
- description "Description for an existing field."
59
- end
60
-
61
- assert_equal "Description for an existing field.", arg.description
62
- end
63
-
64
- it "creates an argument from dsl arguments" do
65
- arg = GraphQL::Argument.from_dsl(
66
- :foo,
67
- GraphQL::STRING_TYPE,
68
- "A Description",
69
- default_value: "Bar"
70
- )
71
-
72
- assert_equal "foo", arg.name
73
- assert_equal GraphQL::STRING_TYPE, arg.type
74
- assert_equal "A Description", arg.description
75
- assert_equal "Bar", arg.default_value
76
- end
77
- end
78
-
79
- it "accepts custom keywords" do
80
- type = GraphQL::ObjectType.define do
81
- name "Something"
82
- field :something, types.String do
83
- argument "flagged", types.Int, metadata_flag: :flag_1
84
- end
85
- end
86
-
87
- arg = type.fields["something"].arguments["flagged"]
88
- assert_equal true, arg.metadata[:flag_1]
89
- end
90
-
91
- it "accepts proc type" do
92
- argument = GraphQL::Argument.define(name: :favoriteFood, type: -> { GraphQL::STRING_TYPE })
93
- assert_equal GraphQL::STRING_TYPE, argument.type
94
- end
95
-
96
- it "accepts a default_value" do
97
- argument = GraphQL::Argument.define(name: :favoriteFood, type: GraphQL::STRING_TYPE, default_value: 'Default')
98
- assert_equal 'Default', argument.default_value
99
- assert argument.default_value?
100
- end
101
-
102
- it "accepts a default_value of nil" do
103
- argument = GraphQL::Argument.define(name: :favoriteFood, type: GraphQL::STRING_TYPE, default_value: nil)
104
- assert argument.default_value.nil?
105
- assert argument.default_value?
106
- end
107
-
108
- it "default_value is optional" do
109
- argument = GraphQL::Argument.define(name: :favoriteFood, type: GraphQL::STRING_TYPE)
110
- assert argument.default_value.nil?
111
- assert !argument.default_value?
112
- end
113
-
114
- describe "#as, #exposed_as" do
115
- it "accepts a `as` property to define the arg name at resolve time" do
116
- argument = GraphQL::Argument.define(name: :favoriteFood, type: GraphQL::STRING_TYPE, as: :favFood)
117
- assert_equal argument.as, :favFood
118
- end
119
-
120
- it "uses `name` or `as` for `expose_as`" do
121
- arg_1 = GraphQL::Argument.define(name: :favoriteFood, type: GraphQL::STRING_TYPE, as: :favFood)
122
- assert_equal arg_1.expose_as, "favFood"
123
- arg_2 = GraphQL::Argument.define(name: :favoriteFood, type: GraphQL::STRING_TYPE)
124
- assert_equal arg_2.expose_as, "favoriteFood"
125
- arg_3 = arg_2.redefine { as :ff }
126
- assert_equal arg_3.expose_as, "ff"
127
- end
128
-
129
- it "can be set in the passed block" do
130
- argument = GraphQL::Argument.define do
131
- name "arg"
132
- as "arg_name"
133
- end
134
- assert_equal "arg_name", argument.as
135
- end
136
- end
137
-
138
- describe "prepare" do
139
- it "accepts a prepare proc and calls it to generate the prepared value" do
140
- prepare_proc = Proc.new { |arg, ctx| arg + ctx[:val] }
141
- argument = GraphQL::Argument.define(name: :plusOne, type: GraphQL::INT_TYPE, prepare: prepare_proc)
142
- assert_equal argument.prepare(1, {val: 1}), 2
143
- end
144
-
145
- it "returns the value itself if no prepare proc is provided" do
146
- argument = GraphQL::Argument.define(name: :someNumber, type: GraphQL::INT_TYPE)
147
- assert_equal argument.prepare(1, nil), 1
148
- end
149
-
150
- it "can be set in the passed block" do
151
- prepare_proc = Proc.new { |arg, ctx| arg + ctx[:val] }
152
- argument = GraphQL::Argument.define do
153
- name "arg"
154
- prepare prepare_proc
155
- end
156
- assert_equal argument.prepare(1, {val: 1}), 2
157
- end
158
- end
159
- end
@@ -1,974 +0,0 @@
1
- # frozen_string_literal: true
2
- require "spec_helper"
3
-
4
- describe GraphQL::Authorization do
5
- module AuthTest
6
- class Box
7
- attr_reader :value
8
- def initialize(value:)
9
- @value = value
10
- end
11
- end
12
-
13
- class BaseArgument < GraphQL::Schema::Argument
14
- def visible?(context)
15
- super && (context[:hide] ? @name != "hidden" : true)
16
- end
17
-
18
- def accessible?(context)
19
- super && (context[:hide] ? @name != "inaccessible" : true)
20
- end
21
-
22
- def authorized?(parent_object, context)
23
- super && parent_object != :hide2
24
- end
25
- end
26
-
27
- class BaseField < GraphQL::Schema::Field
28
- def initialize(*args, edge_class: nil, **kwargs, &block)
29
- @edge_class = edge_class
30
- super(*args, **kwargs, &block)
31
- end
32
-
33
- def to_graphql
34
- field_defn = super
35
- if @edge_class
36
- field_defn.edge_class = @edge_class
37
- end
38
- field_defn
39
- end
40
-
41
- argument_class BaseArgument
42
- def visible?(context)
43
- super && (context[:hide] ? @name != "hidden" : true)
44
- end
45
-
46
- def accessible?(context)
47
- super && (context[:hide] ? @name != "inaccessible" : true)
48
- end
49
-
50
- def authorized?(object, context)
51
- if object == :raise
52
- raise GraphQL::UnauthorizedFieldError.new("raised authorized field error", object: object)
53
- end
54
- super && object != :hide && object != :replace
55
- end
56
- end
57
-
58
- class BaseObject < GraphQL::Schema::Object
59
- field_class BaseField
60
- end
61
-
62
- module BaseInterface
63
- include GraphQL::Schema::Interface
64
- end
65
-
66
- class BaseEnumValue < GraphQL::Schema::EnumValue
67
- def initialize(*args, role: nil, **kwargs)
68
- @role = role
69
- super(*args, **kwargs)
70
- end
71
-
72
- def visible?(context)
73
- super && (context[:hide] ? @role != :hidden : true)
74
- end
75
- end
76
-
77
- class BaseEnum < GraphQL::Schema::Enum
78
- enum_value_class(BaseEnumValue)
79
- end
80
-
81
- module HiddenInterface
82
- include BaseInterface
83
-
84
- def self.visible?(ctx)
85
- super && !ctx[:hide]
86
- end
87
-
88
- def self.resolve_type(obj, ctx)
89
- HiddenObject
90
- end
91
- end
92
-
93
- module HiddenDefaultInterface
94
- include BaseInterface
95
- # visible? will call the super method
96
- def self.resolve_type(obj, ctx)
97
- HiddenObject
98
- end
99
- end
100
-
101
- class HiddenObject < BaseObject
102
- implements HiddenInterface
103
- implements HiddenDefaultInterface
104
- def self.visible?(ctx)
105
- super && !ctx[:hide]
106
- end
107
- end
108
-
109
- class RelayObject < BaseObject
110
- def self.visible?(ctx)
111
- super && !ctx[:hidden_relay]
112
- end
113
-
114
- def self.accessible?(ctx)
115
- super && !ctx[:inaccessible_relay]
116
- end
117
-
118
- def self.authorized?(_val, ctx)
119
- super && !ctx[:unauthorized_relay]
120
- end
121
- end
122
-
123
- # TODO test default behavior for abstract types,
124
- # that they check their concrete types
125
- module InaccessibleInterface
126
- include BaseInterface
127
-
128
- def self.accessible?(ctx)
129
- super && !ctx[:hide]
130
- end
131
-
132
- def self.resolve_type(obj, ctx)
133
- InaccessibleObject
134
- end
135
- end
136
-
137
- module InaccessibleDefaultInterface
138
- include BaseInterface
139
- # accessible? will call the super method
140
- def self.resolve_type(obj, ctx)
141
- InaccessibleObject
142
- end
143
- end
144
-
145
- class InaccessibleObject < BaseObject
146
- implements InaccessibleInterface
147
- implements InaccessibleDefaultInterface
148
- def self.accessible?(ctx)
149
- super && !ctx[:hide]
150
- end
151
- end
152
-
153
- class UnauthorizedObject < BaseObject
154
- def self.authorized?(value, context)
155
- if context[:raise]
156
- raise GraphQL::UnauthorizedError.new("raised authorized object error", object: value.object)
157
- end
158
- super && !context[:hide]
159
- end
160
-
161
- field :value, String, null: false, method: :itself
162
- end
163
-
164
- class UnauthorizedBox < BaseObject
165
- # Hide `"a"`
166
- def self.authorized?(value, context)
167
- super && value != "a"
168
- end
169
-
170
- field :value, String, null: false, method: :itself
171
- end
172
-
173
- module UnauthorizedInterface
174
- include BaseInterface
175
-
176
- def self.resolve_type(obj, ctx)
177
- if obj.is_a?(String)
178
- UnauthorizedCheckBox
179
- else
180
- raise "Unexpected value: #{obj.inspect}"
181
- end
182
- end
183
- end
184
-
185
- class UnauthorizedCheckBox < BaseObject
186
- implements UnauthorizedInterface
187
- # This authorized check returns a lazy object, it should be synced by the runtime.
188
- def self.authorized?(value, context)
189
- if !value.is_a?(String)
190
- raise "Unexpected box value: #{value.inspect}"
191
- end
192
- is_authed = super && value != "a"
193
- # Make it many levels nested just to make sure we support nested lazy objects
194
- Box.new(value: Box.new(value: Box.new(value: Box.new(value: is_authed))))
195
- end
196
-
197
- field :value, String, null: false, method: :itself
198
- end
199
-
200
- class IntegerObject < BaseObject
201
- def self.authorized?(obj, ctx)
202
- if !obj.is_a?(Integer)
203
- raise "Unexpected IntegerObject: #{obj}"
204
- end
205
- is_allowed = !(ctx[:unauthorized_relay] || obj == ctx[:exclude_integer])
206
- Box.new(value: Box.new(value: is_allowed))
207
- end
208
- field :value, Integer, null: false, method: :itself
209
- end
210
-
211
- class IntegerObjectEdge < GraphQL::Types::Relay::BaseEdge
212
- node_type(IntegerObject)
213
- end
214
-
215
- class IntegerObjectConnection < GraphQL::Types::Relay::BaseConnection
216
- edge_type(IntegerObjectEdge)
217
- end
218
-
219
- # This object responds with `replaced => false`,
220
- # but if its replacement value is used, it gives `replaced => true`
221
- class Replaceable
222
- def replacement
223
- { replaced: true }
224
- end
225
-
226
- def replaced
227
- false
228
- end
229
- end
230
-
231
- class ReplacedObject < BaseObject
232
- def self.authorized?(obj, ctx)
233
- super && !ctx[:replace_me]
234
- end
235
-
236
- field :replaced, Boolean, null: false
237
- end
238
-
239
- class LandscapeFeature < BaseEnum
240
- value "MOUNTAIN"
241
- value "STREAM", role: :unauthorized
242
- value "FIELD", role: :inaccessible
243
- value "TAR_PIT", role: :hidden
244
- end
245
-
246
- class Query < BaseObject
247
- def self.authorized?(obj, ctx)
248
- !ctx[:query_unauthorized]
249
- end
250
-
251
- field :hidden, Integer, null: false
252
- field :unauthorized, Integer, null: true, method: :itself
253
- field :int2, Integer, null: true do
254
- argument :int, Integer, required: false
255
- argument :hidden, Integer, required: false
256
- argument :inaccessible, Integer, required: false
257
- argument :unauthorized, Integer, required: false
258
- end
259
-
260
- def int2(**args)
261
- args[:unauthorized] || 1
262
- end
263
-
264
- field :landscape_feature, LandscapeFeature, null: false do
265
- argument :string, String, required: false
266
- argument :enum, LandscapeFeature, required: false
267
- end
268
-
269
- def landscape_feature(string: nil, enum: nil)
270
- string || enum
271
- end
272
-
273
- field :landscape_features, [LandscapeFeature], null: false do
274
- argument :strings, [String], required: false
275
- argument :enums, [LandscapeFeature], required: false
276
- end
277
-
278
- def landscape_features(strings: [], enums: [])
279
- strings + enums
280
- end
281
-
282
- def empty_array; []; end
283
- field :hidden_object, HiddenObject, null: false, resolver_method: :itself
284
- field :hidden_interface, HiddenInterface, null: false, resolver_method: :itself
285
- field :hidden_default_interface, HiddenDefaultInterface, null: false, resolver_method: :itself
286
- field :hidden_connection, RelayObject.connection_type, null: :false, resolver_method: :empty_array
287
- field :hidden_edge, RelayObject.edge_type, null: :false, resolver_method: :edge_object
288
-
289
- field :inaccessible, Integer, null: false, method: :object_id
290
- field :inaccessible_object, InaccessibleObject, null: false, resolver_method: :itself
291
- field :inaccessible_interface, InaccessibleInterface, null: false, resolver_method: :itself
292
- field :inaccessible_default_interface, InaccessibleDefaultInterface, null: false, resolver_method: :itself
293
- field :inaccessible_connection, RelayObject.connection_type, null: :false, resolver_method: :empty_array
294
- field :inaccessible_edge, RelayObject.edge_type, null: :false, resolver_method: :edge_object
295
-
296
- field :unauthorized_object, UnauthorizedObject, null: true, resolver_method: :itself
297
- field :unauthorized_connection, RelayObject.connection_type, null: false, resolver_method: :array_with_item
298
- field :unauthorized_edge, RelayObject.edge_type, null: false, resolver_method: :edge_object
299
-
300
- def edge_object
301
- OpenStruct.new(node: 100)
302
- end
303
-
304
- def array_with_item
305
- [1]
306
- end
307
-
308
- field :unauthorized_lazy_box, UnauthorizedBox, null: true do
309
- argument :value, String, required: true
310
- end
311
- def unauthorized_lazy_box(value:)
312
- # Make it extra nested, just for good measure.
313
- Box.new(value: Box.new(value: value))
314
- end
315
- field :unauthorized_list_items, [UnauthorizedObject], null: true
316
- def unauthorized_list_items
317
- [self, self]
318
- end
319
-
320
- field :unauthorized_lazy_check_box, UnauthorizedCheckBox, null: true, resolver_method: :unauthorized_lazy_box do
321
- argument :value, String, required: true
322
- end
323
-
324
- field :unauthorized_interface, UnauthorizedInterface, null: true, resolver_method: :unauthorized_lazy_box do
325
- argument :value, String, required: true
326
- end
327
-
328
- field :unauthorized_lazy_list_interface, [UnauthorizedInterface, null: true], null: true
329
-
330
- def unauthorized_lazy_list_interface
331
- ["z", Box.new(value: Box.new(value: "z2")), "a", Box.new(value: "a")]
332
- end
333
-
334
- field :integers, IntegerObjectConnection, null: false
335
-
336
- def integers
337
- [1,2,3]
338
- end
339
-
340
- field :lazy_integers, IntegerObjectConnection, null: false
341
-
342
- def lazy_integers
343
- Box.new(value: Box.new(value: [1,2,3]))
344
- end
345
-
346
- field :replaced_object, ReplacedObject, null: false
347
- def replaced_object
348
- Replaceable.new
349
- end
350
- end
351
-
352
- class DoHiddenStuff < GraphQL::Schema::RelayClassicMutation
353
- def self.visible?(ctx)
354
- super && (ctx[:hidden_mutation] ? false : true)
355
- end
356
- end
357
-
358
- class DoHiddenStuff2 < GraphQL::Schema::Mutation
359
- def self.visible?(ctx)
360
- super && !ctx[:hidden_mutation]
361
- end
362
- end
363
-
364
- class DoInaccessibleStuff < GraphQL::Schema::RelayClassicMutation
365
- def self.accessible?(ctx)
366
- super && (ctx[:inaccessible_mutation] ? false : true)
367
- end
368
- end
369
-
370
- class DoUnauthorizedStuff < GraphQL::Schema::RelayClassicMutation
371
- def self.authorized?(obj, ctx)
372
- super && (ctx[:unauthorized_mutation] ? false : true)
373
- end
374
- end
375
-
376
- class Mutation < BaseObject
377
- field :do_hidden_stuff, mutation: DoHiddenStuff
378
- field :do_hidden_stuff2, mutation: DoHiddenStuff2
379
- field :do_inaccessible_stuff, mutation: DoInaccessibleStuff
380
- field :do_unauthorized_stuff, mutation: DoUnauthorizedStuff
381
- end
382
-
383
- class Schema < GraphQL::Schema
384
- if TESTING_INTERPRETER
385
- use GraphQL::Execution::Interpreter
386
- end
387
- query(Query)
388
- mutation(Mutation)
389
-
390
- # Opt in to accessible? checks
391
- query_analyzer GraphQL::Authorization::Analyzer
392
-
393
- lazy_resolve(Box, :value)
394
-
395
- def self.unauthorized_object(err)
396
- if err.object.respond_to?(:replacement)
397
- err.object.replacement
398
- elsif err.object == :replace
399
- 33
400
- elsif err.object == :raise_from_object
401
- raise GraphQL::ExecutionError, err.message
402
- else
403
- raise GraphQL::ExecutionError, "Unauthorized #{err.type.graphql_name}: #{err.object.inspect}"
404
- end
405
- end
406
-
407
- # use GraphQL::Backtrace
408
- end
409
-
410
- class SchemaWithFieldHook < GraphQL::Schema
411
- if TESTING_INTERPRETER
412
- use GraphQL::Execution::Interpreter
413
- end
414
- query(Query)
415
-
416
- def self.unauthorized_field(err)
417
- if err.object == :replace
418
- 42
419
- elsif err.object == :raise
420
- raise GraphQL::ExecutionError, "#{err.message} in field #{err.field.name}"
421
- else
422
- raise GraphQL::ExecutionError, "Unauthorized field #{err.field.graphql_name} on #{err.type.graphql_name}: #{err.object}"
423
- end
424
- end
425
- end
426
- end
427
-
428
- def auth_execute(*args)
429
- AuthTest::Schema.execute(*args)
430
- end
431
-
432
- describe "applying the visible? method" do
433
- it "works in queries" do
434
- res = auth_execute(" { int int2 } ", context: { hide: true })
435
- assert_equal 1, res["errors"].size
436
- end
437
-
438
- it "applies return type visibility to fields" do
439
- error_queries = {
440
- "hiddenObject" => "{ hiddenObject { __typename } }",
441
- "hiddenInterface" => "{ hiddenInterface { __typename } }",
442
- "hiddenDefaultInterface" => "{ hiddenDefaultInterface { __typename } }",
443
- }
444
-
445
- error_queries.each do |name, q|
446
- hidden_res = auth_execute(q, context: { hide: true})
447
- assert_equal ["Field '#{name}' doesn't exist on type 'Query'"], hidden_res["errors"].map { |e| e["message"] }
448
-
449
- visible_res = auth_execute(q)
450
- # Both fields exist; the interface resolves to the object type, though
451
- assert_equal "HiddenObject", visible_res["data"][name]["__typename"]
452
- end
453
- end
454
-
455
- it "uses the mutation for derived fields, inputs and outputs" do
456
- query = "mutation { doHiddenStuff(input: {}) { __typename } }"
457
- res = auth_execute(query, context: { hidden_mutation: true })
458
- assert_equal ["Field 'doHiddenStuff' doesn't exist on type 'Mutation'"], res["errors"].map { |e| e["message"] }
459
-
460
- # `#resolve` isn't implemented, so this errors out:
461
- assert_raises NotImplementedError do
462
- auth_execute(query)
463
- end
464
-
465
- introspection_q = <<-GRAPHQL
466
- {
467
- t1: __type(name: "DoHiddenStuffInput") { name }
468
- t2: __type(name: "DoHiddenStuffPayload") { name }
469
- }
470
- GRAPHQL
471
- hidden_introspection_res = auth_execute(introspection_q, context: { hidden_mutation: true })
472
- assert_nil hidden_introspection_res["data"]["t1"]
473
- assert_nil hidden_introspection_res["data"]["t2"]
474
-
475
- visible_introspection_res = auth_execute(introspection_q)
476
- assert_equal "DoHiddenStuffInput", visible_introspection_res["data"]["t1"]["name"]
477
- assert_equal "DoHiddenStuffPayload", visible_introspection_res["data"]["t2"]["name"]
478
- end
479
-
480
- it "works with Schema::Mutation" do
481
- query = "mutation { doHiddenStuff2 { __typename } }"
482
- res = auth_execute(query, context: { hidden_mutation: true })
483
- assert_equal ["Field 'doHiddenStuff2' doesn't exist on type 'Mutation'"], res["errors"].map { |e| e["message"] }
484
-
485
- # `#resolve` isn't implemented, so this errors out:
486
- assert_raises NotImplementedError do
487
- auth_execute(query)
488
- end
489
- end
490
-
491
- it "uses the base type for edges and connections" do
492
- query = <<-GRAPHQL
493
- {
494
- hiddenConnection { __typename }
495
- hiddenEdge { __typename }
496
- }
497
- GRAPHQL
498
-
499
- hidden_res = auth_execute(query, context: { hidden_relay: true })
500
- assert_equal 2, hidden_res["errors"].size
501
-
502
- visible_res = auth_execute(query)
503
- assert_equal "RelayObjectConnection", visible_res["data"]["hiddenConnection"]["__typename"]
504
- assert_equal "RelayObjectEdge", visible_res["data"]["hiddenEdge"]["__typename"]
505
- end
506
-
507
- it "treats hidden enum values as non-existant, even in lists" do
508
- hidden_res_1 = auth_execute <<-GRAPHQL, context: { hide: true }
509
- {
510
- landscapeFeature(enum: TAR_PIT)
511
- }
512
- GRAPHQL
513
-
514
- assert_equal ["Argument 'enum' on Field 'landscapeFeature' has an invalid value. Expected type 'LandscapeFeature'."], hidden_res_1["errors"].map { |e| e["message"] }
515
-
516
- hidden_res_2 = auth_execute <<-GRAPHQL, context: { hide: true }
517
- {
518
- landscapeFeatures(enums: [STREAM, TAR_PIT])
519
- }
520
- GRAPHQL
521
-
522
- assert_equal ["Argument 'enums' on Field 'landscapeFeatures' has an invalid value. Expected type '[LandscapeFeature!]'."], hidden_res_2["errors"].map { |e| e["message"] }
523
-
524
- success_res = auth_execute <<-GRAPHQL, context: { hide: false }
525
- {
526
- landscapeFeature(enum: TAR_PIT)
527
- landscapeFeatures(enums: [STREAM, TAR_PIT])
528
- }
529
- GRAPHQL
530
-
531
- assert_equal "TAR_PIT", success_res["data"]["landscapeFeature"]
532
- assert_equal ["STREAM", "TAR_PIT"], success_res["data"]["landscapeFeatures"]
533
- end
534
-
535
- it "refuses to resolve to hidden enum values" do
536
- assert_raises(GraphQL::EnumType::UnresolvedValueError) do
537
- auth_execute <<-GRAPHQL, context: { hide: true }
538
- {
539
- landscapeFeature(string: "TAR_PIT")
540
- }
541
- GRAPHQL
542
- end
543
-
544
- assert_raises(GraphQL::EnumType::UnresolvedValueError) do
545
- auth_execute <<-GRAPHQL, context: { hide: true }
546
- {
547
- landscapeFeatures(strings: ["STREAM", "TAR_PIT"])
548
- }
549
- GRAPHQL
550
- end
551
- end
552
-
553
- it "works in introspection" do
554
- res = auth_execute <<-GRAPHQL, context: { hide: true, hidden_mutation: true }
555
- {
556
- query: __type(name: "Query") {
557
- fields {
558
- name
559
- args { name }
560
- }
561
- }
562
-
563
- hiddenObject: __type(name: "HiddenObject") { name }
564
- hiddenInterface: __type(name: "HiddenInterface") { name }
565
- landscapeFeatures: __type(name: "LandscapeFeature") { enumValues { name } }
566
- }
567
- GRAPHQL
568
- query_field_names = res["data"]["query"]["fields"].map { |f| f["name"] }
569
- refute_includes query_field_names, "int"
570
- int2_arg_names = res["data"]["query"]["fields"].find { |f| f["name"] == "int2" }["args"].map { |a| a["name"] }
571
- assert_equal ["int", "inaccessible", "unauthorized"], int2_arg_names
572
-
573
- assert_nil res["data"]["hiddenObject"]
574
- assert_nil res["data"]["hiddenInterface"]
575
-
576
- visible_landscape_features = res["data"]["landscapeFeatures"]["enumValues"].map { |v| v["name"] }
577
- assert_equal ["MOUNTAIN", "STREAM", "FIELD"], visible_landscape_features
578
- end
579
- end
580
-
581
- describe "applying the accessible? method" do
582
- it "works with fields and arguments" do
583
- queries = {
584
- "{ inaccessible }" => ["Some fields in this query are not accessible: inaccessible"],
585
- "{ int2(inaccessible: 1) }" => ["Some fields in this query are not accessible: int2"],
586
- }
587
-
588
- queries.each do |query_str, errors|
589
- res = auth_execute(query_str, context: { hide: true })
590
- assert_equal errors, res.fetch("errors").map { |e| e["message"] }
591
-
592
- res = auth_execute(query_str, context: { hide: false })
593
- refute res.key?("errors")
594
- end
595
- end
596
-
597
- it "works with return types" do
598
- queries = {
599
- "{ inaccessibleObject { __typename } }" => ["Some fields in this query are not accessible: inaccessibleObject"],
600
- "{ inaccessibleInterface { __typename } }" => ["Some fields in this query are not accessible: inaccessibleInterface"],
601
- "{ inaccessibleDefaultInterface { __typename } }" => ["Some fields in this query are not accessible: inaccessibleDefaultInterface"],
602
- }
603
-
604
- queries.each do |query_str, errors|
605
- res = auth_execute(query_str, context: { hide: true })
606
- assert_equal errors, res["errors"].map { |e| e["message"] }
607
-
608
- res = auth_execute(query_str, context: { hide: false })
609
- refute res.key?("errors")
610
- end
611
- end
612
-
613
- it "works with mutations" do
614
- query = "mutation { doInaccessibleStuff(input: {}) { __typename } }"
615
- res = auth_execute(query, context: { inaccessible_mutation: true })
616
- assert_equal ["Some fields in this query are not accessible: doInaccessibleStuff"], res["errors"].map { |e| e["message"] }
617
-
618
- assert_raises NotImplementedError do
619
- auth_execute(query)
620
- end
621
- end
622
-
623
- it "works with edges and connections" do
624
- query = <<-GRAPHQL
625
- {
626
- inaccessibleConnection { __typename }
627
- inaccessibleEdge { __typename }
628
- }
629
- GRAPHQL
630
-
631
- inaccessible_res = auth_execute(query, context: { inaccessible_relay: true })
632
- assert_equal ["Some fields in this query are not accessible: inaccessibleConnection, inaccessibleEdge"], inaccessible_res["errors"].map { |e| e["message"] }
633
-
634
- accessible_res = auth_execute(query)
635
- refute accessible_res.key?("errors")
636
- end
637
- end
638
-
639
- describe "applying the authorized? method" do
640
- it "halts on unauthorized objects, replacing the object with nil" do
641
- query = "{ unauthorizedObject { __typename } }"
642
- hidden_response = auth_execute(query, context: { hide: true })
643
- assert_nil hidden_response["data"].fetch("unauthorizedObject")
644
- visible_response = auth_execute(query, context: {})
645
- assert_equal({ "__typename" => "UnauthorizedObject" }, visible_response["data"]["unauthorizedObject"])
646
- end
647
-
648
- it "halts on unauthorized mutations" do
649
- query = "mutation { doUnauthorizedStuff(input: {}) { __typename } }"
650
- res = auth_execute(query, context: { unauthorized_mutation: true })
651
- assert_nil res["data"].fetch("doUnauthorizedStuff")
652
- assert_raises NotImplementedError do
653
- auth_execute(query)
654
- end
655
- end
656
-
657
- describe "field level authorization" do
658
- describe "unauthorized field" do
659
- describe "with an unauthorized field hook configured" do
660
- describe "when the hook returns a value" do
661
- it "replaces the response with the return value of the unauthorized field hook" do
662
- query = "{ unauthorized }"
663
- response = AuthTest::SchemaWithFieldHook.execute(query, root_value: :replace)
664
- assert_equal 42, response["data"].fetch("unauthorized")
665
- end
666
- end
667
-
668
- describe "when the field hook raises an error" do
669
- it "returns nil" do
670
- query = "{ unauthorized }"
671
- response = AuthTest::SchemaWithFieldHook.execute(query, root_value: :hide)
672
- assert_nil response["data"].fetch("unauthorized")
673
- end
674
-
675
- it "adds the error to the errors key" do
676
- query = "{ unauthorized }"
677
- response = AuthTest::SchemaWithFieldHook.execute(query, root_value: :hide)
678
- assert_equal ["Unauthorized field unauthorized on Query: hide"], response["errors"].map { |e| e["message"] }
679
- end
680
- end
681
-
682
- describe "when the field authorization raises an UnauthorizedFieldError" do
683
- it "receives the raised error" do
684
- query = "{ unauthorized }"
685
- response = AuthTest::SchemaWithFieldHook.execute(query, root_value: :raise)
686
- assert_equal ["raised authorized field error in field unauthorized"], response["errors"].map { |e| e["message"] }
687
- end
688
- end
689
- end
690
-
691
- describe "with an unauthorized field hook not configured" do
692
- describe "When the object hook replaces the field" do
693
- it "delegates to the unauthorized object hook, which replaces the object" do
694
- query = "{ unauthorized }"
695
- response = AuthTest::Schema.execute(query, root_value: :replace)
696
- assert_equal 33, response["data"].fetch("unauthorized")
697
- end
698
- end
699
- describe "When the object hook raises an error" do
700
- it "returns nil" do
701
- query = "{ unauthorized }"
702
- response = AuthTest::Schema.execute(query, root_value: :hide)
703
- assert_nil response["data"].fetch("unauthorized")
704
- end
705
-
706
- it "adds the error to the errors key" do
707
- query = "{ unauthorized }"
708
- response = AuthTest::Schema.execute(query, root_value: :hide)
709
- assert_equal ["Unauthorized Query: :hide"], response["errors"].map { |e| e["message"] }
710
- end
711
- end
712
- end
713
- end
714
-
715
- describe "authorized field" do
716
- it "returns the field data" do
717
- query = "{ unauthorized }"
718
- response = AuthTest::SchemaWithFieldHook.execute(query, root_value: 1)
719
- assert_equal 1, response["data"].fetch("unauthorized")
720
- end
721
- end
722
- end
723
-
724
- it "halts on unauthorized fields, using the parent object" do
725
- query = "{ unauthorized }"
726
- hidden_response = auth_execute(query, root_value: :hide)
727
- assert_nil hidden_response["data"].fetch("unauthorized")
728
- visible_response = auth_execute(query, root_value: 1)
729
- assert_equal 1, visible_response["data"]["unauthorized"]
730
- end
731
-
732
- it "halts on unauthorized arguments, using the parent object" do
733
- query = "{ int2(unauthorized: 5) }"
734
- hidden_response = auth_execute(query, root_value: :hide2)
735
- assert_nil hidden_response["data"].fetch("int2")
736
- visible_response = auth_execute(query)
737
- assert_equal 5, visible_response["data"]["int2"]
738
- end
739
-
740
- it "works with edges and connections" do
741
- query = <<-GRAPHQL
742
- {
743
- unauthorizedConnection {
744
- __typename
745
- edges {
746
- __typename
747
- node {
748
- __typename
749
- }
750
- }
751
- nodes {
752
- __typename
753
- }
754
- }
755
- unauthorizedEdge {
756
- __typename
757
- node {
758
- __typename
759
- }
760
- }
761
- }
762
- GRAPHQL
763
-
764
- unauthorized_res = auth_execute(query, context: { unauthorized_relay: true })
765
- conn = unauthorized_res["data"].fetch("unauthorizedConnection")
766
- assert_equal "RelayObjectConnection", conn.fetch("__typename")
767
- # This is tricky: the previous behavior was to replace the _whole_
768
- # list with `nil`. This was due to an implementation detail:
769
- # The list field's return value (an array of integers) was wrapped
770
- # _before_ returning, and during this wrapping, a cascading error
771
- # caused the entire field to be nilled out.
772
- #
773
- # In the interpreter, each list item is contained and the error doesn't propagate
774
- # up to the whole list.
775
- #
776
- # Originally, I thought that this was a _feature_ that obscured list entries.
777
- # But really, look at the test below: you don't get this "feature" if
778
- # you use `edges { node }`, so it can't be relied on in any way.
779
- #
780
- # All that to say, in the interpreter, `nodes` and `edges { node }` behave
781
- # the same.
782
- #
783
- # TODO revisit the docs for this.
784
- failed_nodes_value = TESTING_INTERPRETER ? [nil] : nil
785
- assert_equal failed_nodes_value, conn.fetch("nodes")
786
- assert_equal [{"node" => nil, "__typename" => "RelayObjectEdge"}], conn.fetch("edges")
787
-
788
- edge = unauthorized_res["data"].fetch("unauthorizedEdge")
789
- assert_nil edge.fetch("node")
790
- assert_equal "RelayObjectEdge", edge["__typename"]
791
-
792
- unauthorized_object_paths = [
793
- ["unauthorizedConnection", "edges", 0, "node"],
794
- TESTING_INTERPRETER ? ["unauthorizedConnection", "nodes", 0] : ["unauthorizedConnection", "nodes"],
795
- ["unauthorizedEdge", "node"]
796
- ]
797
-
798
- assert_equal unauthorized_object_paths, unauthorized_res["errors"].map { |e| e["path"] }
799
-
800
- authorized_res = auth_execute(query)
801
- conn = authorized_res["data"].fetch("unauthorizedConnection")
802
- assert_equal "RelayObjectConnection", conn.fetch("__typename")
803
- assert_equal [{"__typename"=>"RelayObject"}], conn.fetch("nodes")
804
- assert_equal [{"node" => {"__typename" => "RelayObject"}, "__typename" => "RelayObjectEdge"}], conn.fetch("edges")
805
-
806
- edge = authorized_res["data"].fetch("unauthorizedEdge")
807
- assert_equal "RelayObject", edge.fetch("node").fetch("__typename")
808
- assert_equal "RelayObjectEdge", edge["__typename"]
809
- end
810
-
811
- it "authorizes _after_ resolving lazy objects" do
812
- query = <<-GRAPHQL
813
- {
814
- a: unauthorizedLazyBox(value: "a") { value }
815
- b: unauthorizedLazyBox(value: "b") { value }
816
- }
817
- GRAPHQL
818
-
819
- unauthorized_res = auth_execute(query)
820
- assert_nil unauthorized_res["data"].fetch("a")
821
- assert_equal "b", unauthorized_res["data"]["b"]["value"]
822
- end
823
-
824
- it "authorizes items in a list" do
825
- query = <<-GRAPHQL
826
- {
827
- unauthorizedListItems { __typename }
828
- }
829
- GRAPHQL
830
-
831
- unauthorized_res = auth_execute(query, context: { hide: true })
832
-
833
- assert_nil unauthorized_res["data"]["unauthorizedListItems"]
834
- authorized_res = auth_execute(query, context: { hide: false })
835
- assert_equal 2, authorized_res["data"]["unauthorizedListItems"].size
836
- end
837
-
838
- it "syncs lazy objects from authorized? checks" do
839
- query = <<-GRAPHQL
840
- {
841
- a: unauthorizedLazyCheckBox(value: "a") { value }
842
- b: unauthorizedLazyCheckBox(value: "b") { value }
843
- }
844
- GRAPHQL
845
-
846
- unauthorized_res = auth_execute(query)
847
- assert_nil unauthorized_res["data"].fetch("a")
848
- assert_equal "b", unauthorized_res["data"]["b"]["value"]
849
- # Also, the custom handler was called:
850
- assert_equal ["Unauthorized UnauthorizedCheckBox: \"a\""], unauthorized_res["errors"].map { |e| e["message"] }
851
- end
852
-
853
- it "Works for lazy connections" do
854
- query = <<-GRAPHQL
855
- {
856
- lazyIntegers { edges { node { value } } }
857
- }
858
- GRAPHQL
859
- res = auth_execute(query)
860
- assert_equal [1,2,3], res["data"]["lazyIntegers"]["edges"].map { |e| e["node"]["value"] }
861
- end
862
-
863
- it "Works for eager connections" do
864
- query = <<-GRAPHQL
865
- {
866
- integers { edges { node { value } } }
867
- }
868
- GRAPHQL
869
- res = auth_execute(query)
870
- assert_equal [1,2,3], res["data"]["integers"]["edges"].map { |e| e["node"]["value"] }
871
- end
872
-
873
- it "filters out individual nodes by value" do
874
- query = <<-GRAPHQL
875
- {
876
- integers { edges { node { value } } }
877
- }
878
- GRAPHQL
879
- res = auth_execute(query, context: { exclude_integer: 1 })
880
- assert_equal [nil,2,3], res["data"]["integers"]["edges"].map { |e| e["node"] && e["node"]["value"] }
881
- assert_equal ["Unauthorized IntegerObject: 1"], res["errors"].map { |e| e["message"] }
882
- end
883
-
884
- it "works with lazy values / interfaces" do
885
- query = <<-GRAPHQL
886
- query($value: String!){
887
- unauthorizedInterface(value: $value) {
888
- ... on UnauthorizedCheckBox {
889
- value
890
- }
891
- }
892
- }
893
- GRAPHQL
894
-
895
- res = auth_execute(query, variables: { value: "a"})
896
- assert_nil res["data"]["unauthorizedInterface"]
897
-
898
- res2 = auth_execute(query, variables: { value: "b"})
899
- assert_equal "b", res2["data"]["unauthorizedInterface"]["value"]
900
- end
901
-
902
- it "works with lazy values / lists of interfaces" do
903
- query = <<-GRAPHQL
904
- {
905
- unauthorizedLazyListInterface {
906
- ... on UnauthorizedCheckBox {
907
- value
908
- }
909
- }
910
- }
911
- GRAPHQL
912
-
913
- res = auth_execute(query)
914
- # An error from two, values from the others
915
- assert_equal ["Unauthorized UnauthorizedCheckBox: \"a\"", "Unauthorized UnauthorizedCheckBox: \"a\""], res["errors"].map { |e| e["message"] }
916
- assert_equal [{"value" => "z"}, {"value" => "z2"}, nil, nil], res["data"]["unauthorizedLazyListInterface"]
917
- end
918
-
919
- describe "with an unauthorized field hook configured" do
920
- it "replaces objects from the unauthorized_object hook" do
921
- query = "{ replacedObject { replaced } }"
922
- res = auth_execute(query, context: { replace_me: true })
923
- assert_equal true, res["data"]["replacedObject"]["replaced"]
924
-
925
- res = auth_execute(query, context: { replace_me: false })
926
- assert_equal false, res["data"]["replacedObject"]["replaced"]
927
- end
928
-
929
- it "works when the query hook returns false and there's no root object" do
930
- query = "{ __typename }"
931
- res = auth_execute(query)
932
- assert_equal "Query", res["data"]["__typename"]
933
-
934
- unauth_res = auth_execute(query, context: { query_unauthorized: true })
935
- assert_nil unauth_res["data"]
936
- assert_equal [{"message"=>"Unauthorized Query: nil"}], unauth_res["errors"]
937
- end
938
-
939
- describe "when the object authorization raises an UnauthorizedFieldError" do
940
- it "receives the raised error" do
941
- query = "{ unauthorizedObject { value } }"
942
- response = auth_execute(query, context: { raise: true }, root_value: :raise_from_object)
943
- assert_equal ["raised authorized object error"], response["errors"].map { |e| e["message"] }
944
- end
945
- end
946
- end
947
- end
948
-
949
- describe "returning false" do
950
- class FalseSchema < GraphQL::Schema
951
- class Query < GraphQL::Schema::Object
952
- def self.authorized?(obj, ctx)
953
- false
954
- end
955
-
956
- field :int, Integer, null: false
957
-
958
- def int
959
- 1
960
- end
961
- end
962
- query(Query)
963
- if TESTING_INTERPRETER
964
- use GraphQL::Execution::Interpreter
965
- end
966
- end
967
-
968
- it "works out-of-the-box" do
969
- res = FalseSchema.execute("{ int }")
970
- assert_nil res.fetch("data")
971
- refute res.key?("errors")
972
- end
973
- end
974
- end