Skip to content

Commit

Permalink
update readme
Browse files Browse the repository at this point in the history
  • Loading branch information
lucasteles committed Jul 6, 2023
1 parent d85a7a2 commit d70cd39
Show file tree
Hide file tree
Showing 5 changed files with 328 additions and 14 deletions.
175 changes: 174 additions & 1 deletion README.md
Original file line number Diff line number Diff line change
Expand Up @@ -11,7 +11,8 @@

# Resulteles

Practical `Result` data structure for C#
**Resulteles is a practical and lightweight library that define and facilitate the use of a `success|failure` data
structure called `Result` (aka `Either`).**

## Getting started

Expand All @@ -20,3 +21,175 @@ Practical `Result` data structure for C#
```ps
$ dotnet add package Resulteles
```

### Creating Results

You can create any result using the static methods on `Result`

```csharp
// explicitly
var resultSuccess = Result.Ok<int, string>(42);
var resultError = Result.Error<int, string>("Something wrong!");

var processResult = Result.Try(() => { return 1; });
var processAsyncResult = Result.TryAsync(async () => { return await Task.FromResult(2); });
```

In most of the scenarios you will be using result as a return of a method or function, because of that it
implements implicit convert operators from the generic types, reducing the generic type definition noise

```csharp
public Result<int, string> GetEvenNumber(int number)
{
if (number % 2 is 0)
return number;
else
return $"{number} is not even!";
}
```

### Reading Result Value

The `Result` type acts like a blackbox, by default there is no way to just get the typed value mostly because it can be
two things, an `Ok` or an `Error`.

```csharp
Result<int, string> result = GetEvenNumber(2);

// return -1 on error
int value1 = result.Match(
n => n,
error => -1
);

result.Switch(
n => Console.WriteLine($"the number {n} is even"),
error => Console.WriteLine(error)
);

// return 0 on error
var value2 = result.DefaultValue(0);

// lazily define the bad case value
var value3 = result.DefaultWith(e => e.Length);

// get the value or throw an exeption
var value4 = result.GetValueOrThrow();

// try/out pattern
if (result.TryGet(out var ok, out var error))
Console.WriteLine($"the number {ok} is even");
else
Console.WriteLine(error);
```

### Operators

Most types you not even need to extract the value, you can just apply functions or combine with other `Result` types:

```csharp
var result1 = Result.Ok<int, string>(1);

// Map the result value only when it is Ok
var result2 = result1.Select(n => n + 1);

// map for ok or error case
Result<string, int> result3 = result1.Select(n => n.ToString(), err => err.Length);

// SelectMany / Bind / FlatMamap
// Nested results transformation
Result<int, string> resultCombined = result1.SelectMany(n => GetEvenNumber(n));

// Zip Results together into a tuple
// only resolves as Ok when both are Ok
Result<(int, int), string> resultZip = result1.Zip(result2);

// Zip Results together
// only resolves as Ok when both are Ok
Result<int, string> resultSum = result1.Zip(result2, (a, b) => a + b);

// Enumerate result, empty when error, singleton when error (see ToArray() also)
IEnumerable<int> resultEnumerable = result1.AsEnumerable();

// LINQ QUERY
// you can use linq query to easily express map and bind
// simple map
var result2Linq = from r in result1 select r + 1;

// combining multiple results (fail fast at first non ok value)
var resultEvenSum =
from a in GetEvenNumber(2)
from b in GetEvenNumber(4)
from c in GetEvenNumber(6)
select a + b + c;
}
```

### Async/Await

This library implements some helpers to seamless deal with async Result code

```csharp
var result = GetEvenNumber(2);

// async match overload
int value1 = await result.Match(
async n =>
{
await Task.Delay(n);
return n;
},
_ => -1
);

await result.SwitchAsync(
async n => await Console.Out.WriteLineAsync($"the number {n} is even"),
async error => await Console.Error.WriteLineAsync(error)
);

// async projection counterparts
Result<int, string> resultWithDelay = await result.SelectAsync(async n =>
{
await Task.Delay(n);
return n;
});

// Transform Result<Task<int>, string> in a awaitable Task<Result<int, string>>
Result<int, string> resultWithDelay2 = await result.Select(async n =>
{
await Task.Delay(n);
return n;
}).ToTask();
```

# Helper extensions

```csharp
// result collection
var results = new[] { GetEvenNumber(2), GetEvenNumber(3), GetEvenNumber(4) };

// filter ok values
IEnumerable<int> okResultsOnly = results.GetOkValues(); // [2,4]
// map/reduce based on a result function
var numbers = new[] { 1, 2, 3, 4 };
var calculatedResults = numbers.ChooseResult(x => GetEvenNumber(x)); // [2,4]
// group a collection of result into a Result of collection
// collapse to error if any error is in the collection
Result<IReadOnlyList<int>, string> singleResult = results.GroupResults();

// Transform Result<Task<int>, string> in a awaitable Task<Result<int, string>>
Result<int, string> resultWithDelay2 = await results[0].Select(async n =>
{
await Task.Delay(n);
return n;
}).ToTask();

```

# Alternatives

- [FluentResults](https://github.com/altmann/FluentResults)
- [language-ext](https://github.com/louthy/language-ext)
13 changes: 6 additions & 7 deletions src/ResultLinqExtensions.cs
Original file line number Diff line number Diff line change
Expand Up @@ -123,7 +123,7 @@ Func<TOk, Result<TMap, TError>> selector
/// <summary>
/// Return one result with all ok values or first error
/// </summary>
public static Result<IReadOnlyList<TOk>, TError> ToSingleResult<TOk, TError>(
public static Result<IReadOnlyList<TOk>, TError> GroupResults<TOk, TError>(
this IEnumerable<Result<TOk, TError>> results
)
{
Expand All @@ -140,15 +140,15 @@ this IEnumerable<Result<TOk, TError>> results
/// <summary>
/// Return one result with all ok values or first error
/// </summary>
public static Result<IReadOnlyList<TMap>, TError> ToSingleResult<TOk, TError, TMap>(
public static Result<IReadOnlyList<TMap>, TError> GroupResults<TOk, TError, TMap>(
this IEnumerable<TOk> results,
Func<TOk, Result<TMap, TError>> selector
) => results.Select(selector).ToSingleResult();
) => results.Select(selector).GroupResults();

/// <summary>
/// Return one result with all ok or all errors
/// </summary>
public static Result<IReadOnlyList<TOk>, IReadOnlyList<TError>> ToSingleResultWithAllErrors<TOk, TError>(
public static Result<IReadOnlyList<TOk>, IReadOnlyList<TError>> GroupResultsWithErrors<TOk, TError>(
this IEnumerable<Result<TOk, TError>> results
)
{
Expand All @@ -170,9 +170,8 @@ this IEnumerable<Result<TOk, TError>> results
/// <summary>
/// Return one result with all ok values or first error
/// </summary>
public static Result<IReadOnlyList<TMap>, IReadOnlyList<TError>> ToSingleResultWithAllErrors<TOk, TError, TMap>(
public static Result<IReadOnlyList<TMap>, IReadOnlyList<TError>> GroupResultsWithErrors<TOk, TError, TMap>(
this IEnumerable<TOk> results,
Func<TOk, Result<TMap, TError>> selector
) => results.Select(selector).ToSingleResultWithAllErrors();

) => results.Select(selector).GroupResultsWithErrors();
}
1 change: 1 addition & 0 deletions src/ResultStatic.cs
Original file line number Diff line number Diff line change
Expand Up @@ -60,4 +60,5 @@ public static async Task<Result<TOk, Exception>> TryAsync<TOk>(Func<Task<TOk>> f
return e;
}
}

}
12 changes: 6 additions & 6 deletions tests/Resulteles.Tests/ResultLinqTests.cs
Original file line number Diff line number Diff line change
Expand Up @@ -202,7 +202,7 @@ public void ShouldChooseOnlyOkValuesFromEnumerable()
public void ShouldTraverseResultIntoSingleErrorResult()
{
var results = new Result<int, string>[] { "Err1", 42, "Err2", 99 };
results.ToSingleResult().Should().Be(
results.GroupResults().Should().Be(
Result.Error<IReadOnlyList<int>, string>("Err1")
);
}
Expand All @@ -211,7 +211,7 @@ public void ShouldTraverseResultIntoSingleErrorResult()
public void ShouldTraverseResultIntoErrorListResult()
{
var results = new Result<int, string>[] { "Err1", 42, "Err2", 99 };
results.ToSingleResultWithAllErrors().Should()
results.GroupResultsWithErrors().Should()
.BeOfType<Result<IReadOnlyList<int>, IReadOnlyList<string>>>()
.And.BeEquivalentTo(new
{
Expand All @@ -224,7 +224,7 @@ public void ShouldTraverseResultIntoErrorListResult()
public void ShouldTraverseResultIntoOkListResult()
{
var results = new Result<int, string>[] { 42, 99 };
results.ToSingleResult().Should()
results.GroupResults().Should()
.BeOfType<Result<IReadOnlyList<int>, string>>()
.And
.BeEquivalentTo(new
Expand All @@ -237,7 +237,7 @@ public void ShouldTraverseResultIntoOkListResult()
[Test]
public void ShouldTraverseEnumerableIntoSingleErrorResult()
{
var results = new[] { 2, 3, 4, 5 }.ToSingleResult(IsEven);
var results = new[] { 2, 3, 4, 5 }.GroupResults(IsEven);
results.Should().Be(
Result.Error<IReadOnlyList<int>, string>("Invalid 3")
);
Expand All @@ -246,7 +246,7 @@ public void ShouldTraverseEnumerableIntoSingleErrorResult()
[Test]
public void ShouldEnumerableResultIntoErrorListResult()
{
var results = new[] { 2, 3, 4, 5 }.ToSingleResultWithAllErrors(IsEven);
var results = new[] { 2, 3, 4, 5 }.GroupResultsWithErrors(IsEven);
results.Should()
.BeOfType<Result<IReadOnlyList<int>, IReadOnlyList<string>>>()
.And.BeEquivalentTo(new
Expand All @@ -259,7 +259,7 @@ public void ShouldEnumerableResultIntoErrorListResult()
[Test]
public void ShouldEnumerableResultIntoOkListResult()
{
var results = new[] { 2, 4, 6 }.ToSingleResult(IsEven);
var results = new[] { 2, 4, 6 }.GroupResults(IsEven);
results.Should()
.BeOfType<Result<IReadOnlyList<int>, string>>()
.And
Expand Down
Loading

0 comments on commit d70cd39

Please sign in to comment.