// Copyright (c) Microsoft Open Technologies, Inc. All rights reserved. See License.txt in the project root for license information. using System; using System.Diagnostics; using System.Reactive.Concurrency; using System.Reactive.Disposables; using System.Threading; using Microsoft.Reactive.Testing; #if NUNIT using NUnit.Framework; using TestClassAttribute = NUnit.Framework.TestFixtureAttribute; using TestMethodAttribute = NUnit.Framework.TestAttribute; using TestInitializeAttribute = NUnit.Framework.SetUpAttribute; #else using Microsoft.VisualStudio.TestTools.UnitTesting; #endif namespace ReactiveTests.Tests { [TestClass] public class NewThreadSchedulerTest { [TestMethod] public void NewThread_ArgumentChecking() { ReactiveAssert.Throws(() => new NewThreadScheduler(null)); ReactiveAssert.Throws(() => NewThreadScheduler.Default.Schedule(42, default(Func))); ReactiveAssert.Throws(() => NewThreadScheduler.Default.Schedule(42, DateTimeOffset.Now, default(Func))); ReactiveAssert.Throws(() => NewThreadScheduler.Default.Schedule(42, TimeSpan.Zero, default(Func))); ReactiveAssert.Throws(() => NewThreadScheduler.Default.SchedulePeriodic(42, TimeSpan.FromSeconds(1), default(Func))); ReactiveAssert.Throws(() => NewThreadScheduler.Default.ScheduleLongRunning(42, default(Action))); ReactiveAssert.Throws(() => NewThreadScheduler.Default.SchedulePeriodic(42, TimeSpan.FromSeconds(-1), _ => _)); } [TestMethod] public void NewThread_Now() { var res = NewThreadScheduler.Default.Now - DateTime.Now; Assert.IsTrue(res.Seconds < 1); } [TestMethod] public void NewThread_ScheduleAction() { var id = Thread.CurrentThread.ManagedThreadId; var nt = NewThreadScheduler.Default; var evt = new ManualResetEvent(false); nt.Schedule(() => { Assert.AreNotEqual(id, Thread.CurrentThread.ManagedThreadId); evt.Set(); }); evt.WaitOne(); } #if !SILVERLIGHT [TestMethod] [Ignore] public void NewThread_ScheduleActionDue() { var id = Thread.CurrentThread.ManagedThreadId; var nt = NewThreadScheduler.Default; var evt = new ManualResetEvent(false); var sw = new Stopwatch(); sw.Start(); nt.Schedule(TimeSpan.FromSeconds(0.2), () => { sw.Stop(); Assert.AreNotEqual(id, Thread.CurrentThread.ManagedThreadId); evt.Set(); }); evt.WaitOne(); Assert.IsTrue(sw.ElapsedMilliseconds > 180, "due " + sw.ElapsedMilliseconds); } #endif #if !NO_PERF #if !NO_STOPWATCH [TestMethod] public void Stopwatch() { StopwatchTest.Run(NewThreadScheduler.Default); } #endif #endif [TestMethod] public void NewThread_Periodic() { var n = 0; var e = new ManualResetEvent(false); var d = NewThreadScheduler.Default.SchedulePeriodic(TimeSpan.FromMilliseconds(25), () => { if (Interlocked.Increment(ref n) == 10) e.Set(); }); if (!e.WaitOne(10000)) Assert.Fail(); d.Dispose(); } [TestMethod] public void NewThread_Periodic_NonReentrant() { var n = 0; var fail = false; var d = NewThreadScheduler.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.IsFalse(fail); } } }