Add tests and fix exports diagnostics

- Add tests for the following diagnostics: GD0101, GD0102, GD0103, GD0104, GD0105, GD0106, GD0107.
- Fix GD0101 not being reported any more (was filtering static classes before reporting).
- Fix GD0107 not preventing `Node` members from being exported from not-`Node` types.
This commit is contained in:
Paul Joannon 2024-02-18 14:40:31 +01:00
parent b7145638d5
commit 88ad4e6c24
No known key found for this signature in database
GPG key ID: C12F69B0AD0390DD
19 changed files with 242 additions and 9 deletions

View file

@ -0,0 +1,77 @@
using Xunit;
namespace Godot.SourceGenerators.Tests;
public class ExportDiagnosticsTests
{
[Fact]
public async void StaticMembers()
{
await CSharpSourceGeneratorVerifier<ScriptPropertyDefValGenerator>.Verify(
"ExportDiagnostics_GD0101.cs",
"ExportDiagnostics_GD0101_ScriptPropertyDefVal.generated.cs"
);
}
[Fact]
public async void TypeIsNotSupported()
{
await CSharpSourceGeneratorVerifier<ScriptPropertyDefValGenerator>.Verify(
"ExportDiagnostics_GD0102.cs",
"ExportDiagnostics_GD0102_ScriptPropertyDefVal.generated.cs"
);
}
[Fact]
public async void ReadOnly()
{
await CSharpSourceGeneratorVerifier<ScriptPropertyDefValGenerator>.Verify(
"ExportDiagnostics_GD0103.cs",
"ExportDiagnostics_GD0103_ScriptPropertyDefVal.generated.cs"
);
}
[Fact]
public async void WriteOnly()
{
await CSharpSourceGeneratorVerifier<ScriptPropertyDefValGenerator>.Verify(
"ExportDiagnostics_GD0104.cs",
"ExportDiagnostics_GD0104_ScriptPropertyDefVal.generated.cs"
);
}
[Fact]
public async void Indexer()
{
await CSharpSourceGeneratorVerifier<ScriptPropertyDefValGenerator>.Verify(
"ExportDiagnostics_GD0105.cs",
"ExportDiagnostics_GD0105_ScriptPropertyDefVal.generated.cs"
);
}
[Fact]
public async void ExplicitInterfaceImplementation()
{
await CSharpSourceGeneratorVerifier<ScriptPropertyDefValGenerator>.Verify(
new string[] { "ExportDiagnostics_GD0106.cs" },
new string[]
{
"ExportDiagnostics_GD0106_OK_ScriptPropertyDefVal.generated.cs",
"ExportDiagnostics_GD0106_KO_ScriptPropertyDefVal.generated.cs",
}
);
}
[Fact]
public async void NodeExports()
{
await CSharpSourceGeneratorVerifier<ScriptPropertyDefValGenerator>.Verify(
new string[] { "ExportDiagnostics_GD0107.cs" },
new string[]
{
"ExportDiagnostics_GD0107_OK_ScriptPropertyDefVal.generated.cs",
"ExportDiagnostics_GD0107_KO_ScriptPropertyDefVal.generated.cs",
}
);
}
}

View file

@ -0,0 +1,21 @@
partial class ExportDiagnostics_GD0106_OK
{
#pragma warning disable CS0109 // Disable warning about redundant 'new' keyword
#if TOOLS
/// <summary>
/// Get the default values for all properties declared in this class.
/// This method is used by Godot to determine the value that will be
/// used by the inspector when resetting properties.
/// Do not call this method.
/// </summary>
[global::System.ComponentModel.EditorBrowsable(global::System.ComponentModel.EditorBrowsableState.Never)]
internal new static global::System.Collections.Generic.Dictionary<global::Godot.StringName, global::Godot.Variant> GetGodotPropertyDefaultValues()
{
var values = new global::System.Collections.Generic.Dictionary<global::Godot.StringName, global::Godot.Variant>(1);
int __MyProperty_default_value = default;
values.Add(PropertyName.MyProperty, global::Godot.Variant.From<int>(__MyProperty_default_value));
return values;
}
#endif // TOOLS
#pragma warning restore CS0109
}

View file

@ -0,0 +1,23 @@
partial class ExportDiagnostics_GD0107_OK
{
#pragma warning disable CS0109 // Disable warning about redundant 'new' keyword
#if TOOLS
/// <summary>
/// Get the default values for all properties declared in this class.
/// This method is used by Godot to determine the value that will be
/// used by the inspector when resetting properties.
/// Do not call this method.
/// </summary>
[global::System.ComponentModel.EditorBrowsable(global::System.ComponentModel.EditorBrowsableState.Never)]
internal new static global::System.Collections.Generic.Dictionary<global::Godot.StringName, global::Godot.Variant> GetGodotPropertyDefaultValues()
{
var values = new global::System.Collections.Generic.Dictionary<global::Godot.StringName, global::Godot.Variant>(2);
global::Godot.Node __NodeProperty_default_value = default;
values.Add(PropertyName.NodeProperty, global::Godot.Variant.From<global::Godot.Node>(__NodeProperty_default_value));
global::Godot.Node __NodeField_default_value = default;
values.Add(PropertyName.NodeField, global::Godot.Variant.From<global::Godot.Node>(__NodeField_default_value));
return values;
}
#endif // TOOLS
#pragma warning restore CS0109
}

View file

@ -0,0 +1,10 @@
using Godot;
public partial class ExportDiagnostics_GD0101 : Node
{
[Export]
public static string {|GD0101:StaticField|};
[Export]
public static int {|GD0101:StaticProperty|} { get; set; }
}

View file

@ -0,0 +1,12 @@
using Godot;
public partial class ExportDiagnostics_GD0102 : Node
{
public struct MyStruct { }
[Export]
public MyStruct {|GD0102:StructField|};
[Export]
public MyStruct {|GD0102:StructProperty|} { get; set; }
}

View file

@ -0,0 +1,10 @@
using Godot;
public partial class ExportDiagnostics_GD0103 : Node
{
[Export]
public readonly string {|GD0103:ReadOnlyField|};
[Export]
public string {|GD0103:ReadOnlyProperty|} { get; }
}

View file

@ -0,0 +1,7 @@
using Godot;
public partial class ExportDiagnostics_GD0104 : Node
{
[Export]
public string {|GD0104:WriteOnlyProperty|} { set { } }
}

View file

@ -0,0 +1,12 @@
using System;
using Godot;
public partial class ExportDiagnostics_GD0105 : Node
{
[Export]
public int {|GD0105:this|}[int index]
{
get { return index; }
set { }
}
}

View file

@ -0,0 +1,18 @@
using Godot;
public interface MyInterface
{
public int MyProperty { get; set; }
}
public partial class ExportDiagnostics_GD0106_OK : Node, MyInterface
{
[Export]
public int MyProperty { get; set; }
}
public partial class ExportDiagnostics_GD0106_KO : Node, MyInterface
{
[Export]
int MyInterface.{|GD0106:MyProperty|} { get; set; }
}

View file

@ -0,0 +1,19 @@
using Godot;
public partial class ExportDiagnostics_GD0107_OK : Node
{
[Export]
public Node NodeField;
[Export]
public Node NodeProperty { get; set; }
}
public partial class ExportDiagnostics_GD0107_KO : Resource
{
[Export]
public Node {|GD0107:NodeField|};
[Export]
public Node {|GD0107:NodeProperty|} { get; set; }
}

View file

@ -3,6 +3,7 @@ namespace Godot.SourceGenerators
public static class GodotClasses public static class GodotClasses
{ {
public const string GodotObject = "Godot.GodotObject"; public const string GodotObject = "Godot.GodotObject";
public const string Node = "Godot.Node";
public const string AssemblyHasScriptsAttr = "Godot.AssemblyHasScriptsAttribute"; public const string AssemblyHasScriptsAttr = "Godot.AssemblyHasScriptsAttribute";
public const string ExportAttr = "Godot.ExportAttribute"; public const string ExportAttr = "Godot.ExportAttribute";
public const string ExportCategoryAttr = "Godot.ExportCategoryAttribute"; public const string ExportCategoryAttr = "Godot.ExportCategoryAttribute";

View file

@ -66,11 +66,13 @@ namespace Godot.SourceGenerators
) )
{ {
INamespaceSymbol namespaceSymbol = symbol.ContainingNamespace; INamespaceSymbol namespaceSymbol = symbol.ContainingNamespace;
string classNs = namespaceSymbol != null && !namespaceSymbol.IsGlobalNamespace ? string classNs = namespaceSymbol is { IsGlobalNamespace: false }
namespaceSymbol.FullQualifiedNameOmitGlobal() : ? namespaceSymbol.FullQualifiedNameOmitGlobal()
string.Empty; : string.Empty;
bool hasNamespace = classNs.Length != 0; bool hasNamespace = classNs.Length != 0;
bool isNode = symbol.InheritsFrom("GodotSharp", GodotClasses.Node);
bool isInnerClass = symbol.ContainingType != null; bool isInnerClass = symbol.ContainingType != null;
string uniqueHint = symbol.FullQualifiedNameOmitGlobal().SanitizeQualifiedNameForUniqueHint() string uniqueHint = symbol.FullQualifiedNameOmitGlobal().SanitizeQualifiedNameForUniqueHint()
@ -114,14 +116,14 @@ namespace Godot.SourceGenerators
var members = symbol.GetMembers(); var members = symbol.GetMembers();
var exportedProperties = members var exportedProperties = members
.Where(s => !s.IsStatic && s.Kind == SymbolKind.Property) .Where(s => s.Kind == SymbolKind.Property)
.Cast<IPropertySymbol>() .Cast<IPropertySymbol>()
.Where(s => s.GetAttributes() .Where(s => s.GetAttributes()
.Any(a => a.AttributeClass?.IsGodotExportAttribute() ?? false)) .Any(a => a.AttributeClass?.IsGodotExportAttribute() ?? false))
.ToArray(); .ToArray();
var exportedFields = members var exportedFields = members
.Where(s => !s.IsStatic && s.Kind == SymbolKind.Field && !s.IsImplicitlyDeclared) .Where(s => s.Kind == SymbolKind.Field && !s.IsImplicitlyDeclared)
.Cast<IFieldSymbol>() .Cast<IFieldSymbol>()
.Where(s => s.GetAttributes() .Where(s => s.GetAttributes()
.Any(a => a.AttributeClass?.IsGodotExportAttribute() ?? false)) .Any(a => a.AttributeClass?.IsGodotExportAttribute() ?? false))
@ -198,13 +200,13 @@ namespace Godot.SourceGenerators
if (marshalType == MarshalType.GodotObjectOrDerived) if (marshalType == MarshalType.GodotObjectOrDerived)
{ {
if (!symbol.InheritsFrom("GodotSharp", "Godot.Node") && if (!isNode && propertyType.InheritsFrom("GodotSharp", GodotClasses.Node))
propertyType.InheritsFrom("GodotSharp", "Godot.Node"))
{ {
context.ReportDiagnostic(Diagnostic.Create( context.ReportDiagnostic(Diagnostic.Create(
Common.OnlyNodesShouldExportNodesRule, Common.OnlyNodesShouldExportNodesRule,
property.Locations.FirstLocationWithSourceTreeOrDefault() property.Locations.FirstLocationWithSourceTreeOrDefault()
)); ));
continue;
} }
} }
@ -317,13 +319,13 @@ namespace Godot.SourceGenerators
if (marshalType == MarshalType.GodotObjectOrDerived) if (marshalType == MarshalType.GodotObjectOrDerived)
{ {
if (!symbol.InheritsFrom("GodotSharp", "Godot.Node") && if (!isNode && fieldType.InheritsFrom("GodotSharp", GodotClasses.Node))
fieldType.InheritsFrom("GodotSharp", "Godot.Node"))
{ {
context.ReportDiagnostic(Diagnostic.Create( context.ReportDiagnostic(Diagnostic.Create(
Common.OnlyNodesShouldExportNodesRule, Common.OnlyNodesShouldExportNodesRule,
field.Locations.FirstLocationWithSourceTreeOrDefault() field.Locations.FirstLocationWithSourceTreeOrDefault()
)); ));
continue;
} }
} }