added Tap extensions generator
This commit is contained in:
@@ -16,6 +16,7 @@ public class ResultMethodGenerator : IIncrementalGenerator
|
|||||||
new ResultMatchExecutor(),
|
new ResultMatchExecutor(),
|
||||||
new ResultMapExecutor(),
|
new ResultMapExecutor(),
|
||||||
new ResultBindExecutor(),
|
new ResultBindExecutor(),
|
||||||
|
new ResultTapExecutor(),
|
||||||
};
|
};
|
||||||
|
|
||||||
public void Initialize(IncrementalGeneratorInitializationContext context)
|
public void Initialize(IncrementalGeneratorInitializationContext context)
|
||||||
|
|||||||
112
Railway.SourceGenerator/ResultTapExecutor.cs
Normal file
112
Railway.SourceGenerator/ResultTapExecutor.cs
Normal file
@@ -0,0 +1,112 @@
|
|||||||
|
using System.Collections.Immutable;
|
||||||
|
using System.Linq;
|
||||||
|
using System.Text;
|
||||||
|
using Microsoft.CodeAnalysis;
|
||||||
|
|
||||||
|
namespace Just.Railway.SourceGen;
|
||||||
|
|
||||||
|
internal sealed class ResultTapExecutor : ResultExtensionsExecutor
|
||||||
|
{
|
||||||
|
protected override string ExtensionType => "Tap";
|
||||||
|
|
||||||
|
protected override void GenerateMethodsForArgCount(StringBuilder sb, int argCount)
|
||||||
|
{
|
||||||
|
var templateArgNames = Enumerable.Range(1, argCount)
|
||||||
|
.Select(i => $"T{i}")
|
||||||
|
.ToImmutableArray();
|
||||||
|
string separatedTemplateArgs = string.Join(", ", templateArgNames);
|
||||||
|
|
||||||
|
sb.AppendLine($"#region <{separatedTemplateArgs}>");
|
||||||
|
|
||||||
|
string resultValueType = templateArgNames.Length == 1 ? separatedTemplateArgs : $"({separatedTemplateArgs})";
|
||||||
|
string resultValueExpansion = GenerateResultValueExpansion(templateArgNames);
|
||||||
|
|
||||||
|
sb.AppendLine($$"""
|
||||||
|
[PureAttribute]
|
||||||
|
[GeneratedCodeAttribute("{{nameof(ResultTapExecutor)}}", "1.0.0.0")]
|
||||||
|
public static ref readonly Result<{{resultValueType}}> Tap<{{separatedTemplateArgs}}>(this in Result<{{resultValueType}}> result, Action<{{separatedTemplateArgs}}>? onSuccess = null, Action<Error>? onFailure = null)
|
||||||
|
{
|
||||||
|
switch (result.State)
|
||||||
|
{
|
||||||
|
case ResultState.Success:
|
||||||
|
onSuccess?.Invoke({{resultValueExpansion}});
|
||||||
|
break;
|
||||||
|
case ResultState.Error:
|
||||||
|
onFailure?.Invoke(result.Error!);
|
||||||
|
break;
|
||||||
|
|
||||||
|
default: throw new ResultNotInitializedException(nameof(result));
|
||||||
|
}
|
||||||
|
return ref result;
|
||||||
|
}
|
||||||
|
""");
|
||||||
|
|
||||||
|
sb.AppendLine($$"""
|
||||||
|
[PureAttribute]
|
||||||
|
[GeneratedCodeAttribute("{{nameof(ResultTapExecutor)}}", "1.0.0.0")]
|
||||||
|
public static async Task<Result<{{resultValueType}}>> Tap<{{separatedTemplateArgs}}>(this Task<Result<{{resultValueType}}>> resultTask, Action<{{separatedTemplateArgs}}>? onSuccess = null, Action<Error>? onFailure = null)
|
||||||
|
{
|
||||||
|
var result = await resultTask.ConfigureAwait(false);
|
||||||
|
switch (result.State)
|
||||||
|
{
|
||||||
|
case ResultState.Success:
|
||||||
|
onSuccess?.Invoke({{resultValueExpansion}});
|
||||||
|
break;
|
||||||
|
case ResultState.Error:
|
||||||
|
onFailure?.Invoke(result.Error!);
|
||||||
|
break;
|
||||||
|
|
||||||
|
default: throw new ResultNotInitializedException(nameof(resultTask));
|
||||||
|
}
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
""");
|
||||||
|
|
||||||
|
sb.AppendLine($$"""
|
||||||
|
[PureAttribute]
|
||||||
|
[GeneratedCodeAttribute("{{nameof(ResultTapExecutor)}}", "1.0.0.0")]
|
||||||
|
public static async Task<Result<{{resultValueType}}>> Tap<{{separatedTemplateArgs}}>(this Result<{{resultValueType}}> result, Func<{{separatedTemplateArgs}}, Task>? onSuccess = null, Func<Error, Task>? onFailure = null)
|
||||||
|
{
|
||||||
|
switch (result.State)
|
||||||
|
{
|
||||||
|
case ResultState.Success:
|
||||||
|
if (onSuccess is not null)
|
||||||
|
await onSuccess.Invoke({{resultValueExpansion}}).ConfigureAwait(false);
|
||||||
|
break;
|
||||||
|
case ResultState.Error:
|
||||||
|
if (onFailure is not null)
|
||||||
|
await onFailure.Invoke(result.Error!).ConfigureAwait(false);
|
||||||
|
break;
|
||||||
|
|
||||||
|
default: throw new ResultNotInitializedException(nameof(result));
|
||||||
|
}
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
""");
|
||||||
|
|
||||||
|
sb.AppendLine($$"""
|
||||||
|
[PureAttribute]
|
||||||
|
[GeneratedCodeAttribute("{{nameof(ResultTapExecutor)}}", "1.0.0.0")]
|
||||||
|
public static async Task<Result<{{resultValueType}}>> Tap<{{separatedTemplateArgs}}>(this Task<Result<{{resultValueType}}>> resultTask, Func<{{separatedTemplateArgs}}, Task>? onSuccess = null, Func<Error, Task>? onFailure = null)
|
||||||
|
{
|
||||||
|
var result = await resultTask.ConfigureAwait(false);
|
||||||
|
switch (result.State)
|
||||||
|
{
|
||||||
|
case ResultState.Success:
|
||||||
|
if (onSuccess is not null)
|
||||||
|
await onSuccess.Invoke({{resultValueExpansion}}).ConfigureAwait(false);
|
||||||
|
break;
|
||||||
|
case ResultState.Error:
|
||||||
|
if (onFailure is not null)
|
||||||
|
await onFailure.Invoke(result.Error!).ConfigureAwait(false);
|
||||||
|
break;
|
||||||
|
|
||||||
|
default: throw new ResultNotInitializedException(nameof(resultTask));
|
||||||
|
}
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
""");
|
||||||
|
|
||||||
|
sb.AppendLine("#endregion");
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -686,7 +686,7 @@ public static partial class ResultExtensions
|
|||||||
|
|
||||||
#endregion
|
#endregion
|
||||||
|
|
||||||
#region Tap
|
#region Tap<>
|
||||||
[Pure]
|
[Pure]
|
||||||
public static ref readonly Result Tap(this in Result result, Action? onSuccess = null, Action<Error>? onFailure = null)
|
public static ref readonly Result Tap(this in Result result, Action? onSuccess = null, Action<Error>? onFailure = null)
|
||||||
{
|
{
|
||||||
@@ -762,289 +762,6 @@ public static partial class ResultExtensions
|
|||||||
return result;
|
return result;
|
||||||
}
|
}
|
||||||
|
|
||||||
[Pure]
|
|
||||||
public static ref readonly Result<T> Tap<T>(this in Result<T> result, Action<T>? onSuccess = null, Action<Error>? onFailure = null)
|
|
||||||
{
|
|
||||||
switch (result.State)
|
|
||||||
{
|
|
||||||
case ResultState.Success:
|
|
||||||
onSuccess?.Invoke(result.Value!);
|
|
||||||
break;
|
|
||||||
case ResultState.Error:
|
|
||||||
onFailure?.Invoke(result.Error!);
|
|
||||||
break;
|
|
||||||
|
|
||||||
default: throw new ResultNotInitializedException(nameof(result));
|
|
||||||
}
|
|
||||||
return ref result;
|
|
||||||
}
|
|
||||||
[Pure]
|
|
||||||
public static async Task<Result<T>> Tap<T>(this Task<Result<T>> resultTask, Action<T>? onSuccess = null, Action<Error>? onFailure = null)
|
|
||||||
{
|
|
||||||
var result = await resultTask.ConfigureAwait(false);
|
|
||||||
switch (result.State)
|
|
||||||
{
|
|
||||||
case ResultState.Success:
|
|
||||||
onSuccess?.Invoke(result.Value!);
|
|
||||||
break;
|
|
||||||
case ResultState.Error:
|
|
||||||
onFailure?.Invoke(result.Error!);
|
|
||||||
break;
|
|
||||||
|
|
||||||
default: throw new ResultNotInitializedException(nameof(resultTask));
|
|
||||||
}
|
|
||||||
return result;
|
|
||||||
}
|
|
||||||
[Pure]
|
|
||||||
public static async Task<Result<T>> Tap<T>(this Result<T> result, Func<T, Task>? onSuccess = null, Func<Error, Task>? onFailure = null)
|
|
||||||
{
|
|
||||||
switch (result.State)
|
|
||||||
{
|
|
||||||
case ResultState.Success:
|
|
||||||
if (onSuccess is not null)
|
|
||||||
await onSuccess.Invoke(result.Value!).ConfigureAwait(false);
|
|
||||||
break;
|
|
||||||
case ResultState.Error:
|
|
||||||
if (onFailure is not null)
|
|
||||||
await onFailure.Invoke(result.Error!).ConfigureAwait(false);
|
|
||||||
break;
|
|
||||||
|
|
||||||
default: throw new ResultNotInitializedException(nameof(result));
|
|
||||||
}
|
|
||||||
return result;
|
|
||||||
}
|
|
||||||
[Pure]
|
|
||||||
public static async Task<Result<T>> Tap<T>(this Task<Result<T>> resultTask, Func<T, Task>? onSuccess = null, Func<Error, Task>? onFailure = null)
|
|
||||||
{
|
|
||||||
var result = await resultTask.ConfigureAwait(false);
|
|
||||||
switch (result.State)
|
|
||||||
{
|
|
||||||
case ResultState.Success:
|
|
||||||
if (onSuccess is not null)
|
|
||||||
await onSuccess.Invoke(result.Value!).ConfigureAwait(false);
|
|
||||||
break;
|
|
||||||
case ResultState.Error:
|
|
||||||
if (onFailure is not null)
|
|
||||||
await onFailure.Invoke(result.Error!).ConfigureAwait(false);
|
|
||||||
break;
|
|
||||||
|
|
||||||
default: throw new ResultNotInitializedException(nameof(resultTask));
|
|
||||||
}
|
|
||||||
return result;
|
|
||||||
}
|
|
||||||
|
|
||||||
[Pure]
|
|
||||||
public static ref readonly Result<(T1, T2)> Tap<T1, T2>(this in Result<(T1, T2)> result, Action<T1, T2>? onSuccess = null, Action<Error>? onFailure = null)
|
|
||||||
{
|
|
||||||
switch (result.State)
|
|
||||||
{
|
|
||||||
case ResultState.Success:
|
|
||||||
onSuccess?.Invoke(result.Value.Item1, result.Value.Item2);
|
|
||||||
break;
|
|
||||||
case ResultState.Error:
|
|
||||||
onFailure?.Invoke(result.Error!);
|
|
||||||
break;
|
|
||||||
|
|
||||||
default: throw new ResultNotInitializedException(nameof(result));
|
|
||||||
}
|
|
||||||
return ref result;
|
|
||||||
}
|
|
||||||
[Pure]
|
|
||||||
public static async Task<Result<(T1, T2)>> Tap<T1, T2>(this Task<Result<(T1, T2)>> resultTask, Action<T1, T2>? onSuccess = null, Action<Error>? onFailure = null)
|
|
||||||
{
|
|
||||||
var result = await resultTask.ConfigureAwait(false);
|
|
||||||
switch (result.State)
|
|
||||||
{
|
|
||||||
case ResultState.Success:
|
|
||||||
onSuccess?.Invoke(result.Value.Item1, result.Value.Item2);
|
|
||||||
break;
|
|
||||||
case ResultState.Error:
|
|
||||||
onFailure?.Invoke(result.Error!);
|
|
||||||
break;
|
|
||||||
|
|
||||||
default: throw new ResultNotInitializedException(nameof(resultTask));
|
|
||||||
}
|
|
||||||
return result;
|
|
||||||
}
|
|
||||||
[Pure]
|
|
||||||
public static async Task<Result<(T1, T2)>> Tap<T1, T2>(this Result<(T1, T2)> result, Func<T1, T2, Task>? onSuccess = null, Func<Error, Task>? onFailure = null)
|
|
||||||
{
|
|
||||||
switch (result.State)
|
|
||||||
{
|
|
||||||
case ResultState.Success:
|
|
||||||
if (onSuccess is not null)
|
|
||||||
await onSuccess.Invoke(result.Value.Item1, result.Value.Item2).ConfigureAwait(false);
|
|
||||||
break;
|
|
||||||
case ResultState.Error:
|
|
||||||
if (onFailure is not null)
|
|
||||||
await onFailure.Invoke(result.Error!).ConfigureAwait(false);
|
|
||||||
break;
|
|
||||||
|
|
||||||
default: throw new ResultNotInitializedException(nameof(result));
|
|
||||||
}
|
|
||||||
return result;
|
|
||||||
}
|
|
||||||
[Pure]
|
|
||||||
public static async Task<Result<(T1, T2)>> Tap<T1, T2>(this Task<Result<(T1, T2)>> resultTask, Func<T1, T2, Task>? onSuccess = null, Func<Error, Task>? onFailure = null)
|
|
||||||
{
|
|
||||||
var result = await resultTask.ConfigureAwait(false);
|
|
||||||
switch (result.State)
|
|
||||||
{
|
|
||||||
case ResultState.Success:
|
|
||||||
if (onSuccess is not null)
|
|
||||||
await onSuccess.Invoke(result.Value.Item1, result.Value.Item2).ConfigureAwait(false);
|
|
||||||
break;
|
|
||||||
case ResultState.Error:
|
|
||||||
if (onFailure is not null)
|
|
||||||
await onFailure.Invoke(result.Error!).ConfigureAwait(false);
|
|
||||||
break;
|
|
||||||
|
|
||||||
default: throw new ResultNotInitializedException(nameof(resultTask));
|
|
||||||
}
|
|
||||||
return result;
|
|
||||||
}
|
|
||||||
|
|
||||||
[Pure]
|
|
||||||
public static ref readonly Result<(T1, T2, T3)> Tap<T1, T2, T3>(this in Result<(T1, T2, T3)> result, Action<T1, T2, T3>? onSuccess = null, Action<Error>? onFailure = null)
|
|
||||||
{
|
|
||||||
switch (result.State)
|
|
||||||
{
|
|
||||||
case ResultState.Success:
|
|
||||||
onSuccess?.Invoke(result.Value.Item1, result.Value.Item2, result.Value.Item3);
|
|
||||||
break;
|
|
||||||
case ResultState.Error:
|
|
||||||
onFailure?.Invoke(result.Error!);
|
|
||||||
break;
|
|
||||||
|
|
||||||
default: throw new ResultNotInitializedException(nameof(result));
|
|
||||||
}
|
|
||||||
return ref result;
|
|
||||||
}
|
|
||||||
[Pure]
|
|
||||||
public static async Task<Result<(T1, T2, T3)>> Tap<T1, T2, T3>(this Task<Result<(T1, T2, T3)>> resultTask, Action<T1, T2, T3>? onSuccess = null, Action<Error>? onFailure = null)
|
|
||||||
{
|
|
||||||
var result = await resultTask.ConfigureAwait(false);
|
|
||||||
switch (result.State)
|
|
||||||
{
|
|
||||||
case ResultState.Success:
|
|
||||||
onSuccess?.Invoke(result.Value.Item1, result.Value.Item2, result.Value.Item3);
|
|
||||||
break;
|
|
||||||
case ResultState.Error:
|
|
||||||
onFailure?.Invoke(result.Error!);
|
|
||||||
break;
|
|
||||||
|
|
||||||
default: throw new ResultNotInitializedException(nameof(resultTask));
|
|
||||||
}
|
|
||||||
return result;
|
|
||||||
}
|
|
||||||
[Pure]
|
|
||||||
public static async Task<Result<(T1, T2, T3)>> Tap<T1, T2, T3>(this Result<(T1, T2, T3)> result, Func<T1, T2, T3, Task>? onSuccess = null, Func<Error, Task>? onFailure = null)
|
|
||||||
{
|
|
||||||
switch (result.State)
|
|
||||||
{
|
|
||||||
case ResultState.Success:
|
|
||||||
if (onSuccess is not null)
|
|
||||||
await onSuccess.Invoke(result.Value.Item1, result.Value.Item2, result.Value.Item3).ConfigureAwait(false);
|
|
||||||
break;
|
|
||||||
case ResultState.Error:
|
|
||||||
if (onFailure is not null)
|
|
||||||
await onFailure.Invoke(result.Error!).ConfigureAwait(false);
|
|
||||||
break;
|
|
||||||
|
|
||||||
default: throw new ResultNotInitializedException(nameof(result));
|
|
||||||
}
|
|
||||||
return result;
|
|
||||||
}
|
|
||||||
[Pure]
|
|
||||||
public static async Task<Result<(T1, T2, T3)>> Tap<T1, T2, T3>(this Task<Result<(T1, T2, T3)>> resultTask, Func<T1, T2, T3, Task>? onSuccess = null, Func<Error, Task>? onFailure = null)
|
|
||||||
{
|
|
||||||
var result = await resultTask.ConfigureAwait(false);
|
|
||||||
switch (result.State)
|
|
||||||
{
|
|
||||||
case ResultState.Success:
|
|
||||||
if (onSuccess is not null)
|
|
||||||
await onSuccess.Invoke(result.Value.Item1, result.Value.Item2, result.Value.Item3).ConfigureAwait(false);
|
|
||||||
break;
|
|
||||||
case ResultState.Error:
|
|
||||||
if (onFailure is not null)
|
|
||||||
await onFailure.Invoke(result.Error!).ConfigureAwait(false);
|
|
||||||
break;
|
|
||||||
|
|
||||||
default: throw new ResultNotInitializedException(nameof(resultTask));
|
|
||||||
}
|
|
||||||
return result;
|
|
||||||
}
|
|
||||||
|
|
||||||
[Pure]
|
|
||||||
public static ref readonly Result<(T1, T2, T3, T4)> Tap<T1, T2, T3, T4>(this in Result<(T1, T2, T3, T4)> result, Action<T1, T2, T3, T4>? onSuccess = null, Action<Error>? onFailure = null)
|
|
||||||
{
|
|
||||||
switch (result.State)
|
|
||||||
{
|
|
||||||
case ResultState.Success:
|
|
||||||
onSuccess?.Invoke(result.Value.Item1, result.Value.Item2, result.Value.Item3, result.Value.Item4);
|
|
||||||
break;
|
|
||||||
case ResultState.Error:
|
|
||||||
onFailure?.Invoke(result.Error!);
|
|
||||||
break;
|
|
||||||
|
|
||||||
default: throw new ResultNotInitializedException(nameof(result));
|
|
||||||
}
|
|
||||||
return ref result;
|
|
||||||
}
|
|
||||||
[Pure]
|
|
||||||
public static async Task<Result<(T1, T2, T3, T4)>> Tap<T1, T2, T3, T4>(this Task<Result<(T1, T2, T3, T4)>> resultTask, Action<T1, T2, T3, T4>? onSuccess = null, Action<Error>? onFailure = null)
|
|
||||||
{
|
|
||||||
var result = await resultTask.ConfigureAwait(false);
|
|
||||||
switch (result.State)
|
|
||||||
{
|
|
||||||
case ResultState.Success:
|
|
||||||
onSuccess?.Invoke(result.Value.Item1, result.Value.Item2, result.Value.Item3, result.Value.Item4);
|
|
||||||
break;
|
|
||||||
case ResultState.Error:
|
|
||||||
onFailure?.Invoke(result.Error!);
|
|
||||||
break;
|
|
||||||
|
|
||||||
default: throw new ResultNotInitializedException(nameof(resultTask));
|
|
||||||
}
|
|
||||||
return result;
|
|
||||||
}
|
|
||||||
[Pure]
|
|
||||||
public static async Task<Result<(T1, T2, T3, T4)>> Tap<T1, T2, T3, T4>(this Result<(T1, T2, T3, T4)> result, Func<T1, T2, T3, T4, Task>? onSuccess = null, Func<Error, Task>? onFailure = null)
|
|
||||||
{
|
|
||||||
switch (result.State)
|
|
||||||
{
|
|
||||||
case ResultState.Success:
|
|
||||||
if (onSuccess is not null)
|
|
||||||
await onSuccess.Invoke(result.Value.Item1, result.Value.Item2, result.Value.Item3, result.Value.Item4).ConfigureAwait(false);
|
|
||||||
break;
|
|
||||||
case ResultState.Error:
|
|
||||||
if (onFailure is not null)
|
|
||||||
await onFailure.Invoke(result.Error!).ConfigureAwait(false);
|
|
||||||
break;
|
|
||||||
|
|
||||||
default: throw new ResultNotInitializedException(nameof(result));
|
|
||||||
}
|
|
||||||
return result;
|
|
||||||
}
|
|
||||||
[Pure]
|
|
||||||
public static async Task<Result<(T1, T2, T3, T4)>> Tap<T1, T2, T3, T4>(this Task<Result<(T1, T2, T3, T4)>> resultTask, Func<T1, T2, T3, T4, Task>? onSuccess = null, Func<Error, Task>? onFailure = null)
|
|
||||||
{
|
|
||||||
var result = await resultTask.ConfigureAwait(false);
|
|
||||||
switch (result.State)
|
|
||||||
{
|
|
||||||
case ResultState.Success:
|
|
||||||
if (onSuccess is not null)
|
|
||||||
await onSuccess.Invoke(result.Value.Item1, result.Value.Item2, result.Value.Item3, result.Value.Item4).ConfigureAwait(false);
|
|
||||||
break;
|
|
||||||
case ResultState.Error:
|
|
||||||
if (onFailure is not null)
|
|
||||||
await onFailure.Invoke(result.Error!).ConfigureAwait(false);
|
|
||||||
break;
|
|
||||||
|
|
||||||
default: throw new ResultNotInitializedException(nameof(resultTask));
|
|
||||||
}
|
|
||||||
return result;
|
|
||||||
}
|
|
||||||
#endregion
|
#endregion
|
||||||
|
|
||||||
#region Merge
|
#region Merge
|
||||||
|
|||||||
Reference in New Issue
Block a user