Skip to content

Commit 179a728

Browse files
Added result extensions for results that contain optional values. (#61)
1 parent cb7e455 commit 179a728

File tree

2 files changed

+244
-0
lines changed

2 files changed

+244
-0
lines changed
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,180 @@
1+
// Copyright 2020 ONIXLabs
2+
//
3+
// Licensed under the Apache License, Version 2.0 (the "License");
4+
// you may not use this file except in compliance with the License.
5+
// You may obtain a copy of the License at
6+
//
7+
// http://www.apache.org/licenses/LICENSE-2.0
8+
//
9+
// Unless required by applicable law or agreed to in writing, software
10+
// distributed under the License is distributed on an "AS IS" BASIS,
11+
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
12+
// See the License for the specific language governing permissions and
13+
// limitations under the License.
14+
15+
using System;
16+
using Xunit;
17+
18+
namespace OnixLabs.Core.UnitTests;
19+
20+
public sealed class ResultExtensionTests
21+
{
22+
[Fact(DisplayName = "Result.GetValueOrNone should return the Optional value when the result is Success and the Optional value is not None")]
23+
public void ResultGetValueOrNoneShouldReturnOptionalValueWhenResultIsSuccessAndOptionalValueIsNotNone()
24+
{
25+
// Given
26+
Optional<int> expectedNumber = 123;
27+
Optional<string> expectedText = "abc";
28+
Result<Optional<int>> numberResult = expectedNumber;
29+
Result<Optional<string>> textResult = expectedText;
30+
31+
// When
32+
Optional<int> actualNumber = numberResult.GetValueOrNone();
33+
Optional<string> actualText = textResult.GetValueOrNone();
34+
35+
// Then
36+
Assert.Equal(expectedNumber, actualNumber);
37+
Assert.Equal(expectedText, actualText);
38+
}
39+
40+
[Fact(DisplayName = "Result.GetValueOrNone should return None value when the result is Success and the Optional value is None")]
41+
public void ResultGetValueOrNoneShouldReturnNoneWhenResultIsSuccessAndOptionalValueIsNotNone()
42+
{
43+
// Given
44+
Optional<int> expectedNumber = Optional<int>.None;
45+
Optional<string> expectedText = Optional<string>.None;
46+
Result<Optional<int>> numberResult = expectedNumber;
47+
Result<Optional<string>> textResult = expectedText;
48+
49+
// When
50+
Optional<int> actualNumber = numberResult.GetValueOrNone();
51+
Optional<string> actualText = textResult.GetValueOrNone();
52+
53+
// Then
54+
Assert.Equal(expectedNumber, actualNumber);
55+
Assert.Equal(expectedText, actualText);
56+
}
57+
58+
[Fact(DisplayName = "Result.GetValueOrNone should return None value when the result is Failure")]
59+
public void ResultGetValueOrNoneShouldReturnNoneWhenResultIsFailure()
60+
{
61+
// Given
62+
Optional<int> expectedNumber = Optional<int>.None;
63+
Optional<string> expectedText = Optional<string>.None;
64+
Result<Optional<int>> numberResult = Result<Optional<int>>.Failure(new Exception("Result has failed."));
65+
Result<Optional<string>> textResult = Result<Optional<string>>.Failure(new Exception("Result has failed."));
66+
67+
// When
68+
Optional<int> actualNumber = numberResult.GetValueOrNone();
69+
Optional<string> actualText = textResult.GetValueOrNone();
70+
71+
// Then
72+
Assert.Equal(expectedNumber, actualNumber);
73+
Assert.Equal(expectedText, actualText);
74+
}
75+
76+
[Fact(DisplayName = "Result.GetOptionalValueOrThrow should return the Optional value when the Result is Success and the Optional value is not None")]
77+
public void ResultGetOptionalValueOrThrowShouldReturnOptionalValueWhenResultIsSuccessAndOptionalValueIsNotNone()
78+
{
79+
// Given
80+
const int expectedNumber = 123;
81+
const string expectedText = "abc";
82+
Result<Optional<int>> numberResult = Optional<int>.Some(expectedNumber);
83+
Result<Optional<string>> textResult = Optional<string>.Some(expectedText);
84+
85+
// When
86+
int actualNumber = numberResult.GetOptionalValueOrThrow();
87+
string actualText = textResult.GetOptionalValueOrThrow();
88+
89+
// Then
90+
Assert.Equal(expectedNumber, actualNumber);
91+
Assert.Equal(expectedText, actualText);
92+
}
93+
94+
[Fact(DisplayName = "Result.GetOptionalValueOrThrow should throw InvalidOperationException when the Result is Success and the Optional value is None")]
95+
public void ResultGetOptionalValueOrThrowShouldThrowInvalidOperationExceptionWhenResultIsSuccessAndOptionalValueIsNone()
96+
{
97+
// Given
98+
Result<Optional<int>> numberResult = Optional<int>.None;
99+
Result<Optional<string>> textResult = Optional<string>.None;
100+
101+
// When
102+
Exception numberException = Assert.Throws<InvalidOperationException>(() => numberResult.GetOptionalValueOrThrow());
103+
Exception textException = Assert.Throws<InvalidOperationException>(() => textResult.GetOptionalValueOrThrow());
104+
105+
// Then
106+
Assert.Equal("Optional value of type System.Int32 is not present.", numberException.Message);
107+
Assert.Equal("Optional value of type System.String is not present.", textException.Message);
108+
}
109+
110+
[Fact(DisplayName = "Result.GetOptionalValueOrThrow should throw InvalidOperationException when the Result is Failure")]
111+
public void ResultGetOptionalValueOrThrowShouldThrowInvalidOperationExceptionWhenResultIsFailure()
112+
{
113+
// Given
114+
Result<Optional<int>> numberResult = Result<Optional<int>>.Failure(new Exception("Result has failed."));
115+
Result<Optional<string>> textResult = Result<Optional<string>>.Failure(new Exception("Result has failed."));
116+
117+
// When
118+
Exception numberException = Assert.Throws<Exception>(() => numberResult.GetOptionalValueOrThrow());
119+
Exception textException = Assert.Throws<Exception>(() => textResult.GetOptionalValueOrThrow());
120+
121+
// Then
122+
Assert.Equal("Result has failed.", numberException.Message);
123+
Assert.Equal("Result has failed.", textException.Message);
124+
}
125+
126+
127+
[Fact(DisplayName = "Result.GetOptionalValueOrDefault should return the Optional value when the Result is Success and the Optional value is not None")]
128+
public void ResultGetOptionalValueOrDefaultShouldReturnOptionalValueWhenResultIsSuccessAndOptionalValueIsNotNone()
129+
{
130+
// Given
131+
const int expectedNumber = 123;
132+
const string expectedText = "abc";
133+
Result<Optional<int>> numberResult = Optional<int>.Some(expectedNumber);
134+
Result<Optional<string>> textResult = Optional<string>.Some(expectedText);
135+
136+
// When
137+
int actualNumber = numberResult.GetOptionalValueOrDefault(456);
138+
string actualText = textResult.GetOptionalValueOrDefault("xyz");
139+
140+
// Then
141+
Assert.Equal(expectedNumber, actualNumber);
142+
Assert.Equal(expectedText, actualText);
143+
}
144+
145+
[Fact(DisplayName = "Result.GetOptionalValueOrDefault should throw return the default value when the Result is Success and the Optional value is None")]
146+
public void ResultGetOptionalValueOrThrowShouldReturnDefaultValueWhenResultIsSuccessAndOptionalValueIsNone()
147+
{
148+
// Given
149+
const int expectedNumber = 456;
150+
const string expectedText = "xyz";
151+
Result<Optional<int>> numberResult = Optional<int>.None;
152+
Result<Optional<string>> textResult = Optional<string>.None;
153+
154+
// When
155+
int actualNumber = numberResult.GetOptionalValueOrDefault(expectedNumber);
156+
string actualText = textResult.GetOptionalValueOrDefault(expectedText);
157+
158+
// Then
159+
Assert.Equal(expectedNumber, actualNumber);
160+
Assert.Equal(expectedText, actualText);
161+
}
162+
163+
[Fact(DisplayName = "Result.GetOptionalValueOrDefault should return default value when the result is Failure")]
164+
public void ResultGetOptionalValueOrDefaultShouldReturnDefaultValueWhenResultIsFailure()
165+
{
166+
// Given
167+
const int expectedNumber = 456;
168+
const string expectedText = "xyz";
169+
Result<Optional<int>> numberResult = Result<Optional<int>>.Failure(new Exception("Result has failed."));
170+
Result<Optional<string>> textResult = Result<Optional<string>>.Failure(new Exception("Result has failed."));
171+
172+
// When
173+
int actualNumber = numberResult.GetOptionalValueOrDefault(expectedNumber);
174+
string actualText = textResult.GetOptionalValueOrDefault(expectedText);
175+
176+
// Then
177+
Assert.Equal(expectedNumber, actualNumber);
178+
Assert.Equal(expectedText, actualText);
179+
}
180+
}

OnixLabs.Core/Extensions.Result.cs

+64
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,64 @@
1+
// Copyright 2020 ONIXLabs
2+
//
3+
// Licensed under the Apache License, Version 2.0 (the "License");
4+
// you may not use this file except in compliance with the License.
5+
// You may obtain a copy of the License at
6+
//
7+
// http://www.apache.org/licenses/LICENSE-2.0
8+
//
9+
// Unless required by applicable law or agreed to in writing, software
10+
// distributed under the License is distributed on an "AS IS" BASIS,
11+
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
12+
// See the License for the specific language governing permissions and
13+
// limitations under the License.
14+
15+
using System.ComponentModel;
16+
17+
namespace OnixLabs.Core;
18+
19+
/// <summary>
20+
/// Provides extension methods for <see cref="Result{T}"/> instances.
21+
/// </summary>
22+
[EditorBrowsable(EditorBrowsableState.Never)]
23+
public static class ResultExtensions
24+
{
25+
/// <summary>
26+
/// Gets the <see cref="Optional{T}"/> value of the current <see cref="Result{T}"/> instance,
27+
/// or <see cref="Optional{T}.None"/> if the result is in a <see cref="Failure{T}"/> state.
28+
/// </summary>
29+
/// <param name="result">The <see cref="Result{T}"/> instance from which to obtain the <see cref="Optional{T}"/> value.</param>
30+
/// <typeparam name="T">The underlying type of the <see cref="Optional{T}"/> value.</typeparam>
31+
/// <returns>
32+
/// Return the <see cref="Optional{T}"/> value of the current <see cref="Result{T}"/> instance,
33+
/// or <see cref="Optional{T}.None"/> if the result is in a <see cref="Failure{T}"/> state.
34+
/// </returns>
35+
public static Optional<T> GetValueOrNone<T>(this Result<Optional<T>> result) where T : notnull =>
36+
result is Success<Optional<T>> { Value.HasValue: true } ? result.GetValueOrThrow() : Optional<T>.None;
37+
38+
/// <summary>
39+
/// Gets the underlying <typeparamref name="T"/> value from the current <see cref="Result{T}"/> of <see cref="Optional{T}"/>,
40+
/// or throws an exception if the result is in a <see cref="Failure{T}"/> state, or the <see cref="Optional{T}"/> is <see cref="Optional{T}.None"/>.
41+
/// </summary>
42+
/// <param name="result">The <see cref="Result{T}"/> instance from which to obtain the <see cref="Optional{T}"/> value.</param>
43+
/// <typeparam name="T">The underlying type of the <see cref="Optional{T}"/> value.</typeparam>
44+
/// <returns>
45+
/// Returns the underlying <typeparamref name="T"/> value from the current <see cref="Result{T}"/> of <see cref="Optional{T}"/>,
46+
/// or throws an exception if the result is in a <see cref="Failure{T}"/> state, or the <see cref="Optional{T}"/> is <see cref="Optional{T}.None"/>.
47+
/// </returns>
48+
public static T GetOptionalValueOrThrow<T>(this Result<Optional<T>> result) where T : notnull =>
49+
result.GetValueOrThrow().GetValueOrThrow();
50+
51+
/// <summary>
52+
/// Gets the underlying <typeparamref name="T"/> value from the current <see cref="Result{T}"/> of <see cref="Optional{T}"/>,
53+
/// or returns the default value if the result is in a <see cref="Failure{T}"/> state, or the <see cref="Optional{T}"/> is <see cref="Optional{T}.None"/>.
54+
/// </summary>
55+
/// <param name="result">The <see cref="Result{T}"/> instance from which to obtain the <see cref="Optional{T}"/> value.</param>
56+
/// <param name="defaultValue">The default value to return if the result is in a <see cref="Failure{T}"/> state, or the <see cref="Optional{T}"/> is <see cref="Optional{T}.None"/>.</param>
57+
/// <typeparam name="T">The underlying type of the <see cref="Optional{T}"/> value.</typeparam>
58+
/// <returns>
59+
/// Returns the underlying <typeparamref name="T"/> value from the current <see cref="Result{T}"/> of <see cref="Optional{T}"/>,
60+
/// or returns the default value if the result is in a <see cref="Failure{T}"/> state, or the <see cref="Optional{T}"/> is <see cref="Optional{T}.None"/>.
61+
/// </returns>
62+
public static T GetOptionalValueOrDefault<T>(this Result<Optional<T>> result, T defaultValue) where T : notnull =>
63+
result is Success<Optional<T>> { Value.HasValue: true } ? result.GetOptionalValueOrThrow() : defaultValue;
64+
}

0 commit comments

Comments
 (0)