mirror of
https://github.com/GeWuYou/GFramework.git
synced 2026-03-22 10:34:30 +08:00
- 为Result.Failure方法添加参数验证,确保异常和消息参数不为空 - 重构Result<T>内部状态枚举顺序以优化比较逻辑 - 为Result<T>的Map、Bind和MapAsync方法添加参数验证和异常处理 - 在Result<T>比较逻辑中添加类型不可比较时的安全处理 - 添加ResultExtensions.BindAsync扩展方法支持异步绑定操作 - 重写Combine扩展方法中对失败结果的处理逻辑 - 添加完整的Result相关类型单元测试覆盖所有功能场景 - 为Result类型添加详细的XML文档注释和使用示例
603 lines
19 KiB
C#
603 lines
19 KiB
C#
// Copyright (c) 2025 GeWuYou
|
|
// Licensed under the Apache License, Version 2.0 (the "License");
|
|
// you may not use this file except in compliance with the License.
|
|
// You may obtain a copy of the License at
|
|
//
|
|
// http://www.apache.org/licenses/LICENSE-2.0
|
|
//
|
|
// Unless required by applicable law or agreed to in writing, software
|
|
// distributed under the License is distributed on an "AS IS" BASIS,
|
|
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
// See the License for the specific language governing permissions and
|
|
// limitations under the License.
|
|
|
|
using GFramework.Core.Functional;
|
|
using NUnit.Framework;
|
|
|
|
namespace GFramework.Core.Tests.Functional;
|
|
|
|
/// <summary>
|
|
/// Result<A> 泛型类型测试类
|
|
/// </summary>
|
|
[TestFixture]
|
|
public class ResultTTests
|
|
{
|
|
[Test]
|
|
public void Constructor_WithValue_Should_Create_Success_Result()
|
|
{
|
|
var result = new Result<int>(42);
|
|
Assert.That(result.IsSuccess, Is.True);
|
|
Assert.That(result.IsFaulted, Is.False);
|
|
Assert.That(result.IsBottom, Is.False);
|
|
}
|
|
|
|
[Test]
|
|
public void Constructor_WithException_Should_Create_Faulted_Result()
|
|
{
|
|
var exception = new InvalidOperationException("Test error");
|
|
var result = new Result<int>(exception);
|
|
Assert.That(result.IsFaulted, Is.True);
|
|
Assert.That(result.IsSuccess, Is.False);
|
|
Assert.That(result.Exception, Is.SameAs(exception));
|
|
}
|
|
|
|
[Test]
|
|
public void Constructor_WithNullException_Should_Throw_ArgumentNullException()
|
|
{
|
|
Assert.Throws<ArgumentNullException>(() => new Result<int>(null!));
|
|
}
|
|
|
|
[Test]
|
|
public void DefaultConstructor_Should_Create_Bottom_State()
|
|
{
|
|
var result = new Result<int>();
|
|
Assert.That(result.IsBottom, Is.True);
|
|
Assert.That(result.IsSuccess, Is.False);
|
|
Assert.That(result.IsFaulted, Is.False);
|
|
}
|
|
|
|
[Test]
|
|
public void ImplicitConversion_Should_Create_Success_Result()
|
|
{
|
|
Result<int> result = 42;
|
|
Assert.That(result.IsSuccess, Is.True);
|
|
Assert.That(result.Match(succ: v => v, fail: _ => 0), Is.EqualTo(42));
|
|
}
|
|
|
|
[Test]
|
|
public void IsSuccess_Should_Return_True_When_Result_Is_Successful()
|
|
{
|
|
var result = Result<int>.Succeed(42);
|
|
Assert.That(result.IsSuccess, Is.True);
|
|
}
|
|
|
|
[Test]
|
|
public void IsFaulted_Should_Return_True_When_Result_Is_Faulted()
|
|
{
|
|
var result = Result<int>.Fail(new Exception("Error"));
|
|
Assert.That(result.IsFaulted, Is.True);
|
|
}
|
|
|
|
[Test]
|
|
public void IsBottom_Should_Return_True_When_Result_Is_Bottom()
|
|
{
|
|
var result = Result<int>.Bottom;
|
|
Assert.That(result.IsBottom, Is.True);
|
|
}
|
|
|
|
[Test]
|
|
public void Exception_Should_Return_Exception_When_Faulted()
|
|
{
|
|
var exception = new InvalidOperationException("Test");
|
|
var result = Result<int>.Fail(exception);
|
|
Assert.That(result.Exception, Is.SameAs(exception));
|
|
}
|
|
|
|
[Test]
|
|
public void Exception_Should_Return_InvalidOperationException_When_Bottom()
|
|
{
|
|
var result = Result<int>.Bottom;
|
|
Assert.That(result.Exception, Is.TypeOf<InvalidOperationException>());
|
|
}
|
|
|
|
[Test]
|
|
public void Exception_Should_Return_InvalidOperationException_When_Success()
|
|
{
|
|
var result = Result<int>.Succeed(42);
|
|
Assert.That(result.Exception, Is.TypeOf<InvalidOperationException>());
|
|
}
|
|
|
|
[Test]
|
|
public void Succeed_Should_Create_Successful_Result()
|
|
{
|
|
var result = Result<int>.Succeed(42);
|
|
Assert.That(result.IsSuccess, Is.True);
|
|
Assert.That(result.Match(succ: v => v, fail: _ => 0), Is.EqualTo(42));
|
|
}
|
|
|
|
[Test]
|
|
public void Success_Should_Create_Successful_Result()
|
|
{
|
|
var result = Result<int>.Success(42);
|
|
Assert.That(result.IsSuccess, Is.True);
|
|
Assert.That(result.Match(succ: v => v, fail: _ => 0), Is.EqualTo(42));
|
|
}
|
|
|
|
[Test]
|
|
public void Fail_Should_Create_Faulted_Result()
|
|
{
|
|
var exception = new Exception("Error");
|
|
var result = Result<int>.Fail(exception);
|
|
Assert.That(result.IsFaulted, Is.True);
|
|
Assert.That(result.Exception, Is.SameAs(exception));
|
|
}
|
|
|
|
[Test]
|
|
public void Failure_WithException_Should_Create_Faulted_Result()
|
|
{
|
|
var exception = new Exception("Error");
|
|
var result = Result<int>.Failure(exception);
|
|
Assert.That(result.IsFaulted, Is.True);
|
|
Assert.That(result.Exception, Is.SameAs(exception));
|
|
}
|
|
|
|
[Test]
|
|
public void Failure_WithMessage_Should_Create_Faulted_Result()
|
|
{
|
|
var message = "Test error";
|
|
var result = Result<int>.Failure(message);
|
|
Assert.That(result.IsFaulted, Is.True);
|
|
Assert.That(result.Exception.Message, Is.EqualTo(message));
|
|
}
|
|
|
|
[Test]
|
|
public void Try_Should_Return_Success_When_Function_Succeeds()
|
|
{
|
|
var result = Result<int>.Try(() => 42);
|
|
Assert.That(result.IsSuccess, Is.True);
|
|
Assert.That(result.Match(succ: v => v, fail: _ => 0), Is.EqualTo(42));
|
|
}
|
|
|
|
[Test]
|
|
public void Try_Should_Return_Failure_When_Function_Throws()
|
|
{
|
|
var result = Result<int>.Try(() => throw new InvalidOperationException("Error"));
|
|
Assert.That(result.IsFaulted, Is.True);
|
|
Assert.That(result.Exception, Is.TypeOf<InvalidOperationException>());
|
|
}
|
|
|
|
[Test]
|
|
public void IfFail_WithDefaultValue_Should_Return_Value_When_Success()
|
|
{
|
|
var result = Result<int>.Succeed(42);
|
|
var value = result.IfFail(0);
|
|
Assert.That(value, Is.EqualTo(42));
|
|
}
|
|
|
|
[Test]
|
|
public void IfFail_WithDefaultValue_Should_Return_Default_When_Faulted()
|
|
{
|
|
var result = Result<int>.Fail(new Exception("Error"));
|
|
var value = result.IfFail(99);
|
|
Assert.That(value, Is.EqualTo(99));
|
|
}
|
|
|
|
[Test]
|
|
public void IfFail_WithFunction_Should_Return_Value_When_Success()
|
|
{
|
|
var result = Result<int>.Succeed(42);
|
|
var value = result.IfFail(_ => 0);
|
|
Assert.That(value, Is.EqualTo(42));
|
|
}
|
|
|
|
[Test]
|
|
public void IfFail_WithFunction_Should_Execute_Function_When_Faulted()
|
|
{
|
|
var exception = new Exception("Error");
|
|
var result = Result<int>.Fail(exception);
|
|
var executed = false;
|
|
var value = result.IfFail(ex =>
|
|
{
|
|
executed = true;
|
|
return 99;
|
|
});
|
|
Assert.That(executed, Is.True);
|
|
Assert.That(value, Is.EqualTo(99));
|
|
}
|
|
|
|
[Test]
|
|
public void IfFail_WithAction_Should_Execute_Action_When_Faulted()
|
|
{
|
|
var result = Result<int>.Fail(new Exception("Error"));
|
|
var executed = false;
|
|
result.IfFail(_ => executed = true);
|
|
Assert.That(executed, Is.True);
|
|
}
|
|
|
|
[Test]
|
|
public void IfSucc_Should_Execute_Action_When_Success()
|
|
{
|
|
var result = Result<int>.Succeed(42);
|
|
var executed = false;
|
|
result.IfSucc(_ => executed = true);
|
|
Assert.That(executed, Is.True);
|
|
}
|
|
|
|
[Test]
|
|
public void IfSucc_Should_Not_Execute_Action_When_Faulted()
|
|
{
|
|
var result = Result<int>.Fail(new Exception("Error"));
|
|
var executed = false;
|
|
result.IfSucc(_ => executed = true);
|
|
Assert.That(executed, Is.False);
|
|
}
|
|
|
|
[Test]
|
|
public void Map_Should_Transform_Value_When_Success()
|
|
{
|
|
var result = Result<int>.Succeed(42);
|
|
var mapped = result.Map(x => x.ToString());
|
|
Assert.That(mapped.IsSuccess, Is.True);
|
|
Assert.That(mapped.Match(succ: v => v, fail: _ => ""), Is.EqualTo("42"));
|
|
}
|
|
|
|
[Test]
|
|
public void Map_Should_Propagate_Exception_When_Faulted()
|
|
{
|
|
var exception = new Exception("Error");
|
|
var result = Result<int>.Fail(exception);
|
|
var mapped = result.Map(x => x.ToString());
|
|
Assert.That(mapped.IsFaulted, Is.True);
|
|
Assert.That(mapped.Exception, Is.SameAs(exception));
|
|
}
|
|
|
|
[Test]
|
|
public void Map_Should_Handle_Null_Result_From_Mapper()
|
|
{
|
|
var result = Result<int>.Succeed(42);
|
|
var mapped = result.Map<string>(_ => null!);
|
|
Assert.That(mapped.IsSuccess, Is.True);
|
|
}
|
|
|
|
[Test]
|
|
public void Map_Should_Not_Throw_When_Mapper_Returns_Null()
|
|
{
|
|
var result = Result<string>.Succeed("test");
|
|
Assert.DoesNotThrow(() => result.Map<string?>(_ => null));
|
|
}
|
|
|
|
[Test]
|
|
public void Bind_Should_Chain_Success_Results()
|
|
{
|
|
var result = Result<int>.Succeed(42);
|
|
var bound = result.Bind(x => Result<string>.Succeed(x.ToString()));
|
|
Assert.That(bound.IsSuccess, Is.True);
|
|
Assert.That(bound.Match(succ: v => v, fail: _ => ""), Is.EqualTo("42"));
|
|
}
|
|
|
|
[Test]
|
|
public void Bind_Should_Propagate_First_Failure()
|
|
{
|
|
var exception = new Exception("Error");
|
|
var result = Result<int>.Fail(exception);
|
|
var bound = result.Bind(x => Result<string>.Succeed(x.ToString()));
|
|
Assert.That(bound.IsFaulted, Is.True);
|
|
Assert.That(bound.Exception, Is.SameAs(exception));
|
|
}
|
|
|
|
[Test]
|
|
public void Bind_Should_Propagate_Second_Failure()
|
|
{
|
|
var result = Result<int>.Succeed(42);
|
|
var exception = new Exception("Bind error");
|
|
var bound = result.Bind(x => Result<string>.Fail(exception));
|
|
Assert.That(bound.IsFaulted, Is.True);
|
|
Assert.That(bound.Exception, Is.SameAs(exception));
|
|
}
|
|
|
|
[Test]
|
|
public void Bind_Should_Handle_Complex_Chaining()
|
|
{
|
|
var result = Result<int>.Succeed(10)
|
|
.Bind(x => Result<int>.Succeed(x * 2))
|
|
.Bind(x => Result<int>.Succeed(x + 5));
|
|
Assert.That(result.IsSuccess, Is.True);
|
|
Assert.That(result.Match(succ: v => v, fail: _ => 0), Is.EqualTo(25));
|
|
}
|
|
|
|
[Test]
|
|
public async Task MapAsync_Should_Transform_Value_When_Success()
|
|
{
|
|
var result = Result<int>.Succeed(42);
|
|
var mapped = await result.MapAsync(async x =>
|
|
{
|
|
await Task.Delay(1);
|
|
return x.ToString();
|
|
});
|
|
Assert.That(mapped.IsSuccess, Is.True);
|
|
Assert.That(mapped.Match(succ: v => v, fail: _ => ""), Is.EqualTo("42"));
|
|
}
|
|
|
|
[Test]
|
|
public async Task MapAsync_Should_Propagate_Exception_When_Faulted()
|
|
{
|
|
var exception = new Exception("Error");
|
|
var result = Result<int>.Fail(exception);
|
|
var mapped = await result.MapAsync(async x =>
|
|
{
|
|
await Task.Delay(1);
|
|
return x.ToString();
|
|
});
|
|
Assert.That(mapped.IsFaulted, Is.True);
|
|
Assert.That(mapped.Exception, Is.SameAs(exception));
|
|
}
|
|
|
|
[Test]
|
|
public async Task MapAsync_Should_Handle_Async_Exceptions()
|
|
{
|
|
var result = Result<int>.Succeed(42);
|
|
var mapped = await result.MapAsync<string>(async _ =>
|
|
{
|
|
await Task.Delay(1);
|
|
throw new InvalidOperationException("Async error");
|
|
});
|
|
Assert.That(mapped.IsFaulted, Is.True);
|
|
Assert.That(mapped.Exception, Is.TypeOf<InvalidOperationException>());
|
|
}
|
|
|
|
[Test]
|
|
public void Match_WithFunctions_Should_Execute_Succ_When_Success()
|
|
{
|
|
var result = Result<int>.Succeed(42);
|
|
var value = result.Match(succ: x => x * 2, fail: _ => 0);
|
|
Assert.That(value, Is.EqualTo(84));
|
|
}
|
|
|
|
[Test]
|
|
public void Match_WithFunctions_Should_Execute_Fail_When_Faulted()
|
|
{
|
|
var result = Result<int>.Fail(new Exception("Error"));
|
|
var value = result.Match(succ: x => x, fail: _ => 99);
|
|
Assert.That(value, Is.EqualTo(99));
|
|
}
|
|
|
|
[Test]
|
|
public void Match_WithActions_Should_Execute_Succ_When_Success()
|
|
{
|
|
var result = Result<int>.Succeed(42);
|
|
var executed = false;
|
|
result.Match(succ: _ => executed = true, fail: _ => { });
|
|
Assert.That(executed, Is.True);
|
|
}
|
|
|
|
[Test]
|
|
public void Match_WithActions_Should_Execute_Fail_When_Faulted()
|
|
{
|
|
var result = Result<int>.Fail(new Exception("Error"));
|
|
var executed = false;
|
|
result.Match(succ: _ => { }, fail: _ => executed = true);
|
|
Assert.That(executed, Is.True);
|
|
}
|
|
|
|
[Test]
|
|
public void Match_Should_Handle_Bottom_State_Correctly()
|
|
{
|
|
var result = Result<int>.Bottom;
|
|
var value = result.Match(succ: x => x, fail: _ => 99);
|
|
Assert.That(value, Is.EqualTo(99));
|
|
}
|
|
|
|
[Test]
|
|
public void Equals_Should_Return_True_When_Both_Success_With_Same_Value()
|
|
{
|
|
var result1 = Result<int>.Succeed(42);
|
|
var result2 = Result<int>.Succeed(42);
|
|
Assert.That(result1.Equals(result2), Is.True);
|
|
}
|
|
|
|
[Test]
|
|
public void Equals_Should_Return_False_When_Success_Values_Differ()
|
|
{
|
|
var result1 = Result<int>.Succeed(42);
|
|
var result2 = Result<int>.Succeed(43);
|
|
Assert.That(result1.Equals(result2), Is.False);
|
|
}
|
|
|
|
[Test]
|
|
public void Equals_Should_Return_True_When_Both_Faulted_With_Same_Exception()
|
|
{
|
|
var result1 = Result<int>.Fail(new InvalidOperationException("Error"));
|
|
var result2 = Result<int>.Fail(new InvalidOperationException("Error"));
|
|
Assert.That(result1.Equals(result2), Is.True);
|
|
}
|
|
|
|
[Test]
|
|
public void Equals_Should_Return_False_When_Exception_Types_Differ()
|
|
{
|
|
var result1 = Result<int>.Fail(new InvalidOperationException("Error"));
|
|
var result2 = Result<int>.Fail(new ArgumentException("Error"));
|
|
Assert.That(result1.Equals(result2), Is.False);
|
|
}
|
|
|
|
[Test]
|
|
public void Equals_Should_Return_True_When_Both_Bottom()
|
|
{
|
|
var result1 = Result<int>.Bottom;
|
|
var result2 = new Result<int>();
|
|
Assert.That(result1.Equals(result2), Is.True);
|
|
}
|
|
|
|
[Test]
|
|
public void Equals_Should_Return_False_When_States_Differ()
|
|
{
|
|
var result1 = Result<int>.Succeed(42);
|
|
var result2 = Result<int>.Fail(new Exception("Error"));
|
|
Assert.That(result1.Equals(result2), Is.False);
|
|
}
|
|
|
|
[Test]
|
|
public void GetHashCode_Should_Be_Consistent_For_Equal_Results()
|
|
{
|
|
var result1 = Result<int>.Succeed(42);
|
|
var result2 = Result<int>.Succeed(42);
|
|
Assert.That(result1.GetHashCode(), Is.EqualTo(result2.GetHashCode()));
|
|
}
|
|
|
|
[Test]
|
|
public void OperatorEquals_Should_Work_Correctly()
|
|
{
|
|
var result1 = Result<int>.Succeed(42);
|
|
var result2 = Result<int>.Succeed(42);
|
|
var result3 = Result<int>.Succeed(43);
|
|
Assert.That(result1 == result2, Is.True);
|
|
Assert.That(result1 == result3, Is.False);
|
|
}
|
|
|
|
[Test]
|
|
public void OperatorNotEquals_Should_Work_Correctly()
|
|
{
|
|
var result1 = Result<int>.Succeed(42);
|
|
var result2 = Result<int>.Succeed(43);
|
|
Assert.That(result1 != result2, Is.True);
|
|
}
|
|
|
|
[Test]
|
|
public void CompareTo_Should_Order_Bottom_Before_Faulted()
|
|
{
|
|
var bottom = Result<int>.Bottom;
|
|
var faulted = Result<int>.Fail(new Exception("Error"));
|
|
Assert.That(bottom.CompareTo(faulted), Is.LessThan(0));
|
|
}
|
|
|
|
[Test]
|
|
public void CompareTo_Should_Order_Faulted_Before_Success()
|
|
{
|
|
var faulted = Result<int>.Fail(new Exception("Error"));
|
|
var success = Result<int>.Succeed(42);
|
|
Assert.That(faulted.CompareTo(success), Is.LessThan(0));
|
|
}
|
|
|
|
[Test]
|
|
public void CompareTo_Should_Compare_Success_Values_When_Both_Success()
|
|
{
|
|
var result1 = Result<int>.Succeed(10);
|
|
var result2 = Result<int>.Succeed(20);
|
|
Assert.That(result1.CompareTo(result2), Is.LessThan(0));
|
|
}
|
|
|
|
[Test]
|
|
public void CompareTo_Should_Return_Zero_When_Both_Faulted()
|
|
{
|
|
var result1 = Result<int>.Fail(new Exception("Error1"));
|
|
var result2 = Result<int>.Fail(new Exception("Error2"));
|
|
Assert.That(result1.CompareTo(result2), Is.EqualTo(0));
|
|
}
|
|
|
|
[Test]
|
|
public void CompareTo_Should_Return_Zero_When_Both_Bottom()
|
|
{
|
|
var result1 = Result<int>.Bottom;
|
|
var result2 = new Result<int>();
|
|
Assert.That(result1.CompareTo(result2), Is.EqualTo(0));
|
|
}
|
|
|
|
[Test]
|
|
public void OperatorLessThan_Should_Work_Correctly()
|
|
{
|
|
var result1 = Result<int>.Succeed(10);
|
|
var result2 = Result<int>.Succeed(20);
|
|
Assert.That(result1 < result2, Is.True);
|
|
}
|
|
|
|
[Test]
|
|
public void OperatorLessThanOrEqual_Should_Work_Correctly()
|
|
{
|
|
var result1 = Result<int>.Succeed(10);
|
|
var result2 = Result<int>.Succeed(10);
|
|
Assert.That(result1 <= result2, Is.True);
|
|
}
|
|
|
|
[Test]
|
|
public void OperatorGreaterThan_Should_Work_Correctly()
|
|
{
|
|
var result1 = Result<int>.Succeed(20);
|
|
var result2 = Result<int>.Succeed(10);
|
|
Assert.That(result1 > result2, Is.True);
|
|
}
|
|
|
|
[Test]
|
|
public void OperatorGreaterThanOrEqual_Should_Work_Correctly()
|
|
{
|
|
var result1 = Result<int>.Succeed(20);
|
|
var result2 = Result<int>.Succeed(20);
|
|
Assert.That(result1 >= result2, Is.True);
|
|
}
|
|
|
|
[Test]
|
|
public void CompareTo_Should_Handle_NonComparable_Types_Gracefully()
|
|
{
|
|
var result1 = Result<object>.Succeed(new object());
|
|
var result2 = Result<object>.Succeed(new object());
|
|
Assert.DoesNotThrow(() => result1.CompareTo(result2));
|
|
}
|
|
|
|
[Test]
|
|
public void ToString_Should_Return_Value_String_When_Success()
|
|
{
|
|
var result = Result<int>.Succeed(42);
|
|
Assert.That(result.ToString(), Is.EqualTo("42"));
|
|
}
|
|
|
|
[Test]
|
|
public void ToString_Should_Return_Fail_Message_When_Faulted()
|
|
{
|
|
var result = Result<int>.Fail(new Exception("Test error"));
|
|
Assert.That(result.ToString(), Is.EqualTo("Fail(Test error)"));
|
|
}
|
|
|
|
[Test]
|
|
public void ToString_Should_Return_Bottom_When_Bottom()
|
|
{
|
|
var result = Result<int>.Bottom;
|
|
Assert.That(result.ToString(), Is.EqualTo("(Bottom)"));
|
|
}
|
|
|
|
[Test]
|
|
public void Success_WithNullValue_Should_Create_Valid_Result()
|
|
{
|
|
var result = Result<string?>.Succeed(null);
|
|
Assert.That(result.IsSuccess, Is.True);
|
|
}
|
|
|
|
[Test]
|
|
public void Map_WithNullValue_Should_Handle_Correctly()
|
|
{
|
|
var result = Result<string?>.Succeed(null);
|
|
var mapped = result.Map(x => x?.Length ?? 0);
|
|
Assert.That(mapped.IsSuccess, Is.True);
|
|
Assert.That(mapped.Match(succ: v => v, fail: _ => -1), Is.EqualTo(0));
|
|
}
|
|
|
|
[Test]
|
|
public void Equals_WithNullValues_Should_Work_Correctly()
|
|
{
|
|
var result1 = Result<string?>.Succeed(null);
|
|
var result2 = Result<string?>.Succeed(null);
|
|
Assert.That(result1.Equals(result2), Is.True);
|
|
}
|
|
|
|
[Test]
|
|
public void ImplicitConversion_WithNull_Should_Create_Success_With_Null()
|
|
{
|
|
Result<string?> result = null;
|
|
Assert.That(result.IsSuccess, Is.True);
|
|
}
|
|
|
|
[Test]
|
|
public void Bottom_Should_Be_Readonly_And_Immutable()
|
|
{
|
|
var bottom1 = Result<int>.Bottom;
|
|
var bottom2 = Result<int>.Bottom;
|
|
Assert.That(bottom1.Equals(bottom2), Is.True);
|
|
}
|
|
} |