ILookupDefinition.cs 19 KB

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