added Tap extensions generator
This commit is contained in:
@@ -16,6 +16,7 @@ public class ResultMethodGenerator : IIncrementalGenerator
|
||||
new ResultMatchExecutor(),
|
||||
new ResultMapExecutor(),
|
||||
new ResultBindExecutor(),
|
||||
new ResultTapExecutor(),
|
||||
};
|
||||
|
||||
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
|
||||
|
||||
#region Tap
|
||||
#region Tap<>
|
||||
[Pure]
|
||||
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;
|
||||
}
|
||||
|
||||
[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
|
||||
|
||||
#region Merge
|
||||
|
||||
Reference in New Issue
Block a user