4
0

ResourceCollectionMerger.cs 7.0 KB

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