Ametrin.Optional
0.2.6
dotnet add package Ametrin.Optional --version 0.2.6
NuGet\Install-Package Ametrin.Optional -Version 0.2.6
<PackageReference Include="Ametrin.Optional" Version="0.2.6" />
<PackageVersion Include="Ametrin.Optional" Version="0.2.6" />
<PackageReference Include="Ametrin.Optional" />
paket add Ametrin.Optional --version 0.2.6
#r "nuget: Ametrin.Optional, 0.2.6"
#:package Ametrin.Optional@0.2.6
#addin nuget:?package=Ametrin.Optional&version=0.2.6
#tool nuget:?package=Ametrin.Optional&version=0.2.6
Ametrin.Optional
A modern, allocation-free library providing robust optional types for .NET. It offers a flexible and efficient way to handle nullable values, errors, and exceptions with a fluent, monadic API.
dotnet add package Ametrin.Optional
Features
- 🚀 Zero Allocation - Designed for high-performance scenarios
- 🧩 Monadic API - Fluent interface for transformations and error handling
- 🔄 Easy Integration - Seamless integration with existing C# code
- 🎯 Multiple Option Types - Different types for various use cases
- 💪 Async Support - First-class support for async operations
Core Types
Option<T>
Represents a value of type T
or nothing (error state).
// Creating Options
Option<T> b = Option.Success(someT); // explicit success creation (throws if someT is null)
Option<T> c = Option.Error<T>(); // explicit error creation
Option<T> a = Option.Of(someT); // returns Option.Error<T>() if someT is null
Option<T> a = someT; // implicit conversion from T -> Option.Of(someT)
Option<T> d = default; // default results in an error state -> Option.Error<T>()
Result<T>
and Result<T, E>
Like Option<T>
but with error information.
Result<T>
stores an Exception as error
Result<T, E>
stores a custom error type
Result<T> b = Result.Success(someT); // explicit success creation (throws if someT is null)
Result<T> c = Result.Error<T>(new Exception()); // explicit error creation
Result<T> a = Result.Of(someT); // returns Result.Error<T>(new NullReferenceException()) if someT is null
Result<T> a = someT; // implicit conversion from T -> Result.Of(someT)
Result<T> a = new Exception(); // implicit conversion from Exception -> Result.Error<T>(new Exception())
// same applies for Result<T, E> (with generic arguments)
ErrorState
and ErrorState<E>
Represents a success state or an error value
ErrorState
stores an Exception as error
ErrorState<E>
stores a custom error type
ErrorState success = default; // ErrorState.Success()
ErrorState error = new Exception(); // ErrorState.Success(new Exception())
// same applies for ErrorState<E> (with generic arguments)
Option
Represents a success state or error state. Holds no value.
Option success = true; // Option.Success();
Option error = false; // Option.Error();
Core API
// common operations
option.Map(value => value * 2); // transform value if present
option.Require(value => value > 0); // filter based on predicate
option.Reject(value => value > 0); // inverse of Require
option.Or(defaultValue); // return value or defaultValue
option.OrThrow(); // return value or throw if error
option.Match( // reduce to a value (equivalent to .Map(success).Or(error) but supports ref structs)
success: value => value,
error: error => defaultValue
);
option.Consume( // handle success and error
success: value => { },
error: error => { }
);
// all operations have an overload that allows you to pass an argument into the delegate an avoid the closure
Advanced Features
Tuple Operations
// Combine multiple options
(optionA, optionB).Map((a, b) => a + b);
(optionA, optionB).Consume(
success: (a, b) => Console.WriteLine($"{a} + {b} = {a + b}"),
error: () => Console.WriteLine("One of the values was missing")
);
Async Support
// Async operations with full option support
var text = await new FileInfo("file.txt")
.RequireExists()
.MapAsync(f => File.ReadAllTextAsync(f.FullName))
.MapAsync(s => s.ToLower());
await text.ConsumeAsync(text => File.WriteAllTextAsync("output.txt", text));
RefOption<T>
A reduced version of Option<T>
that can hold a ref struct as value
Edge Cases
For edge cases, high-performance scenarios where delegates cannot be static or when you absolutly need to modify the controll flow use Branch
:
if(result.Branch(out var value, out var error))
{
// success
}
else
{
// error
}
If there is no way around it you can get low-level access to all option types through OptionsMarshall
.
Testing
For testing code using Ametrin.Optional types, use the Ametrin.Optional.Testing.TUnit
extensions:
await Assert.That(option).IsSuccess(expectedValue);
await Assert.That(option).IsError();
Contributing
Contributions are welcome! Feel free to:
- Create issues for bugs or feature requests
- Submit pull requests (discuss design first)
- Add extensions for more testing frameworks
Performance
The library is designed with performance in mind and has minimal to no overhead. Example benchmark for parsing a DateTime:
| Method | Mean | Error | StdDev | Allocated |
|------------------ |----------:|---------:|---------:|----------:|
| Default_Success | 291.30 ns | 1.093 ns | 1.022 ns | - |
| Option_Success | 310.56 ns | 1.914 ns | 1.697 ns | - |
| RefOption_Success | 300.98 ns | 1.454 ns | 1.360 ns | - |
| Default_Error | 72.12 ns | 0.289 ns | 0.226 ns | - | // using TryParse
| Option_Error | 67.75 ns | 0.324 ns | 0.287 ns | - |
| RefOption_Error | 72.75 ns | 0.145 ns | 0.136 ns | - |
Product | Versions Compatible and additional computed target framework versions. |
---|---|
.NET | 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. |
-
net9.0
- No dependencies.
NuGet packages (1)
Showing the top 1 NuGet packages that depend on Ametrin.Optional:
Package | Downloads |
---|---|
Ametrin.Optional.Testing.TUnit
Package Description |
GitHub repositories
This package is not used by any popular GitHub repositories.