|  | @@ -18,7 +18,6 @@ namespace Renci.SshNet.Tests.Classes
 | 
											
												
													
														|  |      public class ShellStreamTest_ReadExpect
 |  |      public class ShellStreamTest_ReadExpect
 | 
											
												
													
														|  |      {
 |  |      {
 | 
											
												
													
														|  |          private const int BufferSize = 1024;
 |  |          private const int BufferSize = 1024;
 | 
											
												
													
														|  | -        private const int ExpectSize = BufferSize * 2;
 |  | 
 | 
											
												
													
														|  |          private ShellStream _shellStream;
 |  |          private ShellStream _shellStream;
 | 
											
												
													
														|  |          private ChannelSessionStub _channelSessionStub;
 |  |          private ChannelSessionStub _channelSessionStub;
 | 
											
												
													
														|  |  
 |  |  
 | 
											
										
											
												
													
														|  | @@ -44,8 +43,7 @@ namespace Renci.SshNet.Tests.Classes
 | 
											
												
													
														|  |                  width: 800,
 |  |                  width: 800,
 | 
											
												
													
														|  |                  height: 600,
 |  |                  height: 600,
 | 
											
												
													
														|  |                  terminalModeValues: null,
 |  |                  terminalModeValues: null,
 | 
											
												
													
														|  | -                bufferSize: BufferSize,
 |  | 
 | 
											
												
													
														|  | -                expectSize: ExpectSize);
 |  | 
 | 
											
												
													
														|  | 
 |  | +                bufferSize: BufferSize);
 | 
											
												
													
														|  |          }
 |  |          }
 | 
											
												
													
														|  |  
 |  |  
 | 
											
												
													
														|  |          [TestMethod]
 |  |          [TestMethod]
 | 
											
										
											
												
													
														|  | @@ -74,26 +72,48 @@ namespace Renci.SshNet.Tests.Classes
 | 
											
												
													
														|  |  
 |  |  
 | 
											
												
													
														|  |          [DataTestMethod]
 |  |          [DataTestMethod]
 | 
											
												
													
														|  |          [DataRow("\r\n")]
 |  |          [DataRow("\r\n")]
 | 
											
												
													
														|  | -        //[DataRow("\r")] These currently fail.
 |  | 
 | 
											
												
													
														|  | -        //[DataRow("\n")]
 |  | 
 | 
											
												
													
														|  | 
 |  | +        [DataRow("\r")]
 | 
											
												
													
														|  | 
 |  | +        [DataRow("\n")]
 | 
											
												
													
														|  |          public void ReadLine(string newLine)
 |  |          public void ReadLine(string newLine)
 | 
											
												
													
														|  |          {
 |  |          {
 | 
											
												
													
														|  |              _channelSessionStub.Receive(Encoding.UTF8.GetBytes("Hello "));
 |  |              _channelSessionStub.Receive(Encoding.UTF8.GetBytes("Hello "));
 | 
											
												
													
														|  |              _channelSessionStub.Receive(Encoding.UTF8.GetBytes("World!"));
 |  |              _channelSessionStub.Receive(Encoding.UTF8.GetBytes("World!"));
 | 
											
												
													
														|  |  
 |  |  
 | 
											
												
													
														|  | -            // We specify a nonzero timeout to avoid waiting infinitely.
 |  | 
 | 
											
												
													
														|  | -            Assert.IsNull(_shellStream.ReadLine(TimeSpan.FromTicks(1)));
 |  | 
 | 
											
												
													
														|  | 
 |  | +            // We specify a timeout to avoid waiting infinitely.
 | 
											
												
													
														|  | 
 |  | +            Assert.IsNull(_shellStream.ReadLine(TimeSpan.Zero));
 | 
											
												
													
														|  |  
 |  |  
 | 
											
												
													
														|  |              _channelSessionStub.Receive(Encoding.UTF8.GetBytes(newLine));
 |  |              _channelSessionStub.Receive(Encoding.UTF8.GetBytes(newLine));
 | 
											
												
													
														|  |  
 |  |  
 | 
											
												
													
														|  | -            Assert.AreEqual("Hello World!", _shellStream.ReadLine(TimeSpan.FromTicks(1)));
 |  | 
 | 
											
												
													
														|  | -            Assert.IsNull(_shellStream.ReadLine(TimeSpan.FromTicks(1)));
 |  | 
 | 
											
												
													
														|  | 
 |  | +            Assert.AreEqual("Hello World!", _shellStream.ReadLine(TimeSpan.Zero));
 | 
											
												
													
														|  | 
 |  | +            Assert.IsNull(_shellStream.ReadLine(TimeSpan.Zero));
 | 
											
												
													
														|  |  
 |  |  
 | 
											
												
													
														|  |              _channelSessionStub.Receive(Encoding.UTF8.GetBytes("Second line!" + newLine + "Third line!" + newLine));
 |  |              _channelSessionStub.Receive(Encoding.UTF8.GetBytes("Second line!" + newLine + "Third line!" + newLine));
 | 
											
												
													
														|  |  
 |  |  
 | 
											
												
													
														|  | -            Assert.AreEqual("Second line!", _shellStream.ReadLine(TimeSpan.FromTicks(1)));
 |  | 
 | 
											
												
													
														|  | -            Assert.AreEqual("Third line!", _shellStream.ReadLine(TimeSpan.FromTicks(1)));
 |  | 
 | 
											
												
													
														|  | -            Assert.IsNull(_shellStream.ReadLine(TimeSpan.FromTicks(1)));
 |  | 
 | 
											
												
													
														|  | 
 |  | +            Assert.AreEqual("Second line!", _shellStream.ReadLine(TimeSpan.Zero));
 | 
											
												
													
														|  | 
 |  | +            Assert.AreEqual("Third line!", _shellStream.ReadLine(TimeSpan.Zero));
 | 
											
												
													
														|  | 
 |  | +            Assert.IsNull(_shellStream.ReadLine(TimeSpan.Zero));
 | 
											
												
													
														|  | 
 |  | +
 | 
											
												
													
														|  | 
 |  | +            _channelSessionStub.Receive(Encoding.UTF8.GetBytes("Last line!")); // no newLine at the end
 | 
											
												
													
														|  | 
 |  | +
 | 
											
												
													
														|  | 
 |  | +            Assert.IsNull(_shellStream.ReadLine(TimeSpan.Zero));
 | 
											
												
													
														|  | 
 |  | +
 | 
											
												
													
														|  | 
 |  | +            _channelSessionStub.Close();
 | 
											
												
													
														|  | 
 |  | +
 | 
											
												
													
														|  | 
 |  | +            Assert.AreEqual("Last line!", _shellStream.ReadLine(TimeSpan.Zero));
 | 
											
												
													
														|  | 
 |  | +        }
 | 
											
												
													
														|  | 
 |  | +
 | 
											
												
													
														|  | 
 |  | +        [TestMethod]
 | 
											
												
													
														|  | 
 |  | +        public void ReadLine_DifferentTerminators()
 | 
											
												
													
														|  | 
 |  | +        {
 | 
											
												
													
														|  | 
 |  | +            _channelSessionStub.Receive(Encoding.UTF8.GetBytes("Hello\rWorld!\nWhat's\r\ngoing\n\ron?\n"));
 | 
											
												
													
														|  | 
 |  | +
 | 
											
												
													
														|  | 
 |  | +            Assert.AreEqual("Hello", _shellStream.ReadLine());
 | 
											
												
													
														|  | 
 |  | +            Assert.AreEqual("World!", _shellStream.ReadLine());
 | 
											
												
													
														|  | 
 |  | +            Assert.AreEqual("What's", _shellStream.ReadLine());
 | 
											
												
													
														|  | 
 |  | +            Assert.AreEqual("going", _shellStream.ReadLine());
 | 
											
												
													
														|  | 
 |  | +            Assert.AreEqual("", _shellStream.ReadLine());
 | 
											
												
													
														|  | 
 |  | +            Assert.AreEqual("on?", _shellStream.ReadLine());
 | 
											
												
													
														|  | 
 |  | +            Assert.IsNull(_shellStream.ReadLine(TimeSpan.Zero));
 | 
											
												
													
														|  |          }
 |  |          }
 | 
											
												
													
														|  |  
 |  |  
 | 
											
												
													
														|  |          [DataTestMethod]
 |  |          [DataTestMethod]
 | 
											
										
											
												
													
														|  | @@ -111,54 +131,31 @@ namespace Renci.SshNet.Tests.Classes
 | 
											
												
													
														|  |          }
 |  |          }
 | 
											
												
													
														|  |  
 |  |  
 | 
											
												
													
														|  |          [TestMethod]
 |  |          [TestMethod]
 | 
											
												
													
														|  | -        [Ignore] // Currently returns 0 immediately
 |  | 
 | 
											
												
													
														|  | -        public void Read_NonEmptyArray_OnlyReturnsZeroAfterClose()
 |  | 
 | 
											
												
													
														|  | 
 |  | +        public async Task Read_NonEmptyArray_OnlyReturnsZeroAfterClose()
 | 
											
												
													
														|  |          {
 |  |          {
 | 
											
												
													
														|  | -            Task closeTask = Task.Run(async () =>
 |  | 
 | 
											
												
													
														|  | -            {
 |  | 
 | 
											
												
													
														|  | -                // For the test to have meaning, we should be in
 |  | 
 | 
											
												
													
														|  | -                // the call to Read before closing the channel.
 |  | 
 | 
											
												
													
														|  | -                // Impose a short delay to make that more likely.
 |  | 
 | 
											
												
													
														|  | -                await Task.Delay(50);
 |  | 
 | 
											
												
													
														|  | -
 |  | 
 | 
											
												
													
														|  | -                _channelSessionStub.Close();
 |  | 
 | 
											
												
													
														|  | -            });
 |  | 
 | 
											
												
													
														|  | 
 |  | +            Task<int> readTask = _shellStream.ReadAsync(new byte[16], 0, 16);
 | 
											
												
													
														|  |  
 |  |  
 | 
											
												
													
														|  | -            Assert.AreEqual(0, _shellStream.Read(new byte[16], 0, 16));
 |  | 
 | 
											
												
													
														|  | -            Assert.AreEqual(TaskStatus.RanToCompletion, closeTask.Status);
 |  | 
 | 
											
												
													
														|  | -        }
 |  | 
 | 
											
												
													
														|  | 
 |  | +            await Task.Delay(50);
 | 
											
												
													
														|  |  
 |  |  
 | 
											
												
													
														|  | -        [TestMethod]
 |  | 
 | 
											
												
													
														|  | -        [Ignore] // Currently returns 0 immediately
 |  | 
 | 
											
												
													
														|  | -        public void Read_EmptyArray_OnlyReturnsZeroWhenDataAvailable()
 |  | 
 | 
											
												
													
														|  | -        {
 |  | 
 | 
											
												
													
														|  | -            Task receiveTask = Task.Run(async () =>
 |  | 
 | 
											
												
													
														|  | -            {
 |  | 
 | 
											
												
													
														|  | -                // For the test to have meaning, we should be in
 |  | 
 | 
											
												
													
														|  | -                // the call to Read before receiving the data.
 |  | 
 | 
											
												
													
														|  | -                // Impose a short delay to make that more likely.
 |  | 
 | 
											
												
													
														|  | -                await Task.Delay(50);
 |  | 
 | 
											
												
													
														|  | 
 |  | +            Assert.IsFalse(readTask.IsCompleted);
 | 
											
												
													
														|  |  
 |  |  
 | 
											
												
													
														|  | -                _channelSessionStub.Receive(Encoding.UTF8.GetBytes("Hello World!"));
 |  | 
 | 
											
												
													
														|  | -            });
 |  | 
 | 
											
												
													
														|  | 
 |  | +            _channelSessionStub.Close();
 | 
											
												
													
														|  |  
 |  |  
 | 
											
												
													
														|  | -            Assert.AreEqual(0, _shellStream.Read(Array.Empty<byte>(), 0, 0));
 |  | 
 | 
											
												
													
														|  | -            Assert.AreEqual(TaskStatus.RanToCompletion, receiveTask.Status);
 |  | 
 | 
											
												
													
														|  | 
 |  | +            Assert.AreEqual(0, await readTask);
 | 
											
												
													
														|  |          }
 |  |          }
 | 
											
												
													
														|  |  
 |  |  
 | 
											
												
													
														|  |          [TestMethod]
 |  |          [TestMethod]
 | 
											
												
													
														|  | -        [Ignore] // Currently hangs
 |  | 
 | 
											
												
													
														|  | -        public void ReadLine_NoData_ReturnsNullAfterClose()
 |  | 
 | 
											
												
													
														|  | 
 |  | +        public async Task Read_EmptyArray_OnlyReturnsZeroWhenDataAvailable()
 | 
											
												
													
														|  |          {
 |  |          {
 | 
											
												
													
														|  | -            Task closeTask = Task.Run(async () =>
 |  | 
 | 
											
												
													
														|  | -            {
 |  | 
 | 
											
												
													
														|  | -                await Task.Delay(50);
 |  | 
 | 
											
												
													
														|  | 
 |  | +            Task<int> readTask = _shellStream.ReadAsync(Array.Empty<byte>(), 0, 0);
 | 
											
												
													
														|  |  
 |  |  
 | 
											
												
													
														|  | -                _channelSessionStub.Close();
 |  | 
 | 
											
												
													
														|  | -            });
 |  | 
 | 
											
												
													
														|  | 
 |  | +            await Task.Delay(50);
 | 
											
												
													
														|  |  
 |  |  
 | 
											
												
													
														|  | -            Assert.IsNull(_shellStream.ReadLine());
 |  | 
 | 
											
												
													
														|  | -            Assert.AreEqual(TaskStatus.RanToCompletion, closeTask.Status);
 |  | 
 | 
											
												
													
														|  | 
 |  | +            Assert.IsFalse(readTask.IsCompleted);
 | 
											
												
													
														|  | 
 |  | +
 | 
											
												
													
														|  | 
 |  | +            _channelSessionStub.Receive(Encoding.UTF8.GetBytes("Hello World!"));
 | 
											
												
													
														|  | 
 |  | +
 | 
											
												
													
														|  | 
 |  | +            Assert.AreEqual(0, await readTask);
 | 
											
												
													
														|  |          }
 |  |          }
 | 
											
												
													
														|  |  
 |  |  
 | 
											
												
													
														|  |          [TestMethod]
 |  |          [TestMethod]
 | 
											
										
											
												
													
														|  | @@ -167,14 +164,24 @@ namespace Renci.SshNet.Tests.Classes
 | 
											
												
													
														|  |              _channelSessionStub.Receive(Encoding.UTF8.GetBytes("Hello "));
 |  |              _channelSessionStub.Receive(Encoding.UTF8.GetBytes("Hello "));
 | 
											
												
													
														|  |              _channelSessionStub.Receive(Encoding.UTF8.GetBytes("World!"));
 |  |              _channelSessionStub.Receive(Encoding.UTF8.GetBytes("World!"));
 | 
											
												
													
														|  |  
 |  |  
 | 
											
												
													
														|  | -            Assert.IsNull(_shellStream.Expect("123", TimeSpan.FromTicks(1)));
 |  | 
 | 
											
												
													
														|  | 
 |  | +            Assert.IsNull(_shellStream.Expect("123", TimeSpan.Zero));
 | 
											
												
													
														|  |  
 |  |  
 | 
											
												
													
														|  |              _channelSessionStub.Receive(Encoding.UTF8.GetBytes("\r\n12345"));
 |  |              _channelSessionStub.Receive(Encoding.UTF8.GetBytes("\r\n12345"));
 | 
											
												
													
														|  |  
 |  |  
 | 
											
												
													
														|  | -            // Both of these cases fail
 |  | 
 | 
											
												
													
														|  | -            // Case 1 above.
 |  | 
 | 
											
												
													
														|  | -            Assert.AreEqual("Hello World!\r\n123", _shellStream.Expect("123")); // Fails, returns "Hello World!\r\n12345"
 |  | 
 | 
											
												
													
														|  | -            Assert.AreEqual("45", _shellStream.Read()); // Passes, but should probably fail and return ""
 |  | 
 | 
											
												
													
														|  | 
 |  | +            Assert.AreEqual("Hello World!\r\n123", _shellStream.Expect("123"));
 | 
											
												
													
														|  | 
 |  | +            Assert.AreEqual("45", _shellStream.Read());
 | 
											
												
													
														|  | 
 |  | +        }
 | 
											
												
													
														|  | 
 |  | +
 | 
											
												
													
														|  | 
 |  | +        [TestMethod]
 | 
											
												
													
														|  | 
 |  | +        public void Read_AfterDispose_StillWorks()
 | 
											
												
													
														|  | 
 |  | +        {
 | 
											
												
													
														|  | 
 |  | +            _channelSessionStub.Receive(Encoding.UTF8.GetBytes("Hello World!"));
 | 
											
												
													
														|  | 
 |  | +
 | 
											
												
													
														|  | 
 |  | +            _shellStream.Dispose();
 | 
											
												
													
														|  | 
 |  | +            _shellStream.Dispose(); // Check that multiple Dispose is OK.
 | 
											
												
													
														|  | 
 |  | +
 | 
											
												
													
														|  | 
 |  | +            Assert.AreEqual("Hello World!", _shellStream.ReadLine());
 | 
											
												
													
														|  | 
 |  | +            Assert.IsNull(_shellStream.ReadLine());
 | 
											
												
													
														|  |          }
 |  |          }
 | 
											
												
													
														|  |  
 |  |  
 | 
											
												
													
														|  |          [TestMethod]
 |  |          [TestMethod]
 | 
											
										
											
												
													
														|  | @@ -221,7 +228,7 @@ namespace Renci.SshNet.Tests.Classes
 | 
											
												
													
														|  |          }
 |  |          }
 | 
											
												
													
														|  |  
 |  |  
 | 
											
												
													
														|  |          [TestMethod]
 |  |          [TestMethod]
 | 
											
												
													
														|  | -        public void Expect_String_non_ASCII_characters()
 |  | 
 | 
											
												
													
														|  | 
 |  | +        public void Expect_Regex_non_ASCII_characters()
 | 
											
												
													
														|  |          {
 |  |          {
 | 
											
												
													
														|  |              _channelSessionStub.Receive(Encoding.UTF8.GetBytes("Hello, こんにちは, Bonjour"));
 |  |              _channelSessionStub.Receive(Encoding.UTF8.GetBytes("Hello, こんにちは, Bonjour"));
 | 
											
												
													
														|  |  
 |  |  
 | 
											
										
											
												
													
														|  | @@ -247,13 +254,12 @@ namespace Renci.SshNet.Tests.Classes
 | 
											
												
													
														|  |          }
 |  |          }
 | 
											
												
													
														|  |  
 |  |  
 | 
											
												
													
														|  |          [TestMethod]
 |  |          [TestMethod]
 | 
											
												
													
														|  | -        public void Expect_String_DequeueChecks()
 |  | 
 | 
											
												
													
														|  | 
 |  | +        public void Expect_String_WithLookback()
 | 
											
												
													
														|  |          {
 |  |          {
 | 
											
												
													
														|  |              const string expected = "ccccc";
 |  |              const string expected = "ccccc";
 | 
											
												
													
														|  |  
 |  |  
 | 
											
												
													
														|  |              // Prime buffer
 |  |              // Prime buffer
 | 
											
												
													
														|  |              _channelSessionStub.Receive(Encoding.UTF8.GetBytes(new string(' ', BufferSize)));
 |  |              _channelSessionStub.Receive(Encoding.UTF8.GetBytes(new string(' ', BufferSize)));
 | 
											
												
													
														|  | -            _channelSessionStub.Receive(Encoding.UTF8.GetBytes(new string(' ', ExpectSize)));
 |  | 
 | 
											
												
													
														|  |  
 |  |  
 | 
											
												
													
														|  |              // Test data
 |  |              // Test data
 | 
											
												
													
														|  |              _channelSessionStub.Receive(Encoding.UTF8.GetBytes(new string('a', 100)));
 |  |              _channelSessionStub.Receive(Encoding.UTF8.GetBytes(new string('a', 100)));
 | 
											
										
											
												
													
														|  | @@ -263,14 +269,34 @@ namespace Renci.SshNet.Tests.Classes
 | 
											
												
													
														|  |              _channelSessionStub.Receive(Encoding.UTF8.GetBytes(new string('e', 100)));
 |  |              _channelSessionStub.Receive(Encoding.UTF8.GetBytes(new string('e', 100)));
 | 
											
												
													
														|  |  
 |  |  
 | 
											
												
													
														|  |              // Expected result
 |  |              // Expected result
 | 
											
												
													
														|  | -            var expectedResult = $"{new string(' ', BufferSize)}{new string(' ', ExpectSize)}{new string('a', 100)}{new string('b', 100)}{expected}";
 |  | 
 | 
											
												
													
														|  | 
 |  | +            var expectedResult = $"{new string(' ', BufferSize)}{new string('a', 100)}{new string('b', 100)}{expected}";
 | 
											
												
													
														|  |              var expectedRead = $"{new string('d', 100)}{new string('e', 100)}";
 |  |              var expectedRead = $"{new string('d', 100)}{new string('e', 100)}";
 | 
											
												
													
														|  |  
 |  |  
 | 
											
												
													
														|  | -            Assert.AreEqual(expectedResult, _shellStream.Expect(expected));
 |  | 
 | 
											
												
													
														|  | 
 |  | +            Assert.AreEqual(expectedResult, _shellStream.Expect(expected, TimeSpan.Zero, lookback: 250));
 | 
											
												
													
														|  |  
 |  |  
 | 
											
												
													
														|  |              Assert.AreEqual(expectedRead, _shellStream.Read());
 |  |              Assert.AreEqual(expectedRead, _shellStream.Read());
 | 
											
												
													
														|  |          }
 |  |          }
 | 
											
												
													
														|  |  
 |  |  
 | 
											
												
													
														|  | 
 |  | +        [TestMethod]
 | 
											
												
													
														|  | 
 |  | +        public void Expect_Regex_WithLookback()
 | 
											
												
													
														|  | 
 |  | +        {
 | 
											
												
													
														|  | 
 |  | +            _channelSessionStub.Receive(Encoding.UTF8.GetBytes("0123456789"));
 | 
											
												
													
														|  | 
 |  | +
 | 
											
												
													
														|  | 
 |  | +            Assert.AreEqual("01234567", _shellStream.Expect(new Regex(@"\d"), TimeSpan.Zero, lookback: 3));
 | 
											
												
													
														|  | 
 |  | +
 | 
											
												
													
														|  | 
 |  | +            Assert.AreEqual("89", _shellStream.Read());
 | 
											
												
													
														|  | 
 |  | +        }
 | 
											
												
													
														|  | 
 |  | +
 | 
											
												
													
														|  | 
 |  | +        [TestMethod]
 | 
											
												
													
														|  | 
 |  | +        public void Expect_Regex_WithLookback_non_ASCII_characters()
 | 
											
												
													
														|  | 
 |  | +        {
 | 
											
												
													
														|  | 
 |  | +            _channelSessionStub.Receive(Encoding.UTF8.GetBytes("Hello, こんにちは, Bonjour"));
 | 
											
												
													
														|  | 
 |  | +
 | 
											
												
													
														|  | 
 |  | +            Assert.AreEqual("Hello, こんにち", _shellStream.Expect(new Regex(@"[^\u0000-\u007F]"), TimeSpan.Zero, lookback: 11));
 | 
											
												
													
														|  | 
 |  | +
 | 
											
												
													
														|  | 
 |  | +            Assert.AreEqual("は, Bonjour", _shellStream.Read());
 | 
											
												
													
														|  | 
 |  | +        }
 | 
											
												
													
														|  | 
 |  | +
 | 
											
												
													
														|  |          [TestMethod]
 |  |          [TestMethod]
 | 
											
												
													
														|  |          public void Expect_Timeout()
 |  |          public void Expect_Timeout()
 | 
											
												
													
														|  |          {
 |  |          {
 |