| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178 | // 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;using System.Reactive.Concurrency;using System.Threading;using Microsoft.Reactive.Testing;using Xunit;namespace ReactiveTests.Tests{        public class TaskPoolSchedulerTest    {        [Fact]        public void TaskPool_ArgumentChecking()        {            ReactiveAssert.Throws<ArgumentNullException>(() => new TaskPoolScheduler(null));            ReactiveAssert.Throws<ArgumentNullException>(() => TaskPoolScheduler.Default.Schedule<int>(42, default(Func<IScheduler, int, IDisposable>)));            ReactiveAssert.Throws<ArgumentNullException>(() => TaskPoolScheduler.Default.Schedule<int>(42, DateTimeOffset.Now, default(Func<IScheduler, int, IDisposable>)));            ReactiveAssert.Throws<ArgumentNullException>(() => TaskPoolScheduler.Default.Schedule<int>(42, TimeSpan.Zero, default(Func<IScheduler, int, IDisposable>)));            ReactiveAssert.Throws<ArgumentNullException>(() => TaskPoolScheduler.Default.SchedulePeriodic(42, TimeSpan.FromSeconds(1), default(Func<int, int>)));            ReactiveAssert.Throws<ArgumentOutOfRangeException>(() => TaskPoolScheduler.Default.SchedulePeriodic(42, TimeSpan.FromSeconds(-1), _ => _));        }        [Fact]        public void TaskPool_Now()        {            var res = TaskPoolScheduler.Default.Now - DateTime.Now;            Assert.True(res.Seconds < 1);        }#if !NO_THREAD        [Fact]        public void TaskPool_ScheduleAction()        {            var id = Thread.CurrentThread.ManagedThreadId;            var nt = TaskPoolScheduler.Default;            var evt = new ManualResetEvent(false);            nt.Schedule(() => { Assert.NotEqual(id, Thread.CurrentThread.ManagedThreadId); evt.Set(); });            evt.WaitOne();        }        [Fact]        public void TaskPool_ScheduleActionDueNow()        {            var id = Thread.CurrentThread.ManagedThreadId;            var nt = TaskPoolScheduler.Default;            var evt = new ManualResetEvent(false);            nt.Schedule(TimeSpan.Zero, () => { Assert.NotEqual(id, Thread.CurrentThread.ManagedThreadId); evt.Set(); });            evt.WaitOne();        }        [Fact]        public void TaskPool_ScheduleActionDue()        {            var id = Thread.CurrentThread.ManagedThreadId;            var nt = TaskPoolScheduler.Default;            var evt = new ManualResetEvent(false);            nt.Schedule(TimeSpan.FromMilliseconds(1), () => { Assert.NotEqual(id, Thread.CurrentThread.ManagedThreadId); evt.Set(); });            evt.WaitOne();        }        [Fact]        public void TaskPool_ScheduleActionCancel()        {            var id = Thread.CurrentThread.ManagedThreadId;            var nt = TaskPoolScheduler.Default;            var set = false;            var d = nt.Schedule(TimeSpan.FromSeconds(0.2), () => { Assert.True(false); set = true; });            d.Dispose();            Thread.Sleep(400);            Assert.False(set);        }#if !NO_PERF        [Fact]        public void TaskPool_ScheduleLongRunning()        {            var n = 0;            var e = new ManualResetEvent(false);            var gate = new object();            var d = TaskPoolScheduler.Default.ScheduleLongRunning(42, (x, cancel) =>            {                while (!cancel.IsDisposed)                    lock (gate)                        n++;                e.Set();            });            while (true)            {                lock (gate)                    if (n >= 10)                        break;                Thread.Sleep(10);            }            d.Dispose();            e.WaitOne();            Assert.True(n >= 0);        }#endif#endif#if !NO_PERF        [Fact]        public void Stopwatch()        {            StopwatchTest.Run(TaskPoolScheduler.Default);        }#endif        [Fact]        public void TaskPool_Periodic()        {            var n = 0;            var e = new ManualResetEvent(false);            var d = TaskPoolScheduler.Default.SchedulePeriodic(TimeSpan.FromMilliseconds(25), () =>            {                if (Interlocked.Increment(ref n) == 10)                    e.Set();            });            if (!e.WaitOne(10000))                Assert.True(false);            d.Dispose();        }#if !NO_THREAD        [Fact]        public void TaskPool_Periodic_NonReentrant()        {            var n = 0;            var fail = false;            var d = TaskPoolScheduler.Default.SchedulePeriodic(0, TimeSpan.FromMilliseconds(50), x =>            {                try                {                    if (Interlocked.Increment(ref n) > 1) // Without an AsyncLock this would fail.                        fail = true;                    Thread.Sleep(100);                    return x + 1;                }                finally                {                    Interlocked.Decrement(ref n);                }            });            Thread.Sleep(500);            d.Dispose();            Assert.False(fail);        }#endif        [Fact]        public void TaskPool_Delay_LargerThanIntMaxValue()        {            var dueTime = TimeSpan.FromMilliseconds((double)int.MaxValue + 1);            // Just ensuring the call to Schedule does not throw.            var d = TaskPoolScheduler.Default.Schedule(dueTime, () => { });            d.Dispose();        }    }}
 |