AnsiRequestSchedulerTests.cs 4.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153
  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>> (), 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>> (), 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>> (), 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. private void SetTime (int milliseconds)
  111. {
  112. // This simulates the passing of time by setting the Now function to return a specific time.
  113. var newNow = _staticNow.AddMilliseconds (milliseconds);
  114. _scheduler.Now = () => newNow;
  115. }
  116. }