From dbcfe7ab09da103ba510756f5785faabdc902e61 Mon Sep 17 00:00:00 2001 From: Stephen Toub Date: Sun, 3 Nov 2019 21:59:00 -0500 Subject: [PATCH] Improve tests for StreamReader cancellation (#42171) Existing tests were only validating precancellation and not canceling once a Read{Block}Async call was already in flight. --- .../tests/StreamReader/StreamReaderTests.cs | 44 ++++++++++++++++--- 1 file changed, 39 insertions(+), 5 deletions(-) diff --git a/src/System.IO/tests/StreamReader/StreamReaderTests.cs b/src/System.IO/tests/StreamReader/StreamReaderTests.cs index a66379629fc5..34e724e87300 100644 --- a/src/System.IO/tests/StreamReader/StreamReaderTests.cs +++ b/src/System.IO/tests/StreamReader/StreamReaderTests.cs @@ -4,6 +4,7 @@ using System; using System.Collections.Generic; +using System.IO.Pipes; using System.Linq; using System.Text; using System.Threading; @@ -506,13 +507,46 @@ public async Task ReadBlockAsync_RepeatsReadsUntilReadDesiredAmount() } } - [Fact] - public async Task ReadAsync_Precanceled_ThrowsException() + [Theory] + [InlineData(0, false)] + [InlineData(0, true)] + [InlineData(1, false)] + [InlineData(1, true)] + public async Task ReadAsync_Canceled_ThrowsException(int method, bool precanceled) { - using (var sr = new StreamReader(new MemoryStream())) + Func> func = method switch + { + 0 => (sr, ct) => sr.ReadAsync(new char[1], ct).AsTask(), + 1 => (sr, ct) => sr.ReadBlockAsync(new char[1], ct).AsTask(), + _ => throw new Exception("unknown mode") + }; + + string pipeName = Guid.NewGuid().ToString("N"); + using (var serverStream = new NamedPipeServerStream(pipeName, PipeDirection.Out, 1, PipeTransmissionMode.Byte, PipeOptions.Asynchronous)) + using (var clientStream = new NamedPipeClientStream(".", pipeName, PipeDirection.In, PipeOptions.Asynchronous)) { - await Assert.ThrowsAnyAsync(() => sr.ReadAsync(Memory.Empty, new CancellationToken(true)).AsTask()); - await Assert.ThrowsAnyAsync(() => sr.ReadBlockAsync(Memory.Empty, new CancellationToken(true)).AsTask()); + await Task.WhenAll( + serverStream.WaitForConnectionAsync(), + clientStream.ConnectAsync()); + + using (var sr = new StreamReader(clientStream)) + { + var cts = new CancellationTokenSource(); + + if (precanceled) + { + cts.Cancel(); + } + + Task t = func(sr, cts.Token); + + if (!precanceled) + { + cts.Cancel(); + } + + await Assert.ThrowsAnyAsync(() => t); + } } }