#region License Information
/* HeuristicLab
* Copyright (C) 2002-2018 Heuristic and Evolutionary Algorithms Laboratory (HEAL)
*
* This file is part of HeuristicLab.
*
* HeuristicLab is free software: you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation, either version 3 of the License, or
* (at your option) any later version.
*
* HeuristicLab is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with HeuristicLab. If not, see .
*/
#endregion
using System;
using System.Collections.Generic;
using System.Linq;
using System.Reflection;
using System.Reflection.Emit;
using System.Text;
using HeuristicLab.Persistence.Core;
using HeuristicLab.Persistence.Interfaces;
namespace HeuristicLab.Persistence.Default.CompositeSerializers.Storable {
///
/// Intended for serialization of all custom classes. Classes should have the
/// [StorableClass] attribute set. The default mode is to serialize
/// members with the [Storable] attribute set. Alternatively the
/// storable mode can be set to AllFields, AllProperties
/// or AllFieldsAndAllProperties.
///
[StorableClass]
public sealed class StorableSerializer : ICompositeSerializer {
public StorableSerializer() {
accessorListCache = new AccessorListCache();
accessorCache = new AccessorCache();
constructorCache = new Dictionary();
hookCache = new Dictionary>();
}
[StorableConstructor]
private StorableSerializer(bool deserializing) : this() { }
#region ICompositeSerializer implementation
///
/// Priority 200, one of the first default composite serializers to try.
///
///
public int Priority {
get { return 200; }
}
///
/// Determines for every type whether the composite serializer is applicable.
///
/// The type.
///
/// true if this instance can serialize the specified type; otherwise, false.
///
public bool CanSerialize(Type type) {
var markedStorable = StorableReflection.HasStorableClassAttribute(type);
if (GetConstructor(type) == null)
if (markedStorable)
throw new Exception("[Storable] type has no default constructor and no [StorableConstructor]");
else
return false;
if (!StorableReflection.IsEmptyOrStorableType(type, true))
if (markedStorable)
throw new Exception("[Storable] type has non emtpy, non [Storable] base classes");
else
return false;
return true;
}
///
/// Give a reason if possibly why the given type cannot be serialized by this
/// ICompositeSerializer.
///
/// The type.
///
/// A string justifying why type cannot be serialized.
///
public string JustifyRejection(Type type) {
var sb = new StringBuilder();
if (GetConstructor(type) == null)
sb.Append("class has no default constructor and no [StorableConstructor]");
if (!StorableReflection.IsEmptyOrStorableType(type, true))
sb.Append("class (or one of its bases) is not empty and not marked [Storable]; ");
return sb.ToString();
}
///
/// Creates the meta info.
///
/// The object.
/// A list of storable components.
public IEnumerable CreateMetaInfo(object o) {
InvokeHook(HookType.BeforeSerialization, o);
return new Tag[] { };
}
///
/// Decompose an object into s, the tag name can be null,
/// the order in which elements are generated is guaranteed to be
/// the same as they will be supplied to the Populate method.
///
/// An object.
/// An enumerable of s.
public IEnumerable Decompose(object obj) {
return from accessor in GetStorableAccessors(obj.GetType())
where accessor.Get != null
select new Tag(accessor.Name, accessor.Get(obj));
}
///
/// Create an instance of the object using the provided meta information.
///
/// A type.
/// The meta information.
/// A fresh instance of the provided type.
public object CreateInstance(Type type, IEnumerable metaInfo) {
try {
return GetConstructor(type)();
} catch (TargetInvocationException x) {
throw new PersistenceException(
"Could not instantiate storable object: Encountered exception during constructor call",
x.InnerException);
}
}
///
/// Populates the specified instance.
///
/// The instance.
/// The objects.
/// The type.
public void Populate(object instance, IEnumerable objects, Type type) {
var memberDict = new Dictionary();
var iter = objects.GetEnumerator();
while (iter.MoveNext()) {
memberDict.Add(iter.Current.Name, iter.Current);
}
foreach (var accessor in GetStorableAccessors(instance.GetType())) {
if (accessor.Set != null) {
if (memberDict.ContainsKey(accessor.Name)) {
accessor.Set(instance, memberDict[accessor.Name].Value);
} else if (accessor.DefaultValue != null) {
accessor.Set(instance, accessor.DefaultValue);
}
}
}
InvokeHook(HookType.AfterDeserialization, instance);
}
#endregion
#region constants & private data types
private const BindingFlags ALL_CONSTRUCTORS =
BindingFlags.Instance | BindingFlags.Public | BindingFlags.NonPublic;
private sealed class HookDesignator : Tuple {
public HookDesignator(Type type, HookType hookType) : base(type, hookType) { }
}
private sealed class AccessorListCache : Dictionary> { }
private sealed class AccessorCache : Dictionary { }
private delegate object Constructor();
#endregion
#region caches
private readonly AccessorListCache accessorListCache;
private readonly AccessorCache accessorCache;
private readonly Dictionary constructorCache;
private readonly Dictionary> hookCache;
#endregion
#region attribute access
private IEnumerable GetStorableAccessors(Type type) {
lock (accessorListCache) {
if (accessorListCache.ContainsKey(type))
return accessorListCache[type];
var storableMembers = StorableReflection
.GenerateStorableMembers(type)
.Select(GetMemberAccessor)
.ToList();
accessorListCache[type] = storableMembers;
return storableMembers;
}
}
private DataMemberAccessor GetMemberAccessor(StorableMemberInfo mi) {
lock (accessorCache) {
if (accessorCache.ContainsKey(mi.MemberInfo))
return new DataMemberAccessor(accessorCache[mi.MemberInfo], mi.DisentangledName, mi.DefaultValue);
var dma = new DataMemberAccessor(mi.MemberInfo, mi.DisentangledName, mi.DefaultValue);
accessorCache[mi.MemberInfo] = dma;
return dma;
}
}
private Constructor GetConstructor(Type type) {
lock (constructorCache) {
if (constructorCache.ContainsKey(type))
return constructorCache[type];
var c = FindStorableConstructor(type) ?? GetDefaultConstructor(type);
constructorCache.Add(type, c);
return c;
}
}
private Constructor GetDefaultConstructor(Type type) {
var ci = type.GetConstructor(ALL_CONSTRUCTORS, null, Type.EmptyTypes, null);
if (ci == null)
return null;
var dm = new DynamicMethod("", typeof(object), null, type, true);
var ilgen = dm.GetILGenerator();
ilgen.Emit(OpCodes.Newobj, ci);
ilgen.Emit(OpCodes.Ret);
return (Constructor)dm.CreateDelegate(typeof(Constructor));
}
private Constructor FindStorableConstructor(Type type) {
foreach (var ci in type
.GetConstructors(ALL_CONSTRUCTORS)
.Where(ci => ci.GetCustomAttributes(typeof(StorableConstructorAttribute), false).Length > 0)) {
if (ci.GetParameters().Length != 1 ||
ci.GetParameters()[0].ParameterType != typeof(bool))
throw new PersistenceException("StorableConstructor must have exactly one argument of type bool");
var dm = new DynamicMethod("", typeof(object), null, type, true);
var ilgen = dm.GetILGenerator();
ilgen.Emit(OpCodes.Ldc_I4_1); // load true
ilgen.Emit(OpCodes.Newobj, ci);
ilgen.Emit(OpCodes.Ret);
return (Constructor)dm.CreateDelegate(typeof(Constructor));
}
return null;
}
private void InvokeHook(HookType hookType, object obj) {
if (obj == null)
throw new ArgumentNullException("obj");
foreach (var hook in GetHooks(hookType, obj.GetType())) {
hook(obj);
}
}
private IEnumerable GetHooks(HookType hookType, Type type) {
lock (hookCache) {
List hooks;
var designator = new HookDesignator(type, hookType);
hookCache.TryGetValue(designator, out hooks);
if (hooks != null)
return hooks;
hooks = new List(StorableReflection.CollectHooks(hookType, type));
hookCache.Add(designator, hooks);
return hooks;
}
}
#endregion
}
}