1
0
Bart De Smet 8 жил өмнө
parent
commit
3025b05bd8

+ 297 - 0
AsyncRx.NET/System.Reactive.Async.Linq/System/Reactive/Linq/Operators/Throttle.cs

@@ -0,0 +1,297 @@
+// Licensed to the .NET Foundation under one or more agreements.
+// The .NET Foundation licenses this file to you under the Apache 2.0 License.
+// See the LICENSE file in the project root for more information. 
+
+using System.Reactive.Concurrency;
+using System.Reactive.Disposables;
+using System.Threading;
+
+namespace System.Reactive.Linq
+{
+    partial class AsyncObservable
+    {
+        public static IAsyncObservable<TSource> Throttle<TSource>(this IAsyncObservable<TSource> source, TimeSpan dueTime)
+        {
+            if (source == null)
+                throw new ArgumentNullException(nameof(source));
+            if (dueTime < TimeSpan.Zero)
+                throw new ArgumentOutOfRangeException(nameof(dueTime));
+
+            return Create<TSource>(async observer =>
+            {
+                var d = new CompositeAsyncDisposable();
+
+                var (sink, throttler) = AsyncObserver.Throttle(observer, dueTime);
+
+                await d.AddAsync(throttler).ConfigureAwait(false);
+
+                var sourceSubscription = await source.SubscribeSafeAsync(sink).ConfigureAwait(false);
+
+                await d.AddAsync(sourceSubscription).ConfigureAwait(false);
+
+                return d;
+            });
+        }
+
+        public static IAsyncObservable<TSource> Throttle<TSource>(this IAsyncObservable<TSource> source, TimeSpan dueTime, IAsyncScheduler scheduler)
+        {
+            if (source == null)
+                throw new ArgumentNullException(nameof(source));
+            if (dueTime < TimeSpan.Zero)
+                throw new ArgumentOutOfRangeException(nameof(dueTime));
+            if (scheduler == null)
+                throw new ArgumentNullException(nameof(scheduler));
+
+            return Create<TSource>(async observer =>
+            {
+                var d = new CompositeAsyncDisposable();
+
+                var (sink, throttler) = AsyncObserver.Throttle(observer, dueTime, scheduler);
+
+                await d.AddAsync(throttler).ConfigureAwait(false);
+
+                var sourceSubscription = await source.SubscribeSafeAsync(sink).ConfigureAwait(false);
+
+                await d.AddAsync(sourceSubscription).ConfigureAwait(false);
+
+                return d;
+            });
+        }
+
+        public static IAsyncObservable<TSource> Throttle<TSource, TThrottle>(this IAsyncObservable<TSource> source, Func<TSource, IAsyncObservable<TThrottle>> throttleSelector)
+        {
+            if (source == null)
+                throw new ArgumentNullException(nameof(source));
+            if (throttleSelector == null)
+                throw new ArgumentNullException(nameof(throttleSelector));
+
+            return Create<TSource>(async observer =>
+            {
+                var d = new CompositeAsyncDisposable();
+
+                var (sink, throttler) = AsyncObserver.Throttle(observer, throttleSelector);
+
+                await d.AddAsync(throttler).ConfigureAwait(false);
+
+                var sourceSubscription = await source.SubscribeSafeAsync(sink).ConfigureAwait(false);
+
+                await d.AddAsync(sourceSubscription).ConfigureAwait(false);
+
+                return d;
+            });
+        }
+    }
+
+    partial class AsyncObserver
+    {
+        public static (IAsyncObserver<TSource>, IAsyncDisposable) Throttle<TSource>(IAsyncObserver<TSource> observer, TimeSpan dueTime) => Throttle(observer, dueTime, TaskPoolAsyncScheduler.Default);
+
+        public static (IAsyncObserver<TSource>, IAsyncDisposable) Throttle<TSource>(IAsyncObserver<TSource> observer, TimeSpan dueTime, IAsyncScheduler scheduler)
+        {
+            if (observer == null)
+                throw new ArgumentNullException(nameof(observer));
+            if (dueTime < TimeSpan.Zero)
+                throw new ArgumentOutOfRangeException(nameof(dueTime));
+            if (scheduler == null)
+                throw new ArgumentNullException(nameof(scheduler));
+
+            var gate = new AsyncLock();
+
+            var timer = new SerialAsyncDisposable();
+
+            var hasValue = false;
+            var value = default(TSource);
+            var id = 0UL;
+
+            return
+                (
+                    Create<TSource>(
+                        async x =>
+                        {
+                            var myId = default(ulong);
+
+                            using (await gate.LockAsync().ConfigureAwait(false))
+                            {
+                                hasValue = true;
+                                value = x;
+                                myId = ++id;
+                            }
+
+                            var d = new SingleAssignmentAsyncDisposable();
+                            await timer.AssignAsync(d).ConfigureAwait(false);
+
+                            var t = await scheduler.ScheduleAsync(async ct =>
+                            {
+                                if (!ct.IsCancellationRequested)
+                                {
+                                    using (await gate.LockAsync().ConfigureAwait(false))
+                                    {
+                                        if (hasValue && id == myId)
+                                        {
+                                            await observer.OnNextAsync(value).ConfigureAwait(false);
+                                        }
+
+                                        hasValue = false;
+                                    }
+                                }
+                            }).ConfigureAwait(false);
+                            await d.AssignAsync(t).ConfigureAwait(false);
+                        },
+                        async ex =>
+                        {
+                            await timer.DisposeAsync().ConfigureAwait(false);
+
+                            using (await gate.LockAsync().ConfigureAwait(false))
+                            {
+                                await observer.OnErrorAsync(ex).ConfigureAwait(false);
+
+                                hasValue = false;
+                                id++;
+                            }
+                        },
+                        async () =>
+                        {
+                            await timer.DisposeAsync().ConfigureAwait(false);
+
+                            using (await gate.LockAsync().ConfigureAwait(false))
+                            {
+                                if (hasValue)
+                                {
+                                    await observer.OnNextAsync(value).ConfigureAwait(false);
+                                }
+
+                                await observer.OnCompletedAsync().ConfigureAwait(false);
+
+                                hasValue = false;
+                                id++;
+                            }
+                        }
+                    ),
+                    timer
+                );
+        }
+
+        public static (IAsyncObserver<TSource>, IAsyncDisposable) Throttle<TSource, TThrottle>(IAsyncObserver<TSource> observer, Func<TSource, IAsyncObservable<TThrottle>> throttleSelector)
+        {
+            if (observer == null)
+                throw new ArgumentNullException(nameof(observer));
+            if (throttleSelector == null)
+                throw new ArgumentNullException(nameof(throttleSelector));
+
+            var gate = new AsyncLock();
+
+            var throttler = new SerialAsyncDisposable();
+
+            var hasValue = false;
+            var value = default(TSource);
+            var id = 0UL;
+
+            return
+                (
+                    Create<TSource>(
+                        async x =>
+                        {
+                            var throttleSource = default(IAsyncObservable<TThrottle>);
+
+                            try
+                            {
+                                throttleSource = throttleSelector(x); // REVIEW: Do we need an async variant?
+                            }
+                            catch (Exception ex)
+                            {
+                                using (await gate.LockAsync().ConfigureAwait(false))
+                                {
+                                    await observer.OnErrorAsync(ex).ConfigureAwait(false);
+                                }
+
+                                return;
+                            }
+
+                            var myId = default(ulong);
+
+                            using (await gate.LockAsync().ConfigureAwait(false))
+                            {
+                                hasValue = true;
+                                value = x;
+                                myId = ++id;
+                            }
+
+                            var d = new SingleAssignmentAsyncDisposable();
+                            await throttler.AssignAsync(d).ConfigureAwait(false);
+
+                            var throttleObserver = Create<TThrottle>(
+                                    async y =>
+                                    {
+                                        using (await gate.LockAsync().ConfigureAwait(false))
+                                        {
+                                            if (hasValue && myId == id)
+                                            {
+                                                await observer.OnNextAsync(x).ConfigureAwait(false);
+                                            }
+
+                                            hasValue = false;
+
+                                            await d.DisposeAsync().ConfigureAwait(false);
+                                        }
+                                    },
+                                    async ex =>
+                                    {
+                                        using (await gate.LockAsync().ConfigureAwait(false))
+                                        {
+                                            await observer.OnErrorAsync(ex).ConfigureAwait(false);
+                                        }
+                                    },
+                                    async () =>
+                                    {
+                                        using (await gate.LockAsync().ConfigureAwait(false))
+                                        {
+                                            if (hasValue && myId == id)
+                                            {
+                                                await observer.OnNextAsync(x).ConfigureAwait(false);
+                                            }
+
+                                            hasValue = false;
+
+                                            await d.DisposeAsync().ConfigureAwait(false);
+                                        }
+                                    }
+                                );
+
+                            var t = await throttleSource.SubscribeSafeAsync(throttleObserver).ConfigureAwait(false);
+                            await d.AssignAsync(t).ConfigureAwait(false);
+                        },
+                        async ex =>
+                        {
+                            await throttler.DisposeAsync().ConfigureAwait(false);
+
+                            using (await gate.LockAsync().ConfigureAwait(false))
+                            {
+                                await observer.OnErrorAsync(ex).ConfigureAwait(false);
+
+                                hasValue = false;
+                                id++;
+                            }
+                        },
+                        async () =>
+                        {
+                            await throttler.DisposeAsync().ConfigureAwait(false);
+
+                            using (await gate.LockAsync().ConfigureAwait(false))
+                            {
+                                if (hasValue)
+                                {
+                                    await observer.OnNextAsync(value).ConfigureAwait(false);
+                                }
+
+                                await observer.OnCompletedAsync().ConfigureAwait(false);
+
+                                hasValue = false;
+                                id++;
+                            }
+                        }
+                    ),
+                    throttler
+                );
+        }
+    }
+}