IPCClientTransport.cs 4.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142
  1. using H.Pipes;
  2. using InABox.Core;
  3. using InABox.IPC;
  4. using InABox.IPC.Shared;
  5. using System.Collections.Concurrent;
  6. namespace InABox.Client.IPC
  7. {
  8. public class IPCClientTransport : IDisposable
  9. {
  10. private PipeClient<IPCMessage> Client;
  11. private ConcurrentDictionary<Guid, ManualResetEventSlim> Events = new();
  12. private ConcurrentDictionary<Guid, IPCMessage> Responses = new();
  13. private const int DefaultRequestTimeout = 5 * 60 * 1000; // 5 minutes
  14. public delegate void ConnectEvent();
  15. public delegate void DisconnectEvent();
  16. /// <summary>
  17. /// A handler for any requests pushed from the server, i.e., not initialised by the client.
  18. /// </summary>
  19. public delegate void PushEvent(PipeRequest request);
  20. public bool Disconnected { get; private set; }
  21. public event ConnectEvent? OnConnect;
  22. public event DisconnectEvent? OnDisconnect;
  23. public event PushEvent? OnPush;
  24. public IPCClientTransport(string pipeName)
  25. {
  26. Client = new PipeClient<IPCMessage>(pipeName);
  27. Client.Connected += Client_Connected;
  28. Client.Disconnected += Client_Disconnected;
  29. Client.MessageReceived += Client_MessageReceived;
  30. Client.ExceptionOccurred += Client_ExceptionOccurred;
  31. Client.ConnectAsync();
  32. }
  33. private void Client_ExceptionOccurred(object? sender, H.Pipes.Args.ExceptionEventArgs e)
  34. {
  35. Logger.Send(LogType.Error, "", $"Exception occured: {e.Exception.Message}");
  36. }
  37. public IPCMessage Send(IPCMessage request, int timeout = DefaultRequestTimeout)
  38. {
  39. var start = DateTime.Now;
  40. var ev = Queue(request.RequestID);
  41. Client.WriteAsync(request);
  42. var result = GetResult(request.RequestID, ev, timeout);
  43. return result;
  44. }
  45. public ManualResetEventSlim Queue(Guid id)
  46. {
  47. var ev = new ManualResetEventSlim();
  48. Events[id] = ev;
  49. return ev;
  50. }
  51. public IPCMessage GetResult(Guid id, ManualResetEventSlim ev, int timeout)
  52. {
  53. if (Responses.TryGetValue(id, out var result))
  54. {
  55. Responses.Remove(id, out result);
  56. Events.Remove(id, out ev);
  57. return result;
  58. }
  59. try
  60. {
  61. if (!ev.Wait(timeout))
  62. {
  63. return IPCMessage.Error(RequestError.TIMEOUT);
  64. }
  65. }
  66. catch (Exception e)
  67. {
  68. Console.WriteLine(e);
  69. throw;
  70. }
  71. Responses.Remove(id, out result);
  72. Events.Remove(id, out ev);
  73. return result ?? IPCMessage.Error(RequestError.UNKNOWN);
  74. }
  75. private void Client_MessageReceived(object? sender, H.Pipes.Args.ConnectionMessageEventArgs<IPCMessage?> e)
  76. {
  77. if (Events.TryGetValue(e.Message.RequestID, out var ev))
  78. {
  79. Responses[e.Message.RequestID] = e.Message;
  80. ev.Set();
  81. }
  82. else
  83. {
  84. Task.Run(() =>
  85. {
  86. OnPush?.Invoke(e.Message);
  87. }).ContinueWith(task =>
  88. {
  89. if (task.Exception != null)
  90. {
  91. Logger.Send(LogType.Error, "", $"Error in IPC Client Push: {CoreUtils.FormatException(task.Exception)}");
  92. }
  93. });
  94. }
  95. }
  96. private void Client_Connected(object? sender, H.Pipes.Args.ConnectionEventArgs<IPCMessage> e)
  97. {
  98. Logger.Send(LogType.Information, "", $"Connected to Pipe: {e.Connection.PipeName}");
  99. Disconnected = false;
  100. OnConnect?.Invoke();
  101. }
  102. private void Client_Disconnected(object? sender, H.Pipes.Args.ConnectionEventArgs<IPCMessage> e)
  103. {
  104. Logger.Send(LogType.Information, "", $"Disconnected from Pipe: {e.Connection.PipeName}");
  105. foreach (var ev in Events)
  106. {
  107. Responses.TryAdd(ev.Key, IPCMessage.Error(RequestError.DISCONNECTED));
  108. ev.Value.Set();
  109. }
  110. Disconnected = true;
  111. OnDisconnect?.Invoke();
  112. }
  113. public void Dispose()
  114. {
  115. Client.DisposeAsync().AsTask().Wait();
  116. }
  117. ~IPCClientTransport()
  118. {
  119. Dispose();
  120. }
  121. }
  122. }