Jo Shields a575963da9 Imported Upstream version 3.6.0
Former-commit-id: da6be194a6b1221998fc28233f2503bd61dd9d14
2014-08-13 10:39:27 +01:00

231 lines
11 KiB
C#

// 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<MediaTypeFormatter> _emptyFormatterList = Enumerable.Empty<MediaTypeFormatter>();
private readonly Mock<MediaTypeFormatter> _formatterMock = new Mock<MediaTypeFormatter> { 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<string>(null, _emptyFormatterList), "content");
}
[Fact]
public void ReadAsAsyncOfT_WhenFormattersParameterIsNull_Throws()
{
Assert.ThrowsArgumentNull(() => HttpContentExtensions.ReadAsAsync<string>(new StringContent(""), null), "formatters");
}
[Fact]
public void ReadAsAsyncOfT_WhenContentIsObjectContent_GoesThroughSerializationCycleToConvertTypes()
{
var content = new ObjectContent<int[]>(new int[] { 10, 20, 30, 40 }, new JsonMediaTypeFormatter());
byte[] result = content.ReadAsAsync<byte[]>().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<InvalidOperationException>(() => content.ReadAsAsync<List<string>>(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<InvalidOperationException>(() => content.ReadAsAsync<List<string>>(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<TestableHttpContent> { CallBase = true };
contentMock.Setup(c => c.SerializeToStreamAsyncPublic(It.IsAny<Stream>(), It.IsAny<TransportContext>()))
.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<Stream>(), It.IsAny<HttpContentHeaders>(), It.IsAny<IFormatterLogger>()))
.Returns(TaskHelpers.FromResult<object>(value));
_formatterMock.Setup(f => f.CanReadType(typeof(string))).Returns(true);
var result = content.ReadAsAsync<string>(_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<string>(_formatters);
result.WaitUntilCompleted();
_formatterMock.Verify(f => f.ReadFromStreamAsync(typeof(string), It.IsAny<Stream>(), content.Headers, It.IsAny<IFormatterLogger>()), 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<TestClass>(value, _formatterMock.Object);
Assert.Same(value, content.ReadAsAsync<object>(_formatters).Result);
Assert.Same(value, content.ReadAsAsync<TestClass>(_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<Type>(), It.IsAny<Stream>(), content.Headers, It.IsAny<IFormatterLogger>()), 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<Type>())).Returns(true);
var content = new ObjectContent<object>(null, _formatterMock.Object);
SetupUpRoundTripSerialization(type => null);
Assert.Null(content.ReadAsAsync<object>(_formatters).Result);
Assert.Null(content.ReadAsAsync<TestClass>(_formatters).Result);
Assert.Null(content.ReadAsAsync<Nullable<int>>(_formatters).Result);
Assert.Null(content.ReadAsAsync(typeof(object), _formatters).Result);
Assert.Null(content.ReadAsAsync(typeof(TestClass), _formatters).Result);
Assert.Null(content.ReadAsAsync(typeof(Nullable<int>), _formatters).Result);
_formatterMock.Verify(f => f.ReadFromStreamAsync(It.IsAny<Type>(), It.IsAny<Stream>(), content.Headers, It.IsAny<IFormatterLogger>()), 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<object>(null, _formatterMock.Object, _mediaType.MediaType);
SetupUpRoundTripSerialization();
Assert.IsType<Int32>(content.ReadAsAsync<Int32>(_formatters).Result);
Assert.IsType<Int32>(content.ReadAsAsync(typeof(Int32), _formatters).Result);
_formatterMock.Verify(f => f.ReadFromStreamAsync(It.IsAny<Type>(), It.IsAny<Stream>(), content.Headers, It.IsAny<IFormatterLogger>()), 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<TestClass>(value, _formatterMock.Object, _mediaType.MediaType);
SetupUpRoundTripSerialization(type => new TestClass());
Assert.Throws<InvalidCastException>(() => content.ReadAsAsync<string>(_formatters).RethrowFaultedTaskException());
Assert.IsNotType<string>(content.ReadAsAsync(typeof(string), _formatters).Result);
_formatterMock.Verify(f => f.ReadFromStreamAsync(It.IsAny<Type>(), It.IsAny<Stream>(), content.Headers, It.IsAny<IFormatterLogger>()), Times.Exactly(2));
}
private void SetupUpRoundTripSerialization(Func<Type, object> factory = null)
{
factory = factory ?? Activator.CreateInstance;
_formatterMock.Setup(f => f.WriteToStreamAsync(It.IsAny<Type>(), It.IsAny<object>(), It.IsAny<Stream>(), It.IsAny<HttpContentHeaders>(), It.IsAny<TransportContext>()))
.Returns(TaskHelpers.Completed());
_formatterMock.Setup(f => f.ReadFromStreamAsync(It.IsAny<Type>(), It.IsAny<Stream>(), It.IsAny<HttpContentHeaders>(), It.IsAny<IFormatterLogger>()))
.Returns<Type, Stream, HttpContentHeaders, IFormatterLogger>((type, stream, headers, logger) => TaskHelpers.FromResult<object>(factory(type)));
}
public class TestClass { }
public abstract class TestableHttpContent : HttpContent
{
protected override Task<Stream> CreateContentReadStreamAsync()
{
return CreateContentReadStreamAsyncPublic();
}
public virtual Task<Stream> 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);
}
}
}