EventLoopSchedulerTest.cs 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374
  1. // Copyright (c) Microsoft Open Technologies, Inc. All rights reserved. See License.txt in the project root for license information.
  2. using System;
  3. using System.Collections.Generic;
  4. using System.Diagnostics;
  5. using System.Reactive.Concurrency;
  6. using System.Threading;
  7. using Microsoft.Reactive.Testing;
  8. using Microsoft.VisualStudio.TestTools.UnitTesting;
  9. namespace ReactiveTests.Tests
  10. {
  11. [TestClass]
  12. public class EventLoopSchedulerTest
  13. {
  14. [TestMethod]
  15. public void EventLoop_ArgumentChecking()
  16. {
  17. var el = new EventLoopScheduler();
  18. ReactiveAssert.Throws<ArgumentNullException>(() => new EventLoopScheduler(null));
  19. ReactiveAssert.Throws<ArgumentNullException>(() => el.Schedule<int>(42, default(Func<IScheduler, int, IDisposable>)));
  20. ReactiveAssert.Throws<ArgumentNullException>(() => el.Schedule<int>(42, DateTimeOffset.Now, default(Func<IScheduler, int, IDisposable>)));
  21. ReactiveAssert.Throws<ArgumentNullException>(() => el.Schedule<int>(42, TimeSpan.Zero, default(Func<IScheduler, int, IDisposable>)));
  22. ReactiveAssert.Throws<ArgumentNullException>(() => el.SchedulePeriodic(42, TimeSpan.FromSeconds(1), default(Func<int, int>)));
  23. ReactiveAssert.Throws<ArgumentOutOfRangeException>(() => el.SchedulePeriodic(42, TimeSpan.FromSeconds(-1), _ => _));
  24. }
  25. [TestMethod]
  26. public void EventLoop_Now()
  27. {
  28. var res = new EventLoopScheduler().Now - DateTime.Now;
  29. Assert.IsTrue(res.Seconds < 1);
  30. }
  31. [TestMethod]
  32. public void EventLoop_ScheduleAction()
  33. {
  34. var ran = false;
  35. var gate = new Semaphore(0, 1);
  36. var el = new EventLoopScheduler();
  37. el.Schedule(() => { ran = true;
  38. gate.Release(); });
  39. Assert.IsTrue(gate.WaitOne(TimeSpan.FromSeconds(2)));
  40. Assert.IsTrue(ran);
  41. }
  42. [TestMethod]
  43. public void EventLoop_DifferentThread()
  44. {
  45. var id = default(int);
  46. var gate = new Semaphore(0, 1);
  47. var el = new EventLoopScheduler();
  48. el.Schedule(() =>
  49. {
  50. id = Thread.CurrentThread.ManagedThreadId;
  51. gate.Release();
  52. });
  53. Assert.IsTrue(gate.WaitOne(TimeSpan.FromSeconds(2)));
  54. Assert.AreNotEqual(Thread.CurrentThread.ManagedThreadId, id);
  55. }
  56. [TestMethod]
  57. public void EventLoop_ScheduleOrderedActions()
  58. {
  59. var results = new List<int>();
  60. var gate = new Semaphore(0, 1);
  61. var el = new EventLoopScheduler();
  62. el.Schedule(() => results.Add(0));
  63. el.Schedule(() =>
  64. {
  65. results.Add(1);
  66. gate.Release();
  67. });
  68. Assert.IsTrue(gate.WaitOne(TimeSpan.FromSeconds(2)));
  69. results.AssertEqual(0, 1);
  70. }
  71. [TestMethod]
  72. public void EventLoop_SchedulerDisposed()
  73. {
  74. var d = 0;
  75. var e = new ManualResetEvent(false);
  76. var f = new ManualResetEvent(false);
  77. var results = new List<int>();
  78. var el = new EventLoopScheduler();
  79. el.Schedule(() => results.Add(0));
  80. el.Schedule(() =>
  81. {
  82. el.Dispose();
  83. e.Set();
  84. results.Add(1);
  85. try
  86. {
  87. el.Schedule(() => { throw new Exception("Should be disposed!"); });
  88. f.Set();
  89. }
  90. catch (ObjectDisposedException)
  91. {
  92. // BREAKING CHANGE v2 > v1.x - New exception behavior.
  93. Interlocked.Increment(ref d);
  94. f.Set();
  95. }
  96. });
  97. e.WaitOne();
  98. try
  99. {
  100. el.Schedule(() => results.Add(2));
  101. }
  102. catch (ObjectDisposedException)
  103. {
  104. // BREAKING CHANGE v2 > v1.x - New exception behavior.
  105. Interlocked.Increment(ref d);
  106. }
  107. f.WaitOne();
  108. results.AssertEqual(0, 1);
  109. Assert.AreEqual(2, d);
  110. }
  111. [TestMethod]
  112. public void EventLoop_ScheduleTimeOrderedActions()
  113. {
  114. var results = new List<int>();
  115. var gate = new Semaphore(0, 1);
  116. var el = new EventLoopScheduler();
  117. el.Schedule(TimeSpan.FromMilliseconds(50), () => results.Add(1));
  118. el.Schedule(TimeSpan.FromMilliseconds(100), () =>
  119. {
  120. results.Add(0);
  121. gate.Release();
  122. });
  123. Assert.IsTrue(gate.WaitOne(TimeSpan.FromSeconds(2)));
  124. results.AssertEqual(1, 0);
  125. }
  126. [TestMethod]
  127. public void EventLoop_ScheduleOrderedAndTimedActions()
  128. {
  129. var results = new List<int>();
  130. var gate = new Semaphore(0, 1);
  131. var el = new EventLoopScheduler();
  132. el.Schedule(() => results.Add(1));
  133. el.Schedule(TimeSpan.FromMilliseconds(100), () =>
  134. {
  135. results.Add(0);
  136. gate.Release();
  137. });
  138. Assert.IsTrue(gate.WaitOne(TimeSpan.FromSeconds(2)));
  139. results.AssertEqual(1, 0);
  140. }
  141. [TestMethod]
  142. public void EventLoop_ScheduleTimeOrderedInFlightActions()
  143. {
  144. var results = new List<int>();
  145. var gate = new Semaphore(0, 1);
  146. var el = new EventLoopScheduler();
  147. el.Schedule(TimeSpan.FromMilliseconds(100), () =>
  148. {
  149. results.Add(0);
  150. el.Schedule(TimeSpan.FromMilliseconds(50), () => results.Add(1));
  151. el.Schedule(TimeSpan.FromMilliseconds(100), ()=>
  152. {
  153. results.Add(2);
  154. gate.Release();
  155. });
  156. });
  157. Assert.IsTrue(gate.WaitOne(TimeSpan.FromSeconds(2)));
  158. results.AssertEqual(0, 1, 2);
  159. }
  160. [TestMethod]
  161. public void EventLoop_ScheduleTimeAndOrderedInFlightActions()
  162. {
  163. var results = new List<int>();
  164. var gate = new Semaphore(0, 1);
  165. var el = new EventLoopScheduler();
  166. el.Schedule(TimeSpan.FromMilliseconds(100), () =>
  167. {
  168. results.Add(0);
  169. el.Schedule(() => results.Add(4));
  170. el.Schedule(TimeSpan.FromMilliseconds(50), () => results.Add(1));
  171. el.Schedule(TimeSpan.FromMilliseconds(100), () =>
  172. {
  173. results.Add(2);
  174. gate.Release();
  175. });
  176. });
  177. Assert.IsTrue(gate.WaitOne(TimeSpan.FromSeconds(2)));
  178. results.AssertEqual(0, 4, 1, 2);
  179. }
  180. [TestMethod]
  181. public void EventLoop_ScheduleActionNested()
  182. {
  183. var ran = false;
  184. var el = new EventLoopScheduler();
  185. var gate = new Semaphore(0, 1);
  186. el.Schedule(() => el.Schedule(() => { ran = true;
  187. gate.Release(); }));
  188. Assert.IsTrue(gate.WaitOne(TimeSpan.FromSeconds(2)));
  189. Assert.IsTrue(ran);
  190. }
  191. #if !SILVERLIGHT
  192. [TestMethod]
  193. [Ignore]
  194. public void EventLoop_ScheduleActionDue()
  195. {
  196. var ran = false;
  197. var el = new EventLoopScheduler();
  198. var sw = new Stopwatch();
  199. var gate = new Semaphore(0, 1);
  200. sw.Start();
  201. el.Schedule(TimeSpan.FromSeconds(0.2), () => {
  202. ran = true;
  203. sw.Stop();
  204. gate.Release();
  205. });
  206. Assert.IsTrue(gate.WaitOne(TimeSpan.FromSeconds(2)));
  207. Assert.IsTrue(ran, "ran");
  208. Assert.IsTrue(sw.ElapsedMilliseconds > 180, "due " + sw.ElapsedMilliseconds);
  209. }
  210. [TestMethod]
  211. [Ignore]
  212. public void EventLoop_ScheduleActionDueNested()
  213. {
  214. var ran = false;
  215. var el = new EventLoopScheduler();
  216. var gate = new Semaphore(0, 1);
  217. var sw = new Stopwatch();
  218. sw.Start();
  219. el.Schedule(TimeSpan.FromSeconds(0.2), () =>
  220. {
  221. sw.Stop();
  222. sw.Start();
  223. el.Schedule(TimeSpan.FromSeconds(0.2), () =>
  224. {
  225. sw.Stop();
  226. ran = true;
  227. gate.Release();
  228. });
  229. });
  230. Assert.IsTrue(gate.WaitOne(TimeSpan.FromSeconds(2)));
  231. Assert.IsTrue(ran, "ran");
  232. Assert.IsTrue(sw.ElapsedMilliseconds > 380, "due " + sw.ElapsedMilliseconds);
  233. }
  234. #endif
  235. #if !NO_PERF
  236. #if !NO_STOPWATCH
  237. [TestMethod]
  238. public void Stopwatch()
  239. {
  240. StopwatchTest.Run(new EventLoopScheduler());
  241. }
  242. #endif
  243. #endif
  244. #if !NO_CDS
  245. [TestMethod]
  246. public void EventLoop_Immediate()
  247. {
  248. var M = 1000;
  249. var N = 4;
  250. for (int i = 0; i < N; i++)
  251. {
  252. for (int j = 1; j <= M; j *= 10)
  253. {
  254. using (var e = new EventLoopScheduler())
  255. {
  256. var cd = new CountdownEvent(j);
  257. for (int k = 0; k < j; k++)
  258. e.Schedule(() => cd.Signal());
  259. if (!cd.Wait(10000))
  260. Assert.Fail("j = " + j);
  261. }
  262. }
  263. }
  264. }
  265. [TestMethod]
  266. public void EventLoop_TimeCollisions()
  267. {
  268. var M = 1000;
  269. var N = 4;
  270. for (int i = 0; i < N; i++)
  271. {
  272. for (int j = 1; j <= M; j *= 10)
  273. {
  274. using (var e = new EventLoopScheduler())
  275. {
  276. var cd = new CountdownEvent(j);
  277. for (int k = 0; k < j; k++)
  278. e.Schedule(TimeSpan.FromMilliseconds(100), () => cd.Signal());
  279. if (!cd.Wait(10000))
  280. Assert.Fail("j = " + j);
  281. }
  282. }
  283. }
  284. }
  285. [TestMethod]
  286. public void EventLoop_Spread()
  287. {
  288. var M = 1000;
  289. var N = 4;
  290. for (int i = 0; i < N; i++)
  291. {
  292. for (int j = 1; j <= M; j *= 10)
  293. {
  294. using (var e = new EventLoopScheduler())
  295. {
  296. var cd = new CountdownEvent(j);
  297. for (int k = 0; k < j; k++)
  298. e.Schedule(TimeSpan.FromMilliseconds(k), () => cd.Signal());
  299. if (!cd.Wait(10000))
  300. Assert.Fail("j = " + j);
  301. }
  302. }
  303. }
  304. }
  305. #endif
  306. [TestMethod]
  307. public void EventLoop_Periodic()
  308. {
  309. var n = 0;
  310. using (var s = new EventLoopScheduler())
  311. {
  312. var e = new ManualResetEvent(false);
  313. var d = s.SchedulePeriodic(TimeSpan.FromMilliseconds(25), () =>
  314. {
  315. if (Interlocked.Increment(ref n) == 10)
  316. e.Set();
  317. });
  318. if (!e.WaitOne(10000))
  319. Assert.Fail();
  320. d.Dispose();
  321. }
  322. }
  323. }
  324. }