ClientFactory.cs 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385
  1. using System;
  2. using System.Collections.Generic;
  3. using System.Linq;
  4. using System.Threading.Tasks;
  5. using InABox.Configuration;
  6. using InABox.Core;
  7. using InABox.Mail;
  8. namespace InABox.Clients
  9. {
  10. public class BypassSession : IDisposable
  11. {
  12. public BypassSession()
  13. {
  14. ClientFactory.SetBypass();
  15. }
  16. public void Dispose()
  17. {
  18. ClientFactory.UnsetBypass();
  19. }
  20. }
  21. public static class ClientFactory
  22. {
  23. public static Dictionary<EmailType, Type> MailerTypes = new Dictionary<EmailType, Type>();
  24. public static Guid UserGuid { get; private set; }
  25. public static string UserID { get; private set; }
  26. public static Guid UserSecurityID { get; private set; }
  27. public static Guid SessionID { get; private set; }
  28. public static Guid DatabaseID { get; set; }
  29. public static Platform Platform { get; private set; }
  30. public static string? Version { get; private set; }
  31. public static Type? ClientType { get; private set; }
  32. public static object[]? Parameters { get; private set; }
  33. public static IEnumerable<string>? SupportedTypes { get; private set; }
  34. public static string? Recipient2FA { get; private set; }
  35. public static DateTime PasswordExpiration { get; private set; }
  36. public static int PINLength { get; } = 4;
  37. public static PushHandlers PushHandlers { get; private set; } = new PushHandlers();
  38. public delegate void RequestErrorHandler(RequestException e);
  39. public static event RequestErrorHandler? OnRequestError;
  40. public static void RaiseRequestError(RequestException e)
  41. {
  42. OnRequestError?.Invoke(e);
  43. }
  44. public static void SetClientType(Type type, Platform platform, string? version, params object[]? parameters)
  45. {
  46. ClientType = type;
  47. Platform = platform;
  48. Version = String.IsNullOrEmpty(version) ? Version : version;
  49. Parameters = parameters == null ? Parameters : parameters;
  50. SupportedTypes = parameters == null ? SupportedTypes : null;
  51. ConfigurationCache.ClearAll(ConfigurationCacheType.User);
  52. ConfigurationCache.ClearAll(ConfigurationCacheType.Global);
  53. }
  54. public static void ClearClientType()
  55. {
  56. ClientType = null;
  57. Platform = default(Platform);
  58. Version = "";
  59. Parameters = null;
  60. ConfigurationCache.ClearAll(ConfigurationCacheType.User);
  61. ConfigurationCache.ClearAll(ConfigurationCacheType.Global);
  62. }
  63. // override the need to provide credentials when configuring the database
  64. public static void SetBypass()
  65. {
  66. UserGuid = CoreUtils.FullGuid;
  67. UserID = "";
  68. Request.BeforeRequest = request =>
  69. {
  70. if(request is ValidateRequest validate)
  71. {
  72. var ticks = DateTime.Now.ToUniversalTime().Ticks.ToString();
  73. validate.UserID = Encryption.Encrypt(ticks, "wCq9rryEJEuHIifYrxRjxg", true);
  74. validate.Password = Encryption.Encrypt(ticks, "7mhvLnqMwkCAzN+zNGlyyg", true);
  75. }
  76. };
  77. var ticks = DateTime.Now.ToUniversalTime().Ticks.ToString();
  78. var result = Validate(
  79. Encryption.Encrypt(ticks, "wCq9rryEJEuHIifYrxRjxg", true),
  80. Encryption.Encrypt(ticks, "7mhvLnqMwkCAzN+zNGlyyg", true));
  81. if (result != ValidationStatus.VALID)
  82. {
  83. Logger.Send(LogType.Error, "", "Bypass login failed");
  84. }
  85. //Load up the data model, including Custom Properties
  86. //var props = CreateClient<CustomProperty>().Load();
  87. //DatabaseSchema.Load(props);
  88. }
  89. public static void UnsetBypass()
  90. {
  91. Request.BeforeRequest = null;
  92. UserGuid = Guid.Empty;
  93. }
  94. public static ValidationStatus Validate(string userid, string password, Guid session = default)
  95. {
  96. InvalidateUser();
  97. var result = Client.Validate(userid, password, session);
  98. if (result.Status != ValidationStatus.INVALID)
  99. {
  100. UserGuid = result.UserGuid;
  101. UserID = result.UserID;
  102. UserSecurityID = result.SecurityID;
  103. Recipient2FA = result.Recipient2FA;
  104. SessionID = result.SessionID;
  105. PasswordExpiration = result.PasswordExpiration;
  106. if(result.Status == ValidationStatus.VALID)
  107. {
  108. OnValidLogin();
  109. }
  110. return result.Status;
  111. }
  112. return result.Status;
  113. }
  114. public static ValidationStatus Validate(string pin, Guid session = default)
  115. {
  116. InvalidateUser();
  117. var result = Client.Validate(pin, session);
  118. if (result.Status != ValidationStatus.INVALID)
  119. {
  120. UserGuid = result.UserGuid;
  121. UserID = result.UserID;
  122. UserSecurityID = result.SecurityID;
  123. Recipient2FA = result.Recipient2FA;
  124. SessionID = result.SessionID;
  125. PasswordExpiration = result.PasswordExpiration;
  126. if (result.Status == ValidationStatus.VALID)
  127. {
  128. OnValidLogin();
  129. }
  130. return result.Status;
  131. }
  132. return result.Status;
  133. }
  134. public static ValidationStatus Validate(Guid session)
  135. {
  136. var result = Client.Validate(session);
  137. if (result.Status != ValidationStatus.INVALID)
  138. {
  139. UserGuid = result.UserGuid;
  140. UserID = result.UserID;
  141. UserSecurityID = result.SecurityID;
  142. Recipient2FA = result.Recipient2FA;
  143. SessionID = result.SessionID;
  144. PasswordExpiration = result.PasswordExpiration;
  145. if (result.Status == ValidationStatus.VALID)
  146. {
  147. OnValidLogin();
  148. }
  149. return result.Status;
  150. }
  151. return result.Status;
  152. }
  153. public static bool ValidateUser(string userid, string password)
  154. {
  155. InvalidateUser();
  156. var result = Client.Validate(userid, password);
  157. if (result.Status == ValidationStatus.VALID)
  158. {
  159. UserGuid = result.UserGuid;
  160. UserID = result.UserID;
  161. UserSecurityID = result.SecurityID;
  162. Recipient2FA = result.Recipient2FA;
  163. SessionID = result.SessionID;
  164. PasswordExpiration = result.PasswordExpiration;
  165. if (result.Status == ValidationStatus.VALID)
  166. {
  167. OnValidLogin();
  168. }
  169. return true;
  170. }
  171. return false;
  172. }
  173. public static bool ValidatePIN(string pin)
  174. {
  175. InvalidateUser();
  176. var result = Client.Validate(pin);
  177. if (result.Status == ValidationStatus.VALID)
  178. {
  179. UserGuid = result.UserGuid;
  180. UserID = result.UserID;
  181. UserSecurityID = result.SecurityID;
  182. Recipient2FA = result.Recipient2FA;
  183. SessionID = result.SessionID;
  184. PasswordExpiration = result.PasswordExpiration;
  185. if (result.Status == ValidationStatus.VALID)
  186. {
  187. OnValidLogin();
  188. }
  189. return true;
  190. }
  191. return false;
  192. }
  193. public static bool Check2FA(string code, Guid? session = null)
  194. {
  195. var result = Client.Check2FA(code, session);
  196. if (result)
  197. {
  198. OnValidLogin();
  199. }
  200. return result;
  201. }
  202. public static void OnValidLogin()
  203. {
  204. //Load up the data model, including Custom Properties
  205. var props = new Client<CustomProperty>().Load();
  206. DatabaseSchema.Load(props);
  207. }
  208. public static void InvalidateUser()
  209. {
  210. //CurrentUser = null;
  211. UserGuid = Guid.Empty;
  212. UserID = "";
  213. UserSecurityID = Guid.Empty;
  214. SessionID = Guid.Empty;
  215. InABox.Core.Security.Reset();
  216. }
  217. public static event LogEvent? OnLog;
  218. public static IClient<TEntity> CreateClient<TEntity>() where TEntity : Entity, new()
  219. {
  220. var type = ClientType.MakeGenericType(typeof(TEntity));
  221. var client = (IClient<TEntity>)Activator.CreateInstance(type, Parameters);
  222. client.OnLog += OnLog;
  223. return client;
  224. }
  225. public static IClient CreateClient(Type t)
  226. {
  227. var type = ClientType.MakeGenericType(t);
  228. var client = (IClient)Activator.CreateInstance(type, Parameters);
  229. client.OnLog += OnLog;
  230. return client;
  231. }
  232. public static CoreTable[] MultiQuery(params IQueryDef[] queries)
  233. {
  234. var tasks = new Dictionary<object, Task<CoreTable>>();
  235. for (var i = 0; i < queries.Length; i++)
  236. {
  237. var def = queries[i];
  238. var task = Task.Run(() => { return CreateClient(def.Type).Query(def.Filter, def.Columns, def.SortOrder); });
  239. tasks[i] = task;
  240. }
  241. Task.WaitAll(tasks.Values.ToArray());
  242. var results = new List<CoreTable>();
  243. for (var i = 0; i < queries.Length; i++)
  244. results.Add(tasks[i].Result);
  245. return results.ToArray();
  246. }
  247. public static void MultiQuery(IQueryDef[] queries, Action<CoreTable[]> OnResults)
  248. {
  249. Task.Run(() =>
  250. {
  251. var tasks = new Dictionary<object, Task<CoreTable>>();
  252. for (var i = 0; i < queries.Length; i++)
  253. {
  254. var def = queries[i];
  255. var task = Task.Run(() => { return CreateClient(def.Type).Query(def.Filter, def.Columns, def.SortOrder); });
  256. tasks[i] = task;
  257. }
  258. Task.WaitAll(tasks.Values.ToArray());
  259. var results = new List<CoreTable>();
  260. for (var i = 0; i < queries.Length; i++)
  261. results.Add(tasks[i].Result);
  262. OnResults.Invoke(results.ToArray());
  263. });
  264. }
  265. public static void RegisterMailer(EmailType type, Type mailer)
  266. {
  267. MailerTypes[type] = mailer;
  268. }
  269. public static ICoreMailer? CreateMailer()
  270. {
  271. var row = new Client<User>().Query(
  272. Filter<User>.Where(x => x.ID).IsEqualTo(UserGuid),
  273. Columns.None<User>().Add(
  274. x => x.EmailType,
  275. x => x.EmailHost,
  276. x => x.EmailPort,
  277. x => x.EmailDomain,
  278. x => x.EmailUserID,
  279. x => x.EmailPassword,
  280. x => x.SMTPHost,
  281. x => x.SMPTPort,
  282. x => x.SMTPUserName,
  283. x => x.SMTPPassword)
  284. ).Rows.FirstOrDefault();
  285. if (row == null)
  286. return null;
  287. var type = row.Get<User, EmailType>(x => x.EmailType);
  288. if (type == EmailType.None)
  289. return null;
  290. var result = (Activator.CreateInstance(MailerTypes[type]) as ICoreMailer)!;
  291. result.MailboxHost = row.Get<User, string>(x => x.EmailHost);
  292. result.MailboxPort = row.Get<User, int>(x => x.EmailPort);
  293. result.MailboxDomain = row.Get<User, string>(x => x.EmailDomain);
  294. result.MailboxUserName = row.Get<User, string>(x => x.EmailUserID);
  295. result.MailboxPassword = row.Get<User, string>(x => x.EmailPassword);
  296. result.SMTPHost = row.Get<User, string>(x => x.SMTPHost);
  297. result.SMTPPort = row.Get<User, int>(x => x.SMPTPort);
  298. result.SMTPUserName = row.Get<User, string>(x => x.SMTPUserName);
  299. result.SMTPPassword = row.Get<User, string>(x => x.SMTPPassword);
  300. return result;
  301. }
  302. }
  303. }