12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697 |
- using Cysharp.Threading.Tasks.Internal;
- using System;
- using System.Threading;
- namespace Cysharp.Threading.Tasks.Linq
- {
- public static partial class UniTaskAsyncEnumerable
- {
- public static IObservable<TSource> ToObservable<TSource>(this IUniTaskAsyncEnumerable<TSource> source)
- {
- Error.ThrowArgumentNullException(source, nameof(source));
- return new ToObservable<TSource>(source);
- }
- }
- internal sealed class ToObservable<T> : IObservable<T>
- {
- readonly IUniTaskAsyncEnumerable<T> source;
- public ToObservable(IUniTaskAsyncEnumerable<T> source)
- {
- this.source = source;
- }
- public IDisposable Subscribe(IObserver<T> observer)
- {
- var ctd = new CancellationTokenDisposable();
- RunAsync(source, observer, ctd.Token).Forget();
- return ctd;
- }
- static async UniTaskVoid RunAsync(IUniTaskAsyncEnumerable<T> src, IObserver<T> observer, CancellationToken cancellationToken)
- {
-
-
- var e = src.GetAsyncEnumerator(cancellationToken);
- try
- {
- bool hasNext;
- do
- {
- try
- {
- hasNext = await e.MoveNextAsync();
- }
- catch (Exception ex)
- {
- if (cancellationToken.IsCancellationRequested)
- {
- return;
- }
- observer.OnError(ex);
- return;
- }
- if (hasNext)
- {
- observer.OnNext(e.Current);
- }
- else
- {
- observer.OnCompleted();
- return;
- }
- } while (!cancellationToken.IsCancellationRequested);
- }
- finally
- {
- if (e != null)
- {
- await e.DisposeAsync();
- }
- }
- }
- internal sealed class CancellationTokenDisposable : IDisposable
- {
- readonly CancellationTokenSource cts = new CancellationTokenSource();
- public CancellationToken Token => cts.Token;
- public void Dispose()
- {
- if (!cts.IsCancellationRequested)
- {
- cts.Cancel();
- }
- }
- }
- }
- }
|