RPCServer.cs 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284
  1. using System.Reflection;
  2. using H.Pipes.Args;
  3. using InABox.API;
  4. using InABox.Clients;
  5. using InABox.Core;
  6. namespace InABox.IPC
  7. {
  8. public class RPCServer<TTransport> : IDisposable, IRPCServer where TTransport : IRPCServerTransport
  9. {
  10. private TTransport _transport;
  11. public event LogFunction? OnLog;
  12. IPCNotifyState NotifyState = new();
  13. public RPCServer(Func<TTransport> transport)
  14. {
  15. _transport = transport();
  16. _transport.OnOpen += Transport_OnOpen;
  17. _transport.OnClose += Transport_OnClose;
  18. _transport.OnException += Transport_OnException;
  19. _transport.BeforeMessage += Transport_BeforeMessage;
  20. _transport.AfterMessage += Transport_AfterMessage;
  21. _transport.AddHandler<IRPCServer, RPCInfoCommand, RPCInfoParameters, RPCInfoResponse>(new RPCInfoHandler(this));
  22. _transport.AddHandler<IRPCServer, RPCValidateCommand, RPCValidateParameters, RPCValidateResponse>(new RPCValidateHandler(this));
  23. _transport.AddHandler<IRPCServer, RPCCheck2FACommand, RPCCheck2FAParameters, RPCCheck2FAResponse>(new RPCCheck2FAHandler(this));
  24. _transport.AddHandler<IRPCServer, RPCQueryCommand, RPCQueryParameters, RPCQueryResponse>(new RPCQueryHandler(this));
  25. _transport.AddHandler<IRPCServer, RPCSaveCommand, RPCSaveParameters, RPCSaveResponse>(new RPCSaveHandler(this));
  26. _transport.AddHandler<IRPCServer, RPCDeleteCommand, RPCDeleteParameters, RPCDeleteResponse>(new RPCDeleteHandler(this));
  27. }
  28. private void Transport_OnOpen(IRPCTransport transport, RPCTransportOpenArgs e)
  29. {
  30. OnLog?.Invoke(LogType.Information, $"{e.Session?.ID}", "Client Connected");
  31. }
  32. private void Transport_OnClose(IRPCTransport transport, RPCTransportCloseArgs e)
  33. {
  34. OnLog?.Invoke(LogType.Information, $"{e.Session?.ID}", $"Client Disconnected({e.Type})");
  35. }
  36. private void Transport_BeforeMessage(IRPCTransport transport, RPCTransportMessageArgs e)
  37. {
  38. OnLog?.Invoke(LogType.Information, "", $"Request Received [{e.Session?.ID}]: {e.Message?.Command ?? "Unknown"}");
  39. }
  40. private void Transport_AfterMessage(IRPCTransport transport, RPCTransportMessageArgs e)
  41. {
  42. OnLog?.Invoke(LogType.Information, "", $"Sending Response [{e.Session?.ID}]: {e.Message?.Command ?? "Unknown"}");
  43. }
  44. private void Transport_OnException(IRPCTransport transport, RPCTransportExceptionArgs e)
  45. {
  46. OnLog?.Invoke(LogType.Error, $"", $"Exception Occurredn in {e.Session?.ID}: {e.Exception.Message}");
  47. }
  48. public void Start()
  49. {
  50. _transport.Start();
  51. }
  52. public void Stop()
  53. {
  54. _transport.Stop();
  55. }
  56. private static List<Type>? _persistentRemotable;
  57. private static Type? GetEntity(string entityName)
  58. {
  59. _persistentRemotable ??= CoreUtils.TypeList(
  60. e => e.IsSubclassOf(typeof(Entity)) &&
  61. e.GetInterfaces().Contains(typeof(IRemotable)) &&
  62. e.GetInterfaces().Contains(typeof(IPersistent))).ToList();
  63. return _persistentRemotable.FirstOrDefault(x => x.Name == entityName);
  64. }
  65. private static Type? GetResponseType(Method method, string? entityName)
  66. {
  67. if (entityName != null)
  68. {
  69. var entityType = GetEntity(entityName);
  70. if (entityType != null)
  71. {
  72. var response = method switch
  73. {
  74. Method.Query => typeof(QueryResponse<>).MakeGenericType(entityType),
  75. Method.Delete => typeof(DeleteResponse<>).MakeGenericType(entityType),
  76. Method.MultiDelete => typeof(MultiDeleteResponse<>).MakeGenericType(entityType),
  77. Method.Save => typeof(SaveResponse<>).MakeGenericType(entityType),
  78. Method.MultiSave => typeof(MultiSaveResponse<>).MakeGenericType(entityType),
  79. _ => null
  80. };
  81. if (response != null) return response;
  82. }
  83. }
  84. return method switch
  85. {
  86. Method.QueryMultiple => typeof(MultiQueryResponse),
  87. Method.Validate => typeof(ValidateResponse),
  88. Method.Check2FA => typeof(Check2FAResponse),
  89. _ => null
  90. };
  91. }
  92. private class RequestData
  93. {
  94. public ConnectionMessageEventArgs<IPCMessage?> e { get; }
  95. public RequestData(ConnectionMessageEventArgs<IPCMessage?> e)
  96. {
  97. this.e = e;
  98. }
  99. }
  100. private IPCMessage QueryMultiple(IPCMessage request, RequestData data)
  101. {
  102. var response = RestService.QueryMultiple(request.GetRequest<MultiQueryRequest>(), true);
  103. return request.Respond(response);
  104. }
  105. private IPCMessage Validate(IPCMessage request, RequestData data)
  106. {
  107. var response = RestService.Validate(request.GetRequest<ValidateRequest>());
  108. return request.Respond(response);
  109. }
  110. private IPCMessage Ping(IPCMessage request, RequestData data) => request.Respond(new PingResponse().Status(StatusCode.OK));
  111. private IPCMessage Info(IPCMessage request, RequestData data)
  112. {
  113. var response = RestService.Info(request.GetRequest<InfoRequest>());
  114. return request.Respond(response);
  115. }
  116. private IPCMessage Check2FA(IPCMessage request, RequestData data)
  117. {
  118. var response = RestService.Check2FA(request.GetRequest<Check2FARequest>());
  119. return request.Respond(response);
  120. }
  121. private IPCMessage Query<T>(IPCMessage request, RequestData data) where T : Entity, new()
  122. {
  123. var response = RestService<T>.List(request.GetRequest<QueryRequest<T>>());
  124. return request.Respond(response);
  125. }
  126. private IPCMessage Save<T>(IPCMessage request, RequestData data) where T : Entity, new()
  127. {
  128. var response = RestService<T>.Save(request.GetRequest<SaveRequest<T>>());
  129. return request.Respond(response);
  130. }
  131. private IPCMessage MultiSave<T>(IPCMessage request, RequestData data) where T : Entity, new()
  132. {
  133. var response = RestService<T>.MultiSave(request.GetRequest<MultiSaveRequest<T>>());
  134. return request.Respond(response);
  135. }
  136. private IPCMessage Delete<T>(IPCMessage request, RequestData data) where T : Entity, new()
  137. {
  138. var response = RestService<T>.Delete(request.GetRequest<DeleteRequest<T>>());
  139. return request.Respond(response);
  140. }
  141. private IPCMessage MultiDelete<T>(IPCMessage request, RequestData data) where T : Entity, new()
  142. {
  143. var response = RestService<T>.MultiDelete(request.GetRequest<MultiDeleteRequest<T>>());
  144. return request.Respond(response);
  145. }
  146. private static readonly MethodInfo QueryMethod = GetMethod(nameof(Query));
  147. private static readonly MethodInfo SaveMethod = GetMethod(nameof(Save));
  148. private static readonly MethodInfo MultiSaveMethod = GetMethod(nameof(MultiSave));
  149. private static readonly MethodInfo DeleteMethod = GetMethod(nameof(Delete));
  150. private static readonly MethodInfo MultiDeleteMethod = GetMethod(nameof(MultiDelete));
  151. private static readonly MethodInfo QueryMultipleMethod = GetMethod(nameof(QueryMultiple));
  152. private static readonly MethodInfo ValidateMethod = GetMethod(nameof(Validate));
  153. private static readonly MethodInfo Check2FAMethod = GetMethod(nameof(Check2FA));
  154. private static readonly MethodInfo PingMethod = GetMethod(nameof(Ping));
  155. private static readonly MethodInfo InfoMethod = GetMethod(nameof(Info));
  156. private static MethodInfo GetMethod(string name) =>
  157. typeof(IPCServer).GetMethod(name, BindingFlags.NonPublic | BindingFlags.Instance)
  158. ?? throw new Exception($"Invalid method '{name}'");
  159. private void Server_MessageReceived(object? sender, H.Pipes.Args.ConnectionMessageEventArgs<IPCMessage?> e)
  160. {
  161. Task.Run(() =>
  162. {
  163. var start = DateTime.Now;
  164. try
  165. {
  166. if (e.Message == null) throw new Exception($"Invalid message");
  167. var method = e.Message.Method switch
  168. {
  169. Method.Query => QueryMethod,
  170. Method.QueryMultiple => QueryMultipleMethod,
  171. Method.Delete => DeleteMethod,
  172. Method.MultiDelete => MultiDeleteMethod,
  173. Method.Save => SaveMethod,
  174. Method.MultiSave => MultiSaveMethod,
  175. Method.Check2FA => Check2FAMethod,
  176. Method.Validate => ValidateMethod,
  177. Method.Ping => PingMethod,
  178. Method.Info => InfoMethod,
  179. Method.None or _ => throw new Exception($"Invalid method '{e.Message.Method}'")
  180. };
  181. if (e.Message.Type != null)
  182. {
  183. var entityType = GetEntity(e.Message.Type) ?? throw new Exception($"No entity '{e.Message.Type}'");
  184. method = method.MakeGenericMethod(entityType);
  185. }
  186. var response = method.Invoke(this, new object[] { e.Message, new RequestData(e) }) as IPCMessage;
  187. e.Connection.WriteAsync(response).ContinueWith(task =>
  188. {
  189. if (task.Exception != null)
  190. {
  191. Logger.Send(LogType.Error, "", $"Error in response: {CoreUtils.FormatException(task.Exception)}");
  192. }
  193. });
  194. }
  195. catch (Exception err)
  196. {
  197. Logger.Send(LogType.Error, "", err.Message);
  198. if (e.Message != null)
  199. {
  200. var responseType = GetResponseType(e.Message.Method, e.Message.Type);
  201. if (responseType != null)
  202. {
  203. var response = (Activator.CreateInstance(responseType) as Response)!;
  204. response.Status = StatusCode.Error;
  205. response.Messages.Add(err.Message);
  206. e.Connection.WriteAsync(e.Message.Respond(response)).ContinueWith(task =>
  207. {
  208. if (task.Exception != null)
  209. {
  210. Logger.Send(LogType.Error, "", $"Error in response: {CoreUtils.FormatException(task.Exception)}");
  211. }
  212. });
  213. }
  214. }
  215. }
  216. });
  217. }
  218. private void Server_ClientDisconnected(object? sender, H.Pipes.Args.ConnectionEventArgs<IPCMessage> e)
  219. {
  220. Logger.Send(LogType.Information, "", "Client Disconnected");
  221. var sessionID = NotifyState.SessionMap.Where(x => x.Value.Connection == e.Connection).FirstOrDefault().Key;
  222. NotifyState.SessionMap.TryRemove(sessionID, out var session);
  223. e.Connection.DisposeAsync();
  224. }
  225. private void Server_ClientConnected(object? sender, H.Pipes.Args.ConnectionEventArgs<IPCMessage> e)
  226. {
  227. Logger.Send(LogType.Information, "", "Client Connected");
  228. }
  229. public void Dispose()
  230. {
  231. _transport?.Stop();
  232. }
  233. ~RPCServer()
  234. {
  235. Dispose();
  236. }
  237. }
  238. }