-
Notifications
You must be signed in to change notification settings - Fork 582
/
Copy pathUtils.cs
146 lines (130 loc) · 4.91 KB
/
Utils.cs
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
// Copyright (c) Microsoft Corporation.
// Licensed under the MIT License.
using System;
using System.Collections;
using System.Collections.Generic;
using System.Dynamic;
using System.Linq;
using System.Linq.Dynamic.Core;
namespace RulesEngine.HelperFunctions
{
public static class Utils
{
/* valid only for netstandard 2.0 */
#if NETSTANDARD2_0 || NETSTANDARD2_1
public static IEnumerable<TSource> DistinctBy<TSource, TKey>(this IEnumerable<TSource> source, Func<TSource, TKey> keySelector)
{
HashSet<TKey> seenKeys = new HashSet<TKey>();
foreach (TSource element in source)
{
if (seenKeys.Add(keySelector(element)))
{
yield return element;
}
}
}
#endif
/* **************************** */
public static object GetTypedObject(dynamic input)
{
if (input is ExpandoObject)
{
Type type = CreateAbstractClassType(input);
return CreateObject(type, input);
}
else
{
return input;
}
}
public static Type CreateAbstractClassType(dynamic input)
{
List<DynamicProperty> props = new List<DynamicProperty>();
if (input == null)
{
return typeof(object);
}
if (!(input is ExpandoObject))
{
return input.GetType();
}
else
{
foreach (var expando in (IDictionary<string, object>)input)
{
Type value;
if (expando.Value is IList)
{
if (((IList)expando.Value).Count == 0)
value = typeof(List<object>);
else
{
var internalType = CreateAbstractClassType(((IList)expando.Value)[0]);
value = new List<object>().Cast(internalType).ToList(internalType).GetType();
}
}
else
{
value = CreateAbstractClassType(expando.Value);
}
props.Add(new DynamicProperty(expando.Key, value));
}
}
var type = DynamicClassFactory.CreateType(props);
return type;
}
public static object CreateObject(Type type, dynamic input)
{
if (!(input is ExpandoObject))
{
return Convert.ChangeType(input, type);
}
object obj = Activator.CreateInstance(type);
var typeProps = type.GetProperties().ToDictionary(c => c.Name);
foreach (var expando in (IDictionary<string, object>)input)
{
if (typeProps.ContainsKey(expando.Key) &&
expando.Value != null && (expando.Value.GetType().Name != "DBNull" || expando.Value != DBNull.Value))
{
object val;
var propInfo = typeProps[expando.Key];
if (expando.Value is ExpandoObject)
{
var propType = propInfo.PropertyType;
val = CreateObject(propType, expando.Value);
}
else if (expando.Value is IList)
{
var internalType = propInfo.PropertyType.GenericTypeArguments.FirstOrDefault() ?? typeof(object);
var temp = (IList)expando.Value;
var newList = new List<object>().Cast(internalType).ToList(internalType);
for (int i = 0; i < temp.Count; i++)
{
var child = CreateObject(internalType, temp[i]);
newList.Add(child);
};
val = newList;
}
else
{
val = expando.Value;
}
propInfo.SetValue(obj, val, null);
}
}
return obj;
}
private static IEnumerable Cast(this IEnumerable self, Type innerType)
{
var methodInfo = typeof(Enumerable).GetMethod("Cast");
var genericMethod = methodInfo.MakeGenericMethod(innerType);
return genericMethod.Invoke(null, new[] { self }) as IEnumerable;
}
private static IList ToList(this IEnumerable self, Type innerType)
{
var methodInfo = typeof(Enumerable).GetMethod("ToList");
var genericMethod = methodInfo.MakeGenericMethod(innerType);
return genericMethod.Invoke(null, new[] { self }) as IList;
}
}
}