Skip to content

Commit c8e4edd

Browse files
feature/result-select (#63)
Added `Select` and `SelectMany` functions to allow interchangeable `Result` to `Result<T>` transformations.
1 parent 3234a7d commit c8e4edd

File tree

8 files changed

+254
-48
lines changed

8 files changed

+254
-48
lines changed

OnixLabs.Core.UnitTests/ResultNonGenericTests.cs

+56-4
Original file line numberDiff line numberDiff line change
@@ -236,6 +236,32 @@ public void ResultFailureMatchShouldProduceExpectedResult()
236236

237237
[Fact(DisplayName = "Result Success.Select should produce the expected result")]
238238
public void ResultSuccessSelectShouldProduceExpectedResult()
239+
{
240+
// Given
241+
Result expected = Result.Success();
242+
243+
// When
244+
Result actual = expected.Select(() => { });
245+
246+
// Then
247+
Assert.Equal(expected, actual);
248+
}
249+
250+
[Fact(DisplayName = "Result Failure.Select should produce the expected result")]
251+
public void ResultFailureSelectShouldProduceExpectedResult()
252+
{
253+
// Given
254+
Result expected = Result.Failure(new Exception("Failure"));
255+
256+
// When
257+
Result actual = expected.Select(() => { });
258+
259+
// Then
260+
Assert.Equal(expected, actual);
261+
}
262+
263+
[Fact(DisplayName = "Result Success.Select<TResult> should produce the expected result")]
264+
public void ResultSuccessSelectTResultShouldProduceExpectedResult()
239265
{
240266
// Given
241267
const int expected = 9;
@@ -248,8 +274,8 @@ public void ResultSuccessSelectShouldProduceExpectedResult()
248274
Assert.Equal(expected, actual);
249275
}
250276

251-
[Fact(DisplayName = "Result Failure.Select should produce the expected result")]
252-
public void ResultFailureSelectShouldProduceExpectedResult()
277+
[Fact(DisplayName = "Result Failure.Select<TResult> should produce the expected result")]
278+
public void ResultFailureSelectTResultShouldProduceExpectedResult()
253279
{
254280
// Given
255281
Exception exception = new("failure");
@@ -264,6 +290,32 @@ public void ResultFailureSelectShouldProduceExpectedResult()
264290

265291
[Fact(DisplayName = "Result Success.SelectMany should produce the expected result")]
266292
public void ResultSuccessSelectManyShouldProduceExpectedResult()
293+
{
294+
// Given
295+
Result expected = Result.Success();
296+
297+
// When
298+
Result actual = expected.SelectMany(Result.Success);
299+
300+
// Then
301+
Assert.Equal(expected, actual);
302+
}
303+
304+
[Fact(DisplayName = "Result Failure.SelectMany should produce the expected result")]
305+
public void ResultFailureSelectManyShouldProduceExpectedResult()
306+
{
307+
// Given
308+
Result expected = Result.Failure(new Exception("Failure"));
309+
310+
// When
311+
Result actual = expected.SelectMany(Result.Success);
312+
313+
// Then
314+
Assert.Equal(expected, actual);
315+
}
316+
317+
[Fact(DisplayName = "Result Success.SelectMany<TResult> should produce the expected result")]
318+
public void ResultSuccessSelectTResultManyShouldProduceExpectedResult()
267319
{
268320
// Given
269321
const int expected = 9;
@@ -276,8 +328,8 @@ public void ResultSuccessSelectManyShouldProduceExpectedResult()
276328
Assert.Equal(expected, actual);
277329
}
278330

279-
[Fact(DisplayName = "Result Failure.SelectMany should produce the expected result")]
280-
public void ResultFailureSelectManyShouldProduceExpectedResult()
331+
[Fact(DisplayName = "Result Failure.SelectMany<TResult> should produce the expected result")]
332+
public void ResultFailureSelectTResultManyShouldProduceExpectedResult()
281333
{
282334
// Given
283335
Exception exception = new("failure");

OnixLabs.Core.UnitTests/ResultTests.cs

+65-7
Original file line numberDiff line numberDiff line change
@@ -430,25 +430,54 @@ public void ResultFailureMatchShouldProduceExpectedResult()
430430
public void ResultSuccessSelectShouldProduceExpectedResult()
431431
{
432432
// Given
433-
const int expected = 9;
434-
Result<int> result = 3;
433+
Result expected = Result.Success();
434+
Result<int> result = 123;
435435

436436
// When
437-
Result<int> actual = result.Select(value => value * value);
437+
Result actual = result.Select(_ => { });
438438

439439
// Then
440440
Assert.Equal(expected, actual);
441441
}
442442

443443
[Fact(DisplayName = "Result Failure.Select should produce the expected result")]
444444
public void ResultFailureSelectShouldProduceExpectedResult()
445+
{
446+
// Given
447+
Exception exception = new("Failure");
448+
Result expected = Result.Failure(exception);
449+
Result<int> result = Result<int>.Failure(exception);
450+
451+
// When
452+
Result actual = result.Select(_ => { });
453+
454+
// Then
455+
Assert.Equal(expected, actual);
456+
}
457+
458+
[Fact(DisplayName = "Result Success.Select<TResult> should produce the expected result")]
459+
public void ResultSuccessSelectTResultShouldProduceExpectedResult()
460+
{
461+
// Given
462+
Result<int> expected = 9;
463+
Result<int> result = 3;
464+
465+
// When
466+
Result<int> actual = result.Select(x => x * x);
467+
468+
// Then
469+
Assert.Equal(expected, actual);
470+
}
471+
472+
[Fact(DisplayName = "Result Failure.Select<TResult> should produce the expected result")]
473+
public void ResultFailureSelectTResultShouldProduceExpectedResult()
445474
{
446475
// Given
447476
Exception exception = new("failure");
448477
Result<int> result = Result<int>.Failure(exception);
449478

450479
// When
451-
Result<int> actual = result.Select(value => value * value);
480+
Result<int> actual = result.Select(x => x * x);
452481

453482
// Then
454483
Assert.Equal(Result<int>.Failure(exception), actual);
@@ -458,25 +487,54 @@ public void ResultFailureSelectShouldProduceExpectedResult()
458487
public void ResultSuccessSelectManyShouldProduceExpectedResult()
459488
{
460489
// Given
461-
const int expected = 9;
490+
Result expected = Result.Success();
462491
Result<int> result = 3;
463492

464493
// When
465-
Result<int> actual = result.SelectMany<int>(value => value * value);
494+
Result actual = result.SelectMany(_ => Result.Success());
466495

467496
// Then
468497
Assert.Equal(expected, actual);
469498
}
470499

471500
[Fact(DisplayName = "Result Failure.SelectMany should produce the expected result")]
472501
public void ResultFailureSelectManyShouldProduceExpectedResult()
502+
{
503+
// Given
504+
Exception exception = new("Failure");
505+
Result expected = Result.Failure(exception);
506+
Result<int> result = Result<int>.Failure(exception);
507+
508+
// When
509+
Result actual = result.SelectMany(_ => Result.Success());
510+
511+
// Then
512+
Assert.Equal(expected, actual);
513+
}
514+
515+
[Fact(DisplayName = "Result Success.SelectMany<TResult> should produce the expected result")]
516+
public void ResultSuccessSelectTResultManyShouldProduceExpectedResult()
517+
{
518+
// Given
519+
Result<int> expected = 9;
520+
Result<int> result = 3;
521+
522+
// When
523+
Result<int> actual = result.SelectMany(x => Result<int>.Success(x * x));
524+
525+
// Then
526+
Assert.Equal(expected, actual);
527+
}
528+
529+
[Fact(DisplayName = "Result Failure.SelectMany<TResult> should produce the expected result")]
530+
public void ResultFailureSelectTResultManyShouldProduceExpectedResult()
473531
{
474532
// Given
475533
Exception exception = new("failure");
476534
Result<int> result = Result<int>.Failure(exception);
477535

478536
// When
479-
Result<int> actual = result.SelectMany<int>(value => value * value);
537+
Result<int> actual = result.SelectMany(x => Result<int>.Success(x * x));
480538

481539
// Then
482540
Assert.Equal(Result<int>.Failure(exception), actual);

OnixLabs.Core/OnixLabs.Core.csproj

+1-1
Original file line numberDiff line numberDiff line change
@@ -11,7 +11,7 @@
1111
<NeutralLanguage>en</NeutralLanguage>
1212
<Copyright>Copyright © ONIXLabs 2020</Copyright>
1313
<RepositoryUrl>https://github.com/onix-labs/onixlabs-dotnet</RepositoryUrl>
14-
<PackageVersion>8.7.0</PackageVersion>
14+
<PackageVersion>8.8.0</PackageVersion>
1515
<PackageLicenseUrl></PackageLicenseUrl>
1616
</PropertyGroup>
1717
<PropertyGroup>

OnixLabs.Core/Result.Failure.cs

+42-10
Original file line numberDiff line numberDiff line change
@@ -87,25 +87,41 @@ public sealed class Failure : Result, IValueEquatable<Failure>
8787
/// </returns>
8888
public override TResult Match<TResult>(Func<TResult> success, Func<Exception, TResult> failure) => failure(Exception);
8989

90+
/// <summary>
91+
/// Applies the provided selector action to the value of the current <see cref="Result"/> instance.
92+
/// </summary>
93+
/// <param name="selector">The action to apply to current <see cref="Result"/> instance.</param>
94+
/// <returns>
95+
/// Returns <see cref="Success"/> if the current <see cref="Result"/> is in a successful state, and the action invocation is also successful; otherwise; <see cref="Failure"/>.
96+
/// </returns>
97+
public override Result Select(Action selector) => this;
98+
9099
/// <summary>
91100
/// Applies the provided selector function to the value of the current <see cref="Result"/> instance.
92101
/// </summary>
93-
/// <param name="selector">The function to apply to the value of the current <see cref="Result"/> instance.</param>
102+
/// <param name="selector">The function to apply to the current <see cref="Result"/> instance.</param>
94103
/// <typeparam name="TResult">The underlying type of the result produced by the selector function.</typeparam>
95104
/// <returns>
96-
/// Returns a new <see cref="Result{TResult}"/> instance containing the result of the function if the current
97-
/// <see cref="Result"/> instance is in a successful state; otherwise, returns the current failed <see cref="Result"/> instance.
105+
/// Returns <see cref="Success{T}"/> if the current <see cref="Result"/> is in a successful state, and the action invocation is also successful; otherwise; <see cref="Failure{T}"/>.
98106
/// </returns>
99107
public override Result<TResult> Select<TResult>(Func<TResult> selector) => Result<TResult>.Failure(Exception);
100108

101109
/// <summary>
102110
/// Applies the provided selector function to the value of the current <see cref="Result"/> instance.
103111
/// </summary>
104-
/// <param name="selector">The function to apply to the value of the current <see cref="Result"/> instance.</param>
112+
/// <param name="selector">The action to function to the current <see cref="Result"/> instance.</param>
113+
/// <returns>
114+
/// Returns <see cref="Success"/> if the current <see cref="Result"/> is in a successful state, and the action invocation is also successful; otherwise; <see cref="Failure"/>.
115+
/// </returns>
116+
public override Result SelectMany(Func<Result> selector) => this;
117+
118+
/// <summary>
119+
/// Applies the provided selector function to the value of the current <see cref="Result"/> instance.
120+
/// </summary>
121+
/// <param name="selector">The function to apply to the current <see cref="Result"/> instance.</param>
105122
/// <typeparam name="TResult">The underlying type of the result produced by the selector function.</typeparam>
106123
/// <returns>
107-
/// Returns a new <see cref="Result{TResult}"/> instance containing the result of the function if the current
108-
/// <see cref="Result"/> instance is in a successful state; otherwise, returns the current failed <see cref="Result"/> instance.
124+
/// Returns <see cref="Success{T}"/> if the current <see cref="Result"/> is in a successful state, and the action invocation is also successful; otherwise; <see cref="Failure{T}"/>.
109125
/// </returns>
110126
public override Result<TResult> SelectMany<TResult>(Func<Result<TResult>> selector) => Result<TResult>.Failure(Exception);
111127

@@ -224,25 +240,41 @@ public sealed class Failure<T> : Result<T>, IValueEquatable<Failure<T>>
224240
/// </returns>
225241
public override TResult Match<TResult>(Func<T, TResult> success, Func<Exception, TResult> failure) => failure(Exception);
226242

243+
/// <summary>
244+
/// Applies the provided selector action to the value of the current <see cref="Result{T}"/> instance.
245+
/// </summary>
246+
/// <param name="selector">The action to apply to the value of the current <see cref="Result{T}"/> instance.</param>
247+
/// <returns>
248+
/// Returns <see cref="Success"/> if the current <see cref="Result{T}"/> is in a successful state, and the action invocation is also successful; otherwise; <see cref="Failure"/>.
249+
/// </returns>
250+
public override Result Select(Action<T> selector) => Result.Failure(Exception);
251+
227252
/// <summary>
228253
/// Applies the provided selector function to the value of the current <see cref="Result{T}"/> instance.
229254
/// </summary>
230255
/// <param name="selector">The function to apply to the value of the current <see cref="Result{T}"/> instance.</param>
231256
/// <typeparam name="TResult">The underlying type of the result produced by the selector function.</typeparam>
232257
/// <returns>
233-
/// Returns a new <see cref="Result{TResult}"/> instance containing the result of the function if the current
234-
/// <see cref="Result{T}"/> instance is in a successful state; otherwise, returns the current failed <see cref="Result{T}"/> instance.
258+
/// Returns <see cref="Success{T}"/> if the current <see cref="Result{T}"/> is in a successful state, and the action invocation is also successful; otherwise; <see cref="Failure{T}"/>.
235259
/// </returns>
236260
public override Result<TResult> Select<TResult>(Func<T, TResult> selector) => Result<TResult>.Failure(Exception);
237261

262+
/// <summary>
263+
/// Applies the provided selector function to the value of the current <see cref="Result{T}"/> instance.
264+
/// </summary>
265+
/// <param name="selector">The function to apply to the value of the current <see cref="Result{T}"/> instance.</param>
266+
/// <returns>
267+
/// Returns <see cref="Success"/> if the current <see cref="Result{T}"/> is in a successful state, and the action invocation is also successful; otherwise; <see cref="Failure"/>.
268+
/// </returns>
269+
public override Result SelectMany(Func<T, Result> selector) => Result.Failure(Exception);
270+
238271
/// <summary>
239272
/// Applies the provided selector function to the value of the current <see cref="Result{T}"/> instance.
240273
/// </summary>
241274
/// <param name="selector">The function to apply to the value of the current <see cref="Result{T}"/> instance.</param>
242275
/// <typeparam name="TResult">The underlying type of the result produced by the selector function.</typeparam>
243276
/// <returns>
244-
/// Returns a new <see cref="Result{TResult}"/> instance containing the result of the function if the current
245-
/// <see cref="Result{T}"/> instance is in a successful state; otherwise, returns the current failed <see cref="Result{T}"/> instance.
277+
/// Returns <see cref="Success{T}"/> if the current <see cref="Result{T}"/> is in a successful state, and the action invocation is also successful; otherwise; <see cref="Failure{T}"/>.
246278
/// </returns>
247279
public override Result<TResult> SelectMany<TResult>(Func<T, Result<TResult>> selector) => Result<TResult>.Failure(Exception);
248280

0 commit comments

Comments
 (0)