diff --git a/src/HotChocolate/Fusion-vnext/src/Fusion.Composition/Logging/LogEntryCodes.cs b/src/HotChocolate/Fusion-vnext/src/Fusion.Composition/Logging/LogEntryCodes.cs
index 298f766cf33..cf1339ee726 100644
--- a/src/HotChocolate/Fusion-vnext/src/Fusion.Composition/Logging/LogEntryCodes.cs
+++ b/src/HotChocolate/Fusion-vnext/src/Fusion.Composition/Logging/LogEntryCodes.cs
@@ -3,5 +3,6 @@ namespace HotChocolate.Fusion.Logging;
public static class LogEntryCodes
{
public const string DisallowedInaccessible = "DISALLOWED_INACCESSIBLE";
+ public const string ExternalArgumentDefaultMismatch = "EXTERNAL_ARGUMENT_DEFAULT_MISMATCH";
public const string OutputFieldTypesNotMergeable = "OUTPUT_FIELD_TYPES_NOT_MERGEABLE";
}
diff --git a/src/HotChocolate/Fusion-vnext/src/Fusion.Composition/Logging/LogEntryHelper.cs b/src/HotChocolate/Fusion-vnext/src/Fusion.Composition/Logging/LogEntryHelper.cs
index 15bda6cc934..c59331ee55c 100644
--- a/src/HotChocolate/Fusion-vnext/src/Fusion.Composition/Logging/LogEntryHelper.cs
+++ b/src/HotChocolate/Fusion-vnext/src/Fusion.Composition/Logging/LogEntryHelper.cs
@@ -76,6 +76,16 @@ public static LogEntry DisallowedInaccessibleDirectiveArgument(
new SchemaCoordinate(directiveName, argumentName: argument.Name, ofDirective: true),
schema: schema);
+ public static LogEntry ExternalArgumentDefaultMismatch(string fieldName, string typeName)
+ => new(
+ string.Format(
+ LogEntryHelper_ExternalArgumentDefaultMismatch,
+ fieldName,
+ typeName),
+ LogEntryCodes.ExternalArgumentDefaultMismatch,
+ LogSeverity.Error,
+ new SchemaCoordinate(typeName, fieldName));
+
public static LogEntry OutputFieldTypesNotMergeable(string fieldName, string typeName)
=> new(
string.Format(
diff --git a/src/HotChocolate/Fusion-vnext/src/Fusion.Composition/PreMergeValidation/Rules/ExternalArgumentsDefaultMismatchRule.cs b/src/HotChocolate/Fusion-vnext/src/Fusion.Composition/PreMergeValidation/Rules/ExternalArgumentsDefaultMismatchRule.cs
index 799641a5e6c..8753db86e41 100644
--- a/src/HotChocolate/Fusion-vnext/src/Fusion.Composition/PreMergeValidation/Rules/ExternalArgumentsDefaultMismatchRule.cs
+++ b/src/HotChocolate/Fusion-vnext/src/Fusion.Composition/PreMergeValidation/Rules/ExternalArgumentsDefaultMismatchRule.cs
@@ -1,85 +1,59 @@
using HotChocolate.Fusion.Events;
+using HotChocolate.Language;
using HotChocolate.Skimmed;
using static HotChocolate.Fusion.Logging.LogEntryHelper;
namespace HotChocolate.Fusion.PreMergeValidation.Rules;
///
-/// This rule ensures that certain essential elements of a GraphQL schema, particularly built-in
-/// scalars, directive arguments, and introspection types, cannot be marked as @inaccessible. These
-/// types are fundamental to GraphQL. Making these elements inaccessible would break core GraphQL
-/// functionality.
+/// This rule ensures that arguments on fields marked as @external have default values compatible
+/// with the corresponding arguments on fields from other source schemas where the field is defined (non-@external).
///
-///
+///
/// Specification
///
internal sealed class ExternalArgumentsDefaultMismatchRule
- : IEventHandler
- , IEventHandler
- , IEventHandler
- , IEventHandler
+ : IEventHandler
{
- public void Handle(TypeEvent @event, CompositionContext context)
+ public void Handle(OutputFieldGroupEvent @event, CompositionContext context)
{
- var (type, schema) = @event;
+ var (fieldName, fieldGroup, typeName) = @event;
- // Built-in scalar types must be accessible.
- if (type is ScalarTypeDefinition { IsSpecScalar: true } scalar
- && !ValidationHelper.IsAccessible(scalar))
+ if (fieldGroup.FirstOrDefault(i => ValidationHelper.IsExternal(i.Field)) is { } externalField)
{
- context.Log.Write(DisallowedInaccessibleBuiltInScalar(scalar, schema));
- }
-
- // Introspection types must be accessible.
- if (type.IsIntrospectionType && !ValidationHelper.IsAccessible(type))
- {
- context.Log.Write(DisallowedInaccessibleIntrospectionType(type, schema));
- }
- }
-
- public void Handle(OutputFieldEvent @event, CompositionContext context)
- {
- var (field, type, schema) = @event;
-
- // Introspection fields must be accessible.
- if (type.IsIntrospectionType && !ValidationHelper.IsAccessible(field))
- {
- context.Log.Write(
- DisallowedInaccessibleIntrospectionField(
- field,
- type.Name,
- schema));
- }
- }
-
- public void Handle(FieldArgumentEvent @event, CompositionContext context)
- {
- var (argument, field, type, schema) = @event;
-
- // Introspection arguments must be accessible.
- if (type.IsIntrospectionType && !ValidationHelper.IsAccessible(argument))
- {
- context.Log.Write(
- DisallowedInaccessibleIntrospectionArgument(
- argument,
- field.Name,
- type.Name,
- schema));
- }
- }
-
- public void Handle(DirectiveArgumentEvent @event, CompositionContext context)
- {
- var (argument, directive, schema) = @event;
-
- // Built-in directive arguments must be accessible.
- if (BuiltIns.IsBuiltInDirective(directive.Name) && !ValidationHelper.IsAccessible(argument))
- {
- context.Log.Write(
- DisallowedInaccessibleDirectiveArgument(
- argument,
- directive.Name,
- schema));
+ var argumentNames = fieldGroup.SelectMany(fg => fg.Field.Arguments, (_, arg) => arg.Name).ToHashSet();
+ foreach (var argumentName in argumentNames)
+ {
+ if (!externalField.Field.Arguments.TryGetField(argumentName, out var argumentField))
+ {
+ // Logged in separate rule.
+ continue;
+ }
+
+ var defaultValue = argumentField.DefaultValue;
+ foreach (var currentField in fieldGroup.Except([externalField]))
+ {
+ if (!currentField.Field.Arguments.TryGetField(argumentName, out argumentField))
+ {
+ // Logged in separate rule.
+ continue;
+ }
+
+ var currentValue = argumentField.DefaultValue;
+ var match = (currentValue, defaultValue) switch
+ {
+ (null, null) => true,
+ (not null, null) => false,
+ (null, not null) => false,
+ _ => currentValue.Value!.Equals(defaultValue.Value)
+ };
+
+ if (!match)
+ {
+ context.Log.Write(ExternalArgumentDefaultMismatch(fieldName, typeName));
+ }
+ }
+ }
}
}
}
diff --git a/src/HotChocolate/Fusion-vnext/src/Fusion.Composition/Properties/CompositionResources.Designer.cs b/src/HotChocolate/Fusion-vnext/src/Fusion.Composition/Properties/CompositionResources.Designer.cs
index 8a2c9d46b34..2900d451d0c 100644
--- a/src/HotChocolate/Fusion-vnext/src/Fusion.Composition/Properties/CompositionResources.Designer.cs
+++ b/src/HotChocolate/Fusion-vnext/src/Fusion.Composition/Properties/CompositionResources.Designer.cs
@@ -11,46 +11,32 @@ namespace HotChocolate.Fusion.Properties {
using System;
- ///
- /// A strongly-typed resource class, for looking up localized strings, etc.
- ///
- // This class was auto-generated by the StronglyTypedResourceBuilder
- // class via a tool like ResGen or Visual Studio.
- // To add or remove a member, edit your .ResX file then rerun ResGen
- // with the /str option, or rebuild your VS project.
- [global::System.CodeDom.Compiler.GeneratedCodeAttribute("System.Resources.Tools.StronglyTypedResourceBuilder", "4.0.0.0")]
- [global::System.Diagnostics.DebuggerNonUserCodeAttribute()]
- [global::System.Runtime.CompilerServices.CompilerGeneratedAttribute()]
+ [System.CodeDom.Compiler.GeneratedCodeAttribute("System.Resources.Tools.StronglyTypedResourceBuilder", "4.0.0.0")]
+ [System.Diagnostics.DebuggerNonUserCodeAttribute()]
+ [System.Runtime.CompilerServices.CompilerGeneratedAttribute()]
internal class CompositionResources {
- private static global::System.Resources.ResourceManager resourceMan;
+ private static System.Resources.ResourceManager resourceMan;
- private static global::System.Globalization.CultureInfo resourceCulture;
+ private static System.Globalization.CultureInfo resourceCulture;
- [global::System.Diagnostics.CodeAnalysis.SuppressMessageAttribute("Microsoft.Performance", "CA1811:AvoidUncalledPrivateCode")]
+ [System.Diagnostics.CodeAnalysis.SuppressMessageAttribute("Microsoft.Performance", "CA1811:AvoidUncalledPrivateCode")]
internal CompositionResources() {
}
- ///
- /// Returns the cached ResourceManager instance used by this class.
- ///
- [global::System.ComponentModel.EditorBrowsableAttribute(global::System.ComponentModel.EditorBrowsableState.Advanced)]
- internal static global::System.Resources.ResourceManager ResourceManager {
+ [System.ComponentModel.EditorBrowsableAttribute(System.ComponentModel.EditorBrowsableState.Advanced)]
+ internal static System.Resources.ResourceManager ResourceManager {
get {
- if (object.ReferenceEquals(resourceMan, null)) {
- global::System.Resources.ResourceManager temp = new global::System.Resources.ResourceManager("HotChocolate.Fusion.Properties.CompositionResources", typeof(CompositionResources).Assembly);
+ if (object.Equals(null, resourceMan)) {
+ System.Resources.ResourceManager temp = new System.Resources.ResourceManager("HotChocolate.Fusion.Properties.CompositionResources", typeof(CompositionResources).Assembly);
resourceMan = temp;
}
return resourceMan;
}
}
- ///
- /// Overrides the current thread's CurrentUICulture property for all
- /// resource lookups using this strongly typed resource class.
- ///
- [global::System.ComponentModel.EditorBrowsableAttribute(global::System.ComponentModel.EditorBrowsableState.Advanced)]
- internal static global::System.Globalization.CultureInfo Culture {
+ [System.ComponentModel.EditorBrowsableAttribute(System.ComponentModel.EditorBrowsableState.Advanced)]
+ internal static System.Globalization.CultureInfo Culture {
get {
return resourceCulture;
}
@@ -59,67 +45,52 @@ internal CompositionResources() {
}
}
- ///
- /// Looks up a localized string similar to Pre-merge validation failed. View the composition log for details..
- ///
internal static string ErrorHelper_PreMergeValidationFailed {
get {
return ResourceManager.GetString("ErrorHelper_PreMergeValidationFailed", resourceCulture);
}
}
- ///
- /// Looks up a localized string similar to The built-in scalar type '{0}' is not accessible..
- ///
internal static string LogEntryHelper_DisallowedInaccessibleBuiltInScalar {
get {
return ResourceManager.GetString("LogEntryHelper_DisallowedInaccessibleBuiltInScalar", resourceCulture);
}
}
- ///
- /// Looks up a localized string similar to The argument '{0}' on built-in directive type '{1}' is not accessible..
- ///
- internal static string LogEntryHelper_DisallowedInaccessibleDirectiveArgument {
+ internal static string LogEntryHelper_DisallowedInaccessibleIntrospectionType {
get {
- return ResourceManager.GetString("LogEntryHelper_DisallowedInaccessibleDirectiveArgument", resourceCulture);
+ return ResourceManager.GetString("LogEntryHelper_DisallowedInaccessibleIntrospectionType", resourceCulture);
}
}
- ///
- /// Looks up a localized string similar to The introspection argument '{0}' with schema coordinate '{1}' is not accessible..
- ///
- internal static string LogEntryHelper_DisallowedInaccessibleIntrospectionArgument {
+ internal static string LogEntryHelper_DisallowedInaccessibleIntrospectionField {
get {
- return ResourceManager.GetString("LogEntryHelper_DisallowedInaccessibleIntrospectionArgument", resourceCulture);
+ return ResourceManager.GetString("LogEntryHelper_DisallowedInaccessibleIntrospectionField", resourceCulture);
}
}
- ///
- /// Looks up a localized string similar to The introspection field '{0}' on type '{1}' is not accessible..
- ///
- internal static string LogEntryHelper_DisallowedInaccessibleIntrospectionField {
+ internal static string LogEntryHelper_DisallowedInaccessibleIntrospectionArgument {
get {
- return ResourceManager.GetString("LogEntryHelper_DisallowedInaccessibleIntrospectionField", resourceCulture);
+ return ResourceManager.GetString("LogEntryHelper_DisallowedInaccessibleIntrospectionArgument", resourceCulture);
}
}
- ///
- /// Looks up a localized string similar to The introspection type '{0}' is not accessible..
- ///
- internal static string LogEntryHelper_DisallowedInaccessibleIntrospectionType {
+ internal static string LogEntryHelper_DisallowedInaccessibleDirectiveArgument {
get {
- return ResourceManager.GetString("LogEntryHelper_DisallowedInaccessibleIntrospectionType", resourceCulture);
+ return ResourceManager.GetString("LogEntryHelper_DisallowedInaccessibleDirectiveArgument", resourceCulture);
}
}
- ///
- /// Looks up a localized string similar to Field '{0}' on type '{1}' is not mergeable..
- ///
internal static string LogEntryHelper_OutputFieldTypesNotMergeable {
get {
return ResourceManager.GetString("LogEntryHelper_OutputFieldTypesNotMergeable", resourceCulture);
}
}
+
+ internal static string LogEntryHelper_ExternalArgumentDefaultMismatch {
+ get {
+ return ResourceManager.GetString("LogEntryHelper_ExternalArgumentDefaultMismatch", resourceCulture);
+ }
+ }
}
}
diff --git a/src/HotChocolate/Fusion-vnext/src/Fusion.Composition/Properties/CompositionResources.resx b/src/HotChocolate/Fusion-vnext/src/Fusion.Composition/Properties/CompositionResources.resx
index 78fde65b49b..51125140373 100644
--- a/src/HotChocolate/Fusion-vnext/src/Fusion.Composition/Properties/CompositionResources.resx
+++ b/src/HotChocolate/Fusion-vnext/src/Fusion.Composition/Properties/CompositionResources.resx
@@ -39,4 +39,7 @@
Field '{0}' on type '{1}' is not mergeable.
+
+ Field '{0}' on type '{1}' must have consistent default values.
+
diff --git a/src/HotChocolate/Fusion-vnext/test/Fusion.Composition.Tests/PreMergeValidation/Rules/ExternalArgumentsDefaultMismatchRuleTests.cs b/src/HotChocolate/Fusion-vnext/test/Fusion.Composition.Tests/PreMergeValidation/Rules/ExternalArgumentsDefaultMismatchRuleTests.cs
index 9c2baabd41e..270a8e13c37 100644
--- a/src/HotChocolate/Fusion-vnext/test/Fusion.Composition.Tests/PreMergeValidation/Rules/ExternalArgumentsDefaultMismatchRuleTests.cs
+++ b/src/HotChocolate/Fusion-vnext/test/Fusion.Composition.Tests/PreMergeValidation/Rules/ExternalArgumentsDefaultMismatchRuleTests.cs
@@ -62,6 +62,19 @@ type Product {
}
"""
],
+ () =>
+ [
+ """
+ type Product {
+ name(language: String = "en", localization: String = "sr"): String
+ }
+ """,
+ """
+ type Product {
+ name(language: String = "en", localization: String = "sr"): String @external
+ }
+ """
+ ],
];
}
@@ -95,7 +108,33 @@ type Product {
name(language: String): String @external
}
"""
- ]
+ ],
+ () =>
+ [
+ """
+ type Product {
+ name(language: String = "en", localization: String = "sr"): String
+ }
+ """,
+ """
+ type Product {
+ name(language: String = "en", localization: String = "sa"): String @external
+ }
+ """
+ ],
+ () =>
+ [
+ """
+ type Product {
+ name(language: String = "en", localization: String = "sr"): String
+ }
+ """,
+ """
+ type Product {
+ name(language: String = "en", localization: String): String @external
+ }
+ """
+ ],
];
}
}