From 901267ab7ea92b62632fd337a7e29b9974e76605 Mon Sep 17 00:00:00 2001
From: Tobias Tengler <45513122+tobias-tengler@users.noreply.github.com>
Date: Sun, 17 Nov 2024 22:40:26 +0100
Subject: [PATCH] @semanticNonNull support (#7681)
---
.../Core/src/Abstractions/ErrorCodes.cs | 1 +
.../src/Abstractions/WellKnownDirectives.cs | 10 +
.../src/Abstractions/WellKnownMiddleware.cs | 5 +
.../Core/src/Types/IReadOnlySchemaOptions.cs | 7 +
.../Core/src/Types/SchemaBuilder.cs | 1 +
.../Core/src/Types/SchemaOptions.cs | 162 +--
.../Types/SemanticNonNullTypeInterceptor.cs | 376 ++++++
.../src/Types/Types/Directives/Directives.cs | 6 +
.../Directives/SemanticNonNullDirective.cs | 11 +
.../Execution.Tests/SemanticNonNullTests.cs | 1057 +++++++++++++++++
...s_Null_Should_Null_Item_Without_Error.snap | 13 +
...eturns_Null_Should_Null_Without_Error.snap | 5 +
...eturns_Null_Should_Null_Without_Error.snap | 5 +
...s_Null_Should_Null_Item_Without_Error.snap | 9 +
...eturns_Null_Should_Null_Without_Error.snap | 5 +
...eturns_Null_Should_Null_Without_Error.snap | 5 +
...t_Item_Returns_Null_Should_Error_Item.snap | 31 +
...m_Throwing_Should_Null_And_Error_Item.snap | 44 +
...Object_List_Returns_Null_Should_Error.snap | 22 +
..._List_Throwing_Should_Null_FAnd_Error.snap | 19 +
...sync_Object_Returns_Null_Should_Error.snap | 22 +
...Object_Throwing_Should_Null_And_Error.snap | 19 +
...t_Item_Returns_Null_Should_Error_Item.snap | 27 +
...m_Throwing_Should_Null_And_Error_Item.snap | 40 +
...Scalar_List_Returns_Null_Should_Error.snap | 22 +
...r_List_Throwing_Should_Null_And_Error.snap | 19 +
...sync_Scalar_Returns_Null_Should_Error.snap | 22 +
...Scalar_Throwing_Should_Null_And_Error.snap | 19 +
...sts.Mutation_With_MutationConventions.snap | 7 +
...s_Null_Should_Null_Item_Without_Error.snap | 13 +
...eturns_Null_Should_Null_Without_Error.snap | 5 +
...eturns_Null_Should_Null_Without_Error.snap | 5 +
...s_Null_Should_Null_Item_Without_Error.snap | 9 +
...eturns_Null_Should_Null_Without_Error.snap | 5 +
...eturns_Null_Should_Null_Without_Error.snap | 5 +
...t_Item_Returns_Null_Should_Error_Item.snap | 31 +
...m_Throwing_Should_Null_And_Error_Item.snap | 44 +
...Object_List_Returns_Null_Should_Error.snap | 22 +
..._List_Throwing_Should_Null_FAnd_Error.snap | 19 +
...Pure_Object_Returns_Null_Should_Error.snap | 22 +
...Object_Throwing_Should_Null_And_Error.snap | 19 +
...ner_Return_Null_Should_Null_And_Error.snap | 53 +
...le_Returns_Null_Should_Null_And_Error.snap | 35 +
...t_Item_Returns_Null_Should_Error_Item.snap | 27 +
...m_Throwing_Should_Null_And_Error_Item.snap | 40 +
...Scalar_List_Returns_Null_Should_Error.snap | 22 +
...r_List_Throwing_Should_Null_And_Error.snap | 19 +
...Pure_Scalar_Returns_Null_Should_Error.snap | 22 +
...Scalar_Throwing_Should_Null_And_Error.snap | 19 +
...ticNonNullTests.Query_With_Connection.snap | 34 +
...ts.Query_With_NullableConnectionNodes.snap | 17 +
.../test/Types.Tests/SemanticNonNullTests.cs | 334 ++++++
....Apply_SemanticNonNull_To_SchemaFirst.snap | 27 +
...Derive_SemanticNonNull_From_CodeFirst.snap | 27 +
...anticNonNull_From_ImplementationFirst.snap | 27 +
...ationFirst_With_GraphQLType_As_String.snap | 27 +
...ntationFirst_With_GraphQLType_As_Type.snap | 27 +
...anticNonNullTests.MutationConventions.snap | 24 +
...NonNullTests.Object_Implementing_Node.snap | 22 +
.../SemanticNonNullTests.Pagination.snap | 56 +
60 files changed, 2921 insertions(+), 128 deletions(-)
create mode 100644 src/HotChocolate/Core/src/Types/SemanticNonNullTypeInterceptor.cs
create mode 100644 src/HotChocolate/Core/src/Types/Types/Directives/SemanticNonNullDirective.cs
create mode 100644 src/HotChocolate/Core/test/Execution.Tests/SemanticNonNullTests.cs
create mode 100644 src/HotChocolate/Core/test/Execution.Tests/__snapshots__/SemanticNonNullTests.Async_Nullable_Object_List_Item_Returns_Null_Should_Null_Item_Without_Error.snap
create mode 100644 src/HotChocolate/Core/test/Execution.Tests/__snapshots__/SemanticNonNullTests.Async_Nullable_Object_List_Returns_Null_Should_Null_Without_Error.snap
create mode 100644 src/HotChocolate/Core/test/Execution.Tests/__snapshots__/SemanticNonNullTests.Async_Nullable_Object_Returns_Null_Should_Null_Without_Error.snap
create mode 100644 src/HotChocolate/Core/test/Execution.Tests/__snapshots__/SemanticNonNullTests.Async_Nullable_Scalar_List_Item_Returns_Null_Should_Null_Item_Without_Error.snap
create mode 100644 src/HotChocolate/Core/test/Execution.Tests/__snapshots__/SemanticNonNullTests.Async_Nullable_Scalar_List_Returns_Null_Should_Null_Without_Error.snap
create mode 100644 src/HotChocolate/Core/test/Execution.Tests/__snapshots__/SemanticNonNullTests.Async_Nullable_Scalar_Returns_Null_Should_Null_Without_Error.snap
create mode 100644 src/HotChocolate/Core/test/Execution.Tests/__snapshots__/SemanticNonNullTests.Async_Object_List_Item_Returns_Null_Should_Error_Item.snap
create mode 100644 src/HotChocolate/Core/test/Execution.Tests/__snapshots__/SemanticNonNullTests.Async_Object_List_Item_Throwing_Should_Null_And_Error_Item.snap
create mode 100644 src/HotChocolate/Core/test/Execution.Tests/__snapshots__/SemanticNonNullTests.Async_Object_List_Returns_Null_Should_Error.snap
create mode 100644 src/HotChocolate/Core/test/Execution.Tests/__snapshots__/SemanticNonNullTests.Async_Object_List_Throwing_Should_Null_FAnd_Error.snap
create mode 100644 src/HotChocolate/Core/test/Execution.Tests/__snapshots__/SemanticNonNullTests.Async_Object_Returns_Null_Should_Error.snap
create mode 100644 src/HotChocolate/Core/test/Execution.Tests/__snapshots__/SemanticNonNullTests.Async_Object_Throwing_Should_Null_And_Error.snap
create mode 100644 src/HotChocolate/Core/test/Execution.Tests/__snapshots__/SemanticNonNullTests.Async_Scalar_List_Item_Returns_Null_Should_Error_Item.snap
create mode 100644 src/HotChocolate/Core/test/Execution.Tests/__snapshots__/SemanticNonNullTests.Async_Scalar_List_Item_Throwing_Should_Null_And_Error_Item.snap
create mode 100644 src/HotChocolate/Core/test/Execution.Tests/__snapshots__/SemanticNonNullTests.Async_Scalar_List_Returns_Null_Should_Error.snap
create mode 100644 src/HotChocolate/Core/test/Execution.Tests/__snapshots__/SemanticNonNullTests.Async_Scalar_List_Throwing_Should_Null_And_Error.snap
create mode 100644 src/HotChocolate/Core/test/Execution.Tests/__snapshots__/SemanticNonNullTests.Async_Scalar_Returns_Null_Should_Error.snap
create mode 100644 src/HotChocolate/Core/test/Execution.Tests/__snapshots__/SemanticNonNullTests.Async_Scalar_Throwing_Should_Null_And_Error.snap
create mode 100644 src/HotChocolate/Core/test/Execution.Tests/__snapshots__/SemanticNonNullTests.Mutation_With_MutationConventions.snap
create mode 100644 src/HotChocolate/Core/test/Execution.Tests/__snapshots__/SemanticNonNullTests.Pure_Nullable_Object_List_Item_Returns_Null_Should_Null_Item_Without_Error.snap
create mode 100644 src/HotChocolate/Core/test/Execution.Tests/__snapshots__/SemanticNonNullTests.Pure_Nullable_Object_List_Returns_Null_Should_Null_Without_Error.snap
create mode 100644 src/HotChocolate/Core/test/Execution.Tests/__snapshots__/SemanticNonNullTests.Pure_Nullable_Object_Returns_Null_Should_Null_Without_Error.snap
create mode 100644 src/HotChocolate/Core/test/Execution.Tests/__snapshots__/SemanticNonNullTests.Pure_Nullable_Scalar_List_Item_Returns_Null_Should_Null_Item_Without_Error.snap
create mode 100644 src/HotChocolate/Core/test/Execution.Tests/__snapshots__/SemanticNonNullTests.Pure_Nullable_Scalar_List_Returns_Null_Should_Null_Without_Error.snap
create mode 100644 src/HotChocolate/Core/test/Execution.Tests/__snapshots__/SemanticNonNullTests.Pure_Nullable_Scalar_Returns_Null_Should_Null_Without_Error.snap
create mode 100644 src/HotChocolate/Core/test/Execution.Tests/__snapshots__/SemanticNonNullTests.Pure_Object_List_Item_Returns_Null_Should_Error_Item.snap
create mode 100644 src/HotChocolate/Core/test/Execution.Tests/__snapshots__/SemanticNonNullTests.Pure_Object_List_Item_Throwing_Should_Null_And_Error_Item.snap
create mode 100644 src/HotChocolate/Core/test/Execution.Tests/__snapshots__/SemanticNonNullTests.Pure_Object_List_Returns_Null_Should_Error.snap
create mode 100644 src/HotChocolate/Core/test/Execution.Tests/__snapshots__/SemanticNonNullTests.Pure_Object_List_Throwing_Should_Null_FAnd_Error.snap
create mode 100644 src/HotChocolate/Core/test/Execution.Tests/__snapshots__/SemanticNonNullTests.Pure_Object_Returns_Null_Should_Error.snap
create mode 100644 src/HotChocolate/Core/test/Execution.Tests/__snapshots__/SemanticNonNullTests.Pure_Object_Throwing_Should_Null_And_Error.snap
create mode 100644 src/HotChocolate/Core/test/Execution.Tests/__snapshots__/SemanticNonNullTests.Pure_Scalar_ListOfList_Nullable_Middle_Item_Outer_And_Inner_Return_Null_Should_Null_And_Error.snap
create mode 100644 src/HotChocolate/Core/test/Execution.Tests/__snapshots__/SemanticNonNullTests.Pure_Scalar_ListOfList_Nullable_Outer_And_Inner_Middle_Returns_Null_Should_Null_And_Error.snap
create mode 100644 src/HotChocolate/Core/test/Execution.Tests/__snapshots__/SemanticNonNullTests.Pure_Scalar_List_Item_Returns_Null_Should_Error_Item.snap
create mode 100644 src/HotChocolate/Core/test/Execution.Tests/__snapshots__/SemanticNonNullTests.Pure_Scalar_List_Item_Throwing_Should_Null_And_Error_Item.snap
create mode 100644 src/HotChocolate/Core/test/Execution.Tests/__snapshots__/SemanticNonNullTests.Pure_Scalar_List_Returns_Null_Should_Error.snap
create mode 100644 src/HotChocolate/Core/test/Execution.Tests/__snapshots__/SemanticNonNullTests.Pure_Scalar_List_Throwing_Should_Null_And_Error.snap
create mode 100644 src/HotChocolate/Core/test/Execution.Tests/__snapshots__/SemanticNonNullTests.Pure_Scalar_Returns_Null_Should_Error.snap
create mode 100644 src/HotChocolate/Core/test/Execution.Tests/__snapshots__/SemanticNonNullTests.Pure_Scalar_Throwing_Should_Null_And_Error.snap
create mode 100644 src/HotChocolate/Core/test/Execution.Tests/__snapshots__/SemanticNonNullTests.Query_With_Connection.snap
create mode 100644 src/HotChocolate/Core/test/Execution.Tests/__snapshots__/SemanticNonNullTests.Query_With_NullableConnectionNodes.snap
create mode 100644 src/HotChocolate/Core/test/Types.Tests/SemanticNonNullTests.cs
create mode 100644 src/HotChocolate/Core/test/Types.Tests/__snapshots__/SemanticNonNullTests.Apply_SemanticNonNull_To_SchemaFirst.snap
create mode 100644 src/HotChocolate/Core/test/Types.Tests/__snapshots__/SemanticNonNullTests.Derive_SemanticNonNull_From_CodeFirst.snap
create mode 100644 src/HotChocolate/Core/test/Types.Tests/__snapshots__/SemanticNonNullTests.Derive_SemanticNonNull_From_ImplementationFirst.snap
create mode 100644 src/HotChocolate/Core/test/Types.Tests/__snapshots__/SemanticNonNullTests.Derive_SemanticNonNull_From_ImplementationFirst_With_GraphQLType_As_String.snap
create mode 100644 src/HotChocolate/Core/test/Types.Tests/__snapshots__/SemanticNonNullTests.Derive_SemanticNonNull_From_ImplementationFirst_With_GraphQLType_As_Type.snap
create mode 100644 src/HotChocolate/Core/test/Types.Tests/__snapshots__/SemanticNonNullTests.MutationConventions.snap
create mode 100644 src/HotChocolate/Core/test/Types.Tests/__snapshots__/SemanticNonNullTests.Object_Implementing_Node.snap
create mode 100644 src/HotChocolate/Core/test/Types.Tests/__snapshots__/SemanticNonNullTests.Pagination.snap
diff --git a/src/HotChocolate/Core/src/Abstractions/ErrorCodes.cs b/src/HotChocolate/Core/src/Abstractions/ErrorCodes.cs
index dad7fd68e68..d18b87831d1 100644
--- a/src/HotChocolate/Core/src/Abstractions/ErrorCodes.cs
+++ b/src/HotChocolate/Core/src/Abstractions/ErrorCodes.cs
@@ -49,6 +49,7 @@ public static class Execution
public const string OneSlicingArgumentRequired = "HC0082";
public const string NonNullViolation = "HC0018";
+ public const string SemanticNonNullViolation = "HC0088";
public const string MustBeInputType = "HC0017";
public const string InvalidType = "HC0016";
public const string QueryNotFound = "HC0015";
diff --git a/src/HotChocolate/Core/src/Abstractions/WellKnownDirectives.cs b/src/HotChocolate/Core/src/Abstractions/WellKnownDirectives.cs
index 9126408eb4b..ed741877439 100644
--- a/src/HotChocolate/Core/src/Abstractions/WellKnownDirectives.cs
+++ b/src/HotChocolate/Core/src/Abstractions/WellKnownDirectives.cs
@@ -69,4 +69,14 @@ public static class WellKnownDirectives
/// The name of the @tag argument name.
///
public const string Name = "name";
+
+ ///
+ /// The name of the @semanticNonNull directive.
+ ///
+ public const string SemanticNonNull = "semanticNonNull";
+
+ ///
+ /// The name of the @semanticNonNull argument levels.
+ ///
+ public const string Levels = "levels";
}
diff --git a/src/HotChocolate/Core/src/Abstractions/WellKnownMiddleware.cs b/src/HotChocolate/Core/src/Abstractions/WellKnownMiddleware.cs
index d7d31265e9d..be6fa2bb050 100644
--- a/src/HotChocolate/Core/src/Abstractions/WellKnownMiddleware.cs
+++ b/src/HotChocolate/Core/src/Abstractions/WellKnownMiddleware.cs
@@ -90,4 +90,9 @@ public static class WellKnownMiddleware
/// The key identifies the authorization middleware.
///
public const string Authorization = "HotChocolate.Authorization";
+
+ ///
+ /// This key identifies the semantic-non-null middleware.
+ ///
+ public const string SemanticNonNull = "HotChocolate.Types.SemanticNonNull";
}
diff --git a/src/HotChocolate/Core/src/Types/IReadOnlySchemaOptions.cs b/src/HotChocolate/Core/src/Types/IReadOnlySchemaOptions.cs
index 78b8062ffe9..6509c7e4614 100644
--- a/src/HotChocolate/Core/src/Types/IReadOnlySchemaOptions.cs
+++ b/src/HotChocolate/Core/src/Types/IReadOnlySchemaOptions.cs
@@ -165,6 +165,13 @@ public interface IReadOnlySchemaOptions
///
bool EnableStream { get; }
+ ///
+ /// Enables the @semanticNonNull directive and rewrites Non-Null types to nullable types
+ /// with this directive attached to indicate semantic non-nullability.
+ /// This feature is experimental and might be changed or removed in the future.
+ ///
+ bool EnableSemanticNonNull { get; }
+
///
/// Specifies the maximum allowed nodes that can be fetched at once through the nodes field.
///
diff --git a/src/HotChocolate/Core/src/Types/SchemaBuilder.cs b/src/HotChocolate/Core/src/Types/SchemaBuilder.cs
index e538d629c02..b34a54db89b 100644
--- a/src/HotChocolate/Core/src/Types/SchemaBuilder.cs
+++ b/src/HotChocolate/Core/src/Types/SchemaBuilder.cs
@@ -37,6 +37,7 @@ public partial class SchemaBuilder : ISchemaBuilder
typeof(InterfaceCompletionTypeInterceptor),
typeof(MiddlewareValidationTypeInterceptor),
typeof(EnableTrueNullabilityTypeInterceptor),
+ typeof(SemanticNonNullTypeInterceptor),
];
private SchemaOptions _options = new();
diff --git a/src/HotChocolate/Core/src/Types/SchemaOptions.cs b/src/HotChocolate/Core/src/Types/SchemaOptions.cs
index 800c9e0f827..eb4b277f829 100644
--- a/src/HotChocolate/Core/src/Types/SchemaOptions.cs
+++ b/src/HotChocolate/Core/src/Types/SchemaOptions.cs
@@ -15,58 +15,34 @@ public class SchemaOptions : IReadOnlySchemaOptions
private BindingBehavior _defaultBindingBehavior = BindingBehavior.Implicit;
private FieldBindingFlags _defaultFieldBindingFlags = FieldBindingFlags.Instance;
- ///
- /// Gets or sets the name of the query type.
- ///
+ ///
public string? QueryTypeName { get; set; }
- ///
- /// Gets or sets the name of the mutation type.
- ///
+ ///
public string? MutationTypeName { get; set; }
- ///
- /// Gets or sets the name of the subscription type.
- ///
+ ///
public string? SubscriptionTypeName { get; set; }
- ///
- /// Defines if the schema allows the query type to be omitted.
- ///
+ ///
public bool StrictValidation { get; set; } = true;
- ///
- /// Defines if the CSharp XML documentation shall be integrated.
- ///
+ ///
public bool UseXmlDocumentation { get; set; } = true;
- ///
- /// A delegate which defines the name of the XML documentation file to be read.
- /// Only used if is true.
- ///
+ ///
public Func? ResolveXmlDocumentationFileName { get; set; }
- ///
- /// Defines if fields shall be sorted by name.
- /// Default: false
- ///
+ ///
public bool SortFieldsByName { get; set; }
- ///
- /// Defines if types shall be removed from the schema that are
- /// unreachable from the root types.
- ///
+ ///
public bool RemoveUnreachableTypes { get; set; }
- ///
- /// Defines if unused type system directives shall
- /// be removed from the schema.
- ///
+ ///
public bool RemoveUnusedTypeSystemDirectives { get; set; } = true;
- ///
- /// Defines the default binding behavior.
- ///
+ ///
public BindingBehavior DefaultBindingBehavior
{
get => _defaultBindingBehavior;
@@ -81,10 +57,7 @@ public BindingBehavior DefaultBindingBehavior
}
}
- ///
- /// Defines which members shall be by default inferred as GraphQL fields.
- /// This default applies to and .
- ///
+ ///
public FieldBindingFlags DefaultFieldBindingFlags
{
get => _defaultFieldBindingFlags;
@@ -99,137 +72,69 @@ public FieldBindingFlags DefaultFieldBindingFlags
}
}
- ///
- /// Defines on which fields a middleware pipeline can be applied on.
- ///
+ ///
public FieldMiddlewareApplication FieldMiddleware { get; set; } =
FieldMiddlewareApplication.UserDefinedFields;
- ///
- /// Defines if the experimental directive introspection feature shall be enabled.
- ///
+ ///
public bool EnableDirectiveIntrospection { get; set; }
- ///
- /// The default directive visibility when directive introspection is enabled.
- ///
+ ///
public DirectiveVisibility DefaultDirectiveVisibility { get; set; } =
DirectiveVisibility.Public;
- ///
- /// Defines that the default resolver execution strategy.
- ///
+ ///
public ExecutionStrategy DefaultResolverStrategy { get; set; } =
ExecutionStrategy.Parallel;
- ///
- /// Defines if the order of important middleware components shall be validated.
- ///
+ ///
public bool ValidatePipelineOrder { get; set; } = true;
- ///
- /// Defines if the runtime types of types shall be validated.
- ///
+ ///
public bool StrictRuntimeTypeValidation { get; set; }
- ///
- /// Defines a delegate that determines if a runtime
- /// is an instance of an .
- ///
+ ///
public IsOfTypeFallback? DefaultIsOfTypeCheck { get; set; }
- ///
- /// Defines if the OneOf spec RFC is enabled. This feature is experimental.
- ///
+ ///
public bool EnableOneOf { get; set; }
- ///
- /// Defines if the schema building process shall validate that all nodes are resolvable through `node`.
- ///
+ ///
public bool EnsureAllNodesCanBeResolved { get; set; } = true;
- ///
- /// Defines if flag enums should be inferred as object value nodes
- ///
- ///
- /// Given the following enum
- ///
- ///
- /// [Flags]
- /// public enum Example { First, Second, Third }
- ///
- /// public class Query { public Example Loopback(Example input) => input;
- ///
- ///
- /// The following schema is produced
- ///
- ///
- /// type Query {
- /// loopback(input: ExampleFlagsInput!): ExampleFlags
- /// }
- ///
- /// type ExampleFlags {
- /// isFirst: Boolean!
- /// isSecond: Boolean!
- /// isThird: Boolean!
- /// }
- ///
- /// input ExampleFlagsInput {
- /// isFirst: Boolean
- /// isSecond: Boolean
- /// isThird: Boolean
- /// }
- ///
- ///
+ ///
public bool EnableFlagEnums { get; set; }
- ///
- /// Enables the @defer directive.
- /// Defer and stream both are at the moment preview features.
- ///
+ ///
public bool EnableDefer { get; set; }
- ///
- /// Enables the @stream directive.
- /// Defer and stream both are at the moment preview features.
- ///
+ ///
public bool EnableStream { get; set; }
- ///
- /// Specifies the maximum allowed nodes that can be fetched at once through the nodes field.
- ///
+ ///
+ public bool EnableSemanticNonNull { get; set; }
+
+ ///
public int MaxAllowedNodeBatchSize { get; set; } = 50;
- ///
- /// Specified if the leading I shall be stripped from the interface name.
- ///
+ ///
public bool StripLeadingIFromInterface { get; set; }
- ///
- /// Specifies that the true nullability proto type shall be enabled.
- ///
+ ///
public bool EnableTrueNullability { get; set; }
- ///
- /// Specifies that the @tag directive shall be registered with the type system.
- ///
+ ///
public bool EnableTag { get; set; } = true;
- ///
- /// Defines the default dependency injection scope for query fields.
- ///
+ ///
public DependencyInjectionScope DefaultQueryDependencyInjectionScope { get; set; } =
DependencyInjectionScope.Resolver;
- ///
- /// Defines the default dependency injection scope for mutation fields.
- ///
+ ///
public DependencyInjectionScope DefaultMutationDependencyInjectionScope { get; set; } =
DependencyInjectionScope.Request;
- ///
- /// Defines if the root field pages shall be published to the promise cache.
- ///
+ ///
public bool PublishRootFieldPagesToPromiseCache { get; set; } = true;
///
@@ -263,6 +168,7 @@ public static SchemaOptions FromOptions(IReadOnlySchemaOptions options)
EnableFlagEnums = options.EnableFlagEnums,
EnableDefer = options.EnableDefer,
EnableStream = options.EnableStream,
+ EnableSemanticNonNull = options.EnableSemanticNonNull,
DefaultFieldBindingFlags = options.DefaultFieldBindingFlags,
MaxAllowedNodeBatchSize = options.MaxAllowedNodeBatchSize,
StripLeadingIFromInterface = options.StripLeadingIFromInterface,
diff --git a/src/HotChocolate/Core/src/Types/SemanticNonNullTypeInterceptor.cs b/src/HotChocolate/Core/src/Types/SemanticNonNullTypeInterceptor.cs
new file mode 100644
index 00000000000..1be7f69f7bf
--- /dev/null
+++ b/src/HotChocolate/Core/src/Types/SemanticNonNullTypeInterceptor.cs
@@ -0,0 +1,376 @@
+#nullable enable
+
+using System.Collections;
+using HotChocolate.Configuration;
+using HotChocolate.Language;
+using HotChocolate.Resolvers;
+using HotChocolate.Types;
+using HotChocolate.Types.Descriptors;
+using HotChocolate.Types.Descriptors.Definitions;
+using HotChocolate.Types.Helpers;
+using HotChocolate.Types.Relay;
+
+namespace HotChocolate;
+
+internal sealed class SemanticNonNullTypeInterceptor : TypeInterceptor
+{
+ private ITypeInspector _typeInspector = null!;
+ private ExtendedTypeReference _nodeTypeReference = null!;
+
+ internal override bool IsEnabled(IDescriptorContext context)
+ => context.Options.EnableSemanticNonNull;
+
+ internal override void InitializeContext(
+ IDescriptorContext context,
+ TypeInitializer typeInitializer,
+ TypeRegistry typeRegistry,
+ TypeLookup typeLookup,
+ TypeReferenceResolver typeReferenceResolver)
+ {
+ _typeInspector = context.TypeInspector;
+
+ _nodeTypeReference = _typeInspector.GetTypeRef(typeof(NodeType));
+ }
+
+ ///
+ /// After the root types have been resolved, we go through all the fields of the mutation type
+ /// and undo semantic non-nullability. This is because mutations can be chained and we want to retain
+ /// the null-bubbling so execution is aborted once one non-null mutation field produces an error.
+ /// We have to do this in a different hook because the mutation type is not yet fully resolved in the
+ /// hook.
+ ///
+ public override void OnAfterResolveRootType(
+ ITypeCompletionContext completionContext,
+ ObjectTypeDefinition definition,
+ OperationType operationType)
+ {
+ if (operationType == OperationType.Mutation)
+ {
+ foreach (var field in definition.Fields)
+ {
+ if (field.IsIntrospectionField)
+ {
+ continue;
+ }
+
+ if (!field.HasDirectives)
+ {
+ continue;
+ }
+
+ var semanticNonNullDirective =
+ field.Directives.FirstOrDefault(d => d.Value is SemanticNonNullDirective);
+
+ if (semanticNonNullDirective is not null)
+ {
+ field.Directives.Remove(semanticNonNullDirective);
+ }
+
+ var semanticNonNullFormatterDefinition =
+ field.FormatterDefinitions.FirstOrDefault(fd => fd.Key == WellKnownMiddleware.SemanticNonNull);
+
+ if (semanticNonNullFormatterDefinition is not null)
+ {
+ field.FormatterDefinitions.Remove(semanticNonNullFormatterDefinition);
+ }
+ }
+ }
+ }
+
+ public override void OnAfterCompleteName(ITypeCompletionContext completionContext, DefinitionBase definition)
+ {
+ if (completionContext.IsIntrospectionType)
+ {
+ return;
+ }
+
+ if (definition is ObjectTypeDefinition objectDef)
+ {
+ if (objectDef.Name is "CollectionSegmentInfo" or "PageInfo")
+ {
+ return;
+ }
+
+ var implementsNode = objectDef.Interfaces.Any(i => i.Equals(_nodeTypeReference));
+
+ foreach (var field in objectDef.Fields)
+ {
+ if (field.IsIntrospectionField)
+ {
+ continue;
+ }
+
+ if (implementsNode && field.Name == "id")
+ {
+ continue;
+ }
+
+ if (field.Type is null)
+ {
+ continue;
+ }
+
+ var levels = GetSemanticNonNullLevels(field.Type);
+
+ if (levels.Count < 1)
+ {
+ continue;
+ }
+
+ ApplySemanticNonNullDirective(field, completionContext, levels);
+
+ field.FormatterDefinitions.Add(CreateSemanticNonNullResultFormatterDefinition(levels));
+ }
+ }
+ else if (definition is InterfaceTypeDefinition interfaceDef)
+ {
+ if (interfaceDef.Name == "Node")
+ {
+ // The Node interface is well defined, so we don't want to go and change the type of its fields.
+ return;
+ }
+
+ foreach (var field in interfaceDef.Fields)
+ {
+ if (field.Type is null)
+ {
+ continue;
+ }
+
+ var levels = GetSemanticNonNullLevels(field.Type);
+
+ if (levels.Count < 1)
+ {
+ continue;
+ }
+
+ ApplySemanticNonNullDirective(field, completionContext, levels);
+ }
+ }
+ }
+
+ private void ApplySemanticNonNullDirective(
+ OutputFieldDefinitionBase field,
+ ITypeCompletionContext completionContext,
+ HashSet levels)
+ {
+ var directiveDependency = new TypeDependency(
+ _typeInspector.GetTypeRef(typeof(SemanticNonNullDirective)),
+ TypeDependencyFulfilled.Completed);
+
+ ((RegisteredType)completionContext).Dependencies.Add(directiveDependency);
+
+ field.AddDirective(new SemanticNonNullDirective(levels.ToList()), _typeInspector);
+
+ field.Type = BuildNullableTypeStructure(field.Type!, _typeInspector);
+ }
+
+ private static HashSet GetSemanticNonNullLevels(TypeReference typeReference)
+ {
+ if (typeReference is ExtendedTypeReference extendedTypeReference)
+ {
+ return GetSemanticNonNullLevelsFromReference(extendedTypeReference);
+ }
+
+ if (typeReference is SchemaTypeReference schemaRef)
+ {
+ return GetSemanticNonNullLevelsFromReference(schemaRef);
+ }
+
+ if (typeReference is SyntaxTypeReference syntaxRef)
+ {
+ return GetSemanticNonNullLevelsFromReference(syntaxRef);
+ }
+
+ return [];
+ }
+
+ private static HashSet GetSemanticNonNullLevelsFromReference(ExtendedTypeReference typeReference)
+ {
+ var levels = new HashSet();
+
+ var currentType = typeReference.Type;
+ var index = 0;
+
+ do
+ {
+ if (currentType.IsArrayOrList)
+ {
+ if (!currentType.IsNullable)
+ {
+ levels.Add(index);
+ }
+
+ index++;
+ currentType = currentType.ElementType;
+ }
+ else if (!currentType.IsNullable)
+ {
+ levels.Add(index);
+ break;
+ }
+ else
+ {
+ break;
+ }
+ } while (currentType is not null);
+
+ return levels;
+ }
+
+ private static HashSet GetSemanticNonNullLevelsFromReference(SchemaTypeReference typeReference)
+ {
+ var levels = new HashSet();
+
+ var currentType = typeReference.Type;
+ var index = 0;
+
+ while (true)
+ {
+ if (currentType is ListType listType)
+ {
+ index++;
+ currentType = listType.ElementType;
+ }
+ else if (currentType is NonNullType nonNullType)
+ {
+ levels.Add(index);
+ currentType = nonNullType.Type;
+ }
+ else
+ {
+ break;
+ }
+ }
+
+ return levels;
+ }
+
+ private static HashSet GetSemanticNonNullLevelsFromReference(SyntaxTypeReference typeReference)
+ {
+ var levels = new HashSet();
+
+ var currentType = typeReference.Type;
+ var index = 0;
+
+ while (true)
+ {
+ if (currentType is ListTypeNode listType)
+ {
+ index++;
+ currentType = listType.Type;
+ }
+ else if (currentType is NonNullTypeNode nonNullType)
+ {
+ levels.Add(index);
+ currentType = nonNullType.Type;
+ }
+ else
+ {
+ break;
+ }
+ }
+
+ return levels;
+ }
+
+ private static readonly bool?[] _fullNullablePattern = Enumerable.Range(0, 32).Select(_ => (bool?)true).ToArray();
+
+ private static TypeReference BuildNullableTypeStructure(
+ TypeReference typeReference,
+ ITypeInspector typeInspector)
+ {
+ if (typeReference is ExtendedTypeReference extendedTypeRef)
+ {
+ return extendedTypeRef.WithType(typeInspector.ChangeNullability(extendedTypeRef.Type,
+ _fullNullablePattern));
+ }
+
+ if (typeReference is SchemaTypeReference schemaRef)
+ {
+ return schemaRef.WithType(BuildNullableTypeStructure(schemaRef.Type));
+ }
+
+ if (typeReference is SyntaxTypeReference syntaxRef)
+ {
+ return syntaxRef.WithType(BuildNullableTypeStructure(syntaxRef.Type));
+ }
+
+ throw new NotSupportedException();
+ }
+
+ private static IType BuildNullableTypeStructure(ITypeSystemMember typeSystemMember)
+ {
+ if (typeSystemMember is ListType listType)
+ {
+ return new ListType(BuildNullableTypeStructure(listType.ElementType));
+ }
+
+ if (typeSystemMember is NonNullType nonNullType)
+ {
+ return BuildNullableTypeStructure(nonNullType.Type);
+ }
+
+ return (IType)typeSystemMember;
+ }
+
+ private static ITypeNode BuildNullableTypeStructure(ITypeNode typeNode)
+ {
+ if (typeNode is ListTypeNode listType)
+ {
+ return new ListTypeNode(BuildNullableTypeStructure(listType.Type));
+ }
+
+ if (typeNode is NonNullTypeNode nonNullType)
+ {
+ return BuildNullableTypeStructure(nonNullType.Type);
+ }
+
+ return typeNode;
+ }
+
+ private static ResultFormatterDefinition CreateSemanticNonNullResultFormatterDefinition(HashSet levels)
+ => new((context, result) =>
+ {
+ CheckResultForSemanticNonNullViolations(result, context, context.Path, levels, 0);
+
+ return result;
+ },
+ key: WellKnownMiddleware.SemanticNonNull,
+ isRepeatable: false);
+
+ private static void CheckResultForSemanticNonNullViolations(object? result, IResolverContext context, Path path,
+ HashSet levels,
+ int currentLevel)
+ {
+ if (result is null && levels.Contains(currentLevel))
+ {
+ context.ReportError(CreateSemanticNonNullViolationError(path));
+ return;
+ }
+
+ if (result is IEnumerable enumerable)
+ {
+ if (currentLevel >= 32)
+ {
+ // We bail if we're at a depth of 32 as this would mean that we're dealing with an AnyType or another structure.
+ return;
+ }
+
+ var index = 0;
+ foreach (var item in enumerable)
+ {
+ CheckResultForSemanticNonNullViolations(item, context, path.Append(index), levels, currentLevel + 1);
+
+ index++;
+ }
+ }
+ }
+
+ private static IError CreateSemanticNonNullViolationError(Path path)
+ => ErrorBuilder.New()
+ .SetMessage("Cannot return null for semantic non-null field.")
+ .SetCode(ErrorCodes.Execution.SemanticNonNullViolation)
+ .SetPath(path)
+ .Build();
+}
diff --git a/src/HotChocolate/Core/src/Types/Types/Directives/Directives.cs b/src/HotChocolate/Core/src/Types/Types/Directives/Directives.cs
index 716b0259114..362a883a4d1 100644
--- a/src/HotChocolate/Core/src/Types/Types/Directives/Directives.cs
+++ b/src/HotChocolate/Core/src/Types/Types/Directives/Directives.cs
@@ -15,6 +15,7 @@ public static class Directives
WellKnownDirectives.Stream,
WellKnownDirectives.Defer,
WellKnownDirectives.OneOf,
+ WellKnownDirectives.SemanticNonNull
];
internal static IReadOnlyList CreateReferences(
@@ -38,6 +39,11 @@ internal static IReadOnlyList CreateReferences(
directiveTypes.Add(typeInspector.GetTypeRef(typeof(StreamDirectiveType)));
}
+ if (descriptorContext.Options.EnableSemanticNonNull)
+ {
+ directiveTypes.Add(typeInspector.GetTypeRef(typeof(SemanticNonNullDirective)));
+ }
+
if (descriptorContext.Options.EnableTag)
{
directiveTypes.Add(typeInspector.GetTypeRef(typeof(Tag)));
diff --git a/src/HotChocolate/Core/src/Types/Types/Directives/SemanticNonNullDirective.cs b/src/HotChocolate/Core/src/Types/Types/Directives/SemanticNonNullDirective.cs
new file mode 100644
index 00000000000..1f64e212a95
--- /dev/null
+++ b/src/HotChocolate/Core/src/Types/Types/Directives/SemanticNonNullDirective.cs
@@ -0,0 +1,11 @@
+#nullable enable
+
+namespace HotChocolate.Types;
+
+[DirectiveType(WellKnownDirectives.SemanticNonNull, DirectiveLocation.FieldDefinition, IsRepeatable = false)]
+public sealed class SemanticNonNullDirective(IReadOnlyList levels)
+{
+ [GraphQLType>>]
+ [DefaultValueSyntax("[0]")]
+ public IReadOnlyList? Levels { get; } = levels;
+}
diff --git a/src/HotChocolate/Core/test/Execution.Tests/SemanticNonNullTests.cs b/src/HotChocolate/Core/test/Execution.Tests/SemanticNonNullTests.cs
new file mode 100644
index 00000000000..48088a0729f
--- /dev/null
+++ b/src/HotChocolate/Core/test/Execution.Tests/SemanticNonNullTests.cs
@@ -0,0 +1,1057 @@
+using CookieCrumble;
+using HotChocolate.Resolvers;
+using HotChocolate.Types;
+using Microsoft.Extensions.DependencyInjection;
+
+namespace HotChocolate.Execution;
+
+public class SemanticNonNullTests
+{
+ #region Scalar
+
+ [Fact]
+ public async Task Async_Scalar_Returns_Null_Should_Error()
+ {
+ var result = await new ServiceCollection()
+ .AddGraphQL()
+ .ModifyOptions(o =>
+ {
+ o.EnableSemanticNonNull = true;
+ })
+ .AddQueryType()
+ .ExecuteRequestAsync("""
+ {
+ scalarReturningNull
+ }
+ """);
+
+ result.MatchSnapshot();
+ }
+
+ [Fact]
+ public async Task Async_Scalar_Throwing_Should_Null_And_Error()
+ {
+ var result = await new ServiceCollection()
+ .AddGraphQL()
+ .ModifyOptions(o =>
+ {
+ o.EnableSemanticNonNull = true;
+ })
+ .AddQueryType()
+ .ExecuteRequestAsync("""
+ {
+ scalarThrowingError
+ }
+ """);
+
+ result.MatchSnapshot();
+ }
+
+ [Fact]
+ public async Task Async_Nullable_Scalar_Returns_Null_Should_Null_Without_Error()
+ {
+ var result = await new ServiceCollection()
+ .AddGraphQL()
+ .ModifyOptions(o =>
+ {
+ o.EnableSemanticNonNull = true;
+ })
+ .AddQueryType()
+ .ExecuteRequestAsync("""
+ {
+ nullableScalarReturningNull
+ }
+ """);
+
+ result.MatchSnapshot();
+ }
+
+ [Fact]
+ public async Task Pure_Scalar_Returns_Null_Should_Error()
+ {
+ var result = await new ServiceCollection()
+ .AddGraphQL()
+ .ModifyOptions(o =>
+ {
+ o.EnableSemanticNonNull = true;
+ })
+ .AddQueryType()
+ .ExecuteRequestAsync("""
+ {
+ pureScalarReturningNull
+ }
+ """);
+
+ result.MatchSnapshot();
+ }
+
+ [Fact]
+ public async Task Pure_Scalar_Throwing_Should_Null_And_Error()
+ {
+ var result = await new ServiceCollection()
+ .AddGraphQL()
+ .ModifyOptions(o =>
+ {
+ o.EnableSemanticNonNull = true;
+ })
+ .AddQueryType()
+ .ExecuteRequestAsync("""
+ {
+ pureScalarThrowingError
+ }
+ """);
+
+ result.MatchSnapshot();
+ }
+
+ [Fact]
+ public async Task Pure_Nullable_Scalar_Returns_Null_Should_Null_Without_Error()
+ {
+ var result = await new ServiceCollection()
+ .AddGraphQL()
+ .ModifyOptions(o =>
+ {
+ o.EnableSemanticNonNull = true;
+ })
+ .AddQueryType()
+ .ExecuteRequestAsync("""
+ {
+ pureNullableScalarReturningNull
+ }
+ """);
+
+ result.MatchSnapshot();
+ }
+
+ #endregion
+
+ #region Scalar List
+
+ [Fact]
+ public async Task Async_Scalar_List_Returns_Null_Should_Error()
+ {
+ var result = await new ServiceCollection()
+ .AddGraphQL()
+ .ModifyOptions(o =>
+ {
+ o.EnableSemanticNonNull = true;
+ })
+ .AddQueryType()
+ .ExecuteRequestAsync("""
+ {
+ scalarListReturningNull
+ }
+ """);
+
+ result.MatchSnapshot();
+ }
+
+ [Fact]
+ public async Task Async_Scalar_List_Throwing_Should_Null_And_Error()
+ {
+ var result = await new ServiceCollection()
+ .AddGraphQL()
+ .ModifyOptions(o =>
+ {
+ o.EnableSemanticNonNull = true;
+ })
+ .AddQueryType()
+ .ExecuteRequestAsync("""
+ {
+ scalarListThrowingError
+ }
+ """);
+
+ result.MatchSnapshot();
+ }
+
+ [Fact]
+ public async Task Async_Nullable_Scalar_List_Returns_Null_Should_Null_Without_Error()
+ {
+ var result = await new ServiceCollection()
+ .AddGraphQL()
+ .ModifyOptions(o =>
+ {
+ o.EnableSemanticNonNull = true;
+ })
+ .AddQueryType()
+ .ExecuteRequestAsync("""
+ {
+ nullableScalarListReturningNull
+ }
+ """);
+
+ result.MatchSnapshot();
+ }
+
+ [Fact]
+ public async Task Pure_Scalar_List_Returns_Null_Should_Error()
+ {
+ var result = await new ServiceCollection()
+ .AddGraphQL()
+ .ModifyOptions(o =>
+ {
+ o.EnableSemanticNonNull = true;
+ })
+ .AddQueryType()
+ .ExecuteRequestAsync("""
+ {
+ pureScalarListReturningNull
+ }
+ """);
+
+ result.MatchSnapshot();
+ }
+
+ [Fact]
+ public async Task Pure_Scalar_List_Throwing_Should_Null_And_Error()
+ {
+ var result = await new ServiceCollection()
+ .AddGraphQL()
+ .ModifyOptions(o =>
+ {
+ o.EnableSemanticNonNull = true;
+ })
+ .AddQueryType()
+ .ExecuteRequestAsync("""
+ {
+ pureScalarListThrowingError
+ }
+ """);
+
+ result.MatchSnapshot();
+ }
+
+ [Fact]
+ public async Task Pure_Nullable_Scalar_List_Returns_Null_Should_Null_Without_Error()
+ {
+ var result = await new ServiceCollection()
+ .AddGraphQL()
+ .ModifyOptions(o =>
+ {
+ o.EnableSemanticNonNull = true;
+ })
+ .AddQueryType()
+ .ExecuteRequestAsync("""
+ {
+ pureNullableScalarListReturningNull
+ }
+ """);
+
+ result.MatchSnapshot();
+ }
+
+ #endregion
+
+ #region Scalar List Item
+
+ [Fact]
+ public async Task Async_Scalar_List_Item_Returns_Null_Should_Error_Item()
+ {
+ var result = await new ServiceCollection()
+ .AddGraphQL()
+ .ModifyOptions(o =>
+ {
+ o.EnableSemanticNonNull = true;
+ })
+ .AddQueryType()
+ .ExecuteRequestAsync("""
+ {
+ scalarListItemReturningNull
+ }
+ """);
+
+ result.MatchSnapshot();
+ }
+
+ [Fact]
+ public async Task Async_Scalar_List_Item_Throwing_Should_Null_And_Error_Item()
+ {
+ var result = await new ServiceCollection()
+ .AddGraphQL()
+ .ModifyOptions(o =>
+ {
+ o.EnableSemanticNonNull = true;
+ })
+ .AddQueryType()
+ .ExecuteRequestAsync("""
+ {
+ scalarListItemThrowingError
+ }
+ """);
+
+ result.MatchSnapshot();
+ }
+
+ [Fact]
+ public async Task Async_Nullable_Scalar_List_Item_Returns_Null_Should_Null_Item_Without_Error()
+ {
+ var result = await new ServiceCollection()
+ .AddGraphQL()
+ .ModifyOptions(o =>
+ {
+ o.EnableSemanticNonNull = true;
+ })
+ .AddQueryType()
+ .ExecuteRequestAsync("""
+ {
+ nullableScalarListItemReturningNull
+ }
+ """);
+
+ result.MatchSnapshot();
+ }
+
+ [Fact]
+ public async Task Pure_Scalar_List_Item_Returns_Null_Should_Error_Item()
+ {
+ var result = await new ServiceCollection()
+ .AddGraphQL()
+ .ModifyOptions(o =>
+ {
+ o.EnableSemanticNonNull = true;
+ })
+ .AddQueryType()
+ .ExecuteRequestAsync("""
+ {
+ pureScalarListItemReturningNull
+ }
+ """);
+
+ result.MatchSnapshot();
+ }
+
+ [Fact]
+ public async Task Pure_Scalar_List_Item_Throwing_Should_Null_And_Error_Item()
+ {
+ var result = await new ServiceCollection()
+ .AddGraphQL()
+ .ModifyOptions(o =>
+ {
+ o.EnableSemanticNonNull = true;
+ })
+ .AddQueryType()
+ .ExecuteRequestAsync("""
+ {
+ pureScalarListItemThrowingError
+ }
+ """);
+
+ result.MatchSnapshot();
+ }
+
+ [Fact]
+ public async Task Pure_Nullable_Scalar_List_Item_Returns_Null_Should_Null_Item_Without_Error()
+ {
+ var result = await new ServiceCollection()
+ .AddGraphQL()
+ .ModifyOptions(o =>
+ {
+ o.EnableSemanticNonNull = true;
+ })
+ .AddQueryType()
+ .ExecuteRequestAsync("""
+ {
+ pureNullableScalarListItemReturningNull
+ }
+ """);
+
+ result.MatchSnapshot();
+ }
+
+ #endregion
+
+ #region Object
+
+ [Fact]
+ public async Task Async_Object_Returns_Null_Should_Error()
+ {
+ var result = await new ServiceCollection()
+ .AddGraphQL()
+ .ModifyOptions(o =>
+ {
+ o.EnableSemanticNonNull = true;
+ })
+ .AddQueryType()
+ .ExecuteRequestAsync("""
+ {
+ objectReturningNull {
+ property
+ }
+ }
+ """);
+
+ result.MatchSnapshot();
+ }
+
+ [Fact]
+ public async Task Async_Object_Throwing_Should_Null_And_Error()
+ {
+ var result = await new ServiceCollection()
+ .AddGraphQL()
+ .ModifyOptions(o =>
+ {
+ o.EnableSemanticNonNull = true;
+ })
+ .AddQueryType()
+ .ExecuteRequestAsync("""
+ {
+ objectThrowingError {
+ property
+ }
+ }
+ """);
+
+ result.MatchSnapshot();
+ }
+
+ [Fact]
+ public async Task Async_Nullable_Object_Returns_Null_Should_Null_Without_Error()
+ {
+ var result = await new ServiceCollection()
+ .AddGraphQL()
+ .ModifyOptions(o =>
+ {
+ o.EnableSemanticNonNull = true;
+ })
+ .AddQueryType()
+ .ExecuteRequestAsync("""
+ {
+ nullableObjectReturningNull {
+ property
+ }
+ }
+ """);
+
+ result.MatchSnapshot();
+ }
+
+ [Fact]
+ public async Task Pure_Object_Returns_Null_Should_Error()
+ {
+ var result = await new ServiceCollection()
+ .AddGraphQL()
+ .ModifyOptions(o =>
+ {
+ o.EnableSemanticNonNull = true;
+ })
+ .AddQueryType()
+ .ExecuteRequestAsync("""
+ {
+ pureObjectReturningNull {
+ property
+ }
+ }
+ """);
+
+ result.MatchSnapshot();
+ }
+
+ [Fact]
+ public async Task Pure_Object_Throwing_Should_Null_And_Error()
+ {
+ var result = await new ServiceCollection()
+ .AddGraphQL()
+ .ModifyOptions(o =>
+ {
+ o.EnableSemanticNonNull = true;
+ })
+ .AddQueryType()
+ .ExecuteRequestAsync("""
+ {
+ pureObjectThrowingError {
+ property
+ }
+ }
+ """);
+
+ result.MatchSnapshot();
+ }
+
+ [Fact]
+ public async Task Pure_Nullable_Object_Returns_Null_Should_Null_Without_Error()
+ {
+ var result = await new ServiceCollection()
+ .AddGraphQL()
+ .ModifyOptions(o =>
+ {
+ o.EnableSemanticNonNull = true;
+ })
+ .AddQueryType()
+ .ExecuteRequestAsync("""
+ {
+ pureNullableObjectReturningNull {
+ property
+ }
+ }
+ """);
+
+ result.MatchSnapshot();
+ }
+
+ #endregion
+
+ #region Object List
+
+ [Fact]
+ public async Task Async_Object_List_Returns_Null_Should_Error()
+ {
+ var result = await new ServiceCollection()
+ .AddGraphQL()
+ .ModifyOptions(o =>
+ {
+ o.EnableSemanticNonNull = true;
+ })
+ .AddQueryType()
+ .ExecuteRequestAsync("""
+ {
+ objectListReturningNull {
+ property
+ }
+ }
+ """);
+
+ result.MatchSnapshot();
+ }
+
+ [Fact]
+ public async Task Async_Object_List_Throwing_Should_Null_FAnd_Error()
+ {
+ var result = await new ServiceCollection()
+ .AddGraphQL()
+ .ModifyOptions(o =>
+ {
+ o.EnableSemanticNonNull = true;
+ })
+ .AddQueryType()
+ .ExecuteRequestAsync("""
+ {
+ objectListThrowingError {
+ property
+ }
+ }
+ """);
+
+ result.MatchSnapshot();
+ }
+
+ [Fact]
+ public async Task Async_Nullable_Object_List_Returns_Null_Should_Null_Without_Error()
+ {
+ var result = await new ServiceCollection()
+ .AddGraphQL()
+ .ModifyOptions(o =>
+ {
+ o.EnableSemanticNonNull = true;
+ })
+ .AddQueryType()
+ .ExecuteRequestAsync("""
+ {
+ nullableObjectListReturningNull {
+ property
+ }
+ }
+ """);
+
+ result.MatchSnapshot();
+ }
+
+ [Fact]
+ public async Task Pure_Object_List_Returns_Null_Should_Error()
+ {
+ var result = await new ServiceCollection()
+ .AddGraphQL()
+ .ModifyOptions(o =>
+ {
+ o.EnableSemanticNonNull = true;
+ })
+ .AddQueryType()
+ .ExecuteRequestAsync("""
+ {
+ pureObjectListReturningNull {
+ property
+ }
+ }
+ """);
+
+ result.MatchSnapshot();
+ }
+
+ [Fact]
+ public async Task Pure_Object_List_Throwing_Should_Null_FAnd_Error()
+ {
+ var result = await new ServiceCollection()
+ .AddGraphQL()
+ .ModifyOptions(o =>
+ {
+ o.EnableSemanticNonNull = true;
+ })
+ .AddQueryType()
+ .ExecuteRequestAsync("""
+ {
+ pureObjectListThrowingError {
+ property
+ }
+ }
+ """);
+
+ result.MatchSnapshot();
+ }
+
+ [Fact]
+ public async Task Pure_Nullable_Object_List_Returns_Null_Should_Null_Without_Error()
+ {
+ var result = await new ServiceCollection()
+ .AddGraphQL()
+ .ModifyOptions(o =>
+ {
+ o.EnableSemanticNonNull = true;
+ })
+ .AddQueryType()
+ .ExecuteRequestAsync("""
+ {
+ pureNullableObjectListReturningNull {
+ property
+ }
+ }
+ """);
+
+ result.MatchSnapshot();
+ }
+
+ #endregion
+
+ #region Object List Item
+
+ [Fact]
+ public async Task Async_Object_List_Item_Returns_Null_Should_Error_Item()
+ {
+ var result = await new ServiceCollection()
+ .AddGraphQL()
+ .ModifyOptions(o =>
+ {
+ o.EnableSemanticNonNull = true;
+ })
+ .AddQueryType()
+ .ExecuteRequestAsync("""
+ {
+ objectListItemReturningNull {
+ property
+ }
+ }
+ """);
+
+ result.MatchSnapshot();
+ }
+
+ [Fact]
+ public async Task Async_Object_List_Item_Throwing_Should_Null_And_Error_Item()
+ {
+ var result = await new ServiceCollection()
+ .AddGraphQL()
+ .ModifyOptions(o =>
+ {
+ o.EnableSemanticNonNull = true;
+ })
+ .AddQueryType()
+ .ExecuteRequestAsync("""
+ {
+ objectListItemThrowingError {
+ property
+ }
+ }
+ """);
+
+ result.MatchSnapshot();
+ }
+
+ [Fact]
+ public async Task Async_Nullable_Object_List_Item_Returns_Null_Should_Null_Item_Without_Error()
+ {
+ var result = await new ServiceCollection()
+ .AddGraphQL()
+ .ModifyOptions(o =>
+ {
+ o.EnableSemanticNonNull = true;
+ })
+ .AddQueryType()
+ .ExecuteRequestAsync("""
+ {
+ nullableObjectListItemReturningNull {
+ property
+ }
+ }
+ """);
+
+ result.MatchSnapshot();
+ }
+
+ [Fact]
+ public async Task Pure_Object_List_Item_Returns_Null_Should_Error_Item()
+ {
+ var result = await new ServiceCollection()
+ .AddGraphQL()
+ .ModifyOptions(o =>
+ {
+ o.EnableSemanticNonNull = true;
+ })
+ .AddQueryType()
+ .ExecuteRequestAsync("""
+ {
+ pureObjectListItemReturningNull {
+ property
+ }
+ }
+ """);
+
+ result.MatchSnapshot();
+ }
+
+ [Fact]
+ public async Task Pure_Object_List_Item_Throwing_Should_Null_And_Error_Item()
+ {
+ var result = await new ServiceCollection()
+ .AddGraphQL()
+ .ModifyOptions(o =>
+ {
+ o.EnableSemanticNonNull = true;
+ })
+ .AddQueryType()
+ .ExecuteRequestAsync("""
+ {
+ pureObjectListItemThrowingError {
+ property
+ }
+ }
+ """);
+
+ result.MatchSnapshot();
+ }
+
+ [Fact]
+ public async Task Pure_Nullable_Object_List_Item_Returns_Null_Should_Null_Item_Without_Error()
+ {
+ var result = await new ServiceCollection()
+ .AddGraphQL()
+ .ModifyOptions(o =>
+ {
+ o.EnableSemanticNonNull = true;
+ })
+ .AddQueryType()
+ .ExecuteRequestAsync("""
+ {
+ pureNullableObjectListItemReturningNull {
+ property
+ }
+ }
+ """);
+
+ result.MatchSnapshot();
+ }
+
+ #endregion
+
+ [Fact]
+ public async Task Mutation_With_MutationConventions()
+ {
+ var result = await new ServiceCollection()
+ .AddGraphQL()
+ .ModifyOptions(o =>
+ {
+ o.StrictValidation = false;
+ o.EnableSemanticNonNull = true;
+ })
+ .AddMutationConventions()
+ .AddMutationType()
+ .ExecuteRequestAsync("""
+ mutation {
+ someMutationReturningNull {
+ scalarReturningNull
+ }
+ }
+ """);
+
+ result.MatchSnapshot();
+ }
+
+ [Fact]
+ public async Task Query_With_Connection()
+ {
+ var result = await new ServiceCollection()
+ .AddGraphQL()
+ .ModifyOptions(o =>
+ {
+ o.EnableSemanticNonNull = true;
+ })
+ .AddMutationConventions()
+ .AddQueryType()
+ .ExecuteRequestAsync("""
+ {
+ scalarConnection {
+ edges {
+ node
+ }
+ }
+ }
+ """);
+
+ result.MatchSnapshot();
+ }
+
+ [Fact]
+ public async Task Query_With_NullableConnectionNodes()
+ {
+ var result = await new ServiceCollection()
+ .AddGraphQL()
+ .ModifyOptions(o =>
+ {
+ o.EnableSemanticNonNull = true;
+ })
+ .AddMutationConventions()
+ .AddQueryType()
+ .ExecuteRequestAsync("""
+ {
+ nullableScalarConnection {
+ edges {
+ node
+ }
+ }
+ }
+ """);
+
+ result.MatchSnapshot();
+ }
+
+ [Fact]
+ public async Task Pure_Scalar_ListOfList_Nullable_Outer_And_Inner_Middle_Returns_Null_Should_Null_And_Error()
+ {
+ var result = await new ServiceCollection()
+ .AddGraphQL()
+ .ModifyOptions(o =>
+ {
+ o.EnableSemanticNonNull = true;
+ })
+ .AddMutationConventions()
+ .AddQueryType()
+ .ExecuteRequestAsync("""
+ {
+ nestedScalarArrayNullableOuterItems
+ }
+ """);
+
+ result.MatchSnapshot();
+ }
+
+ [Fact]
+ public async Task Pure_Scalar_ListOfList_Nullable_Middle_Item_Outer_And_Inner_Return_Null_Should_Null_And_Error()
+ {
+ var result = await new ServiceCollection()
+ .AddGraphQL()
+ .ModifyOptions(o =>
+ {
+ o.EnableSemanticNonNull = true;
+ })
+ .AddMutationConventions()
+ .AddQueryType()
+ .ExecuteRequestAsync("""
+ {
+ nestedScalarArrayNullableMiddleItem
+ }
+ """);
+
+ result.MatchSnapshot();
+ }
+
+ public class Query
+ {
+ #region Scalar
+
+ public Task ScalarReturningNull()
+ {
+ return Task.FromResult(null!);
+ }
+
+ public Task ScalarThrowingError()
+ {
+ throw new Exception("Something went wrong");
+ }
+
+ public Task NullableScalarReturningNull()
+ {
+ return Task.FromResult(null);
+ }
+
+ public string PureScalarReturningNull => null!;
+
+ public string PureScalarThrowingError => throw new Exception("Somethin went wrong");
+
+ public string? PureNullableScalarReturningNull => null;
+
+ #endregion
+
+ #region Scalar List
+
+ public Task ScalarListReturningNull()
+ {
+ return Task.FromResult(null!);
+ }
+
+ public Task ScalarListThrowingError()
+ {
+ throw new Exception("Something went wrong");
+ }
+
+ public Task NullableScalarListReturningNull()
+ {
+ return Task.FromResult(null);
+ }
+
+ public string[] PureScalarListReturningNull => null!;
+
+ public string[] PureScalarListThrowingError => throw new Exception("Somethin went wrong");
+
+ public string[]? PureNullableScalarListReturningNull => null;
+
+ #endregion
+
+ #region Scalar List Item
+
+ public Task ScalarListItemReturningNull()
+ {
+ return Task.FromResult(["a", null!, "c"]);
+ }
+
+ public Task ScalarListItemThrowingError(IResolverContext context)
+ {
+ // TODO: How can you create a terminating error for a single item?
+ context.ReportError(ErrorBuilder.New().SetMessage("Another error").SetPath(context.Path.Append(1)).Build());
+ return Task.FromResult(["a", null!, "c"]);
+ }
+
+ public Task NullableScalarListItemReturningNull()
+ {
+ return Task.FromResult(["a", null, "c"]);
+ }
+
+ public string[] PureScalarListItemReturningNull => ["a", null!, "c"];
+
+ // TODO: This is no longer a pure resolver as soon as it access the IResolverContext, right?
+ public string[] PureScalarListItemThrowingError(IResolverContext context)
+ {
+ // TODO: How can you create a terminating error for a single item?
+ context.ReportError(ErrorBuilder.New().SetMessage("Another error").SetPath(context.Path.Append(1)).Build());
+ return ["a", null!, "c"];
+ }
+
+ public string?[] PureNullableScalarListItemReturningNull => ["a", null, "c"];
+
+ #endregion
+
+ #region Object
+
+ public Task ObjectReturningNull()
+ {
+ return Task.FromResult(null!);
+ }
+
+ public Task ObjectThrowingError()
+ {
+ throw new Exception("Something went wrong");
+ }
+
+ public Task NullableObjectReturningNull()
+ {
+ return Task.FromResult(null);
+ }
+
+ public SomeObject PureObjectReturningNull => null!;
+
+ public SomeObject PureObjectThrowingError => throw new Exception("Somethin went wrong");
+
+ public SomeObject? PureNullableObjectReturningNull => null;
+
+ #endregion
+
+ #region Object List
+
+ public Task ObjectListReturningNull()
+ {
+ return Task.FromResult(null!);
+ }
+
+ public Task ObjectListThrowingError()
+ {
+ throw new Exception("Something went wrong");
+ }
+
+ public Task NullableObjectListReturningNull()
+ {
+ return Task.FromResult(null);
+ }
+
+ public SomeObject[] PureObjectListReturningNull => null!;
+
+ public SomeObject[] PureObjectListThrowingError => throw new Exception("Somethin went wrong");
+
+ public SomeObject[]? PureNullableObjectListReturningNull => null;
+
+ #endregion
+
+ #region Object List Item
+
+ public Task ObjectListItemReturningNull()
+ {
+ return Task.FromResult([new("a"), null!, new("c")]);
+ }
+
+ public Task ObjectListItemThrowingError(IResolverContext context)
+ {
+ context.ReportError(ErrorBuilder.New().SetMessage("Another error").SetPath(context.Path.Append(1)).Build());
+ return Task.FromResult([new("a"), null!, new("c")]);
+ }
+
+ public Task NullableObjectListItemReturningNull()
+ {
+ return Task.FromResult([new("a"), null, new("c")]);
+ }
+
+ public SomeObject[] PureObjectListItemReturningNull => [new("a"), null!, new("c")];
+
+ // TODO: This is no longer a pure resolver as soon as it access the IResolverContext, right?
+ public SomeObject[] PureObjectListItemThrowingError(IResolverContext context)
+ {
+ context.ReportError(ErrorBuilder.New().SetMessage("Another error").SetPath(context.Path.Append(1)).Build());
+ return [new("a"), null!, new("c")];;
+ }
+
+ public SomeObject?[] PureNullableObjectListItemReturningNull => [new("a"), null, new("c")];
+
+ #endregion
+
+ #region Nested Array
+ public string?[][]? NestedScalarArrayNullableOuterItems()
+ {
+ return [["a1", null!, "c1"], null!, ["a2", null!, "c2"]];
+ }
+
+ public string[]?[] NestedScalarArrayNullableMiddleItem()
+ {
+ return [["a1", null!, "c1"], null!, ["a2", null!, "c2"]];
+ }
+ #endregion
+
+ [UsePaging]
+ public string[] ScalarConnection() => new[] { "a", null!, "c" };
+
+ [UsePaging]
+ public string?[] NullableScalarConnection() => new[] { "a", null, "c" };
+ }
+
+ public record SomeObject(string Property);
+
+ public class Mutation
+ {
+ [UseMutationConvention(PayloadFieldName = "scalarReturningNull")]
+ public Task SomeMutationReturningNull() => Task.FromResult(null!);
+ }
+}
diff --git a/src/HotChocolate/Core/test/Execution.Tests/__snapshots__/SemanticNonNullTests.Async_Nullable_Object_List_Item_Returns_Null_Should_Null_Item_Without_Error.snap b/src/HotChocolate/Core/test/Execution.Tests/__snapshots__/SemanticNonNullTests.Async_Nullable_Object_List_Item_Returns_Null_Should_Null_Item_Without_Error.snap
new file mode 100644
index 00000000000..5bf14273af3
--- /dev/null
+++ b/src/HotChocolate/Core/test/Execution.Tests/__snapshots__/SemanticNonNullTests.Async_Nullable_Object_List_Item_Returns_Null_Should_Null_Item_Without_Error.snap
@@ -0,0 +1,13 @@
+{
+ "data": {
+ "nullableObjectListItemReturningNull": [
+ {
+ "property": "a"
+ },
+ null,
+ {
+ "property": "c"
+ }
+ ]
+ }
+}
diff --git a/src/HotChocolate/Core/test/Execution.Tests/__snapshots__/SemanticNonNullTests.Async_Nullable_Object_List_Returns_Null_Should_Null_Without_Error.snap b/src/HotChocolate/Core/test/Execution.Tests/__snapshots__/SemanticNonNullTests.Async_Nullable_Object_List_Returns_Null_Should_Null_Without_Error.snap
new file mode 100644
index 00000000000..80512055ffc
--- /dev/null
+++ b/src/HotChocolate/Core/test/Execution.Tests/__snapshots__/SemanticNonNullTests.Async_Nullable_Object_List_Returns_Null_Should_Null_Without_Error.snap
@@ -0,0 +1,5 @@
+{
+ "data": {
+ "nullableObjectListReturningNull": null
+ }
+}
diff --git a/src/HotChocolate/Core/test/Execution.Tests/__snapshots__/SemanticNonNullTests.Async_Nullable_Object_Returns_Null_Should_Null_Without_Error.snap b/src/HotChocolate/Core/test/Execution.Tests/__snapshots__/SemanticNonNullTests.Async_Nullable_Object_Returns_Null_Should_Null_Without_Error.snap
new file mode 100644
index 00000000000..d6ed49ca517
--- /dev/null
+++ b/src/HotChocolate/Core/test/Execution.Tests/__snapshots__/SemanticNonNullTests.Async_Nullable_Object_Returns_Null_Should_Null_Without_Error.snap
@@ -0,0 +1,5 @@
+{
+ "data": {
+ "nullableObjectReturningNull": null
+ }
+}
diff --git a/src/HotChocolate/Core/test/Execution.Tests/__snapshots__/SemanticNonNullTests.Async_Nullable_Scalar_List_Item_Returns_Null_Should_Null_Item_Without_Error.snap b/src/HotChocolate/Core/test/Execution.Tests/__snapshots__/SemanticNonNullTests.Async_Nullable_Scalar_List_Item_Returns_Null_Should_Null_Item_Without_Error.snap
new file mode 100644
index 00000000000..b3c51911a4c
--- /dev/null
+++ b/src/HotChocolate/Core/test/Execution.Tests/__snapshots__/SemanticNonNullTests.Async_Nullable_Scalar_List_Item_Returns_Null_Should_Null_Item_Without_Error.snap
@@ -0,0 +1,9 @@
+{
+ "data": {
+ "nullableScalarListItemReturningNull": [
+ "a",
+ null,
+ "c"
+ ]
+ }
+}
diff --git a/src/HotChocolate/Core/test/Execution.Tests/__snapshots__/SemanticNonNullTests.Async_Nullable_Scalar_List_Returns_Null_Should_Null_Without_Error.snap b/src/HotChocolate/Core/test/Execution.Tests/__snapshots__/SemanticNonNullTests.Async_Nullable_Scalar_List_Returns_Null_Should_Null_Without_Error.snap
new file mode 100644
index 00000000000..68f0bbd3b50
--- /dev/null
+++ b/src/HotChocolate/Core/test/Execution.Tests/__snapshots__/SemanticNonNullTests.Async_Nullable_Scalar_List_Returns_Null_Should_Null_Without_Error.snap
@@ -0,0 +1,5 @@
+{
+ "data": {
+ "nullableScalarListReturningNull": null
+ }
+}
diff --git a/src/HotChocolate/Core/test/Execution.Tests/__snapshots__/SemanticNonNullTests.Async_Nullable_Scalar_Returns_Null_Should_Null_Without_Error.snap b/src/HotChocolate/Core/test/Execution.Tests/__snapshots__/SemanticNonNullTests.Async_Nullable_Scalar_Returns_Null_Should_Null_Without_Error.snap
new file mode 100644
index 00000000000..161eb3e36df
--- /dev/null
+++ b/src/HotChocolate/Core/test/Execution.Tests/__snapshots__/SemanticNonNullTests.Async_Nullable_Scalar_Returns_Null_Should_Null_Without_Error.snap
@@ -0,0 +1,5 @@
+{
+ "data": {
+ "nullableScalarReturningNull": null
+ }
+}
diff --git a/src/HotChocolate/Core/test/Execution.Tests/__snapshots__/SemanticNonNullTests.Async_Object_List_Item_Returns_Null_Should_Error_Item.snap b/src/HotChocolate/Core/test/Execution.Tests/__snapshots__/SemanticNonNullTests.Async_Object_List_Item_Returns_Null_Should_Error_Item.snap
new file mode 100644
index 00000000000..cef5849184d
--- /dev/null
+++ b/src/HotChocolate/Core/test/Execution.Tests/__snapshots__/SemanticNonNullTests.Async_Object_List_Item_Returns_Null_Should_Error_Item.snap
@@ -0,0 +1,31 @@
+{
+ "errors": [
+ {
+ "message": "Cannot return null for semantic non-null field.",
+ "locations": [
+ {
+ "line": 2,
+ "column": 3
+ }
+ ],
+ "path": [
+ "objectListItemReturningNull",
+ 1
+ ],
+ "extensions": {
+ "code": "HC0088"
+ }
+ }
+ ],
+ "data": {
+ "objectListItemReturningNull": [
+ {
+ "property": "a"
+ },
+ null,
+ {
+ "property": "c"
+ }
+ ]
+ }
+}
diff --git a/src/HotChocolate/Core/test/Execution.Tests/__snapshots__/SemanticNonNullTests.Async_Object_List_Item_Throwing_Should_Null_And_Error_Item.snap b/src/HotChocolate/Core/test/Execution.Tests/__snapshots__/SemanticNonNullTests.Async_Object_List_Item_Throwing_Should_Null_And_Error_Item.snap
new file mode 100644
index 00000000000..c136aaacabf
--- /dev/null
+++ b/src/HotChocolate/Core/test/Execution.Tests/__snapshots__/SemanticNonNullTests.Async_Object_List_Item_Throwing_Should_Null_And_Error_Item.snap
@@ -0,0 +1,44 @@
+{
+ "errors": [
+ {
+ "message": "Another error",
+ "locations": [
+ {
+ "line": 2,
+ "column": 3
+ }
+ ],
+ "path": [
+ "objectListItemThrowingError",
+ 1
+ ]
+ },
+ {
+ "message": "Cannot return null for semantic non-null field.",
+ "locations": [
+ {
+ "line": 2,
+ "column": 3
+ }
+ ],
+ "path": [
+ "objectListItemThrowingError",
+ 1
+ ],
+ "extensions": {
+ "code": "HC0088"
+ }
+ }
+ ],
+ "data": {
+ "objectListItemThrowingError": [
+ {
+ "property": "a"
+ },
+ null,
+ {
+ "property": "c"
+ }
+ ]
+ }
+}
diff --git a/src/HotChocolate/Core/test/Execution.Tests/__snapshots__/SemanticNonNullTests.Async_Object_List_Returns_Null_Should_Error.snap b/src/HotChocolate/Core/test/Execution.Tests/__snapshots__/SemanticNonNullTests.Async_Object_List_Returns_Null_Should_Error.snap
new file mode 100644
index 00000000000..5272b1193ac
--- /dev/null
+++ b/src/HotChocolate/Core/test/Execution.Tests/__snapshots__/SemanticNonNullTests.Async_Object_List_Returns_Null_Should_Error.snap
@@ -0,0 +1,22 @@
+{
+ "errors": [
+ {
+ "message": "Cannot return null for semantic non-null field.",
+ "locations": [
+ {
+ "line": 2,
+ "column": 3
+ }
+ ],
+ "path": [
+ "objectListReturningNull"
+ ],
+ "extensions": {
+ "code": "HC0088"
+ }
+ }
+ ],
+ "data": {
+ "objectListReturningNull": null
+ }
+}
diff --git a/src/HotChocolate/Core/test/Execution.Tests/__snapshots__/SemanticNonNullTests.Async_Object_List_Throwing_Should_Null_FAnd_Error.snap b/src/HotChocolate/Core/test/Execution.Tests/__snapshots__/SemanticNonNullTests.Async_Object_List_Throwing_Should_Null_FAnd_Error.snap
new file mode 100644
index 00000000000..584bc579f4d
--- /dev/null
+++ b/src/HotChocolate/Core/test/Execution.Tests/__snapshots__/SemanticNonNullTests.Async_Object_List_Throwing_Should_Null_FAnd_Error.snap
@@ -0,0 +1,19 @@
+{
+ "errors": [
+ {
+ "message": "Unexpected Execution Error",
+ "locations": [
+ {
+ "line": 2,
+ "column": 3
+ }
+ ],
+ "path": [
+ "objectListThrowingError"
+ ]
+ }
+ ],
+ "data": {
+ "objectListThrowingError": null
+ }
+}
diff --git a/src/HotChocolate/Core/test/Execution.Tests/__snapshots__/SemanticNonNullTests.Async_Object_Returns_Null_Should_Error.snap b/src/HotChocolate/Core/test/Execution.Tests/__snapshots__/SemanticNonNullTests.Async_Object_Returns_Null_Should_Error.snap
new file mode 100644
index 00000000000..fa19729099c
--- /dev/null
+++ b/src/HotChocolate/Core/test/Execution.Tests/__snapshots__/SemanticNonNullTests.Async_Object_Returns_Null_Should_Error.snap
@@ -0,0 +1,22 @@
+{
+ "errors": [
+ {
+ "message": "Cannot return null for semantic non-null field.",
+ "locations": [
+ {
+ "line": 2,
+ "column": 3
+ }
+ ],
+ "path": [
+ "objectReturningNull"
+ ],
+ "extensions": {
+ "code": "HC0088"
+ }
+ }
+ ],
+ "data": {
+ "objectReturningNull": null
+ }
+}
diff --git a/src/HotChocolate/Core/test/Execution.Tests/__snapshots__/SemanticNonNullTests.Async_Object_Throwing_Should_Null_And_Error.snap b/src/HotChocolate/Core/test/Execution.Tests/__snapshots__/SemanticNonNullTests.Async_Object_Throwing_Should_Null_And_Error.snap
new file mode 100644
index 00000000000..de64fcf0f25
--- /dev/null
+++ b/src/HotChocolate/Core/test/Execution.Tests/__snapshots__/SemanticNonNullTests.Async_Object_Throwing_Should_Null_And_Error.snap
@@ -0,0 +1,19 @@
+{
+ "errors": [
+ {
+ "message": "Unexpected Execution Error",
+ "locations": [
+ {
+ "line": 2,
+ "column": 3
+ }
+ ],
+ "path": [
+ "objectThrowingError"
+ ]
+ }
+ ],
+ "data": {
+ "objectThrowingError": null
+ }
+}
diff --git a/src/HotChocolate/Core/test/Execution.Tests/__snapshots__/SemanticNonNullTests.Async_Scalar_List_Item_Returns_Null_Should_Error_Item.snap b/src/HotChocolate/Core/test/Execution.Tests/__snapshots__/SemanticNonNullTests.Async_Scalar_List_Item_Returns_Null_Should_Error_Item.snap
new file mode 100644
index 00000000000..6e3b5178b29
--- /dev/null
+++ b/src/HotChocolate/Core/test/Execution.Tests/__snapshots__/SemanticNonNullTests.Async_Scalar_List_Item_Returns_Null_Should_Error_Item.snap
@@ -0,0 +1,27 @@
+{
+ "errors": [
+ {
+ "message": "Cannot return null for semantic non-null field.",
+ "locations": [
+ {
+ "line": 2,
+ "column": 3
+ }
+ ],
+ "path": [
+ "scalarListItemReturningNull",
+ 1
+ ],
+ "extensions": {
+ "code": "HC0088"
+ }
+ }
+ ],
+ "data": {
+ "scalarListItemReturningNull": [
+ "a",
+ null,
+ "c"
+ ]
+ }
+}
diff --git a/src/HotChocolate/Core/test/Execution.Tests/__snapshots__/SemanticNonNullTests.Async_Scalar_List_Item_Throwing_Should_Null_And_Error_Item.snap b/src/HotChocolate/Core/test/Execution.Tests/__snapshots__/SemanticNonNullTests.Async_Scalar_List_Item_Throwing_Should_Null_And_Error_Item.snap
new file mode 100644
index 00000000000..87f194c0037
--- /dev/null
+++ b/src/HotChocolate/Core/test/Execution.Tests/__snapshots__/SemanticNonNullTests.Async_Scalar_List_Item_Throwing_Should_Null_And_Error_Item.snap
@@ -0,0 +1,40 @@
+{
+ "errors": [
+ {
+ "message": "Another error",
+ "locations": [
+ {
+ "line": 2,
+ "column": 3
+ }
+ ],
+ "path": [
+ "scalarListItemThrowingError",
+ 1
+ ]
+ },
+ {
+ "message": "Cannot return null for semantic non-null field.",
+ "locations": [
+ {
+ "line": 2,
+ "column": 3
+ }
+ ],
+ "path": [
+ "scalarListItemThrowingError",
+ 1
+ ],
+ "extensions": {
+ "code": "HC0088"
+ }
+ }
+ ],
+ "data": {
+ "scalarListItemThrowingError": [
+ "a",
+ null,
+ "c"
+ ]
+ }
+}
diff --git a/src/HotChocolate/Core/test/Execution.Tests/__snapshots__/SemanticNonNullTests.Async_Scalar_List_Returns_Null_Should_Error.snap b/src/HotChocolate/Core/test/Execution.Tests/__snapshots__/SemanticNonNullTests.Async_Scalar_List_Returns_Null_Should_Error.snap
new file mode 100644
index 00000000000..136f20e321e
--- /dev/null
+++ b/src/HotChocolate/Core/test/Execution.Tests/__snapshots__/SemanticNonNullTests.Async_Scalar_List_Returns_Null_Should_Error.snap
@@ -0,0 +1,22 @@
+{
+ "errors": [
+ {
+ "message": "Cannot return null for semantic non-null field.",
+ "locations": [
+ {
+ "line": 2,
+ "column": 3
+ }
+ ],
+ "path": [
+ "scalarListReturningNull"
+ ],
+ "extensions": {
+ "code": "HC0088"
+ }
+ }
+ ],
+ "data": {
+ "scalarListReturningNull": null
+ }
+}
diff --git a/src/HotChocolate/Core/test/Execution.Tests/__snapshots__/SemanticNonNullTests.Async_Scalar_List_Throwing_Should_Null_And_Error.snap b/src/HotChocolate/Core/test/Execution.Tests/__snapshots__/SemanticNonNullTests.Async_Scalar_List_Throwing_Should_Null_And_Error.snap
new file mode 100644
index 00000000000..fe169a3cc33
--- /dev/null
+++ b/src/HotChocolate/Core/test/Execution.Tests/__snapshots__/SemanticNonNullTests.Async_Scalar_List_Throwing_Should_Null_And_Error.snap
@@ -0,0 +1,19 @@
+{
+ "errors": [
+ {
+ "message": "Unexpected Execution Error",
+ "locations": [
+ {
+ "line": 2,
+ "column": 3
+ }
+ ],
+ "path": [
+ "scalarListThrowingError"
+ ]
+ }
+ ],
+ "data": {
+ "scalarListThrowingError": null
+ }
+}
diff --git a/src/HotChocolate/Core/test/Execution.Tests/__snapshots__/SemanticNonNullTests.Async_Scalar_Returns_Null_Should_Error.snap b/src/HotChocolate/Core/test/Execution.Tests/__snapshots__/SemanticNonNullTests.Async_Scalar_Returns_Null_Should_Error.snap
new file mode 100644
index 00000000000..d6f046b374e
--- /dev/null
+++ b/src/HotChocolate/Core/test/Execution.Tests/__snapshots__/SemanticNonNullTests.Async_Scalar_Returns_Null_Should_Error.snap
@@ -0,0 +1,22 @@
+{
+ "errors": [
+ {
+ "message": "Cannot return null for semantic non-null field.",
+ "locations": [
+ {
+ "line": 2,
+ "column": 3
+ }
+ ],
+ "path": [
+ "scalarReturningNull"
+ ],
+ "extensions": {
+ "code": "HC0088"
+ }
+ }
+ ],
+ "data": {
+ "scalarReturningNull": null
+ }
+}
diff --git a/src/HotChocolate/Core/test/Execution.Tests/__snapshots__/SemanticNonNullTests.Async_Scalar_Throwing_Should_Null_And_Error.snap b/src/HotChocolate/Core/test/Execution.Tests/__snapshots__/SemanticNonNullTests.Async_Scalar_Throwing_Should_Null_And_Error.snap
new file mode 100644
index 00000000000..f992ef3496b
--- /dev/null
+++ b/src/HotChocolate/Core/test/Execution.Tests/__snapshots__/SemanticNonNullTests.Async_Scalar_Throwing_Should_Null_And_Error.snap
@@ -0,0 +1,19 @@
+{
+ "errors": [
+ {
+ "message": "Unexpected Execution Error",
+ "locations": [
+ {
+ "line": 2,
+ "column": 3
+ }
+ ],
+ "path": [
+ "scalarThrowingError"
+ ]
+ }
+ ],
+ "data": {
+ "scalarThrowingError": null
+ }
+}
diff --git a/src/HotChocolate/Core/test/Execution.Tests/__snapshots__/SemanticNonNullTests.Mutation_With_MutationConventions.snap b/src/HotChocolate/Core/test/Execution.Tests/__snapshots__/SemanticNonNullTests.Mutation_With_MutationConventions.snap
new file mode 100644
index 00000000000..bd13441af00
--- /dev/null
+++ b/src/HotChocolate/Core/test/Execution.Tests/__snapshots__/SemanticNonNullTests.Mutation_With_MutationConventions.snap
@@ -0,0 +1,7 @@
+{
+ "data": {
+ "someMutationReturningNull": {
+ "scalarReturningNull": null
+ }
+ }
+}
diff --git a/src/HotChocolate/Core/test/Execution.Tests/__snapshots__/SemanticNonNullTests.Pure_Nullable_Object_List_Item_Returns_Null_Should_Null_Item_Without_Error.snap b/src/HotChocolate/Core/test/Execution.Tests/__snapshots__/SemanticNonNullTests.Pure_Nullable_Object_List_Item_Returns_Null_Should_Null_Item_Without_Error.snap
new file mode 100644
index 00000000000..bc867086e8a
--- /dev/null
+++ b/src/HotChocolate/Core/test/Execution.Tests/__snapshots__/SemanticNonNullTests.Pure_Nullable_Object_List_Item_Returns_Null_Should_Null_Item_Without_Error.snap
@@ -0,0 +1,13 @@
+{
+ "data": {
+ "pureNullableObjectListItemReturningNull": [
+ {
+ "property": "a"
+ },
+ null,
+ {
+ "property": "c"
+ }
+ ]
+ }
+}
diff --git a/src/HotChocolate/Core/test/Execution.Tests/__snapshots__/SemanticNonNullTests.Pure_Nullable_Object_List_Returns_Null_Should_Null_Without_Error.snap b/src/HotChocolate/Core/test/Execution.Tests/__snapshots__/SemanticNonNullTests.Pure_Nullable_Object_List_Returns_Null_Should_Null_Without_Error.snap
new file mode 100644
index 00000000000..2abd30023ac
--- /dev/null
+++ b/src/HotChocolate/Core/test/Execution.Tests/__snapshots__/SemanticNonNullTests.Pure_Nullable_Object_List_Returns_Null_Should_Null_Without_Error.snap
@@ -0,0 +1,5 @@
+{
+ "data": {
+ "pureNullableObjectListReturningNull": null
+ }
+}
diff --git a/src/HotChocolate/Core/test/Execution.Tests/__snapshots__/SemanticNonNullTests.Pure_Nullable_Object_Returns_Null_Should_Null_Without_Error.snap b/src/HotChocolate/Core/test/Execution.Tests/__snapshots__/SemanticNonNullTests.Pure_Nullable_Object_Returns_Null_Should_Null_Without_Error.snap
new file mode 100644
index 00000000000..472da5015a4
--- /dev/null
+++ b/src/HotChocolate/Core/test/Execution.Tests/__snapshots__/SemanticNonNullTests.Pure_Nullable_Object_Returns_Null_Should_Null_Without_Error.snap
@@ -0,0 +1,5 @@
+{
+ "data": {
+ "pureNullableObjectReturningNull": null
+ }
+}
diff --git a/src/HotChocolate/Core/test/Execution.Tests/__snapshots__/SemanticNonNullTests.Pure_Nullable_Scalar_List_Item_Returns_Null_Should_Null_Item_Without_Error.snap b/src/HotChocolate/Core/test/Execution.Tests/__snapshots__/SemanticNonNullTests.Pure_Nullable_Scalar_List_Item_Returns_Null_Should_Null_Item_Without_Error.snap
new file mode 100644
index 00000000000..65e0c7452b1
--- /dev/null
+++ b/src/HotChocolate/Core/test/Execution.Tests/__snapshots__/SemanticNonNullTests.Pure_Nullable_Scalar_List_Item_Returns_Null_Should_Null_Item_Without_Error.snap
@@ -0,0 +1,9 @@
+{
+ "data": {
+ "pureNullableScalarListItemReturningNull": [
+ "a",
+ null,
+ "c"
+ ]
+ }
+}
diff --git a/src/HotChocolate/Core/test/Execution.Tests/__snapshots__/SemanticNonNullTests.Pure_Nullable_Scalar_List_Returns_Null_Should_Null_Without_Error.snap b/src/HotChocolate/Core/test/Execution.Tests/__snapshots__/SemanticNonNullTests.Pure_Nullable_Scalar_List_Returns_Null_Should_Null_Without_Error.snap
new file mode 100644
index 00000000000..859a7fd908e
--- /dev/null
+++ b/src/HotChocolate/Core/test/Execution.Tests/__snapshots__/SemanticNonNullTests.Pure_Nullable_Scalar_List_Returns_Null_Should_Null_Without_Error.snap
@@ -0,0 +1,5 @@
+{
+ "data": {
+ "pureNullableScalarListReturningNull": null
+ }
+}
diff --git a/src/HotChocolate/Core/test/Execution.Tests/__snapshots__/SemanticNonNullTests.Pure_Nullable_Scalar_Returns_Null_Should_Null_Without_Error.snap b/src/HotChocolate/Core/test/Execution.Tests/__snapshots__/SemanticNonNullTests.Pure_Nullable_Scalar_Returns_Null_Should_Null_Without_Error.snap
new file mode 100644
index 00000000000..e05006cc33f
--- /dev/null
+++ b/src/HotChocolate/Core/test/Execution.Tests/__snapshots__/SemanticNonNullTests.Pure_Nullable_Scalar_Returns_Null_Should_Null_Without_Error.snap
@@ -0,0 +1,5 @@
+{
+ "data": {
+ "pureNullableScalarReturningNull": null
+ }
+}
diff --git a/src/HotChocolate/Core/test/Execution.Tests/__snapshots__/SemanticNonNullTests.Pure_Object_List_Item_Returns_Null_Should_Error_Item.snap b/src/HotChocolate/Core/test/Execution.Tests/__snapshots__/SemanticNonNullTests.Pure_Object_List_Item_Returns_Null_Should_Error_Item.snap
new file mode 100644
index 00000000000..20741ef4898
--- /dev/null
+++ b/src/HotChocolate/Core/test/Execution.Tests/__snapshots__/SemanticNonNullTests.Pure_Object_List_Item_Returns_Null_Should_Error_Item.snap
@@ -0,0 +1,31 @@
+{
+ "errors": [
+ {
+ "message": "Cannot return null for semantic non-null field.",
+ "locations": [
+ {
+ "line": 2,
+ "column": 3
+ }
+ ],
+ "path": [
+ "pureObjectListItemReturningNull",
+ 1
+ ],
+ "extensions": {
+ "code": "HC0088"
+ }
+ }
+ ],
+ "data": {
+ "pureObjectListItemReturningNull": [
+ {
+ "property": "a"
+ },
+ null,
+ {
+ "property": "c"
+ }
+ ]
+ }
+}
diff --git a/src/HotChocolate/Core/test/Execution.Tests/__snapshots__/SemanticNonNullTests.Pure_Object_List_Item_Throwing_Should_Null_And_Error_Item.snap b/src/HotChocolate/Core/test/Execution.Tests/__snapshots__/SemanticNonNullTests.Pure_Object_List_Item_Throwing_Should_Null_And_Error_Item.snap
new file mode 100644
index 00000000000..338ea110d60
--- /dev/null
+++ b/src/HotChocolate/Core/test/Execution.Tests/__snapshots__/SemanticNonNullTests.Pure_Object_List_Item_Throwing_Should_Null_And_Error_Item.snap
@@ -0,0 +1,44 @@
+{
+ "errors": [
+ {
+ "message": "Another error",
+ "locations": [
+ {
+ "line": 2,
+ "column": 3
+ }
+ ],
+ "path": [
+ "pureObjectListItemThrowingError",
+ 1
+ ]
+ },
+ {
+ "message": "Cannot return null for semantic non-null field.",
+ "locations": [
+ {
+ "line": 2,
+ "column": 3
+ }
+ ],
+ "path": [
+ "pureObjectListItemThrowingError",
+ 1
+ ],
+ "extensions": {
+ "code": "HC0088"
+ }
+ }
+ ],
+ "data": {
+ "pureObjectListItemThrowingError": [
+ {
+ "property": "a"
+ },
+ null,
+ {
+ "property": "c"
+ }
+ ]
+ }
+}
diff --git a/src/HotChocolate/Core/test/Execution.Tests/__snapshots__/SemanticNonNullTests.Pure_Object_List_Returns_Null_Should_Error.snap b/src/HotChocolate/Core/test/Execution.Tests/__snapshots__/SemanticNonNullTests.Pure_Object_List_Returns_Null_Should_Error.snap
new file mode 100644
index 00000000000..dd9c523f560
--- /dev/null
+++ b/src/HotChocolate/Core/test/Execution.Tests/__snapshots__/SemanticNonNullTests.Pure_Object_List_Returns_Null_Should_Error.snap
@@ -0,0 +1,22 @@
+{
+ "errors": [
+ {
+ "message": "Cannot return null for semantic non-null field.",
+ "locations": [
+ {
+ "line": 2,
+ "column": 3
+ }
+ ],
+ "path": [
+ "pureObjectListReturningNull"
+ ],
+ "extensions": {
+ "code": "HC0088"
+ }
+ }
+ ],
+ "data": {
+ "pureObjectListReturningNull": null
+ }
+}
diff --git a/src/HotChocolate/Core/test/Execution.Tests/__snapshots__/SemanticNonNullTests.Pure_Object_List_Throwing_Should_Null_FAnd_Error.snap b/src/HotChocolate/Core/test/Execution.Tests/__snapshots__/SemanticNonNullTests.Pure_Object_List_Throwing_Should_Null_FAnd_Error.snap
new file mode 100644
index 00000000000..89587618619
--- /dev/null
+++ b/src/HotChocolate/Core/test/Execution.Tests/__snapshots__/SemanticNonNullTests.Pure_Object_List_Throwing_Should_Null_FAnd_Error.snap
@@ -0,0 +1,19 @@
+{
+ "errors": [
+ {
+ "message": "Unexpected Execution Error",
+ "locations": [
+ {
+ "line": 2,
+ "column": 3
+ }
+ ],
+ "path": [
+ "pureObjectListThrowingError"
+ ]
+ }
+ ],
+ "data": {
+ "pureObjectListThrowingError": null
+ }
+}
diff --git a/src/HotChocolate/Core/test/Execution.Tests/__snapshots__/SemanticNonNullTests.Pure_Object_Returns_Null_Should_Error.snap b/src/HotChocolate/Core/test/Execution.Tests/__snapshots__/SemanticNonNullTests.Pure_Object_Returns_Null_Should_Error.snap
new file mode 100644
index 00000000000..81b0b6ca595
--- /dev/null
+++ b/src/HotChocolate/Core/test/Execution.Tests/__snapshots__/SemanticNonNullTests.Pure_Object_Returns_Null_Should_Error.snap
@@ -0,0 +1,22 @@
+{
+ "errors": [
+ {
+ "message": "Cannot return null for semantic non-null field.",
+ "locations": [
+ {
+ "line": 2,
+ "column": 3
+ }
+ ],
+ "path": [
+ "pureObjectReturningNull"
+ ],
+ "extensions": {
+ "code": "HC0088"
+ }
+ }
+ ],
+ "data": {
+ "pureObjectReturningNull": null
+ }
+}
diff --git a/src/HotChocolate/Core/test/Execution.Tests/__snapshots__/SemanticNonNullTests.Pure_Object_Throwing_Should_Null_And_Error.snap b/src/HotChocolate/Core/test/Execution.Tests/__snapshots__/SemanticNonNullTests.Pure_Object_Throwing_Should_Null_And_Error.snap
new file mode 100644
index 00000000000..76230f78282
--- /dev/null
+++ b/src/HotChocolate/Core/test/Execution.Tests/__snapshots__/SemanticNonNullTests.Pure_Object_Throwing_Should_Null_And_Error.snap
@@ -0,0 +1,19 @@
+{
+ "errors": [
+ {
+ "message": "Unexpected Execution Error",
+ "locations": [
+ {
+ "line": 2,
+ "column": 3
+ }
+ ],
+ "path": [
+ "pureObjectThrowingError"
+ ]
+ }
+ ],
+ "data": {
+ "pureObjectThrowingError": null
+ }
+}
diff --git a/src/HotChocolate/Core/test/Execution.Tests/__snapshots__/SemanticNonNullTests.Pure_Scalar_ListOfList_Nullable_Middle_Item_Outer_And_Inner_Return_Null_Should_Null_And_Error.snap b/src/HotChocolate/Core/test/Execution.Tests/__snapshots__/SemanticNonNullTests.Pure_Scalar_ListOfList_Nullable_Middle_Item_Outer_And_Inner_Return_Null_Should_Null_And_Error.snap
new file mode 100644
index 00000000000..7c42ba3a2a3
--- /dev/null
+++ b/src/HotChocolate/Core/test/Execution.Tests/__snapshots__/SemanticNonNullTests.Pure_Scalar_ListOfList_Nullable_Middle_Item_Outer_And_Inner_Return_Null_Should_Null_And_Error.snap
@@ -0,0 +1,53 @@
+{
+ "errors": [
+ {
+ "message": "Cannot return null for semantic non-null field.",
+ "locations": [
+ {
+ "line": 2,
+ "column": 3
+ }
+ ],
+ "path": [
+ "nestedScalarArrayNullableMiddleItem",
+ 0,
+ 1
+ ],
+ "extensions": {
+ "code": "HC0088"
+ }
+ },
+ {
+ "message": "Cannot return null for semantic non-null field.",
+ "locations": [
+ {
+ "line": 2,
+ "column": 3
+ }
+ ],
+ "path": [
+ "nestedScalarArrayNullableMiddleItem",
+ 2,
+ 1
+ ],
+ "extensions": {
+ "code": "HC0088"
+ }
+ }
+ ],
+ "data": {
+ "nestedScalarArrayNullableMiddleItem": [
+ [
+ "a1",
+ null,
+ "c1"
+ ],
+ null,
+ [
+ "a2",
+ null,
+ "c2"
+ ]
+ ]
+ }
+}
diff --git a/src/HotChocolate/Core/test/Execution.Tests/__snapshots__/SemanticNonNullTests.Pure_Scalar_ListOfList_Nullable_Outer_And_Inner_Middle_Returns_Null_Should_Null_And_Error.snap b/src/HotChocolate/Core/test/Execution.Tests/__snapshots__/SemanticNonNullTests.Pure_Scalar_ListOfList_Nullable_Outer_And_Inner_Middle_Returns_Null_Should_Null_And_Error.snap
new file mode 100644
index 00000000000..5b68ea3fafb
--- /dev/null
+++ b/src/HotChocolate/Core/test/Execution.Tests/__snapshots__/SemanticNonNullTests.Pure_Scalar_ListOfList_Nullable_Outer_And_Inner_Middle_Returns_Null_Should_Null_And_Error.snap
@@ -0,0 +1,35 @@
+{
+ "errors": [
+ {
+ "message": "Cannot return null for semantic non-null field.",
+ "locations": [
+ {
+ "line": 2,
+ "column": 3
+ }
+ ],
+ "path": [
+ "nestedScalarArrayNullableOuterItems",
+ 1
+ ],
+ "extensions": {
+ "code": "HC0088"
+ }
+ }
+ ],
+ "data": {
+ "nestedScalarArrayNullableOuterItems": [
+ [
+ "a1",
+ null,
+ "c1"
+ ],
+ null,
+ [
+ "a2",
+ null,
+ "c2"
+ ]
+ ]
+ }
+}
diff --git a/src/HotChocolate/Core/test/Execution.Tests/__snapshots__/SemanticNonNullTests.Pure_Scalar_List_Item_Returns_Null_Should_Error_Item.snap b/src/HotChocolate/Core/test/Execution.Tests/__snapshots__/SemanticNonNullTests.Pure_Scalar_List_Item_Returns_Null_Should_Error_Item.snap
new file mode 100644
index 00000000000..f3a0b0bb1ca
--- /dev/null
+++ b/src/HotChocolate/Core/test/Execution.Tests/__snapshots__/SemanticNonNullTests.Pure_Scalar_List_Item_Returns_Null_Should_Error_Item.snap
@@ -0,0 +1,27 @@
+{
+ "errors": [
+ {
+ "message": "Cannot return null for semantic non-null field.",
+ "locations": [
+ {
+ "line": 2,
+ "column": 3
+ }
+ ],
+ "path": [
+ "pureScalarListItemReturningNull",
+ 1
+ ],
+ "extensions": {
+ "code": "HC0088"
+ }
+ }
+ ],
+ "data": {
+ "pureScalarListItemReturningNull": [
+ "a",
+ null,
+ "c"
+ ]
+ }
+}
diff --git a/src/HotChocolate/Core/test/Execution.Tests/__snapshots__/SemanticNonNullTests.Pure_Scalar_List_Item_Throwing_Should_Null_And_Error_Item.snap b/src/HotChocolate/Core/test/Execution.Tests/__snapshots__/SemanticNonNullTests.Pure_Scalar_List_Item_Throwing_Should_Null_And_Error_Item.snap
new file mode 100644
index 00000000000..e5ecb022752
--- /dev/null
+++ b/src/HotChocolate/Core/test/Execution.Tests/__snapshots__/SemanticNonNullTests.Pure_Scalar_List_Item_Throwing_Should_Null_And_Error_Item.snap
@@ -0,0 +1,40 @@
+{
+ "errors": [
+ {
+ "message": "Another error",
+ "locations": [
+ {
+ "line": 2,
+ "column": 3
+ }
+ ],
+ "path": [
+ "pureScalarListItemThrowingError",
+ 1
+ ]
+ },
+ {
+ "message": "Cannot return null for semantic non-null field.",
+ "locations": [
+ {
+ "line": 2,
+ "column": 3
+ }
+ ],
+ "path": [
+ "pureScalarListItemThrowingError",
+ 1
+ ],
+ "extensions": {
+ "code": "HC0088"
+ }
+ }
+ ],
+ "data": {
+ "pureScalarListItemThrowingError": [
+ "a",
+ null,
+ "c"
+ ]
+ }
+}
diff --git a/src/HotChocolate/Core/test/Execution.Tests/__snapshots__/SemanticNonNullTests.Pure_Scalar_List_Returns_Null_Should_Error.snap b/src/HotChocolate/Core/test/Execution.Tests/__snapshots__/SemanticNonNullTests.Pure_Scalar_List_Returns_Null_Should_Error.snap
new file mode 100644
index 00000000000..e6c89302539
--- /dev/null
+++ b/src/HotChocolate/Core/test/Execution.Tests/__snapshots__/SemanticNonNullTests.Pure_Scalar_List_Returns_Null_Should_Error.snap
@@ -0,0 +1,22 @@
+{
+ "errors": [
+ {
+ "message": "Cannot return null for semantic non-null field.",
+ "locations": [
+ {
+ "line": 2,
+ "column": 3
+ }
+ ],
+ "path": [
+ "pureScalarListReturningNull"
+ ],
+ "extensions": {
+ "code": "HC0088"
+ }
+ }
+ ],
+ "data": {
+ "pureScalarListReturningNull": null
+ }
+}
diff --git a/src/HotChocolate/Core/test/Execution.Tests/__snapshots__/SemanticNonNullTests.Pure_Scalar_List_Throwing_Should_Null_And_Error.snap b/src/HotChocolate/Core/test/Execution.Tests/__snapshots__/SemanticNonNullTests.Pure_Scalar_List_Throwing_Should_Null_And_Error.snap
new file mode 100644
index 00000000000..87eb04d1690
--- /dev/null
+++ b/src/HotChocolate/Core/test/Execution.Tests/__snapshots__/SemanticNonNullTests.Pure_Scalar_List_Throwing_Should_Null_And_Error.snap
@@ -0,0 +1,19 @@
+{
+ "errors": [
+ {
+ "message": "Unexpected Execution Error",
+ "locations": [
+ {
+ "line": 2,
+ "column": 3
+ }
+ ],
+ "path": [
+ "pureScalarListThrowingError"
+ ]
+ }
+ ],
+ "data": {
+ "pureScalarListThrowingError": null
+ }
+}
diff --git a/src/HotChocolate/Core/test/Execution.Tests/__snapshots__/SemanticNonNullTests.Pure_Scalar_Returns_Null_Should_Error.snap b/src/HotChocolate/Core/test/Execution.Tests/__snapshots__/SemanticNonNullTests.Pure_Scalar_Returns_Null_Should_Error.snap
new file mode 100644
index 00000000000..4ddf157a011
--- /dev/null
+++ b/src/HotChocolate/Core/test/Execution.Tests/__snapshots__/SemanticNonNullTests.Pure_Scalar_Returns_Null_Should_Error.snap
@@ -0,0 +1,22 @@
+{
+ "errors": [
+ {
+ "message": "Cannot return null for semantic non-null field.",
+ "locations": [
+ {
+ "line": 2,
+ "column": 3
+ }
+ ],
+ "path": [
+ "pureScalarReturningNull"
+ ],
+ "extensions": {
+ "code": "HC0088"
+ }
+ }
+ ],
+ "data": {
+ "pureScalarReturningNull": null
+ }
+}
diff --git a/src/HotChocolate/Core/test/Execution.Tests/__snapshots__/SemanticNonNullTests.Pure_Scalar_Throwing_Should_Null_And_Error.snap b/src/HotChocolate/Core/test/Execution.Tests/__snapshots__/SemanticNonNullTests.Pure_Scalar_Throwing_Should_Null_And_Error.snap
new file mode 100644
index 00000000000..a1980169f92
--- /dev/null
+++ b/src/HotChocolate/Core/test/Execution.Tests/__snapshots__/SemanticNonNullTests.Pure_Scalar_Throwing_Should_Null_And_Error.snap
@@ -0,0 +1,19 @@
+{
+ "errors": [
+ {
+ "message": "Unexpected Execution Error",
+ "locations": [
+ {
+ "line": 2,
+ "column": 3
+ }
+ ],
+ "path": [
+ "pureScalarThrowingError"
+ ]
+ }
+ ],
+ "data": {
+ "pureScalarThrowingError": null
+ }
+}
diff --git a/src/HotChocolate/Core/test/Execution.Tests/__snapshots__/SemanticNonNullTests.Query_With_Connection.snap b/src/HotChocolate/Core/test/Execution.Tests/__snapshots__/SemanticNonNullTests.Query_With_Connection.snap
new file mode 100644
index 00000000000..af087216eda
--- /dev/null
+++ b/src/HotChocolate/Core/test/Execution.Tests/__snapshots__/SemanticNonNullTests.Query_With_Connection.snap
@@ -0,0 +1,34 @@
+{
+ "errors": [
+ {
+ "message": "Unexpected Execution Error",
+ "locations": [
+ {
+ "line": 4,
+ "column": 7
+ }
+ ],
+ "path": [
+ "scalarConnection",
+ "edges",
+ 1,
+ "node"
+ ]
+ }
+ ],
+ "data": {
+ "scalarConnection": {
+ "edges": [
+ {
+ "node": "a"
+ },
+ {
+ "node": null
+ },
+ {
+ "node": "c"
+ }
+ ]
+ }
+ }
+}
diff --git a/src/HotChocolate/Core/test/Execution.Tests/__snapshots__/SemanticNonNullTests.Query_With_NullableConnectionNodes.snap b/src/HotChocolate/Core/test/Execution.Tests/__snapshots__/SemanticNonNullTests.Query_With_NullableConnectionNodes.snap
new file mode 100644
index 00000000000..ba3ef427b57
--- /dev/null
+++ b/src/HotChocolate/Core/test/Execution.Tests/__snapshots__/SemanticNonNullTests.Query_With_NullableConnectionNodes.snap
@@ -0,0 +1,17 @@
+{
+ "data": {
+ "nullableScalarConnection": {
+ "edges": [
+ {
+ "node": "a"
+ },
+ {
+ "node": null
+ },
+ {
+ "node": "c"
+ }
+ ]
+ }
+ }
+}
diff --git a/src/HotChocolate/Core/test/Types.Tests/SemanticNonNullTests.cs b/src/HotChocolate/Core/test/Types.Tests/SemanticNonNullTests.cs
new file mode 100644
index 00000000000..f20ad4eaa9e
--- /dev/null
+++ b/src/HotChocolate/Core/test/Types.Tests/SemanticNonNullTests.cs
@@ -0,0 +1,334 @@
+#nullable enable
+
+using HotChocolate.Execution;
+using HotChocolate.Tests;
+using HotChocolate.Types;
+using HotChocolate.Types.Relay;
+using Microsoft.Extensions.DependencyInjection;
+
+namespace HotChocolate;
+
+public class SemanticNonNullTests
+{
+ [Fact]
+ public async Task Object_Implementing_Node()
+ {
+ await new ServiceCollection()
+ .AddGraphQL()
+ .ModifyOptions(o =>
+ {
+ o.EnableSemanticNonNull = true;
+ o.EnsureAllNodesCanBeResolved = false;
+ })
+ .AddQueryType()
+ .AddGlobalObjectIdentification()
+ .BuildSchemaAsync()
+ .MatchSnapshotAsync();
+ }
+
+ [Fact]
+ public async Task MutationConventions()
+ {
+ await new ServiceCollection()
+ .AddGraphQL()
+ .ModifyOptions(o =>
+ {
+ o.StrictValidation = false;
+ o.EnableSemanticNonNull = true;
+ })
+ .AddMutationConventions()
+ .AddMutationType()
+ .BuildSchemaAsync()
+ .MatchSnapshotAsync();
+ }
+
+ [Fact]
+ public async Task Pagination()
+ {
+ await new ServiceCollection()
+ .AddGraphQL()
+ .ModifyOptions(o =>
+ {
+ o.EnableSemanticNonNull = true;
+ })
+ .AddQueryType()
+ .BuildSchemaAsync()
+ .MatchSnapshotAsync();
+ }
+
+ [Fact]
+ public async Task Derive_SemanticNonNull_From_ImplementationFirst()
+ {
+ await new ServiceCollection()
+ .AddGraphQL()
+ .ModifyOptions(o => o.EnableSemanticNonNull = true)
+ .AddQueryType()
+ .BuildSchemaAsync()
+ .MatchSnapshotAsync();
+ }
+
+ [Fact]
+ public async Task Derive_SemanticNonNull_From_ImplementationFirst_With_GraphQLType_As_Type()
+ {
+ await new ServiceCollection()
+ .AddGraphQL()
+ .ModifyOptions(o => o.EnableSemanticNonNull = true)
+ .AddQueryType()
+ .BuildSchemaAsync()
+ .MatchSnapshotAsync();
+ }
+
+ [Fact]
+ public async Task Derive_SemanticNonNull_From_ImplementationFirst_With_GraphQLType_As_String()
+ {
+ await new ServiceCollection()
+ .AddGraphQL()
+ .ModifyOptions(o => o.EnableSemanticNonNull = true)
+ .AddType()
+ .AddQueryType()
+ .BuildSchemaAsync()
+ .MatchSnapshotAsync();
+ }
+
+ [Fact]
+ public async Task Derive_SemanticNonNull_From_CodeFirst()
+ {
+ await new ServiceCollection()
+ .AddGraphQL()
+ .ModifyOptions(o => o.EnableSemanticNonNull = true)
+ .AddQueryType()
+ .UseField(_ => _ => default)
+ .BuildSchemaAsync()
+ .MatchSnapshotAsync();
+ }
+
+ [Fact]
+ public async Task Apply_SemanticNonNull_To_SchemaFirst()
+ {
+ await new ServiceCollection()
+ .AddGraphQL()
+ .ModifyOptions(o => o.EnableSemanticNonNull = true)
+ .AddDocumentFromString(
+ """
+ type Query {
+ scalar: String
+ nonNulScalar: String!
+ scalarArray: [String]
+ nonNullScalarArray: [String!]!
+ outerNonNullScalarArray: [String]!
+ scalarNestedArray: [[String]]
+ nonNullScalarNestedArray: [[String!]!]!
+ innerNonNullScalarNestedArray: [[String!]]!
+ object: Foo
+ nonNullObject: Foo!
+ objectArray: [Foo]
+ nonNullObjectArray: [Foo!]!
+ objectNestedArray: [[Foo]]
+ nonNullObjectNestedArray: [[Foo!]!]!
+ innerNonNullObjectNestedArray: [[Foo!]]!
+ }
+
+ type Foo {
+ bar: String!
+ }
+ """)
+ .UseField(_ => _ => default)
+ .BuildSchemaAsync()
+ .MatchSnapshotAsync();
+ }
+
+ public class QueryType : ObjectType
+ {
+ protected override void Configure(IObjectTypeDescriptor descriptor)
+ {
+ descriptor.Name("Query");
+ descriptor.Field("scalar").Type();
+ descriptor.Field("nonNulScalar").Type>();
+ descriptor.Field("scalarArray").Type>();
+ descriptor.Field("nonNullScalarArray").Type>>>();
+ descriptor.Field("outerNonNullScalarArray").Type>>();
+ descriptor.Field("scalarNestedArray").Type>>();
+ descriptor.Field("nonNullScalarNestedArray").Type>>>>>();
+ descriptor.Field("innerNonNullScalarNestedArray").Type>>>>();
+ descriptor.Field("object").Type();
+ descriptor.Field("nonNullObject").Type>();
+ descriptor.Field("objectArray").Type>();
+ descriptor.Field("nonNullObjectArray").Type>>>();
+ descriptor.Field("objectNestedArray").Type>>();
+ descriptor.Field("nonNullObjectNestedArray").Type>>>>>();
+ descriptor.Field("innerNonNullObjectNestedArray").Type>>>>();
+ }
+ }
+
+ public class FooType : ObjectType
+ {
+ protected override void Configure(IObjectTypeDescriptor descriptor)
+ {
+ descriptor.Name("Foo");
+ descriptor.Field("bar").Type>();
+ }
+ }
+
+ public class Query
+ {
+ public string? Scalar { get; }
+
+ public string NonNulScalar { get; } = null!;
+
+ public string?[]? ScalarArray { get; }
+
+ public string[] NonNullScalarArray { get; } = null!;
+
+ public string?[] OuterNonNullScalarArray { get; } = null!;
+
+ public string?[]?[]? ScalarNestedArray { get; }
+
+ public string[][] NonNullScalarNestedArray { get; } = null!;
+
+ public string[]?[] InnerNonNullScalarNestedArray { get; } = null!;
+
+ public Foo? Object { get; }
+
+ public Foo NonNullObject { get; } = null!;
+
+ public Foo?[]? ObjectArray { get; }
+
+ public Foo[] NonNullObjectArray { get; } = null!;
+
+ public Foo?[]?[]? ObjectNestedArray { get; }
+
+ public Foo[][] NonNullObjectNestedArray { get; } = null!;
+
+ public Foo[]?[] InnerNonNullObjectNestedArray { get; } = null!;
+ }
+
+ [ObjectType("Query")]
+ public class QueryWithTypeAttribute
+ {
+ [GraphQLType]
+ public string? Scalar { get; }
+
+ [GraphQLType>]
+ public string NonNulScalar { get; } = null!;
+
+ [GraphQLType>]
+ public string?[]? ScalarArray { get; }
+
+ [GraphQLType>>>]
+ public string[] NonNullScalarArray { get; } = null!;
+
+ [GraphQLType>>]
+ public string?[] OuterNonNullScalarArray { get; } = null!;
+
+ [GraphQLType>>]
+ public string?[]?[]? ScalarNestedArray { get; }
+
+ [GraphQLType>>>>>]
+ public string[][] NonNullScalarNestedArray { get; } = null!;
+
+ [GraphQLType>>>>]
+ public string[]?[] InnerNonNullScalarNestedArray { get; } = null!;
+
+ [GraphQLType]
+ public Foo? Object { get; }
+
+ [GraphQLType>]
+ public Foo NonNullObject { get; } = null!;
+
+ [GraphQLType>]
+ public Foo?[]? ObjectArray { get; }
+
+ [GraphQLType>>>]
+ public Foo[] NonNullObjectArray { get; } = null!;
+
+ [GraphQLType>>]
+ public Foo?[]?[]? ObjectNestedArray { get; }
+
+ [GraphQLType>>>>>]
+ public Foo[][] NonNullObjectNestedArray { get; } = null!;
+
+ [GraphQLType>>>>]
+ public Foo[]?[] InnerNonNullObjectNestedArray { get; } = null!;
+ }
+
+ [ObjectType("Query")]
+ public class QueryWithTypeAttributeAsString
+ {
+ [GraphQLType("String")]
+ public string? Scalar { get; }
+
+ [GraphQLType("String!")]
+ public string NonNulScalar { get; } = null!;
+
+ [GraphQLType("[String]")]
+ public string?[]? ScalarArray { get; }
+
+ [GraphQLType("[String!]!")]
+ public string[] NonNullScalarArray { get; } = null!;
+
+ [GraphQLType("[String]!")]
+ public string?[] OuterNonNullScalarArray { get; } = null!;
+
+ [GraphQLType("[[String]]")]
+ public string?[]?[]? ScalarNestedArray { get; }
+
+ [GraphQLType("[[String!]!]!")]
+ public string[][] NonNullScalarNestedArray { get; } = null!;
+
+ [GraphQLType("[[String!]]!")]
+ public string[]?[] InnerNonNullScalarNestedArray { get; } = null!;
+
+ [GraphQLType("Foo")]
+ public Foo? Object { get; }
+
+ [GraphQLType("Foo!")]
+ public Foo NonNullObject { get; } = null!;
+
+ [GraphQLType("[Foo]")]
+ public Foo?[]? ObjectArray { get; }
+
+ [GraphQLType("[Foo!]!")]
+ public Foo[] NonNullObjectArray { get; } = null!;
+
+ [GraphQLType("[[Foo]]")]
+ public Foo?[]?[]? ObjectNestedArray { get; }
+
+ [GraphQLType("[[Foo!]!]!")]
+ public Foo[][] NonNullObjectNestedArray { get; } = null!;
+
+ [GraphQLType("[[Foo!]]!")]
+ public Foo[]?[] InnerNonNullObjectNestedArray { get; } = null!;
+ }
+
+ public class Foo
+ {
+ public string Bar { get; } = default!;
+ }
+
+ [ObjectType("Query")]
+ public class QueryWithNode
+ {
+ public MyNode GetMyNode() => new(1);
+ }
+
+ [Node]
+ public record MyNode([property: ID] int Id);
+
+ public class Mutation
+ {
+ [UseMutationConvention]
+ [Error]
+ public bool DoSomething() => true;
+ }
+
+ public class MyException : Exception;
+
+ public class QueryWithPagination
+ {
+ [UsePaging]
+ public string[] GetCursorPagination() => [];
+
+ [UseOffsetPaging]
+ public string[] GetOffsetPagination() => [];
+ }
+}
diff --git a/src/HotChocolate/Core/test/Types.Tests/__snapshots__/SemanticNonNullTests.Apply_SemanticNonNull_To_SchemaFirst.snap b/src/HotChocolate/Core/test/Types.Tests/__snapshots__/SemanticNonNullTests.Apply_SemanticNonNull_To_SchemaFirst.snap
new file mode 100644
index 00000000000..08e902a8348
--- /dev/null
+++ b/src/HotChocolate/Core/test/Types.Tests/__snapshots__/SemanticNonNullTests.Apply_SemanticNonNull_To_SchemaFirst.snap
@@ -0,0 +1,27 @@
+schema {
+ query: Query
+}
+
+type Foo {
+ bar: String @semanticNonNull
+}
+
+type Query {
+ scalar: String
+ nonNulScalar: String @semanticNonNull
+ scalarArray: [String]
+ nonNullScalarArray: [String] @semanticNonNull(levels: [ 0, 1 ])
+ outerNonNullScalarArray: [String] @semanticNonNull
+ scalarNestedArray: [[String]]
+ nonNullScalarNestedArray: [[String]] @semanticNonNull(levels: [ 0, 1, 2 ])
+ innerNonNullScalarNestedArray: [[String]] @semanticNonNull(levels: [ 0, 2 ])
+ object: Foo
+ nonNullObject: Foo @semanticNonNull
+ objectArray: [Foo]
+ nonNullObjectArray: [Foo] @semanticNonNull(levels: [ 0, 1 ])
+ objectNestedArray: [[Foo]]
+ nonNullObjectNestedArray: [[Foo]] @semanticNonNull(levels: [ 0, 1, 2 ])
+ innerNonNullObjectNestedArray: [[Foo]] @semanticNonNull(levels: [ 0, 2 ])
+}
+
+directive @semanticNonNull(levels: [Int!] = [ 0 ]) on FIELD_DEFINITION
diff --git a/src/HotChocolate/Core/test/Types.Tests/__snapshots__/SemanticNonNullTests.Derive_SemanticNonNull_From_CodeFirst.snap b/src/HotChocolate/Core/test/Types.Tests/__snapshots__/SemanticNonNullTests.Derive_SemanticNonNull_From_CodeFirst.snap
new file mode 100644
index 00000000000..08e902a8348
--- /dev/null
+++ b/src/HotChocolate/Core/test/Types.Tests/__snapshots__/SemanticNonNullTests.Derive_SemanticNonNull_From_CodeFirst.snap
@@ -0,0 +1,27 @@
+schema {
+ query: Query
+}
+
+type Foo {
+ bar: String @semanticNonNull
+}
+
+type Query {
+ scalar: String
+ nonNulScalar: String @semanticNonNull
+ scalarArray: [String]
+ nonNullScalarArray: [String] @semanticNonNull(levels: [ 0, 1 ])
+ outerNonNullScalarArray: [String] @semanticNonNull
+ scalarNestedArray: [[String]]
+ nonNullScalarNestedArray: [[String]] @semanticNonNull(levels: [ 0, 1, 2 ])
+ innerNonNullScalarNestedArray: [[String]] @semanticNonNull(levels: [ 0, 2 ])
+ object: Foo
+ nonNullObject: Foo @semanticNonNull
+ objectArray: [Foo]
+ nonNullObjectArray: [Foo] @semanticNonNull(levels: [ 0, 1 ])
+ objectNestedArray: [[Foo]]
+ nonNullObjectNestedArray: [[Foo]] @semanticNonNull(levels: [ 0, 1, 2 ])
+ innerNonNullObjectNestedArray: [[Foo]] @semanticNonNull(levels: [ 0, 2 ])
+}
+
+directive @semanticNonNull(levels: [Int!] = [ 0 ]) on FIELD_DEFINITION
diff --git a/src/HotChocolate/Core/test/Types.Tests/__snapshots__/SemanticNonNullTests.Derive_SemanticNonNull_From_ImplementationFirst.snap b/src/HotChocolate/Core/test/Types.Tests/__snapshots__/SemanticNonNullTests.Derive_SemanticNonNull_From_ImplementationFirst.snap
new file mode 100644
index 00000000000..08e902a8348
--- /dev/null
+++ b/src/HotChocolate/Core/test/Types.Tests/__snapshots__/SemanticNonNullTests.Derive_SemanticNonNull_From_ImplementationFirst.snap
@@ -0,0 +1,27 @@
+schema {
+ query: Query
+}
+
+type Foo {
+ bar: String @semanticNonNull
+}
+
+type Query {
+ scalar: String
+ nonNulScalar: String @semanticNonNull
+ scalarArray: [String]
+ nonNullScalarArray: [String] @semanticNonNull(levels: [ 0, 1 ])
+ outerNonNullScalarArray: [String] @semanticNonNull
+ scalarNestedArray: [[String]]
+ nonNullScalarNestedArray: [[String]] @semanticNonNull(levels: [ 0, 1, 2 ])
+ innerNonNullScalarNestedArray: [[String]] @semanticNonNull(levels: [ 0, 2 ])
+ object: Foo
+ nonNullObject: Foo @semanticNonNull
+ objectArray: [Foo]
+ nonNullObjectArray: [Foo] @semanticNonNull(levels: [ 0, 1 ])
+ objectNestedArray: [[Foo]]
+ nonNullObjectNestedArray: [[Foo]] @semanticNonNull(levels: [ 0, 1, 2 ])
+ innerNonNullObjectNestedArray: [[Foo]] @semanticNonNull(levels: [ 0, 2 ])
+}
+
+directive @semanticNonNull(levels: [Int!] = [ 0 ]) on FIELD_DEFINITION
diff --git a/src/HotChocolate/Core/test/Types.Tests/__snapshots__/SemanticNonNullTests.Derive_SemanticNonNull_From_ImplementationFirst_With_GraphQLType_As_String.snap b/src/HotChocolate/Core/test/Types.Tests/__snapshots__/SemanticNonNullTests.Derive_SemanticNonNull_From_ImplementationFirst_With_GraphQLType_As_String.snap
new file mode 100644
index 00000000000..08e902a8348
--- /dev/null
+++ b/src/HotChocolate/Core/test/Types.Tests/__snapshots__/SemanticNonNullTests.Derive_SemanticNonNull_From_ImplementationFirst_With_GraphQLType_As_String.snap
@@ -0,0 +1,27 @@
+schema {
+ query: Query
+}
+
+type Foo {
+ bar: String @semanticNonNull
+}
+
+type Query {
+ scalar: String
+ nonNulScalar: String @semanticNonNull
+ scalarArray: [String]
+ nonNullScalarArray: [String] @semanticNonNull(levels: [ 0, 1 ])
+ outerNonNullScalarArray: [String] @semanticNonNull
+ scalarNestedArray: [[String]]
+ nonNullScalarNestedArray: [[String]] @semanticNonNull(levels: [ 0, 1, 2 ])
+ innerNonNullScalarNestedArray: [[String]] @semanticNonNull(levels: [ 0, 2 ])
+ object: Foo
+ nonNullObject: Foo @semanticNonNull
+ objectArray: [Foo]
+ nonNullObjectArray: [Foo] @semanticNonNull(levels: [ 0, 1 ])
+ objectNestedArray: [[Foo]]
+ nonNullObjectNestedArray: [[Foo]] @semanticNonNull(levels: [ 0, 1, 2 ])
+ innerNonNullObjectNestedArray: [[Foo]] @semanticNonNull(levels: [ 0, 2 ])
+}
+
+directive @semanticNonNull(levels: [Int!] = [ 0 ]) on FIELD_DEFINITION
diff --git a/src/HotChocolate/Core/test/Types.Tests/__snapshots__/SemanticNonNullTests.Derive_SemanticNonNull_From_ImplementationFirst_With_GraphQLType_As_Type.snap b/src/HotChocolate/Core/test/Types.Tests/__snapshots__/SemanticNonNullTests.Derive_SemanticNonNull_From_ImplementationFirst_With_GraphQLType_As_Type.snap
new file mode 100644
index 00000000000..08e902a8348
--- /dev/null
+++ b/src/HotChocolate/Core/test/Types.Tests/__snapshots__/SemanticNonNullTests.Derive_SemanticNonNull_From_ImplementationFirst_With_GraphQLType_As_Type.snap
@@ -0,0 +1,27 @@
+schema {
+ query: Query
+}
+
+type Foo {
+ bar: String @semanticNonNull
+}
+
+type Query {
+ scalar: String
+ nonNulScalar: String @semanticNonNull
+ scalarArray: [String]
+ nonNullScalarArray: [String] @semanticNonNull(levels: [ 0, 1 ])
+ outerNonNullScalarArray: [String] @semanticNonNull
+ scalarNestedArray: [[String]]
+ nonNullScalarNestedArray: [[String]] @semanticNonNull(levels: [ 0, 1, 2 ])
+ innerNonNullScalarNestedArray: [[String]] @semanticNonNull(levels: [ 0, 2 ])
+ object: Foo
+ nonNullObject: Foo @semanticNonNull
+ objectArray: [Foo]
+ nonNullObjectArray: [Foo] @semanticNonNull(levels: [ 0, 1 ])
+ objectNestedArray: [[Foo]]
+ nonNullObjectNestedArray: [[Foo]] @semanticNonNull(levels: [ 0, 1, 2 ])
+ innerNonNullObjectNestedArray: [[Foo]] @semanticNonNull(levels: [ 0, 2 ])
+}
+
+directive @semanticNonNull(levels: [Int!] = [ 0 ]) on FIELD_DEFINITION
diff --git a/src/HotChocolate/Core/test/Types.Tests/__snapshots__/SemanticNonNullTests.MutationConventions.snap b/src/HotChocolate/Core/test/Types.Tests/__snapshots__/SemanticNonNullTests.MutationConventions.snap
new file mode 100644
index 00000000000..9441a12814c
--- /dev/null
+++ b/src/HotChocolate/Core/test/Types.Tests/__snapshots__/SemanticNonNullTests.MutationConventions.snap
@@ -0,0 +1,24 @@
+schema {
+ mutation: Mutation
+}
+
+interface Error {
+ message: String @semanticNonNull
+}
+
+type DoSomethingPayload {
+ boolean: Boolean
+ errors: [DoSomethingError] @semanticNonNull(levels: [ 1 ])
+}
+
+type Mutation {
+ doSomething: DoSomethingPayload!
+}
+
+type MyError implements Error {
+ message: String @semanticNonNull
+}
+
+union DoSomethingError = MyError
+
+directive @semanticNonNull(levels: [Int!] = [ 0 ]) on FIELD_DEFINITION
diff --git a/src/HotChocolate/Core/test/Types.Tests/__snapshots__/SemanticNonNullTests.Object_Implementing_Node.snap b/src/HotChocolate/Core/test/Types.Tests/__snapshots__/SemanticNonNullTests.Object_Implementing_Node.snap
new file mode 100644
index 00000000000..454e9932455
--- /dev/null
+++ b/src/HotChocolate/Core/test/Types.Tests/__snapshots__/SemanticNonNullTests.Object_Implementing_Node.snap
@@ -0,0 +1,22 @@
+schema {
+ query: Query
+}
+
+"The node interface is implemented by entities that have a global unique identifier."
+interface Node {
+ id: ID!
+}
+
+type MyNode implements Node {
+ id: ID!
+}
+
+type Query {
+ "Fetches an object given its ID."
+ node("ID of the object." id: ID!): Node
+ "Lookup nodes by a list of IDs."
+ nodes("The list of node IDs." ids: [ID!]!): [Node]!
+ myNode: MyNode @semanticNonNull
+}
+
+directive @semanticNonNull(levels: [Int!] = [ 0 ]) on FIELD_DEFINITION
diff --git a/src/HotChocolate/Core/test/Types.Tests/__snapshots__/SemanticNonNullTests.Pagination.snap b/src/HotChocolate/Core/test/Types.Tests/__snapshots__/SemanticNonNullTests.Pagination.snap
new file mode 100644
index 00000000000..146dc8ea148
--- /dev/null
+++ b/src/HotChocolate/Core/test/Types.Tests/__snapshots__/SemanticNonNullTests.Pagination.snap
@@ -0,0 +1,56 @@
+schema {
+ query: QueryWithPagination
+}
+
+"Information about the offset pagination."
+type CollectionSegmentInfo {
+ "Indicates whether more items exist following the set defined by the clients arguments."
+ hasNextPage: Boolean!
+ "Indicates whether more items exist prior the set defined by the clients arguments."
+ hasPreviousPage: Boolean!
+}
+
+"A connection to a list of items."
+type CursorPaginationConnection {
+ "Information to aid in pagination."
+ pageInfo: PageInfo @semanticNonNull
+ "A list of edges."
+ edges: [CursorPaginationEdge] @semanticNonNull(levels: [ 1 ])
+ "A flattened list of the nodes."
+ nodes: [String] @semanticNonNull(levels: [ 1 ])
+}
+
+"An edge in a connection."
+type CursorPaginationEdge {
+ "A cursor for use in pagination."
+ cursor: String @semanticNonNull
+ "The item at the end of the edge."
+ node: String @semanticNonNull
+}
+
+"A segment of a collection."
+type OffsetPaginationCollectionSegment {
+ "Information to aid in pagination."
+ pageInfo: CollectionSegmentInfo @semanticNonNull
+ "A flattened list of the items."
+ items: [String] @semanticNonNull(levels: [ 1 ])
+}
+
+"Information about pagination in a connection."
+type PageInfo {
+ "Indicates whether more edges exist following the set defined by the clients arguments."
+ hasNextPage: Boolean!
+ "Indicates whether more edges exist prior the set defined by the clients arguments."
+ hasPreviousPage: Boolean!
+ "When paginating backwards, the cursor to continue."
+ startCursor: String
+ "When paginating forwards, the cursor to continue."
+ endCursor: String
+}
+
+type QueryWithPagination {
+ cursorPagination("Returns the first _n_ elements from the list." first: Int "Returns the elements in the list that come after the specified cursor." after: String "Returns the last _n_ elements from the list." last: Int "Returns the elements in the list that come before the specified cursor." before: String): CursorPaginationConnection
+ offsetPagination(skip: Int take: Int): OffsetPaginationCollectionSegment
+}
+
+directive @semanticNonNull(levels: [Int!] = [ 0 ]) on FIELD_DEFINITION