AnsiRequestSchedulerTests.cs 7.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236
  1. using Moq;
  2. namespace UnitTests.ConsoleDrivers;
  3. public class AnsiRequestSchedulerTests
  4. {
  5. private readonly Mock<IAnsiResponseParser> _parserMock;
  6. private readonly AnsiRequestScheduler _scheduler;
  7. private static DateTime _staticNow; // Static value to hold the current time
  8. public AnsiRequestSchedulerTests ()
  9. {
  10. _parserMock = new Mock<IAnsiResponseParser> (MockBehavior.Strict);
  11. _staticNow = DateTime.UtcNow; // Initialize static time
  12. _scheduler = new AnsiRequestScheduler (_parserMock.Object, () => _staticNow);
  13. }
  14. [Fact]
  15. public void SendOrSchedule_SendsDeviceAttributeRequest_WhenNoOutstandingRequests ()
  16. {
  17. // Arrange
  18. var request = new AnsiEscapeSequenceRequest
  19. {
  20. Request = "\u001b[0c", // ESC [ c
  21. Terminator = "c",
  22. ResponseReceived = r => { }
  23. };
  24. // we have no outstanding for c already
  25. _parserMock.Setup (p => p.IsExpecting ("c")).Returns (false).Verifiable(Times.Once);
  26. // then we should execute our request
  27. _parserMock.Setup (p => p.ExpectResponse ("c", It.IsAny<Action<string>> (), null, false)).Verifiable (Times.Once);
  28. // Act
  29. bool result = _scheduler.SendOrSchedule (request);
  30. // Assert
  31. Assert.Empty (_scheduler.QueuedRequests); // We sent it i.e. we did not queue it for later
  32. Assert.True (result); // Should send immediately
  33. _parserMock.Verify ();
  34. }
  35. [Fact]
  36. public void SendOrSchedule_QueuesRequest_WhenOutstandingRequestExists ()
  37. {
  38. // Arrange
  39. var request1 = new AnsiEscapeSequenceRequest
  40. {
  41. Request = "\u001b[0c", // ESC [ 0 c
  42. Terminator = "c",
  43. ResponseReceived = r => { }
  44. };
  45. // Parser already has an ongoing request for "c"
  46. _parserMock.Setup (p => p.IsExpecting ("c")).Returns (true).Verifiable (Times.Once);
  47. // Act
  48. var result = _scheduler.SendOrSchedule (request1);
  49. // Assert
  50. Assert.Single (_scheduler.QueuedRequests); // Ensure only one request is in the queue
  51. Assert.False (result); // Should be queued
  52. _parserMock.Verify ();
  53. }
  54. [Fact]
  55. public void RunSchedule_ThrottleNotExceeded_AllowSend ()
  56. {
  57. // Arrange
  58. var request = new AnsiEscapeSequenceRequest
  59. {
  60. Request = "\u001b[0c", // ESC [ 0 c
  61. Terminator = "c",
  62. ResponseReceived = r => { }
  63. };
  64. // Set up to expect no outstanding request for "c" i.e. parser instantly gets response and resolves it
  65. _parserMock.Setup (p => p.IsExpecting ("c")).Returns (false).Verifiable(Times.Exactly (2));
  66. _parserMock.Setup (p => p.ExpectResponse ("c", It.IsAny<Action<string>> (), null, false)).Verifiable (Times.Exactly (2));
  67. _scheduler.SendOrSchedule (request);
  68. // Simulate time passing beyond throttle
  69. SetTime (101); // Exceed throttle limit
  70. // Act
  71. // Send another request after the throttled time limit
  72. var result = _scheduler.SendOrSchedule (request);
  73. // Assert
  74. Assert.Empty (_scheduler.QueuedRequests); // Should send and clear the request
  75. Assert.True (result); // Should have found and sent the request
  76. _parserMock.Verify ();
  77. }
  78. [Fact]
  79. public void RunSchedule_ThrottleExceeded_QueueRequest ()
  80. {
  81. // Arrange
  82. var request = new AnsiEscapeSequenceRequest
  83. {
  84. Request = "\u001b[0c", // ESC [ 0 c
  85. Terminator = "c",
  86. ResponseReceived = r => { }
  87. };
  88. // Set up to expect no outstanding request for "c" i.e. parser instantly gets response and resolves it
  89. _parserMock.Setup (p => p.IsExpecting ("c")).Returns (false).Verifiable (Times.Exactly (2));
  90. _parserMock.Setup (p => p.ExpectResponse ("c", It.IsAny<Action<string>> (), null, false)).Verifiable (Times.Exactly (2));
  91. _scheduler.SendOrSchedule (request);
  92. // Simulate time passing
  93. SetTime (55); // Does not exceed throttle limit
  94. // Act
  95. // Send another request after the throttled time limit
  96. var result = _scheduler.SendOrSchedule (request);
  97. // Assert
  98. Assert.Single (_scheduler.QueuedRequests); // Should have been queued
  99. Assert.False(result); // Should have been queued
  100. // Throttle still not exceeded
  101. Assert.False(_scheduler.RunSchedule ());
  102. SetTime (90);
  103. // Throttle still not exceeded
  104. Assert.False (_scheduler.RunSchedule ());
  105. SetTime (105);
  106. // Throttle exceeded - so send the request
  107. Assert.True (_scheduler.RunSchedule ());
  108. _parserMock.Verify ();
  109. }
  110. [Fact]
  111. public void EvictStaleRequests_RemovesStaleRequest_AfterTimeout ()
  112. {
  113. // Arrange
  114. var request1 = new AnsiEscapeSequenceRequest
  115. {
  116. Request = "\u001b[0c",
  117. Terminator = "c",
  118. ResponseReceived = r => { }
  119. };
  120. // Send
  121. _parserMock.Setup (p => p.IsExpecting ("c")).Returns (false).Verifiable (Times.Once);
  122. _parserMock.Setup (p => p.ExpectResponse ("c", It.IsAny<Action<string>> (), null, false)).Verifiable (Times.Exactly (2));
  123. Assert.True (_scheduler.SendOrSchedule (request1));
  124. // Parser already has an ongoing request for "c"
  125. _parserMock.Setup (p => p.IsExpecting ("c")).Returns (true).Verifiable (Times.Exactly (2));
  126. // Cannot send because there is already outstanding request
  127. Assert.False(_scheduler.SendOrSchedule (request1));
  128. Assert.Single (_scheduler.QueuedRequests);
  129. // Simulate request going stale
  130. SetTime (5001); // Exceeds stale timeout
  131. // Parser should be told to give up on this one (evicted)
  132. _parserMock.Setup (p => p.StopExpecting ("c", false))
  133. .Callback (() =>
  134. {
  135. // When we tell parser to evict - it should now tell us it is no longer expecting
  136. _parserMock.Setup (p => p.IsExpecting ("c")).Returns (false).Verifiable (Times.Once);
  137. }).Verifiable ();
  138. // When we send again the evicted one should be
  139. var evicted = _scheduler.RunSchedule ();
  140. Assert.True (evicted); // Stale request should be evicted
  141. Assert.Empty (_scheduler.QueuedRequests);
  142. // Assert
  143. _parserMock.Verify ();
  144. }
  145. [Fact]
  146. public void RunSchedule_DoesNothing_WhenQueueIsEmpty ()
  147. {
  148. // Act
  149. var result = _scheduler.RunSchedule ();
  150. // Assert
  151. Assert.False (result); // No requests to process
  152. Assert.Empty (_scheduler.QueuedRequests);
  153. }
  154. [Fact]
  155. public void SendOrSchedule_ManagesIndependentTerminatorsCorrectly ()
  156. {
  157. // Arrange
  158. var request1 = new AnsiEscapeSequenceRequest { Request = "\u001b[0c", Terminator = "c", ResponseReceived = r => { } };
  159. var request2 = new AnsiEscapeSequenceRequest { Request = "\u001b[0x", Terminator = "x", ResponseReceived = r => { } };
  160. // Already have a 'c' ongoing
  161. _parserMock.Setup (p => p.IsExpecting ("c")).Returns (true).Verifiable (Times.Once);
  162. // 'x' is free
  163. _parserMock.Setup (p => p.IsExpecting ("x")).Returns (false).Verifiable (Times.Once);
  164. _parserMock.Setup (p => p.ExpectResponse ("x", It.IsAny<Action<string>> (), null, false)).Verifiable (Times.Once);
  165. // Act
  166. var a = _scheduler.SendOrSchedule (request1);
  167. var b = _scheduler.SendOrSchedule (request2);
  168. // Assert
  169. Assert.False (a);
  170. Assert.True (b);
  171. Assert.Equal(request1, Assert.Single (_scheduler.QueuedRequests));
  172. _parserMock.Verify ();
  173. }
  174. private void SetTime (int milliseconds)
  175. {
  176. // This simulates the passing of time by setting the Now function to return a specific time.
  177. var newNow = _staticNow.AddMilliseconds (milliseconds);
  178. _scheduler.Now = () => newNow;
  179. }
  180. }