// Copyright (c) .NET Foundation. All rights reserved. // Licensed under the Apache License, Version 2.0. See License.txt in the project root for license information. using System; using System.Collections.Generic; #if DNX451 using System.Globalization; using System.Linq; #endif using System.Threading.Tasks; using Microsoft.AspNet.Http.Internal; #if DNX451 using Moq; #endif using Xunit; namespace Microsoft.AspNet.Mvc.ModelBinding.Test { public class CollectionModelBinderTest { #if DNX451 [Fact] public async Task BindComplexCollectionFromIndexes_FiniteIndexes() { // Arrange var valueProvider = new SimpleValueProvider { { "someName[foo]", "42" }, { "someName[baz]", "200" } }; var bindingContext = GetModelBindingContext(valueProvider); var binder = new CollectionModelBinder(); // Act var boundCollection = await binder.BindComplexCollectionFromIndexes(bindingContext, new[] { "foo", "bar", "baz" }); // Assert Assert.Equal(new[] { 42, 0, 200 }, boundCollection.Model.ToArray()); Assert.Equal( new[] { "someName[foo]", "someName[baz]" }, boundCollection.ValidationNode.ChildNodes.Select(o => o.Key).ToArray()); } [Fact] public async Task BindComplexCollectionFromIndexes_InfiniteIndexes() { // Arrange var valueProvider = new SimpleValueProvider { { "someName[0]", "42" }, { "someName[1]", "100" }, { "someName[3]", "400" } }; var bindingContext = GetModelBindingContext(valueProvider); var binder = new CollectionModelBinder(); // Act var boundCollection = await binder.BindComplexCollectionFromIndexes(bindingContext, indexNames: null); // Assert Assert.Equal(new[] { 42, 100 }, boundCollection.Model.ToArray()); Assert.Equal( new[] { "someName[0]", "someName[1]" }, boundCollection.ValidationNode.ChildNodes.Select(o => o.Key).ToArray()); } [Theory] [InlineData(false)] [InlineData(true)] public async Task BindModel_ComplexCollection_Succeeds(bool isReadOnly) { // Arrange var valueProvider = new SimpleValueProvider { { "someName.index", new[] { "foo", "bar", "baz" } }, { "someName[foo]", "42" }, { "someName[bar]", "100" }, { "someName[baz]", "200" } }; var bindingContext = GetModelBindingContext(valueProvider, isReadOnly); var modelState = bindingContext.ModelState; var binder = new CollectionModelBinder(); // Act var result = await binder.BindModelAsync(bindingContext); // Assert Assert.NotNull(result); Assert.True(result.IsModelSet); var list = Assert.IsAssignableFrom>(result.Model); Assert.Equal(new[] { 42, 100, 200 }, list.ToArray()); Assert.True(modelState.IsValid); } [Theory] [InlineData(false)] [InlineData(true)] public async Task BindModel_ComplexCollection_BindingContextModelNonNull_Succeeds(bool isReadOnly) { // Arrange var valueProvider = new SimpleValueProvider { { "someName.index", new[] { "foo", "bar", "baz" } }, { "someName[foo]", "42" }, { "someName[bar]", "100" }, { "someName[baz]", "200" } }; var bindingContext = GetModelBindingContext(valueProvider, isReadOnly); var modelState = bindingContext.ModelState; var list = new List(); bindingContext.Model = list; var binder = new CollectionModelBinder(); // Act var result = await binder.BindModelAsync(bindingContext); // Assert Assert.NotNull(result); Assert.True(result.IsModelSet); Assert.Same(list, result.Model); Assert.Equal(new[] { 42, 100, 200 }, list.ToArray()); Assert.True(modelState.IsValid); } [Theory] [InlineData(false)] [InlineData(true)] public async Task BindModel_SimpleCollection_Succeeds(bool isReadOnly) { // Arrange var valueProvider = new SimpleValueProvider { { "someName", new[] { "42", "100", "200" } } }; var bindingContext = GetModelBindingContext(valueProvider, isReadOnly); var modelState = bindingContext.ModelState; var binder = new CollectionModelBinder(); // Act var result = await binder.BindModelAsync(bindingContext); // Assert Assert.NotNull(result); Assert.True(result.IsModelSet); var list = Assert.IsAssignableFrom>(result.Model); Assert.Equal(new[] { 42, 100, 200 }, list.ToArray()); } [Theory] [InlineData(false)] [InlineData(true)] public async Task BindModel_SimpleCollection_BindingContextModelNonNull_Succeeds(bool isReadOnly) { // Arrange var valueProvider = new SimpleValueProvider { { "someName", new[] { "42", "100", "200" } } }; var bindingContext = GetModelBindingContext(valueProvider, isReadOnly); var modelState = bindingContext.ModelState; var list = new List(); bindingContext.Model = list; var binder = new CollectionModelBinder(); // Act var result = await binder.BindModelAsync(bindingContext); // Assert Assert.NotNull(result); Assert.True(result.IsModelSet); Assert.Same(list, result.Model); Assert.Equal(new[] { 42, 100, 200 }, list.ToArray()); } [Fact] public async Task BindModelAsync_SimpleCollectionWithNullValue_Succeeds() { // Arrange var binder = new CollectionModelBinder(); var valueProvider = new SimpleValueProvider { { "someName", null }, }; var bindingContext = GetModelBindingContext(valueProvider, isReadOnly: false); var modelState = bindingContext.ModelState; // Act var result = await binder.BindModelAsync(bindingContext); // Assert Assert.NotNull(result); Assert.True(result.IsModelSet); Assert.NotNull(result.Model); Assert.NotNull(result.ValidationNode); var model = Assert.IsType>(result.Model); Assert.Empty(model); } #endif [Fact] public async Task BindSimpleCollection_RawValueIsEmptyCollection_ReturnsEmptyList() { // Arrange var binder = new CollectionModelBinder(); var context = new ModelBindingContext() { OperationBindingContext = new OperationBindingContext() { MetadataProvider = TestModelMetadataProvider.CreateDefaultProvider(), }, }; // Act var boundCollection = await binder.BindSimpleCollection(context, new ValueProviderResult(new string[0])); // Assert Assert.NotNull(boundCollection.Model); Assert.Empty(boundCollection.Model); } [Fact] public async Task CollectionModelBinder_DoesNotCreateCollection_IfIsTopLevelObjectAndIsFirstChanceBinding() { // Arrange var binder = new CollectionModelBinder(); var context = CreateContext(); context.IsTopLevelObject = true; context.IsFirstChanceBinding = true; // Explicit prefix and empty model name both ignored. context.BinderModelName = "prefix"; context.ModelName = string.Empty; var metadataProvider = context.OperationBindingContext.MetadataProvider; context.ModelMetadata = metadataProvider.GetMetadataForType(typeof(List)); context.ValueProvider = new TestValueProvider(new Dictionary()); // Act var result = await binder.BindModelAsync(context); // Assert Assert.Null(result); } [Fact] public async Task CollectionModelBinder_CreatesEmptyCollection_IfIsTopLevelObjectAndNotIsFirstChanceBinding() { // Arrange var binder = new CollectionModelBinder(); var context = CreateContext(); context.IsTopLevelObject = true; // Lack of prefix and non-empty model name both ignored. context.ModelName = "modelName"; var metadataProvider = context.OperationBindingContext.MetadataProvider; context.ModelMetadata = metadataProvider.GetMetadataForType(typeof(List)); context.ValueProvider = new TestValueProvider(new Dictionary()); // Act var result = await binder.BindModelAsync(context); // Assert Assert.NotNull(result); Assert.Empty(Assert.IsType>(result.Model)); Assert.Equal("modelName", result.Key); Assert.True(result.IsModelSet); Assert.Same(result.ValidationNode.Model, result.Model); Assert.Same(result.ValidationNode.Key, result.Key); Assert.Same(result.ValidationNode.ModelMetadata, context.ModelMetadata); } // Setup like CollectionModelBinder_CreatesEmptyCollection_IfIsTopLevelObjectAndNotIsFirstChanceBinding except // Model already has a value. [Fact] public async Task CollectionModelBinder_DoesNotCreateEmptyCollection_IfModelNonNull() { // Arrange var binder = new CollectionModelBinder(); var context = CreateContext(); context.IsTopLevelObject = true; var list = new List(); context.Model = list; // Lack of prefix and non-empty model name both ignored. context.ModelName = "modelName"; var metadataProvider = context.OperationBindingContext.MetadataProvider; context.ModelMetadata = metadataProvider.GetMetadataForType(typeof(List)); context.ValueProvider = new TestValueProvider(new Dictionary()); // Act var result = await binder.BindModelAsync(context); // Assert Assert.NotNull(result); Assert.Same(list, result.Model); Assert.Empty(list); Assert.Equal("modelName", result.Key); Assert.True(result.IsModelSet); Assert.Same(result.ValidationNode.Model, result.Model); Assert.Same(result.ValidationNode.Key, result.Key); Assert.Same(result.ValidationNode.ModelMetadata, context.ModelMetadata); } [Theory] [InlineData("")] [InlineData("param")] public async Task CollectionModelBinder_DoesNotCreateCollection_IfNotIsTopLevelObject(string prefix) { // Arrange var binder = new CollectionModelBinder(); var context = CreateContext(); context.ModelName = ModelNames.CreatePropertyModelName(prefix, "ListProperty"); var metadataProvider = context.OperationBindingContext.MetadataProvider; context.ModelMetadata = metadataProvider.GetMetadataForProperty( typeof(ModelWithListProperty), nameof(ModelWithListProperty.ListProperty)); context.ValueProvider = new TestValueProvider(new Dictionary()); // Act var result = await binder.BindModelAsync(context); // Assert Assert.Null(result); } // Model type -> can create instance. public static TheoryData CanCreateInstanceData { get { return new TheoryData { { typeof(IEnumerable), true }, { typeof(ICollection), true }, { typeof(IList), true }, { typeof(List), true }, { typeof(LinkedList), true }, { typeof(ISet), false }, { typeof(ListWithInternalConstructor), false }, { typeof(ListWithThrowingConstructor), false }, }; } } [Theory] [MemberData(nameof(CanCreateInstanceData))] public void CanCreateInstance_ReturnsExpectedValue(Type modelType, bool expectedResult) { // Arrange var binder = new CollectionModelBinder(); // Act var result = binder.CanCreateInstance(modelType); // Assert Assert.Equal(expectedResult, result); } #if DNX451 [Fact] public async Task BindSimpleCollection_SubBindingSucceeds() { // Arrange var culture = new CultureInfo("fr-FR"); var bindingContext = GetModelBindingContext(new SimpleValueProvider()); ModelValidationNode childValidationNode = null; Mock.Get(bindingContext.OperationBindingContext.ModelBinder) .Setup(o => o.BindModelAsync(It.IsAny())) .Returns((ModelBindingContext mbc) => { Assert.Equal("someName", mbc.ModelName); childValidationNode = new ModelValidationNode("someName", mbc.ModelMetadata, mbc.Model); return Task.FromResult(new ModelBindingResult(42, mbc.ModelName, true, childValidationNode)); }); var modelBinder = new CollectionModelBinder(); // Act var boundCollection = await modelBinder.BindSimpleCollection( bindingContext, new ValueProviderResult(new string[] { "0" })); // Assert Assert.Equal(new[] { 42 }, boundCollection.Model.ToArray()); Assert.Equal(new[] { childValidationNode }, boundCollection.ValidationNode.ChildNodes.ToArray()); } private static ModelBindingContext GetModelBindingContext( IValueProvider valueProvider, bool isReadOnly = false) { var metadataProvider = new TestModelMetadataProvider(); metadataProvider.ForType>().BindingDetails(bd => bd.IsReadOnly = isReadOnly); var bindingContext = new ModelBindingContext { ModelMetadata = metadataProvider.GetMetadataForType(typeof(IList)), ModelName = "someName", ValueProvider = valueProvider, OperationBindingContext = new OperationBindingContext { ModelBinder = CreateIntBinder(), MetadataProvider = metadataProvider } }; return bindingContext; } private static IModelBinder CreateIntBinder() { Mock mockIntBinder = new Mock(); mockIntBinder .Setup(o => o.BindModelAsync(It.IsAny())) .Returns((ModelBindingContext mbc) => { var value = mbc.ValueProvider.GetValue(mbc.ModelName); if (value != ValueProviderResult.None) { var model = value.ConvertTo(mbc.ModelType); var modelValidationNode = new ModelValidationNode(mbc.ModelName, mbc.ModelMetadata, model); return Task.FromResult(new ModelBindingResult( model, mbc.ModelName, model != null, modelValidationNode)); } return Task.FromResult(null); }); return mockIntBinder.Object; } #endif private static ModelBindingContext CreateContext() { var modelBindingContext = new ModelBindingContext() { OperationBindingContext = new OperationBindingContext() { HttpContext = new DefaultHttpContext(), MetadataProvider = new TestModelMetadataProvider(), } }; return modelBindingContext; } private class ModelWithListProperty { public List ListProperty { get; set; } } private class ModelWithSimpleProperties { public int Id { get; set; } public string Name { get; set; } } private class ListWithInternalConstructor : List { internal ListWithInternalConstructor() : base() { } } private class ListWithThrowingConstructor : List { public ListWithThrowingConstructor() : base() { throw new ApplicationException("No, don't do this."); } } } }