diff --git a/.github/workflows/publish.yml b/.github/workflows/publish.yml
index 478276d..396363c 100644
--- a/.github/workflows/publish.yml
+++ b/.github/workflows/publish.yml
@@ -60,5 +60,10 @@ jobs:
dotnet build ./src/LightResults.Extensions.Json/LightResults.Extensions.Json.csproj --configuration Release --no-restore
dotnet pack ./src/LightResults.Extensions.Json/LightResults.Extensions.Json.csproj --configuration Release --no-build --output .
+ - name: Pack GeneratedIdentifier
+ run: |
+ dotnet build ./src/LightResults.Extensions.GeneratedIdentifier/LightResults.Extensions.GeneratedIdentifier.csproj --configuration Release --no-restore
+ dotnet pack ./src/LightResults.Extensions.GeneratedIdentifier/LightResults.Extensions.GeneratedIdentifier.csproj --configuration Release --no-build --output .
+
- name: Push to NuGet
run: dotnet nuget push "*.nupkg" --api-key ${{secrets.NUGET_API_KEY}} --source https://api.nuget.org/v3/index.json --skip-duplicate
diff --git a/ComparisonBenchmarks/ComparisonBenchmarks.csproj b/ComparisonBenchmarks/ComparisonBenchmarks.csproj
index bf78d27..8ae5d28 100644
--- a/ComparisonBenchmarks/ComparisonBenchmarks.csproj
+++ b/ComparisonBenchmarks/ComparisonBenchmarks.csproj
@@ -11,14 +11,14 @@
-
-
-
-
+
+
+
+
-
+
diff --git a/LightResults.Extensions.sln b/LightResults.Extensions.sln
index dacdd2c..4fb9a40 100644
--- a/LightResults.Extensions.sln
+++ b/LightResults.Extensions.sln
@@ -20,6 +20,12 @@ Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "Benchmarks", "tools\Benchma
EndProject
Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "LightResults.Extensions.Operations.Tests", "tests\LightResults.Extensions.Operations.Tests\LightResults.Extensions.Operations.Tests.csproj", "{D4E8F259-596E-412D-A757-769383865764}"
EndProject
+Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "LightResults.Extensions.GeneratedIdentifier", "src\LightResults.Extensions.GeneratedIdentifier\LightResults.Extensions.GeneratedIdentifier.csproj", "{2D558C96-F052-4959-B996-5B03A66E4FCF}"
+EndProject
+Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "LightResults.Extensions.GeneratedIdentifier.Fixtures", "tests\LightResults.Extensions.GeneratedIdentifier.Fixtures\LightResults.Extensions.GeneratedIdentifier.Fixtures.csproj", "{06BDEA4C-9036-44B7-AB44-DBD088556726}"
+EndProject
+Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "LightResults.Extensions.GeneratedIdentifier.Tests", "tests\LightResults.Extensions.GeneratedIdentifier.Tests\LightResults.Extensions.GeneratedIdentifier.Tests.csproj", "{45AF3136-2F1A-48C5-A76F-52635D7F0B90}"
+EndProject
Global
GlobalSection(SolutionConfigurationPlatforms) = preSolution
Debug|Any CPU = Debug|Any CPU
@@ -58,6 +64,18 @@ Global
{D4E8F259-596E-412D-A757-769383865764}.Debug|Any CPU.Build.0 = Debug|Any CPU
{D4E8F259-596E-412D-A757-769383865764}.Release|Any CPU.ActiveCfg = Release|Any CPU
{D4E8F259-596E-412D-A757-769383865764}.Release|Any CPU.Build.0 = Release|Any CPU
+ {2D558C96-F052-4959-B996-5B03A66E4FCF}.Debug|Any CPU.ActiveCfg = Debug|Any CPU
+ {2D558C96-F052-4959-B996-5B03A66E4FCF}.Debug|Any CPU.Build.0 = Debug|Any CPU
+ {2D558C96-F052-4959-B996-5B03A66E4FCF}.Release|Any CPU.ActiveCfg = Release|Any CPU
+ {2D558C96-F052-4959-B996-5B03A66E4FCF}.Release|Any CPU.Build.0 = Release|Any CPU
+ {06BDEA4C-9036-44B7-AB44-DBD088556726}.Debug|Any CPU.ActiveCfg = Debug|Any CPU
+ {06BDEA4C-9036-44B7-AB44-DBD088556726}.Debug|Any CPU.Build.0 = Debug|Any CPU
+ {06BDEA4C-9036-44B7-AB44-DBD088556726}.Release|Any CPU.ActiveCfg = Release|Any CPU
+ {06BDEA4C-9036-44B7-AB44-DBD088556726}.Release|Any CPU.Build.0 = Release|Any CPU
+ {45AF3136-2F1A-48C5-A76F-52635D7F0B90}.Debug|Any CPU.ActiveCfg = Debug|Any CPU
+ {45AF3136-2F1A-48C5-A76F-52635D7F0B90}.Debug|Any CPU.Build.0 = Debug|Any CPU
+ {45AF3136-2F1A-48C5-A76F-52635D7F0B90}.Release|Any CPU.ActiveCfg = Release|Any CPU
+ {45AF3136-2F1A-48C5-A76F-52635D7F0B90}.Release|Any CPU.Build.0 = Release|Any CPU
EndGlobalSection
GlobalSection(NestedProjects) = preSolution
{93C8BC67-7F34-4935-A671-F0B12AEDB072} = {A6C9FED0-42B6-488C-8961-DE13F291434B}
@@ -65,5 +83,7 @@ Global
{93C8BC67-7F34-4935-A671-F0B12AEDB072} = {A6C9FED0-42B6-488C-8961-DE13F291434B}
{6DC31D8F-B71A-4A08-A892-5593F54EB82C} = {D10E009A-B3B8-4FB3-8B01-F21C383CD513}
{D4E8F259-596E-412D-A757-769383865764} = {A6C9FED0-42B6-488C-8961-DE13F291434B}
+ {06BDEA4C-9036-44B7-AB44-DBD088556726} = {A6C9FED0-42B6-488C-8961-DE13F291434B}
+ {45AF3136-2F1A-48C5-A76F-52635D7F0B90} = {A6C9FED0-42B6-488C-8961-DE13F291434B}
EndGlobalSection
EndGlobal
diff --git a/README.md b/README.md
index 9f88748..38714d0 100644
--- a/README.md
+++ b/README.md
@@ -21,6 +21,11 @@ Extensions for [LightResults](https://github.com/jscarle/LightResults), an extre
[![nuget](https://img.shields.io/nuget/v/LightResults.Extensions.Json)](https://www.nuget.org/packages/LightResults.Extensions.Json)
[![downloads](https://img.shields.io/nuget/dt/LightResults.Extensions.Json)](https://www.nuget.org/packages/LightResults.Extensions.Json)
+- [Json](https://jscarle.github.io/LightResults.Extensions/docs/generatedidentifier.html) - Provides strongly-typed identifiers.
+
+ [![nuget](https://img.shields.io/nuget/v/LightResults.Extensions.GeneratedIdentifier)](https://www.nuget.org/packages/LightResults.Extensions.GeneratedIdentifier)
+ [![downloads](https://img.shields.io/nuget/dt/LightResults.Extensions.GeneratedIdentifier)](https://www.nuget.org/packages/LightResults.Extensions.GeneratedIdentifier)
+
## Documentation
Make sure to [read the docs](https://jscarle.github.io/LightResults.Extensions/) for the full API.
diff --git a/docfx/docs/generatedidentifier.md b/docfx/docs/generatedidentifier.md
new file mode 100644
index 0000000..01b4f9f
--- /dev/null
+++ b/docfx/docs/generatedidentifier.md
@@ -0,0 +1,7 @@
+# GeneratedIdentifier
+
+A Roslyn source generator that provides strongly-typed identifiers using LightResults.
+
+[![main](https://img.shields.io/github/actions/workflow/status/jscarle/LightResults.Extensions/main.yml?logo=github)](https://github.com/jscarle/LightResults.Extensions)
+[![nuget](https://img.shields.io/nuget/v/LightResults.Extensions.GeneratedIdentifier)](https://www.nuget.org/packages/LightResults.Extensions.GeneratedIdentifier)
+[![downloads](https://img.shields.io/nuget/dt/LightResults.Extensions.GeneratedIdentifier)](https://www.nuget.org/packages/LightResults.Extensions.GeneratedIdentifier)
diff --git a/src/LightResults.Extensions.GeneratedIdentifier/Common/Declaration.cs b/src/LightResults.Extensions.GeneratedIdentifier/Common/Declaration.cs
new file mode 100644
index 0000000..e562850
--- /dev/null
+++ b/src/LightResults.Extensions.GeneratedIdentifier/Common/Declaration.cs
@@ -0,0 +1,59 @@
+namespace LightResults.Extensions.GeneratedIdentifier.Common;
+
+/// Represents a declaration.
+public sealed record Declaration
+{
+ /// Initializes a new instance of the class with the specified type, name, and generic parameters.
+ /// The type of declaration.
+ /// The name of the declaration.
+ /// A read-only list of generic parameter names, or an empty list if not generic.
+ internal Declaration(DeclarationType type, string name, EquatableImmutableArray genericParameters)
+ {
+ Type = type;
+ Name = name;
+ GenericParameters = genericParameters;
+ }
+
+ /// Gets the type of declaration.
+ public DeclarationType Type { get; }
+
+ /// Gets the name of the declaration.
+ public string Name { get; }
+
+ /// Gets a read-only list of generic parameter names for generic declarations, or an empty list otherwise.
+ public EquatableImmutableArray GenericParameters { get; }
+
+ /// Returns a string representation of the declaration in the appropriate format for its type.
+ /// A string representation of the declaration.
+ public override string ToString()
+ {
+ switch (Type)
+ {
+ case DeclarationType.Namespace:
+ return $"namespace {Name}";
+ case DeclarationType.Interface:
+ case DeclarationType.Class:
+ case DeclarationType.Record:
+ case DeclarationType.Struct:
+ case DeclarationType.RecordStruct:
+ var keyword = ToKeyword(Type);
+ return GenericParameters.Count == 0 ? $"{keyword} {Name}" : $"{keyword} {Name}<{string.Join(", ", GenericParameters)}>";
+ default:
+ return base.ToString();
+ }
+ }
+
+ private static string ToKeyword(DeclarationType declarationType)
+ {
+ return declarationType switch
+ {
+ DeclarationType.Namespace => "namespace",
+ DeclarationType.Interface => "interface",
+ DeclarationType.Class => "class",
+ DeclarationType.Record => "record",
+ DeclarationType.Struct => "struct",
+ DeclarationType.RecordStruct => "record struct",
+ _ => throw new InvalidOperationException(),
+ };
+ }
+}
diff --git a/src/LightResults.Extensions.GeneratedIdentifier/Common/DeclarationExtensions.cs b/src/LightResults.Extensions.GeneratedIdentifier/Common/DeclarationExtensions.cs
new file mode 100644
index 0000000..16db73b
--- /dev/null
+++ b/src/LightResults.Extensions.GeneratedIdentifier/Common/DeclarationExtensions.cs
@@ -0,0 +1,53 @@
+using System.Text;
+
+namespace LightResults.Extensions.GeneratedIdentifier.Common;
+
+/// Provides extension methods for working with declarations.
+internal static class DeclarationExtensions
+{
+ /// Converts a list of declarations to their corresponding namespace.
+ /// The list of declarations to convert.
+ /// The namespace represented by the declarations.
+ public static string ToNamespace(this EquatableImmutableArray declarations)
+ {
+ var builder = new StringBuilder();
+
+ for (var index = 0; index < declarations.Count; index++)
+ {
+ var declaration = declarations[index];
+ if (declaration.Type != DeclarationType.Namespace)
+ continue;
+
+ if (builder.Length > 0)
+ builder.Append('.');
+ builder.Append(declaration.Name);
+ }
+
+ return builder.ToString();
+ }
+
+ /// Converts a list of declarations to their fully qualified name.
+ /// The list of declarations to convert.
+ /// The fully qualified name represented by the declarations.
+ public static string ToFullyQualifiedName(this EquatableImmutableArray declarations)
+ {
+ var builder = new StringBuilder();
+
+ for (var index = 0; index < declarations.Count; index++)
+ {
+ var declaration = declarations[index];
+
+ if (builder.Length > 0)
+ builder.Append('.');
+ builder.Append(declaration.Name);
+
+ if (declaration.GenericParameters.Count == 0)
+ continue;
+
+ builder.Append('`');
+ builder.Append(declaration.GenericParameters.Count);
+ }
+
+ return builder.ToString();
+ }
+}
diff --git a/src/LightResults.Extensions.GeneratedIdentifier/Common/DeclarationType.cs b/src/LightResults.Extensions.GeneratedIdentifier/Common/DeclarationType.cs
new file mode 100644
index 0000000..7d971d1
--- /dev/null
+++ b/src/LightResults.Extensions.GeneratedIdentifier/Common/DeclarationType.cs
@@ -0,0 +1,23 @@
+namespace LightResults.Extensions.GeneratedIdentifier.Common;
+
+/// Specifies the kind of declaration.
+public enum DeclarationType
+{
+ /// Represents a namespace declaration.
+ Namespace = 0,
+
+ /// Represents an interface declaration.
+ Interface = 1,
+
+ /// Represents a class declaration.
+ Class = 2,
+
+ /// Represents a record declaration.
+ Record = 3,
+
+ /// Represents a struct declaration.
+ Struct = 4,
+
+ /// Represents a record struct declaration.
+ RecordStruct = 5,
+}
diff --git a/src/LightResults.Extensions.GeneratedIdentifier/Common/EquatableImmutableArray.cs b/src/LightResults.Extensions.GeneratedIdentifier/Common/EquatableImmutableArray.cs
new file mode 100644
index 0000000..6664e9d
--- /dev/null
+++ b/src/LightResults.Extensions.GeneratedIdentifier/Common/EquatableImmutableArray.cs
@@ -0,0 +1,15 @@
+using System.Collections.Immutable;
+
+namespace LightResults.Extensions.GeneratedIdentifier.Common;
+
+/// Provides extension methods to convert various collections to an .
+internal static class EquatableImmutableArray
+{
+ /// Converts an to an .
+ /// The to convert.
+ /// An containing the same elements as the original enumerable.
+ public static EquatableImmutableArray ToEquatableImmutableArray(this IEnumerable enumerable)
+ {
+ return new EquatableImmutableArray(enumerable.ToImmutableArray());
+ }
+}
diff --git a/src/LightResults.Extensions.GeneratedIdentifier/Common/EquatableImmutableArray`1.cs b/src/LightResults.Extensions.GeneratedIdentifier/Common/EquatableImmutableArray`1.cs
new file mode 100644
index 0000000..ccd0315
--- /dev/null
+++ b/src/LightResults.Extensions.GeneratedIdentifier/Common/EquatableImmutableArray`1.cs
@@ -0,0 +1,87 @@
+using System.Collections;
+using System.Collections.Immutable;
+
+namespace LightResults.Extensions.GeneratedIdentifier.Common;
+
+/// Represents an immutable array that implements .
+/// The type of elements in the array.
+public readonly struct EquatableImmutableArray : IEquatable>, IReadOnlyList
+{
+ /// Gets an empty .
+ internal static EquatableImmutableArray Empty { get; } = new(ImmutableArray.Empty);
+
+ /// Gets the number of elements in the array.
+ public int Count => Array.Length;
+
+ /// Gets the element at the specified index.
+ /// The zero-based index of the element to get.
+ /// The element at the specified index.
+ public T this[int index] => Array[index];
+
+ private ImmutableArray Array => _array ?? ImmutableArray.Empty;
+ private readonly ImmutableArray? _array;
+
+ internal EquatableImmutableArray(ImmutableArray? array)
+ {
+ _array = array;
+ }
+
+ ///
+ public bool Equals(EquatableImmutableArray other)
+ {
+ return this.SequenceEqual(other);
+ }
+
+ ///
+ public override bool Equals(object? obj)
+ {
+ return obj is EquatableImmutableArray other && Equals(other);
+ }
+
+ ///
+ public override int GetHashCode()
+ {
+ var hashCode = new HashCode();
+
+ foreach (var item in Array)
+ hashCode.Add(item);
+
+ return hashCode.ToHashCode();
+ }
+
+ /// Determines whether two instances are equal.
+ /// The first to compare.
+ /// The second to compare.
+ /// if the two instances are equal; otherwise, .
+ public static bool operator ==(EquatableImmutableArray left, EquatableImmutableArray right)
+ {
+ return left.Equals(right);
+ }
+
+ /// Determines whether two instances are not equal.
+ /// The first to compare.
+ /// The second to compare.
+ /// if the two instances are not equal; otherwise, .
+ public static bool operator !=(EquatableImmutableArray left, EquatableImmutableArray right)
+ {
+ return !left.Equals(right);
+ }
+
+ ///
+ public IEnumerator GetEnumerator()
+ {
+ // ReSharper disable once ForCanBeConvertedToForeach
+ // ReSharper disable once LoopCanBeConvertedToQuery
+ for (var index = 0; index < Array.Length; index++)
+ {
+ var item = Array[index];
+ yield return item;
+ }
+ }
+
+ ///
+ IEnumerator IEnumerable.GetEnumerator()
+ {
+ return GetEnumerator();
+ }
+}
diff --git a/src/LightResults.Extensions.GeneratedIdentifier/Common/IncrementalValueProviderExtensions.cs b/src/LightResults.Extensions.GeneratedIdentifier/Common/IncrementalValueProviderExtensions.cs
new file mode 100644
index 0000000..2f18bd3
--- /dev/null
+++ b/src/LightResults.Extensions.GeneratedIdentifier/Common/IncrementalValueProviderExtensions.cs
@@ -0,0 +1,13 @@
+using Microsoft.CodeAnalysis;
+
+namespace LightResults.Extensions.GeneratedIdentifier.Common;
+
+internal static class IncrementalValueProviderExtensions
+{
+ public static IncrementalValuesProvider WhereNotNull(this IncrementalValuesProvider source)
+ where TSource : struct
+ {
+ return source.Where(x => x is not null)
+ .Select((x, _) => x!.Value);
+ }
+}
diff --git a/src/LightResults.Extensions.GeneratedIdentifier/Common/SymbolExtensions.cs b/src/LightResults.Extensions.GeneratedIdentifier/Common/SymbolExtensions.cs
new file mode 100644
index 0000000..7460ab5
--- /dev/null
+++ b/src/LightResults.Extensions.GeneratedIdentifier/Common/SymbolExtensions.cs
@@ -0,0 +1,100 @@
+using Microsoft.CodeAnalysis;
+
+namespace LightResults.Extensions.GeneratedIdentifier.Common;
+
+/// Provides extension methods for working with symbols.
+internal static class SymbolExtensions
+{
+ /// Gets a list of declarations representing the hierarchy containing the given symbol.
+ /// The to get the containing declarations for.
+ /// The cancellation token.
+ /// An of objects representing the hierarchy.
+ public static EquatableImmutableArray GetContainingDeclarations(this ISymbol symbol, CancellationToken cancellationToken)
+ {
+ cancellationToken.ThrowIfCancellationRequested();
+
+ var declarations = new Stack();
+
+ BuildContainingSymbolHierarchy(symbol, in declarations, cancellationToken);
+
+ return declarations.ToEquatableImmutableArray();
+ }
+
+ private static void BuildContainingSymbolHierarchy(ISymbol symbol, in Stack declarations, CancellationToken cancellationToken)
+ {
+ cancellationToken.ThrowIfCancellationRequested();
+
+ switch (symbol.ContainingSymbol)
+ {
+ case INamespaceSymbol namespaceSymbol:
+ BuildNamespaceHierarchy(namespaceSymbol, declarations, cancellationToken);
+ break;
+ case INamedTypeSymbol namedTypeSymbol:
+ BuildTypeHierarchy(namedTypeSymbol, declarations, cancellationToken);
+ break;
+ }
+ }
+
+ private static void BuildNamespaceHierarchy(INamespaceSymbol symbol, in Stack declarations, CancellationToken cancellationToken)
+ {
+ cancellationToken.ThrowIfCancellationRequested();
+
+ if (!symbol.IsGlobalNamespace)
+ {
+ var namespaceDeclaration = new Declaration(DeclarationType.Namespace, symbol.Name, EquatableImmutableArray.Empty);
+ declarations.Push(namespaceDeclaration);
+ }
+
+ if (symbol.ContainingNamespace is not null && !symbol.ContainingNamespace.IsGlobalNamespace)
+ BuildNamespaceHierarchy(symbol.ContainingNamespace, declarations, cancellationToken);
+ }
+
+ private static void BuildTypeHierarchy(INamedTypeSymbol symbol, in Stack declarations, CancellationToken cancellationToken)
+ {
+ cancellationToken.ThrowIfCancellationRequested();
+
+ var declarationType = symbol.GetDeclarationType(cancellationToken);
+ if (declarationType is null)
+ return;
+
+ var genericTypeParameters = symbol.GetGenericTypeParameters(cancellationToken);
+
+ var typeDeclaration = new Declaration(declarationType.Value, symbol.Name, genericTypeParameters);
+ declarations.Push(typeDeclaration);
+
+ BuildContainingSymbolHierarchy(symbol, declarations, cancellationToken);
+ }
+
+ private static EquatableImmutableArray GetGenericTypeParameters(this INamedTypeSymbol symbol, CancellationToken cancellationToken)
+ {
+ cancellationToken.ThrowIfCancellationRequested();
+
+ if (!symbol.IsGenericType)
+ return EquatableImmutableArray.Empty;
+
+ var genericTypeParameters = new List();
+
+ for (var index = 0; index < symbol.TypeParameters.Length; index++)
+ {
+ var typeParameter = symbol.TypeParameters[index];
+ genericTypeParameters.Add(typeParameter.Name);
+ }
+
+ return genericTypeParameters.ToEquatableImmutableArray();
+ }
+
+ private static DeclarationType? GetDeclarationType(this ITypeSymbol symbol, CancellationToken cancellationToken)
+ {
+ cancellationToken.ThrowIfCancellationRequested();
+
+ return symbol switch
+ {
+ { IsReferenceType: true, TypeKind: TypeKind.Interface } => DeclarationType.Interface,
+ { IsReferenceType: true, IsRecord: true } => DeclarationType.Record,
+ { IsReferenceType: true } => DeclarationType.Class,
+ { IsValueType: true, IsRecord: true } => DeclarationType.RecordStruct,
+ { IsValueType: true } => DeclarationType.Struct,
+ _ => null,
+ };
+ }
+}
diff --git a/src/LightResults.Extensions.GeneratedIdentifier/GeneratedIdentifierSourceGenerator.cs b/src/LightResults.Extensions.GeneratedIdentifier/GeneratedIdentifierSourceGenerator.cs
new file mode 100644
index 0000000..59025a1
--- /dev/null
+++ b/src/LightResults.Extensions.GeneratedIdentifier/GeneratedIdentifierSourceGenerator.cs
@@ -0,0 +1,527 @@
+using System.Collections.Immutable;
+using System.Text;
+using LightResults.Extensions.GeneratedIdentifier.Common;
+using Microsoft.CodeAnalysis;
+using Microsoft.CodeAnalysis.CSharp;
+using Microsoft.CodeAnalysis.Text;
+
+namespace LightResults.Extensions.GeneratedIdentifier;
+
+[Generator]
+public sealed class GeneratedIdentifierSourceGenerator : IIncrementalGenerator
+{
+ private const string AttributesNamespace = "LightResults.Extensions.GeneratedIdentifier";
+ private const string GeneratedIdentifierAttributeName = "GeneratedIdentifierAttribute";
+ private const string GeneratedIdentifierAttributeFullyQualifiedName = $"{AttributesNamespace}.{GeneratedIdentifierAttributeName}`1";
+ private const string GeneratedIdentifierAttributeHint = $"{GeneratedIdentifierAttributeFullyQualifiedName}.g.cs";
+
+ private static readonly string FileHeader = $"""
+ //-----------------------------------------------------------------------------
+ //
+ // This code was generated by {nameof(GeneratedIdentifierSourceGenerator)} which
+ // can be found in the {typeof(GeneratedIdentifierSourceGenerator).Namespace} namespace.
+ //
+ // Changes to this file may cause incorrect behavior
+ // and will be lost if the code is regenerated.
+ //
+ //-----------------------------------------------------------------------------
+ #nullable enable
+ """;
+
+ public void Initialize(IncrementalGeneratorInitializationContext context)
+ {
+ context.RegisterPostInitializationOutput(RegisterAttributes);
+
+ var generatedIdentifiers = context.SyntaxProvider
+ .ForAttributeWithMetadataName(GeneratedIdentifierAttributeFullyQualifiedName, Filter, Transform)
+ .WhereNotNull()
+ .Collect();
+
+ context.RegisterSourceOutput(generatedIdentifiers, GenerateIdentifier);
+ }
+
+ private static void RegisterAttributes(IncrementalGeneratorPostInitializationContext context)
+ {
+ var source = $"""
+ {FileHeader}
+
+ using System;
+
+ namespace {AttributesNamespace};
+
+ [AttributeUsage(AttributeTargets.Struct)]
+ public sealed class {GeneratedIdentifierAttributeName} : Attribute;
+ """;
+ context.AddSource(GeneratedIdentifierAttributeHint, SourceText.From(source, Encoding.UTF8));
+ }
+
+ private static bool Filter(SyntaxNode syntaxNode, CancellationToken cancellationToken)
+ {
+ cancellationToken.ThrowIfCancellationRequested();
+
+ return syntaxNode.IsKind(SyntaxKind.StructDeclaration);
+ }
+
+ private static Identifier? Transform(GeneratorAttributeSyntaxContext context, CancellationToken cancellationToken)
+ {
+ cancellationToken.ThrowIfCancellationRequested();
+
+ if (context.TargetSymbol is not INamedTypeSymbol namedTypeSymbol)
+ return null;
+
+ var containingDeclarations = namedTypeSymbol.GetContainingDeclarations(cancellationToken);
+ var symbolName = namedTypeSymbol.Name;
+
+ var attribute = context.Attributes[0].AttributeClass!;
+ if (attribute.TypeArguments.Length != 1)
+ return null;
+
+ var typeArgument = attribute.TypeArguments[0];
+
+ string? declaredValueType;
+ string? fullValueType;
+ switch (typeArgument.SpecialType)
+ {
+ case SpecialType.System_Int16:
+ declaredValueType = "short";
+ fullValueType = "Int16";
+ break;
+ case SpecialType.System_Int32:
+ declaredValueType = "int";
+ fullValueType = "Int32";
+ break;
+ case SpecialType.System_Int64:
+ declaredValueType = "long";
+ fullValueType = "Int64";
+ break;
+ default:
+ if (typeArgument is not { Name: "Guid", ContainingNamespace: { Name: "System", ContainingNamespace.IsGlobalNamespace: true } })
+ return null;
+ declaredValueType = "Guid";
+ fullValueType = "Guid";
+ break;
+ }
+
+ var symbol = new Identifier(containingDeclarations, symbolName, declaredValueType, fullValueType);
+
+ return symbol;
+ }
+
+ private static void GenerateIdentifier(SourceProductionContext context, ImmutableArray generatedIdentifiers)
+ {
+ foreach (var symbol in generatedIdentifiers)
+ {
+ var structNamespace = symbol.ContainingDeclarations.ToNamespace();
+ var structName = symbol.Name;
+ var declaredValueType = symbol.DeclaredValueType;
+ var fullValueType = symbol.FullValueType;
+
+ var source = new StringBuilder();
+
+ source.AppendLine($"""
+ //-----------------------------------------------------------------------------
+ //
+ // This code was generated by {nameof(GeneratedIdentifierSourceGenerator)} which
+ // can be found in the {typeof(GeneratedIdentifierSourceGenerator).Namespace} namespace.
+ //
+ // Changes to this file may cause incorrect behavior
+ // and will be lost if the code is regenerated.
+ //
+ //-----------------------------------------------------------------------------
+
+ #nullable enable
+
+ using System.ComponentModel;
+ using System.Globalization;
+ using System.Text.Json;
+ using System.Text.Json.Serialization;
+ using LightResults;
+ using GeneratedIdentifier.Common.ValueObjects;
+
+ """
+ );
+
+ if (structNamespace.Length > 0)
+ source.AppendLine($"""
+ namespace {structNamespace};
+
+ """
+ );
+
+ source.AppendLine($$"""
+ [TypeConverter(typeof({{structName}}TypeConverter))]
+ [JsonConverter(typeof({{structName}}JsonConverter))]
+ readonly partial struct {{structName}} :
+ ICreatableValueObject<{{declaredValueType}}, {{structName}}>,
+ IParsableValueObject<{{structName}}>,
+ IValueObject<{{declaredValueType}}, {{structName}}>,
+ IComparable<{{structName}}>,
+ IComparable
+ {
+ """
+ );
+
+ source.AppendLine("""
+ /// Gets whether this identifier is the default value.
+ public bool IsDefault => _value == default;
+
+ """
+ );
+
+ source.AppendLine($"""
+ {declaredValueType} IValueObject<{declaredValueType}, {structName}>.Value => _value;
+
+ private readonly {declaredValueType} _value;
+
+ """
+ );
+
+ source.AppendLine($$"""
+ private {{structName}}({{declaredValueType}} value, bool skipValidation = false)
+ {
+ if (!skipValidation)
+ ValueObjectException.ThrowIfFailed(Validate(value));
+
+ _value = value;
+ }
+
+ """
+ );
+
+ source.AppendLine($$"""
+ ///
+ public static {{structName}} Create({{declaredValueType}} value)
+ {
+ var result = TryCreate(value);
+ if (result.IsSuccess(out var identifier, out var error))
+ return identifier;
+
+ throw new ValueObjectException(error.Message);
+ }
+
+ """
+ );
+
+ source.AppendLine($$"""
+ ///
+ public static Result<{{structName}}> TryCreate({{declaredValueType}} value)
+ {
+ var validation = Validate(value);
+ if (validation.IsFailed(out var error))
+ return Result.Fail<{{structName}}>(error);
+
+ return Result.Ok<{{structName}}>(new {{structName}}(value, true));
+ }
+
+ """
+ );
+
+ source.AppendLine($$"""
+ ///
+ public static {{structName}} Parse(string s)
+ {
+ var result = TryParse(s);
+ if (result.IsSuccess(out var identifier, out var error))
+ return identifier;
+
+ throw new ValueObjectException(error.Message);
+ }
+
+ """
+ );
+
+ source.AppendLine($$"""
+ ///
+ public static Result<{{structName}}> TryParse(string s)
+ {
+ if ({{declaredValueType}}.TryParse(s, out var value))
+ return TryCreate(value);
+
+ return Result.Fail<{{structName}}>("The string is not a valid identifier.");
+ }
+
+ """
+ );
+
+ source.AppendLine($$"""
+ ///
+ public static bool TryParse(string s, out {{structName}} identifier)
+ {
+ return TryParse(s).IsSuccess(out identifier);
+ }
+
+ """
+ );
+
+ source.AppendLine($$"""
+ ///
+ public static bool TryParse(string s, IFormatProvider provider, out {{structName}} identifier)
+ {
+ return TryParse(s).IsSuccess(out identifier);
+ }
+
+ """
+ );
+
+ source.AppendLine($$"""
+ ///
+ public bool Equals({{structName}} other)
+ {
+ return _value == other._value;
+ }
+
+ """
+ );
+
+ source.AppendLine($$"""
+ ///
+ public override bool Equals(object? obj)
+ {
+ return obj is {{structName}} other && Equals(other);
+ }
+
+ """
+ );
+
+ if (declaredValueType == "Guid")
+ source.AppendLine("""
+ ///
+ public override int GetHashCode()
+ {
+ return _value.GetHashCode();
+ }
+
+ """
+ );
+ else
+ source.AppendLine("""
+ ///
+ public override int GetHashCode()
+ {
+ return _value;
+ }
+
+ """
+ );
+
+ source.AppendLine($$"""
+ /// Determines whether two instances of are equal.
+ /// The first instance to compare.
+ /// The second instance to compare.
+ /// true if the instances are equal; otherwise, false.
+ public static bool operator ==({{structName}} left, {{structName}} right)
+ {
+ return left.Equals(right);
+ }
+
+ """
+ );
+
+ source.AppendLine($$"""
+ /// Determines whether two instances of are not equal.
+ /// The first instance to compare.
+ /// The second instance to compare.
+ /// true if the instances are not equal; otherwise, false.
+ public static bool operator !=({{structName}} left, {{structName}} right)
+ {
+ return !left.Equals(right);
+ }
+
+ """
+ );
+
+ source.AppendLine($$"""
+ ///
+ public int CompareTo({{structName}} other)
+ {
+ return _value.CompareTo(other._value);
+ }
+
+ """
+ );
+
+ source.AppendLine($$"""
+ ///
+ public int CompareTo(object? obj)
+ {
+ if (ReferenceEquals(null, obj)) return 1;
+ return obj is {{structName}} other ? CompareTo(other) : throw new ArgumentException($"Object must be of type {nameof({{structName}})}");
+ }
+
+ """
+ );
+
+ source.AppendLine($$"""
+ /// Determines whether the first instance of is less than the second instance.
+ /// The first instance to compare.
+ /// The second instance to compare.
+ /// true if the first instance is less than the second instance; otherwise, false.
+ public static bool operator <({{structName}} left, {{structName}} right)
+ {
+ return left.CompareTo(right) < 0;
+ }
+
+ """
+ );
+
+ source.AppendLine($$"""
+ /// Determines whether the first instance of is greater than the second instance.
+ /// The first instance to compare.
+ /// The second instance to compare.
+ /// true if the first instance is greater than the second instance; otherwise, false.
+ public static bool operator >({{structName}} left, {{structName}} right)
+ {
+ return left.CompareTo(right) > 0;
+ }
+
+ """
+ );
+
+ source.AppendLine($$"""
+ /// Determines whether the first instance of is less than or equal to the second instance.
+ /// The first instance to compare.
+ /// The second instance to compare.
+ /// true if the first instance is less than or equal to the second instance; otherwise, false.
+ public static bool operator <=({{structName}} left, {{structName}} right)
+ {
+ return left.CompareTo(right) <= 0;
+ }
+
+ """
+ );
+
+ source.AppendLine($$"""
+ /// Determines whether the first instance of is greater than or equal to the second instance.
+ /// The first instance to compare.
+ /// The second instance to compare.
+ /// true if the first instance is greater than or equal to the second instance; otherwise, false.
+ public static bool operator >=({{structName}} left, {{structName}} right)
+ {
+ return left.CompareTo(right) >= 0;
+ }
+
+ """
+ );
+
+ source.AppendLine($$"""
+ /// Gets the underlying value of the .
+ /// The underlying value of the .
+ public {{declaredValueType}} To{{fullValueType}}()
+ {
+ return _value;
+ }
+
+ """
+ );
+
+ if (declaredValueType == "Guid")
+ source.AppendLine("""
+ ///
+ public override string ToString()
+ {
+ return _value.ToString();
+ }
+
+ """
+ );
+ else
+ source.AppendLine("""
+ ///
+ public override string ToString()
+ {
+ return _value.ToString(CultureInfo.InvariantCulture);
+ }
+
+ """
+ );
+
+ if (declaredValueType == "Guid")
+ source.AppendLine($$"""
+ private static Result Validate({{declaredValueType}} value)
+ {
+ return Result.Ok();
+ }
+ """
+ );
+ else
+ source.AppendLine($$"""
+ private static Result Validate({{declaredValueType}} value)
+ {
+ if (value < 0)
+ return Result.Fail("The value must be equal to or greater than zero.");
+
+ return Result.Ok();
+ }
+ """
+ );
+
+ source.AppendLine("""
+ }
+
+ """
+ );
+
+ source.AppendLine($$"""
+ public class {{structName}}TypeConverter : TypeConverter
+ {
+ public override bool CanConvertFrom(ITypeDescriptorContext? context, Type sourceType)
+ {
+ return sourceType == typeof({{declaredValueType}}) || base.CanConvertFrom(context, sourceType);
+ }
+
+ public override object? ConvertFrom(ITypeDescriptorContext? context, CultureInfo? culture, object value)
+ {
+ if (value is {{declaredValueType}} identifierValue)
+ return {{structName}}.Create(identifierValue);
+
+ return base.ConvertFrom(context, culture, value);
+ }
+ }
+
+ """
+ );
+ source.AppendLine($$"""
+ public class {{structName}}JsonConverter : JsonConverter<{{structName}}>
+ {
+ public override void Write(Utf8JsonWriter writer, {{structName}} identifier, JsonSerializerOptions options)
+ {
+ var value = ((IValueObject<{{declaredValueType}}, {{structName}}>)identifier).Value;
+ """
+ );
+
+ if (declaredValueType == "Guid")
+ source.AppendLine(""" writer.WriteStringValue(value.ToString());""");
+ else
+ source.AppendLine(""" writer.WriteNumberValue(value);""");
+
+ source.Append($$"""
+ }
+
+ public override {{structName}} Read(ref Utf8JsonReader reader, Type typeToConvert, JsonSerializerOptions options)
+ {
+ var value = reader.Get{{fullValueType}}();
+ return {{structName}}.Create(value);
+ }
+ }
+
+ """
+ );
+
+ var hint = $"{symbol.ContainingDeclarations.ToFullyQualifiedName()}.{symbol.Name}.g.cs";
+ context.AddSource(hint, source.ToString());
+ }
+ }
+
+ private readonly record struct Identifier(
+ EquatableImmutableArray ContainingDeclarations,
+ string Name,
+ string DeclaredValueType,
+ string FullValueType
+ )
+ {
+ public EquatableImmutableArray ContainingDeclarations { get; } = ContainingDeclarations;
+ public string Name { get; } = Name;
+ public string DeclaredValueType { get; } = DeclaredValueType;
+ public string FullValueType { get; } = FullValueType;
+ }
+}
diff --git a/src/LightResults.Extensions.GeneratedIdentifier/LightResults.Extensions.GeneratedIdentifier.csproj b/src/LightResults.Extensions.GeneratedIdentifier/LightResults.Extensions.GeneratedIdentifier.csproj
new file mode 100644
index 0000000..524093c
--- /dev/null
+++ b/src/LightResults.Extensions.GeneratedIdentifier/LightResults.Extensions.GeneratedIdentifier.csproj
@@ -0,0 +1,83 @@
+
+
+
+
+ LightResults.Extensions.GeneratedIdentifier
+ netstandard2.0
+ enable
+ enable
+ latest
+
+
+
+
+ latest-all
+ true
+ true
+ true
+ $(NoWarn);NU5128
+
+
+
+
+
+
+ all
+ runtime; build; native; contentfiles; analyzers; buildtransitive
+
+
+
+
+ all
+ runtime; build; native; contentfiles; analyzers; buildtransitive
+
+
+
+
+
+ LightResults.Extensions.GeneratedIdentifier
+ 9.0.0-preview.1
+ 9.0.0.0
+ 9.0.0.0
+ en-US
+ false
+
+
+
+
+ true
+ LightResults.Extensions.GeneratedIdentifier
+ LightResults.Extensions.GeneratedIdentifier
+ A Roslyn source generator that automatically creates strongly-typed identifier structs.
+ README.md
+ https://github.com/jscarle/LightResults.Extensions
+ result results pattern lightresults source-generator strongly-typed identifie
+ https://github.com/jscarle/LightResults.Extensions
+ git
+ Jean-Sebastien Carle
+ Copyright © Jean-Sebastien Carle 2024
+ LICENSE.md
+ Icon.png
+
+
+
+
+
+
+
+
+
+
+
+
+ true
+ $(GetTargetPathDependsOn);GetDependencyTargetPaths
+
+
+
+
+
+
+
+
+
diff --git a/src/LightResults.Extensions.GeneratedIdentifier/Properties/launchSettings.json b/src/LightResults.Extensions.GeneratedIdentifier/Properties/launchSettings.json
new file mode 100644
index 0000000..29c3748
--- /dev/null
+++ b/src/LightResults.Extensions.GeneratedIdentifier/Properties/launchSettings.json
@@ -0,0 +1,9 @@
+{
+ "$schema": "https://json.schemastore.org/launchsettings.json",
+ "profiles": {
+ "DebugRoslynSourceGenerator": {
+ "commandName": "DebugRoslynComponent",
+ "targetProject": "../GeneratedIdentifier.Sample/GeneratedIdentifier.Sample.csproj"
+ }
+ }
+}
\ No newline at end of file
diff --git a/tests/LightResults.Extensions.GeneratedIdentifier.Fixtures/Identifiers/TestGuidId.cs b/tests/LightResults.Extensions.GeneratedIdentifier.Fixtures/Identifiers/TestGuidId.cs
new file mode 100644
index 0000000..ecfebad
--- /dev/null
+++ b/tests/LightResults.Extensions.GeneratedIdentifier.Fixtures/Identifiers/TestGuidId.cs
@@ -0,0 +1,6 @@
+using GeneratedIdentifier;
+
+namespace LightResults.Extensions.GeneratedIdentifier.Fixtures.Identifiers;
+
+[GeneratedIdentifier]
+public partial struct TestGuidId;
diff --git a/tests/LightResults.Extensions.GeneratedIdentifier.Fixtures/Identifiers/TestIntId.cs b/tests/LightResults.Extensions.GeneratedIdentifier.Fixtures/Identifiers/TestIntId.cs
new file mode 100644
index 0000000..e6c515b
--- /dev/null
+++ b/tests/LightResults.Extensions.GeneratedIdentifier.Fixtures/Identifiers/TestIntId.cs
@@ -0,0 +1,6 @@
+using GeneratedIdentifier;
+
+namespace LightResults.Extensions.GeneratedIdentifier.Fixtures.Identifiers;
+
+[GeneratedIdentifier]
+public partial struct TestIntId;
diff --git a/tests/LightResults.Extensions.GeneratedIdentifier.Fixtures/Identifiers/TestShortId.cs b/tests/LightResults.Extensions.GeneratedIdentifier.Fixtures/Identifiers/TestShortId.cs
new file mode 100644
index 0000000..baee8e1
--- /dev/null
+++ b/tests/LightResults.Extensions.GeneratedIdentifier.Fixtures/Identifiers/TestShortId.cs
@@ -0,0 +1,6 @@
+using GeneratedIdentifier;
+
+namespace LightResults.Extensions.GeneratedIdentifier.Fixtures.Identifiers;
+
+[GeneratedIdentifier]
+public partial struct TestShortId;
diff --git a/tests/LightResults.Extensions.GeneratedIdentifier.Fixtures/Interfaces/ICloneableValueObject.cs b/tests/LightResults.Extensions.GeneratedIdentifier.Fixtures/Interfaces/ICloneableValueObject.cs
new file mode 100644
index 0000000..7651e14
--- /dev/null
+++ b/tests/LightResults.Extensions.GeneratedIdentifier.Fixtures/Interfaces/ICloneableValueObject.cs
@@ -0,0 +1,9 @@
+// Resharper disable CheckNamespace
+
+namespace GeneratedIdentifier.Common.ValueObjects;
+
+public interface ICloneableValueObject : IValueObject
+ where TSelf : notnull
+{
+ TSelf Clone();
+}
diff --git a/tests/LightResults.Extensions.GeneratedIdentifier.Fixtures/Interfaces/IConvertibleValueObject.cs b/tests/LightResults.Extensions.GeneratedIdentifier.Fixtures/Interfaces/IConvertibleValueObject.cs
new file mode 100644
index 0000000..0e1e792
--- /dev/null
+++ b/tests/LightResults.Extensions.GeneratedIdentifier.Fixtures/Interfaces/IConvertibleValueObject.cs
@@ -0,0 +1,17 @@
+using System.Diagnostics.CodeAnalysis;
+using LightResults;
+
+// Resharper disable CheckNamespace
+namespace GeneratedIdentifier.Common.ValueObjects;
+
+[SuppressMessage(
+ "Design",
+ "CA1000: Do not declare static members on generic types",
+ Justification = "This is required for handling value objects generically."
+)]
+public interface IConvertibleValueObject : IValueObject
+ where TSelf : notnull
+{
+ static abstract TSelf Convert(TSource source);
+ static abstract Result TryConvert(TSource source);
+}
diff --git a/tests/LightResults.Extensions.GeneratedIdentifier.Fixtures/Interfaces/ICreatableValueObject.cs b/tests/LightResults.Extensions.GeneratedIdentifier.Fixtures/Interfaces/ICreatableValueObject.cs
new file mode 100644
index 0000000..a203424
--- /dev/null
+++ b/tests/LightResults.Extensions.GeneratedIdentifier.Fixtures/Interfaces/ICreatableValueObject.cs
@@ -0,0 +1,17 @@
+using System.Diagnostics.CodeAnalysis;
+using LightResults;
+
+// Resharper disable CheckNamespace
+namespace GeneratedIdentifier.Common.ValueObjects;
+
+[SuppressMessage(
+ "Design",
+ "CA1000: Do not declare static members on generic types",
+ Justification = "This is required for handling value objects generically."
+)]
+public interface ICreatableValueObject : IValueObject
+ where TSelf : notnull
+{
+ static abstract TSelf Create(TValue value);
+ static abstract Result TryCreate(TValue value);
+}
diff --git a/tests/LightResults.Extensions.GeneratedIdentifier.Fixtures/Interfaces/IParsableValueObject.cs b/tests/LightResults.Extensions.GeneratedIdentifier.Fixtures/Interfaces/IParsableValueObject.cs
new file mode 100644
index 0000000..9979b84
--- /dev/null
+++ b/tests/LightResults.Extensions.GeneratedIdentifier.Fixtures/Interfaces/IParsableValueObject.cs
@@ -0,0 +1,19 @@
+using System.Diagnostics.CodeAnalysis;
+using LightResults;
+
+// Resharper disable CheckNamespace
+namespace GeneratedIdentifier.Common.ValueObjects;
+
+[SuppressMessage(
+ "Design",
+ "CA1000: Do not declare static members on generic types",
+ Justification = "This is required for handling value objects generically."
+)]
+public interface IParsableValueObject : IValueObject
+ where TSelf : notnull
+{
+ static abstract TSelf Parse(string s);
+ static abstract Result TryParse(string s);
+ static abstract bool TryParse(string s, out TSelf result);
+ static abstract bool TryParse(string s, IFormatProvider provider, out TSelf result);
+}
diff --git a/tests/LightResults.Extensions.GeneratedIdentifier.Fixtures/Interfaces/IValueObject.cs b/tests/LightResults.Extensions.GeneratedIdentifier.Fixtures/Interfaces/IValueObject.cs
new file mode 100644
index 0000000..a9591f5
--- /dev/null
+++ b/tests/LightResults.Extensions.GeneratedIdentifier.Fixtures/Interfaces/IValueObject.cs
@@ -0,0 +1,14 @@
+// Resharper disable CheckNamespace
+
+namespace GeneratedIdentifier.Common.ValueObjects;
+
+public interface IValueObject : IValueObject
+ where TSelf : notnull
+{
+ TValue Value { get; }
+}
+
+public interface IValueObject : IValueObject, IEquatable, IComparable, IComparable
+ where TSelf : notnull;
+
+public interface IValueObject;
diff --git a/tests/LightResults.Extensions.GeneratedIdentifier.Fixtures/Interfaces/ValueObjectValidationException.cs b/tests/LightResults.Extensions.GeneratedIdentifier.Fixtures/Interfaces/ValueObjectValidationException.cs
new file mode 100644
index 0000000..23fd614
--- /dev/null
+++ b/tests/LightResults.Extensions.GeneratedIdentifier.Fixtures/Interfaces/ValueObjectValidationException.cs
@@ -0,0 +1,27 @@
+using IResult = LightResults.IResult;
+
+// Resharper disable CheckNamespace
+namespace GeneratedIdentifier.Common.ValueObjects;
+
+public sealed class ValueObjectException : Exception
+{
+ public ValueObjectException()
+ {
+ }
+
+ public ValueObjectException(string message)
+ : base(message)
+ {
+ }
+
+ public ValueObjectException(string message, Exception innerException)
+ : base(message, innerException)
+ {
+ }
+
+ public static void ThrowIfFailed(IResult result)
+ {
+ if (result.IsFailed(out var error))
+ throw new ValueObjectException(error.Message);
+ }
+}
diff --git a/tests/LightResults.Extensions.GeneratedIdentifier.Fixtures/LightResults.Extensions.GeneratedIdentifier.Fixtures.csproj b/tests/LightResults.Extensions.GeneratedIdentifier.Fixtures/LightResults.Extensions.GeneratedIdentifier.Fixtures.csproj
new file mode 100644
index 0000000..931a9f0
--- /dev/null
+++ b/tests/LightResults.Extensions.GeneratedIdentifier.Fixtures/LightResults.Extensions.GeneratedIdentifier.Fixtures.csproj
@@ -0,0 +1,28 @@
+
+
+
+ net8.0
+ enable
+ enable
+ latest
+ LightResults.Extensions.GeneratedIdentifier.Fixtures
+ LightResults.Extensions.GeneratedIdentifier.Fixtures
+ latest-Default
+ false
+ false
+
+
+
+
+
+
+
+
+
+
+
+
diff --git a/tests/LightResults.Extensions.GeneratedIdentifier.Fixtures/Program.cs b/tests/LightResults.Extensions.GeneratedIdentifier.Fixtures/Program.cs
new file mode 100644
index 0000000..5d25b47
--- /dev/null
+++ b/tests/LightResults.Extensions.GeneratedIdentifier.Fixtures/Program.cs
@@ -0,0 +1,8 @@
+namespace LightResults.Extensions.GeneratedIdentifier.Fixtures;
+
+public static class Program
+{
+ public static void Main()
+ {
+ }
+}
diff --git a/tests/LightResults.Extensions.GeneratedIdentifier.Fixtures/Properties/launchSettings.json b/tests/LightResults.Extensions.GeneratedIdentifier.Fixtures/Properties/launchSettings.json
new file mode 100644
index 0000000..b7cdb7f
--- /dev/null
+++ b/tests/LightResults.Extensions.GeneratedIdentifier.Fixtures/Properties/launchSettings.json
@@ -0,0 +1,12 @@
+{
+ "profiles": {
+ "Api.SourceGenerators.Fixtures": {
+ "commandName": "Project",
+ "launchBrowser": true,
+ "environmentVariables": {
+ "ASPNETCORE_ENVIRONMENT": "Development"
+ },
+ "applicationUrl": "https://localhost:59000;http://localhost:59001"
+ }
+ }
+}
\ No newline at end of file
diff --git a/tests/LightResults.Extensions.GeneratedIdentifier.Tests/GeneratedIdentifierSourceGeneratorTests.GenerateGuidIdentifier_WithNamespace.verified.txt b/tests/LightResults.Extensions.GeneratedIdentifier.Tests/GeneratedIdentifierSourceGeneratorTests.GenerateGuidIdentifier_WithNamespace.verified.txt
new file mode 100644
index 0000000..0436efc
--- /dev/null
+++ b/tests/LightResults.Extensions.GeneratedIdentifier.Tests/GeneratedIdentifierSourceGeneratorTests.GenerateGuidIdentifier_WithNamespace.verified.txt
@@ -0,0 +1,230 @@
+//-----------------------------------------------------------------------------
+//
+// This code was generated by GeneratedIdentifierSourceGenerator which
+// can be found in the LightResults.Extensions.GeneratedIdentifier namespace.
+//
+// Changes to this file may cause incorrect behavior
+// and will be lost if the code is regenerated.
+//
+//-----------------------------------------------------------------------------
+
+#nullable enable
+
+using System.ComponentModel;
+using System.Globalization;
+using System.Text.Json;
+using System.Text.Json.Serialization;
+using LightResults;
+using GeneratedIdentifier.Common.ValueObjects;
+
+namespace MyProject.Identifiers;
+
+[TypeConverter(typeof(TestGuidIdTypeConverter))]
+[JsonConverter(typeof(TestGuidIdJsonConverter))]
+readonly partial struct TestGuidId :
+ ICreatableValueObject,
+ IParsableValueObject,
+ IValueObject,
+ IComparable,
+ IComparable
+{
+ /// Gets whether this identifier is the default value.
+ public bool IsDefault => _value == default;
+
+ Guid IValueObject.Value => _value;
+
+ private readonly Guid _value;
+
+ private TestGuidId(Guid value, bool skipValidation = false)
+ {
+ if (!skipValidation)
+ ValueObjectException.ThrowIfFailed(Validate(value));
+
+ _value = value;
+ }
+
+ ///
+ public static TestGuidId Create(Guid value)
+ {
+ var result = TryCreate(value);
+ if (result.IsSuccess(out var identifier, out var error))
+ return identifier;
+
+ throw new ValueObjectException(error.Message);
+ }
+
+ ///
+ public static Result TryCreate(Guid value)
+ {
+ var validation = Validate(value);
+ if (validation.IsFailed(out var error))
+ return Result.Fail(error);
+
+ return Result.Ok(new TestGuidId(value, true));
+ }
+
+ ///
+ public static TestGuidId Parse(string s)
+ {
+ var result = TryParse(s);
+ if (result.IsSuccess(out var identifier, out var error))
+ return identifier;
+
+ throw new ValueObjectException(error.Message);
+ }
+
+ ///
+ public static Result TryParse(string s)
+ {
+ if (Guid.TryParse(s, out var value))
+ return TryCreate(value);
+
+ return Result.Fail("The string is not a valid identifier.");
+ }
+
+ ///
+ public static bool TryParse(string s, out TestGuidId identifier)
+ {
+ return TryParse(s).IsSuccess(out identifier);
+ }
+
+ ///
+ public static bool TryParse(string s, IFormatProvider provider, out TestGuidId identifier)
+ {
+ return TryParse(s).IsSuccess(out identifier);
+ }
+
+ ///
+ public bool Equals(TestGuidId other)
+ {
+ return _value == other._value;
+ }
+
+ ///
+ public override bool Equals(object? obj)
+ {
+ return obj is TestGuidId other && Equals(other);
+ }
+
+ ///
+ public override int GetHashCode()
+ {
+ return _value.GetHashCode();
+ }
+
+ /// Determines whether two instances of are equal.
+ /// The first instance to compare.
+ /// The second instance to compare.
+ /// true if the instances are equal; otherwise, false.
+ public static bool operator ==(TestGuidId left, TestGuidId right)
+ {
+ return left.Equals(right);
+ }
+
+ /// Determines whether two instances of are not equal.
+ /// The first instance to compare.
+ /// The second instance to compare.
+ /// true if the instances are not equal; otherwise, false.
+ public static bool operator !=(TestGuidId left, TestGuidId right)
+ {
+ return !left.Equals(right);
+ }
+
+ ///
+ public int CompareTo(TestGuidId other)
+ {
+ return _value.CompareTo(other._value);
+ }
+
+ ///
+ public int CompareTo(object? obj)
+ {
+ if (ReferenceEquals(null, obj)) return 1;
+ return obj is TestGuidId other ? CompareTo(other) : throw new ArgumentException($"Object must be of type {nameof(TestGuidId)}");
+ }
+
+ /// Determines whether the first instance of is less than the second instance.
+ /// The first instance to compare.
+ /// The second instance to compare.
+ /// true if the first instance is less than the second instance; otherwise, false.
+ public static bool operator <(TestGuidId left, TestGuidId right)
+ {
+ return left.CompareTo(right) < 0;
+ }
+
+ /// Determines whether the first instance of is greater than the second instance.
+ /// The first instance to compare.
+ /// The second instance to compare.
+ /// true if the first instance is greater than the second instance; otherwise, false.
+ public static bool operator >(TestGuidId left, TestGuidId right)
+ {
+ return left.CompareTo(right) > 0;
+ }
+
+ /// Determines whether the first instance of is less than or equal to the second instance.
+ /// The first instance to compare.
+ /// The second instance to compare.
+ /// true if the first instance is less than or equal to the second instance; otherwise, false.
+ public static bool operator <=(TestGuidId left, TestGuidId right)
+ {
+ return left.CompareTo(right) <= 0;
+ }
+
+ /// Determines whether the first instance of is greater than or equal to the second instance.
+ /// The first instance to compare.
+ /// The second instance to compare.
+ /// true if the first instance is greater than or equal to the second instance; otherwise, false.
+ public static bool operator >=(TestGuidId left, TestGuidId right)
+ {
+ return left.CompareTo(right) >= 0;
+ }
+
+ /// Gets the underlying value of the .
+ /// The underlying value of the .
+ public Guid ToGuid()
+ {
+ return _value;
+ }
+
+ ///
+ public override string ToString()
+ {
+ return _value.ToString();
+ }
+
+ private static Result Validate(Guid value)
+ {
+ return Result.Ok();
+ }
+}
+
+public class TestGuidIdTypeConverter : TypeConverter
+{
+ public override bool CanConvertFrom(ITypeDescriptorContext? context, Type sourceType)
+ {
+ return sourceType == typeof(Guid) || base.CanConvertFrom(context, sourceType);
+ }
+
+ public override object? ConvertFrom(ITypeDescriptorContext? context, CultureInfo? culture, object value)
+ {
+ if (value is Guid identifierValue)
+ return TestGuidId.Create(identifierValue);
+
+ return base.ConvertFrom(context, culture, value);
+ }
+}
+
+public class TestGuidIdJsonConverter : JsonConverter
+{
+ public override void Write(Utf8JsonWriter writer, TestGuidId identifier, JsonSerializerOptions options)
+ {
+ var value = ((IValueObject)identifier).Value;
+ writer.WriteStringValue(value.ToString());
+ }
+
+ public override TestGuidId Read(ref Utf8JsonReader reader, Type typeToConvert, JsonSerializerOptions options)
+ {
+ var value = reader.GetGuid();
+ return TestGuidId.Create(value);
+ }
+}
diff --git a/tests/LightResults.Extensions.GeneratedIdentifier.Tests/GeneratedIdentifierSourceGeneratorTests.GenerateGuidIdentifier_WithoutNamespace.verified.txt b/tests/LightResults.Extensions.GeneratedIdentifier.Tests/GeneratedIdentifierSourceGeneratorTests.GenerateGuidIdentifier_WithoutNamespace.verified.txt
new file mode 100644
index 0000000..98ca6ee
--- /dev/null
+++ b/tests/LightResults.Extensions.GeneratedIdentifier.Tests/GeneratedIdentifierSourceGeneratorTests.GenerateGuidIdentifier_WithoutNamespace.verified.txt
@@ -0,0 +1,228 @@
+//-----------------------------------------------------------------------------
+//
+// This code was generated by GeneratedIdentifierSourceGenerator which
+// can be found in the LightResults.Extensions.GeneratedIdentifier namespace.
+//
+// Changes to this file may cause incorrect behavior
+// and will be lost if the code is regenerated.
+//
+//-----------------------------------------------------------------------------
+
+#nullable enable
+
+using System.ComponentModel;
+using System.Globalization;
+using System.Text.Json;
+using System.Text.Json.Serialization;
+using LightResults;
+using GeneratedIdentifier.Common.ValueObjects;
+
+[TypeConverter(typeof(TestGuidIdTypeConverter))]
+[JsonConverter(typeof(TestGuidIdJsonConverter))]
+readonly partial struct TestGuidId :
+ ICreatableValueObject,
+ IParsableValueObject,
+ IValueObject,
+ IComparable,
+ IComparable
+{
+ /// Gets whether this identifier is the default value.
+ public bool IsDefault => _value == default;
+
+ Guid IValueObject.Value => _value;
+
+ private readonly Guid _value;
+
+ private TestGuidId(Guid value, bool skipValidation = false)
+ {
+ if (!skipValidation)
+ ValueObjectException.ThrowIfFailed(Validate(value));
+
+ _value = value;
+ }
+
+ ///
+ public static TestGuidId Create(Guid value)
+ {
+ var result = TryCreate(value);
+ if (result.IsSuccess(out var identifier, out var error))
+ return identifier;
+
+ throw new ValueObjectException(error.Message);
+ }
+
+ ///
+ public static Result TryCreate(Guid value)
+ {
+ var validation = Validate(value);
+ if (validation.IsFailed(out var error))
+ return Result.Fail(error);
+
+ return Result.Ok(new TestGuidId(value, true));
+ }
+
+ ///
+ public static TestGuidId Parse(string s)
+ {
+ var result = TryParse(s);
+ if (result.IsSuccess(out var identifier, out var error))
+ return identifier;
+
+ throw new ValueObjectException(error.Message);
+ }
+
+ ///
+ public static Result TryParse(string s)
+ {
+ if (Guid.TryParse(s, out var value))
+ return TryCreate(value);
+
+ return Result.Fail("The string is not a valid identifier.");
+ }
+
+ ///
+ public static bool TryParse(string s, out TestGuidId identifier)
+ {
+ return TryParse(s).IsSuccess(out identifier);
+ }
+
+ ///
+ public static bool TryParse(string s, IFormatProvider provider, out TestGuidId identifier)
+ {
+ return TryParse(s).IsSuccess(out identifier);
+ }
+
+ ///
+ public bool Equals(TestGuidId other)
+ {
+ return _value == other._value;
+ }
+
+ ///
+ public override bool Equals(object? obj)
+ {
+ return obj is TestGuidId other && Equals(other);
+ }
+
+ ///
+ public override int GetHashCode()
+ {
+ return _value.GetHashCode();
+ }
+
+ /// Determines whether two instances of are equal.
+ /// The first instance to compare.
+ /// The second instance to compare.
+ /// true if the instances are equal; otherwise, false.
+ public static bool operator ==(TestGuidId left, TestGuidId right)
+ {
+ return left.Equals(right);
+ }
+
+ /// Determines whether two instances of are not equal.
+ /// The first instance to compare.
+ /// The second instance to compare.
+ /// true if the instances are not equal; otherwise, false.
+ public static bool operator !=(TestGuidId left, TestGuidId right)
+ {
+ return !left.Equals(right);
+ }
+
+ ///
+ public int CompareTo(TestGuidId other)
+ {
+ return _value.CompareTo(other._value);
+ }
+
+ ///
+ public int CompareTo(object? obj)
+ {
+ if (ReferenceEquals(null, obj)) return 1;
+ return obj is TestGuidId other ? CompareTo(other) : throw new ArgumentException($"Object must be of type {nameof(TestGuidId)}");
+ }
+
+ /// Determines whether the first instance of is less than the second instance.
+ /// The first instance to compare.
+ /// The second instance to compare.
+ /// true if the first instance is less than the second instance; otherwise, false.
+ public static bool operator <(TestGuidId left, TestGuidId right)
+ {
+ return left.CompareTo(right) < 0;
+ }
+
+ /// Determines whether the first instance of is greater than the second instance.
+ /// The first instance to compare.
+ /// The second instance to compare.
+ /// true if the first instance is greater than the second instance; otherwise, false.
+ public static bool operator >(TestGuidId left, TestGuidId right)
+ {
+ return left.CompareTo(right) > 0;
+ }
+
+ /// Determines whether the first instance of is less than or equal to the second instance.
+ /// The first instance to compare.
+ /// The second instance to compare.
+ /// true if the first instance is less than or equal to the second instance; otherwise, false.
+ public static bool operator <=(TestGuidId left, TestGuidId right)
+ {
+ return left.CompareTo(right) <= 0;
+ }
+
+ /// Determines whether the first instance of is greater than or equal to the second instance.
+ /// The first instance to compare.
+ /// The second instance to compare.
+ /// true if the first instance is greater than or equal to the second instance; otherwise, false.
+ public static bool operator >=(TestGuidId left, TestGuidId right)
+ {
+ return left.CompareTo(right) >= 0;
+ }
+
+ /// Gets the underlying value of the .
+ /// The underlying value of the .
+ public Guid ToGuid()
+ {
+ return _value;
+ }
+
+ ///
+ public override string ToString()
+ {
+ return _value.ToString();
+ }
+
+ private static Result Validate(Guid value)
+ {
+ return Result.Ok();
+ }
+}
+
+public class TestGuidIdTypeConverter : TypeConverter
+{
+ public override bool CanConvertFrom(ITypeDescriptorContext? context, Type sourceType)
+ {
+ return sourceType == typeof(Guid) || base.CanConvertFrom(context, sourceType);
+ }
+
+ public override object? ConvertFrom(ITypeDescriptorContext? context, CultureInfo? culture, object value)
+ {
+ if (value is Guid identifierValue)
+ return TestGuidId.Create(identifierValue);
+
+ return base.ConvertFrom(context, culture, value);
+ }
+}
+
+public class TestGuidIdJsonConverter : JsonConverter
+{
+ public override void Write(Utf8JsonWriter writer, TestGuidId identifier, JsonSerializerOptions options)
+ {
+ var value = ((IValueObject)identifier).Value;
+ writer.WriteStringValue(value.ToString());
+ }
+
+ public override TestGuidId Read(ref Utf8JsonReader reader, Type typeToConvert, JsonSerializerOptions options)
+ {
+ var value = reader.GetGuid();
+ return TestGuidId.Create(value);
+ }
+}
diff --git a/tests/LightResults.Extensions.GeneratedIdentifier.Tests/GeneratedIdentifierSourceGeneratorTests.GenerateIntIdentifier_WithNamespace.verified.txt b/tests/LightResults.Extensions.GeneratedIdentifier.Tests/GeneratedIdentifierSourceGeneratorTests.GenerateIntIdentifier_WithNamespace.verified.txt
new file mode 100644
index 0000000..8d23b64
--- /dev/null
+++ b/tests/LightResults.Extensions.GeneratedIdentifier.Tests/GeneratedIdentifierSourceGeneratorTests.GenerateIntIdentifier_WithNamespace.verified.txt
@@ -0,0 +1,233 @@
+//-----------------------------------------------------------------------------
+//
+// This code was generated by GeneratedIdentifierSourceGenerator which
+// can be found in the LightResults.Extensions.GeneratedIdentifier namespace.
+//
+// Changes to this file may cause incorrect behavior
+// and will be lost if the code is regenerated.
+//
+//-----------------------------------------------------------------------------
+
+#nullable enable
+
+using System.ComponentModel;
+using System.Globalization;
+using System.Text.Json;
+using System.Text.Json.Serialization;
+using LightResults;
+using GeneratedIdentifier.Common.ValueObjects;
+
+namespace MyProject.Identifiers;
+
+[TypeConverter(typeof(TestIntIdTypeConverter))]
+[JsonConverter(typeof(TestIntIdJsonConverter))]
+readonly partial struct TestIntId :
+ ICreatableValueObject,
+ IParsableValueObject,
+ IValueObject,
+ IComparable,
+ IComparable
+{
+ /// Gets whether this identifier is the default value.
+ public bool IsDefault => _value == default;
+
+ int IValueObject.Value => _value;
+
+ private readonly int _value;
+
+ private TestIntId(int value, bool skipValidation = false)
+ {
+ if (!skipValidation)
+ ValueObjectException.ThrowIfFailed(Validate(value));
+
+ _value = value;
+ }
+
+ ///
+ public static TestIntId Create(int value)
+ {
+ var result = TryCreate(value);
+ if (result.IsSuccess(out var identifier, out var error))
+ return identifier;
+
+ throw new ValueObjectException(error.Message);
+ }
+
+ ///
+ public static Result TryCreate(int value)
+ {
+ var validation = Validate(value);
+ if (validation.IsFailed(out var error))
+ return Result.Fail(error);
+
+ return Result.Ok(new TestIntId(value, true));
+ }
+
+ ///
+ public static TestIntId Parse(string s)
+ {
+ var result = TryParse(s);
+ if (result.IsSuccess(out var identifier, out var error))
+ return identifier;
+
+ throw new ValueObjectException(error.Message);
+ }
+
+ ///
+ public static Result TryParse(string s)
+ {
+ if (int.TryParse(s, out var value))
+ return TryCreate(value);
+
+ return Result.Fail("The string is not a valid identifier.");
+ }
+
+ ///
+ public static bool TryParse(string s, out TestIntId identifier)
+ {
+ return TryParse(s).IsSuccess(out identifier);
+ }
+
+ ///
+ public static bool TryParse(string s, IFormatProvider provider, out TestIntId identifier)
+ {
+ return TryParse(s).IsSuccess(out identifier);
+ }
+
+ ///
+ public bool Equals(TestIntId other)
+ {
+ return _value == other._value;
+ }
+
+ ///
+ public override bool Equals(object? obj)
+ {
+ return obj is TestIntId other && Equals(other);
+ }
+
+ ///
+ public override int GetHashCode()
+ {
+ return _value;
+ }
+
+ /// Determines whether two instances of are equal.
+ /// The first instance to compare.
+ /// The second instance to compare.
+ /// true if the instances are equal; otherwise, false.
+ public static bool operator ==(TestIntId left, TestIntId right)
+ {
+ return left.Equals(right);
+ }
+
+ /// Determines whether two instances of are not equal.
+ /// The first instance to compare.
+ /// The second instance to compare.
+ /// true if the instances are not equal; otherwise, false.
+ public static bool operator !=(TestIntId left, TestIntId right)
+ {
+ return !left.Equals(right);
+ }
+
+ ///
+ public int CompareTo(TestIntId other)
+ {
+ return _value.CompareTo(other._value);
+ }
+
+ ///
+ public int CompareTo(object? obj)
+ {
+ if (ReferenceEquals(null, obj)) return 1;
+ return obj is TestIntId other ? CompareTo(other) : throw new ArgumentException($"Object must be of type {nameof(TestIntId)}");
+ }
+
+ /// Determines whether the first instance of is less than the second instance.
+ /// The first instance to compare.
+ /// The second instance to compare.
+ /// true if the first instance is less than the second instance; otherwise, false.
+ public static bool operator <(TestIntId left, TestIntId right)
+ {
+ return left.CompareTo(right) < 0;
+ }
+
+ /// Determines whether the first instance of is greater than the second instance.
+ /// The first instance to compare.
+ /// The second instance to compare.
+ /// true if the first instance is greater than the second instance; otherwise, false.
+ public static bool operator >(TestIntId left, TestIntId right)
+ {
+ return left.CompareTo(right) > 0;
+ }
+
+ /// Determines whether the first instance of is less than or equal to the second instance.
+ /// The first instance to compare.
+ /// The second instance to compare.
+ /// true if the first instance is less than or equal to the second instance; otherwise, false.
+ public static bool operator <=(TestIntId left, TestIntId right)
+ {
+ return left.CompareTo(right) <= 0;
+ }
+
+ /// Determines whether the first instance of is greater than or equal to the second instance.
+ /// The first instance to compare.
+ /// The second instance to compare.
+ /// true if the first instance is greater than or equal to the second instance; otherwise, false.
+ public static bool operator >=(TestIntId left, TestIntId right)
+ {
+ return left.CompareTo(right) >= 0;
+ }
+
+ /// Gets the underlying value of the .
+ /// The underlying value of the .
+ public int ToInt32()
+ {
+ return _value;
+ }
+
+ ///
+ public override string ToString()
+ {
+ return _value.ToString(CultureInfo.InvariantCulture);
+ }
+
+ private static Result Validate(int value)
+ {
+ if (value < 0)
+ return Result.Fail("The value must be equal to or greater than zero.");
+
+ return Result.Ok();
+ }
+}
+
+public class TestIntIdTypeConverter : TypeConverter
+{
+ public override bool CanConvertFrom(ITypeDescriptorContext? context, Type sourceType)
+ {
+ return sourceType == typeof(int) || base.CanConvertFrom(context, sourceType);
+ }
+
+ public override object? ConvertFrom(ITypeDescriptorContext? context, CultureInfo? culture, object value)
+ {
+ if (value is int identifierValue)
+ return TestIntId.Create(identifierValue);
+
+ return base.ConvertFrom(context, culture, value);
+ }
+}
+
+public class TestIntIdJsonConverter : JsonConverter
+{
+ public override void Write(Utf8JsonWriter writer, TestIntId identifier, JsonSerializerOptions options)
+ {
+ var value = ((IValueObject)identifier).Value;
+ writer.WriteNumberValue(value);
+ }
+
+ public override TestIntId Read(ref Utf8JsonReader reader, Type typeToConvert, JsonSerializerOptions options)
+ {
+ var value = reader.GetInt32();
+ return TestIntId.Create(value);
+ }
+}
diff --git a/tests/LightResults.Extensions.GeneratedIdentifier.Tests/GeneratedIdentifierSourceGeneratorTests.GenerateIntIdentifier_WithoutNamespace.verified.txt b/tests/LightResults.Extensions.GeneratedIdentifier.Tests/GeneratedIdentifierSourceGeneratorTests.GenerateIntIdentifier_WithoutNamespace.verified.txt
new file mode 100644
index 0000000..a90adba
--- /dev/null
+++ b/tests/LightResults.Extensions.GeneratedIdentifier.Tests/GeneratedIdentifierSourceGeneratorTests.GenerateIntIdentifier_WithoutNamespace.verified.txt
@@ -0,0 +1,231 @@
+//-----------------------------------------------------------------------------
+//
+// This code was generated by GeneratedIdentifierSourceGenerator which
+// can be found in the LightResults.Extensions.GeneratedIdentifier namespace.
+//
+// Changes to this file may cause incorrect behavior
+// and will be lost if the code is regenerated.
+//
+//-----------------------------------------------------------------------------
+
+#nullable enable
+
+using System.ComponentModel;
+using System.Globalization;
+using System.Text.Json;
+using System.Text.Json.Serialization;
+using LightResults;
+using GeneratedIdentifier.Common.ValueObjects;
+
+[TypeConverter(typeof(TestIntIdTypeConverter))]
+[JsonConverter(typeof(TestIntIdJsonConverter))]
+readonly partial struct TestIntId :
+ ICreatableValueObject,
+ IParsableValueObject,
+ IValueObject,
+ IComparable,
+ IComparable
+{
+ /// Gets whether this identifier is the default value.
+ public bool IsDefault => _value == default;
+
+ int IValueObject.Value => _value;
+
+ private readonly int _value;
+
+ private TestIntId(int value, bool skipValidation = false)
+ {
+ if (!skipValidation)
+ ValueObjectException.ThrowIfFailed(Validate(value));
+
+ _value = value;
+ }
+
+ ///
+ public static TestIntId Create(int value)
+ {
+ var result = TryCreate(value);
+ if (result.IsSuccess(out var identifier, out var error))
+ return identifier;
+
+ throw new ValueObjectException(error.Message);
+ }
+
+ ///
+ public static Result TryCreate(int value)
+ {
+ var validation = Validate(value);
+ if (validation.IsFailed(out var error))
+ return Result.Fail(error);
+
+ return Result.Ok(new TestIntId(value, true));
+ }
+
+ ///
+ public static TestIntId Parse(string s)
+ {
+ var result = TryParse(s);
+ if (result.IsSuccess(out var identifier, out var error))
+ return identifier;
+
+ throw new ValueObjectException(error.Message);
+ }
+
+ ///
+ public static Result TryParse(string s)
+ {
+ if (int.TryParse(s, out var value))
+ return TryCreate(value);
+
+ return Result.Fail("The string is not a valid identifier.");
+ }
+
+ ///
+ public static bool TryParse(string s, out TestIntId identifier)
+ {
+ return TryParse(s).IsSuccess(out identifier);
+ }
+
+ ///
+ public static bool TryParse(string s, IFormatProvider provider, out TestIntId identifier)
+ {
+ return TryParse(s).IsSuccess(out identifier);
+ }
+
+ ///
+ public bool Equals(TestIntId other)
+ {
+ return _value == other._value;
+ }
+
+ ///
+ public override bool Equals(object? obj)
+ {
+ return obj is TestIntId other && Equals(other);
+ }
+
+ ///
+ public override int GetHashCode()
+ {
+ return _value;
+ }
+
+ /// Determines whether two instances of are equal.
+ /// The first instance to compare.
+ /// The second instance to compare.
+ /// true if the instances are equal; otherwise, false.
+ public static bool operator ==(TestIntId left, TestIntId right)
+ {
+ return left.Equals(right);
+ }
+
+ /// Determines whether two instances of are not equal.
+ /// The first instance to compare.
+ /// The second instance to compare.
+ /// true if the instances are not equal; otherwise, false.
+ public static bool operator !=(TestIntId left, TestIntId right)
+ {
+ return !left.Equals(right);
+ }
+
+ ///
+ public int CompareTo(TestIntId other)
+ {
+ return _value.CompareTo(other._value);
+ }
+
+ ///
+ public int CompareTo(object? obj)
+ {
+ if (ReferenceEquals(null, obj)) return 1;
+ return obj is TestIntId other ? CompareTo(other) : throw new ArgumentException($"Object must be of type {nameof(TestIntId)}");
+ }
+
+ /// Determines whether the first instance of is less than the second instance.
+ /// The first instance to compare.
+ /// The second instance to compare.
+ /// true if the first instance is less than the second instance; otherwise, false.
+ public static bool operator <(TestIntId left, TestIntId right)
+ {
+ return left.CompareTo(right) < 0;
+ }
+
+ /// Determines whether the first instance of is greater than the second instance.
+ /// The first instance to compare.
+ /// The second instance to compare.
+ /// true if the first instance is greater than the second instance; otherwise, false.
+ public static bool operator >(TestIntId left, TestIntId right)
+ {
+ return left.CompareTo(right) > 0;
+ }
+
+ /// Determines whether the first instance of is less than or equal to the second instance.
+ /// The first instance to compare.
+ /// The second instance to compare.
+ /// true if the first instance is less than or equal to the second instance; otherwise, false.
+ public static bool operator <=(TestIntId left, TestIntId right)
+ {
+ return left.CompareTo(right) <= 0;
+ }
+
+ /// Determines whether the first instance of is greater than or equal to the second instance.
+ /// The first instance to compare.
+ /// The second instance to compare.
+ /// true if the first instance is greater than or equal to the second instance; otherwise, false.
+ public static bool operator >=(TestIntId left, TestIntId right)
+ {
+ return left.CompareTo(right) >= 0;
+ }
+
+ /// Gets the underlying value of the .
+ /// The underlying value of the .
+ public int ToInt32()
+ {
+ return _value;
+ }
+
+ ///
+ public override string ToString()
+ {
+ return _value.ToString(CultureInfo.InvariantCulture);
+ }
+
+ private static Result Validate(int value)
+ {
+ if (value < 0)
+ return Result.Fail("The value must be equal to or greater than zero.");
+
+ return Result.Ok();
+ }
+}
+
+public class TestIntIdTypeConverter : TypeConverter
+{
+ public override bool CanConvertFrom(ITypeDescriptorContext? context, Type sourceType)
+ {
+ return sourceType == typeof(int) || base.CanConvertFrom(context, sourceType);
+ }
+
+ public override object? ConvertFrom(ITypeDescriptorContext? context, CultureInfo? culture, object value)
+ {
+ if (value is int identifierValue)
+ return TestIntId.Create(identifierValue);
+
+ return base.ConvertFrom(context, culture, value);
+ }
+}
+
+public class TestIntIdJsonConverter : JsonConverter
+{
+ public override void Write(Utf8JsonWriter writer, TestIntId identifier, JsonSerializerOptions options)
+ {
+ var value = ((IValueObject)identifier).Value;
+ writer.WriteNumberValue(value);
+ }
+
+ public override TestIntId Read(ref Utf8JsonReader reader, Type typeToConvert, JsonSerializerOptions options)
+ {
+ var value = reader.GetInt32();
+ return TestIntId.Create(value);
+ }
+}
diff --git a/tests/LightResults.Extensions.GeneratedIdentifier.Tests/GeneratedIdentifierSourceGeneratorTests.GenerateShortIdentifier_WithNamespace.verified.txt b/tests/LightResults.Extensions.GeneratedIdentifier.Tests/GeneratedIdentifierSourceGeneratorTests.GenerateShortIdentifier_WithNamespace.verified.txt
new file mode 100644
index 0000000..c676b2c
--- /dev/null
+++ b/tests/LightResults.Extensions.GeneratedIdentifier.Tests/GeneratedIdentifierSourceGeneratorTests.GenerateShortIdentifier_WithNamespace.verified.txt
@@ -0,0 +1,233 @@
+//-----------------------------------------------------------------------------
+//
+// This code was generated by GeneratedIdentifierSourceGenerator which
+// can be found in the LightResults.Extensions.GeneratedIdentifier namespace.
+//
+// Changes to this file may cause incorrect behavior
+// and will be lost if the code is regenerated.
+//
+//-----------------------------------------------------------------------------
+
+#nullable enable
+
+using System.ComponentModel;
+using System.Globalization;
+using System.Text.Json;
+using System.Text.Json.Serialization;
+using LightResults;
+using GeneratedIdentifier.Common.ValueObjects;
+
+namespace MyProject.Identifiers;
+
+[TypeConverter(typeof(TestShortIdTypeConverter))]
+[JsonConverter(typeof(TestShortIdJsonConverter))]
+readonly partial struct TestShortId :
+ ICreatableValueObject,
+ IParsableValueObject,
+ IValueObject,
+ IComparable,
+ IComparable
+{
+ /// Gets whether this identifier is the default value.
+ public bool IsDefault => _value == default;
+
+ short IValueObject.Value => _value;
+
+ private readonly short _value;
+
+ private TestShortId(short value, bool skipValidation = false)
+ {
+ if (!skipValidation)
+ ValueObjectException.ThrowIfFailed(Validate(value));
+
+ _value = value;
+ }
+
+ ///
+ public static TestShortId Create(short value)
+ {
+ var result = TryCreate(value);
+ if (result.IsSuccess(out var identifier, out var error))
+ return identifier;
+
+ throw new ValueObjectException(error.Message);
+ }
+
+ ///
+ public static Result TryCreate(short value)
+ {
+ var validation = Validate(value);
+ if (validation.IsFailed(out var error))
+ return Result.Fail(error);
+
+ return Result.Ok(new TestShortId(value, true));
+ }
+
+ ///
+ public static TestShortId Parse(string s)
+ {
+ var result = TryParse(s);
+ if (result.IsSuccess(out var identifier, out var error))
+ return identifier;
+
+ throw new ValueObjectException(error.Message);
+ }
+
+ ///
+ public static Result TryParse(string s)
+ {
+ if (short.TryParse(s, out var value))
+ return TryCreate(value);
+
+ return Result.Fail("The string is not a valid identifier.");
+ }
+
+ ///
+ public static bool TryParse(string s, out TestShortId identifier)
+ {
+ return TryParse(s).IsSuccess(out identifier);
+ }
+
+ ///
+ public static bool TryParse(string s, IFormatProvider provider, out TestShortId identifier)
+ {
+ return TryParse(s).IsSuccess(out identifier);
+ }
+
+ ///
+ public bool Equals(TestShortId other)
+ {
+ return _value == other._value;
+ }
+
+ ///
+ public override bool Equals(object? obj)
+ {
+ return obj is TestShortId other && Equals(other);
+ }
+
+ ///
+ public override int GetHashCode()
+ {
+ return _value;
+ }
+
+ /// Determines whether two instances of are equal.
+ /// The first instance to compare.
+ /// The second instance to compare.
+ /// true if the instances are equal; otherwise, false.
+ public static bool operator ==(TestShortId left, TestShortId right)
+ {
+ return left.Equals(right);
+ }
+
+ /// Determines whether two instances of are not equal.
+ /// The first instance to compare.
+ /// The second instance to compare.
+ /// true if the instances are not equal; otherwise, false.
+ public static bool operator !=(TestShortId left, TestShortId right)
+ {
+ return !left.Equals(right);
+ }
+
+ ///
+ public int CompareTo(TestShortId other)
+ {
+ return _value.CompareTo(other._value);
+ }
+
+ ///
+ public int CompareTo(object? obj)
+ {
+ if (ReferenceEquals(null, obj)) return 1;
+ return obj is TestShortId other ? CompareTo(other) : throw new ArgumentException($"Object must be of type {nameof(TestShortId)}");
+ }
+
+ /// Determines whether the first instance of is less than the second instance.
+ /// The first instance to compare.
+ /// The second instance to compare.
+ /// true if the first instance is less than the second instance; otherwise, false.
+ public static bool operator <(TestShortId left, TestShortId right)
+ {
+ return left.CompareTo(right) < 0;
+ }
+
+ /// Determines whether the first instance of is greater than the second instance.
+ /// The first instance to compare.
+ /// The second instance to compare.
+ /// true if the first instance is greater than the second instance; otherwise, false.
+ public static bool operator >(TestShortId left, TestShortId right)
+ {
+ return left.CompareTo(right) > 0;
+ }
+
+ /// Determines whether the first instance of is less than or equal to the second instance.
+ /// The first instance to compare.
+ /// The second instance to compare.
+ /// true if the first instance is less than or equal to the second instance; otherwise, false.
+ public static bool operator <=(TestShortId left, TestShortId right)
+ {
+ return left.CompareTo(right) <= 0;
+ }
+
+ /// Determines whether the first instance of is greater than or equal to the second instance.
+ /// The first instance to compare.
+ /// The second instance to compare.
+ /// true if the first instance is greater than or equal to the second instance; otherwise, false.
+ public static bool operator >=(TestShortId left, TestShortId right)
+ {
+ return left.CompareTo(right) >= 0;
+ }
+
+ /// Gets the underlying value of the .
+ /// The underlying value of the .
+ public short ToInt16()
+ {
+ return _value;
+ }
+
+ ///
+ public override string ToString()
+ {
+ return _value.ToString(CultureInfo.InvariantCulture);
+ }
+
+ private static Result Validate(short value)
+ {
+ if (value < 0)
+ return Result.Fail("The value must be equal to or greater than zero.");
+
+ return Result.Ok();
+ }
+}
+
+public class TestShortIdTypeConverter : TypeConverter
+{
+ public override bool CanConvertFrom(ITypeDescriptorContext? context, Type sourceType)
+ {
+ return sourceType == typeof(short) || base.CanConvertFrom(context, sourceType);
+ }
+
+ public override object? ConvertFrom(ITypeDescriptorContext? context, CultureInfo? culture, object value)
+ {
+ if (value is short identifierValue)
+ return TestShortId.Create(identifierValue);
+
+ return base.ConvertFrom(context, culture, value);
+ }
+}
+
+public class TestShortIdJsonConverter : JsonConverter
+{
+ public override void Write(Utf8JsonWriter writer, TestShortId identifier, JsonSerializerOptions options)
+ {
+ var value = ((IValueObject)identifier).Value;
+ writer.WriteNumberValue(value);
+ }
+
+ public override TestShortId Read(ref Utf8JsonReader reader, Type typeToConvert, JsonSerializerOptions options)
+ {
+ var value = reader.GetInt16();
+ return TestShortId.Create(value);
+ }
+}
diff --git a/tests/LightResults.Extensions.GeneratedIdentifier.Tests/GeneratedIdentifierSourceGeneratorTests.GenerateShortIdentifier_WithoutNamespace.verified.txt b/tests/LightResults.Extensions.GeneratedIdentifier.Tests/GeneratedIdentifierSourceGeneratorTests.GenerateShortIdentifier_WithoutNamespace.verified.txt
new file mode 100644
index 0000000..21dacba
--- /dev/null
+++ b/tests/LightResults.Extensions.GeneratedIdentifier.Tests/GeneratedIdentifierSourceGeneratorTests.GenerateShortIdentifier_WithoutNamespace.verified.txt
@@ -0,0 +1,231 @@
+//-----------------------------------------------------------------------------
+//
+// This code was generated by GeneratedIdentifierSourceGenerator which
+// can be found in the LightResults.Extensions.GeneratedIdentifier namespace.
+//
+// Changes to this file may cause incorrect behavior
+// and will be lost if the code is regenerated.
+//
+//-----------------------------------------------------------------------------
+
+#nullable enable
+
+using System.ComponentModel;
+using System.Globalization;
+using System.Text.Json;
+using System.Text.Json.Serialization;
+using LightResults;
+using GeneratedIdentifier.Common.ValueObjects;
+
+[TypeConverter(typeof(TestShortIdTypeConverter))]
+[JsonConverter(typeof(TestShortIdJsonConverter))]
+readonly partial struct TestShortId :
+ ICreatableValueObject,
+ IParsableValueObject,
+ IValueObject,
+ IComparable,
+ IComparable
+{
+ /// Gets whether this identifier is the default value.
+ public bool IsDefault => _value == default;
+
+ short IValueObject.Value => _value;
+
+ private readonly short _value;
+
+ private TestShortId(short value, bool skipValidation = false)
+ {
+ if (!skipValidation)
+ ValueObjectException.ThrowIfFailed(Validate(value));
+
+ _value = value;
+ }
+
+ ///
+ public static TestShortId Create(short value)
+ {
+ var result = TryCreate(value);
+ if (result.IsSuccess(out var identifier, out var error))
+ return identifier;
+
+ throw new ValueObjectException(error.Message);
+ }
+
+ ///
+ public static Result TryCreate(short value)
+ {
+ var validation = Validate(value);
+ if (validation.IsFailed(out var error))
+ return Result.Fail(error);
+
+ return Result.Ok(new TestShortId(value, true));
+ }
+
+ ///
+ public static TestShortId Parse(string s)
+ {
+ var result = TryParse(s);
+ if (result.IsSuccess(out var identifier, out var error))
+ return identifier;
+
+ throw new ValueObjectException(error.Message);
+ }
+
+ ///
+ public static Result TryParse(string s)
+ {
+ if (short.TryParse(s, out var value))
+ return TryCreate(value);
+
+ return Result.Fail("The string is not a valid identifier.");
+ }
+
+ ///
+ public static bool TryParse(string s, out TestShortId identifier)
+ {
+ return TryParse(s).IsSuccess(out identifier);
+ }
+
+ ///
+ public static bool TryParse(string s, IFormatProvider provider, out TestShortId identifier)
+ {
+ return TryParse(s).IsSuccess(out identifier);
+ }
+
+ ///
+ public bool Equals(TestShortId other)
+ {
+ return _value == other._value;
+ }
+
+ ///
+ public override bool Equals(object? obj)
+ {
+ return obj is TestShortId other && Equals(other);
+ }
+
+ ///
+ public override int GetHashCode()
+ {
+ return _value;
+ }
+
+ /// Determines whether two instances of are equal.
+ /// The first instance to compare.
+ /// The second instance to compare.
+ /// true if the instances are equal; otherwise, false.
+ public static bool operator ==(TestShortId left, TestShortId right)
+ {
+ return left.Equals(right);
+ }
+
+ /// Determines whether two instances of are not equal.
+ /// The first instance to compare.
+ /// The second instance to compare.
+ /// true if the instances are not equal; otherwise, false.
+ public static bool operator !=(TestShortId left, TestShortId right)
+ {
+ return !left.Equals(right);
+ }
+
+ ///
+ public int CompareTo(TestShortId other)
+ {
+ return _value.CompareTo(other._value);
+ }
+
+ ///
+ public int CompareTo(object? obj)
+ {
+ if (ReferenceEquals(null, obj)) return 1;
+ return obj is TestShortId other ? CompareTo(other) : throw new ArgumentException($"Object must be of type {nameof(TestShortId)}");
+ }
+
+ /// Determines whether the first instance of is less than the second instance.
+ /// The first instance to compare.
+ /// The second instance to compare.
+ /// true if the first instance is less than the second instance; otherwise, false.
+ public static bool operator <(TestShortId left, TestShortId right)
+ {
+ return left.CompareTo(right) < 0;
+ }
+
+ /// Determines whether the first instance of is greater than the second instance.
+ /// The first instance to compare.
+ /// The second instance to compare.
+ /// true if the first instance is greater than the second instance; otherwise, false.
+ public static bool operator >(TestShortId left, TestShortId right)
+ {
+ return left.CompareTo(right) > 0;
+ }
+
+ /// Determines whether the first instance of is less than or equal to the second instance.
+ /// The first instance to compare.
+ /// The second instance to compare.
+ /// true if the first instance is less than or equal to the second instance; otherwise, false.
+ public static bool operator <=(TestShortId left, TestShortId right)
+ {
+ return left.CompareTo(right) <= 0;
+ }
+
+ /// Determines whether the first instance of is greater than or equal to the second instance.
+ /// The first instance to compare.
+ /// The second instance to compare.
+ /// true if the first instance is greater than or equal to the second instance; otherwise, false.
+ public static bool operator >=(TestShortId left, TestShortId right)
+ {
+ return left.CompareTo(right) >= 0;
+ }
+
+ /// Gets the underlying value of the .
+ /// The underlying value of the .
+ public short ToInt16()
+ {
+ return _value;
+ }
+
+ ///
+ public override string ToString()
+ {
+ return _value.ToString(CultureInfo.InvariantCulture);
+ }
+
+ private static Result Validate(short value)
+ {
+ if (value < 0)
+ return Result.Fail("The value must be equal to or greater than zero.");
+
+ return Result.Ok();
+ }
+}
+
+public class TestShortIdTypeConverter : TypeConverter
+{
+ public override bool CanConvertFrom(ITypeDescriptorContext? context, Type sourceType)
+ {
+ return sourceType == typeof(short) || base.CanConvertFrom(context, sourceType);
+ }
+
+ public override object? ConvertFrom(ITypeDescriptorContext? context, CultureInfo? culture, object value)
+ {
+ if (value is short identifierValue)
+ return TestShortId.Create(identifierValue);
+
+ return base.ConvertFrom(context, culture, value);
+ }
+}
+
+public class TestShortIdJsonConverter : JsonConverter
+{
+ public override void Write(Utf8JsonWriter writer, TestShortId identifier, JsonSerializerOptions options)
+ {
+ var value = ((IValueObject)identifier).Value;
+ writer.WriteNumberValue(value);
+ }
+
+ public override TestShortId Read(ref Utf8JsonReader reader, Type typeToConvert, JsonSerializerOptions options)
+ {
+ var value = reader.GetInt16();
+ return TestShortId.Create(value);
+ }
+}
diff --git a/tests/LightResults.Extensions.GeneratedIdentifier.Tests/GeneratedIdentifierSourceGeneratorTests.cs b/tests/LightResults.Extensions.GeneratedIdentifier.Tests/GeneratedIdentifierSourceGeneratorTests.cs
new file mode 100644
index 0000000..ba8db70
--- /dev/null
+++ b/tests/LightResults.Extensions.GeneratedIdentifier.Tests/GeneratedIdentifierSourceGeneratorTests.cs
@@ -0,0 +1,104 @@
+using System.Collections.Immutable;
+using Basic.Reference.Assemblies;
+using LightResults.Extensions.GeneratedIdentifier;
+using LightResults.Extensions.GeneratedIdentifier.Tests;
+using Microsoft.CodeAnalysis;
+using SourceGeneratorTestHelpers;
+using SourceGeneratorTestHelpers.XUnit;
+
+namespace GeneratedIdentifier.Tests;
+
+public sealed class GeneratedIdentifierSourceGeneratorTests
+{
+ static GeneratedIdentifierSourceGeneratorTests()
+ {
+ ModuleInitializer.Initialize();
+ }
+
+ [Theory]
+ [InlineData(true)]
+ [InlineData(false)]
+ public async Task GenerateGuidIdentifier(bool withNamespace)
+ {
+ var sources = GetSources("""
+ /// Represents an identifier.
+ [GeneratedIdentifier]
+ public partial struct TestGuidId;
+ """, withNamespace
+ );
+
+ var result = RunGenerator(sources);
+ await result.VerifyAsync("TestGuidId.g.cs")
+ .UseMethodName($"{nameof(GenerateGuidIdentifier)}_With{(withNamespace ? "" : "out")}Namespace");
+ }
+
+ [Theory]
+ [InlineData(true)]
+ [InlineData(false)]
+ public async Task GenerateIntIdentifier(bool withNamespace)
+ {
+ var sources = GetSources("""
+ /// Represents an identifier.
+ [GeneratedIdentifier]
+ public partial struct TestIntId;
+ """, withNamespace
+ );
+
+ var result = RunGenerator(sources);
+ await result.VerifyAsync("TestIntId.g.cs")
+ .UseMethodName($"{nameof(GenerateIntIdentifier)}_With{(withNamespace ? "" : "out")}Namespace");
+ }
+
+ [Theory]
+ [InlineData(true)]
+ [InlineData(false)]
+ public async Task GenerateShortIdentifier(bool withNamespace)
+ {
+ var sources = GetSources("""
+ /// Represents an identifier.
+ [GeneratedIdentifier]
+ public partial struct TestShortId;
+ """, withNamespace
+ );
+
+ var result = RunGenerator(sources);
+ await result.VerifyAsync("TestShortId.g.cs")
+ .UseMethodName($"{nameof(GenerateShortIdentifier)}_With{(withNamespace ? "" : "out")}Namespace");
+ }
+
+ private static IEnumerable GetSources(string source, bool withNamespace = true)
+ {
+ const string usingStatements = """
+ using System;
+ using LightResults.Extensions.GeneratedIdentifier;
+ """;
+
+ if (withNamespace)
+ yield return $"""
+ {usingStatements}
+
+ namespace MyProject.Identifiers;
+
+ {source}
+ """;
+ else
+ yield return $"""
+ {usingStatements}
+
+ {source}
+ """;
+ yield return """
+ using System;
+
+ namespace LightResults.Extensions.GeneratedIdentifier;
+
+ [AttributeUsage(AttributeTargets.Struct)]
+ public sealed class GeneratedIdentifierAttribute : Attribute;
+ """;
+ }
+
+ private static (ImmutableArray Diagnostics, GeneratorDriverRunResult Result) RunGenerator(IEnumerable sources)
+ {
+ return IncrementalGenerator.RunWithDiagnostics(sources, metadataReferences: ReferenceAssemblies.Net80);
+ }
+}
diff --git a/tests/LightResults.Extensions.GeneratedIdentifier.Tests/LightResults.Extensions.GeneratedIdentifier.Tests.csproj b/tests/LightResults.Extensions.GeneratedIdentifier.Tests/LightResults.Extensions.GeneratedIdentifier.Tests.csproj
new file mode 100644
index 0000000..402a32f
--- /dev/null
+++ b/tests/LightResults.Extensions.GeneratedIdentifier.Tests/LightResults.Extensions.GeneratedIdentifier.Tests.csproj
@@ -0,0 +1,66 @@
+
+
+
+ net8.0
+ enable
+ enable
+ latest
+ LightResults.Extensions.GeneratedIdentifier.Tests
+ LightResults.Extensions.GeneratedIdentifier.Tests
+ latest-Default
+ false
+ true
+
+
+
+
+
+
+
+
+
+
+
+ all
+ runtime; build; native; contentfiles; analyzers; buildtransitive
+
+
+
+
+
+
+
+
+ runtime; build; native; contentfiles; analyzers; buildtransitive
+ all
+
+
+
+
+
+ IdentifierSourceGeneratorTests.cs
+
+
+ IdentifierSourceGeneratorTests.cs
+
+
+ IdentifierSourceGeneratorTests.cs
+
+
+ IdentifierSourceGeneratorTests.cs
+
+
+ IdentifierSourceGeneratorTests.cs
+
+
+ IdentifierSourceGeneratorTests.cs
+
+
+ GeneratedIdentifierSourceGeneratorTests.cs
+
+
+ GeneratedIdentifierSourceGeneratorTests.cs
+
+
+
+
diff --git a/tests/LightResults.Extensions.GeneratedIdentifier.Tests/ModuleInitializer.cs b/tests/LightResults.Extensions.GeneratedIdentifier.Tests/ModuleInitializer.cs
new file mode 100644
index 0000000..6addfa9
--- /dev/null
+++ b/tests/LightResults.Extensions.GeneratedIdentifier.Tests/ModuleInitializer.cs
@@ -0,0 +1,16 @@
+using DiffEngine;
+using VerifyTests.DiffPlex;
+
+namespace LightResults.Extensions.GeneratedIdentifier.Tests;
+
+public static class ModuleInitializer
+{
+ public static void Initialize()
+ {
+ DiffRunner.Disabled = true;
+ VerifyDiffPlex.Initialize(OutputType.Compact);
+ VerifierSettings.InitializePlugins();
+ VerifierSettings.ScrubLinesContaining("DiffEngineTray");
+ VerifierSettings.IgnoreStackTrace();
+ }
+}
diff --git a/tests/LightResults.Extensions.GeneratedIdentifier.Tests/TestGuidIdTest.cs b/tests/LightResults.Extensions.GeneratedIdentifier.Tests/TestGuidIdTest.cs
new file mode 100644
index 0000000..a4a1c0b
--- /dev/null
+++ b/tests/LightResults.Extensions.GeneratedIdentifier.Tests/TestGuidIdTest.cs
@@ -0,0 +1,329 @@
+using FluentAssertions;
+using GeneratedIdentifier.Common.ValueObjects;
+using LightResults.Extensions.GeneratedIdentifier.Fixtures.Identifiers;
+
+// ReSharper disable SuspiciousTypeConversion.Global
+// ReSharper disable EqualExpressionComparison
+#pragma warning disable CS1718 // Comparison made to same variable
+
+namespace LightResults.Extensions.GeneratedIdentifier.Tests;
+
+public sealed class TestGuidIdTest
+{
+ private static readonly Guid Guid1 = Guid.Parse("bcbac1e2-de00-47e4-9891-58774a68668f");
+ private static readonly Guid Guid2 = Guid.Parse("e8d81b8f-127f-4e7d-b7fb-c9ab6f34be72");
+
+ [Fact]
+ public void Create_ValidValue_ShouldSucceed()
+ {
+ // Arrange
+ var validValue = Guid1;
+
+ // Act
+ var id = TestGuidId.Create(validValue);
+
+ // Assert
+ id.Should().NotBeNull();
+ id.ToGuid().Should().Be(validValue);
+ }
+
+ [Fact]
+ public void TryCreate_ValidValue_ShouldSucceed()
+ {
+ // Arrange
+ var validValue = Guid1;
+
+ // Act
+ var result = TestGuidId.TryCreate(validValue);
+
+ // Assert
+ result.IsSuccess(out var id).Should().BeTrue();
+ id.Should().NotBeNull();
+ id.ToGuid().Should().Be(validValue);
+ }
+
+ [Fact]
+ public void Parse_ValidString_ShouldSucceed()
+ {
+ // Arrange
+ const string validString = "5528cc73-cba9-4960-85c1-ed96dc4c7f95";
+
+ // Act
+ var result = TestGuidId.Parse(validString);
+
+ // Assert
+
+ result.ToGuid().Should().Be(Guid.Parse(validString));
+ }
+
+ [Theory]
+ [InlineData("invalid")]
+ [InlineData("g528cc73-cba9-4960-85c1-ed96dc4c7f95")]
+ public void Parse_InvalidString_ShouldThrowException(string invalidString)
+ {
+ // Act
+ var parse = () => TestGuidId.Parse(invalidString);
+
+ // Assert
+ parse.Should().Throw();
+ }
+
+ [Fact]
+ public void TryParse_ValidString_ShouldSucceed()
+ {
+ // Arrange
+ const string validString = "5528cc73-cba9-4960-85c1-ed96dc4c7f95";
+
+ // Act
+ var result = TestGuidId.TryParse(validString);
+
+ // Assert
+ result.IsSuccess(out var id).Should().BeTrue();
+ id.Should().NotBeNull();
+ id.ToGuid().Should().Be(Guid.Parse(validString));
+ }
+
+ [Theory]
+ [InlineData("invalid")]
+ [InlineData("-1")]
+ public void TryParse_InvalidString_ShouldFail(string invalidString)
+ {
+ // Act
+ var result = TestGuidId.TryParse(invalidString);
+
+ // Assert
+ result.IsFailed().Should().BeTrue();
+ result.Errors.Should().ContainSingle();
+ }
+
+ [Fact]
+ public void Equals_SameValues_ShouldBeEqual()
+ {
+ // Arrange
+ var id1 = TestGuidId.Create(Guid1);
+ var id2 = TestGuidId.Create(Guid1);
+
+ // Assert
+ id1.Should().Be(id2);
+ (id1 == id2).Should().BeTrue();
+ (id1 != id2).Should().BeFalse();
+ }
+
+ [Fact]
+ public void Equals_DifferentValues_ShouldNotBeEqual()
+ {
+ // Arrange
+ var id1 = TestGuidId.Create(Guid1);
+ var id2 = TestGuidId.Create(Guid2);
+
+ // Assert
+ id1.Should().NotBe(id2);
+ (id1 == id2).Should().BeFalse();
+ (id1 != id2).Should().BeTrue();
+ }
+
+ [Fact]
+ public void Equals_ObjectIsNull_ShouldReturnFalse()
+ {
+ // Arrange
+ var id = TestGuidId.Create(Guid1);
+
+ // Act
+ var result = id.Equals(null);
+
+ // Assert
+ result.Should().BeFalse();
+ }
+
+ [Fact]
+ public void Equals_ObjectIsNotTestGuidId_ShouldReturnFalse()
+ {
+ // Arrange
+ var id = TestGuidId.Create(Guid1);
+
+ // Act
+ var result = id.Equals("not an TestGuidId");
+
+ // Assert
+ result.Should().BeFalse();
+ }
+
+ [Fact]
+ public void GetHashCode_ShouldReturnCorrectValue()
+ {
+ // Arrange
+ var underlyingValue = Guid1;
+ var id = TestGuidId.Create(underlyingValue);
+
+ // Act
+ var hashCode1 = id.GetHashCode();
+ var hashCode2 = underlyingValue.GetHashCode();
+
+ // Assert
+ hashCode1.Should().Be(hashCode2);
+ }
+
+ [Fact]
+ public void Operators_Equality_ShouldReturnTrue()
+ {
+ // Arrange
+ var id1 = TestGuidId.Create(Guid1);
+ var id2 = TestGuidId.Create(Guid1);
+
+ // Assert
+ (id1 == id2).Should().BeTrue();
+ (id1 != id2).Should().BeFalse();
+ }
+
+ [Fact]
+ public void Operators_Inequality_ShouldReturnTrue()
+ {
+ // Arrange
+ var id1 = TestGuidId.Create(Guid1);
+ var id2 = TestGuidId.Create(Guid2);
+
+ // Assert
+ (id1 != id2).Should().BeTrue();
+ (id1 == id2).Should().BeFalse();
+ }
+
+ [Fact]
+ public void CompareTo_SameValue_ShouldReturnZero()
+ {
+ // Arrange
+ var id1 = TestGuidId.Create(Guid1);
+ var id2 = TestGuidId.Create(Guid1);
+
+ // Act
+ var result = id1.CompareTo(id2);
+
+ // Assert
+ result.Should().Be(0);
+ }
+
+ [Fact]
+ public void CompareTo_LesserValue_ShouldReturnNegative()
+ {
+ // Arrange
+ var id1 = TestGuidId.Create(Guid1);
+ var id2 = TestGuidId.Create(Guid2);
+
+ // Act
+ var result = id1.CompareTo(id2);
+
+ // Assert
+ result.Should().BeNegative();
+ }
+
+ [Fact]
+ public void CompareTo_GreaterValue_ShouldReturnPositive()
+ {
+ // Arrange
+ var id1 = TestGuidId.Create(Guid2);
+ var id2 = TestGuidId.Create(Guid1);
+
+ // Act
+ var result = id1.CompareTo(id2);
+
+ // Assert
+ result.Should().BePositive();
+ }
+
+ [Fact]
+ public void Operators_LessThan_ShouldReturnTrue()
+ {
+ // Arrange
+ var id1 = TestGuidId.Create(Guid1);
+ var id2 = TestGuidId.Create(Guid2);
+
+ // Assert
+ (id1 < id2).Should().BeTrue();
+ }
+
+ [Fact]
+ public void Operators_GreaterThan_ShouldReturnTrue()
+ {
+ // Arrange
+ var id1 = TestGuidId.Create(Guid2);
+ var id2 = TestGuidId.Create(Guid1);
+
+ // Assert
+ (id1 > id2).Should().BeTrue();
+ }
+
+ [Fact]
+ public void Operators_LessThanOrEqual_ShouldReturnTrue()
+ {
+ // Arrange
+ var id1 = TestGuidId.Create(Guid1);
+ var id2 = TestGuidId.Create(Guid2);
+
+ // Assert
+ (id1 <= id2).Should().BeTrue();
+ (id1 <= id1).Should().BeTrue();
+ }
+
+ [Fact]
+ public void Operators_GreaterThanOrEqual_ShouldReturnTrue()
+ {
+ // Arrange
+ var id1 = TestGuidId.Create(Guid2);
+ var id2 = TestGuidId.Create(Guid1);
+
+ // Assert
+ (id1 >= id2).Should().BeTrue();
+ (id1 >= id1).Should().BeTrue();
+ }
+
+ [Fact]
+ public void CompareTo_ObjectIsNull_ShouldReturnPositive()
+ {
+ // Arrange
+ var id = TestGuidId.Create(Guid1);
+
+ // Act
+ var result = id.CompareTo(null);
+
+ // Assert
+ result.Should().BePositive();
+ }
+
+ [Fact]
+ public void CompareTo_ObjectIsNotTestGuidId_ShouldThrowException()
+ {
+ // Arrange
+ var id = TestGuidId.Create(Guid1);
+
+ // Act
+ var compareTo = () => id.CompareTo("not an TestGuidId");
+
+ // Assert
+ compareTo.Should().Throw();
+ }
+
+ [Fact]
+ public void ToInt_ShouldReturnCorrectValue()
+ {
+ // Arrange
+ var id = TestGuidId.Create(Guid1);
+
+ // Act
+ var integerValue = id.ToGuid();
+
+ // Assert
+ integerValue.Should().Be(Guid1);
+ }
+
+ [Fact]
+ public void ToString_ShouldReturnStringValue()
+ {
+ // Arrange
+ var id = TestGuidId.Create(Guid1);
+
+ // Act
+ var stringValue = id.ToString();
+
+ // Assert
+ stringValue.Should().Be(Guid1.ToString());
+ }
+}
diff --git a/tests/LightResults.Extensions.GeneratedIdentifier.Tests/TestIntIdTest.cs b/tests/LightResults.Extensions.GeneratedIdentifier.Tests/TestIntIdTest.cs
new file mode 100644
index 0000000..3f0a47e
--- /dev/null
+++ b/tests/LightResults.Extensions.GeneratedIdentifier.Tests/TestIntIdTest.cs
@@ -0,0 +1,353 @@
+using FluentAssertions;
+using GeneratedIdentifier.Common.ValueObjects;
+using LightResults.Extensions.GeneratedIdentifier.Fixtures.Identifiers;
+
+// ReSharper disable SuspiciousTypeConversion.Global
+// ReSharper disable EqualExpressionComparison
+#pragma warning disable CS1718 // Comparison made to same variable
+
+namespace LightResults.Extensions.GeneratedIdentifier.Tests;
+
+public sealed class TestIntIdTest
+{
+ [Fact]
+ public void Create_ValidValue_ShouldSucceed()
+ {
+ // Arrange
+ const int validValue = 42;
+
+ // Act
+ var id = TestIntId.Create(validValue);
+
+ // Assert
+ id.Should().NotBeNull();
+ id.ToInt32().Should().Be(validValue);
+ }
+
+ [Fact]
+ public void Create_InvalidValue_ShouldThrowException()
+ {
+ // Arrange
+ const int invalidValue = -1;
+
+ // Act
+ var create = () => TestIntId.Create(invalidValue);
+
+ // Assert
+ create.Should().Throw();
+ }
+
+ [Fact]
+ public void TryCreate_ValidValue_ShouldSucceed()
+ {
+ // Arrange
+ const int validValue = 42;
+
+ // Act
+ var result = TestIntId.TryCreate(validValue);
+
+ // Assert
+ result.IsSuccess(out var id).Should().BeTrue();
+ id.Should().NotBeNull();
+ id.ToInt32().Should().Be(validValue);
+ }
+
+ [Fact]
+ public void TryCreate_InvalidValue_ShouldFail()
+ {
+ // Arrange
+ const int invalidValue = -1;
+
+ // Act
+ var result = TestIntId.TryCreate(invalidValue);
+
+ // Assert
+ result.IsFailed().Should().BeTrue();
+ result.Errors.Should().ContainSingle();
+ }
+
+ [Fact]
+ public void Parse_ValidString_ShouldSucceed()
+ {
+ // Arrange
+ const string validString = "42";
+
+ // Act
+ var result = TestIntId.Parse(validString);
+
+ // Assert
+
+ result.ToInt32().Should().Be(int.Parse(validString));
+ }
+
+ [Theory]
+ [InlineData("invalid")]
+ [InlineData("-1")]
+ public void Parse_InvalidString_ShouldThrowException(string invalidString)
+ {
+ // Act
+ var parse = () => TestIntId.Parse(invalidString);
+
+ // Assert
+ parse.Should().Throw();
+ }
+
+ [Fact]
+ public void TryParse_ValidString_ShouldSucceed()
+ {
+ // Arrange
+ const string validString = "42";
+
+ // Act
+ var result = TestIntId.TryParse(validString);
+
+ // Assert
+ result.IsSuccess(out var id).Should().BeTrue();
+ id.Should().NotBeNull();
+ id.ToInt32().Should().Be(int.Parse(validString));
+ }
+
+ [Theory]
+ [InlineData("invalid")]
+ [InlineData("-1")]
+ public void TryParse_InvalidString_ShouldFail(string invalidString)
+ {
+ // Act
+ var result = TestIntId.TryParse(invalidString);
+
+ // Assert
+ result.IsFailed().Should().BeTrue();
+ result.Errors.Should().ContainSingle();
+ }
+
+ [Fact]
+ public void Equals_SameValues_ShouldBeEqual()
+ {
+ // Arrange
+ var id1 = TestIntId.Create(42);
+ var id2 = TestIntId.Create(42);
+
+ // Assert
+ id1.Should().Be(id2);
+ (id1 == id2).Should().BeTrue();
+ (id1 != id2).Should().BeFalse();
+ }
+
+ [Fact]
+ public void Equals_DifferentValues_ShouldNotBeEqual()
+ {
+ // Arrange
+ var id1 = TestIntId.Create(42);
+ var id2 = TestIntId.Create(99);
+
+ // Assert
+ id1.Should().NotBe(id2);
+ (id1 == id2).Should().BeFalse();
+ (id1 != id2).Should().BeTrue();
+ }
+
+ [Fact]
+ public void Equals_ObjectIsNull_ShouldReturnFalse()
+ {
+ // Arrange
+ var id = TestIntId.Create(42);
+
+ // Act
+ var result = id.Equals(null);
+
+ // Assert
+ result.Should().BeFalse();
+ }
+
+ [Fact]
+ public void Equals_ObjectIsNotTestIntId_ShouldReturnFalse()
+ {
+ // Arrange
+ var id = TestIntId.Create(42);
+
+ // Act
+ var result = id.Equals("not an TestIntId");
+
+ // Assert
+ result.Should().BeFalse();
+ }
+
+ [Fact]
+ public void GetHashCode_ShouldReturnCorrectValue()
+ {
+ // Arrange
+ const int underlyingValue = 42;
+ var id = TestIntId.Create(underlyingValue);
+
+ // Act
+ var hashCode1 = id.GetHashCode();
+ var hashCode2 = underlyingValue.GetHashCode();
+
+ // Assert
+ hashCode1.Should().Be(hashCode2);
+ }
+
+ [Fact]
+ public void Operators_Equality_ShouldReturnTrue()
+ {
+ // Arrange
+ var id1 = TestIntId.Create(42);
+ var id2 = TestIntId.Create(42);
+
+ // Assert
+ (id1 == id2).Should().BeTrue();
+ (id1 != id2).Should().BeFalse();
+ }
+
+ [Fact]
+ public void Operators_Inequality_ShouldReturnTrue()
+ {
+ // Arrange
+ var id1 = TestIntId.Create(42);
+ var id2 = TestIntId.Create(99);
+
+ // Assert
+ (id1 != id2).Should().BeTrue();
+ (id1 == id2).Should().BeFalse();
+ }
+
+ [Fact]
+ public void CompareTo_SameValue_ShouldReturnZero()
+ {
+ // Arrange
+ var id1 = TestIntId.Create(42);
+ var id2 = TestIntId.Create(42);
+
+ // Act
+ var result = id1.CompareTo(id2);
+
+ // Assert
+ result.Should().Be(0);
+ }
+
+ [Fact]
+ public void CompareTo_LesserValue_ShouldReturnNegative()
+ {
+ // Arrange
+ var id1 = TestIntId.Create(42);
+ var id2 = TestIntId.Create(99);
+
+ // Act
+ var result = id1.CompareTo(id2);
+
+ // Assert
+ result.Should().BeNegative();
+ }
+
+ [Fact]
+ public void CompareTo_GreaterValue_ShouldReturnPositive()
+ {
+ // Arrange
+ var id1 = TestIntId.Create(99);
+ var id2 = TestIntId.Create(42);
+
+ // Act
+ var result = id1.CompareTo(id2);
+
+ // Assert
+ result.Should().BePositive();
+ }
+
+ [Fact]
+ public void Operators_LessThan_ShouldReturnTrue()
+ {
+ // Arrange
+ var id1 = TestIntId.Create(42);
+ var id2 = TestIntId.Create(99);
+
+ // Assert
+ (id1 < id2).Should().BeTrue();
+ }
+
+ [Fact]
+ public void Operators_GreaterThan_ShouldReturnTrue()
+ {
+ // Arrange
+ var id1 = TestIntId.Create(99);
+ var id2 = TestIntId.Create(42);
+
+ // Assert
+ (id1 > id2).Should().BeTrue();
+ }
+
+ [Fact]
+ public void Operators_LessThanOrEqual_ShouldReturnTrue()
+ {
+ // Arrange
+ var id1 = TestIntId.Create(42);
+ var id2 = TestIntId.Create(99);
+
+ // Assert
+ (id1 <= id2).Should().BeTrue();
+ (id1 <= id1).Should().BeTrue();
+ }
+
+ [Fact]
+ public void Operators_GreaterThanOrEqual_ShouldReturnTrue()
+ {
+ // Arrange
+ var id1 = TestIntId.Create(99);
+ var id2 = TestIntId.Create(42);
+
+ // Assert
+ (id1 >= id2).Should().BeTrue();
+ (id1 >= id1).Should().BeTrue();
+ }
+
+ [Fact]
+ public void CompareTo_ObjectIsNull_ShouldReturnPositive()
+ {
+ // Arrange
+ var id = TestIntId.Create(42);
+
+ // Act
+ var result = id.CompareTo(null);
+
+ // Assert
+ result.Should().BePositive();
+ }
+
+ [Fact]
+ public void CompareTo_ObjectIsNotTestIntId_ShouldThrowException()
+ {
+ // Arrange
+ var id = TestIntId.Create(42);
+
+ // Act
+ var compareTo = () => id.CompareTo("not an TestIntId");
+
+ // Assert
+ compareTo.Should().Throw();
+ }
+
+ [Fact]
+ public void ToInt_ShouldReturnCorrectValue()
+ {
+ // Arrange
+ var id = TestIntId.Create(42);
+
+ // Act
+ var integerValue = id.ToInt32();
+
+ // Assert
+ integerValue.Should().Be(42);
+ }
+
+ [Fact]
+ public void ToString_ShouldReturnStringValue()
+ {
+ // Arrange
+ var id = TestIntId.Create(42);
+
+ // Act
+ var stringValue = id.ToString();
+
+ // Assert
+ stringValue.Should().Be("42");
+ }
+}
diff --git a/tests/LightResults.Extensions.GeneratedIdentifier.Tests/TestShortIdTest.cs b/tests/LightResults.Extensions.GeneratedIdentifier.Tests/TestShortIdTest.cs
new file mode 100644
index 0000000..ecb96ae
--- /dev/null
+++ b/tests/LightResults.Extensions.GeneratedIdentifier.Tests/TestShortIdTest.cs
@@ -0,0 +1,353 @@
+using FluentAssertions;
+using GeneratedIdentifier.Common.ValueObjects;
+using LightResults.Extensions.GeneratedIdentifier.Fixtures.Identifiers;
+
+// ReSharper disable SuspiciousTypeConversion.Global
+// ReSharper disable EqualExpressionComparison
+#pragma warning disable CS1718 // Comparison made to same variable
+
+namespace LightResults.Extensions.GeneratedIdentifier.Tests;
+
+public sealed class TestShortIdTest
+{
+ [Fact]
+ public void Create_ValidValue_ShouldSucceed()
+ {
+ // Arrange
+ const int validValue = 42;
+
+ // Act
+ var id = TestShortId.Create(validValue);
+
+ // Assert
+ id.Should().NotBeNull();
+ id.ToInt16().Should().Be(validValue);
+ }
+
+ [Fact]
+ public void Create_InvalidValue_ShouldThrowException()
+ {
+ // Arrange
+ const int invalidValue = -1;
+
+ // Act
+ var create = () => TestShortId.Create(invalidValue);
+
+ // Assert
+ create.Should().Throw();
+ }
+
+ [Fact]
+ public void TryCreate_ValidValue_ShouldSucceed()
+ {
+ // Arrange
+ const int validValue = 42;
+
+ // Act
+ var result = TestShortId.TryCreate(validValue);
+
+ // Assert
+ result.IsSuccess(out var id).Should().BeTrue();
+ id.Should().NotBeNull();
+ id.ToInt16().Should().Be(validValue);
+ }
+
+ [Fact]
+ public void TryCreate_InvalidValue_ShouldFail()
+ {
+ // Arrange
+ const int invalidValue = -1;
+
+ // Act
+ var result = TestShortId.TryCreate(invalidValue);
+
+ // Assert
+ result.IsFailed().Should().BeTrue();
+ result.Errors.Should().ContainSingle();
+ }
+
+ [Fact]
+ public void Parse_ValidString_ShouldSucceed()
+ {
+ // Arrange
+ const string validString = "42";
+
+ // Act
+ var result = TestShortId.Parse(validString);
+
+ // Assert
+
+ result.ToInt16().Should().Be(short.Parse(validString));
+ }
+
+ [Theory]
+ [InlineData("invalid")]
+ [InlineData("-1")]
+ public void Parse_InvalidString_ShouldThrowException(string invalidString)
+ {
+ // Act
+ var parse = () => TestShortId.Parse(invalidString);
+
+ // Assert
+ parse.Should().Throw();
+ }
+
+ [Fact]
+ public void TryParse_ValidString_ShouldSucceed()
+ {
+ // Arrange
+ const string validString = "42";
+
+ // Act
+ var result = TestShortId.TryParse(validString);
+
+ // Assert
+ result.IsSuccess(out var id).Should().BeTrue();
+ id.Should().NotBeNull();
+ id.ToInt16().Should().Be(short.Parse(validString));
+ }
+
+ [Theory]
+ [InlineData("invalid")]
+ [InlineData("-1")]
+ public void TryParse_InvalidString_ShouldFail(string invalidString)
+ {
+ // Act
+ var result = TestShortId.TryParse(invalidString);
+
+ // Assert
+ result.IsFailed().Should().BeTrue();
+ result.Errors.Should().ContainSingle();
+ }
+
+ [Fact]
+ public void Equals_SameValues_ShouldBeEqual()
+ {
+ // Arrange
+ var id1 = TestShortId.Create(42);
+ var id2 = TestShortId.Create(42);
+
+ // Assert
+ id1.Should().Be(id2);
+ (id1 == id2).Should().BeTrue();
+ (id1 != id2).Should().BeFalse();
+ }
+
+ [Fact]
+ public void Equals_DifferentValues_ShouldNotBeEqual()
+ {
+ // Arrange
+ var id1 = TestShortId.Create(42);
+ var id2 = TestShortId.Create(99);
+
+ // Assert
+ id1.Should().NotBe(id2);
+ (id1 == id2).Should().BeFalse();
+ (id1 != id2).Should().BeTrue();
+ }
+
+ [Fact]
+ public void Equals_ObjectIsNull_ShouldReturnFalse()
+ {
+ // Arrange
+ var id = TestShortId.Create(42);
+
+ // Act
+ var result = id.Equals(null);
+
+ // Assert
+ result.Should().BeFalse();
+ }
+
+ [Fact]
+ public void Equals_ObjectIsNotTestShortId_ShouldReturnFalse()
+ {
+ // Arrange
+ var id = TestShortId.Create(42);
+
+ // Act
+ var result = id.Equals("not an TestShortId");
+
+ // Assert
+ result.Should().BeFalse();
+ }
+
+ [Fact]
+ public void GetHashCode_ShouldReturnCorrectValue()
+ {
+ // Arrange
+ const int underlyingValue = 42;
+ var id = TestShortId.Create(underlyingValue);
+
+ // Act
+ var hashCode1 = id.GetHashCode();
+ var hashCode2 = underlyingValue.GetHashCode();
+
+ // Assert
+ hashCode1.Should().Be(hashCode2);
+ }
+
+ [Fact]
+ public void Operators_Equality_ShouldReturnTrue()
+ {
+ // Arrange
+ var id1 = TestShortId.Create(42);
+ var id2 = TestShortId.Create(42);
+
+ // Assert
+ (id1 == id2).Should().BeTrue();
+ (id1 != id2).Should().BeFalse();
+ }
+
+ [Fact]
+ public void Operators_Inequality_ShouldReturnTrue()
+ {
+ // Arrange
+ var id1 = TestShortId.Create(42);
+ var id2 = TestShortId.Create(99);
+
+ // Assert
+ (id1 != id2).Should().BeTrue();
+ (id1 == id2).Should().BeFalse();
+ }
+
+ [Fact]
+ public void CompareTo_SameValue_ShouldReturnZero()
+ {
+ // Arrange
+ var id1 = TestShortId.Create(42);
+ var id2 = TestShortId.Create(42);
+
+ // Act
+ var result = id1.CompareTo(id2);
+
+ // Assert
+ result.Should().Be(0);
+ }
+
+ [Fact]
+ public void CompareTo_LesserValue_ShouldReturnNegative()
+ {
+ // Arrange
+ var id1 = TestShortId.Create(42);
+ var id2 = TestShortId.Create(99);
+
+ // Act
+ var result = id1.CompareTo(id2);
+
+ // Assert
+ result.Should().BeNegative();
+ }
+
+ [Fact]
+ public void CompareTo_GreaterValue_ShouldReturnPositive()
+ {
+ // Arrange
+ var id1 = TestShortId.Create(99);
+ var id2 = TestShortId.Create(42);
+
+ // Act
+ var result = id1.CompareTo(id2);
+
+ // Assert
+ result.Should().BePositive();
+ }
+
+ [Fact]
+ public void Operators_LessThan_ShouldReturnTrue()
+ {
+ // Arrange
+ var id1 = TestShortId.Create(42);
+ var id2 = TestShortId.Create(99);
+
+ // Assert
+ (id1 < id2).Should().BeTrue();
+ }
+
+ [Fact]
+ public void Operators_GreaterThan_ShouldReturnTrue()
+ {
+ // Arrange
+ var id1 = TestShortId.Create(99);
+ var id2 = TestShortId.Create(42);
+
+ // Assert
+ (id1 > id2).Should().BeTrue();
+ }
+
+ [Fact]
+ public void Operators_LessThanOrEqual_ShouldReturnTrue()
+ {
+ // Arrange
+ var id1 = TestShortId.Create(42);
+ var id2 = TestShortId.Create(99);
+
+ // Assert
+ (id1 <= id2).Should().BeTrue();
+ (id1 <= id1).Should().BeTrue();
+ }
+
+ [Fact]
+ public void Operators_GreaterThanOrEqual_ShouldReturnTrue()
+ {
+ // Arrange
+ var id1 = TestShortId.Create(99);
+ var id2 = TestShortId.Create(42);
+
+ // Assert
+ (id1 >= id2).Should().BeTrue();
+ (id1 >= id1).Should().BeTrue();
+ }
+
+ [Fact]
+ public void CompareTo_ObjectIsNull_ShouldReturnPositive()
+ {
+ // Arrange
+ var id = TestShortId.Create(42);
+
+ // Act
+ var result = id.CompareTo(null);
+
+ // Assert
+ result.Should().BePositive();
+ }
+
+ [Fact]
+ public void CompareTo_ObjectIsNotTestShortId_ShouldThrowException()
+ {
+ // Arrange
+ var id = TestShortId.Create(42);
+
+ // Act
+ var compareTo = () => id.CompareTo("not an TestShortId");
+
+ // Assert
+ compareTo.Should().Throw();
+ }
+
+ [Fact]
+ public void ToInt_ShouldReturnCorrectValue()
+ {
+ // Arrange
+ var id = TestShortId.Create(42);
+
+ // Act
+ var integerValue = id.ToInt16();
+
+ // Assert
+ integerValue.Should().Be(42);
+ }
+
+ [Fact]
+ public void ToString_ShouldReturnStringValue()
+ {
+ // Arrange
+ var id = TestShortId.Create(42);
+
+ // Act
+ var stringValue = id.ToString();
+
+ // Assert
+ stringValue.Should().Be("42");
+ }
+}
diff --git a/tools/Benchmarks/Benchmarks.csproj b/tools/Benchmarks/Benchmarks.csproj
index 5a511ea..04ed832 100644
--- a/tools/Benchmarks/Benchmarks.csproj
+++ b/tools/Benchmarks/Benchmarks.csproj
@@ -11,8 +11,8 @@
-
-
+
+