You cannot select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
230 lines
8.5 KiB
C#
230 lines
8.5 KiB
C#
using Cysharp.Threading.Tasks.Internal;
|
|
using System;
|
|
using System.Threading;
|
|
|
|
namespace Cysharp.Threading.Tasks.Linq
|
|
{
|
|
public static partial class UniTaskAsyncEnumerable
|
|
{
|
|
public static UniTask<TSource> SingleAsync<TSource>(this IUniTaskAsyncEnumerable<TSource> source, CancellationToken cancellationToken = default)
|
|
{
|
|
Error.ThrowArgumentNullException(source, nameof(source));
|
|
|
|
return SingleOperator.SingleAsync(source, cancellationToken, false);
|
|
}
|
|
|
|
public static UniTask<TSource> SingleAsync<TSource>(this IUniTaskAsyncEnumerable<TSource> source, Func<TSource, Boolean> predicate, CancellationToken cancellationToken = default)
|
|
{
|
|
Error.ThrowArgumentNullException(source, nameof(source));
|
|
Error.ThrowArgumentNullException(predicate, nameof(predicate));
|
|
|
|
return SingleOperator.SingleAsync(source, predicate, cancellationToken, false);
|
|
}
|
|
|
|
public static UniTask<TSource> SingleAwaitAsync<TSource>(this IUniTaskAsyncEnumerable<TSource> source, Func<TSource, UniTask<Boolean>> predicate, CancellationToken cancellationToken = default)
|
|
{
|
|
Error.ThrowArgumentNullException(source, nameof(source));
|
|
Error.ThrowArgumentNullException(predicate, nameof(predicate));
|
|
|
|
return SingleOperator.SingleAwaitAsync(source, predicate, cancellationToken, false);
|
|
}
|
|
|
|
public static UniTask<TSource> SingleAwaitWithCancellationAsync<TSource>(this IUniTaskAsyncEnumerable<TSource> source, Func<TSource, CancellationToken, UniTask<Boolean>> predicate, CancellationToken cancellationToken = default)
|
|
{
|
|
Error.ThrowArgumentNullException(source, nameof(source));
|
|
Error.ThrowArgumentNullException(predicate, nameof(predicate));
|
|
|
|
return SingleOperator.SingleAwaitWithCancellationAsync(source, predicate, cancellationToken, false);
|
|
}
|
|
|
|
public static UniTask<TSource> SingleOrDefaultAsync<TSource>(this IUniTaskAsyncEnumerable<TSource> source, CancellationToken cancellationToken = default)
|
|
{
|
|
Error.ThrowArgumentNullException(source, nameof(source));
|
|
|
|
return SingleOperator.SingleAsync(source, cancellationToken, true);
|
|
}
|
|
|
|
public static UniTask<TSource> SingleOrDefaultAsync<TSource>(this IUniTaskAsyncEnumerable<TSource> source, Func<TSource, Boolean> predicate, CancellationToken cancellationToken = default)
|
|
{
|
|
Error.ThrowArgumentNullException(source, nameof(source));
|
|
Error.ThrowArgumentNullException(predicate, nameof(predicate));
|
|
|
|
return SingleOperator.SingleAsync(source, predicate, cancellationToken, true);
|
|
}
|
|
|
|
public static UniTask<TSource> SingleOrDefaultAwaitAsync<TSource>(this IUniTaskAsyncEnumerable<TSource> source, Func<TSource, UniTask<Boolean>> predicate, CancellationToken cancellationToken = default)
|
|
{
|
|
Error.ThrowArgumentNullException(source, nameof(source));
|
|
Error.ThrowArgumentNullException(predicate, nameof(predicate));
|
|
|
|
return SingleOperator.SingleAwaitAsync(source, predicate, cancellationToken, true);
|
|
}
|
|
|
|
public static UniTask<TSource> SingleOrDefaultAwaitWithCancellationAsync<TSource>(this IUniTaskAsyncEnumerable<TSource> source, Func<TSource, CancellationToken, UniTask<Boolean>> predicate, CancellationToken cancellationToken = default)
|
|
{
|
|
Error.ThrowArgumentNullException(source, nameof(source));
|
|
Error.ThrowArgumentNullException(predicate, nameof(predicate));
|
|
|
|
return SingleOperator.SingleAwaitWithCancellationAsync(source, predicate, cancellationToken, true);
|
|
}
|
|
}
|
|
|
|
internal static class SingleOperator
|
|
{
|
|
public static async UniTask<TSource> SingleAsync<TSource>(IUniTaskAsyncEnumerable<TSource> source, CancellationToken cancellationToken, bool defaultIfEmpty)
|
|
{
|
|
var e = source.GetAsyncEnumerator(cancellationToken);
|
|
try
|
|
{
|
|
if (await e.MoveNextAsync())
|
|
{
|
|
var v = e.Current;
|
|
if (!await e.MoveNextAsync())
|
|
{
|
|
return v;
|
|
}
|
|
|
|
throw Error.MoreThanOneElement();
|
|
}
|
|
else
|
|
{
|
|
if (defaultIfEmpty)
|
|
{
|
|
return default;
|
|
}
|
|
else
|
|
{
|
|
throw Error.NoElements();
|
|
}
|
|
}
|
|
}
|
|
finally
|
|
{
|
|
if (e != null)
|
|
{
|
|
await e.DisposeAsync();
|
|
}
|
|
}
|
|
}
|
|
|
|
public static async UniTask<TSource> SingleAsync<TSource>(IUniTaskAsyncEnumerable<TSource> source, Func<TSource, Boolean> predicate, CancellationToken cancellationToken, bool defaultIfEmpty)
|
|
{
|
|
var e = source.GetAsyncEnumerator(cancellationToken);
|
|
try
|
|
{
|
|
TSource value = default;
|
|
bool found = false;
|
|
while (await e.MoveNextAsync())
|
|
{
|
|
var v = e.Current;
|
|
if (predicate(v))
|
|
{
|
|
if (found)
|
|
{
|
|
throw Error.MoreThanOneElement();
|
|
}
|
|
else
|
|
{
|
|
found = true;
|
|
value = v;
|
|
}
|
|
}
|
|
}
|
|
|
|
if (found || defaultIfEmpty)
|
|
{
|
|
return value;
|
|
}
|
|
|
|
throw Error.NoElements();
|
|
}
|
|
finally
|
|
{
|
|
if (e != null)
|
|
{
|
|
await e.DisposeAsync();
|
|
}
|
|
}
|
|
}
|
|
|
|
public static async UniTask<TSource> SingleAwaitAsync<TSource>(IUniTaskAsyncEnumerable<TSource> source, Func<TSource, UniTask<Boolean>> predicate, CancellationToken cancellationToken, bool defaultIfEmpty)
|
|
{
|
|
var e = source.GetAsyncEnumerator(cancellationToken);
|
|
try
|
|
{
|
|
TSource value = default;
|
|
bool found = false;
|
|
while (await e.MoveNextAsync())
|
|
{
|
|
var v = e.Current;
|
|
if (await predicate(v))
|
|
{
|
|
if (found)
|
|
{
|
|
throw Error.MoreThanOneElement();
|
|
}
|
|
else
|
|
{
|
|
found = true;
|
|
value = v;
|
|
}
|
|
}
|
|
}
|
|
|
|
if (found || defaultIfEmpty)
|
|
{
|
|
return value;
|
|
}
|
|
|
|
throw Error.NoElements();
|
|
}
|
|
finally
|
|
{
|
|
if (e != null)
|
|
{
|
|
await e.DisposeAsync();
|
|
}
|
|
}
|
|
}
|
|
|
|
public static async UniTask<TSource> SingleAwaitWithCancellationAsync<TSource>(IUniTaskAsyncEnumerable<TSource> source, Func<TSource, CancellationToken, UniTask<Boolean>> predicate, CancellationToken cancellationToken, bool defaultIfEmpty)
|
|
{
|
|
var e = source.GetAsyncEnumerator(cancellationToken);
|
|
try
|
|
{
|
|
TSource value = default;
|
|
bool found = false;
|
|
while (await e.MoveNextAsync())
|
|
{
|
|
var v = e.Current;
|
|
if (await predicate(v, cancellationToken))
|
|
{
|
|
if (found)
|
|
{
|
|
throw Error.MoreThanOneElement();
|
|
}
|
|
else
|
|
{
|
|
found = true;
|
|
value = v;
|
|
}
|
|
}
|
|
}
|
|
|
|
if (found || defaultIfEmpty)
|
|
{
|
|
return value;
|
|
}
|
|
|
|
throw Error.NoElements();
|
|
}
|
|
finally
|
|
{
|
|
if (e != null)
|
|
{
|
|
await e.DisposeAsync();
|
|
}
|
|
}
|
|
}
|
|
}
|
|
} |