ResourceCollectionMerger.cs 7.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227
  1. using RackPeek.Domain.Resources;
  2. using System.Reflection;
  3. using System.Text.Json;
  4. using System.Text.Json.Serialization;
  5. using System.Text.Json.Serialization.Metadata;
  6. using RackPeek.Domain.Resources.AccessPoints;
  7. using RackPeek.Domain.Resources.Desktops;
  8. using RackPeek.Domain.Resources.Firewalls;
  9. using RackPeek.Domain.Resources.Laptops;
  10. using RackPeek.Domain.Resources.Routers;
  11. using RackPeek.Domain.Resources.Servers;
  12. using RackPeek.Domain.Resources.Services;
  13. using RackPeek.Domain.Resources.Switches;
  14. using RackPeek.Domain.Resources.SystemResources;
  15. using RackPeek.Domain.Resources.UpsUnits;
  16. namespace RackPeek.Domain.Persistence;
  17. public enum MergeMode
  18. {
  19. Replace,
  20. Merge
  21. }
  22. public static class ResourceCollectionMerger
  23. {
  24. private static readonly JsonSerializerOptions CloneJsonOptions = new()
  25. {
  26. PropertyNameCaseInsensitive = true,
  27. WriteIndented = false,
  28. DefaultIgnoreCondition = JsonIgnoreCondition.WhenWritingNull,
  29. ReferenceHandler = ReferenceHandler.IgnoreCycles,
  30. TypeInfoResolver = ResourcePolymorphismResolver.Create()
  31. };
  32. public static List<Resource> Merge(
  33. IEnumerable<Resource> original,
  34. IEnumerable<Resource> incoming,
  35. MergeMode mode)
  36. {
  37. var originalClone = DeepCloneList(original);
  38. var incomingClone = DeepCloneList(incoming);
  39. var result = originalClone.ToDictionary(r => r.Name, r => r, StringComparer.OrdinalIgnoreCase);
  40. foreach (var newResource in incomingClone)
  41. {
  42. if (!result.TryGetValue(newResource.Name, out var existing))
  43. {
  44. result[newResource.Name] = newResource;
  45. continue;
  46. }
  47. if (mode == MergeMode.Replace ||
  48. existing.GetType() != newResource.GetType())
  49. {
  50. result[newResource.Name] = newResource;
  51. continue;
  52. }
  53. DeepMerge(existing, newResource, mode);
  54. }
  55. return result.Values.ToList();
  56. }
  57. private static List<Resource> DeepCloneList(IEnumerable<Resource> resources)
  58. {
  59. var json = JsonSerializer.Serialize(resources, CloneJsonOptions);
  60. return JsonSerializer.Deserialize<List<Resource>>(json, CloneJsonOptions) ?? new List<Resource>();
  61. }
  62. private static void DeepMerge(object target, object source, MergeMode mode)
  63. {
  64. var type = target.GetType();
  65. foreach (var prop in type.GetProperties(BindingFlags.Public | BindingFlags.Instance))
  66. {
  67. if (!prop.CanRead || !prop.CanWrite)
  68. continue;
  69. var sourceValue = prop.GetValue(source);
  70. if (sourceValue == null)
  71. continue;
  72. var targetValue = prop.GetValue(target);
  73. var propType = prop.PropertyType;
  74. // Simple types → overwrite
  75. if (IsSimple(propType))
  76. {
  77. prop.SetValue(target, sourceValue);
  78. continue;
  79. }
  80. // Dictionary
  81. if (IsDictionary(propType))
  82. {
  83. if (mode == MergeMode.Merge && IsDictionaryEmpty(sourceValue))
  84. continue;
  85. MergeDictionaries(targetValue, sourceValue);
  86. continue;
  87. }
  88. // List / collection
  89. if (IsEnumerable(propType))
  90. {
  91. if (mode == MergeMode.Merge && IsEnumerableEmpty(sourceValue))
  92. continue;
  93. prop.SetValue(target, sourceValue);
  94. continue;
  95. }
  96. // Complex object → recursive merge
  97. if (targetValue == null)
  98. {
  99. prop.SetValue(target, sourceValue);
  100. }
  101. else
  102. {
  103. DeepMerge(targetValue, sourceValue, mode);
  104. }
  105. }
  106. }
  107. private static bool IsSimple(Type type)
  108. {
  109. return type.IsPrimitive
  110. || type == typeof(string)
  111. || type == typeof(decimal)
  112. || type == typeof(DateTime)
  113. || type == typeof(Guid)
  114. || Nullable.GetUnderlyingType(type)?.IsPrimitive == true;
  115. }
  116. private static bool IsDictionary(Type type)
  117. {
  118. return type.IsGenericType &&
  119. type.GetGenericTypeDefinition() == typeof(Dictionary<,>);
  120. }
  121. private static bool IsEnumerable(Type type)
  122. {
  123. return typeof(System.Collections.IEnumerable).IsAssignableFrom(type)
  124. && type != typeof(string)
  125. && !IsDictionary(type);
  126. }
  127. private static bool IsEnumerableEmpty(object value)
  128. {
  129. var enumerable = (System.Collections.IEnumerable)value;
  130. return !enumerable.GetEnumerator().MoveNext();
  131. }
  132. private static bool IsDictionaryEmpty(object value)
  133. {
  134. var dict = (System.Collections.IDictionary)value;
  135. return dict.Count == 0;
  136. }
  137. private static void MergeDictionaries(object? target, object source)
  138. {
  139. if (target == null) return;
  140. var targetDict = (System.Collections.IDictionary)target;
  141. var sourceDict = (System.Collections.IDictionary)source;
  142. foreach (var key in sourceDict.Keys)
  143. {
  144. targetDict[key] = sourceDict[key];
  145. }
  146. }
  147. }
  148. internal static class ResourcePolymorphismResolver
  149. {
  150. public static IJsonTypeInfoResolver Create()
  151. {
  152. var resolver = new DefaultJsonTypeInfoResolver();
  153. resolver.Modifiers.Add(typeInfo =>
  154. {
  155. if (typeInfo.Type == typeof(Resource))
  156. {
  157. typeInfo.PolymorphismOptions = new JsonPolymorphismOptions
  158. {
  159. TypeDiscriminatorPropertyName = "kind",
  160. IgnoreUnrecognizedTypeDiscriminators = false,
  161. UnknownDerivedTypeHandling = JsonUnknownDerivedTypeHandling.FailSerialization
  162. };
  163. typeInfo.PolymorphismOptions.DerivedTypes.Add(
  164. new JsonDerivedType(typeof(Server), Server.KindLabel));
  165. typeInfo.PolymorphismOptions.DerivedTypes.Add(
  166. new JsonDerivedType(typeof(Switch), Switch.KindLabel));
  167. typeInfo.PolymorphismOptions.DerivedTypes.Add(
  168. new JsonDerivedType(typeof(Firewall), Firewall.KindLabel));
  169. typeInfo.PolymorphismOptions.DerivedTypes.Add(
  170. new JsonDerivedType(typeof(Router), Router.KindLabel));
  171. typeInfo.PolymorphismOptions.DerivedTypes.Add(
  172. new JsonDerivedType(typeof(Desktop), Desktop.KindLabel));
  173. typeInfo.PolymorphismOptions.DerivedTypes.Add(
  174. new JsonDerivedType(typeof(Laptop), Laptop.KindLabel));
  175. typeInfo.PolymorphismOptions.DerivedTypes.Add(
  176. new JsonDerivedType(typeof(AccessPoint), AccessPoint.KindLabel));
  177. typeInfo.PolymorphismOptions.DerivedTypes.Add(
  178. new JsonDerivedType(typeof(Ups), Ups.KindLabel));
  179. typeInfo.PolymorphismOptions.DerivedTypes.Add(
  180. new JsonDerivedType(typeof(SystemResource), SystemResource.KindLabel));
  181. typeInfo.PolymorphismOptions.DerivedTypes.Add(
  182. new JsonDerivedType(typeof(Service), Service.KindLabel));
  183. }
  184. });
  185. return resolver;
  186. }
  187. }