/* Copyright 2010-present MongoDB Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ using System; using System.Collections; using System.Collections.Generic; using System.Reflection; namespace MongoDB.Bson.Serialization.Serializers { /// /// Represents a serializer for a class that implements IEnumerable. /// /// The type of the value. public class EnumerableInterfaceImplementerSerializer : EnumerableInterfaceImplementerSerializerBase, IChildSerializerConfigurable where TValue : class, IList, new() { // constructors /// /// Initializes a new instance of the class. /// public EnumerableInterfaceImplementerSerializer() { } /// /// Initializes a new instance of the class. /// /// The item serializer. public EnumerableInterfaceImplementerSerializer(IBsonSerializer itemSerializer) : base(itemSerializer) { } /// /// Initializes a new instance of the class. /// /// public EnumerableInterfaceImplementerSerializer(IBsonSerializerRegistry serializerRegistry) : base(serializerRegistry) { } // public methods /// /// Returns a serializer that has been reconfigured with the specified item serializer. /// /// The item serializer. /// The reconfigured serializer. public EnumerableInterfaceImplementerSerializer WithItemSerializer(IBsonSerializer itemSerializer) { return new EnumerableInterfaceImplementerSerializer(itemSerializer); } // protected methods /// /// Creates the accumulator. /// /// The accumulator. protected override object CreateAccumulator() { return new TValue(); } // explicit interface implementations IBsonSerializer IChildSerializerConfigurable.ChildSerializer { get { return ItemSerializer; } } IBsonSerializer IChildSerializerConfigurable.WithChildSerializer(IBsonSerializer childSerializer) { return WithItemSerializer(childSerializer); } } /// /// Represents a serializer for a class that implementes . /// /// The type of the value. /// The type of the item. public class EnumerableInterfaceImplementerSerializer : EnumerableInterfaceImplementerSerializerBase, IChildSerializerConfigurable where TValue : class, IEnumerable { // constructors /// /// Initializes a new instance of the class. /// public EnumerableInterfaceImplementerSerializer() { } /// /// Initializes a new instance of the class. /// /// The item serializer. public EnumerableInterfaceImplementerSerializer(IBsonSerializer itemSerializer) : base(itemSerializer) { } /// /// Initializes a new instance of the class. /// /// The serializer registry. public EnumerableInterfaceImplementerSerializer(IBsonSerializerRegistry serializerRegistry) : base(serializerRegistry) { } // public methods /// /// Returns a serializer that has been reconfigured with the specified item serializer. /// /// The item serializer. /// The reconfigured serializer. public EnumerableInterfaceImplementerSerializer WithItemSerializer(IBsonSerializer itemSerializer) { return new EnumerableInterfaceImplementerSerializer(itemSerializer); } // protected methods /// /// Creates the accumulator. /// /// The accumulator. protected override object CreateAccumulator() { return new List(); } /// /// Finalizes the result. /// /// The accumulator. /// The final result. protected override TValue FinalizeResult(object accumulator) { // find and call a constructor that we can pass the accumulator to var accumulatorType = accumulator.GetType(); foreach (var constructorInfo in typeof(TValue).GetTypeInfo().GetConstructors()) { var parameterInfos = constructorInfo.GetParameters(); if (parameterInfos.Length == 1 && parameterInfos[0].ParameterType.GetTypeInfo().IsAssignableFrom(accumulatorType)) { return (TValue)constructorInfo.Invoke(new object[] { accumulator }); } } // otherwise try to find a no-argument constructor and an Add method var valueTypeInfo = typeof(TValue).GetTypeInfo(); var noArgumentConstructorInfo = valueTypeInfo.GetConstructor(new Type[] { }); var addMethodInfo = typeof(TValue).GetTypeInfo().GetMethod("Add", new Type[] { typeof(TItem) }); if (noArgumentConstructorInfo != null && addMethodInfo != null) { var value = (TValue)noArgumentConstructorInfo.Invoke(new Type[] { }); foreach (var item in (IEnumerable)accumulator) { addMethodInfo.Invoke(value, new object[] { item }); } return value; } var message = string.Format("Type '{0}' does not have a suitable constructor or Add method.", typeof(TValue).FullName); throw new BsonSerializationException(message); } // explicit interface implementations IBsonSerializer IChildSerializerConfigurable.ChildSerializer { get { return ItemSerializer; } } IBsonSerializer IChildSerializerConfigurable.WithChildSerializer(IBsonSerializer childSerializer) { return WithItemSerializer((IBsonSerializer)childSerializer); } } }