using Cysharp.Threading.Tasks.Internal; using System; using System.Threading; namespace Cysharp.Threading.Tasks.Linq { public static partial class UniTaskAsyncEnumerable { public static IUniTaskAsyncEnumerable Select(this IUniTaskAsyncEnumerable source, Func selector) { Error.ThrowArgumentNullException(source, nameof(source)); Error.ThrowArgumentNullException(selector, nameof(selector)); return new Cysharp.Threading.Tasks.Linq.Select(source, selector); } public static IUniTaskAsyncEnumerable Select(this IUniTaskAsyncEnumerable source, Func selector) { Error.ThrowArgumentNullException(source, nameof(source)); Error.ThrowArgumentNullException(selector, nameof(selector)); return new Cysharp.Threading.Tasks.Linq.SelectInt(source, selector); } public static IUniTaskAsyncEnumerable SelectAwait(this IUniTaskAsyncEnumerable source, Func> selector) { Error.ThrowArgumentNullException(source, nameof(source)); Error.ThrowArgumentNullException(selector, nameof(selector)); return new Cysharp.Threading.Tasks.Linq.SelectAwait(source, selector); } public static IUniTaskAsyncEnumerable SelectAwait(this IUniTaskAsyncEnumerable source, Func> selector) { Error.ThrowArgumentNullException(source, nameof(source)); Error.ThrowArgumentNullException(selector, nameof(selector)); return new Cysharp.Threading.Tasks.Linq.SelectIntAwait(source, selector); } public static IUniTaskAsyncEnumerable SelectAwaitWithCancellation(this IUniTaskAsyncEnumerable source, Func> selector) { Error.ThrowArgumentNullException(source, nameof(source)); Error.ThrowArgumentNullException(selector, nameof(selector)); return new Cysharp.Threading.Tasks.Linq.SelectAwaitWithCancellation(source, selector); } public static IUniTaskAsyncEnumerable SelectAwaitWithCancellation(this IUniTaskAsyncEnumerable source, Func> selector) { Error.ThrowArgumentNullException(source, nameof(source)); Error.ThrowArgumentNullException(selector, nameof(selector)); return new Cysharp.Threading.Tasks.Linq.SelectIntAwaitWithCancellation(source, selector); } } internal sealed class Select : IUniTaskAsyncEnumerable { readonly IUniTaskAsyncEnumerable source; readonly Func selector; public Select(IUniTaskAsyncEnumerable source, Func selector) { this.source = source; this.selector = selector; } public IUniTaskAsyncEnumerator GetAsyncEnumerator(CancellationToken cancellationToken = default) { return new _Select(source, selector, cancellationToken); } sealed class _Select : MoveNextSource, IUniTaskAsyncEnumerator { readonly IUniTaskAsyncEnumerable source; readonly Func selector; readonly CancellationToken cancellationToken; int state = -1; IUniTaskAsyncEnumerator enumerator; UniTask.Awaiter awaiter; Action moveNextAction; public _Select(IUniTaskAsyncEnumerable source, Func selector, CancellationToken cancellationToken) { this.source = source; this.selector = selector; this.cancellationToken = cancellationToken; this.moveNextAction = MoveNext; TaskTracker.TrackActiveTask(this, 3); } public TResult Current { get; private set; } public UniTask MoveNextAsync() { if (state == -2) return default; completionSource.Reset(); MoveNext(); return new UniTask(this, completionSource.Version); } void MoveNext() { 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 = selector(enumerator.Current); goto CONTINUE; } 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 SelectInt : IUniTaskAsyncEnumerable { readonly IUniTaskAsyncEnumerable source; readonly Func selector; public SelectInt(IUniTaskAsyncEnumerable source, Func selector) { this.source = source; this.selector = selector; } public IUniTaskAsyncEnumerator GetAsyncEnumerator(CancellationToken cancellationToken = default) { return new _Select(source, selector, cancellationToken); } sealed class _Select : MoveNextSource, IUniTaskAsyncEnumerator { readonly IUniTaskAsyncEnumerable source; readonly Func selector; readonly CancellationToken cancellationToken; int state = -1; IUniTaskAsyncEnumerator enumerator; UniTask.Awaiter awaiter; Action moveNextAction; int index; public _Select(IUniTaskAsyncEnumerable source, Func selector, CancellationToken cancellationToken) { this.source = source; this.selector = selector; this.cancellationToken = cancellationToken; this.moveNextAction = MoveNext; TaskTracker.TrackActiveTask(this, 3); } public TResult Current { get; private set; } public UniTask MoveNextAsync() { if (state == -2) return default; completionSource.Reset(); MoveNext(); return new UniTask(this, completionSource.Version); } void MoveNext() { 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 = selector(enumerator.Current, checked(index++)); goto CONTINUE; } 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 SelectAwait : IUniTaskAsyncEnumerable { readonly IUniTaskAsyncEnumerable source; readonly Func> selector; public SelectAwait(IUniTaskAsyncEnumerable source, Func> selector) { this.source = source; this.selector = selector; } public IUniTaskAsyncEnumerator GetAsyncEnumerator(CancellationToken cancellationToken = default) { return new _SelectAwait(source, selector, cancellationToken); } sealed class _SelectAwait : MoveNextSource, IUniTaskAsyncEnumerator { readonly IUniTaskAsyncEnumerable source; readonly Func> selector; readonly CancellationToken cancellationToken; int state = -1; IUniTaskAsyncEnumerator enumerator; UniTask.Awaiter awaiter; UniTask.Awaiter awaiter2; Action moveNextAction; public _SelectAwait(IUniTaskAsyncEnumerable source, Func> selector, CancellationToken cancellationToken) { this.source = source; this.selector = selector; this.cancellationToken = cancellationToken; this.moveNextAction = MoveNext; TaskTracker.TrackActiveTask(this, 3); } public TResult Current { get; private set; } public UniTask MoveNextAsync() { if (state == -2) return default; completionSource.Reset(); MoveNext(); return new UniTask(this, completionSource.Version); } void MoveNext() { 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()) { awaiter2 = selector(enumerator.Current).GetAwaiter(); if (awaiter2.IsCompleted) { goto case 2; } else { state = 2; awaiter2.UnsafeOnCompleted(moveNextAction); return; } } else { goto DONE; } case 2: Current = awaiter2.GetResult(); goto CONTINUE; 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 SelectIntAwait : IUniTaskAsyncEnumerable { readonly IUniTaskAsyncEnumerable source; readonly Func> selector; public SelectIntAwait(IUniTaskAsyncEnumerable source, Func> selector) { this.source = source; this.selector = selector; } public IUniTaskAsyncEnumerator GetAsyncEnumerator(CancellationToken cancellationToken = default) { return new _SelectAwait(source, selector, cancellationToken); } sealed class _SelectAwait : MoveNextSource, IUniTaskAsyncEnumerator { readonly IUniTaskAsyncEnumerable source; readonly Func> selector; readonly CancellationToken cancellationToken; int state = -1; IUniTaskAsyncEnumerator enumerator; UniTask.Awaiter awaiter; UniTask.Awaiter awaiter2; Action moveNextAction; int index; public _SelectAwait(IUniTaskAsyncEnumerable source, Func> selector, CancellationToken cancellationToken) { this.source = source; this.selector = selector; this.cancellationToken = cancellationToken; this.moveNextAction = MoveNext; TaskTracker.TrackActiveTask(this, 3); } public TResult Current { get; private set; } public UniTask MoveNextAsync() { if (state == -2) return default; completionSource.Reset(); MoveNext(); return new UniTask(this, completionSource.Version); } void MoveNext() { 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()) { awaiter2 = selector(enumerator.Current, checked(index++)).GetAwaiter(); if (awaiter2.IsCompleted) { goto case 2; } else { state = 2; awaiter2.UnsafeOnCompleted(moveNextAction); return; } } else { goto DONE; } case 2: Current = awaiter2.GetResult(); goto CONTINUE; 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 SelectAwaitWithCancellation : IUniTaskAsyncEnumerable { readonly IUniTaskAsyncEnumerable source; readonly Func> selector; public SelectAwaitWithCancellation(IUniTaskAsyncEnumerable source, Func> selector) { this.source = source; this.selector = selector; } public IUniTaskAsyncEnumerator GetAsyncEnumerator(CancellationToken cancellationToken = default) { return new _SelectAwaitWithCancellation(source, selector, cancellationToken); } sealed class _SelectAwaitWithCancellation : MoveNextSource, IUniTaskAsyncEnumerator { readonly IUniTaskAsyncEnumerable source; readonly Func> selector; readonly CancellationToken cancellationToken; int state = -1; IUniTaskAsyncEnumerator enumerator; UniTask.Awaiter awaiter; UniTask.Awaiter awaiter2; Action moveNextAction; public _SelectAwaitWithCancellation(IUniTaskAsyncEnumerable source, Func> selector, CancellationToken cancellationToken) { this.source = source; this.selector = selector; this.cancellationToken = cancellationToken; this.moveNextAction = MoveNext; TaskTracker.TrackActiveTask(this, 3); } public TResult Current { get; private set; } public UniTask MoveNextAsync() { if (state == -2) return default; completionSource.Reset(); MoveNext(); return new UniTask(this, completionSource.Version); } void MoveNext() { 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()) { awaiter2 = selector(enumerator.Current, cancellationToken).GetAwaiter(); if (awaiter2.IsCompleted) { goto case 2; } else { state = 2; awaiter2.UnsafeOnCompleted(moveNextAction); return; } } else { goto DONE; } case 2: Current = awaiter2.GetResult(); goto CONTINUE; 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 SelectIntAwaitWithCancellation : IUniTaskAsyncEnumerable { readonly IUniTaskAsyncEnumerable source; readonly Func> selector; public SelectIntAwaitWithCancellation(IUniTaskAsyncEnumerable source, Func> selector) { this.source = source; this.selector = selector; } public IUniTaskAsyncEnumerator GetAsyncEnumerator(CancellationToken cancellationToken = default) { return new _SelectAwaitWithCancellation(source, selector, cancellationToken); } sealed class _SelectAwaitWithCancellation : MoveNextSource, IUniTaskAsyncEnumerator { readonly IUniTaskAsyncEnumerable source; readonly Func> selector; readonly CancellationToken cancellationToken; int state = -1; IUniTaskAsyncEnumerator enumerator; UniTask.Awaiter awaiter; UniTask.Awaiter awaiter2; Action moveNextAction; int index; public _SelectAwaitWithCancellation(IUniTaskAsyncEnumerable source, Func> selector, CancellationToken cancellationToken) { this.source = source; this.selector = selector; this.cancellationToken = cancellationToken; this.moveNextAction = MoveNext; TaskTracker.TrackActiveTask(this, 3); } public TResult Current { get; private set; } public UniTask MoveNextAsync() { if (state == -2) return default; completionSource.Reset(); MoveNext(); return new UniTask(this, completionSource.Version); } void MoveNext() { 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()) { awaiter2 = selector(enumerator.Current, checked(index++), cancellationToken).GetAwaiter(); if (awaiter2.IsCompleted) { goto case 2; } else { state = 2; awaiter2.UnsafeOnCompleted(moveNextAction); return; } } else { goto DONE; } case 2: Current = awaiter2.GetResult(); goto CONTINUE; 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(); } } } }