PipeIPCClient.cs 14 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402
  1. using InABox.Clients;
  2. using InABox.Core;
  3. using InABox.IPC.Shared;
  4. using System;
  5. using System.Collections;
  6. using System.Collections.Concurrent;
  7. using System.Collections.Generic;
  8. using System.Linq;
  9. using System.Runtime.CompilerServices;
  10. using System.Text;
  11. using System.Threading.Tasks;
  12. namespace InABox.Client.IPC
  13. {
  14. internal static class LocalCache
  15. {
  16. public static string Password { get; set; }
  17. }
  18. public class PipeIPCClient<TEntity> : BaseClient<TEntity> where TEntity : Entity, new()
  19. {
  20. private IPCClient Client;
  21. public PipeIPCClient(string pipeName)
  22. {
  23. Client = IPCClientFactory.GetClient(pipeName);
  24. Timeout = TimeSpan.FromSeconds(300);
  25. }
  26. private static string[]? _types;
  27. public override string[] SupportedTypes()
  28. {
  29. _types ??= CoreUtils.Entities
  30. .Where(x => x.GetInterfaces().Contains(typeof(IPersistent)))
  31. .Select(x => x.EntityName().Replace(".", "_"))
  32. .ToArray();
  33. return _types;
  34. }
  35. public override DatabaseInfo Info()
  36. {
  37. try
  38. {
  39. var request = new InfoRequest();
  40. PrepareRequest(request, false);
  41. var response = Send(PipeRequest.Info(request)).GetResponse<InfoResponse>();
  42. return response.Info;
  43. }
  44. catch (Exception)
  45. {
  46. return new DatabaseInfo();
  47. }
  48. }
  49. private void PrepareRequest(Request request, bool doCredentials = true)
  50. {
  51. if(request is not ValidateRequest && Client.Disconnected)
  52. {
  53. ClientFactory.Validate(ClientFactory.UserID, LocalCache.Password);
  54. }
  55. if (doCredentials)
  56. {
  57. request.Credentials.Platform = ClientFactory.Platform;
  58. request.Credentials.Version = ClientFactory.Version;
  59. request.Credentials.Session = ClientFactory.SessionID;
  60. }
  61. Request.BeforeRequest?.Invoke(request);
  62. }
  63. private PipeRequest Send(PipeRequest request, int? timeout = null)
  64. {
  65. return Client.Send(request, timeout ?? Convert.ToInt32(Timeout.TotalMilliseconds));
  66. }
  67. protected override bool DoCheck2FA(string code, Guid? session)
  68. {
  69. var request = new Check2FARequest { Code = code };
  70. PrepareRequest(request);
  71. var response = Send(PipeRequest.Check2FA(request)).GetResponse<Check2FAResponse>();
  72. if (response != null)
  73. {
  74. return response.Status switch
  75. {
  76. StatusCode.OK => response.Valid,
  77. StatusCode.Unauthenticated => false,
  78. _ => throw new IPCException(response.Messages),
  79. };
  80. }
  81. return false;
  82. }
  83. protected override bool DoPing()
  84. {
  85. try
  86. {
  87. var request = new PingRequest();
  88. PrepareRequest(request);
  89. var response = Send(PipeRequest.Ping(request), 10_000).GetResponse<PingResponse>();
  90. if (response != null)
  91. {
  92. return response.Status switch
  93. {
  94. StatusCode.Error or StatusCode.BadServer or StatusCode.Incomplete => throw new IPCException(response.Messages),
  95. _ => true
  96. };
  97. }
  98. }
  99. catch (Exception) { }
  100. return false;
  101. }
  102. protected override void DoDelete(TEntity entity, string auditnote)
  103. {
  104. var request = new DeleteRequest<TEntity> { Item = entity };
  105. PrepareRequest(request);
  106. var response = Send(PipeRequest.Delete(request)).GetResponse<DeleteResponse<TEntity>>();
  107. switch (response.Status)
  108. {
  109. case StatusCode.OK:
  110. break;
  111. case StatusCode.Unauthenticated:
  112. throw new IPCException("Client not authenticated");
  113. default:
  114. throw new IPCException(response.Messages);
  115. }
  116. }
  117. protected override void DoDelete(IList<TEntity> entities, string auditnote)
  118. {
  119. var items = entities.ToArray();
  120. var request = new MultiDeleteRequest<TEntity> { Items = items, AuditNote = auditnote };
  121. PrepareRequest(request);
  122. var response = Send(PipeRequest.MultiDelete(request)).GetResponse<MultiDeleteResponse<TEntity>>();
  123. switch (response.Status)
  124. {
  125. case StatusCode.OK:
  126. break;
  127. case StatusCode.Unauthenticated:
  128. throw new IPCException("Client not authenticated");
  129. default:
  130. throw new IPCException(response.Messages);
  131. }
  132. }
  133. protected override TEntity[] DoLoad(Filter<TEntity> filter = null, SortOrder<TEntity> sort = null)
  134. {
  135. var request = new QueryRequest<TEntity>
  136. {
  137. Filter = filter,
  138. Sort = sort
  139. };
  140. PrepareRequest(request);
  141. var result = new List<TEntity>();
  142. var response = Send(PipeRequest.Query(request)).GetResponse<QueryResponse<TEntity>>();
  143. if (response.Items != null)
  144. foreach (var row in response.Items.Rows)
  145. result.Add(row.ToObject<TEntity>());
  146. return result.ToArray();
  147. }
  148. protected override CoreTable DoQuery(Filter<TEntity>? filter, Columns<TEntity>? columns, SortOrder<TEntity>? sort = null)
  149. {
  150. var request = new QueryRequest<TEntity>
  151. {
  152. Columns = columns,
  153. Filter = filter,
  154. Sort = sort
  155. };
  156. PrepareRequest(request);
  157. var response = Send(PipeRequest.Query(request)).GetResponse<QueryResponse<TEntity>>();
  158. if (response != null)
  159. {
  160. return response.Status switch
  161. {
  162. StatusCode.OK => response.Items,
  163. StatusCode.Unauthenticated => throw new IPCException("Client not authenticated", StatusCode.Unauthenticated),
  164. _ => throw new IPCException(response.Messages),
  165. };
  166. }
  167. return null;
  168. }
  169. protected override Dictionary<string, CoreTable> DoQueryMultiple(Dictionary<string, IQueryDef> queries)
  170. {
  171. var request = new MultiQueryRequest
  172. {
  173. TableTypes = new(),
  174. Filters = new(),
  175. Columns = new(),
  176. Sorts = new()
  177. };
  178. foreach (var item in queries)
  179. {
  180. request.TableTypes[item.Key] = item.Value.Type.EntityName();
  181. request.Filters[item.Key] = Serialization.Serialize(item.Value.Filter);
  182. request.Columns[item.Key] = Serialization.Serialize(item.Value.Columns);
  183. request.Sorts[item.Key] = Serialization.Serialize(item.Value.SortOrder);
  184. }
  185. PrepareRequest(request);
  186. var response = Send(PipeRequest.QueryMultiple(request)).GetResponse<MultiQueryResponse>();
  187. if (response != null)
  188. {
  189. return response.Status switch
  190. {
  191. StatusCode.OK => response.Tables,
  192. StatusCode.Unauthenticated => throw new IPCException("Client not authenticated"),
  193. _ => throw new IPCException(response.Messages),
  194. };
  195. }
  196. return null;
  197. }
  198. protected override void DoSave(TEntity entity, string auditnote)
  199. {
  200. var request = new SaveRequest<TEntity>
  201. {
  202. Item = entity,
  203. AuditNote = auditnote,
  204. ReturnOnlyChanged = true
  205. };
  206. PrepareRequest(request);
  207. var response = Send(PipeRequest.Save(request)).GetResponse<SaveResponse<TEntity>>();
  208. switch (response.Status)
  209. {
  210. case StatusCode.OK:
  211. /*var props = CoreUtils.PropertyList(typeof(TEntity), x => true, true);
  212. entity.SetObserving(false);
  213. foreach (var prop in props.Keys)
  214. {
  215. var value = CoreUtils.GetPropertyValue(response.Item, prop);
  216. CoreUtils.SetPropertyValue(entity, prop, value);
  217. }
  218. entity.CommitChanges();
  219. entity.SetObserving(true);*/
  220. entity.SetObserving(false);
  221. foreach (var (key, value) in response.ChangedValues)
  222. {
  223. if (CoreUtils.TryGetProperty<TEntity>(key, out var property))
  224. {
  225. CoreUtils.SetPropertyValue(entity, key, CoreUtils.ChangeType(value, property.PropertyType));
  226. }
  227. }
  228. entity.CommitChanges();
  229. entity.SetObserving(true);
  230. break;
  231. case StatusCode.Unauthenticated:
  232. throw new IPCException("Client not authenticated");
  233. default:
  234. throw new IPCException(response.Messages);
  235. }
  236. }
  237. protected override void DoSave(IEnumerable<TEntity> entities, string auditnote)
  238. {
  239. var items = entities.ToArray();
  240. var request = new MultiSaveRequest<TEntity>
  241. {
  242. Items = items,
  243. AuditNote = auditnote,
  244. ReturnOnlyChanged = true
  245. };
  246. PrepareRequest(request);
  247. var response = Send(PipeRequest.MultiSave(request)).GetResponse<MultiSaveResponse<TEntity>>();
  248. switch (response.Status)
  249. {
  250. case StatusCode.OK:
  251. /*var props = CoreUtils.PropertyList(typeof(TEntity), x => true, true);
  252. for (var i = 0; i < items.Length; i++)
  253. {
  254. items[i].SetObserving(false);
  255. foreach (var prop in props.Keys)
  256. {
  257. var value = CoreUtils.GetPropertyValue(response.Items[i], prop);
  258. CoreUtils.SetPropertyValue(items[i], prop, value);
  259. }
  260. //CoreUtils.DeepClone<TEntity>(response.Items[i], items[i]);
  261. items[i].CommitChanges();
  262. items[i].SetObserving(true);
  263. }*/
  264. for (int i = 0; i < items.Length; ++i)
  265. {
  266. var entity = items[i];
  267. var changedValues = response.ChangedValues[i];
  268. entity.SetObserving(false);
  269. foreach (var (key, value) in changedValues)
  270. {
  271. if (CoreUtils.TryGetProperty<TEntity>(key, out var property))
  272. {
  273. CoreUtils.SetPropertyValue(entity, key, CoreUtils.ChangeType(value, property.PropertyType));
  274. }
  275. }
  276. entity.CommitChanges();
  277. entity.SetObserving(true);
  278. }
  279. break;
  280. case StatusCode.Unauthenticated:
  281. throw new IPCException("Client not authenticated");
  282. default:
  283. throw new IPCException(response.Messages);
  284. }
  285. }
  286. protected override ValidationData DoValidate(Guid session)
  287. {
  288. return Validate(
  289. null, null, false, session);
  290. }
  291. protected override ValidationData DoValidate(string pin, Guid session)
  292. {
  293. return Validate(
  294. null, pin, true, session);
  295. }
  296. protected override ValidationData DoValidate(string userid, string password, Guid session)
  297. {
  298. return Validate(
  299. userid, password, false, session);
  300. }
  301. private ValidationData Validate(string? userid, string? password, bool usePin, Guid session = default)
  302. {
  303. var ticks = DateTime.Now.ToUniversalTime().Ticks.ToString();
  304. var request = new ValidateRequest { UsePIN = usePin };
  305. if (usePin)
  306. {
  307. request.UserID = Encryption.Encrypt(ticks, "wCq9rryEJEuHIifYrxRjxg", true);
  308. request.Password = Encryption.Encrypt(ticks, "7mhvLnqMwkCAzN+zNGlyyg", true);
  309. request.PIN = password;
  310. }
  311. else
  312. {
  313. request.UserID = userid;
  314. request.Password = password;
  315. }
  316. request.Credentials.Platform = ClientFactory.Platform;
  317. request.Credentials.Version = ClientFactory.Version;
  318. PrepareRequest(request, false);
  319. if(session != Guid.Empty)
  320. {
  321. request.Credentials.Session = session;
  322. }
  323. var response = Send(PipeRequest.Validate(request), 10000).GetResponse<ValidateResponse>();
  324. if (response != null)
  325. if (response.Status.Equals(StatusCode.OK))
  326. {
  327. LocalCache.Password = password;
  328. return new ValidationData(
  329. response.ValidationResult,
  330. response.UserID,
  331. response.UserGuid,
  332. response.SecurityID,
  333. response.Session,
  334. response.Recipient2FA,
  335. response.PasswordExpiration
  336. );
  337. }
  338. else if (response.Status == StatusCode.BadServer)
  339. {
  340. throw new IPCException(response.Messages);
  341. }
  342. return new ValidationData(
  343. ValidationResult.INVALID,
  344. "",
  345. Guid.Empty,
  346. Guid.Empty,
  347. Guid.Empty,
  348. null,
  349. DateTime.MinValue
  350. );
  351. }
  352. }
  353. }