From 26958f1659769da1e826989d281b58e944a713c9 Mon Sep 17 00:00:00 2001 From: Stephen Toub Date: Wed, 4 Nov 2020 10:20:19 -0500 Subject: [PATCH 1/2] Add stream conformance tests for TranscodingStream --- .../tests/Encoding/TranscodingStreamTests.cs | 54 ++++++++++++++++++- .../tests/System.Text.Encoding.Tests.csproj | 6 +++ 2 files changed, 58 insertions(+), 2 deletions(-) diff --git a/src/libraries/System.Text.Encoding/tests/Encoding/TranscodingStreamTests.cs b/src/libraries/System.Text.Encoding/tests/Encoding/TranscodingStreamTests.cs index 5eb16c1149c6a..c93779302a0bf 100644 --- a/src/libraries/System.Text.Encoding/tests/Encoding/TranscodingStreamTests.cs +++ b/src/libraries/System.Text.Encoding/tests/Encoding/TranscodingStreamTests.cs @@ -2,7 +2,9 @@ // The .NET Foundation licenses this file to you under the MIT license. using System.Collections.Generic; +using System.Diagnostics; using System.IO; +using System.IO.Tests; using System.Threading; using System.Threading.Tasks; using Moq; @@ -10,8 +12,20 @@ namespace System.Text.Tests { - public class TranscodingStreamTests + public class TranscodingStreamTests : ConnectedStreamConformanceTests { + protected override bool FlushRequiredToWriteData => true; + protected override bool FlushGuaranteesAllDataWritten => false; + protected override bool BlocksOnZeroByteReads => true; + + protected override Task CreateConnectedStreamsAsync() + { + (Stream stream1, Stream stream2) = ConnectedStreams.CreateBidirectional(); + return Task.FromResult( + (Encoding.CreateTranscodingStream(stream1, new IdentityEncoding(), new IdentityEncoding()), + Encoding.CreateTranscodingStream(stream2, new IdentityEncoding(), new IdentityEncoding()))); + } + public static IEnumerable ReadWriteTestBufferLengths { get @@ -668,7 +682,7 @@ public void Write_WithInvalidArgs_Throws() } [Fact] - public async Task WriteAsync() + public async Task WriteAsync_WithFullData() { MemoryStream sink = new MemoryStream(); CancellationToken expectedFlushAsyncCancellationToken = new CancellationTokenSource().Token; @@ -892,5 +906,41 @@ public override bool MovePrevious() } } } + + /// A custom encoding that's used to roundtrip from bytes to bytes through a string. + private sealed class IdentityEncoding : Encoding + { + public override int GetByteCount(char[] chars, int index, int count) => count; + + public override int GetBytes(char[] chars, int charIndex, int charCount, byte[] bytes, int byteIndex) + { + Span span = chars.AsSpan(charIndex, charCount); + for (int i = 0; i < span.Length; i++) + { + Debug.Assert(span[i] <= 0xFF); + bytes[byteIndex + i] = (byte)span[i]; + } + return charCount; + } + + public override int GetCharCount(byte[] bytes, int index, int count) => count; + + public override int GetChars(byte[] bytes, int byteIndex, int byteCount, char[] chars, int charIndex) + { + Span span = bytes.AsSpan(byteIndex, byteCount); + for (int i = 0; i < span.Length; i++) + { + Debug.Assert(span[i] <= 0xFF); + chars[charIndex + i] = (char)span[i]; + } + return byteCount; + } + + public override int GetMaxByteCount(int charCount) => charCount; + + public override int GetMaxCharCount(int byteCount) => byteCount; + + public override byte[] GetPreamble() => Array.Empty(); + } } } diff --git a/src/libraries/System.Text.Encoding/tests/System.Text.Encoding.Tests.csproj b/src/libraries/System.Text.Encoding/tests/System.Text.Encoding.Tests.csproj index 78c22e0243c91..74bcfdc4feaa3 100644 --- a/src/libraries/System.Text.Encoding/tests/System.Text.Encoding.Tests.csproj +++ b/src/libraries/System.Text.Encoding/tests/System.Text.Encoding.Tests.csproj @@ -79,6 +79,12 @@ + + + + + + From 328b46dd66fef845e557141b890b7523b1cfca2b Mon Sep 17 00:00:00 2001 From: Stephen Toub Date: Wed, 4 Nov 2020 17:50:01 -0500 Subject: [PATCH 2/2] Special-case 0-length input buffers to TranscodingStream.Write{Async} The base implementation of Encoder.Convert doesn't like empty inputs. Regardless, if the input is empty, we can avoid a whole bunch of unnecessary work. --- .../src/System/Text/TranscodingStream.cs | 10 ++++++++++ 1 file changed, 10 insertions(+) diff --git a/src/libraries/System.Private.CoreLib/src/System/Text/TranscodingStream.cs b/src/libraries/System.Private.CoreLib/src/System/Text/TranscodingStream.cs index 4bc5f5f509cef..71a8d4bc592ac 100644 --- a/src/libraries/System.Private.CoreLib/src/System/Text/TranscodingStream.cs +++ b/src/libraries/System.Private.CoreLib/src/System/Text/TranscodingStream.cs @@ -459,6 +459,11 @@ public override void Write(ReadOnlySpan buffer) { EnsurePreWriteConditions(); + if (buffer.IsEmpty) + { + return; + } + int rentalLength = Math.Clamp(buffer.Length, MinWriteRentedArraySize, MaxWriteRentedArraySize); char[] scratchChars = ArrayPool.Shared.Rent(rentalLength); @@ -527,6 +532,11 @@ public override ValueTask WriteAsync(ReadOnlyMemory buffer, CancellationTo return ValueTask.FromCanceled(cancellationToken); } + if (buffer.IsEmpty) + { + return ValueTask.CompletedTask; + } + return WriteAsyncCore(buffer, cancellationToken); async ValueTask WriteAsyncCore(ReadOnlyMemory remainingOuterEncodedBytes, CancellationToken cancellationToken) {