-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathResult.cs
116 lines (91 loc) · 3.86 KB
/
Result.cs
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
namespace ResultPattern;
/// <summary>
/// Base Result class that represents the outcome of an operation without a value.
/// </summary>
public class Result
{
public bool IsSuccess { get; }
public bool IsFailure => !IsSuccess;
public string? Error { get; }
// Protected constructor for base class
protected Result(bool isSuccess, string? error)
{
if (isSuccess && error != null)
throw new InvalidOperationException("A success result cannot have an error");
if (!isSuccess && string.IsNullOrWhiteSpace(error))
throw new InvalidOperationException("A failure result must have an error");
IsSuccess = isSuccess;
Error = error;
}
// Factory methods for creating Results
public static Result Success() => new(true, null);
public static Result Failure(string error) => new(false, error);
// Factory methods for creating generic Results
public static Result<T> Success<T>(T value) => Result<T>.Success(value);
public static Result<T> Failure<T>(string error) => Result<T>.Failure(error);
public void Match(Action onSuccess, Action<string> onFailure)
{
if (IsSuccess)
onSuccess();
else
onFailure(Error!);
}
public TResult Match<TResult>(Func<TResult> onSuccess, Func<string, TResult> onFailure)
=> IsSuccess ? onSuccess() : onFailure(Error!);
}
/// <summary>
/// Generic Result class that inherits from base Result and adds a value.
/// </summary>
public class Result<T> : Result
{
private readonly T? _value;
public T? Value => IsSuccess ? _value : default;
private Result(bool isSuccess, T? value, string? error)
: base(isSuccess, error)
{
if (isSuccess && value == null)
throw new InvalidOperationException("A success result must have a value");
_value = value;
}
public new static Result<T> Success(T value) => new(true, value, null);
public new static Result<T> Failure(string error) => new(false, default, error);
public Result<TNew> Map<TNew>(Func<T, TNew> mapper)
=> IsSuccess ? Result<TNew>.Success(mapper(Value!)) : Result<TNew>.Failure(Error!);
public async Task<Result<TNew>> MapAsync<TNew>(Func<T, Task<TNew>> mapper)
=> IsSuccess ? Result<TNew>.Success(await mapper(Value!)) : Result<TNew>.Failure(Error!);
public Result<TNew> Bind<TNew>(Func<T, Result<TNew>> binder)
=> IsSuccess ? binder(Value!) : Result<TNew>.Failure(Error!);
public async Task<Result<TNew>> BindAsync<TNew>(Func<T, Task<Result<TNew>>> binder)
=> IsSuccess ? await binder(Value!) : Result<TNew>.Failure(Error!);
public T Unwrap() => IsSuccess ? Value! : throw new InvalidOperationException($"Cannot unwrap failure result: {Error}");
public T UnwrapOr(T defaultValue) => IsSuccess ? Value! : defaultValue;
public new void Match(Action<T> onSuccess, Action<string> onFailure)
{
if (IsSuccess)
onSuccess(Value!);
else
onFailure(Error!);
}
public new TResult Match<TResult>(Func<T, TResult> onSuccess, Func<string, TResult> onFailure)
=> IsSuccess ? onSuccess(Value!) : onFailure(Error!);
}
// Extension methods
public static class ResultExtensions
{
public static Result<T> ToResult<T>(this T value)
=> Result<T>.Success(value);
public static Result<T> ToFailure<T>(this string error)
=> Result<T>.Failure(error);
// Combine multiple results
public static Result CombineAll(this IEnumerable<Result> results)
{
var failureResults = results.Where(r => r.IsFailure).ToList();
if (failureResults.Any())
{
var errorMessage = string.Join(Environment.NewLine,
failureResults.Select(r => r.Error));
return Result.Failure(errorMessage);
}
return Result.Success();
}
}