using Cysharp.Threading.Tasks.Internal; using System; using System.Threading; namespace Cysharp.Threading.Tasks.Linq { public static partial class UniTaskAsyncEnumerable { public static IUniTaskAsyncEnumerable Where(this IUniTaskAsyncEnumerable source, Func predicate) { Error.ThrowArgumentNullException(source, nameof(source)); Error.ThrowArgumentNullException(predicate, nameof(predicate)); return new Where(source, predicate); } public static IUniTaskAsyncEnumerable Where(this IUniTaskAsyncEnumerable source, Func predicate) { Error.ThrowArgumentNullException(source, nameof(source)); Error.ThrowArgumentNullException(predicate, nameof(predicate)); return new WhereInt(source, predicate); } public static IUniTaskAsyncEnumerable WhereAwait(this IUniTaskAsyncEnumerable source, Func> predicate) { Error.ThrowArgumentNullException(source, nameof(source)); Error.ThrowArgumentNullException(predicate, nameof(predicate)); return new WhereAwait(source, predicate); } public static IUniTaskAsyncEnumerable WhereAwait(this IUniTaskAsyncEnumerable source, Func> predicate) { Error.ThrowArgumentNullException(source, nameof(source)); Error.ThrowArgumentNullException(predicate, nameof(predicate)); return new WhereIntAwait(source, predicate); } public static IUniTaskAsyncEnumerable WhereAwaitWithCancellation(this IUniTaskAsyncEnumerable source, Func> predicate) { Error.ThrowArgumentNullException(source, nameof(source)); Error.ThrowArgumentNullException(predicate, nameof(predicate)); return new WhereAwaitWithCancellation(source, predicate); } public static IUniTaskAsyncEnumerable WhereAwaitWithCancellation(this IUniTaskAsyncEnumerable source, Func> predicate) { Error.ThrowArgumentNullException(source, nameof(source)); Error.ThrowArgumentNullException(predicate, nameof(predicate)); return new WhereIntAwaitWithCancellation(source, predicate); } } internal sealed class Where : IUniTaskAsyncEnumerable { readonly IUniTaskAsyncEnumerable source; readonly Func predicate; public Where(IUniTaskAsyncEnumerable source, Func predicate) { this.source = source; this.predicate = predicate; } public IUniTaskAsyncEnumerator GetAsyncEnumerator(CancellationToken cancellationToken = default) { return new _Where(source, predicate, cancellationToken); } sealed class _Where : MoveNextSource, IUniTaskAsyncEnumerator { readonly IUniTaskAsyncEnumerable source; readonly Func predicate; readonly CancellationToken cancellationToken; int state = -1; IUniTaskAsyncEnumerator enumerator; UniTask.Awaiter awaiter; Action moveNextAction; public _Where(IUniTaskAsyncEnumerable source, Func predicate, CancellationToken cancellationToken) { this.source = source; this.predicate = predicate; this.cancellationToken = cancellationToken; this.moveNextAction = MoveNext; TaskTracker.TrackActiveTask(this, 3); } public TSource Current { get; private set; } public UniTask MoveNextAsync() { if (state == -2) return default; completionSource.Reset(); MoveNext(); return new UniTask(this, completionSource.Version); } void MoveNext() { REPEAT: try { switch (state) { case -1: // init enumerator = source.GetAsyncEnumerator(cancellationToken); goto case 0; case 0: awaiter = enumerator.MoveNextAsync().GetAwaiter(); if (awaiter.IsCompleted) { goto case 1; } else { state = 1; awaiter.UnsafeOnCompleted(moveNextAction); return; } case 1: if (awaiter.GetResult()) { Current = enumerator.Current; if (predicate(Current)) { goto CONTINUE; } else { state = 0; goto REPEAT; } } else { goto DONE; } default: goto DONE; } } catch (Exception ex) { state = -2; completionSource.TrySetException(ex); return; } DONE: state = -2; completionSource.TrySetResult(false); return; CONTINUE: state = 0; completionSource.TrySetResult(true); return; } public UniTask DisposeAsync() { TaskTracker.RemoveTracking(this); return enumerator.DisposeAsync(); } } } internal sealed class WhereInt : IUniTaskAsyncEnumerable { readonly IUniTaskAsyncEnumerable source; readonly Func predicate; public WhereInt(IUniTaskAsyncEnumerable source, Func predicate) { this.source = source; this.predicate = predicate; } public IUniTaskAsyncEnumerator GetAsyncEnumerator(CancellationToken cancellationToken = default) { return new _Where(source, predicate, cancellationToken); } sealed class _Where : MoveNextSource, IUniTaskAsyncEnumerator { readonly IUniTaskAsyncEnumerable source; readonly Func predicate; readonly CancellationToken cancellationToken; int state = -1; IUniTaskAsyncEnumerator enumerator; UniTask.Awaiter awaiter; Action moveNextAction; int index; public _Where(IUniTaskAsyncEnumerable source, Func predicate, CancellationToken cancellationToken) { this.source = source; this.predicate = predicate; this.cancellationToken = cancellationToken; this.moveNextAction = MoveNext; TaskTracker.TrackActiveTask(this, 3); } public TSource Current { get; private set; } public UniTask MoveNextAsync() { if (state == -2) return default; completionSource.Reset(); MoveNext(); return new UniTask(this, completionSource.Version); } void MoveNext() { REPEAT: try { switch (state) { case -1: // init enumerator = source.GetAsyncEnumerator(cancellationToken); goto case 0; case 0: awaiter = enumerator.MoveNextAsync().GetAwaiter(); if (awaiter.IsCompleted) { goto case 1; } else { state = 1; awaiter.UnsafeOnCompleted(moveNextAction); return; } case 1: if (awaiter.GetResult()) { Current = enumerator.Current; if (predicate(Current, checked(index++))) { goto CONTINUE; } else { state = 0; goto REPEAT; } } else { goto DONE; } default: goto DONE; } } catch (Exception ex) { state = -2; completionSource.TrySetException(ex); return; } DONE: state = -2; completionSource.TrySetResult(false); return; CONTINUE: state = 0; completionSource.TrySetResult(true); return; } public UniTask DisposeAsync() { TaskTracker.RemoveTracking(this); return enumerator.DisposeAsync(); } } } internal sealed class WhereAwait : IUniTaskAsyncEnumerable { readonly IUniTaskAsyncEnumerable source; readonly Func> predicate; public WhereAwait(IUniTaskAsyncEnumerable source, Func> predicate) { this.source = source; this.predicate = predicate; } public IUniTaskAsyncEnumerator GetAsyncEnumerator(CancellationToken cancellationToken = default) { return new _WhereAwait(source, predicate, cancellationToken); } sealed class _WhereAwait : MoveNextSource, IUniTaskAsyncEnumerator { readonly IUniTaskAsyncEnumerable source; readonly Func> predicate; readonly CancellationToken cancellationToken; int state = -1; IUniTaskAsyncEnumerator enumerator; UniTask.Awaiter awaiter; UniTask.Awaiter awaiter2; Action moveNextAction; public _WhereAwait(IUniTaskAsyncEnumerable source, Func> predicate, CancellationToken cancellationToken) { this.source = source; this.predicate = predicate; this.cancellationToken = cancellationToken; this.moveNextAction = MoveNext; TaskTracker.TrackActiveTask(this, 3); } public TSource Current { get; private set; } public UniTask MoveNextAsync() { if (state == -2) return default; completionSource.Reset(); MoveNext(); return new UniTask(this, completionSource.Version); } void MoveNext() { REPEAT: try { switch (state) { case -1: // init enumerator = source.GetAsyncEnumerator(cancellationToken); goto case 0; case 0: awaiter = enumerator.MoveNextAsync().GetAwaiter(); if (awaiter.IsCompleted) { goto case 1; } else { state = 1; awaiter.UnsafeOnCompleted(moveNextAction); return; } case 1: if (awaiter.GetResult()) { Current = enumerator.Current; awaiter2 = predicate(Current).GetAwaiter(); if (awaiter2.IsCompleted) { goto case 2; } else { state = 2; awaiter2.UnsafeOnCompleted(moveNextAction); return; } } else { goto DONE; } case 2: if (awaiter2.GetResult()) { goto CONTINUE; } else { state = 0; goto REPEAT; } default: goto DONE; } } catch (Exception ex) { state = -2; completionSource.TrySetException(ex); return; } DONE: state = -2; completionSource.TrySetResult(false); return; CONTINUE: state = 0; completionSource.TrySetResult(true); return; } public UniTask DisposeAsync() { TaskTracker.RemoveTracking(this); return enumerator.DisposeAsync(); } } } internal sealed class WhereIntAwait : IUniTaskAsyncEnumerable { readonly IUniTaskAsyncEnumerable source; readonly Func> predicate; public WhereIntAwait(IUniTaskAsyncEnumerable source, Func> predicate) { this.source = source; this.predicate = predicate; } public IUniTaskAsyncEnumerator GetAsyncEnumerator(CancellationToken cancellationToken = default) { return new _WhereAwait(source, predicate, cancellationToken); } sealed class _WhereAwait : MoveNextSource, IUniTaskAsyncEnumerator { readonly IUniTaskAsyncEnumerable source; readonly Func> predicate; readonly CancellationToken cancellationToken; int state = -1; IUniTaskAsyncEnumerator enumerator; UniTask.Awaiter awaiter; UniTask.Awaiter awaiter2; Action moveNextAction; int index; public _WhereAwait(IUniTaskAsyncEnumerable source, Func> predicate, CancellationToken cancellationToken) { this.source = source; this.predicate = predicate; this.cancellationToken = cancellationToken; this.moveNextAction = MoveNext; TaskTracker.TrackActiveTask(this, 3); } public TSource Current { get; private set; } public UniTask MoveNextAsync() { if (state == -2) return default; completionSource.Reset(); MoveNext(); return new UniTask(this, completionSource.Version); } void MoveNext() { REPEAT: try { switch (state) { case -1: // init enumerator = source.GetAsyncEnumerator(cancellationToken); goto case 0; case 0: awaiter = enumerator.MoveNextAsync().GetAwaiter(); if (awaiter.IsCompleted) { goto case 1; } else { state = 1; awaiter.UnsafeOnCompleted(moveNextAction); return; } case 1: if (awaiter.GetResult()) { Current = enumerator.Current; awaiter2 = predicate(Current, checked(index++)).GetAwaiter(); if (awaiter2.IsCompleted) { goto case 2; } else { state = 2; awaiter2.UnsafeOnCompleted(moveNextAction); return; } } else { goto DONE; } case 2: if (awaiter2.GetResult()) { goto CONTINUE; } else { state = 0; goto REPEAT; } default: goto DONE; } } catch (Exception ex) { state = -2; completionSource.TrySetException(ex); return; } DONE: state = -2; completionSource.TrySetResult(false); return; CONTINUE: state = 0; completionSource.TrySetResult(true); return; } public UniTask DisposeAsync() { TaskTracker.RemoveTracking(this); return enumerator.DisposeAsync(); } } } internal sealed class WhereAwaitWithCancellation : IUniTaskAsyncEnumerable { readonly IUniTaskAsyncEnumerable source; readonly Func> predicate; public WhereAwaitWithCancellation(IUniTaskAsyncEnumerable source, Func> predicate) { this.source = source; this.predicate = predicate; } public IUniTaskAsyncEnumerator GetAsyncEnumerator(CancellationToken cancellationToken = default) { return new _WhereAwaitWithCancellation(source, predicate, cancellationToken); } sealed class _WhereAwaitWithCancellation : MoveNextSource, IUniTaskAsyncEnumerator { readonly IUniTaskAsyncEnumerable source; readonly Func> predicate; readonly CancellationToken cancellationToken; int state = -1; IUniTaskAsyncEnumerator enumerator; UniTask.Awaiter awaiter; UniTask.Awaiter awaiter2; Action moveNextAction; public _WhereAwaitWithCancellation(IUniTaskAsyncEnumerable source, Func> predicate, CancellationToken cancellationToken) { this.source = source; this.predicate = predicate; this.cancellationToken = cancellationToken; this.moveNextAction = MoveNext; TaskTracker.TrackActiveTask(this, 3); } public TSource Current { get; private set; } public UniTask MoveNextAsync() { if (state == -2) return default; completionSource.Reset(); MoveNext(); return new UniTask(this, completionSource.Version); } void MoveNext() { REPEAT: try { switch (state) { case -1: // init enumerator = source.GetAsyncEnumerator(cancellationToken); goto case 0; case 0: awaiter = enumerator.MoveNextAsync().GetAwaiter(); if (awaiter.IsCompleted) { goto case 1; } else { state = 1; awaiter.UnsafeOnCompleted(moveNextAction); return; } case 1: if (awaiter.GetResult()) { Current = enumerator.Current; awaiter2 = predicate(Current, cancellationToken).GetAwaiter(); if (awaiter2.IsCompleted) { goto case 2; } else { state = 2; awaiter2.UnsafeOnCompleted(moveNextAction); return; } } else { goto DONE; } case 2: if (awaiter2.GetResult()) { goto CONTINUE; } else { state = 0; goto REPEAT; } default: goto DONE; } } catch (Exception ex) { state = -2; completionSource.TrySetException(ex); return; } DONE: state = -2; completionSource.TrySetResult(false); return; CONTINUE: state = 0; completionSource.TrySetResult(true); return; } public UniTask DisposeAsync() { TaskTracker.RemoveTracking(this); return enumerator.DisposeAsync(); } } } internal sealed class WhereIntAwaitWithCancellation : IUniTaskAsyncEnumerable { readonly IUniTaskAsyncEnumerable source; readonly Func> predicate; public WhereIntAwaitWithCancellation(IUniTaskAsyncEnumerable source, Func> predicate) { this.source = source; this.predicate = predicate; } public IUniTaskAsyncEnumerator GetAsyncEnumerator(CancellationToken cancellationToken = default) { return new _WhereAwaitWithCancellation(source, predicate, cancellationToken); } sealed class _WhereAwaitWithCancellation : MoveNextSource, IUniTaskAsyncEnumerator { readonly IUniTaskAsyncEnumerable source; readonly Func> predicate; readonly CancellationToken cancellationToken; int state = -1; IUniTaskAsyncEnumerator enumerator; UniTask.Awaiter awaiter; UniTask.Awaiter awaiter2; Action moveNextAction; int index; public _WhereAwaitWithCancellation(IUniTaskAsyncEnumerable source, Func> predicate, CancellationToken cancellationToken) { this.source = source; this.predicate = predicate; this.cancellationToken = cancellationToken; this.moveNextAction = MoveNext; TaskTracker.TrackActiveTask(this, 3); } public TSource Current { get; private set; } public UniTask MoveNextAsync() { if (state == -2) return default; completionSource.Reset(); MoveNext(); return new UniTask(this, completionSource.Version); } void MoveNext() { REPEAT: try { switch (state) { case -1: // init enumerator = source.GetAsyncEnumerator(cancellationToken); goto case 0; case 0: awaiter = enumerator.MoveNextAsync().GetAwaiter(); if (awaiter.IsCompleted) { goto case 1; } else { state = 1; awaiter.UnsafeOnCompleted(moveNextAction); return; } case 1: if (awaiter.GetResult()) { Current = enumerator.Current; awaiter2 = predicate(Current, checked(index++), cancellationToken).GetAwaiter(); if (awaiter2.IsCompleted) { goto case 2; } else { state = 2; awaiter2.UnsafeOnCompleted(moveNextAction); return; } } else { goto DONE; } case 2: if (awaiter2.GetResult()) { goto CONTINUE; } else { state = 0; goto REPEAT; } default: goto DONE; } } catch (Exception ex) { state = -2; completionSource.TrySetException(ex); return; } DONE: state = -2; completionSource.TrySetResult(false); return; CONTINUE: state = 0; completionSource.TrySetResult(true); return; } public UniTask DisposeAsync() { TaskTracker.RemoveTracking(this); return enumerator.DisposeAsync(); } } } }