Tuxedo.SourceGenerator
0.0.0-beta.1
See the version list below for details.
dotnet add package Tuxedo.SourceGenerator --version 0.0.0-beta.1
NuGet\Install-Package Tuxedo.SourceGenerator -Version 0.0.0-beta.1
<PackageReference Include="Tuxedo.SourceGenerator" Version="0.0.0-beta.1" />
paket add Tuxedo.SourceGenerator --version 0.0.0-beta.1
#r "nuget: Tuxedo.SourceGenerator, 0.0.0-beta.1"
// Install Tuxedo.SourceGenerator as a Cake Addin #addin nuget:?package=Tuxedo.SourceGenerator&version=0.0.0-beta.1&prerelease // Install Tuxedo.SourceGenerator as a Cake Tool #tool nuget:?package=Tuxedo.SourceGenerator&version=0.0.0-beta.1&prerelease
Tuxedo
More precise types can help to reduce the number of bugs in your code. Tuxedo provides a source generator that can turn simple conditions (predicates) into refined types. There is no other runtime dependency, so it can be used in libraries without issue.
This provides advantages over the 'Guard' based approaches because once the check on the type has been done, it carries with it proof of refinement, eliminating the need to ever check it again.
This allows for opt in fail fast behaviour, while eliminating shotgun parsing where the same checks are repeated throughout the program "Just in case".
For example, take a simple method that divides 2 floats,
public static float MyCustomDivide(float dividend, float divisor)
{
// some check that throws early if the divisor is zero
// this could also be some fancy "Guard" library it has the same effect
ArgumentOutOfRangeException.ThrowIfZero(divisor);
// now we are safe and can do the work
return dividend / divisor;
}
This is fine, as long as the method does not pass its parameters on to a new method. Those nested methods would then need to do the same checks again to ensure that the float is indeed not zero.
Instead, lets use Tuxedo to refine the float type so that it will always be greater than zero.
// this is the predicate that is used to refine the float
// it will produce a new type called a PositiveFloat
// only floats that meet the predicate can ever be one
[Refinement("The float must be positive, but was '{value}'")]
public static bool Positive(float value) => value > 0;
public static float MyCustomDivide(float dividend, PositiveFloat divisor)
{
return dividend / divisor; // will never fail at this point
}
A PositiveFloat
is a refined version of a float, it must be positive.
That is what a refined type is, a type paired with a predicate that it passes. Tuxedo makes creating these types simple.
Let's create the canonical example of a non-empty list.
/// <summary>
/// This is a non-empty list
/// </summary>
/// <typeparam name="T">some T</typeparam>
public readonly partial struct NonEmptyList<T>
{
/// <summary>
/// The head of the list
/// </summary>
public T Head => Value[0];
/// <summary>
/// Refinement that ensures the list is non-empty
/// </summary>
/// <param name="list">list</param>
/// <typeparam name="T">some T, required again here for the source generator</typeparam>
/// <returns>true if the list is non-empty</returns>
[Refinement("The list must not be empty.")]
private static bool NonEmpty<T>(List<T> list) => list.Count != 0;
}
This new type will convert to its underlying List of T type and has the
convenience Head
method added to it.
Using it looks like this,
// some list we want to refine
List<string> list = new List<string>{ "Some", "Great", "Values" };
// now we can try and convert that list
if(NonEmptyList<string>.TryParse(
list,
out NonEmptyList<string> nel,
out string failureMessage))
{
// we have a safe non-empty list in here
// which we can get the head of
var head = nel.Head;
// or assign to a list variable
List<string> lst = nel;
// or enumerate via the standard Value property
foreach(var e in nel.Value)
{
...
}
}
// or if we want we can just fail fast and do an explicit conversion which
// thows if the refinement fails
var nel = (NonEmptyList<string>) list;
All refined types are structs and will not allocate, making them cost-effective.
They can also be used to parse, not just validate.
A very common use case is running a predicate which also results in some alternative type being created via the process.
Take a string that should also be a valid Guid. It would be good to produce a valid guid as a result of the refinement process.
This can be done with Tuxedo,
/// <summary>
/// Represents a string that is also a valid Guid
/// </summary>
public readonly partial struct GuidString
{
// custom fields and methods can be added to the refined type
public byte[] Bytes => RefinedValue.ToByteArray();
public bool IsEmpty => RefinedValue == System.Guid.Empty;
[Refinement("The value must be a valid GUID, but was '{value}'")]
private static bool Guid(string value, out Guid guid) =>
// the standard TryParse on guid returns true if it's a valid guid
// and also produces a guid as a result, which gets captured as
// part of constructing the GuidString
System.Guid.TryParse(value, out guid);
}
A GuidString
is now a valid string and guid.
It will implicitly convert to both, and will use TryParse on guid internally to determine if the string is a valid guid, so in one pass also produces the guid.
// convert the string to a GuidString
var refined = (GuidString)"6192C5ED-505C-4558-B87C-CA6E7D612B31";
// now we have a valid Guid
refined.RawValue.Should().Be("6192C5ED-505C-4558-B87C-CA6E7D612B31"); // string
refined.RefinedValue.Should().Be(new Guid(
"6192C5ED-505C-4558-B87C-CA6E7D612B31")); // guid
// and can also use custom methods defined on GuidString
refined.Bytes.Should().NotBeEmpty();
refined.IsEmpty.Should().BeFalse();
If the alternative refined value is the same as the raw value, then only one implicit conversion method will be generated for the refined value. not the raw value.
Are these Refinement types?
Sort of, but at this stage there is no compile time checks performed.
I have future plans to perform refinement checks on all statically known values. This would take the form of an analyser, and would be enabled through the use of the refined types.
But for now think of it as moving the guard clauses to the type level and making it more explicit what the method requirements are.
When should I not use this?
Refinement types are not designed to integrate with System.Text.Json or any other serialization layer. That is not to say that they could not be made to, but at this stage it's a non-requirement.
Just keep the refined types inside the domain boundary of your application and use the simpler raw types on the edges.
For more details/information keep reading the docs or have a look at the test projects or create an issue.
Product | Versions Compatible and additional computed target framework versions. |
---|---|
.NET | net5.0 was computed. net5.0-windows was computed. net6.0 was computed. net6.0-android was computed. net6.0-ios was computed. net6.0-maccatalyst was computed. net6.0-macos was computed. net6.0-tvos was computed. net6.0-windows was computed. net7.0 was computed. net7.0-android was computed. net7.0-ios was computed. net7.0-maccatalyst was computed. net7.0-macos was computed. net7.0-tvos was computed. net7.0-windows was computed. net8.0 was computed. net8.0-android was computed. net8.0-browser was computed. net8.0-ios was computed. net8.0-maccatalyst was computed. net8.0-macos was computed. net8.0-tvos was computed. net8.0-windows was computed. net9.0 was computed. net9.0-android was computed. net9.0-browser was computed. net9.0-ios was computed. net9.0-maccatalyst was computed. net9.0-macos was computed. net9.0-tvos was computed. net9.0-windows was computed. |
.NET Core | netcoreapp2.0 was computed. netcoreapp2.1 was computed. netcoreapp2.2 was computed. netcoreapp3.0 was computed. netcoreapp3.1 was computed. |
.NET Standard | netstandard2.0 is compatible. netstandard2.1 was computed. |
.NET Framework | net461 was computed. net462 was computed. net463 was computed. net47 was computed. net471 was computed. net472 was computed. net48 was computed. net481 was computed. |
MonoAndroid | monoandroid was computed. |
MonoMac | monomac was computed. |
MonoTouch | monotouch was computed. |
Tizen | tizen40 was computed. tizen60 was computed. |
Xamarin.iOS | xamarinios was computed. |
Xamarin.Mac | xamarinmac was computed. |
Xamarin.TVOS | xamarintvos was computed. |
Xamarin.WatchOS | xamarinwatchos was computed. |
-
.NETStandard 2.0
- Microsoft.CodeAnalysis.CSharp (>= 4.11.0)
- Microsoft.CodeAnalysis.CSharp.Workspaces (>= 4.11.0)
NuGet packages
This package is not used by any NuGet packages.
GitHub repositories
This package is not used by any popular GitHub repositories.
Version | Downloads | Last updated |
---|---|---|
0.0.0-beta.10 | 16 | 1/8/2025 |
0.0.0-beta.9 | 39 | 1/6/2025 |
0.0.0-beta.8 | 42 | 1/5/2025 |
0.0.0-beta.7 | 56 | 1/5/2025 |
0.0.0-beta.6 | 52 | 1/4/2025 |
0.0.0-beta.5 | 56 | 1/1/2025 |
0.0.0-beta.4 | 49 | 1/1/2025 |
0.0.0-beta.3 | 38 | 12/31/2024 |
0.0.0-beta.2 | 44 | 12/27/2024 |
0.0.0-beta.1 | 44 | 12/27/2024 |