Thinktecture.Runtime.Extensions 9.7.0-beta01

This is a prerelease version of Thinktecture.Runtime.Extensions.
There is a newer prerelease version of this package available.
See the version list below for details.
dotnet add package Thinktecture.Runtime.Extensions --version 9.7.0-beta01
                    
NuGet\Install-Package Thinktecture.Runtime.Extensions -Version 9.7.0-beta01
                    
This command is intended to be used within the Package Manager Console in Visual Studio, as it uses the NuGet module's version of Install-Package.
<PackageReference Include="Thinktecture.Runtime.Extensions" Version="9.7.0-beta01" />
                    
For projects that support PackageReference, copy this XML node into the project file to reference the package.
<PackageVersion Include="Thinktecture.Runtime.Extensions" Version="9.7.0-beta01" />
                    
Directory.Packages.props
<PackageReference Include="Thinktecture.Runtime.Extensions" />
                    
Project file
For projects that support Central Package Management (CPM), copy this XML node into the solution Directory.Packages.props file to version the package.
paket add Thinktecture.Runtime.Extensions --version 9.7.0-beta01
                    
#r "nuget: Thinktecture.Runtime.Extensions, 9.7.0-beta01"
                    
#r directive can be used in F# Interactive and Polyglot Notebooks. Copy this into the interactive tool or source code of the script to reference the package.
#:package Thinktecture.Runtime.Extensions@9.7.0-beta01
                    
#:package directive can be used in C# file-based apps starting in .NET 10 preview 4. Copy this into a .cs file before any lines of code to reference the package.
#addin nuget:?package=Thinktecture.Runtime.Extensions&version=9.7.0-beta01&prerelease
                    
Install as a Cake Addin
#tool nuget:?package=Thinktecture.Runtime.Extensions&version=9.7.0-beta01&prerelease
                    
Install as a Cake Tool

Build TestResults NuGet Downloads

Thinktecture.Runtime.Extensions
Thinktecture.Runtime.Extensions.EntityFrameworkCore7
Thinktecture.Runtime.Extensions.EntityFrameworkCore8
Thinktecture.Runtime.Extensions.EntityFrameworkCore9
Thinktecture.Runtime.Extensions.Json
Thinktecture.Runtime.Extensions.Newtonsoft.Json
Thinktecture.Runtime.Extensions.MessagePack
Thinktecture.Runtime.Extensions.AspNetCore
Thinktecture.Runtime.Extensions.Swashbuckle

This library provides some interfaces, classes, Roslyn Source Generators, Roslyn Analyzers and Roslyn CodeFixes for implementation of Smart Enums, Value Objects and Discriminated Unions.

Documentation

See wiki for more documentation.

Value Objects articles:

Smart Enums articles:

Discriminated Unions articles:

Requirements

  • C# 11 (or higher) for generated code
  • SDK 8.0.400 (or higher) for building projects

Migrations

Ideas and real-world use cases

Smart Enums:

Value objects:

Discriminated Unions:

Smart Enums

Smart Enums provide a powerful alternative to traditional C# enums, offering type-safety, extensibility, and rich behavior. Unlike regular C# enums which are limited to numeric values and lack extensibility, Smart Enums can:

  • Use any type as the underlying type (e.g., strings, integers) or none at all
  • Include additional fields, properties and behavior
  • Use polymorphism to define custom behavior for each value
  • Prevent creation of invalid values
  • Integrate seamlessly with JSON serializers, MessagePack, Entity Framework Core, ASP.NET Core and Swashbuckle (OpenAPI)

Install: Install-Package Thinktecture.Runtime.Extensions

Documentation: Smart Enums

Some of the Key Features are:

  • Choice between always-valid and maybe-valid Smart Enum
  • Reflection-free iteration over all items
  • Fast lookup/conversion from underlying type to Smart Enum and vice versa
  • Allows custom properties and methods
  • Exhaustive pattern matching with Switch/Map methods
  • Provides appropriate constructor, based on the specified properties/fields
  • Proper implementation of Equals, GetHashCode, ToString and equality operators
  • Provides implementation of IComparable, IComparable<T>, IFormattable, IParsable<T> and comparison operators <, <=, >, >= (if applicable to the underlying type)
  • Custom comparer and equality comparer

Roslyn Analyzers and CodeFixes help the developers to implement the Smart Enums correctly

Provides support for:

  • JSON (System.Text.Json and Newtonsoft)
  • Minimal Api Parameter Binding and ASP.NET Core Model Binding
  • Entity Framework Core
  • MessagePack

Definition of a Smart Enum with custom properties and methods.

[SmartEnum<string>]
public partial class ShippingMethod
{
   public static readonly ShippingMethod Standard = new(
      "STANDARD",
      basePrice: 5.99m,
      weightMultiplier: 0.5m,
      estimatedDays: 5,
      requiresSignature: false);

   public static readonly ShippingMethod Express = new(
      "EXPRESS",
      basePrice: 15.99m,
      weightMultiplier: 0.75m,
      estimatedDays: 2,
      requiresSignature: true);

   public static readonly ShippingMethod NextDay = new(
      "NEXT_DAY",
      basePrice: 29.99m,
      weightMultiplier: 1.0m,
      estimatedDays: 1,
      requiresSignature: true);

   private readonly decimal _basePrice;
   private readonly decimal _weightMultiplier;
   private readonly int _estimatedDays;

   public bool RequiresSignature { get; }

   public decimal CalculatePrice(decimal orderWeight)
   {
      return _basePrice + (orderWeight * _weightMultiplier);
   }

   public DateTime GetEstimatedDeliveryDate()
   {
      return DateTime.Today.AddDays(_estimatedDays);
   }
}

Behind the scenes a Roslyn Source Generator generates additional code. Some of the features that are now available are ...

Basic Operations

[SmartEnum<string>]
public partial class ProductType
{
    // The source generator creates a private constructor
    public static readonly ProductType Groceries = new("Groceries");
}

// Enumeration over all defined items
IReadOnlyList<ProductType> allTypes = ProductType.Items;

// Value retrieval
ProductType productType = ProductType.Get("Groceries");        // Get by key (throws if not found)
ProductType productType = (ProductType)"Groceries";            // Same as above but by using a cast
bool found = ProductType.TryGet("Groceries", out var productType);  // Safe retrieval (returns false if not found)

// Validation with detailed error information
ValidationError? error = ProductType.Validate("Groceries", null, out ProductType? productType);

// IParsable<T> (useful for Minimal APIs)
bool parsed = ProductType.TryParse("Groceries", null, out ProductType? parsedType);

// IFormattable (e.g. for numeric keys)
string formatted = ProductGroup.Fruits.ToString("000", CultureInfo.InvariantCulture);  // "001"

// IComparable
int comparison = ProductGroup.Fruits.CompareTo(ProductGroup.Vegetables);
bool isGreater = ProductGroup.Fruits > ProductGroup.Vegetables;  // Comparison operators

Type Conversion and Equality

// Implicit conversion to key type
string key = ProductType.Groceries;  // Returns "Groceries"

// Equality comparison
bool equal = ProductType.Groceries.Equals(ProductType.Groceries);
bool equal = ProductType.Groceries == ProductType.Groceries;  // Operator overloading
bool notEqual = ProductType.Groceries != ProductType.Housewares;

// Methods inherited from Object
int hashCode = ProductType.Groceries.GetHashCode();
string key = ProductType.Groceries.ToString();  // Returns "Groceries"

// TypeConverter
var converter = TypeDescriptor.GetConverter(typeof(ProductType));
string? keyStr = (string?)converter.ConvertTo(ProductType.Groceries, typeof(string));
ProductType? converted = (ProductType?)converter.ConvertFrom("Groceries");

Pattern Matching with Switch/Map

All Switch/Map methods are exhaustive by default ensuring all cases are handled correctly.

ProductType productType = ProductType.Groceries;

// Execute different actions based on the enum value (void return)
productType.Switch(
    groceries: () => Console.WriteLine("Processing groceries order"),
    housewares: () => Console.WriteLine("Processing housewares order")
);

// Transform enum values into different types
string department = productType.Switch(
    groceries: () => "Food and Beverages",
    housewares: () => "Home and Kitchen"
);

// Direct mapping to values - clean and concise
decimal discount = productType.Map(
    groceries: 0.05m,    // 5% off groceries
    housewares: 0.10m    // 10% off housewares
);

For optimal performance Smart Enums provide overloads that prevent closures.

ILogger logger = ...;

// Prevent closures by passing the parameter as first method argument
productType.Switch(logger,
    groceries: static l => l.LogInformation("Processing groceries order"),
    housewares: static l => l.LogInformation("Processing housewares order")
);

// Use a tuple to pass multiple values
var context = (Logger: logger, OrderId: "123");

productType.Switch(context,
    groceries: static ctx => ctx.Logger.LogInformation("Processing groceries order {OrderId}", ctx.OrderId),
    housewares: static ctx => ctx.Logger.LogInformation("Processing housewares order {OrderId}", ctx.OrderId)
);

Value Objects

Install: Install-Package Thinktecture.Runtime.Extensions

Documentation: Value Objects

Value objects help solve several common problems in software development:

  1. Type Safety: Prevent mixing up different concepts that share the same primitive type

    // Problem: Easy to accidentally swap parameters
    void ProcessOrder(int customerId, int orderId) { ... }
    ProcessOrder(orderId, customerId); // Compiles but wrong!
    
    // Solution: Value objects make it type-safe
    [ValueObject<int>]
    public partial struct CustomerId { }
    
    [ValueObject<int>]
    public partial struct OrderId { }
    
    void ProcessOrder(CustomerId customerId, OrderId orderId) { ... }
    ProcessOrder(orderId, customerId); // Won't compile!
    
  2. Built-in Validation: Ensure data consistency at creation time

    [ValueObject<decimal>]
    public partial struct Amount
    {
        static partial void ValidateFactoryArguments(ref ValidationError? validationError, ref decimal value)
        {
            if (value < 0)
            {
                validationError = new ValidationError("Amount cannot be negative");
                return;
            }
            // Normalize to two decimal places
            value = Math.Round(value, 2);
        }
    }
    
    var amount = Amount.Create(100.50m);     // Success: 100.50
    var invalid = Amount.Create(-50m);       // Throws ValidationException
    
  3. Immutability: Prevent accidental modifications and ensure thread safety

  4. Complex Value Objects: Encapsulate multiple related values with validation

    [ComplexValueObject]
    public partial class DateRange
    {
        public DateOnly Start { get; }
        public DateOnly End { get; }
    
        static partial void ValidateFactoryArguments(
            ref ValidationError? validationError,
            ref DateOnly start,
            ref DateOnly end)
        {
            if (end < start)
            {
                validationError = new ValidationError(
                    $"End date '{end}' cannot be before start date '{start}'");
                return;
            }
    
            // Ensure dates are not in the past
            var today = DateOnly.FromDateTime(DateTime.Today);
            if (start < today)
            {
                validationError = new ValidationError("Start date cannot be in the past");
                return;
            }
        }
    
        public int DurationInDays => End.DayNumber - Start.DayNumber + 1;
    
        public bool Contains(DateOnly date) => date >= Start && date <= End;
    }
    
    // Usage
    var range = DateRange.Create(
        start: DateOnly.FromDateTime(DateTime.Today),
        end: DateOnly.FromDateTime(DateTime.Today.AddDays(7))
    );
    
    Console.WriteLine(range.DurationInDays);    // 8
    Console.WriteLine(range.Contains(range.Start));  // true
    

Key Features:

  • Two types of value objects:
    • Simple value objects (wrapper around a single value with validation)
    • Complex value objects (multiple properties representing a single concept)
  • Comprehensive validation support with descriptive error messages
  • Framework integration:
    • JSON serialization (System.Text.Json and Newtonsoft.Json)
    • Entity Framework Core support
    • ASP.NET Core Model Binding
    • Swashbuckle (OpenAPI)
    • MessagePack serialization
  • Rich feature set:
    • Type conversion and comparison operators
    • Custom equality comparison
    • Proper implementation of standard interfaces (IComparable, IFormattable, etc.)
    • Configurable null and empty string handling
  • Development support:
    • Roslyn Analyzers and CodeFixes for correct implementation
    • Logging for debugging and insights

For more examples and detailed documentation, see the wiki.

Discriminated Unions

Install: Install-Package Thinktecture.Runtime.Extensions

Documentation: Discriminated Unions

Discriminated unions are a powerful feature that allows a type to hold a value that could be one of several different types. They provide type safety, exhaustive pattern matching, and elegant handling of complex domain scenarios. Key benefits include:

  • Type-safe representation of values that can be one of several types
  • Exhaustive pattern matching ensuring all cases are handled
  • Elegant modeling of domain concepts with multiple states
  • Clean handling of success/failure scenarios without exceptions

The library provides two types of unions to suit different needs:

Ad hoc unions

Perfect for simple scenarios where you need to combine a few types quickly. Features:

  • Type-safe combination of up to 5 different types
  • Implicit conversions and type checking
  • Exhaustive pattern matching with Switch/Map methods
  • Built-in equality comparison
  • Support for class, struct, or ref struct implementations
// Quick combination of types
[Union<string, int>]
public partial class TextOrNumber;

// Create and use the union
TextOrNumber value = "Hello";           // Implicit conversion
TextOrNumber number = 42;               // Works with any defined type

// Type-safe access
if (value.IsString)
{
    string text = value.AsString;       // Type-safe access
    Console.WriteLine(text);
}

// Exhaustive pattern matching
var result = value.Switch(
    @string: text => $"Text: {text}",
    int32: num => $"Number: {num}"
);

// Custom property names for clarity
[Union<string, int>(T1Name = "Text", T2Name = "Number")]
public partial class BetterNamed;
// Now use .IsText, .IsNumber, .AsText, .AsNumber

Regular unions

Ideal for modeling domain concepts and complex hierarchies. Features:

  • Inheritance-based approach for complex scenarios
  • Support for both classes and records
  • Integration with value objects
  • Generic type support
  • Exhaustive pattern matching

Perfect for modeling domain concepts:

// Model domain concepts clearly
[Union]
public partial record OrderStatus
{
    public record Pending : OrderStatus;
    public record Processing(DateTime StartedAt) : OrderStatus;
    public record Completed(DateTime CompletedAt, string TrackingNumber) : OrderStatus;
    public record Cancelled(string Reason) : OrderStatus;
}

// Generic result type for error handling
[Union]
public partial record Result<T>
{
    public record Success(T Value) : Result<T>;
    public record Failure(string Error) : Result<T>;

    // Implicit conversions from T and string are implemented automatically 
}

// Usage
Result<int> result = await GetDataAsync();

var message = result.Switch(
    success: s => $"Got value: {s.Value}",
    failure: f => $"Error: {f.Error}"
);
Product Compatible and additional computed target framework versions.
.NET net7.0 is compatible.  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 is compatible.  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.  net10.0 was computed.  net10.0-android was computed.  net10.0-browser was computed.  net10.0-ios was computed.  net10.0-maccatalyst was computed.  net10.0-macos was computed.  net10.0-tvos was computed.  net10.0-windows was computed. 
Compatible target framework(s)
Included target framework(s) (in package)
Learn more about Target Frameworks and .NET Standard.

NuGet packages (12)

Showing the top 5 NuGet packages that depend on Thinktecture.Runtime.Extensions:

Package Downloads
Thinktecture.Runtime.Extensions.Json

Adds JSON support to components from Thinktecture.Runtime.Extensions when using System.Text.Json.

Thinktecture.Runtime.Extensions.AspNetCore

Adds ASP.NET Core support to components from Thinktecture.Runtime.Extensions.

Thinktecture.Runtime.Extensions.EntityFrameworkCore

Extends Entity Framework Core to support some components from Thinktecture.Runtime.Extensions.

Thinktecture.Runtime.Extensions.EntityFrameworkCore8

Extends Entity Framework Core to support some components from Thinktecture.Runtime.Extensions.

Thinktecture.Runtime.Extensions.Newtonsoft.Json

Adds better JSON support to components from Thinktecture.Runtime.Extensions when using Newtonsoft.Json.

GitHub repositories

This package is not used by any popular GitHub repositories.

Version Downloads Last Updated
9.7.0-beta04 251 10/21/2025
9.7.0-beta03 289 10/19/2025
9.7.0-beta02 303 10/16/2025
9.7.0-beta01 291 10/15/2025
9.6.4 293 10/21/2025
9.6.3 450 10/19/2025
9.6.2 1,238 9/30/2025
9.6.1 326 9/27/2025
9.6.0 552 9/18/2025
9.5.3 469 9/16/2025
9.5.2 951 8/26/2025
9.5.1 324 8/24/2025
9.5.0 1,686 8/5/2025
9.4.0 2,378 6/23/2025
9.4.0-beta02 310 6/19/2025
9.4.0-beta01 277 6/18/2025
9.3.1 839 6/17/2025
9.3.0 308 6/16/2025
9.2.0 445 6/12/2025
9.1.0 327 5/23/2025
9.0.1 326 5/21/2025
9.0.0 1,651 5/11/2025
8.10.1 321 6/17/2025
8.10.0 1,565 5/23/2025
8.9.2 358 5/21/2025
8.9.1 771 5/8/2025
8.9.0 397 5/7/2025
8.8.1 367 5/4/2025
8.8.0 725 4/27/2025
8.8.0-beta02 359 4/22/2025
8.8.0-beta01 587 4/21/2025
8.7.0 436 4/16/2025
8.7.0-beta03 359 4/15/2025
8.7.0-beta02 360 4/10/2025
8.7.0-beta01 371 4/9/2025
8.6.1 2,583 4/2/2025
8.6.0 751 3/29/2025
8.5.5 460 3/27/2025
8.5.4 950 3/24/2025
8.5.3 1,280 3/13/2025
8.5.2 305 3/12/2025
8.5.1 729 3/8/2025
8.5.0 581 3/4/2025
8.5.0-beta01 249 2/26/2025
8.4.1 2,928 2/24/2025
8.4.0 302 2/23/2025
8.3.0 1,623 2/16/2025
8.2.0 566 1/23/2025
8.1.0 300 1/17/2025
8.0.2 2,621 12/18/2024
8.0.1 414 12/11/2024
8.0.0 354 12/9/2024
8.0.0-beta13 274 11/21/2024
8.0.0-beta12 251 11/19/2024
8.0.0-beta11 258 11/15/2024
8.0.0-beta10 263 10/25/2024
8.0.0-beta09 306 10/23/2024
8.0.0-beta08 251 10/23/2024
8.0.0-beta07 253 9/27/2024
8.0.0-beta06 331 9/19/2024
8.0.0-beta05 325 9/9/2024
8.0.0-beta04 287 9/9/2024
8.0.0-beta03 262 9/8/2024
8.0.0-beta02 280 9/3/2024
8.0.0-beta01 272 8/13/2024
7.6.1 537 11/21/2024
7.6.0 18,216 11/13/2024
7.5.3 2,298 10/23/2024
7.5.2 5,882 9/27/2024
7.5.1 592 9/5/2024
7.5.0 2,404 7/9/2024
7.4.0 1,169 6/13/2024
7.3.0 691 4/18/2024
7.2.1 336 4/14/2024
7.2.0 23,899 1/31/2024
7.1.0 4,126 12/11/2023
7.0.0 398 12/10/2023
7.0.0-beta10 354 11/30/2023
7.0.0-beta09 318 11/26/2023
7.0.0-beta08 292 11/19/2023
7.0.0-beta07 325 11/17/2023
7.0.0-beta06 325 11/14/2023
7.0.0-beta05 310 11/14/2023
7.0.0-beta04 261 11/7/2023
7.0.0-beta03 306 10/19/2023
7.0.0-beta02 274 10/10/2023
7.0.0-beta01 276 10/8/2023
6.6.0 381 11/30/2023
6.5.2 428 11/17/2023
6.5.1 343 11/11/2023
6.5.0 331 11/11/2023
6.5.0-beta03 293 11/6/2023
6.5.0-beta02 267 11/5/2023
6.5.0-beta01 265 11/5/2023
6.4.1 327 11/9/2023
6.4.0 2,279 9/3/2023
6.3.0 487 8/31/2023
6.2.0 3,308 4/2/2023
6.1.0 580 3/22/2023
6.1.0-beta02 364 3/19/2023
6.1.0-beta01 354 3/19/2023
6.0.1 537 3/12/2023
6.0.0 531 3/9/2023
6.0.0-beta03 373 3/6/2023
6.0.0-beta02 370 3/5/2023
6.0.0-beta01 349 3/2/2023
5.2.0 1,808 2/6/2023
5.1.0 33,293 11/25/2022
5.0.1 3,702 10/6/2022
5.0.0 1,787 9/28/2022
5.0.0-beta03 499 9/7/2022
5.0.0-beta02 410 9/4/2022
5.0.0-beta01 861 9/4/2022
4.4.0-beta02 781 8/31/2022
4.4.0-beta01 772 8/30/2022
4.3.3 36,625 8/17/2022
4.3.3-beta01 819 8/17/2022
4.3.2 5,948 6/15/2022
4.3.2-beta02 837 6/15/2022
4.3.2-beta01 844 5/18/2022
4.3.1 12,436 5/17/2022
4.3.0 2,198 5/16/2022
4.2.0 3,332 4/24/2022
4.2.0-beta01 850 4/23/2022
4.1.3 2,180 4/13/2022
4.1.2 4,114 2/10/2022
4.1.1 2,207 2/6/2022
4.1.0 2,272 2/5/2022
4.0.1 38,190 1/16/2022
4.0.0 2,152 1/14/2022
4.0.0-beta03 847 12/17/2021
4.0.0-beta02 886 12/14/2021
4.0.0-beta01 891 12/13/2021
3.1.0 3,959 9/25/2021
3.0.0 2,068 7/10/2021
3.0.0-beta10 1,116 7/10/2021
3.0.0-beta09 1,028 4/1/2021
3.0.0-beta08 1,158 3/15/2021
3.0.0-beta07 1,037 3/14/2021
3.0.0-beta06 1,080 3/11/2021
3.0.0-beta05 1,577 2/12/2021
3.0.0-beta04 1,012 2/7/2021
3.0.0-beta03 996 1/30/2021
3.0.0-beta02 1,138 1/11/2021
3.0.0-beta01 1,080 1/10/2021
2.1.0-beta02 1,214 11/6/2020
2.1.0-beta01 462 11/6/2020
2.0.1 63,925 10/8/2019
2.0.0 1,615 10/8/2019
2.0.0-beta02 1,092 10/8/2019
2.0.0-beta01 1,228 9/30/2019
1.2.0 3,795 9/29/2019
1.2.0-beta04 1,570 8/29/2019
1.2.0-beta03 1,284 8/26/2019
1.2.0-beta02 1,222 8/12/2019
1.2.0-beta01 1,189 8/4/2019
1.1.0 5,573 5/31/2018
1.0.0 1,565 2/26/2018
0.1.0-beta5 1,250 2/23/2018
0.1.0-beta4 1,210 2/22/2018
0.1.0-beta3 1,216 2/19/2018
0.1.0-beta2 1,255 2/16/2018
0.1.0-beta1 1,319 2/11/2018