| 12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082 | // Copyright (c) Microsoft Open Technologies, Inc. All rights reserved. See License.txt in the project root for license information.using System;using System.Collections.Generic;using System.Linq;using System.Reactive.Concurrency;using System.Reactive.Disposables;using System.Reactive.Linq;using System.Reactive.PlatformServices;using System.Runtime.CompilerServices;using System.Threading;using Microsoft.VisualStudio.TestTools.UnitTesting;using Microsoft.Reactive.Testing;#if HAS_WINFORMSusing System.Windows.Forms;#endifnamespace ReactiveTests.Tests{    [TestClass]    public class SchedulerTest    {        #region IScheduler        [TestMethod]        public void Scheduler_ArgumentChecks()        {            var ms = new MyScheduler();            ReactiveAssert.Throws<ArgumentNullException>(() => Scheduler.Schedule(default(IScheduler), a => { }));            ReactiveAssert.Throws<ArgumentNullException>(() => Scheduler.Schedule(default(IScheduler), () => { }));            ReactiveAssert.Throws<ArgumentNullException>(() => Scheduler.Schedule(default(IScheduler), 1, (a, s) => { }));            ReactiveAssert.Throws<ArgumentNullException>(() => Scheduler.Schedule(ms, default(Action<Action>)));            ReactiveAssert.Throws<ArgumentNullException>(() => Scheduler.Schedule(ms, 1, default(Action<int, Action<int>>)));            ReactiveAssert.Throws<ArgumentNullException>(() => Scheduler.Schedule(default(IScheduler), DateTimeOffset.Now, a => { }));            ReactiveAssert.Throws<ArgumentNullException>(() => Scheduler.Schedule(default(IScheduler), DateTimeOffset.Now, () => { }));            ReactiveAssert.Throws<ArgumentNullException>(() => Scheduler.Schedule(default(IScheduler), 1, DateTimeOffset.Now, (a, s) => { }));            ReactiveAssert.Throws<ArgumentNullException>(() => Scheduler.Schedule(ms, DateTimeOffset.Now, default(Action<Action<DateTimeOffset>>)));            ReactiveAssert.Throws<ArgumentNullException>(() => Scheduler.Schedule(ms, 1, DateTimeOffset.Now, default(Action<int, Action<int, DateTimeOffset>>)));            ReactiveAssert.Throws<ArgumentNullException>(() => Scheduler.Schedule(default(IScheduler), TimeSpan.Zero, a => { }));            ReactiveAssert.Throws<ArgumentNullException>(() => Scheduler.Schedule(default(IScheduler), TimeSpan.Zero, () => { }));            ReactiveAssert.Throws<ArgumentNullException>(() => Scheduler.Schedule(default(IScheduler), 1, TimeSpan.Zero, (a, s) => { }));            ReactiveAssert.Throws<ArgumentNullException>(() => Scheduler.Schedule(ms, TimeSpan.Zero, default(Action<Action<TimeSpan>>)));            ReactiveAssert.Throws<ArgumentNullException>(() => Scheduler.Schedule(ms, 1, TimeSpan.Zero, default(Action<int, Action<int, TimeSpan>>)));        }        [TestMethod]        public void Schedulers_ArgumentChecks()        {            ReactiveAssert.Throws<ArgumentNullException>(() => Scheduler.CurrentThread.Schedule(default(Action)));            ReactiveAssert.Throws<ArgumentNullException>(() => Scheduler.CurrentThread.Schedule(TimeSpan.Zero, default(Action)));            ReactiveAssert.Throws<ArgumentNullException>(() => Scheduler.CurrentThread.Schedule(DateTimeOffset.MaxValue, default(Action)));            ReactiveAssert.Throws<ArgumentNullException>(() => DispatcherScheduler.Instance.Schedule(default(Action)));            ReactiveAssert.Throws<ArgumentNullException>(() => DispatcherScheduler.Instance.Schedule(TimeSpan.Zero, default(Action)));            ReactiveAssert.Throws<ArgumentNullException>(() => DispatcherScheduler.Instance.Schedule(DateTimeOffset.MaxValue, default(Action)));            ReactiveAssert.Throws<ArgumentNullException>(() => Scheduler.Immediate.Schedule(default(Action)));            ReactiveAssert.Throws<ArgumentNullException>(() => Scheduler.Immediate.Schedule(TimeSpan.Zero, default(Action)));            ReactiveAssert.Throws<ArgumentNullException>(() => Scheduler.Immediate.Schedule(DateTimeOffset.MaxValue, default(Action)));            ReactiveAssert.Throws<ArgumentNullException>(() => NewThreadScheduler.Default.Schedule(default(Action)));            ReactiveAssert.Throws<ArgumentNullException>(() => NewThreadScheduler.Default.Schedule(TimeSpan.Zero, default(Action)));            ReactiveAssert.Throws<ArgumentNullException>(() => NewThreadScheduler.Default.Schedule(DateTimeOffset.MaxValue, default(Action)));#if !NO_TPL            ReactiveAssert.Throws<ArgumentNullException>(() => TaskPoolScheduler.Default.Schedule(default(Action)));            ReactiveAssert.Throws<ArgumentNullException>(() => TaskPoolScheduler.Default.Schedule(TimeSpan.Zero, default(Action)));            ReactiveAssert.Throws<ArgumentNullException>(() => TaskPoolScheduler.Default.Schedule(DateTimeOffset.MaxValue, default(Action)));#endif            ReactiveAssert.Throws<ArgumentNullException>(() => DefaultScheduler.Instance.Schedule(default(Action)));            ReactiveAssert.Throws<ArgumentNullException>(() => DefaultScheduler.Instance.Schedule(TimeSpan.Zero, default(Action)));            ReactiveAssert.Throws<ArgumentNullException>(() => DefaultScheduler.Instance.Schedule(DateTimeOffset.MaxValue, default(Action)));            ReactiveAssert.Throws<ArgumentNullException>(() => DefaultScheduler.Instance.SchedulePeriodic(42, TimeSpan.FromSeconds(1), default(Func<int, int>)));            ReactiveAssert.Throws<ArgumentOutOfRangeException>(() => DefaultScheduler.Instance.SchedulePeriodic(42, TimeSpan.FromSeconds(-1), _ => _));#if HAS_WINFORMS            var lbl = new Label();            ReactiveAssert.Throws<ArgumentNullException>(() => new ControlScheduler(lbl).Schedule(default(Action)));            ReactiveAssert.Throws<ArgumentNullException>(() => new ControlScheduler(lbl).Schedule(TimeSpan.Zero, default(Action)));            ReactiveAssert.Throws<ArgumentNullException>(() => new ControlScheduler(lbl).Schedule(DateTimeOffset.MaxValue, default(Action)));#endif            var ctx = new SynchronizationContext();            ReactiveAssert.Throws<ArgumentNullException>(() => new SynchronizationContextScheduler(ctx).Schedule(default(Action)));            ReactiveAssert.Throws<ArgumentNullException>(() => new SynchronizationContextScheduler(ctx).Schedule(TimeSpan.Zero, default(Action)));            ReactiveAssert.Throws<ArgumentNullException>(() => new SynchronizationContextScheduler(ctx).Schedule(DateTimeOffset.MaxValue, default(Action)));        }        [TestMethod]        public void Scheduler_ScheduleNonRecursive()        {            var ms = new MyScheduler();            var res = false;            Scheduler.Schedule(ms, a => { res = true; });            Assert.IsTrue(res);        }        [TestMethod]        public void Scheduler_ScheduleRecursive()        {            var ms = new MyScheduler();            var i = 0;            Scheduler.Schedule(ms, a => { if (++i < 10) a(); });            Assert.AreEqual(10, i);        }        [TestMethod]        public void Scheduler_ScheduleWithTimeNonRecursive()        {            var now = DateTimeOffset.Now;            var ms = new MyScheduler(now) { Check = (a, s, t) => { Assert.IsTrue(t == TimeSpan.Zero); } };            var res = false;            Scheduler.Schedule(ms, now, a => { res = true; });            Assert.IsTrue(res);            Assert.IsTrue(ms.WaitCycles == 0);        }        [TestMethod]        public void Scheduler_ScheduleWithTimeRecursive()        {            var now = DateTimeOffset.Now;            var i = 0;            var ms = new MyScheduler(now) { Check = (a, s, t) => { Assert.IsTrue(t == TimeSpan.Zero); } };            Scheduler.Schedule(ms, now, a => { if (++i < 10) a(now); });            Assert.IsTrue(ms.WaitCycles == 0);            Assert.AreEqual(10, i);        }        [TestMethod]        public void Scheduler_ScheduleWithTimeSpanNonRecursive()        {            var now = DateTimeOffset.Now;            var ms = new MyScheduler(now) { Check = (a, s, t) => { Assert.IsTrue(t == TimeSpan.Zero); } };            var res = false;            Scheduler.Schedule(ms, TimeSpan.Zero, a => { res = true; });            Assert.IsTrue(res);            Assert.IsTrue(ms.WaitCycles == 0);        }        [TestMethod]        public void Scheduler_ScheduleWithTimeSpanRecursive()        {            var now = DateTimeOffset.Now;            var ms = new MyScheduler(now) { Check = (a, s, t) => { Assert.IsTrue(t < TimeSpan.FromTicks(10)); } };            var i = 0;            Scheduler.Schedule(ms, TimeSpan.Zero, a => { if (++i < 10) a(TimeSpan.FromTicks(i)); });            Assert.IsTrue(ms.WaitCycles == Enumerable.Range(1, 9).Sum());            Assert.AreEqual(10, i);        }        [TestMethod]        public void Scheduler_StateThreading()        {            var lst = new List<int>();            Scheduler.Immediate.Schedule(0, (i, a) =>            {                lst.Add(i);                if (i < 9)                    a(i + 1);            });            Assert.IsTrue(lst.SequenceEqual(Enumerable.Range(0, 10)));        }        [TestMethod]        public void Scheduler_Builtins()        {            // Default            {                var e = new ManualResetEvent(false);                Scheduler.Default.Schedule(() => e.Set());                e.WaitOne();            }            if (!Utils.IsRunningWithPortableLibraryBinaries())            {                Scheduler_Builtins_NoPlib();            }        }        [MethodImpl(MethodImplOptions.NoInlining)]        private void Scheduler_Builtins_NoPlib()        {#if !PLIB            // ThreadPool            {                var e = new ManualResetEvent(false);                Scheduler.ThreadPool.Schedule(() => e.Set());                e.WaitOne();            }#endif#if !NO_THREAD            // NewThread            {                var e = new ManualResetEvent(false);                Scheduler.NewThread.Schedule(() => e.Set());                e.WaitOne();            }#endif#if !NO_TPL            // TaskPool            {                var e = new ManualResetEvent(false);                Scheduler.TaskPool.Schedule(() => e.Set());                e.WaitOne();            }#endif        }        #endregion        #region ISchedulerLongRunning#if !NO_PERF        [TestMethod]        public void Scheduler_LongRunning_ArgumentChecking()        {            ReactiveAssert.Throws<ArgumentNullException>(() => Scheduler.ScheduleLongRunning(null, c => { }));            ReactiveAssert.Throws<ArgumentNullException>(() => Scheduler.ScheduleLongRunning(ThreadPoolScheduler.Instance, default(Action<ICancelable>)));        }        [TestMethod]        public void Scheduler_Periodic_ArgumentChecking()        {            ReactiveAssert.Throws<ArgumentNullException>(() => Scheduler.SchedulePeriodic(null, TimeSpan.FromSeconds(1), () => { }));            ReactiveAssert.Throws<ArgumentOutOfRangeException>(() => Scheduler.SchedulePeriodic(ThreadPoolScheduler.Instance, TimeSpan.FromSeconds(-1), () => { }));            ReactiveAssert.Throws<ArgumentNullException>(() => Scheduler.SchedulePeriodic(ThreadPoolScheduler.Instance, TimeSpan.FromSeconds(1), default(Action)));            ReactiveAssert.Throws<ArgumentNullException>(() => Scheduler.SchedulePeriodic<int>(null, 42, TimeSpan.FromSeconds(1), _ => { }));            ReactiveAssert.Throws<ArgumentOutOfRangeException>(() => Scheduler.SchedulePeriodic<int>(ThreadPoolScheduler.Instance, 42, TimeSpan.FromSeconds(-1), _ => { }));            ReactiveAssert.Throws<ArgumentNullException>(() => Scheduler.SchedulePeriodic<int>(ThreadPoolScheduler.Instance, 42, TimeSpan.FromSeconds(1), default(Action<int>)));            ReactiveAssert.Throws<ArgumentNullException>(() => Scheduler.SchedulePeriodic<int>(null, 42, TimeSpan.FromSeconds(1), _ => _));            ReactiveAssert.Throws<ArgumentOutOfRangeException>(() => Scheduler.SchedulePeriodic<int>(ThreadPoolScheduler.Instance, 42, TimeSpan.FromSeconds(-1), _ => _));            ReactiveAssert.Throws<ArgumentNullException>(() => Scheduler.SchedulePeriodic<int>(ThreadPoolScheduler.Instance, 42, TimeSpan.FromSeconds(1), default(Func<int, int>)));        }        [TestMethod]        public void Scheduler_Stopwatch_Emulation()        {            ReactiveAssert.Throws<ArgumentNullException>(() => Scheduler.StartStopwatch(null));        }#if !NO_TPL        [TestMethod]        public void Scheduler_LongRunning1()        {            var s = TaskPoolScheduler.Default;            var x = new ManualResetEvent(false);            var e = new ManualResetEvent(false);            var d = s.ScheduleLongRunning(42, (state, cancel) =>            {                while (!cancel.IsDisposed)                    x.Set();                e.Set();            });            x.WaitOne();            d.Dispose();            e.WaitOne();        }        [TestMethod]        public void Scheduler_LongRunning2()        {            var s = TaskPoolScheduler.Default;            var x = new ManualResetEvent(false);            var e = new ManualResetEvent(false);            var d = s.ScheduleLongRunning(cancel =>            {                while (!cancel.IsDisposed)                    x.Set();                e.Set();            });            x.WaitOne();            d.Dispose();            e.WaitOne();        }#endif#endif        #endregion        #region ISchedulerPeriodic#if !NO_PERF        [TestMethod]        public void Scheduler_Periodic1()        {            var n = 0;            var e = new ManualResetEvent(false);            var d = ThreadPoolScheduler.Instance.SchedulePeriodic(TimeSpan.FromMilliseconds(50), () =>            {                if (n++ == 10)                    e.Set();            });            e.WaitOne();            d.Dispose();        }        [TestMethod]        public void Scheduler_Periodic2()        {            var n = 0;            var e = new ManualResetEvent(false);            var d = ThreadPoolScheduler.Instance.SchedulePeriodic(42, TimeSpan.FromMilliseconds(50), x =>            {                Assert.AreEqual(42, x);                if (n++ == 10)                    e.Set();            });            e.WaitOne();            d.Dispose();        }#if DESKTOPCLR        [TestMethod]        public void Scheduler_Periodic_HostLifecycleManagement()        {            var cur = AppDomain.CurrentDomain.BaseDirectory;            var domain = AppDomain.CreateDomain("HLN", null, new AppDomainSetup { ApplicationBase = cur });            domain.DoCallBack(Scheduler_Periodic_HostLifecycleManagement_Callback);        }        private static void Scheduler_Periodic_HostLifecycleManagement_Callback()        {            var pep = PlatformEnlightenmentProvider.Current;            try            {                var hln = new HLN();                PlatformEnlightenmentProvider.Current = new PEP(hln);                var s = ThreadPoolScheduler.Instance.DisableOptimizations(typeof(ISchedulerPeriodic));                var n = 0;                var e = new ManualResetEvent(false);                var d = Observable.Interval(TimeSpan.FromMilliseconds(100), s).Subscribe(_ =>                {                    if (n++ == 10)                        e.Set();                });                hln.OnSuspending();                hln.OnResuming();                Thread.Sleep(250);                hln.OnSuspending();                Thread.Sleep(150);                hln.OnResuming();                Thread.Sleep(50);                hln.OnSuspending();                hln.OnResuming();                e.WaitOne();                d.Dispose();            }            finally            {                PlatformEnlightenmentProvider.Current = pep;            }        }        class PEP : IPlatformEnlightenmentProvider        {            private readonly IPlatformEnlightenmentProvider _old;            private readonly IHostLifecycleNotifications _hln;            public PEP(HLN hln)            {                _old = PlatformEnlightenmentProvider.Current;                _hln = hln;            }            public T GetService<T>(params object[] args) where T : class            {                if (typeof(T) == typeof(IHostLifecycleNotifications))                {                    return (T)(object)_hln;                }                return _old.GetService<T>(args);            }        }        class HLN : IHostLifecycleNotifications        {            public event EventHandler<HostSuspendingEventArgs> Suspending;            public event EventHandler<HostResumingEventArgs> Resuming;            public void OnSuspending()            {                var s = Suspending;                if (s != null)                    s(this, null);            }            public void OnResuming()            {                var s = Resuming;                if (s != null)                    s(this, null);            }        }#endif#endif        #endregion        #region DisableOptimizations#if !NO_PERF        [TestMethod]        public void DisableOptimizations_ArgumentChecking()        {            ReactiveAssert.Throws<ArgumentNullException>(() => Scheduler.DisableOptimizations(default(IScheduler)));            ReactiveAssert.Throws<ArgumentNullException>(() => Scheduler.DisableOptimizations(default(IScheduler), new Type[0]));            ReactiveAssert.Throws<ArgumentNullException>(() => Scheduler.DisableOptimizations(ThreadPoolScheduler.Instance, default(Type[])));            ReactiveAssert.Throws<ArgumentNullException>(() => Scheduler.DisableOptimizations(Scheduler.Default).Schedule(42, default(Func<IScheduler, int, IDisposable>)));            ReactiveAssert.Throws<ArgumentNullException>(() => Scheduler.DisableOptimizations(Scheduler.Default).Schedule(42, TimeSpan.FromSeconds(1), default(Func<IScheduler, int, IDisposable>)));            ReactiveAssert.Throws<ArgumentNullException>(() => Scheduler.DisableOptimizations(Scheduler.Default).Schedule(42, DateTimeOffset.Now, default(Func<IScheduler, int, IDisposable>)));        }#if !NO_TPL        [TestMethod]        public void DisableOptimizations1()        {            var s = TaskPoolScheduler.Default;            Assert.IsTrue(s is IServiceProvider);            var t = s.DisableOptimizations();            var d = t.Now - s.Now;            Assert.IsTrue(d.TotalSeconds < 1);            var e1 = new ManualResetEvent(false);            t.Schedule(42, (self, state) =>            {                e1.Set();                return Disposable.Empty;            });            e1.WaitOne();            var e2 = new ManualResetEvent(false);            t.Schedule(42, TimeSpan.FromMilliseconds(1), (self, state) =>            {                e2.Set();                return Disposable.Empty;            });            e2.WaitOne();            var e3 = new ManualResetEvent(false);            t.Schedule(42, DateTimeOffset.Now.AddMilliseconds(10), (self, state) =>            {                e3.Set();                return Disposable.Empty;            });            e3.WaitOne();        }        [TestMethod]        public void DisableOptimizations2()        {            var s = TaskPoolScheduler.Default;            Assert.IsTrue(s is IServiceProvider);            var lr1 = ((IServiceProvider)s).GetService(typeof(ISchedulerLongRunning));            Assert.IsNotNull(lr1);            var e1 = new ManualResetEvent(false);            s.Schedule(42, (self, state) =>            {                Assert.IsTrue(self is IServiceProvider);                var lrr1 = ((IServiceProvider)self).GetService(typeof(ISchedulerLongRunning));                Assert.IsNotNull(lrr1);                e1.Set();                return Disposable.Empty;            });            e1.WaitOne();            var t = s.DisableOptimizations();            Assert.IsTrue(t is IServiceProvider);            var lr2 = ((IServiceProvider)t).GetService(typeof(ISchedulerLongRunning));            Assert.IsNull(lr2);            var e2 = new ManualResetEvent(false);            t.Schedule(42, (self, state) =>            {                Assert.IsTrue(self is IServiceProvider);                var lrr2 = ((IServiceProvider)self).GetService(typeof(ISchedulerLongRunning));                Assert.IsNull(lrr2);                e2.Set();                return Disposable.Empty;            });            e2.WaitOne();        }        [TestMethod]        public void DisableOptimizations3()        {            var s = TaskPoolScheduler.Default;            Assert.IsTrue(s is IServiceProvider);            var lr1 = ((IServiceProvider)s).GetService(typeof(ISchedulerLongRunning));            Assert.IsNotNull(lr1);            var p1 = ((IServiceProvider)s).GetService(typeof(ISchedulerPeriodic));            Assert.IsNotNull(p1);            var e1 = new ManualResetEvent(false);            s.Schedule(42, (self, state) =>            {                Assert.IsTrue(self is IServiceProvider);                var lrr1 = ((IServiceProvider)self).GetService(typeof(ISchedulerLongRunning));                Assert.IsNotNull(lrr1);                var pr1 = ((IServiceProvider)self).GetService(typeof(ISchedulerPeriodic));                Assert.IsNotNull(pr1);                e1.Set();                return Disposable.Empty;            });            e1.WaitOne();            var t = s.DisableOptimizations(typeof(ISchedulerLongRunning));            Assert.IsTrue(t is IServiceProvider);            var lr2 = ((IServiceProvider)t).GetService(typeof(ISchedulerLongRunning));            Assert.IsNull(lr2);            var p2 = ((IServiceProvider)t).GetService(typeof(ISchedulerPeriodic));            Assert.IsNotNull(p2);            var e2 = new ManualResetEvent(false);            t.Schedule(42, (self, state) =>            {                Assert.IsTrue(self is IServiceProvider);                var lrr2 = ((IServiceProvider)self).GetService(typeof(ISchedulerLongRunning));                Assert.IsNull(lrr2);                var pr2 = ((IServiceProvider)self).GetService(typeof(ISchedulerPeriodic));                Assert.IsNotNull(pr2);                e2.Set();                return Disposable.Empty;            });            e2.WaitOne();        }#endif#endif        [TestMethod]        public void DisableOptimizations_UnknownService()        {            var s = new MyScheduler();            Assert.IsFalse(s is IServiceProvider);            var d = s.DisableOptimizations();            Assert.IsTrue(d is IServiceProvider);            Assert.IsNull(((IServiceProvider)d).GetService(typeof(bool)));        }        class MyScheduler : IScheduler        {            public MyScheduler()                : this(DateTimeOffset.Now)            {            }            public MyScheduler(DateTimeOffset now)            {                Now = now;            }            public DateTimeOffset Now            {                get;                private set;            }            public IDisposable Schedule<T>(T state, Func<IScheduler, T, IDisposable> action)            {                return action(this, state);            }            public Action<Action<object>, object, TimeSpan> Check { get; set; }            public long WaitCycles { get; set; }            public IDisposable Schedule<T>(T state, TimeSpan dueTime, Func<IScheduler, T, IDisposable> action)            {                Check(o => action(this, (T)o), state, dueTime);                WaitCycles += dueTime.Ticks;                return action(this, state);            }            public IDisposable Schedule<T>(T state, DateTimeOffset dueTime, Func<IScheduler, T, IDisposable> action)            {                return Schedule(state, dueTime - Now, action);            }        }        #endregion        #region Catch        [TestMethod]        public void Catch_ArgumentChecking()        {            ReactiveAssert.Throws<ArgumentNullException>(() => Scheduler.Catch<Exception>(default(IScheduler), _ => true));            ReactiveAssert.Throws<ArgumentNullException>(() => Scheduler.Catch<Exception>(Scheduler.Default, default(Func<Exception, bool>)));            ReactiveAssert.Throws<ArgumentNullException>(() => Scheduler.Catch<Exception>(Scheduler.Default, _ => true).Schedule(42, default(Func<IScheduler, int, IDisposable>)));            ReactiveAssert.Throws<ArgumentNullException>(() => Scheduler.Catch<Exception>(Scheduler.Default, _ => true).Schedule(42, TimeSpan.FromSeconds(1), default(Func<IScheduler, int, IDisposable>)));            ReactiveAssert.Throws<ArgumentNullException>(() => Scheduler.Catch<Exception>(Scheduler.Default, _ => true).Schedule(42, DateTimeOffset.Now, default(Func<IScheduler, int, IDisposable>)));        }        [TestMethod]        public void Catch_Builtin_Swallow_Shallow()        {            var done = new ManualResetEvent(false);            var swallow = Scheduler.Default.Catch<InvalidOperationException>(_ => { done.Set(); return true; });            swallow.Schedule(() =>            {                throw new InvalidOperationException();            });            done.WaitOne();        }        [TestMethod]        public void Catch_Builtin_Swallow_Recursive()        {            var done = new ManualResetEvent(false);            var swallow = Scheduler.Default.Catch<InvalidOperationException>(_ => { done.Set(); return true; });            swallow.Schedule(42, (self, state) =>            {                return self.Schedule(() =>                {                    throw new InvalidOperationException();                });            });            done.WaitOne();        }        [TestMethod]        public void Catch_Custom_Unhandled()        {            var err = default(Exception);            var scheduler = new MyExceptionScheduler(ex_ => err = ex_);            scheduler.Catch<InvalidOperationException>(_ => true).Schedule(() =>            {                throw new InvalidOperationException();            });            Assert.IsNull(err);            var ex = new ArgumentException();            scheduler.Catch<InvalidOperationException>(_ => true).Schedule(() =>            {                throw ex;            });            Assert.AreSame(ex, err);        }        [TestMethod]        public void Catch_Custom_Rethrow()        {            var err = default(Exception);            var scheduler = new MyExceptionScheduler(ex_ => err = ex_);            var ex = new InvalidOperationException();            scheduler.Catch<InvalidOperationException>(_ => false).Schedule(() =>            {                throw ex;            });            Assert.AreSame(ex, err);        }        [TestMethod]        public void Catch_Custom_LongRunning_Caught()        {            var err = default(Exception);            var scheduler = new MyExceptionScheduler(ex_ => err = ex_);            var caught = false;            var @catch = scheduler.Catch<InvalidOperationException>(_ => caught = true);            var slr = (ISchedulerLongRunning)((IServiceProvider)@catch).GetService(typeof(ISchedulerLongRunning));            slr.ScheduleLongRunning(cancel =>            {                throw new InvalidOperationException();            });            Assert.IsTrue(caught);            Assert.IsNull(err);            var ex = new ArgumentException();            slr.ScheduleLongRunning(cancel =>            {                throw ex;            });            Assert.AreSame(ex, err);        }        [TestMethod]        public void Catch_Custom_LongRunning_Rethrow()        {            var err = default(Exception);            var scheduler = new MyExceptionScheduler(ex_ => err = ex_);            var @catch = scheduler.Catch<InvalidOperationException>(_ => false);            var slr = (ISchedulerLongRunning)((IServiceProvider)@catch).GetService(typeof(ISchedulerLongRunning));            var done = false;            slr.ScheduleLongRunning(cancel =>            {                done = true;            });            Assert.IsTrue(done);            var ex = new InvalidOperationException();            slr.ScheduleLongRunning(cancel =>            {                throw ex;            });            Assert.AreSame(ex, err);        }        [TestMethod]        public void Catch_Custom_Periodic_Regular()        {            var scheduler = new MyExceptionScheduler(_ => { });            scheduler.PeriodicStopped = new ManualResetEvent(false);            var @catch = scheduler.Catch<InvalidOperationException>(_ => true);            var per = (ISchedulerPeriodic)((IServiceProvider)@catch).GetService(typeof(ISchedulerPeriodic));            var madeProgress = new ManualResetEvent(false);            var d = per.SchedulePeriodic(0, TimeSpan.Zero, x =>            {                if (x > 10)                    madeProgress.Set();                return x + 1;            });            madeProgress.WaitOne();            d.Dispose();            scheduler.PeriodicStopped.WaitOne();        }        [TestMethod]        public void Catch_Custom_Periodic_Uncaught1()        {            var err = default(Exception);            var done = new ManualResetEvent(false);            var scheduler = new MyExceptionScheduler(ex_ => { err = ex_; done.Set(); });            scheduler.PeriodicStopped = new ManualResetEvent(false);            var @catch = scheduler.Catch<InvalidOperationException>(_ => true);            var per = (ISchedulerPeriodic)((IServiceProvider)@catch).GetService(typeof(ISchedulerPeriodic));            var ex = new ArgumentException();            per.SchedulePeriodic(42, TimeSpan.Zero, x =>            {                throw ex;            });            scheduler.PeriodicStopped.WaitOne();            done.WaitOne();            Assert.AreSame(ex, err);        }        [TestMethod]        public void Catch_Custom_Periodic_Uncaught2()        {            var err = default(Exception);            var done = new ManualResetEvent(false);            var scheduler = new MyExceptionScheduler(ex_ => { err = ex_; done.Set(); });            scheduler.PeriodicStopped = new ManualResetEvent(false);            var @catch = scheduler.Catch<InvalidOperationException>(_ => false);            var per = (ISchedulerPeriodic)((IServiceProvider)@catch).GetService(typeof(ISchedulerPeriodic));            var ex = new InvalidOperationException();            per.SchedulePeriodic(42, TimeSpan.Zero, x =>            {                throw ex;            });            scheduler.PeriodicStopped.WaitOne();            done.WaitOne();            Assert.AreSame(ex, err);        }        [TestMethod]        public void Catch_Custom_Periodic_Caught()        {            var err = default(Exception);            var scheduler = new MyExceptionScheduler(ex_ => err = ex_);            scheduler.PeriodicStopped = new ManualResetEvent(false);            var caught = new ManualResetEvent(false);            var @catch = scheduler.Catch<InvalidOperationException>(_ => { caught.Set(); return true; });            var per = (ISchedulerPeriodic)((IServiceProvider)@catch).GetService(typeof(ISchedulerPeriodic));            per.SchedulePeriodic(42, TimeSpan.Zero, x =>            {                throw new InvalidOperationException();            });            scheduler.PeriodicStopped.WaitOne();            caught.WaitOne();            Assert.IsNull(err);        }        class MyExceptionScheduler : LocalScheduler, ISchedulerLongRunning, ISchedulerPeriodic        {            private readonly Action<Exception> _onError;            public MyExceptionScheduler(Action<Exception> onError)            {                _onError = onError;            }            public override IDisposable Schedule<TState>(TState state, Func<IScheduler, TState, IDisposable> action)            {                try                {                    return action(this, state);                }                catch (Exception exception)                {                    _onError(exception);                    return Disposable.Empty;                }            }            public override IDisposable Schedule<TState>(TState state, TimeSpan dueTime, Func<IScheduler, TState, IDisposable> action)            {                throw new NotImplementedException();            }            public IDisposable ScheduleLongRunning<TState>(TState state, Action<TState, ICancelable> action)            {                var b = new BooleanDisposable();                try                {                    action(state, b);                }                catch (Exception exception)                {                    _onError(exception);                    return Disposable.Empty;                }                return b;            }            public ManualResetEvent PeriodicStopped { get; set; }            public IDisposable SchedulePeriodic<TState>(TState state, TimeSpan period, Func<TState, TState> action)            {                var b = new BooleanDisposable();                Scheduler.Default.Schedule(() =>                {                    try                    {                        var s = state;                        for (int i = 0; true; i++)                        {                            if (i > 100 /* mimic delayed cancellation */ && b.IsDisposed)                                break;                            s = action(s);                        }                    }                    catch (Exception exception)                    {                        _onError(exception);                    }                    finally                    {                        PeriodicStopped.Set();                    }                });                return b;            }        }        #endregion        #region Services        [TestMethod]        public void InvalidService_Null()        {            var s = new MySchedulerWithoutServices();            Assert.IsNull(((IServiceProvider)s).GetService(typeof(IAsyncResult)));        }        class MySchedulerWithoutServices : LocalScheduler        {            public override IDisposable Schedule<TState>(TState state, TimeSpan dueTime, Func<IScheduler, TState, IDisposable> action)            {                throw new NotImplementedException();            }        }        [TestMethod]        public void DetectServices_Null_1()        {            var s = new MyDumbScheduler1();            Assert.IsNull(Scheduler.AsLongRunning(s));            Assert.IsNull(Scheduler.AsPeriodic(s));            Assert.IsNull(Scheduler.AsStopwatchProvider(s));        }        class MyDumbScheduler1 : IScheduler        {            public DateTimeOffset Now            {                get { throw new NotImplementedException(); }            }            public IDisposable Schedule<TState>(TState state, Func<IScheduler, TState, IDisposable> action)            {                throw new NotImplementedException();            }            public IDisposable Schedule<TState>(TState state, TimeSpan dueTime, Func<IScheduler, TState, IDisposable> action)            {                throw new NotImplementedException();            }            public IDisposable Schedule<TState>(TState state, DateTimeOffset dueTime, Func<IScheduler, TState, IDisposable> action)            {                throw new NotImplementedException();            }        }        [TestMethod]        public void DetectServices_Null_2()        {            var s = new MyDumbScheduler2(new Dictionary<Type, object>());            Assert.IsNull(Scheduler.AsLongRunning(s));            Assert.IsNull(Scheduler.AsPeriodic(s));            Assert.IsNull(Scheduler.AsStopwatchProvider(s));        }        [TestMethod]        public void DetectServices_Found()        {            {                var x = new MyLongRunning();                var s = new MyDumbScheduler2(new Dictionary<Type, object>                {                    { typeof(ISchedulerLongRunning), x }                });                Assert.AreEqual(x, Scheduler.AsLongRunning(s));            }            {                var x = new MyStopwatchProvider();                var s = new MyDumbScheduler2(new Dictionary<Type, object>                {                    { typeof(IStopwatchProvider), x }                });                Assert.AreEqual(x, Scheduler.AsStopwatchProvider(s));            }            {                var x = new MyPeriodic();                var s = new MyDumbScheduler2(new Dictionary<Type, object>                {                    { typeof(ISchedulerPeriodic), x }                });                Assert.AreEqual(x, Scheduler.AsPeriodic(s));            }        }        class MyDumbScheduler2 : IScheduler, IServiceProvider        {            private readonly Dictionary<Type, object> _services;            public MyDumbScheduler2(Dictionary<Type, object> services)            {                _services = services;            }            public DateTimeOffset Now            {                get { throw new NotImplementedException(); }            }            public IDisposable Schedule<TState>(TState state, Func<IScheduler, TState, IDisposable> action)            {                throw new NotImplementedException();            }            public IDisposable Schedule<TState>(TState state, TimeSpan dueTime, Func<IScheduler, TState, IDisposable> action)            {                throw new NotImplementedException();            }            public IDisposable Schedule<TState>(TState state, DateTimeOffset dueTime, Func<IScheduler, TState, IDisposable> action)            {                throw new NotImplementedException();            }            public object GetService(Type serviceType)            {                var res = default(object);                if (_services.TryGetValue(serviceType, out res))                    return res;                return null;            }        }        class MyLongRunning : ISchedulerLongRunning        {            public IDisposable ScheduleLongRunning<TState>(TState state, Action<TState, ICancelable> action)            {                throw new NotImplementedException();            }        }        class MyStopwatchProvider : IStopwatchProvider        {            public IStopwatch StartStopwatch()            {                throw new NotImplementedException();            }        }        class MyPeriodic : ISchedulerPeriodic        {            public IDisposable SchedulePeriodic<TState>(TState state, TimeSpan period, Func<TState, TState> action)            {                throw new NotImplementedException();            }        }        #endregion    }}
 |