ILookupDefinition.cs 17 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462
  1. using System;
  2. using System.Collections;
  3. using System.Collections.Generic;
  4. using System.Linq;
  5. using System.Linq.Expressions;
  6. using System.Reflection;
  7. namespace InABox.Core
  8. {
  9. public interface ILookupDefinition<TLookup> where TLookup : BaseObject, new()
  10. {
  11. Filter<TLookup> DefineFilter();
  12. Columns<TLookup> DefineColumns();
  13. Columns<TLookup> RequiredColumns();
  14. SortOrder<TLookup> DefineSortOrder();
  15. string FormatLookup(Dictionary<string, object> values, IEnumerable<string> exclude);
  16. }
  17. public interface ILookupDefinition<TLookup, TEntity> where TLookup : Entity
  18. {
  19. Filter<TLookup> DefineFilter(TEntity[] items);
  20. }
  21. public interface IStaticLookupDefinition<TEntity>
  22. {
  23. Dictionary<object, object> DefineLookups(PropertyInfo property);
  24. }
  25. public static class LookupFactory
  26. {
  27. #region LookupCache
  28. private class LookupCacheEntry
  29. {
  30. public LookupCacheEntry(object factory)
  31. {
  32. DefaultFactory = factory;
  33. CustomFactories = new Dictionary<Type, object>();
  34. StaticFactory = null;
  35. }
  36. public object DefaultFactory { get; }
  37. public Dictionary<Type, object> CustomFactories { get; }
  38. public object StaticFactory { get; set; }
  39. public object GetFactory(Type type = null)
  40. {
  41. if (type == null)
  42. return DefaultFactory;
  43. if (CustomFactories.ContainsKey(type))
  44. return CustomFactories[type];
  45. return DefaultFactory;
  46. }
  47. }
  48. private class LookupCache
  49. {
  50. private static Dictionary<Type, LookupCacheEntry> _cache;
  51. public LookupCache()
  52. {
  53. _cache = new Dictionary<Type, LookupCacheEntry>();
  54. // Load up the default types
  55. var defaulttypes = CoreUtils.TypeList(
  56. AppDomain.CurrentDomain.GetAssemblies(),
  57. x => !x.IsAbstract &&
  58. x.GetInterfaces().Any(i => i.IsGenericType && i.GetGenericTypeDefinition().Equals(typeof(ILookupDefinition<>))
  59. )
  60. );
  61. foreach (var type in defaulttypes)
  62. try
  63. {
  64. var intfs = type.GetInterfaces().Where(i =>
  65. i.IsGenericType
  66. && i.GetGenericTypeDefinition().Equals(typeof(ILookupDefinition<>))
  67. );
  68. foreach (var intf in intfs)
  69. _cache[intf.GenericTypeArguments.First()] = new LookupCacheEntry(Activator.CreateInstance(type));
  70. }
  71. catch (Exception e)
  72. {
  73. Logger.Send(LogType.Error, "", string.Format("*** Unknown Error: {0}\n{1}", e.Message, e.StackTrace));
  74. }
  75. // Now load up the specific types
  76. var specifictypes = CoreUtils.TypeList(
  77. AppDomain.CurrentDomain.GetAssemblies(),
  78. x =>
  79. !x.IsAbstract &&
  80. x.GetInterfaces().Any(i => i.IsGenericType && i.GetGenericTypeDefinition().Equals(typeof(ILookupDefinition<,>))
  81. )
  82. );
  83. foreach (var type in specifictypes)
  84. {
  85. var intfs = type.GetInterfaces().Where(i =>
  86. i.IsGenericType
  87. && i.GetGenericTypeDefinition().Equals(typeof(ILookupDefinition<,>))
  88. );
  89. foreach (var intf in intfs)
  90. {
  91. var lookuptype = intf.GenericTypeArguments.First();
  92. var entitytype = intf.GenericTypeArguments.Last();
  93. if (!_cache.ContainsKey(lookuptype))
  94. _cache[lookuptype] = new LookupCacheEntry(null);
  95. _cache[lookuptype].CustomFactories[entitytype] = Activator.CreateInstance(type);
  96. }
  97. }
  98. // Load up the static lookups
  99. var staticlookups = CoreUtils.TypeList(
  100. AppDomain.CurrentDomain.GetAssemblies(),
  101. x =>
  102. !x.IsAbstract &&
  103. x.GetInterfaces().Any(i => i.IsGenericType && i.GetGenericTypeDefinition().Equals(typeof(IStaticLookupDefinition<>))
  104. )
  105. );
  106. foreach (var type in staticlookups)
  107. {
  108. var intf = type.GetInterfaces().First(i =>
  109. i.IsGenericType
  110. && i.GetGenericTypeDefinition().Equals(typeof(IStaticLookupDefinition<>))
  111. );
  112. var iType = intf.GenericTypeArguments.First();
  113. if (!_cache.ContainsKey(iType))
  114. _cache[iType] = new LookupCacheEntry(null);
  115. _cache[iType].StaticFactory = Activator.CreateInstance(type);
  116. }
  117. }
  118. public object GetFactory(Type lookup, Type entity = null)
  119. {
  120. if (!_cache.ContainsKey(lookup))
  121. return null;
  122. if (entity == null)
  123. return _cache[lookup].DefaultFactory;
  124. if (_cache[lookup].CustomFactories.ContainsKey(entity))
  125. return _cache[lookup].CustomFactories[entity];
  126. return null;
  127. }
  128. public object GetStaticFactory(Type lookup)
  129. {
  130. if (!_cache.ContainsKey(lookup))
  131. return null;
  132. return _cache[lookup].StaticFactory;
  133. }
  134. }
  135. private static readonly LookupCache _cache = new LookupCache();
  136. #endregion
  137. #region General Factory Methods
  138. private static object DoInvoke(Type TLookup, Type TResult, string Method)
  139. {
  140. var factory = _cache.GetFactory(TLookup);
  141. if (factory != null)
  142. {
  143. var filtertype = TResult.MakeGenericType(TLookup);
  144. var method = factory.GetType().GetMethods().FirstOrDefault(m =>
  145. m.Name.Equals(Method)
  146. && !m.GetParameters().Any()
  147. && m.ReturnType == filtertype
  148. );
  149. if (method != null)
  150. try
  151. {
  152. object[] parameters = { };
  153. return method.Invoke(factory, parameters);
  154. }
  155. catch (Exception e)
  156. {
  157. Logger.Send(LogType.Error, "", string.Format("*** Unknown Error: {0}\n{1}", e.Message, e.StackTrace));
  158. }
  159. }
  160. return null;
  161. }
  162. // If a specific Entity/Lookup match can't be found,
  163. // this will fall back to the global Lookup filter (if there is one)
  164. // Otherwise, it will return null
  165. private static object DoInvoke(Type TLookup, Type TEntity, IEnumerable items, Type TResult, string Method)
  166. {
  167. var factory = _cache.GetFactory(TLookup, TEntity);
  168. if (factory != null)
  169. {
  170. var filtertype = typeof(Filter<>).MakeGenericType(TLookup);
  171. var method = factory.GetType().GetMethods().FirstOrDefault(m =>
  172. m.Name.Equals("DefineFilter")
  173. && m.GetParameters().Any()
  174. && m.ReturnType == filtertype
  175. && m.GetParameters().First().ParameterType == items.GetType()
  176. );
  177. if (method != null)
  178. try
  179. {
  180. object[] parameters = { items };
  181. return method.Invoke(factory, parameters);
  182. }
  183. catch (Exception e)
  184. {
  185. Logger.Send(LogType.Error, "", string.Format("*** Unknown Error: {0}\n{1}", e.Message, e.StackTrace));
  186. }
  187. }
  188. return DoInvoke(TLookup, TResult, Method);
  189. }
  190. private static Dictionary<object, object> DoInvoke(Type TLookup, PropertyInfo property, string Method)
  191. {
  192. var factory = _cache.GetStaticFactory(TLookup);
  193. if (factory != null)
  194. {
  195. var method = factory.GetType().GetMethods().Where(m =>
  196. m.Name.Equals(Method) &&
  197. m.GetParameters().SingleOrDefault() != null &&
  198. m.ReturnType.Equals(typeof(Dictionary<object, object>))
  199. ).FirstOrDefault();
  200. if (method != null)
  201. try
  202. {
  203. object[] parameters = { property };
  204. return method.Invoke(factory, parameters) as Dictionary<object, object>;
  205. }
  206. catch (Exception e)
  207. {
  208. Logger.Send(LogType.Error, "", string.Format("*** Unknown Error: {0}\n{1}", e.Message, e.StackTrace));
  209. }
  210. }
  211. return null;
  212. }
  213. private static object DoInvoke(Type TLookup, Type TReturn, string Method, params object[] args)
  214. {
  215. var factory = _cache.GetFactory(TLookup);
  216. if (factory != null)
  217. {
  218. var method = factory.GetType().GetMethods().Where(m =>
  219. m.Name.Equals(Method) &&
  220. m.GetParameters().Length == args.Length &&
  221. m.ReturnType.Equals(TReturn)
  222. ).FirstOrDefault();
  223. if (method != null)
  224. try
  225. {
  226. return method.Invoke(factory, args);
  227. }
  228. catch (Exception e)
  229. {
  230. Logger.Send(LogType.Error, "", string.Format("*** Unknown Error: {0}\n{1}", e.Message, e.StackTrace));
  231. }
  232. }
  233. return null;
  234. }
  235. #endregion
  236. #region Formats
  237. public static string DefaultFormatLookup(Dictionary<string, object> values, IEnumerable<string> exclude)
  238. {
  239. return string.Join(": ",
  240. values.Where(x => x.Value != null && x.Value.GetType() != typeof(Guid) && (exclude == null || !exclude.Contains(x.Key)))
  241. .Select(p => p.Value));
  242. }
  243. public static string FormatLookup(Type TLookup, Dictionary<string, object> values, IEnumerable<string> exclude)
  244. {
  245. return DoInvoke(TLookup, typeof(string), "FormatLookup", values, exclude) as string;
  246. }
  247. public static string FormatLookup<TLookup>(Dictionary<string, object> values, IEnumerable<string> exclude)
  248. {
  249. return FormatLookup(typeof(TLookup), values, exclude);
  250. }
  251. #endregion
  252. #region Filters
  253. public static IFilter DefineFilter(Type TLookup)
  254. {
  255. return DoInvoke(TLookup, typeof(Filter<>), "DefineFilter") as IFilter;
  256. //var factory =_cache.GetFactory(TLookup);
  257. //if (factory != null)
  258. //{
  259. // Type filtertype = typeof(Filter<>).MakeGenericType(TLookup);
  260. // var method = factory.GetType().GetMethods().Where(m =>
  261. // m.Name.Equals("DefineFilter") &&
  262. // m.GetParameters().Any() &&
  263. // m.GetParameters().First().ParameterType.IsByRef &&
  264. // m.GetParameters().First().ParameterType.GetElementType().Equals(filtertype)).FirstOrDefault();
  265. // if (method != null)
  266. // {
  267. // try
  268. // {
  269. // object[] parameters = new object[] { null };
  270. // method.Invoke(factory, parameters);
  271. // return parameters[0];
  272. // }
  273. // catch (Exception e)
  274. // {
  275. // }
  276. // }
  277. //}
  278. //return null;
  279. }
  280. public static Filter<TLookup> DefineFilter<TLookup>()
  281. {
  282. return DefineFilter(typeof(TLookup)) as Filter<TLookup>;
  283. }
  284. public static IFilter DefineFilter(Type TLookup, Type TEntity, IEnumerable items)
  285. {
  286. return DoInvoke(TLookup, TEntity, items, typeof(Filter<>), "DefineFilter") as IFilter;
  287. }
  288. public static IFilter DefineFilter<TEntity>(IEnumerable<TEntity> items, Type TLookup)
  289. {
  290. return DoInvoke(TLookup, typeof(TEntity), items, typeof(Filter<>), "DefineFilter") as IFilter;
  291. }
  292. public static Filter<TLookup> DefineFilter<TEntity, TLookup>(TEntity[] items) where TEntity : Entity where TLookup : Entity
  293. {
  294. return DefineFilter(items, typeof(TLookup)) as Filter<TLookup>;
  295. }
  296. #endregion
  297. #region Columns
  298. public static IColumns DefineColumns(Type TLookup)
  299. {
  300. var result = DoInvoke(TLookup, typeof(Columns<>), "DefineColumns") as IColumns;
  301. if (result == null)
  302. {
  303. var type = typeof(Columns<>).MakeGenericType(TLookup);
  304. result = Activator.CreateInstance(type) as IColumns;
  305. result = result.DefaultColumns();
  306. }
  307. return result;
  308. }
  309. public static Columns<TLookup> DefineColumns<TLookup>()
  310. {
  311. return DefineColumns(typeof(TLookup)) as Columns<TLookup>;
  312. }
  313. #endregion
  314. #region RequiredColumns
  315. public static IColumns RequiredColumns(Type TLookup)
  316. {
  317. var result = DoInvoke(TLookup, typeof(Columns<>), "RequiredColumns") as IColumns;
  318. if (result == null)
  319. {
  320. var type = typeof(Columns<>).MakeGenericType(TLookup);
  321. result = Activator.CreateInstance(type) as IColumns;
  322. result = result.DefaultColumns();
  323. }
  324. return result;
  325. }
  326. public static Columns<TLookup> RequiredColumns<TLookup>()
  327. {
  328. return RequiredColumns(typeof(TLookup)) as Columns<TLookup>;
  329. }
  330. #endregion
  331. #region SortOrder
  332. public static ISortOrder DefineSort(Type TLookup)
  333. {
  334. return DoInvoke(TLookup, typeof(SortOrder<>), "DefineSortOrder") as ISortOrder;
  335. }
  336. public static SortOrder<TLookup> DefineSort<TLookup>()
  337. {
  338. return DefineSort(typeof(TLookup)) as SortOrder<TLookup>;
  339. }
  340. #endregion
  341. #region Static Lookups
  342. public static Dictionary<object, object> DefineLookups(Type type, PropertyInfo property)
  343. {
  344. return DoInvoke(type, property, "DefineLookups");
  345. }
  346. public static Dictionary<object, object> DefineLookups<TLookup>(Expression<Func<TLookup, object>> property)
  347. {
  348. return DefineLookups(typeof(TLookup), CoreUtils.GetPropertyFromExpression(property));
  349. }
  350. #endregion
  351. }
  352. public abstract class BaseObjectLookup<TLookup> : ILookupDefinition<TLookup> where TLookup : BaseObject, new()
  353. {
  354. public virtual string FormatLookup(Dictionary<string, object> values, IEnumerable<string> exclude)
  355. {
  356. var filtered = new Dictionary<string, object>();
  357. var cols = DefineColumns();
  358. foreach (var col in cols.Items)
  359. if (values.ContainsKey(col.Property) && values[col.Property] != null && values[col.Property].GetType() != typeof(Guid))
  360. filtered[col.Property] = values[col.Property];
  361. return LookupFactory.DefaultFormatLookup(filtered, exclude);
  362. }
  363. public abstract Columns<TLookup> DefineColumns();
  364. public abstract Columns<TLookup> RequiredColumns();
  365. public abstract Filter<TLookup> DefineFilter();
  366. public abstract SortOrder<TLookup> DefineSortOrder();
  367. }
  368. public abstract class EntityLookup<TLookup> : BaseObjectLookup<TLookup> where TLookup : Entity, new()
  369. {
  370. public override Columns<TLookup> DefineColumns()
  371. {
  372. return new Columns<TLookup>(x => x.ID);
  373. }
  374. public override Columns<TLookup> RequiredColumns()
  375. {
  376. var result = new Columns<TLookup>();
  377. var props = DatabaseSchema.Properties(typeof(TLookup)).Where(x => x.Required);
  378. //CoreUtils.PropertyList(typeof(TLookup), (p) => p.GetCustomAttribute<RequiredColumnAttribute>() != null, true);
  379. foreach (var prop in props)
  380. result.Add(prop.Name);
  381. return result;
  382. }
  383. }
  384. }