A dotnet library providing Option and Result data types, based on the concepts behind 'Railway Oriented Programming'
- Result data type encapsulating computational result that represents the result that is either a success or a failure
- Option data type that represents a value that is either present or not present
- Based on the concepts behind Railway Oriented Programming
- Extension methods that allows interacting with and going into and out of the Result and Option data types
- Fluent discoverable and commented API
- Tested solution > 99% test coverage
- Tested with Stryker mutation testing > 95%
- Integration with Polly via Futurum.Core.Polly
The Result data type is a type that represents the result of a computation that can either be a success or a failure. It is a type that is either a success or a failure.
It is a struct, preventing you from returning null instead.
This come in two forms:
var result = Result.Ok();
var result = Result.Fail("Error message");
var result = Result.Ok(2);
var result = Result.Fail<int>("Error message");
Any method that can fail should return a Result or Result<T>.
There are extension methods that allow you to easily convert dotnet or third-party dotnet libraries to return Result.
NOTE: There are async version of all the extension methods.
There are extension methods that allow you to easily convert dotnet or third-party dotnet libraries to return Result.
NOTE: There are async version of all the extension methods.
Allows you to combine multiple Result's into one, if they have the same payload types.
- If all of the Results passed are success, then return a Result success
- If any of the Results passed are failed, then return a Result failure with a ResultError of all of the failure ResultError's
var result1 = Result.Ok();
var result2 = Result.Ok();
var result3 = Result.Ok();
Result result = Result.Combine(result1, result2, result3);
- If all of the Results passed are success, then return a Result success with an IEnumerable of all the Result payloads
- If any of the Results passed are failed, then return a Result failure with a ResultError of all of the failure ResultError's
var result1 = Result.Ok(Guid.NewGuid());
var result2 = Result.Ok(Guid.NewGuid());
var result3 = Result.Ok(Guid.NewGuid());
Result<IEnumerable<Guid>> result = Result.Combine(result1, result2, result3);
- If all of the Results passed are success, then return a Result success with an IEnumerable of all the Result payloads concatenated together
- If any of the Results passed are failed, then return a Result failure with a ResultError of all of the failure ResultError's
var result1 = Result.Ok(new[] {Guid.NewGuid()}.AsEnumerable());
var result2 = Result.Ok(new[] {Guid.NewGuid()}.AsEnumerable());
var result3 = Result.Ok(new[] {Guid.NewGuid()}.AsEnumerable());
Result<IEnumerable<Guid>> result = Result.Combine(result1, result2, result3);
Allows you to combine Result's that have different payload types.
You additionally pass in a selector func.
var result1 = Result.Ok(Guid.NewGuid());
var result2 = Result.Ok(Guid.NewGuid().ToString());
Result<(Guid, string)> result = Result.CombineAll(result1, result2);
Allows you to provide a compensating action (that also returns a Result), if the Result is:
- a failure
var result = Result.Fail<int>("Error message").Compensate(() => Result.Ok(2));
Allows you to provide a compensating action (that also returns a Result), if the Result is:
- a failure of a specific type
- matches a predicate optional
var result = Result.Fail<int>("Error message").CompensateWhen<InvalidOperationException>(() => Result.Ok(2));
Allows you to provide a compensating action (that also returns a Result), if the Result is:
- a failure of a specific type
- matches a predicate optional
NOTE: this will flatten a ResultErrorComposite.
var result = Result.Fail<int>("Error message").CompensateWhenAny<InvalidOperationException>(() => Result.Ok(2));
Executes a side-effect if the Result is success.
var result = Result.Ok(2)
.Do(x =>
{
// do something
});
Executes a side-effect if the Result is failure.
var result = Result.Fail<int>("Error message")
.DoWhenFailure(x =>
{
// do something
});
Allows you to perform a side-effect on an Result<T>.
You have to say what to do when the value is success and failure.
var result = Result.Fail<int>("Error message")
.DoSwitch(x =>
{
// do something on success
},
error =>
{
// do something on failure
});
Allows you to provide additional context to a failure.
NOTE: This will create a ResultErrorComposite, with the parent IResultError being the error passed in and the existing IResultError its children.
var result = Result.Fail<int>("Error message").EnhanceWithError("Additional context");
- Creates a success Result
- Creates a success Result<T> with the payload provided
var result = Result.Ok();
var result = Result.Ok(2);
- Creates a failure Result with the error provided
- Creates a failure Result<T> with the error provided
var result = Result.Fail("Error message");
var result = Result.Fail<int>("Error message");
Creates a success Result<T> with the payload provided
var result = 2.ToResultOk();
Ignores the failure on Result and always returns Result.Ok().
var resultInput = Result.Fail(ErrorMessage);
var resultOutput = resultInput.IgnoreFailure();
Transforms an Result<T> to an Result<TR>.
The transformation will only occur if there is the Result is success.
var result = Result.Ok(1).Map(x => x.ToString());
NOTE It is assumed that the transformation func that you pass in cannot fail, i.e. throw an exception.
NOTE If it can throw an exception, use Then, returning a Result<TR> from the transformation func.
Transforms an Result<T> to a Result<TR>, allows you to pass in seperate transformations for success and failure.
var result = Result.Ok(1).MapSwitch(x => x + 1, () => 0);
Allows you to transform an Result<T> to a TR.
You have to say what to return when the value is success and failure.
int result = Result.Ok(1).Switch(x => x + 1, () => 0);
Allows you to bind two Result or Result<T> methods together.
Result resultInput = Result.Ok();
Result resultOutput = resultInput.Then(Result.Ok);
Result<int> resultInput = Result.Ok(1);
Result<string> resultOutput = resultInput.Then(value => Result.Ok(value.ToString()));
Transforms an Result<T> to a Result<TR>
Allows you to specify a predicate to change the way the Result is transformed.
Allows you to execute a func wrapping it in a try / catch
- If an exception is thrown, it will return a Result.Fail
- If no exception is thrown it will return a Result.Ok
The result of this is then passed to the then func
var result = await Result.ThenTryAsync(func, () => ERROR_MESSAGE);
A nice pattern to follow using local functions is as follows:
public Task<Result<string>> ExecuteAsync()
{
return Result.Ok().ThenTryAsync(Execute, () => ERROR_MESSAGE);
Task<string> Execute()
{
// Do something
}
}
public Task<Result<string>> ExecuteAsync()
{
return Result.Ok().ThenTryAsync(Execute, () => ERROR_MESSAGE);
Task<Result<string>> Execute()
{
// Do something
}
}
public Task<Result> ExecuteAsync()
{
return Result.Ok().ThenTryAsync(Execute, () => ERROR_MESSAGE);
Task Execute()
{
// Do something
}
}
public Task<Result> ExecuteAsync()
{
return Result.Ok().ThenTryAsync(Execute, () => ERROR_MESSAGE);
Task<Result> Execute()
{
// Do something
}
}
NOTE
- This is really Result.Try and Result.Then combined.
- This can be a good way to integrate third-party libraries.
- There can be no return value
- The return value can be a T or a Result<T>
Discards the payload.
var result = Result.Ok(2).ToNonGeneric();
Will run the func wrapping it in a try / catch.
- If an exception is thrown, it will return a Result.Fail, setting the ResultError to the exception
- If no exception is thrown it will return a Result.Ok, with the return value if there is one
var result = await Result.TryAsync(func, () => ERROR_MESSAGE);
A nice pattern to follow using local functions is as follows:
public Task<Result<string>> ExecuteAsync()
{
return Result.TryAsync(Execute, () => ERROR_MESSAGE);
Task<string> Execute()
{
// Do something
}
}
public Task<Result<string>> ExecuteAsync()
{
return Result.TryAsync(Execute, () => ERROR_MESSAGE);
Task<Result<string>> Execute()
{
// Do something
}
}
public Task<Result> ExecuteAsync()
{
return Result.TryAsync(Execute, () => ERROR_MESSAGE);
Task Execute()
{
// Do something
}
}
public Task<Result> ExecuteAsync()
{
return Result.TryAsync(Execute, () => ERROR_MESSAGE);
Task<Result> Execute()
{
// Do something
}
}
NOTE
- This can be a good way to integrate third-party libraries.
- There can be no return value
- The return value can be a T or a Result<T>
The TryGetValue extension methods work with dictionary.
They get the value from a dictionary as an Result<T>.
var values = Enumerable.Range(0, 10)
.ToDictionary(x => x, x => x);
var result = values.TryGetValue(2, ERROR_MESSAGE);
Comes out of the Result monad.
- If the Result is Ok, then return the value if there is one
- If the Result is Fail, then throw an exception, passing through the ResultError data
Result.Ok().Unwrap();
var value = Result.Ok(1).Unwrap();
Result.Fail(ErrorMessage).Unwrap();
var value = Result.Fail<int>(ErrorMessage).Unwrap();
Return only those elements of a sequence of Result<T> that are success.
If none of the elements in the sequence are success, then a Result<T> failure is returned.
var result1 = Result.Ok(1);
var result2 = Result.Ok(2);
var result3 = Result.Ok(3);
IEnumerable<int> values = new[] {result1, result2, result3}.Choose()
Filters an IEnumerable>Result<T>> based on a predicate
var values = Enumerable.Range(0, 10);
var resultInput = Result.Ok(values);
Result<IEnumerable<int>> resultOutput = resultInput.Filter(x => x % 2 == 0);
Transforms each element of a sequence to an IEnumerable<T> and flattens the resulting sequences into one sequence.
(IEnumerable<T>, Func<T, Result>) -> Result
var input = new[] {1, 2};
Result outputResult = input.FlatMap(x => Result.Ok());
(IEnumerable<T>, Func<T, Result<TR>>) -> Result<IEnumerable<TR>>
var input = new[] {1, 2};
Result outputResult = input.FlatMap(x => Result.Ok());
(IEnumerable<T>, Func<T, IEnumerable<TR>>) -> Result<IEnumerable<TR>>
var input = new[] {1, 2};
Result<IEnumerable<int>> outputResult = input.FlatMap(x =>
{
if (x == 1) return Result.Fail<IEnumerable<int>>(ErrorMessage1);
return Result.Fail<IEnumerable<int>>(ErrorMessage2);
});
(Result<IEnumerable<T>>, Func<T, Result<IEnumerable<TR>>>) -> Result<IEnumerable<TR>>
var input = new[] {1, 2}.AsEnumerable().ToResultOk();
Result<IEnumerable<int>> outputResult = input.FlatMap(x =>
{
if (x == 1) return Result.Fail<IEnumerable<int>>(ErrorMessage1);
return Result.Fail<IEnumerable<int>>(ErrorMessage2);
});
(Result<IEnumerable<T>>, Func<T, Result<TR>>) -> Result<IEnumerable<TR>>
var input = new[] {1, 2}.AsEnumerable().ToResultOk();
Result<IEnumerable<int>> outputResult = input.FlatMap(x =>
{
if (x == 1) return Result.Fail<int>(ErrorMessage1);
return Result.Fail<int>(ErrorMessage2);
});
Result<IEnumerable<IEnumerable<T>>> -> Result<IEnumerable<T>>
var input = new[] {1, 2};
Result<IEnumerable<int>> outputResult = input.FlatMap(x => new[] { x * 2 }.AsEnumerable().ToResultOk());
NOTE
- If ParallelOptions is not specified, then it will use default ParallelOptions.
- There are overloads that take a ParallelOptions, so you can control the level of concurrency.
Transforms an Result<IEnumerable<T>> to a Result<IEnumerable>TR>>
public record Container(IEnumerable<int> Values);
var values = Enumerable.Range(0, 1)
.Select(x => new Container(new []{x}));
Result<IEnumerable<Container>> resultInput = Result.Ok(values);
Result<IEnumerable<int>> resultOutput = resultInput.FlatMapAs(x => x.Values, x -> x * 2);
Transforms each element of a sequence to an IEnumerable<T> and flattens the resulting sequences into one sequence.
NOTE This runs sequentially, i.e. runs with a ParallelOptions MaxDegreeOfParallelism = 1
var input = new[] { 1, 2 };
Result outputResult = await input.FlatMapSequentialAsync(x =>
{
if (x == 1) return Result.FailAsync(ErrorMessage1);
return Result.FailAsync(ErrorMessage2);
});
Transforms each element of a sequence to an IEnumerable<T>, returning the first error it finds.
var input = new [] {1,2, 3};
Result outputResult = input.FlatMapSequentialUntilFailure(x => x == 2 ? Result.Fail(ErrorMessage1) : Result.Ok());
Transforms an Result<IEnumerable<T>> to a Result<IEnumerable>TR>>
var values = Enumerable.Range(0, 1);
var resultInput = Result.Ok(values);
Result<IEnumerable<int>> resultOutput = resultInput.MapAs(x => x + 1);
Return the first element of a sequence of Result<T> that are success.
If none of the elements in the sequence match are success, then a Result<T> failure is returned.
var result1 = Result.Ok(1);
var result2 = Result.Ok(2);
var result3 = Result.Ok(3);
Result<int> result = new[] {result1, result2, result3}.Pick(ErrorMessage);
Return the first element of a sequence of Result that are failure.
If none of the elements in the sequence match are failure, then a Result failure is returned.
var result1 = Result.Ok();
var result2 = Result.Ok();
var result3 = Result.Ok();
Result result = new[] {result1, result2, result3}.PickFailureOrSuccess();
Transforms an Result<IEnumerable<T>> to a Result<IEnumerable>TR>>
var values = Enumerable.Range(0, 1);
var resultInput = Result.Ok(values);
Result<int> resultOutput = resultInput.ThenAs(x => (x + 1).ToResultOk());
Sequentially runs through a sequence of elements applying a function that returns a Result<TR>. The first one that is success is returned.
var input = new[] { 1, 2, 3 };
Result<string> result = input.TryFirst(x =>
{
if (x == 1)
{
return Result.Fail<string>(ErrorMessage);
}
return x.ToString().ToResultOk();
},
ErrorMessage);
Try to create a Dictionary<TKey, TElement;> from an IEnumerable<T> according to specified key selector and element selector functions.
NOTE: This will explicitly checks for duplicate keys and return a Result<Dictionary<TKey, TElement>> failure if there are any.
var numbers = Enumerable.Range(0, 10)
.ToList();
Result<Dictionary<int, int>> result = numbers.TryToDictionary(x => x, x => x);
Return only those elements of a sequence of Result<T> that are success.
If none of the elements in the sequence are success, then a Result<T> failure is returned.
IAsyncEnumerable<Result<int>> inputResults = AsyncEnumerable.Range(1, 3)
.Select(x => Core.Result.Result.Ok(x));
List<int> values = await inputResults.Choose().ToListAsync();
Filters an IEnumerable>Result<T>> based on a predicate
var values = AsyncEnumerable.Range(0, 1);
var resultInput = Result.Ok(values);
Result<IAsyncEnumerable<int>> returnedResult = resultInput.Filter(x => x % 2 == 0);
Transforms an Result<IEnumerable<T>> to a Result<IEnumerable>TR>>
var values = AsyncEnumerable.Range(0, 1);
var resultInput = Result.Ok(values);
Result<IAsyncEnumerable<int>> returnedResult = resultInput.MapAs(x => x * 2);
You can move between an Option and a Result using the 'ToResult' extension method. You just need to provide the error if the Option is not present.
var option = "hello".ToOption();
var result = option.ToResult(ErrorMessage);
This is not supported as it would result in the loss of the error context.
Failures are represented as by the IResultError interface. These represent the context of the error.
This interface has two implementations:
- IResultErrorNonComposite
- IResultErrorComposite
This is the base interface for all IResultError implementations that are not composite, i.e. have no children.
This is the base interface for all IResultError implementations that are composite, i.e. have children.
There are a number of built-in IResultError implementations.
You are free to create your own, by implementing either the IResultErrorNonComposite or IResultErrorComposite interfaces.
Represents a string error message, without any other context.
You can transform a string to a ResultErrorMessage using the extension method ToResultErrorMessage.
var resultError = "error message".ToResultError();
NOTE Most extension methods have an overload that allows you to pass in a error string.
Represents a exception error.
You can transform an Exception to a ResultErrorMessage using the extension method ToResultErrorMessage.
var resultError = new Exception("error message").ToResultError();
Represents an empty error message, this is mainly used internally in the library.
var resultError = ResultErrorEmpty.Value;
This is a special IResultError, it allows you to have a tree structure of IResultError's.
var resultError = resultErrors.ToResultError();
var resultError = ResultErrorCompositeExtensions.ToResultError(parentResultError, resultErrors);
There are four supported ways to get the data from an IResultError:
- GetErrorString
- GetErrorStringSafe
- GetErrorStructure
- GetErrorStructureSafe
This gets a string representing the error(s). Different implementation of IResultError deal with this differently.
ResultErrorComposite will call GetErrorString on all of their children, separating them with the specified separator.
This gets a string representing the error(s). Different implementation of IResultError deal with this differently.
ResultErrorComposite will call GetErrorStringSafe on all of their children, separating them with the specified separator.
NOTE: The error message will be safe to return to the client, that is, it will not contain any sensitive information e.g. StackTrace.
This gets a tree structure representing the error(s). This tree structure is serializable to Json.
ResultErrorComposite will call GetErrorStructure on all of their children, and add them as children to the ResultErrorComposite.
This gets a tree structure representing the error(s). This tree structure is serializable to Json.
ResultErrorComposite will call GetErrorStructureSafe on all of their children, and add them as children to the ResultErrorComposite.
NOTE: The error message will be safe to return to the client, that is, it will not contain any sensitive information e.g. StackTrace.
The Option data type is a type that represents a value that is either present or not present. It is a type that is either HasValue
or HasNoValue
.
It is a struct, preventing you from returning null instead.
It can be used for value and reference types.
Allows you to transform an Option<T> to an Option<TR>.
The transformation will only occur if there is a value present.
var option = Option.From(1).Map(x => x + 1);
Allows you to transform an Option<T> to a TR.
You have to say what to return when the value is present and not present.
var option = Option.From(1).Switch(x => x + 1, () => 0);
Allows you to perform a side-effect on an Option<T>.
You have to say what to do when the value is present and not present.
var option = Option.From(1).Switch(x =>
{
// do something when the value is present
},
() =>
{
// do something when the value is not present
});
Gets the value from a dictionary as an Option<T>.
var values = Enumerable.Range(0, 10)
.ToDictionary(x => x, x => x);
var option = values.TryGetValue(2);
There are a number of factory extension methods.
Converts a T to an Option<T>
If the value is not null, then it will be a Option<T> with the value on it.
var option = Option.From(1);
var option = Option.From("hello");
If the value is null, then it will be a Option<T> with the value not set.
var option = Option.From((int?)null);
var option = Option.From((object)null));
Returns a Option<T> with the value not set.
var option = Option<int>.None;
var option = Option<string>.None;
Converts a T to an Option<T>
If the value is not null, then it will be a Option<T> with the value on it.
var option = 1.ToOption();
var option = "hello".ToOption();
If the value is null, then it will be a Option<T> with the value not set.
var option = ((int?)null).ToOption();
var option = ((object)null).ToOption();
There is an implicit conversion in place. So if your method returns an T, then it will be automatically converted to an Option<T>.
The OrElse extension methods are good when you have a precedence order.
The GetValueOrDefault extension methods are used to as a terminating method and will take you out if an Option, returning the default value provided if the Option has no value.
var value = Option.From(1)
.OrElse(Option.From(2))
.OrElse(Option.From(3))
.GetValueOrDefault(4);
There are a lot of TryParse extension methods provided, for a lot of the built-in framework types:
var option = true.ToString().TryParseBool();
var option = 1.ToString().TryParseInt();
var option = 1.ToString().TryParseLong();
var option = 1m.ToString().TryParseDecimal();
var option = 1d.ToString().TryParseDouble();
var option = 1f.ToString().TryParseFloat();
var option = Guid.NewGuid().ToString().TryParseGuid();
var option = DateTime.Now.ToString().TryParseDateTime();
var option = DateOnly.FromDateTime(DateTime.Now).ToString().TryParseDateOnly();
var option = TimeOnly.FromDateTime(DateTime.Now).ToString().TryParseTimeOnly();
var option = EnumValues.Value2.ToString().TryParseEnum<EnumValues>();
Return only those elements of a sequence of Option<T> that have a value.
var options = Enumerable.Range(0, 10).Select(i => i.ToOption());
var returnValues = options.MapSwitch(value => trueValue, () => falseValue);
Transforms each element of a sequence into a new form.
The transformation will only occur if there is a value present.
var options = Enumerable.Range(0, 5).Select(i => i.ToOption());
var values = options.Map(x => x + 1);
Transforms each element of a sequence based, allows you to pass in seperate transformations for if the value is there or not.
var options = Enumerable.Range(0, 10).Select(i => i.ToOption());
var returnValues = options.MapSwitch(value => trueValue, () => falseValue);
Return the first element of a sequence based on if the value is there.
If none of the elements in the sequence match the predicate, then Option<T>.None is returned.
var option1 = Core.Option.Option.From(1);
var option2 = Core.Option.Option.From(2);
var option3 = Core.Option.Option.From(3);
var option = new[] {option1, option2, option3}.Pick();
Returns the element at a specified index in a sequence.
- If the index is out of range, then an Option<T>.None is returned.
- If the index is in range, the element at the specified position in the source sequence is returned as a Option<T>.
var option = Enumerable.Range(0, 10).TryElementAt(2);
Returns the first element of a sequence.
- If the sequence is empty, then an Option<T>.None is returned.
- If the sequence is not empty, the first element is returned as a Option<T>.
var option = Enumerable.Range(0, 10).TryFirst();
Returns the last element of a sequence.
- If the sequence is empty, then an Option<T>.None is returned.
- If the sequence is not empty, the last element is returned as a Option<T>.
var option = Enumerable.Range(0, 10).TryLast();
Returns the only element of a sequence.
- If the sequence is empty, then an Option<T>.None is returned, wrapped in a Result<T>.IsSuccess.
- If the sequence has 1 value, then the value is returned as a Option<T>, wrapped in a Result<T>.IsSuccess.
var option = Enumerable.Range(0, 10).TrySingle();
For structs there is interop with Nullable
Use the ToOption extension method.
var option = ((int?)null).ToOption();
var option = ((int?)1).ToOption();
Use the ToNullable extension method.
int? value = null;
var option = value.ToOption();
var nullable = option.ToNullable();
int? value = 1;
var option = value.ToOption();
var nullable = option.ToNullable();
You can move between an Option and a Result using the 'ToResult' extension method. You just need to provide the error if the Option is not present.
var option = "hello".ToOption();
var result = option.ToResult(ErrorMessage);
This is not supported as it would result in the loss of the error context.
dotnet tool install -g dotnet-stryker
dotnet build ./test/Futurum.Core.Tests
dotnet stryker