File: XmlDataContractSerializerInputFormatterTest.cs
Web Access
Project: src\src\Mvc\Mvc.Formatters.Xml\test\Microsoft.AspNetCore.Mvc.Formatters.Xml.Test.csproj (Microsoft.AspNetCore.Mvc.Formatters.Xml.Test)
// Licensed to the .NET Foundation under one or more agreements.
// The .NET Foundation licenses this file to you under the MIT license.
 
using System.Globalization;
using System.Runtime.Serialization;
using System.Text;
using System.Xml;
using Microsoft.AspNetCore.Http;
using Microsoft.AspNetCore.Http.Features;
using Microsoft.AspNetCore.Mvc.ModelBinding;
using Microsoft.AspNetCore.WebUtilities;
using Moq;
 
namespace Microsoft.AspNetCore.Mvc.Formatters.Xml;
 
public class XmlDataContractSerializerInputFormatterTest
{
    [DataContract(Name = "DummyClass", Namespace = "")]
    public class DummyClass
    {
        [DataMember]
        public int SampleInt { get; set; }
    }
 
    [DataContract(Name = "SomeDummyClass", Namespace = "")]
    public class SomeDummyClass : DummyClass
    {
        [DataMember]
        public string SampleString { get; set; }
    }
 
    [DataContract(Name = "TestLevelOne", Namespace = "")]
    public class TestLevelOne
    {
        [DataMember]
        public int SampleInt { get; set; }
        [DataMember]
        public string sampleString;
        public DateTime SampleDate { get; set; }
    }
 
    [DataContract(Name = "TestLevelTwo", Namespace = "")]
    public class TestLevelTwo
    {
        [DataMember]
        public string SampleString { get; set; }
        [DataMember]
        public TestLevelOne TestOne { get; set; }
    }
 
    [Theory]
    [InlineData("application/xml", true)]
    [InlineData("application/*", false)]
    [InlineData("*/*", false)]
    [InlineData("text/xml", true)]
    [InlineData("text/*", false)]
    [InlineData("text/json", false)]
    [InlineData("application/json", false)]
    [InlineData("application/some.entity+xml", true)]
    [InlineData("application/some.entity+xml;v=2", true)]
    [InlineData("application/some.entity+json", false)]
    [InlineData("application/some.entity+*", false)]
    [InlineData("text/some.entity+json", false)]
    [InlineData("", false)]
    [InlineData(null, false)]
    [InlineData("invalid", false)]
    public void CanRead_ReturnsTrueForAnySupportedContentType(string requestContentType, bool expectedCanRead)
    {
        // Arrange
        var formatter = new XmlDataContractSerializerInputFormatter(new MvcOptions());
        var contentBytes = Encoding.UTF8.GetBytes("content");
 
        var modelState = new ModelStateDictionary();
        var httpContext = GetHttpContext(contentBytes, contentType: requestContentType);
        var provider = new EmptyModelMetadataProvider();
        var metadata = provider.GetMetadataForType(typeof(string));
        var formatterContext = new InputFormatterContext(
            httpContext,
            modelName: string.Empty,
            modelState: modelState,
            metadata: metadata,
            readerFactory: new TestHttpRequestStreamReaderFactory().CreateReader);
 
        // Act
        var result = formatter.CanRead(formatterContext);
 
        // Assert
        Assert.Equal(expectedCanRead, result);
    }
 
    [Fact]
    public void XmlDataContractSerializer_CachesSerializerForType()
    {
        // Arrange
        var input = "<?xml version=\"1.0\" encoding=\"UTF-8\"?>" +
            "<DummyClass><SampleInt>10</SampleInt></DummyClass>";
        var formatter = new TestXmlDataContractSerializerInputFormatter();
        var contentBytes = Encoding.UTF8.GetBytes(input);
        var context = GetInputFormatterContext(contentBytes, typeof(DummyClass));
 
        // Act
        formatter.CanRead(context);
        formatter.CanRead(context);
 
        // Assert
        Assert.Equal(1, formatter.createSerializerCalledCount);
    }
 
    [Fact]
    public void HasProperSupportedMediaTypes()
    {
        // Arrange & Act
        var formatter = new XmlDataContractSerializerInputFormatter(new MvcOptions());
 
        // Assert
        Assert.Contains("application/xml", formatter.SupportedMediaTypes
                             .Select(content => content.ToString()));
        Assert.Contains("text/xml", formatter.SupportedMediaTypes
                             .Select(content => content.ToString()));
    }
 
    [Fact]
    public void HasProperSupportedEncodings()
    {
        // Arrange & Act
        var formatter = new XmlDataContractSerializerInputFormatter(new MvcOptions());
 
        // Assert
        Assert.Contains(formatter.SupportedEncodings, i => i.WebName == "utf-8");
        Assert.Contains(formatter.SupportedEncodings, i => i.WebName == "utf-16");
    }
 
    [Fact]
    public async Task BuffersRequestBody_ByDefault()
    {
        // Arrange
        var expectedInt = 10;
        var expectedString = "TestString";
 
        var input = "<?xml version=\"1.0\" encoding=\"UTF-8\"?>" +
            "<TestLevelOne><SampleInt>" + expectedInt + "</SampleInt>" +
            "<sampleString>" + expectedString + "</sampleString></TestLevelOne>";
 
        var formatter = new XmlDataContractSerializerInputFormatter(new MvcOptions());
 
        var contentBytes = Encoding.UTF8.GetBytes(input);
        var httpContext = new DefaultHttpContext();
        httpContext.Features.Set<IHttpResponseFeature>(new TestResponseFeature());
        httpContext.Request.Body = new NonSeekableReadStream(contentBytes, allowSyncReads: true);
        httpContext.Request.ContentType = "application/json";
        var context = GetInputFormatterContext(httpContext, typeof(TestLevelOne));
 
        // Act
        var result = await formatter.ReadAsync(context);
 
        // Assert
        Assert.NotNull(result);
        Assert.False(result.HasError);
        var model = Assert.IsType<TestLevelOne>(result.Model);
 
        Assert.Equal(expectedInt, model.SampleInt);
        Assert.Equal(expectedString, model.sampleString);
    }
 
    [Fact]
    public async Task ReadAsync_DoesNotDisposeBufferedStreamIfItDidNotCreateIt()
    {
        // Arrange
        var expectedInt = 10;
        var expectedString = "TestString";
 
        var input = "<?xml version=\"1.0\" encoding=\"UTF-8\"?>" +
            "<TestLevelOne><SampleInt>" + expectedInt + "</SampleInt>" +
            "<sampleString>" + expectedString + "</sampleString></TestLevelOne>";
 
        var formatter = new XmlDataContractSerializerInputFormatter(new MvcOptions());
 
        var contentBytes = Encoding.UTF8.GetBytes(input);
        var httpContext = new DefaultHttpContext();
        var testBufferedReadStream = new VerifyDisposeFileBufferingReadStream(new MemoryStream(contentBytes), 1024);
        httpContext.Request.Body = testBufferedReadStream;
        var context = GetInputFormatterContext(httpContext, typeof(TestLevelOne));
 
        // Act
        var result = await formatter.ReadAsync(context);
 
        // Assert
        Assert.NotNull(result);
        Assert.False(result.HasError);
        var model = Assert.IsType<TestLevelOne>(result.Model);
 
        Assert.Equal(expectedInt, model.SampleInt);
        Assert.Equal(expectedString, model.sampleString);
        Assert.False(testBufferedReadStream.Disposed);
    }
 
    [Fact]
    public async Task SuppressInputFormatterBufferingSetToTrue_DoesNotBufferRequestBody()
    {
        // Arrange
        var expectedInt = 10;
        var expectedString = "TestString";
 
        var input = "<?xml version=\"1.0\" encoding=\"UTF-8\"?>" +
            "<TestLevelOne><SampleInt>" + expectedInt + "</SampleInt>" +
            "<sampleString>" + expectedString + "</sampleString></TestLevelOne>";
 
        var formatter = new XmlDataContractSerializerInputFormatter(new MvcOptions { SuppressInputFormatterBuffering = true });
 
        var contentBytes = Encoding.UTF8.GetBytes(input);
        var httpContext = new DefaultHttpContext();
        httpContext.Features.Set<IHttpResponseFeature>(new TestResponseFeature());
        httpContext.Request.Body = new NonSeekableReadStream(contentBytes);
        httpContext.Request.ContentType = "application/xml";
        var context = GetInputFormatterContext(httpContext, typeof(TestLevelOne));
 
        // Act
        var result = await formatter.ReadAsync(context);
 
        // Assert
        Assert.NotNull(result);
        Assert.False(result.HasError);
        var model = Assert.IsType<TestLevelOne>(result.Model);
 
        Assert.Equal(expectedInt, model.SampleInt);
        Assert.Equal(expectedString, model.sampleString);
 
        // Reading again should fail as buffering request body is disabled
        await Assert.ThrowsAsync<XmlException>(() => formatter.ReadAsync(context));
    }
 
    [Fact]
    public async Task BuffersRequestBody_ByDefaultUsingMvcOptions()
    {
        // Arrange
        var expectedInt = 10;
        var expectedString = "TestString";
 
        var input = "<?xml version=\"1.0\" encoding=\"UTF-8\"?>" +
            "<TestLevelOne><SampleInt>" + expectedInt + "</SampleInt>" +
            "<sampleString>" + expectedString + "</sampleString></TestLevelOne>";
 
        var formatter = new XmlDataContractSerializerInputFormatter(new MvcOptions());
        var contentBytes = Encoding.UTF8.GetBytes(input);
        var httpContext = new DefaultHttpContext();
 
        httpContext.Features.Set<IHttpResponseFeature>(new TestResponseFeature());
        httpContext.Request.Body = new NonSeekableReadStream(contentBytes, allowSyncReads: false);
        httpContext.Request.ContentType = "application/json";
        var context = GetInputFormatterContext(httpContext, typeof(TestLevelOne));
 
        // Act
        var result = await formatter.ReadAsync(context);
 
        // Assert
        Assert.NotNull(result);
        Assert.False(result.HasError);
        var model = Assert.IsType<TestLevelOne>(result.Model);
 
        Assert.Equal(expectedInt, model.SampleInt);
        Assert.Equal(expectedString, model.sampleString);
    }
 
    [Fact]
    public async Task SuppressInputFormatterBufferingSetToTrue_DoesNotBufferRequestBody_UsingMvcOptions()
    {
        // Arrange
        var expectedInt = 10;
        var expectedString = "TestString";
 
        var input = "<?xml version=\"1.0\" encoding=\"UTF-8\"?>" +
            "<TestLevelOne><SampleInt>" + expectedInt + "</SampleInt>" +
            "<sampleString>" + expectedString + "</sampleString></TestLevelOne>";
 
        var formatter = new XmlDataContractSerializerInputFormatter(
            new MvcOptions() { SuppressInputFormatterBuffering = true });
        var contentBytes = Encoding.UTF8.GetBytes(input);
        var httpContext = new DefaultHttpContext();
        httpContext.Features.Set<IHttpResponseFeature>(new TestResponseFeature());
        httpContext.Request.Body = new NonSeekableReadStream(contentBytes);
        httpContext.Request.ContentType = "application/xml";
        var context = GetInputFormatterContext(httpContext, typeof(TestLevelOne));
 
        // Act
        var result = await formatter.ReadAsync(context);
 
        // Assert
        Assert.NotNull(result);
        Assert.False(result.HasError);
        var model = Assert.IsType<TestLevelOne>(result.Model);
 
        Assert.Equal(expectedInt, model.SampleInt);
        Assert.Equal(expectedString, model.sampleString);
    }
 
    [Fact]
    public async Task ReadAsync_ReadsSimpleTypes()
    {
        // Arrange
        var expectedInt = 10;
        var expectedString = "TestString";
 
        var input = "<?xml version=\"1.0\" encoding=\"UTF-8\"?>" +
            "<TestLevelOne><SampleInt>" + expectedInt + "</SampleInt>" +
            "<sampleString>" + expectedString + "</sampleString></TestLevelOne>";
 
        var formatter = new XmlDataContractSerializerInputFormatter(new MvcOptions());
        var contentBytes = Encoding.UTF8.GetBytes(input);
        var context = GetInputFormatterContext(contentBytes, typeof(TestLevelOne));
 
        // Act
        var result = await formatter.ReadAsync(context);
 
        // Assert
        Assert.NotNull(result);
        Assert.False(result.HasError);
        var model = Assert.IsType<TestLevelOne>(result.Model);
 
        Assert.Equal(expectedInt, model.SampleInt);
        Assert.Equal(expectedString, model.sampleString);
    }
 
    [Fact]
    public async Task SuppressInputFormatterBufferingSetToTrue_UsingMutatedOptions()
    {
        // Arrange
        var expectedInt = 10;
        var expectedString = "TestString";
 
        var input = "<?xml version=\"1.0\" encoding=\"UTF-8\"?>" +
            "<TestLevelOne><SampleInt>" + expectedInt + "</SampleInt>" +
            "<sampleString>" + expectedString + "</sampleString></TestLevelOne>";
 
        var mvcOptions = new MvcOptions() { SuppressInputFormatterBuffering = false };
        var formatter = new XmlDataContractSerializerInputFormatter(mvcOptions);
        var contentBytes = Encoding.UTF8.GetBytes(input);
        var httpContext = new DefaultHttpContext();
        httpContext.Features.Set<IHttpResponseFeature>(new TestResponseFeature());
        httpContext.Request.Body = new NonSeekableReadStream(contentBytes);
        httpContext.Request.ContentType = "application/xml";
        var context = GetInputFormatterContext(httpContext, typeof(TestLevelOne));
 
        // Act
        // Mutate options after passing into the constructor to make sure that the value type is not store in the constructor
        mvcOptions.SuppressInputFormatterBuffering = true;
        var result = await formatter.ReadAsync(context);
 
        // Assert
        Assert.NotNull(result);
        Assert.False(result.HasError);
        var model = Assert.IsType<TestLevelOne>(result.Model);
 
        Assert.Equal(expectedInt, model.SampleInt);
        Assert.Equal(expectedString, model.sampleString);
 
        // Reading again should fail as buffering request body is disabled
        await Assert.ThrowsAsync<XmlException>(() => formatter.ReadAsync(context));
    }
 
    [Fact]
    public async Task ReadAsync_ReadsComplexTypes()
    {
        // Arrange
        var expectedInt = 10;
        var expectedString = "TestString";
        var expectedLevelTwoString = "102";
 
        var input = "<?xml version=\"1.0\" encoding=\"UTF-8\"?>" +
            "<TestLevelTwo><SampleString>" + expectedLevelTwoString + "</SampleString>" +
            "<TestOne><SampleInt>" + expectedInt + "</SampleInt>" +
            "<sampleString>" + expectedString + "</sampleString></TestOne></TestLevelTwo>";
 
        var formatter = new XmlDataContractSerializerInputFormatter(new MvcOptions());
        var contentBytes = Encoding.UTF8.GetBytes(input);
        var context = GetInputFormatterContext(contentBytes, typeof(TestLevelTwo));
 
        // Act
        var result = await formatter.ReadAsync(context);
 
        // Assert
        Assert.NotNull(result);
        Assert.False(result.HasError);
        var model = Assert.IsType<TestLevelTwo>(result.Model);
 
        Assert.Equal(expectedLevelTwoString, model.SampleString);
        Assert.Equal(expectedInt, model.TestOne.SampleInt);
        Assert.Equal(expectedString, model.TestOne.sampleString);
    }
 
    [Fact]
    public async Task ReadAsync_ReadsWhenMaxDepthIsModified()
    {
        // Arrange
        var expectedInt = 10;
 
        var input = "<?xml version=\"1.0\" encoding=\"UTF-8\"?>" +
            "<DummyClass><SampleInt>" + expectedInt + "</SampleInt></DummyClass>";
        var formatter = new XmlDataContractSerializerInputFormatter(new MvcOptions());
        formatter.MaxDepth = 10;
        var contentBytes = Encoding.UTF8.GetBytes(input);
        var context = GetInputFormatterContext(contentBytes, typeof(DummyClass));
 
        // Act
        var result = await formatter.ReadAsync(context);
 
        // Assert
        Assert.NotNull(result);
        Assert.False(result.HasError);
        var model = Assert.IsType<DummyClass>(result.Model);
        Assert.Equal(expectedInt, model.SampleInt);
    }
 
    [Fact]
    public async Task ReadAsync_ThrowsOnExceededMaxDepth()
    {
        // Arrange
        var input = "<?xml version=\"1.0\" encoding=\"UTF-8\"?>" +
            "<TestLevelTwo><SampleString>test</SampleString>" +
            "<TestOne><SampleInt>10</SampleInt><sampleString>test</sampleString></TestOne></TestLevelTwo>";
        var formatter = new XmlDataContractSerializerInputFormatter(new MvcOptions());
        formatter.MaxDepth = 1;
        var contentBytes = Encoding.UTF8.GetBytes(input);
        var context = GetInputFormatterContext(contentBytes, typeof(TestLevelTwo));
 
        // Act & Assert
        await Assert.ThrowsAsync<InputFormatterException>(async () => await formatter.ReadAsync(context));
    }
 
    [Fact]
    public async Task ReadAsync_ThrowsWhenReaderQuotasAreChanged()
    {
        // Arrange
        var input = "<?xml version=\"1.0\" encoding=\"UTF-8\"?>" +
            "<TestLevelTwo><SampleString>test</SampleString><TestOne><SampleInt>10</SampleInt>" +
            "<sampleString>test</sampleString></TestOne></TestLevelTwo>";
        var formatter = new XmlDataContractSerializerInputFormatter(new MvcOptions());
        formatter.XmlDictionaryReaderQuotas.MaxStringContentLength = 2;
        var contentBytes = Encoding.UTF8.GetBytes(input);
        var context = GetInputFormatterContext(contentBytes, typeof(TestLevelTwo));
 
        // Act & Assert
        await Assert.ThrowsAsync<InputFormatterException>(async () => await formatter.ReadAsync(context));
    }
 
    [Fact]
    public void SetMaxDepth_ThrowsWhenMaxDepthIsBelowOne()
    {
        // Arrange
        var formatter = new XmlDataContractSerializerInputFormatter(new MvcOptions());
 
        // Act & Assert
        Assert.Throws<ArgumentOutOfRangeException>(() => formatter.MaxDepth = 0);
    }
 
    [Fact]
    public async Task ReadAsync_VerifyStreamIsOpenAfterRead()
    {
        // Arrange
        var input = "<?xml version=\"1.0\" encoding=\"UTF-8\"?>" +
            "<DummyClass><SampleInt>10</SampleInt></DummyClass>";
        var formatter = new XmlDataContractSerializerInputFormatter(new MvcOptions());
        var contentBytes = Encoding.UTF8.GetBytes(input);
        var context = GetInputFormatterContext(contentBytes, typeof(DummyClass));
 
        // Act
        var result = await formatter.ReadAsync(context);
 
        // Assert
        Assert.NotNull(result);
        Assert.False(result.HasError);
        Assert.NotNull(result.Model);
        Assert.True(context.HttpContext.Request.Body.CanRead);
    }
 
    [Fact]
    public async Task ReadAsync_FallsbackToUTF8_WhenCharSet_NotInContentType()
    {
        // Arrange
        var expectedException = typeof(XmlException);
        var inpStart = Encoding.Unicode.GetBytes("<?xml version=\"1.0\" encoding=\"UTF-16\"?>" +
            "<DummyClass><SampleInt>");
        byte[] inp = { 192, 193 };
        var inpEnd = Encoding.Unicode.GetBytes("</SampleInt></DummyClass>");
 
        var contentBytes = new byte[inpStart.Length + inp.Length + inpEnd.Length];
        Buffer.BlockCopy(inpStart, 0, contentBytes, 0, inpStart.Length);
        Buffer.BlockCopy(inp, 0, contentBytes, inpStart.Length, inp.Length);
        Buffer.BlockCopy(inpEnd, 0, contentBytes, inpStart.Length + inp.Length, inpEnd.Length);
 
        var formatter = new XmlDataContractSerializerInputFormatter(new MvcOptions());
        var context = GetInputFormatterContext(contentBytes, typeof(TestLevelTwo));
 
        // Act
        var ex = await Assert.ThrowsAsync(expectedException, () => formatter.ReadAsync(context));
        Assert.Contains("utf-8", ex.Message);
        Assert.Contains("utf-16LE", ex.Message);
    }
 
    [Fact]
    public async Task ReadAsync_UsesContentTypeCharSet_ToReadStream()
    {
        // Arrange
        var expectedException = typeof(XmlException);
 
        var inputBytes = Encoding.UTF8.GetBytes("<?xml version=\"1.0\" encoding=\"UTF-8\"?>" +
            "<DummyClass><SampleInt>1000</SampleInt></DummyClass>");
 
        var formatter = new XmlDataContractSerializerInputFormatter(new MvcOptions());
 
        var modelState = new ModelStateDictionary();
        var httpContext = GetHttpContext(inputBytes, contentType: "application/xml; charset=utf-16");
 
        var provider = new EmptyModelMetadataProvider();
        var metadata = provider.GetMetadataForType(typeof(TestLevelOne));
        var context = new InputFormatterContext(
            httpContext,
            modelName: string.Empty,
            modelState: modelState,
            metadata: metadata,
            readerFactory: new TestHttpRequestStreamReaderFactory().CreateReader);
 
        // Act
        var ex = await Assert.ThrowsAsync(expectedException, () => formatter.ReadAsync(context));
        Assert.Contains("utf-16LE", ex.Message);
        Assert.Contains("utf-8", ex.Message);
    }
 
    [Fact]
    public async Task ReadAsync_IgnoresBOMCharacters()
    {
        // Arrange
        var sampleString = "Test";
        var sampleStringBytes = Encoding.UTF8.GetBytes(sampleString);
        var inputStart = Encoding.UTF8.GetBytes("<?xml version=\"1.0\" encoding=\"UTF-8\"?>" + Environment.NewLine +
            "<TestLevelTwo><SampleString>" + sampleString);
        byte[] bom = { 0xef, 0xbb, 0xbf };
        var inputEnd = Encoding.UTF8.GetBytes("</SampleString></TestLevelTwo>");
        var expectedBytes = new byte[sampleString.Length + bom.Length];
 
        var contentBytes = new byte[inputStart.Length + bom.Length + inputEnd.Length];
        Buffer.BlockCopy(inputStart, 0, contentBytes, 0, inputStart.Length);
        Buffer.BlockCopy(bom, 0, contentBytes, inputStart.Length, bom.Length);
        Buffer.BlockCopy(inputEnd, 0, contentBytes, inputStart.Length + bom.Length, inputEnd.Length);
 
        var formatter = new XmlDataContractSerializerInputFormatter(new MvcOptions());
        var context = GetInputFormatterContext(contentBytes, typeof(TestLevelTwo));
 
        // Act
        var result = await formatter.ReadAsync(context);
 
        // Assert
        Assert.NotNull(result);
        Assert.False(result.HasError);
        var model = Assert.IsType<TestLevelTwo>(result.Model);
        Buffer.BlockCopy(sampleStringBytes, 0, expectedBytes, 0, sampleStringBytes.Length);
        Buffer.BlockCopy(bom, 0, expectedBytes, sampleStringBytes.Length, bom.Length);
        Assert.Equal(expectedBytes, Encoding.UTF8.GetBytes(model.SampleString));
    }
 
    [Fact]
    public async Task ReadAsync_AcceptsUTF16Characters()
    {
        // Arrange
        var expectedInt = 10;
        var expectedString = "TestString";
 
        var input = "<?xml version=\"1.0\" encoding=\"UTF-16\"?>" +
                            "<TestLevelOne><SampleInt>" + expectedInt + "</SampleInt>" +
                            "<sampleString>" + expectedString + "</sampleString></TestLevelOne>";
 
        var formatter = new XmlDataContractSerializerInputFormatter(new MvcOptions());
        var contentBytes = Encoding.Unicode.GetBytes(input);
 
        var modelState = new ModelStateDictionary();
        var httpContext = GetHttpContext(contentBytes, contentType: "application/xml; charset=utf-16");
 
        var provider = new EmptyModelMetadataProvider();
        var metadata = provider.GetMetadataForType(typeof(TestLevelOne));
        var context = new InputFormatterContext(
            httpContext,
            modelName: string.Empty,
            modelState: modelState,
            metadata: metadata,
            readerFactory: new TestHttpRequestStreamReaderFactory().CreateReader);
 
        // Act
        var result = await formatter.ReadAsync(context);
 
        // Assert
        Assert.NotNull(result);
        Assert.False(result.HasError);
        var model = Assert.IsType<TestLevelOne>(result.Model);
 
        Assert.Equal(expectedInt, model.SampleInt);
        Assert.Equal(expectedString, model.sampleString);
    }
 
    [Fact]
    public async Task ReadAsync_ThrowsWhenNotConfiguredWithRootName()
    {
        // Arrange
        var SubstituteRootName = "SomeOtherClass";
        var SubstituteRootNamespace = "http://tempuri.org";
 
        var input = string.Format(
            CultureInfo.InvariantCulture,
            "<{0} xmlns=\"{1}\"><SampleInt xmlns=\"\">1</SampleInt></{0}>",
            SubstituteRootName,
            SubstituteRootNamespace);
        var formatter = new XmlDataContractSerializerInputFormatter(new MvcOptions());
        var contentBytes = Encoding.UTF8.GetBytes(input);
        var context = GetInputFormatterContext(contentBytes, typeof(DummyClass));
 
        // Act & Assert
        await Assert.ThrowsAsync<InputFormatterException>(async () => await formatter.ReadAsync(context));
    }
 
    [Fact]
    public async Task ReadAsync_ReadsWhenConfiguredWithRootName()
    {
        // Arrange
        var expectedInt = 10;
        var SubstituteRootName = "SomeOtherClass";
        var SubstituteRootNamespace = "http://tempuri.org";
 
        var input = string.Format(
            CultureInfo.InvariantCulture,
            "<{0} xmlns=\"{1}\"><SampleInt xmlns=\"\">{2}</SampleInt></{0}>",
            SubstituteRootName,
            SubstituteRootNamespace,
            expectedInt);
 
        var dictionary = new XmlDictionary();
        var settings = new DataContractSerializerSettings
        {
            RootName = dictionary.Add(SubstituteRootName),
            RootNamespace = dictionary.Add(SubstituteRootNamespace)
        };
        var formatter = new XmlDataContractSerializerInputFormatter(new MvcOptions())
        {
            SerializerSettings = settings
        };
        var contentBytes = Encoding.UTF8.GetBytes(input);
        var context = GetInputFormatterContext(contentBytes, typeof(DummyClass));
 
        // Act
        var result = await formatter.ReadAsync(context);
 
        // Assert
        Assert.NotNull(result);
        Assert.False(result.HasError);
        var model = Assert.IsType<DummyClass>(result.Model);
        Assert.Equal(expectedInt, model.SampleInt);
    }
 
    [Fact]
    public async Task ReadAsync_ThrowsWhenNotConfiguredWithKnownTypes()
    {
        // Arrange
        var KnownTypeName = "SomeDummyClass";
        var InstanceNamespace = "http://www.w3.org/2001/XMLSchema-instance";
 
        var input = string.Format(
            CultureInfo.InvariantCulture,
            "<DummyClass i:type=\"{0}\" xmlns:i=\"{1}\"><SampleInt>1</SampleInt>"
            + "<SampleString>Some text</SampleString></DummyClass>",
            KnownTypeName,
            InstanceNamespace);
        var formatter = new XmlDataContractSerializerInputFormatter(new MvcOptions());
        var contentBytes = Encoding.UTF8.GetBytes(input);
        var context = GetInputFormatterContext(contentBytes, typeof(DummyClass));
 
        // Act & Assert
        await Assert.ThrowsAsync<InputFormatterException>(async () => await formatter.ReadAsync(context));
    }
 
    [Fact]
    public async Task ReadAsync_ReadsWhenConfiguredWithKnownTypes()
    {
        // Arrange
        var expectedInt = 10;
        var expectedString = "TestString";
        var KnownTypeName = "SomeDummyClass";
        var InstanceNamespace = "http://www.w3.org/2001/XMLSchema-instance";
 
        var input = string.Format(
            CultureInfo.InvariantCulture,
            "<DummyClass i:type=\"{0}\" xmlns:i=\"{1}\"><SampleInt>{2}</SampleInt>"
            + "<SampleString>{3}</SampleString></DummyClass>",
            KnownTypeName,
            InstanceNamespace,
            expectedInt,
            expectedString);
        var settings = new DataContractSerializerSettings
        {
            KnownTypes = new[] { typeof(SomeDummyClass) }
        };
        var formatter = new XmlDataContractSerializerInputFormatter(new MvcOptions())
        {
            SerializerSettings = settings
        };
        var contentBytes = Encoding.UTF8.GetBytes(input);
        var context = GetInputFormatterContext(contentBytes, typeof(DummyClass));
 
        // Act
        var result = await formatter.ReadAsync(context);
 
        // Assert
        Assert.NotNull(result);
        Assert.False(result.HasError);
        var model = Assert.IsType<SomeDummyClass>(result.Model);
        Assert.Equal(expectedInt, model.SampleInt);
        Assert.Equal(expectedString, model.SampleString);
    }
 
    private InputFormatterContext GetInputFormatterContext(byte[] contentBytes, Type modelType)
    {
        var httpContext = GetHttpContext(contentBytes);
        return GetInputFormatterContext(httpContext, modelType);
    }
 
    private InputFormatterContext GetInputFormatterContext(HttpContext httpContext, Type modelType)
    {
        var provider = new EmptyModelMetadataProvider();
        var metadata = provider.GetMetadataForType(modelType);
        return new InputFormatterContext(
            httpContext,
            modelName: string.Empty,
            modelState: new ModelStateDictionary(),
            metadata: metadata,
            readerFactory: new TestHttpRequestStreamReaderFactory().CreateReader);
    }
 
    private static HttpContext GetHttpContext(
        byte[] contentBytes,
        string contentType = "application/xml")
    {
        var request = new Mock<HttpRequest>();
        var headers = new Mock<IHeaderDictionary>();
        request.SetupGet(r => r.Headers).Returns(headers.Object);
        request.SetupGet(f => f.Body).Returns(new MemoryStream(contentBytes));
        request.SetupGet(f => f.ContentType).Returns(contentType);
        request.SetupGet(f => f.ContentLength).Returns(contentBytes.Length);
 
        var httpContext = new Mock<HttpContext>();
        var features = new Mock<IFeatureCollection>();
        httpContext.SetupGet(c => c.Request).Returns(request.Object);
        httpContext.SetupGet(c => c.Request).Returns(request.Object);
        httpContext.SetupGet(c => c.Features).Returns(features.Object);
        return httpContext.Object;
    }
 
    private class TestXmlDataContractSerializerInputFormatter : XmlDataContractSerializerInputFormatter
    {
        public int createSerializerCalledCount = 0;
 
        public TestXmlDataContractSerializerInputFormatter()
            : base(new MvcOptions())
        {
        }
 
        protected override DataContractSerializer CreateSerializer(Type type)
        {
            createSerializerCalledCount++;
            return base.CreateSerializer(type);
        }
    }
 
    private class TestResponseFeature : HttpResponseFeature
    {
        public override void OnCompleted(Func<object, Task> callback, object state)
        {
            // do not do anything
        }
    }
 
    private class VerifyDisposeFileBufferingReadStream : FileBufferingReadStream
    {
        public bool Disposed { get; private set; }
        public VerifyDisposeFileBufferingReadStream(Stream inner, int memoryThreshold) : base(inner, memoryThreshold)
        {
        }
 
        protected override void Dispose(bool disposing)
        {
            Disposed = true;
            base.Dispose(disposing);
        }
 
        public override ValueTask DisposeAsync()
        {
            Disposed = true;
            return base.DisposeAsync();
        }
    }
}