/* 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);
}
}
}