graphql_cody 1.13.0
Sign up to get free protection for your applications and to get access to all the features.
- checksums.yaml +7 -0
- data/.yardopts +5 -0
- data/MIT-LICENSE +20 -0
- data/lib/generators/graphql/core.rb +74 -0
- data/lib/generators/graphql/enum_generator.rb +33 -0
- data/lib/generators/graphql/install_generator.rb +190 -0
- data/lib/generators/graphql/interface_generator.rb +27 -0
- data/lib/generators/graphql/loader_generator.rb +21 -0
- data/lib/generators/graphql/mutation_generator.rb +55 -0
- data/lib/generators/graphql/object_generator.rb +79 -0
- data/lib/generators/graphql/relay.rb +63 -0
- data/lib/generators/graphql/relay_generator.rb +21 -0
- data/lib/generators/graphql/scalar_generator.rb +20 -0
- data/lib/generators/graphql/templates/base_argument.erb +6 -0
- data/lib/generators/graphql/templates/base_connection.erb +8 -0
- data/lib/generators/graphql/templates/base_edge.erb +8 -0
- data/lib/generators/graphql/templates/base_enum.erb +6 -0
- data/lib/generators/graphql/templates/base_field.erb +7 -0
- data/lib/generators/graphql/templates/base_input_object.erb +7 -0
- data/lib/generators/graphql/templates/base_interface.erb +9 -0
- data/lib/generators/graphql/templates/base_mutation.erb +10 -0
- data/lib/generators/graphql/templates/base_object.erb +7 -0
- data/lib/generators/graphql/templates/base_scalar.erb +6 -0
- data/lib/generators/graphql/templates/base_union.erb +6 -0
- data/lib/generators/graphql/templates/enum.erb +7 -0
- data/lib/generators/graphql/templates/graphql_controller.erb +52 -0
- data/lib/generators/graphql/templates/interface.erb +8 -0
- data/lib/generators/graphql/templates/loader.erb +19 -0
- data/lib/generators/graphql/templates/mutation.erb +16 -0
- data/lib/generators/graphql/templates/mutation_type.erb +12 -0
- data/lib/generators/graphql/templates/node_type.erb +9 -0
- data/lib/generators/graphql/templates/object.erb +8 -0
- data/lib/generators/graphql/templates/query_type.erb +15 -0
- data/lib/generators/graphql/templates/scalar.erb +15 -0
- data/lib/generators/graphql/templates/schema.erb +27 -0
- data/lib/generators/graphql/templates/union.erb +7 -0
- data/lib/generators/graphql/type_generator.rb +98 -0
- data/lib/generators/graphql/union_generator.rb +33 -0
- data/lib/graphql/analysis/analyze_query.rb +98 -0
- data/lib/graphql/analysis/ast/analyzer.rb +84 -0
- data/lib/graphql/analysis/ast/field_usage.rb +51 -0
- data/lib/graphql/analysis/ast/max_query_complexity.rb +23 -0
- data/lib/graphql/analysis/ast/max_query_depth.rb +22 -0
- data/lib/graphql/analysis/ast/query_complexity.rb +230 -0
- data/lib/graphql/analysis/ast/query_depth.rb +56 -0
- data/lib/graphql/analysis/ast/visitor.rb +268 -0
- data/lib/graphql/analysis/ast.rb +91 -0
- data/lib/graphql/analysis/field_usage.rb +45 -0
- data/lib/graphql/analysis/max_query_complexity.rb +26 -0
- data/lib/graphql/analysis/max_query_depth.rb +26 -0
- data/lib/graphql/analysis/query_complexity.rb +88 -0
- data/lib/graphql/analysis/query_depth.rb +43 -0
- data/lib/graphql/analysis/reducer_state.rb +48 -0
- data/lib/graphql/analysis.rb +9 -0
- data/lib/graphql/analysis_error.rb +5 -0
- data/lib/graphql/argument.rb +131 -0
- data/lib/graphql/authorization.rb +82 -0
- data/lib/graphql/backtrace/inspect_result.rb +50 -0
- data/lib/graphql/backtrace/legacy_tracer.rb +56 -0
- data/lib/graphql/backtrace/table.rb +159 -0
- data/lib/graphql/backtrace/traced_error.rb +54 -0
- data/lib/graphql/backtrace/tracer.rb +81 -0
- data/lib/graphql/backtrace.rb +64 -0
- data/lib/graphql/backwards_compatibility.rb +61 -0
- data/lib/graphql/base_type.rb +230 -0
- data/lib/graphql/boolean_type.rb +2 -0
- data/lib/graphql/coercion_error.rb +13 -0
- data/lib/graphql/compatibility/execution_specification/counter_schema.rb +53 -0
- data/lib/graphql/compatibility/execution_specification/specification_schema.rb +200 -0
- data/lib/graphql/compatibility/execution_specification.rb +436 -0
- data/lib/graphql/compatibility/lazy_execution_specification/lazy_schema.rb +111 -0
- data/lib/graphql/compatibility/lazy_execution_specification.rb +215 -0
- data/lib/graphql/compatibility/query_parser_specification/parse_error_specification.rb +87 -0
- data/lib/graphql/compatibility/query_parser_specification/query_assertions.rb +79 -0
- data/lib/graphql/compatibility/query_parser_specification.rb +266 -0
- data/lib/graphql/compatibility/schema_parser_specification.rb +682 -0
- data/lib/graphql/compatibility.rb +5 -0
- data/lib/graphql/dataloader/null_dataloader.rb +22 -0
- data/lib/graphql/dataloader/request.rb +19 -0
- data/lib/graphql/dataloader/request_all.rb +19 -0
- data/lib/graphql/dataloader/source.rb +155 -0
- data/lib/graphql/dataloader.rb +308 -0
- data/lib/graphql/define/assign_argument.rb +12 -0
- data/lib/graphql/define/assign_connection.rb +13 -0
- data/lib/graphql/define/assign_enum_value.rb +18 -0
- data/lib/graphql/define/assign_global_id_field.rb +11 -0
- data/lib/graphql/define/assign_mutation_function.rb +34 -0
- data/lib/graphql/define/assign_object_field.rb +42 -0
- data/lib/graphql/define/defined_object_proxy.rb +53 -0
- data/lib/graphql/define/instance_definable.rb +240 -0
- data/lib/graphql/define/no_definition_error.rb +7 -0
- data/lib/graphql/define/non_null_with_bang.rb +16 -0
- data/lib/graphql/define/type_definer.rb +31 -0
- data/lib/graphql/define.rb +31 -0
- data/lib/graphql/deprecated_dsl.rb +55 -0
- data/lib/graphql/deprecation.rb +9 -0
- data/lib/graphql/dig.rb +19 -0
- data/lib/graphql/directive/deprecated_directive.rb +2 -0
- data/lib/graphql/directive/include_directive.rb +2 -0
- data/lib/graphql/directive/skip_directive.rb +2 -0
- data/lib/graphql/directive.rb +107 -0
- data/lib/graphql/enum_type.rb +133 -0
- data/lib/graphql/execution/directive_checks.rb +37 -0
- data/lib/graphql/execution/errors.rb +163 -0
- data/lib/graphql/execution/execute.rb +333 -0
- data/lib/graphql/execution/flatten.rb +40 -0
- data/lib/graphql/execution/instrumentation.rb +92 -0
- data/lib/graphql/execution/interpreter/argument_value.rb +28 -0
- data/lib/graphql/execution/interpreter/arguments.rb +88 -0
- data/lib/graphql/execution/interpreter/arguments_cache.rb +103 -0
- data/lib/graphql/execution/interpreter/execution_errors.rb +29 -0
- data/lib/graphql/execution/interpreter/handles_raw_value.rb +18 -0
- data/lib/graphql/execution/interpreter/resolve.rb +70 -0
- data/lib/graphql/execution/interpreter/runtime.rb +949 -0
- data/lib/graphql/execution/interpreter.rb +122 -0
- data/lib/graphql/execution/lazy/lazy_method_map.rb +98 -0
- data/lib/graphql/execution/lazy/resolve.rb +91 -0
- data/lib/graphql/execution/lazy.rb +83 -0
- data/lib/graphql/execution/lookahead.rb +307 -0
- data/lib/graphql/execution/multiplex.rb +214 -0
- data/lib/graphql/execution/typecast.rb +50 -0
- data/lib/graphql/execution.rb +11 -0
- data/lib/graphql/execution_error.rb +58 -0
- data/lib/graphql/field/resolve.rb +59 -0
- data/lib/graphql/field.rb +226 -0
- data/lib/graphql/filter.rb +53 -0
- data/lib/graphql/float_type.rb +2 -0
- data/lib/graphql/function.rb +128 -0
- data/lib/graphql/id_type.rb +2 -0
- data/lib/graphql/input_object_type.rb +138 -0
- data/lib/graphql/int_type.rb +2 -0
- data/lib/graphql/integer_decoding_error.rb +17 -0
- data/lib/graphql/integer_encoding_error.rb +36 -0
- data/lib/graphql/interface_type.rb +72 -0
- data/lib/graphql/internal_representation/document.rb +27 -0
- data/lib/graphql/internal_representation/node.rb +206 -0
- data/lib/graphql/internal_representation/print.rb +51 -0
- data/lib/graphql/internal_representation/rewrite.rb +184 -0
- data/lib/graphql/internal_representation/scope.rb +88 -0
- data/lib/graphql/internal_representation/visit.rb +36 -0
- data/lib/graphql/internal_representation.rb +7 -0
- data/lib/graphql/introspection/base_object.rb +13 -0
- data/lib/graphql/introspection/directive_location_enum.rb +15 -0
- data/lib/graphql/introspection/directive_type.rb +29 -0
- data/lib/graphql/introspection/dynamic_fields.rb +17 -0
- data/lib/graphql/introspection/entry_points.rb +35 -0
- data/lib/graphql/introspection/enum_value_type.rb +23 -0
- data/lib/graphql/introspection/field_type.rb +28 -0
- data/lib/graphql/introspection/input_value_type.rb +67 -0
- data/lib/graphql/introspection/introspection_query.rb +7 -0
- data/lib/graphql/introspection/schema_type.rb +44 -0
- data/lib/graphql/introspection/type_kind_enum.rb +13 -0
- data/lib/graphql/introspection/type_type.rb +95 -0
- data/lib/graphql/introspection.rb +114 -0
- data/lib/graphql/invalid_name_error.rb +11 -0
- data/lib/graphql/invalid_null_error.rb +50 -0
- data/lib/graphql/language/block_string.rb +99 -0
- data/lib/graphql/language/cache.rb +37 -0
- data/lib/graphql/language/definition_slice.rb +41 -0
- data/lib/graphql/language/document_from_schema_definition.rb +347 -0
- data/lib/graphql/language/generation.rb +24 -0
- data/lib/graphql/language/lexer.rb +1467 -0
- data/lib/graphql/language/lexer.rl +258 -0
- data/lib/graphql/language/nodes.rb +707 -0
- data/lib/graphql/language/parser.rb +1974 -0
- data/lib/graphql/language/parser.y +544 -0
- data/lib/graphql/language/printer.rb +366 -0
- data/lib/graphql/language/sanitized_printer.rb +222 -0
- data/lib/graphql/language/token.rb +34 -0
- data/lib/graphql/language/visitor.rb +242 -0
- data/lib/graphql/language.rb +36 -0
- data/lib/graphql/list_type.rb +80 -0
- data/lib/graphql/load_application_object_failed_error.rb +22 -0
- data/lib/graphql/name_validator.rb +11 -0
- data/lib/graphql/non_null_type.rb +71 -0
- data/lib/graphql/object_type.rb +130 -0
- data/lib/graphql/pagination/active_record_relation_connection.rb +48 -0
- data/lib/graphql/pagination/array_connection.rb +77 -0
- data/lib/graphql/pagination/connection.rb +226 -0
- data/lib/graphql/pagination/connections.rb +160 -0
- data/lib/graphql/pagination/mongoid_relation_connection.rb +25 -0
- data/lib/graphql/pagination/relation_connection.rb +196 -0
- data/lib/graphql/pagination/sequel_dataset_connection.rb +28 -0
- data/lib/graphql/pagination.rb +6 -0
- data/lib/graphql/parse_error.rb +24 -0
- data/lib/graphql/query/arguments.rb +189 -0
- data/lib/graphql/query/arguments_cache.rb +24 -0
- data/lib/graphql/query/context.rb +371 -0
- data/lib/graphql/query/executor.rb +52 -0
- data/lib/graphql/query/fingerprint.rb +26 -0
- data/lib/graphql/query/input_validation_result.rb +43 -0
- data/lib/graphql/query/literal_input.rb +136 -0
- data/lib/graphql/query/null_context.rb +55 -0
- data/lib/graphql/query/result.rb +63 -0
- data/lib/graphql/query/serial_execution/field_resolution.rb +92 -0
- data/lib/graphql/query/serial_execution/operation_resolution.rb +19 -0
- data/lib/graphql/query/serial_execution/selection_resolution.rb +23 -0
- data/lib/graphql/query/serial_execution/value_resolution.rb +87 -0
- data/lib/graphql/query/serial_execution.rb +40 -0
- data/lib/graphql/query/validation_pipeline.rb +139 -0
- data/lib/graphql/query/variable_validation_error.rb +44 -0
- data/lib/graphql/query/variables.rb +78 -0
- data/lib/graphql/query.rb +454 -0
- data/lib/graphql/railtie.rb +117 -0
- data/lib/graphql/rake_task/validate.rb +63 -0
- data/lib/graphql/rake_task.rb +145 -0
- data/lib/graphql/relay/array_connection.rb +83 -0
- data/lib/graphql/relay/base_connection.rb +189 -0
- data/lib/graphql/relay/connection_instrumentation.rb +54 -0
- data/lib/graphql/relay/connection_resolve.rb +43 -0
- data/lib/graphql/relay/connection_type.rb +41 -0
- data/lib/graphql/relay/edge.rb +27 -0
- data/lib/graphql/relay/edge_type.rb +19 -0
- data/lib/graphql/relay/edges_instrumentation.rb +39 -0
- data/lib/graphql/relay/global_id_resolve.rb +18 -0
- data/lib/graphql/relay/mongo_relation_connection.rb +50 -0
- data/lib/graphql/relay/mutation/instrumentation.rb +23 -0
- data/lib/graphql/relay/mutation/resolve.rb +56 -0
- data/lib/graphql/relay/mutation/result.rb +38 -0
- data/lib/graphql/relay/mutation.rb +106 -0
- data/lib/graphql/relay/node.rb +39 -0
- data/lib/graphql/relay/page_info.rb +7 -0
- data/lib/graphql/relay/range_add.rb +59 -0
- data/lib/graphql/relay/relation_connection.rb +188 -0
- data/lib/graphql/relay/type_extensions.rb +32 -0
- data/lib/graphql/relay.rb +18 -0
- data/lib/graphql/rubocop/graphql/base_cop.rb +36 -0
- data/lib/graphql/rubocop/graphql/default_null_true.rb +43 -0
- data/lib/graphql/rubocop/graphql/default_required_true.rb +43 -0
- data/lib/graphql/rubocop.rb +4 -0
- data/lib/graphql/runtime_type_error.rb +5 -0
- data/lib/graphql/scalar_type.rb +91 -0
- data/lib/graphql/schema/addition.rb +247 -0
- data/lib/graphql/schema/argument.rb +383 -0
- data/lib/graphql/schema/base_64_bp.rb +26 -0
- data/lib/graphql/schema/base_64_encoder.rb +21 -0
- data/lib/graphql/schema/build_from_definition/resolve_map/default_resolve.rb +47 -0
- data/lib/graphql/schema/build_from_definition/resolve_map.rb +78 -0
- data/lib/graphql/schema/build_from_definition.rb +477 -0
- data/lib/graphql/schema/built_in_types.rb +12 -0
- data/lib/graphql/schema/catchall_middleware.rb +35 -0
- data/lib/graphql/schema/default_parse_error.rb +10 -0
- data/lib/graphql/schema/default_type_error.rb +17 -0
- data/lib/graphql/schema/directive/deprecated.rb +18 -0
- data/lib/graphql/schema/directive/feature.rb +66 -0
- data/lib/graphql/schema/directive/flagged.rb +57 -0
- data/lib/graphql/schema/directive/include.rb +25 -0
- data/lib/graphql/schema/directive/skip.rb +25 -0
- data/lib/graphql/schema/directive/transform.rb +60 -0
- data/lib/graphql/schema/directive.rb +210 -0
- data/lib/graphql/schema/enum.rb +193 -0
- data/lib/graphql/schema/enum_value.rb +97 -0
- data/lib/graphql/schema/field/connection_extension.rb +76 -0
- data/lib/graphql/schema/field/scope_extension.rb +22 -0
- data/lib/graphql/schema/field.rb +880 -0
- data/lib/graphql/schema/field_extension.rb +69 -0
- data/lib/graphql/schema/find_inherited_value.rb +36 -0
- data/lib/graphql/schema/finder.rb +155 -0
- data/lib/graphql/schema/input_object.rb +253 -0
- data/lib/graphql/schema/interface.rb +136 -0
- data/lib/graphql/schema/introspection_system.rb +169 -0
- data/lib/graphql/schema/invalid_type_error.rb +7 -0
- data/lib/graphql/schema/late_bound_type.rb +33 -0
- data/lib/graphql/schema/list.rb +75 -0
- data/lib/graphql/schema/loader.rb +226 -0
- data/lib/graphql/schema/member/accepts_definition.rb +159 -0
- data/lib/graphql/schema/member/base_dsl_methods.rb +129 -0
- data/lib/graphql/schema/member/build_type.rb +180 -0
- data/lib/graphql/schema/member/cached_graphql_definition.rb +31 -0
- data/lib/graphql/schema/member/graphql_type_names.rb +21 -0
- data/lib/graphql/schema/member/has_arguments.rb +332 -0
- data/lib/graphql/schema/member/has_ast_node.rb +20 -0
- data/lib/graphql/schema/member/has_deprecation_reason.rb +25 -0
- data/lib/graphql/schema/member/has_directives.rb +98 -0
- data/lib/graphql/schema/member/has_fields.rb +163 -0
- data/lib/graphql/schema/member/has_interfaces.rb +90 -0
- data/lib/graphql/schema/member/has_path.rb +25 -0
- data/lib/graphql/schema/member/has_unresolved_type_error.rb +15 -0
- data/lib/graphql/schema/member/has_validators.rb +31 -0
- data/lib/graphql/schema/member/instrumentation.rb +131 -0
- data/lib/graphql/schema/member/relay_shortcuts.rb +47 -0
- data/lib/graphql/schema/member/scoped.rb +21 -0
- data/lib/graphql/schema/member/type_system_helpers.rb +38 -0
- data/lib/graphql/schema/member/validates_input.rb +33 -0
- data/lib/graphql/schema/member.rb +161 -0
- data/lib/graphql/schema/middleware_chain.rb +82 -0
- data/lib/graphql/schema/mutation.rb +94 -0
- data/lib/graphql/schema/non_null.rb +67 -0
- data/lib/graphql/schema/null_mask.rb +11 -0
- data/lib/graphql/schema/object.rb +150 -0
- data/lib/graphql/schema/possible_types.rb +44 -0
- data/lib/graphql/schema/printer.rb +100 -0
- data/lib/graphql/schema/relay_classic_mutation.rb +160 -0
- data/lib/graphql/schema/rescue_middleware.rb +60 -0
- data/lib/graphql/schema/resolver/has_payload_type.rb +96 -0
- data/lib/graphql/schema/resolver.rb +397 -0
- data/lib/graphql/schema/scalar.rb +69 -0
- data/lib/graphql/schema/subscription.rb +155 -0
- data/lib/graphql/schema/timeout.rb +123 -0
- data/lib/graphql/schema/timeout_middleware.rb +88 -0
- data/lib/graphql/schema/traversal.rb +228 -0
- data/lib/graphql/schema/type_expression.rb +43 -0
- data/lib/graphql/schema/type_membership.rb +48 -0
- data/lib/graphql/schema/union.rb +95 -0
- data/lib/graphql/schema/unique_within_type.rb +34 -0
- data/lib/graphql/schema/validation.rb +313 -0
- data/lib/graphql/schema/validator/allow_blank_validator.rb +29 -0
- data/lib/graphql/schema/validator/allow_null_validator.rb +26 -0
- data/lib/graphql/schema/validator/exclusion_validator.rb +33 -0
- data/lib/graphql/schema/validator/format_validator.rb +48 -0
- data/lib/graphql/schema/validator/inclusion_validator.rb +35 -0
- data/lib/graphql/schema/validator/length_validator.rb +59 -0
- data/lib/graphql/schema/validator/numericality_validator.rb +82 -0
- data/lib/graphql/schema/validator/required_validator.rb +68 -0
- data/lib/graphql/schema/validator.rb +174 -0
- data/lib/graphql/schema/warden.rb +409 -0
- data/lib/graphql/schema/wrapper.rb +29 -0
- data/lib/graphql/schema.rb +1945 -0
- data/lib/graphql/static_validation/all_rules.rb +40 -0
- data/lib/graphql/static_validation/base_visitor.rb +217 -0
- data/lib/graphql/static_validation/default_visitor.rb +15 -0
- data/lib/graphql/static_validation/definition_dependencies.rb +198 -0
- data/lib/graphql/static_validation/error.rb +46 -0
- data/lib/graphql/static_validation/interpreter_visitor.rb +14 -0
- data/lib/graphql/static_validation/literal_validator.rb +139 -0
- data/lib/graphql/static_validation/no_validate_visitor.rb +10 -0
- data/lib/graphql/static_validation/rules/argument_literals_are_compatible.rb +66 -0
- data/lib/graphql/static_validation/rules/argument_literals_are_compatible_error.rb +48 -0
- data/lib/graphql/static_validation/rules/argument_names_are_unique.rb +31 -0
- data/lib/graphql/static_validation/rules/argument_names_are_unique_error.rb +30 -0
- data/lib/graphql/static_validation/rules/arguments_are_defined.rb +71 -0
- data/lib/graphql/static_validation/rules/arguments_are_defined_error.rb +37 -0
- data/lib/graphql/static_validation/rules/directives_are_defined.rb +23 -0
- data/lib/graphql/static_validation/rules/directives_are_defined_error.rb +29 -0
- data/lib/graphql/static_validation/rules/directives_are_in_valid_locations.rb +65 -0
- data/lib/graphql/static_validation/rules/directives_are_in_valid_locations_error.rb +31 -0
- data/lib/graphql/static_validation/rules/fields_are_defined_on_type.rb +30 -0
- data/lib/graphql/static_validation/rules/fields_are_defined_on_type_error.rb +32 -0
- data/lib/graphql/static_validation/rules/fields_have_appropriate_selections.rb +73 -0
- data/lib/graphql/static_validation/rules/fields_have_appropriate_selections_error.rb +31 -0
- data/lib/graphql/static_validation/rules/fields_will_merge.rb +418 -0
- data/lib/graphql/static_validation/rules/fields_will_merge_error.rb +53 -0
- data/lib/graphql/static_validation/rules/fragment_names_are_unique.rb +30 -0
- data/lib/graphql/static_validation/rules/fragment_names_are_unique_error.rb +29 -0
- data/lib/graphql/static_validation/rules/fragment_spreads_are_possible.rb +73 -0
- data/lib/graphql/static_validation/rules/fragment_spreads_are_possible_error.rb +35 -0
- data/lib/graphql/static_validation/rules/fragment_types_exist.rb +39 -0
- data/lib/graphql/static_validation/rules/fragment_types_exist_error.rb +29 -0
- data/lib/graphql/static_validation/rules/fragments_are_finite.rb +21 -0
- data/lib/graphql/static_validation/rules/fragments_are_finite_error.rb +29 -0
- data/lib/graphql/static_validation/rules/fragments_are_named.rb +16 -0
- data/lib/graphql/static_validation/rules/fragments_are_named_error.rb +26 -0
- data/lib/graphql/static_validation/rules/fragments_are_on_composite_types.rb +37 -0
- data/lib/graphql/static_validation/rules/fragments_are_on_composite_types_error.rb +30 -0
- data/lib/graphql/static_validation/rules/fragments_are_used.rb +32 -0
- data/lib/graphql/static_validation/rules/fragments_are_used_error.rb +29 -0
- data/lib/graphql/static_validation/rules/input_object_names_are_unique.rb +30 -0
- data/lib/graphql/static_validation/rules/input_object_names_are_unique_error.rb +30 -0
- data/lib/graphql/static_validation/rules/mutation_root_exists.rb +17 -0
- data/lib/graphql/static_validation/rules/mutation_root_exists_error.rb +26 -0
- data/lib/graphql/static_validation/rules/no_definitions_are_present.rb +41 -0
- data/lib/graphql/static_validation/rules/no_definitions_are_present_error.rb +25 -0
- data/lib/graphql/static_validation/rules/operation_names_are_valid.rb +36 -0
- data/lib/graphql/static_validation/rules/operation_names_are_valid_error.rb +28 -0
- data/lib/graphql/static_validation/rules/required_arguments_are_present.rb +37 -0
- data/lib/graphql/static_validation/rules/required_arguments_are_present_error.rb +35 -0
- data/lib/graphql/static_validation/rules/required_input_object_attributes_are_present.rb +59 -0
- data/lib/graphql/static_validation/rules/required_input_object_attributes_are_present_error.rb +35 -0
- data/lib/graphql/static_validation/rules/subscription_root_exists.rb +17 -0
- data/lib/graphql/static_validation/rules/subscription_root_exists_error.rb +26 -0
- data/lib/graphql/static_validation/rules/unique_directives_per_location.rb +50 -0
- data/lib/graphql/static_validation/rules/unique_directives_per_location_error.rb +29 -0
- data/lib/graphql/static_validation/rules/variable_default_values_are_correctly_typed.rb +46 -0
- data/lib/graphql/static_validation/rules/variable_default_values_are_correctly_typed_error.rb +39 -0
- data/lib/graphql/static_validation/rules/variable_names_are_unique.rb +24 -0
- data/lib/graphql/static_validation/rules/variable_names_are_unique_error.rb +29 -0
- data/lib/graphql/static_validation/rules/variable_usages_are_allowed.rb +153 -0
- data/lib/graphql/static_validation/rules/variable_usages_are_allowed_error.rb +38 -0
- data/lib/graphql/static_validation/rules/variables_are_input_types.rb +39 -0
- data/lib/graphql/static_validation/rules/variables_are_input_types_error.rb +32 -0
- data/lib/graphql/static_validation/rules/variables_are_used_and_defined.rb +155 -0
- data/lib/graphql/static_validation/rules/variables_are_used_and_defined_error.rb +37 -0
- data/lib/graphql/static_validation/type_stack.rb +216 -0
- data/lib/graphql/static_validation/validation_context.rb +49 -0
- data/lib/graphql/static_validation/validation_timeout_error.rb +25 -0
- data/lib/graphql/static_validation/validator.rb +96 -0
- data/lib/graphql/static_validation.rb +19 -0
- data/lib/graphql/string_encoding_error.rb +20 -0
- data/lib/graphql/string_type.rb +2 -0
- data/lib/graphql/subscriptions/action_cable_subscriptions.rb +245 -0
- data/lib/graphql/subscriptions/broadcast_analyzer.rb +81 -0
- data/lib/graphql/subscriptions/default_subscription_resolve_extension.rb +21 -0
- data/lib/graphql/subscriptions/event.rb +144 -0
- data/lib/graphql/subscriptions/instrumentation.rb +79 -0
- data/lib/graphql/subscriptions/serialize.rb +138 -0
- data/lib/graphql/subscriptions/subscription_root.rb +76 -0
- data/lib/graphql/subscriptions.rb +299 -0
- data/lib/graphql/tracing/active_support_notifications_tracing.rb +35 -0
- data/lib/graphql/tracing/appoptics_tracing.rb +173 -0
- data/lib/graphql/tracing/appsignal_tracing.rb +51 -0
- data/lib/graphql/tracing/data_dog_tracing.rb +76 -0
- data/lib/graphql/tracing/new_relic_tracing.rb +51 -0
- data/lib/graphql/tracing/platform_tracing.rb +139 -0
- data/lib/graphql/tracing/prometheus_tracing/graphql_collector.rb +32 -0
- data/lib/graphql/tracing/prometheus_tracing.rb +67 -0
- data/lib/graphql/tracing/scout_tracing.rb +54 -0
- data/lib/graphql/tracing/skylight_tracing.rb +70 -0
- data/lib/graphql/tracing/statsd_tracing.rb +42 -0
- data/lib/graphql/tracing.rb +95 -0
- data/lib/graphql/type_kinds.rb +77 -0
- data/lib/graphql/types/big_int.rb +23 -0
- data/lib/graphql/types/boolean.rb +18 -0
- data/lib/graphql/types/float.rb +19 -0
- data/lib/graphql/types/id.rb +24 -0
- data/lib/graphql/types/int.rb +36 -0
- data/lib/graphql/types/iso_8601_date.rb +34 -0
- data/lib/graphql/types/iso_8601_date_time.rb +65 -0
- data/lib/graphql/types/json.rb +25 -0
- data/lib/graphql/types/relay/base_connection.rb +39 -0
- data/lib/graphql/types/relay/base_edge.rb +29 -0
- data/lib/graphql/types/relay/connection_behaviors.rb +156 -0
- data/lib/graphql/types/relay/default_relay.rb +27 -0
- data/lib/graphql/types/relay/edge_behaviors.rb +53 -0
- data/lib/graphql/types/relay/has_node_field.rb +41 -0
- data/lib/graphql/types/relay/has_nodes_field.rb +41 -0
- data/lib/graphql/types/relay/node.rb +15 -0
- data/lib/graphql/types/relay/node_behaviors.rb +15 -0
- data/lib/graphql/types/relay/node_field.rb +25 -0
- data/lib/graphql/types/relay/nodes_field.rb +27 -0
- data/lib/graphql/types/relay/page_info.rb +11 -0
- data/lib/graphql/types/relay/page_info_behaviors.rb +25 -0
- data/lib/graphql/types/relay.rb +41 -0
- data/lib/graphql/types/string.rb +29 -0
- data/lib/graphql/types.rb +11 -0
- data/lib/graphql/unauthorized_error.rb +29 -0
- data/lib/graphql/unauthorized_field_error.rb +23 -0
- data/lib/graphql/union_type.rb +115 -0
- data/lib/graphql/unresolved_type_error.rb +35 -0
- data/lib/graphql/upgrader/member.rb +937 -0
- data/lib/graphql/upgrader/schema.rb +38 -0
- data/lib/graphql/version.rb +4 -0
- data/lib/graphql.rb +168 -0
- data/readme.md +49 -0
- metadata +714 -0
@@ -0,0 +1,418 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
|
3
|
+
# frozen_string_literal: true
|
4
|
+
module GraphQL
|
5
|
+
module StaticValidation
|
6
|
+
module FieldsWillMerge
|
7
|
+
# Validates that a selection set is valid if all fields (including spreading any
|
8
|
+
# fragments) either correspond to distinct response names or can be merged
|
9
|
+
# without ambiguity.
|
10
|
+
#
|
11
|
+
# Original Algorithm: https://github.com/graphql/graphql-js/blob/master/src/validation/rules/OverlappingFieldsCanBeMerged.js
|
12
|
+
NO_ARGS = {}.freeze
|
13
|
+
|
14
|
+
Field = Struct.new(:node, :definition, :owner_type, :parents)
|
15
|
+
FragmentSpread = Struct.new(:name, :parents)
|
16
|
+
|
17
|
+
def initialize(*)
|
18
|
+
super
|
19
|
+
@visited_fragments = {}
|
20
|
+
@compared_fragments = {}
|
21
|
+
@conflict_count = 0
|
22
|
+
end
|
23
|
+
|
24
|
+
def on_operation_definition(node, _parent)
|
25
|
+
setting_errors { conflicts_within_selection_set(node, type_definition) }
|
26
|
+
super
|
27
|
+
end
|
28
|
+
|
29
|
+
def on_field(node, _parent)
|
30
|
+
setting_errors { conflicts_within_selection_set(node, type_definition) }
|
31
|
+
super
|
32
|
+
end
|
33
|
+
|
34
|
+
private
|
35
|
+
|
36
|
+
def field_conflicts
|
37
|
+
@field_conflicts ||= Hash.new do |errors, field|
|
38
|
+
errors[field] = GraphQL::StaticValidation::FieldsWillMergeError.new(kind: :field, field_name: field)
|
39
|
+
end
|
40
|
+
end
|
41
|
+
|
42
|
+
def arg_conflicts
|
43
|
+
@arg_conflicts ||= Hash.new do |errors, field|
|
44
|
+
errors[field] = GraphQL::StaticValidation::FieldsWillMergeError.new(kind: :argument, field_name: field)
|
45
|
+
end
|
46
|
+
end
|
47
|
+
|
48
|
+
def setting_errors
|
49
|
+
@field_conflicts = nil
|
50
|
+
@arg_conflicts = nil
|
51
|
+
|
52
|
+
yield
|
53
|
+
|
54
|
+
field_conflicts.each_value { |error| add_error(error) }
|
55
|
+
arg_conflicts.each_value { |error| add_error(error) }
|
56
|
+
end
|
57
|
+
|
58
|
+
def conflicts_within_selection_set(node, parent_type)
|
59
|
+
return if parent_type.nil?
|
60
|
+
|
61
|
+
fields, fragment_spreads = fields_and_fragments_from_selection(node, owner_type: parent_type, parents: [])
|
62
|
+
|
63
|
+
# (A) Find find all conflicts "within" the fields of this selection set.
|
64
|
+
find_conflicts_within(fields)
|
65
|
+
|
66
|
+
fragment_spreads.each_with_index do |fragment_spread, i|
|
67
|
+
are_mutually_exclusive = mutually_exclusive?(
|
68
|
+
fragment_spread.parents,
|
69
|
+
[parent_type]
|
70
|
+
)
|
71
|
+
|
72
|
+
# (B) Then find conflicts between these fields and those represented by
|
73
|
+
# each spread fragment name found.
|
74
|
+
find_conflicts_between_fields_and_fragment(
|
75
|
+
fragment_spread,
|
76
|
+
fields,
|
77
|
+
mutually_exclusive: are_mutually_exclusive,
|
78
|
+
)
|
79
|
+
|
80
|
+
# (C) Then compare this fragment with all other fragments found in this
|
81
|
+
# selection set to collect conflicts between fragments spread together.
|
82
|
+
# This compares each item in the list of fragment names to every other
|
83
|
+
# item in that same list (except for itself).
|
84
|
+
fragment_spreads[i + 1..-1].each do |fragment_spread2|
|
85
|
+
are_mutually_exclusive = mutually_exclusive?(
|
86
|
+
fragment_spread.parents,
|
87
|
+
fragment_spread2.parents
|
88
|
+
)
|
89
|
+
|
90
|
+
find_conflicts_between_fragments(
|
91
|
+
fragment_spread,
|
92
|
+
fragment_spread2,
|
93
|
+
mutually_exclusive: are_mutually_exclusive,
|
94
|
+
)
|
95
|
+
end
|
96
|
+
end
|
97
|
+
end
|
98
|
+
|
99
|
+
def find_conflicts_between_fragments(fragment_spread1, fragment_spread2, mutually_exclusive:)
|
100
|
+
fragment_name1 = fragment_spread1.name
|
101
|
+
fragment_name2 = fragment_spread2.name
|
102
|
+
return if fragment_name1 == fragment_name2
|
103
|
+
|
104
|
+
cache_key = compared_fragments_key(
|
105
|
+
fragment_name1,
|
106
|
+
fragment_name2,
|
107
|
+
mutually_exclusive,
|
108
|
+
)
|
109
|
+
if @compared_fragments.key?(cache_key)
|
110
|
+
return
|
111
|
+
else
|
112
|
+
@compared_fragments[cache_key] = true
|
113
|
+
end
|
114
|
+
|
115
|
+
fragment1 = context.fragments[fragment_name1]
|
116
|
+
fragment2 = context.fragments[fragment_name2]
|
117
|
+
|
118
|
+
return if fragment1.nil? || fragment2.nil?
|
119
|
+
|
120
|
+
fragment_type1 = context.warden.get_type(fragment1.type.name)
|
121
|
+
fragment_type2 = context.warden.get_type(fragment2.type.name)
|
122
|
+
|
123
|
+
return if fragment_type1.nil? || fragment_type2.nil?
|
124
|
+
|
125
|
+
fragment_fields1, fragment_spreads1 = fields_and_fragments_from_selection(
|
126
|
+
fragment1,
|
127
|
+
owner_type: fragment_type1,
|
128
|
+
parents: [*fragment_spread1.parents, fragment_type1]
|
129
|
+
)
|
130
|
+
fragment_fields2, fragment_spreads2 = fields_and_fragments_from_selection(
|
131
|
+
fragment2,
|
132
|
+
owner_type: fragment_type1,
|
133
|
+
parents: [*fragment_spread2.parents, fragment_type2]
|
134
|
+
)
|
135
|
+
|
136
|
+
# (F) First, find all conflicts between these two collections of fields
|
137
|
+
# (not including any nested fragments).
|
138
|
+
find_conflicts_between(
|
139
|
+
fragment_fields1,
|
140
|
+
fragment_fields2,
|
141
|
+
mutually_exclusive: mutually_exclusive,
|
142
|
+
)
|
143
|
+
|
144
|
+
# (G) Then collect conflicts between the first fragment and any nested
|
145
|
+
# fragments spread in the second fragment.
|
146
|
+
fragment_spreads2.each do |fragment_spread|
|
147
|
+
find_conflicts_between_fragments(
|
148
|
+
fragment_spread1,
|
149
|
+
fragment_spread,
|
150
|
+
mutually_exclusive: mutually_exclusive,
|
151
|
+
)
|
152
|
+
end
|
153
|
+
|
154
|
+
# (G) Then collect conflicts between the first fragment and any nested
|
155
|
+
# fragments spread in the second fragment.
|
156
|
+
fragment_spreads1.each do |fragment_spread|
|
157
|
+
find_conflicts_between_fragments(
|
158
|
+
fragment_spread2,
|
159
|
+
fragment_spread,
|
160
|
+
mutually_exclusive: mutually_exclusive,
|
161
|
+
)
|
162
|
+
end
|
163
|
+
end
|
164
|
+
|
165
|
+
def find_conflicts_between_fields_and_fragment(fragment_spread, fields, mutually_exclusive:)
|
166
|
+
fragment_name = fragment_spread.name
|
167
|
+
return if @visited_fragments.key?(fragment_name)
|
168
|
+
@visited_fragments[fragment_name] = true
|
169
|
+
|
170
|
+
fragment = context.fragments[fragment_name]
|
171
|
+
return if fragment.nil?
|
172
|
+
|
173
|
+
fragment_type = context.warden.get_type(fragment.type.name)
|
174
|
+
return if fragment_type.nil?
|
175
|
+
|
176
|
+
fragment_fields, fragment_spreads = fields_and_fragments_from_selection(fragment, owner_type: fragment_type, parents: [*fragment_spread.parents, fragment_type])
|
177
|
+
|
178
|
+
# (D) First find any conflicts between the provided collection of fields
|
179
|
+
# and the collection of fields represented by the given fragment.
|
180
|
+
find_conflicts_between(
|
181
|
+
fields,
|
182
|
+
fragment_fields,
|
183
|
+
mutually_exclusive: mutually_exclusive,
|
184
|
+
)
|
185
|
+
|
186
|
+
# (E) Then collect any conflicts between the provided collection of fields
|
187
|
+
# and any fragment names found in the given fragment.
|
188
|
+
fragment_spreads.each do |fragment_spread|
|
189
|
+
find_conflicts_between_fields_and_fragment(
|
190
|
+
fragment_spread,
|
191
|
+
fields,
|
192
|
+
mutually_exclusive: mutually_exclusive,
|
193
|
+
)
|
194
|
+
end
|
195
|
+
end
|
196
|
+
|
197
|
+
def find_conflicts_within(response_keys)
|
198
|
+
response_keys.each do |key, fields|
|
199
|
+
next if fields.size < 2
|
200
|
+
# find conflicts within nodes
|
201
|
+
for i in 0..fields.size - 1
|
202
|
+
for j in i + 1..fields.size - 1
|
203
|
+
find_conflict(key, fields[i], fields[j])
|
204
|
+
end
|
205
|
+
end
|
206
|
+
end
|
207
|
+
end
|
208
|
+
|
209
|
+
def find_conflict(response_key, field1, field2, mutually_exclusive: false)
|
210
|
+
return if @conflict_count >= context.max_errors
|
211
|
+
|
212
|
+
node1 = field1.node
|
213
|
+
node2 = field2.node
|
214
|
+
|
215
|
+
are_mutually_exclusive = mutually_exclusive ||
|
216
|
+
mutually_exclusive?(field1.parents, field2.parents)
|
217
|
+
|
218
|
+
if !are_mutually_exclusive
|
219
|
+
if node1.name != node2.name
|
220
|
+
conflict = field_conflicts[response_key]
|
221
|
+
|
222
|
+
conflict.add_conflict(node1, node1.name)
|
223
|
+
conflict.add_conflict(node2, node2.name)
|
224
|
+
|
225
|
+
@conflict_count += 1
|
226
|
+
end
|
227
|
+
|
228
|
+
if !same_arguments?(node1, node2)
|
229
|
+
conflict = arg_conflicts[response_key]
|
230
|
+
|
231
|
+
conflict.add_conflict(node1, GraphQL::Language.serialize(serialize_field_args(node1)))
|
232
|
+
conflict.add_conflict(node2, GraphQL::Language.serialize(serialize_field_args(node2)))
|
233
|
+
|
234
|
+
@conflict_count += 1
|
235
|
+
end
|
236
|
+
end
|
237
|
+
|
238
|
+
find_conflicts_between_sub_selection_sets(
|
239
|
+
field1,
|
240
|
+
field2,
|
241
|
+
mutually_exclusive: are_mutually_exclusive,
|
242
|
+
)
|
243
|
+
end
|
244
|
+
|
245
|
+
def find_conflicts_between_sub_selection_sets(field1, field2, mutually_exclusive:)
|
246
|
+
return if field1.definition.nil? || field2.definition.nil?
|
247
|
+
|
248
|
+
return_type1 = field1.definition.type.unwrap
|
249
|
+
return_type2 = field2.definition.type.unwrap
|
250
|
+
parents1 = [return_type1]
|
251
|
+
parents2 = [return_type2]
|
252
|
+
|
253
|
+
fields, fragment_spreads = fields_and_fragments_from_selection(
|
254
|
+
field1.node,
|
255
|
+
owner_type: return_type1,
|
256
|
+
parents: parents1
|
257
|
+
)
|
258
|
+
|
259
|
+
fields2, fragment_spreads2 = fields_and_fragments_from_selection(
|
260
|
+
field2.node,
|
261
|
+
owner_type: return_type2,
|
262
|
+
parents: parents2
|
263
|
+
)
|
264
|
+
|
265
|
+
# (H) First, collect all conflicts between these two collections of field.
|
266
|
+
find_conflicts_between(fields, fields2, mutually_exclusive: mutually_exclusive)
|
267
|
+
|
268
|
+
# (I) Then collect conflicts between the first collection of fields and
|
269
|
+
# those referenced by each fragment name associated with the second.
|
270
|
+
fragment_spreads2.each do |fragment_spread|
|
271
|
+
find_conflicts_between_fields_and_fragment(
|
272
|
+
fragment_spread,
|
273
|
+
fields,
|
274
|
+
mutually_exclusive: mutually_exclusive,
|
275
|
+
)
|
276
|
+
end
|
277
|
+
|
278
|
+
# (I) Then collect conflicts between the second collection of fields and
|
279
|
+
# those referenced by each fragment name associated with the first.
|
280
|
+
fragment_spreads.each do |fragment_spread|
|
281
|
+
find_conflicts_between_fields_and_fragment(
|
282
|
+
fragment_spread,
|
283
|
+
fields2,
|
284
|
+
mutually_exclusive: mutually_exclusive,
|
285
|
+
)
|
286
|
+
end
|
287
|
+
|
288
|
+
# (J) Also collect conflicts between any fragment names by the first and
|
289
|
+
# fragment names by the second. This compares each item in the first set of
|
290
|
+
# names to each item in the second set of names.
|
291
|
+
fragment_spreads.each do |frag1|
|
292
|
+
fragment_spreads2.each do |frag2|
|
293
|
+
find_conflicts_between_fragments(
|
294
|
+
frag1,
|
295
|
+
frag2,
|
296
|
+
mutually_exclusive: mutually_exclusive
|
297
|
+
)
|
298
|
+
end
|
299
|
+
end
|
300
|
+
end
|
301
|
+
|
302
|
+
def find_conflicts_between(response_keys, response_keys2, mutually_exclusive:)
|
303
|
+
response_keys.each do |key, fields|
|
304
|
+
fields2 = response_keys2[key]
|
305
|
+
if fields2
|
306
|
+
fields.each do |field|
|
307
|
+
fields2.each do |field2|
|
308
|
+
find_conflict(
|
309
|
+
key,
|
310
|
+
field,
|
311
|
+
field2,
|
312
|
+
mutually_exclusive: mutually_exclusive,
|
313
|
+
)
|
314
|
+
end
|
315
|
+
end
|
316
|
+
end
|
317
|
+
end
|
318
|
+
end
|
319
|
+
|
320
|
+
NO_SELECTIONS = [{}.freeze, [].freeze].freeze
|
321
|
+
|
322
|
+
def fields_and_fragments_from_selection(node, owner_type:, parents:)
|
323
|
+
if node.selections.empty?
|
324
|
+
NO_SELECTIONS
|
325
|
+
else
|
326
|
+
fields, fragment_spreads = find_fields_and_fragments(node.selections, owner_type: owner_type, parents: parents, fields: [], fragment_spreads: [])
|
327
|
+
response_keys = fields.group_by { |f| f.node.alias || f.node.name }
|
328
|
+
[response_keys, fragment_spreads]
|
329
|
+
end
|
330
|
+
end
|
331
|
+
|
332
|
+
def find_fields_and_fragments(selections, owner_type:, parents:, fields:, fragment_spreads:)
|
333
|
+
selections.each do |node|
|
334
|
+
case node
|
335
|
+
when GraphQL::Language::Nodes::Field
|
336
|
+
definition = context.query.get_field(owner_type, node.name)
|
337
|
+
fields << Field.new(node, definition, owner_type, parents)
|
338
|
+
when GraphQL::Language::Nodes::InlineFragment
|
339
|
+
fragment_type = node.type ? context.warden.get_type(node.type.name) : owner_type
|
340
|
+
find_fields_and_fragments(node.selections, parents: [*parents, fragment_type], owner_type: owner_type, fields: fields, fragment_spreads: fragment_spreads) if fragment_type
|
341
|
+
when GraphQL::Language::Nodes::FragmentSpread
|
342
|
+
fragment_spreads << FragmentSpread.new(node.name, parents)
|
343
|
+
end
|
344
|
+
end
|
345
|
+
|
346
|
+
[fields, fragment_spreads]
|
347
|
+
end
|
348
|
+
|
349
|
+
def same_arguments?(field1, field2)
|
350
|
+
# Check for incompatible / non-identical arguments on this node:
|
351
|
+
arguments1 = field1.arguments
|
352
|
+
arguments2 = field2.arguments
|
353
|
+
|
354
|
+
return false if arguments1.length != arguments2.length
|
355
|
+
|
356
|
+
arguments1.all? do |argument1|
|
357
|
+
argument2 = arguments2.find { |argument| argument.name == argument1.name }
|
358
|
+
return false if argument2.nil?
|
359
|
+
|
360
|
+
serialize_arg(argument1.value) == serialize_arg(argument2.value)
|
361
|
+
end
|
362
|
+
end
|
363
|
+
|
364
|
+
def serialize_arg(arg_value)
|
365
|
+
case arg_value
|
366
|
+
when GraphQL::Language::Nodes::AbstractNode
|
367
|
+
arg_value.to_query_string
|
368
|
+
when Array
|
369
|
+
"[#{arg_value.map { |a| serialize_arg(a) }.join(", ")}]"
|
370
|
+
else
|
371
|
+
GraphQL::Language.serialize(arg_value)
|
372
|
+
end
|
373
|
+
end
|
374
|
+
|
375
|
+
def serialize_field_args(field)
|
376
|
+
serialized_args = {}
|
377
|
+
field.arguments.each do |argument|
|
378
|
+
serialized_args[argument.name] = serialize_arg(argument.value)
|
379
|
+
end
|
380
|
+
serialized_args
|
381
|
+
end
|
382
|
+
|
383
|
+
def compared_fragments_key(frag1, frag2, exclusive)
|
384
|
+
# Cache key to not compare two fragments more than once.
|
385
|
+
# The key includes both fragment names sorted (this way we
|
386
|
+
# avoid computing "A vs B" and "B vs A"). It also includes
|
387
|
+
# "exclusive" since the result may change depending on the parent_type
|
388
|
+
"#{[frag1, frag2].sort.join('-')}-#{exclusive}"
|
389
|
+
end
|
390
|
+
|
391
|
+
# Given two list of parents, find out if they are mutually exclusive
|
392
|
+
# In this context, `parents` represends the "self scope" of the field,
|
393
|
+
# what types may be found at this point in the query.
|
394
|
+
def mutually_exclusive?(parents1, parents2)
|
395
|
+
if parents1.empty? || parents2.empty?
|
396
|
+
false
|
397
|
+
elsif parents1.length == parents2.length
|
398
|
+
parents1.length.times.any? do |i|
|
399
|
+
type1 = parents1[i - 1]
|
400
|
+
type2 = parents2[i - 1]
|
401
|
+
if type1 == type2
|
402
|
+
# If the types we're comparing are the same type,
|
403
|
+
# then they aren't mutually exclusive
|
404
|
+
false
|
405
|
+
else
|
406
|
+
# Check if these two scopes have _any_ types in common.
|
407
|
+
possible_right_types = context.query.possible_types(type1)
|
408
|
+
possible_left_types = context.query.possible_types(type2)
|
409
|
+
(possible_right_types & possible_left_types).empty?
|
410
|
+
end
|
411
|
+
end
|
412
|
+
else
|
413
|
+
true
|
414
|
+
end
|
415
|
+
end
|
416
|
+
end
|
417
|
+
end
|
418
|
+
end
|
@@ -0,0 +1,53 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
module GraphQL
|
3
|
+
module StaticValidation
|
4
|
+
class FieldsWillMergeError < StaticValidation::Error
|
5
|
+
attr_reader :field_name
|
6
|
+
attr_reader :kind
|
7
|
+
|
8
|
+
def initialize(kind:, field_name:)
|
9
|
+
super(nil)
|
10
|
+
|
11
|
+
@field_name = field_name
|
12
|
+
@kind = kind
|
13
|
+
@conflicts = []
|
14
|
+
end
|
15
|
+
|
16
|
+
def message
|
17
|
+
"Field '#{field_name}' has #{kind == :argument ? 'an' : 'a'} #{kind} conflict: #{conflicts}?"
|
18
|
+
end
|
19
|
+
|
20
|
+
def path
|
21
|
+
[]
|
22
|
+
end
|
23
|
+
|
24
|
+
def conflicts
|
25
|
+
@conflicts.join(' or ')
|
26
|
+
end
|
27
|
+
|
28
|
+
def add_conflict(node, conflict_str)
|
29
|
+
return if nodes.include?(node)
|
30
|
+
|
31
|
+
@nodes << node
|
32
|
+
@conflicts << conflict_str
|
33
|
+
end
|
34
|
+
|
35
|
+
# A hash representation of this Message
|
36
|
+
def to_h
|
37
|
+
extensions = {
|
38
|
+
"code" => code,
|
39
|
+
"fieldName" => field_name,
|
40
|
+
"conflicts" => conflicts
|
41
|
+
}
|
42
|
+
|
43
|
+
super.merge({
|
44
|
+
"extensions" => extensions
|
45
|
+
})
|
46
|
+
end
|
47
|
+
|
48
|
+
def code
|
49
|
+
"fieldConflict"
|
50
|
+
end
|
51
|
+
end
|
52
|
+
end
|
53
|
+
end
|
@@ -0,0 +1,30 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
module GraphQL
|
3
|
+
module StaticValidation
|
4
|
+
module FragmentNamesAreUnique
|
5
|
+
|
6
|
+
def initialize(*)
|
7
|
+
super
|
8
|
+
@fragments_by_name = Hash.new { |h, k| h[k] = [] }
|
9
|
+
end
|
10
|
+
|
11
|
+
def on_fragment_definition(node, parent)
|
12
|
+
@fragments_by_name[node.name] << node
|
13
|
+
super
|
14
|
+
end
|
15
|
+
|
16
|
+
def on_document(_n, _p)
|
17
|
+
super
|
18
|
+
@fragments_by_name.each do |name, fragments|
|
19
|
+
if fragments.length > 1
|
20
|
+
add_error(GraphQL::StaticValidation::FragmentNamesAreUniqueError.new(
|
21
|
+
%|Fragment name "#{name}" must be unique|,
|
22
|
+
nodes: fragments,
|
23
|
+
name: name
|
24
|
+
))
|
25
|
+
end
|
26
|
+
end
|
27
|
+
end
|
28
|
+
end
|
29
|
+
end
|
30
|
+
end
|
@@ -0,0 +1,29 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
module GraphQL
|
3
|
+
module StaticValidation
|
4
|
+
class FragmentNamesAreUniqueError < StaticValidation::Error
|
5
|
+
attr_reader :fragment_name
|
6
|
+
|
7
|
+
def initialize(message, path: nil, nodes: [], name:)
|
8
|
+
super(message, path: path, nodes: nodes)
|
9
|
+
@fragment_name = name
|
10
|
+
end
|
11
|
+
|
12
|
+
# A hash representation of this Message
|
13
|
+
def to_h
|
14
|
+
extensions = {
|
15
|
+
"code" => code,
|
16
|
+
"fragmentName" => fragment_name
|
17
|
+
}
|
18
|
+
|
19
|
+
super.merge({
|
20
|
+
"extensions" => extensions
|
21
|
+
})
|
22
|
+
end
|
23
|
+
|
24
|
+
def code
|
25
|
+
"fragmentNotUnique"
|
26
|
+
end
|
27
|
+
end
|
28
|
+
end
|
29
|
+
end
|
@@ -0,0 +1,73 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
module GraphQL
|
3
|
+
module StaticValidation
|
4
|
+
module FragmentSpreadsArePossible
|
5
|
+
def initialize(*)
|
6
|
+
super
|
7
|
+
@spreads_to_validate = []
|
8
|
+
end
|
9
|
+
|
10
|
+
def on_inline_fragment(node, parent)
|
11
|
+
fragment_parent = context.object_types[-2]
|
12
|
+
fragment_child = context.object_types.last
|
13
|
+
if fragment_child
|
14
|
+
validate_fragment_in_scope(fragment_parent, fragment_child, node, context, context.path)
|
15
|
+
end
|
16
|
+
super
|
17
|
+
end
|
18
|
+
|
19
|
+
def on_fragment_spread(node, parent)
|
20
|
+
fragment_parent = context.object_types.last
|
21
|
+
@spreads_to_validate << FragmentSpread.new(node: node, parent_type: fragment_parent, path: context.path)
|
22
|
+
super
|
23
|
+
end
|
24
|
+
|
25
|
+
def on_document(node, parent)
|
26
|
+
super
|
27
|
+
@spreads_to_validate.each do |frag_spread|
|
28
|
+
frag_node = context.fragments[frag_spread.node.name]
|
29
|
+
if frag_node
|
30
|
+
fragment_child_name = frag_node.type.name
|
31
|
+
fragment_child = context.warden.get_type(fragment_child_name)
|
32
|
+
# Might be non-existent type name
|
33
|
+
if fragment_child
|
34
|
+
validate_fragment_in_scope(frag_spread.parent_type, fragment_child, frag_spread.node, context, frag_spread.path)
|
35
|
+
end
|
36
|
+
end
|
37
|
+
end
|
38
|
+
end
|
39
|
+
|
40
|
+
private
|
41
|
+
|
42
|
+
def validate_fragment_in_scope(parent_type, child_type, node, context, path)
|
43
|
+
if !child_type.kind.fields?
|
44
|
+
# It's not a valid fragment type, this error was handled someplace else
|
45
|
+
return
|
46
|
+
end
|
47
|
+
parent_types = context.warden.possible_types(parent_type.unwrap)
|
48
|
+
child_types = context.warden.possible_types(child_type.unwrap)
|
49
|
+
|
50
|
+
if child_types.none? { |c| parent_types.include?(c) }
|
51
|
+
name = node.respond_to?(:name) ? " #{node.name}" : ""
|
52
|
+
add_error(GraphQL::StaticValidation::FragmentSpreadsArePossibleError.new(
|
53
|
+
"Fragment#{name} on #{child_type.graphql_name} can't be spread inside #{parent_type.graphql_name}",
|
54
|
+
nodes: node,
|
55
|
+
path: path,
|
56
|
+
fragment_name: name.empty? ? "unknown" : name,
|
57
|
+
type: child_type.graphql_name,
|
58
|
+
parent: parent_type.graphql_name
|
59
|
+
))
|
60
|
+
end
|
61
|
+
end
|
62
|
+
|
63
|
+
class FragmentSpread
|
64
|
+
attr_reader :node, :parent_type, :path
|
65
|
+
def initialize(node:, parent_type:, path:)
|
66
|
+
@node = node
|
67
|
+
@parent_type = parent_type
|
68
|
+
@path = path
|
69
|
+
end
|
70
|
+
end
|
71
|
+
end
|
72
|
+
end
|
73
|
+
end
|
@@ -0,0 +1,35 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
module GraphQL
|
3
|
+
module StaticValidation
|
4
|
+
class FragmentSpreadsArePossibleError < StaticValidation::Error
|
5
|
+
attr_reader :type_name
|
6
|
+
attr_reader :fragment_name
|
7
|
+
attr_reader :parent_name
|
8
|
+
|
9
|
+
def initialize(message, path: nil, nodes: [], type:, fragment_name:, parent:)
|
10
|
+
super(message, path: path, nodes: nodes)
|
11
|
+
@type_name = type
|
12
|
+
@fragment_name = fragment_name
|
13
|
+
@parent_name = parent
|
14
|
+
end
|
15
|
+
|
16
|
+
# A hash representation of this Message
|
17
|
+
def to_h
|
18
|
+
extensions = {
|
19
|
+
"code" => code,
|
20
|
+
"typeName" => type_name,
|
21
|
+
"fragmentName" => fragment_name,
|
22
|
+
"parentName" => parent_name
|
23
|
+
}
|
24
|
+
|
25
|
+
super.merge({
|
26
|
+
"extensions" => extensions
|
27
|
+
})
|
28
|
+
end
|
29
|
+
|
30
|
+
def code
|
31
|
+
"cannotSpreadFragment"
|
32
|
+
end
|
33
|
+
end
|
34
|
+
end
|
35
|
+
end
|
@@ -0,0 +1,39 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
module GraphQL
|
3
|
+
module StaticValidation
|
4
|
+
module FragmentTypesExist
|
5
|
+
def on_fragment_definition(node, _parent)
|
6
|
+
if validate_type_exists(node)
|
7
|
+
super
|
8
|
+
end
|
9
|
+
end
|
10
|
+
|
11
|
+
def on_inline_fragment(node, _parent)
|
12
|
+
if validate_type_exists(node)
|
13
|
+
super
|
14
|
+
end
|
15
|
+
end
|
16
|
+
|
17
|
+
private
|
18
|
+
|
19
|
+
def validate_type_exists(fragment_node)
|
20
|
+
if !fragment_node.type
|
21
|
+
true
|
22
|
+
else
|
23
|
+
type_name = fragment_node.type.name
|
24
|
+
type = context.warden.get_type(type_name)
|
25
|
+
if type.nil?
|
26
|
+
add_error(GraphQL::StaticValidation::FragmentTypesExistError.new(
|
27
|
+
"No such type #{type_name}, so it can't be a fragment condition",
|
28
|
+
nodes: fragment_node,
|
29
|
+
type: type_name
|
30
|
+
))
|
31
|
+
false
|
32
|
+
else
|
33
|
+
true
|
34
|
+
end
|
35
|
+
end
|
36
|
+
end
|
37
|
+
end
|
38
|
+
end
|
39
|
+
end
|