// Copyright (c) Microsoft Corporation. All rights reserved. See License.txt in the project root for license information. using System.Collections.Generic; using System.IO; using System.Linq; using System.Net.Http.Formatting; using System.Net.Http.Headers; using System.Threading.Tasks; using Moq; using Xunit; using Assert = Microsoft.TestCommon.AssertEx; namespace System.Net.Http { public class HttpContentExtensionsTest { private static readonly IEnumerable _emptyFormatterList = Enumerable.Empty(); private readonly Mock _formatterMock = new Mock { CallBase = true }; private readonly MediaTypeHeaderValue _mediaType = new MediaTypeHeaderValue("foo/bar"); private readonly MediaTypeFormatter[] _formatters; public HttpContentExtensionsTest() { _formatterMock.Object.SupportedMediaTypes.Add(_mediaType); _formatters = new[] { _formatterMock.Object }; } [Fact] public void ReadAsAsync_WhenContentParameterIsNull_Throws() { Assert.ThrowsArgumentNull(() => HttpContentExtensions.ReadAsAsync(null, typeof(string), _emptyFormatterList), "content"); } [Fact] public void ReadAsAsync_WhenTypeParameterIsNull_Throws() { Assert.ThrowsArgumentNull(() => HttpContentExtensions.ReadAsAsync(new StringContent(""), null, _emptyFormatterList), "type"); } [Fact] public void ReadAsAsync_WhenFormattersParameterIsNull_Throws() { Assert.ThrowsArgumentNull(() => HttpContentExtensions.ReadAsAsync(new StringContent(""), typeof(string), null), "formatters"); } [Fact] public void ReadAsAsyncOfT_WhenContentParameterIsNull_Throws() { Assert.ThrowsArgumentNull(() => HttpContentExtensions.ReadAsAsync(null, _emptyFormatterList), "content"); } [Fact] public void ReadAsAsyncOfT_WhenFormattersParameterIsNull_Throws() { Assert.ThrowsArgumentNull(() => HttpContentExtensions.ReadAsAsync(new StringContent(""), null), "formatters"); } [Fact] public void ReadAsAsyncOfT_WhenContentIsObjectContent_GoesThroughSerializationCycleToConvertTypes() { var content = new ObjectContent(new int[] { 10, 20, 30, 40 }, new JsonMediaTypeFormatter()); byte[] result = content.ReadAsAsync().Result; Assert.Equal(new byte[] { 10, 20, 30, 40 }, result); } [Fact] public void ReadAsAsyncOfT_WhenNoMatchingFormatterFound_Throws() { var content = new StringContent("{}"); content.Headers.ContentType = _mediaType; content.Headers.ContentType.CharSet = "utf-16"; var formatters = new MediaTypeFormatter[] { new JsonMediaTypeFormatter() }; Assert.Throws(() => content.ReadAsAsync>(formatters), "No MediaTypeFormatter is available to read an object of type 'List`1' from content with media type 'foo/bar'."); } [Fact] public void ReadAsAsyncOfT_WhenNoMatchingFormatterFoundForContentWithNoMediaType_Throws() { var content = new StringContent("{}"); content.Headers.ContentType = null; var formatters = new MediaTypeFormatter[] { new JsonMediaTypeFormatter() }; Assert.Throws(() => content.ReadAsAsync>(formatters), "No MediaTypeFormatter is available to read an object of type 'List`1' from content with media type ''undefined''."); } [Fact] public void ReadAsAsyncOfT_ReadsFromContent_ThenInvokesFormattersReadFromStreamMethod() { Stream contentStream = null; string value = "42"; var contentMock = new Mock { CallBase = true }; contentMock.Setup(c => c.SerializeToStreamAsyncPublic(It.IsAny(), It.IsAny())) .Returns(TaskHelpers.Completed) .Callback((Stream s, TransportContext _) => contentStream = s) .Verifiable(); HttpContent content = contentMock.Object; content.Headers.ContentType = _mediaType; _formatterMock .Setup(f => f.ReadFromStreamAsync(typeof(string), It.IsAny(), It.IsAny(), It.IsAny())) .Returns(TaskHelpers.FromResult(value)); _formatterMock.Setup(f => f.CanReadType(typeof(string))).Returns(true); var result = content.ReadAsAsync(_formatters); var resultValue = result.Result; Assert.Same(value, resultValue); contentMock.Verify(); _formatterMock.Verify(f => f.ReadFromStreamAsync(typeof(string), contentStream, content.Headers, null), Times.Once()); } [Fact] public void ReadAsAsyncOfT_InvokesFormatterEvenIfContentLengthIsZero() { var content = new StringContent(""); _formatterMock.Setup(f => f.CanReadType(typeof(string))).Returns(true); _formatterMock.Object.SupportedMediaTypes.Add(content.Headers.ContentType); var result = content.ReadAsAsync(_formatters); result.WaitUntilCompleted(); _formatterMock.Verify(f => f.ReadFromStreamAsync(typeof(string), It.IsAny(), content.Headers, It.IsAny()), Times.Once()); } [Fact] public void ReadAsAsync_WhenContentIsObjectContentAndValueIsCompatibleType_ReadsValueFromObjectContent() { _formatterMock.Setup(f => f.CanWriteType(typeof(TestClass))).Returns(true); var value = new TestClass(); var content = new ObjectContent(value, _formatterMock.Object); Assert.Same(value, content.ReadAsAsync(_formatters).Result); Assert.Same(value, content.ReadAsAsync(_formatters).Result); Assert.Same(value, content.ReadAsAsync(typeof(object), _formatters).Result); Assert.Same(value, content.ReadAsAsync(typeof(TestClass), _formatters).Result); _formatterMock.Verify(f => f.ReadFromStreamAsync(It.IsAny(), It.IsAny(), content.Headers, It.IsAny()), Times.Never()); } [Fact] public void ReadAsAsync_WhenContentIsObjectContentAndValueIsNull_IfTypeIsNullable_SerializesAndDeserializesValue() { _formatterMock.Setup(f => f.CanWriteType(typeof(object))).Returns(true); _formatterMock.Setup(f => f.CanReadType(It.IsAny())).Returns(true); var content = new ObjectContent(null, _formatterMock.Object); SetupUpRoundTripSerialization(type => null); Assert.Null(content.ReadAsAsync(_formatters).Result); Assert.Null(content.ReadAsAsync(_formatters).Result); Assert.Null(content.ReadAsAsync>(_formatters).Result); Assert.Null(content.ReadAsAsync(typeof(object), _formatters).Result); Assert.Null(content.ReadAsAsync(typeof(TestClass), _formatters).Result); Assert.Null(content.ReadAsAsync(typeof(Nullable), _formatters).Result); _formatterMock.Verify(f => f.ReadFromStreamAsync(It.IsAny(), It.IsAny(), content.Headers, It.IsAny()), Times.Exactly(6)); } [Fact] public void ReadAsAsync_WhenContentIsObjectContentAndValueIsNull_IfTypeIsNotNullable_SerializesAndDeserializesValue() { _formatterMock.Setup(f => f.CanWriteType(typeof(object))).Returns(true); _formatterMock.Setup(f => f.CanReadType(typeof(Int32))).Returns(true); var content = new ObjectContent(null, _formatterMock.Object, _mediaType.MediaType); SetupUpRoundTripSerialization(); Assert.IsType(content.ReadAsAsync(_formatters).Result); Assert.IsType(content.ReadAsAsync(typeof(Int32), _formatters).Result); _formatterMock.Verify(f => f.ReadFromStreamAsync(It.IsAny(), It.IsAny(), content.Headers, It.IsAny()), Times.Exactly(2)); } [Fact] public void ReadAsAsync_WhenContentIsObjectContentAndValueIsNotCompatibleType_SerializesAndDeserializesValue() { _formatterMock.Setup(f => f.CanWriteType(typeof(TestClass))).Returns(true); _formatterMock.Setup(f => f.CanReadType(typeof(string))).Returns(true); var value = new TestClass(); var content = new ObjectContent(value, _formatterMock.Object, _mediaType.MediaType); SetupUpRoundTripSerialization(type => new TestClass()); Assert.Throws(() => content.ReadAsAsync(_formatters).RethrowFaultedTaskException()); Assert.IsNotType(content.ReadAsAsync(typeof(string), _formatters).Result); _formatterMock.Verify(f => f.ReadFromStreamAsync(It.IsAny(), It.IsAny(), content.Headers, It.IsAny()), Times.Exactly(2)); } private void SetupUpRoundTripSerialization(Func factory = null) { factory = factory ?? Activator.CreateInstance; _formatterMock.Setup(f => f.WriteToStreamAsync(It.IsAny(), It.IsAny(), It.IsAny(), It.IsAny(), It.IsAny())) .Returns(TaskHelpers.Completed()); _formatterMock.Setup(f => f.ReadFromStreamAsync(It.IsAny(), It.IsAny(), It.IsAny(), It.IsAny())) .Returns((type, stream, headers, logger) => TaskHelpers.FromResult(factory(type))); } public class TestClass { } public abstract class TestableHttpContent : HttpContent { protected override Task CreateContentReadStreamAsync() { return CreateContentReadStreamAsyncPublic(); } public virtual Task CreateContentReadStreamAsyncPublic() { return base.CreateContentReadStreamAsync(); } protected override Task SerializeToStreamAsync(Stream stream, TransportContext context) { return SerializeToStreamAsyncPublic(stream, context); } public abstract Task SerializeToStreamAsyncPublic(Stream stream, TransportContext context); protected override bool TryComputeLength(out long length) { return TryComputeLengthPublic(out length); } public abstract bool TryComputeLengthPublic(out long length); } } }