IPCClientTransport.cs 4.4 KB

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