#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; using System.Collections.Generic; using System.Collections.Specialized; using System.Linq; using System.Reflection; using System.Security; using System.Threading; namespace HeuristicLab.Common { [AttributeUsage(AttributeTargets.Field)] // this attribute can be used to mark fields that should be excluded from object graph traversal public class ExcludeFromObjectGraphTraversalAttribute : Attribute { } public static class ObjectExtensions { public static IEnumerable ToEnumerable(this T obj) { yield return obj; } public static IEnumerable GetObjectGraphObjects(this object obj, HashSet excludedMembers = null, bool excludeStaticMembers = false) { if (obj == null) return Enumerable.Empty(); if (excludedMembers == null) excludedMembers = new HashSet(); var fieldInfos = new Dictionary(); var objects = new HashSet(); var stack = new Stack(); stack.Push(obj); while (stack.Count > 0) { object current = stack.Pop(); objects.Add(current); foreach (object o in GetChildObjects(current, excludedMembers, excludeStaticMembers, fieldInfos)) { if (o == null) continue; if (ExcludeType(o.GetType())) continue; if (objects.Contains(o)) continue; stack.Push(o); } } return objects; } /// /// Types not collected: /// * System.Delegate /// * System.Reflection.Pointer /// * Primitives (Boolean, Byte, SByte, Int16, UInt16, Int32, UInt32, Int64, UInt64, IntPtr, UIntPtr, Char, Double, Single) /// * string, decimal, DateTime /// * Arrays of types not collected /// * All types from System.Reflection.Emit /// private static bool ExcludeType(Type type) { return type.IsPrimitive || type == typeof(string) || type == typeof(string[]) || type == typeof(decimal) || type == typeof(decimal[]) || type == typeof(DateTime) || type == typeof(DateTime[]) || typeof(Delegate).IsAssignableFrom(type) || typeof(Pointer).IsAssignableFrom(type) || type.Namespace == "System.Reflection.Emit" || type.Namespace == "System.Net.Sockets" || (type.HasElementType && ExcludeType(type.GetElementType())); } private static IEnumerable GetChildObjects(object obj, HashSet excludedMembers, bool excludeStaticMembers, Dictionary fieldInfos) { Type type = obj.GetType(); if (type.IsSubclassOfRawGeneric(typeof(ThreadLocal<>))) { PropertyInfo info = type.GetProperty("Value"); object value = info.GetValue(obj, null); if (value != null && !excludedMembers.Contains(value)) yield return value; } else if (type.IsSubclassOfRawGeneric(typeof(Dictionary<,>)) || type.IsSubclassOfRawGeneric(typeof(SortedDictionary<,>)) || type.IsSubclassOfRawGeneric(typeof(SortedList<,>)) || obj is SortedList || obj is OrderedDictionary || obj is ListDictionary || obj is Hashtable) { var dictionary = obj as IDictionary; foreach (object value in dictionary.Keys) { if (excludedMembers.Contains(value)) continue; yield return value; } foreach (object value in dictionary.Values) { if (excludedMembers.Contains(value)) continue; yield return value; } } else if (type.IsArray || type.IsSubclassOfRawGeneric(typeof(HashSet<>))) { var enumerable = obj as IEnumerable; foreach (var value in enumerable) { if (excludedMembers.Contains(value)) continue; yield return value; } } else { FieldInfo[] fieldInfo; if (!fieldInfos.TryGetValue(type, out fieldInfo)) { fieldInfo = type.GetAllFields() .Where(fi => !Attribute.IsDefined(fi, typeof(ExcludeFromObjectGraphTraversalAttribute))) .ToArray(); fieldInfos.Add(type, fieldInfo); } foreach (FieldInfo f in fieldInfo) { if (excludeStaticMembers && f.IsStatic) continue; object fieldValue; try { fieldValue = f.GetValue(obj); } catch (SecurityException) { continue; } if (excludedMembers.Contains(fieldValue)) continue; yield return fieldValue; } } } } }