Nie możesz wybrać więcej, niż 25 tematów Tematy muszą się zaczynać od litery lub cyfry, mogą zawierać myślniki ('-') i mogą mieć do 35 znaków.

MqttServerAdapter.cs 5.3 KiB

7 lat temu
7 lat temu
7 lat temu
7 lat temu
123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129
  1. using System;
  2. using System.Net;
  3. using System.Net.Security;
  4. using System.Net.Sockets;
  5. using System.Security.Authentication;
  6. using System.Security.Cryptography.X509Certificates;
  7. using System.Threading;
  8. using System.Threading.Tasks;
  9. using MQTTnet.Core.Adapter;
  10. using MQTTnet.Core.Diagnostics;
  11. using MQTTnet.Core.Serializer;
  12. using MQTTnet.Core.Server;
  13. namespace MQTTnet.Implementations
  14. {
  15. public class MqttServerAdapter : IMqttServerAdapter, IDisposable
  16. {
  17. private CancellationTokenSource _cancellationTokenSource;
  18. private Socket _defaultEndpointSocket;
  19. private Socket _tlsEndpointSocket;
  20. private X509Certificate2 _tlsCertificate;
  21. private bool _isRunning;
  22. public event EventHandler<MqttClientConnectedEventArgs> ClientConnected;
  23. public void Start(MqttServerOptions options)
  24. {
  25. if (options == null) throw new ArgumentNullException(nameof(options));
  26. if (_isRunning) throw new InvalidOperationException("Server is already started.");
  27. _isRunning = true;
  28. _cancellationTokenSource = new CancellationTokenSource();
  29. if (options.DefaultEndpointOptions.IsEnabled)
  30. {
  31. _defaultEndpointSocket = new Socket(SocketType.Stream, ProtocolType.Tcp);
  32. _defaultEndpointSocket.Bind(new IPEndPoint(IPAddress.Any, options.GetDefaultEndpointPort()));
  33. _defaultEndpointSocket.Listen(options.ConnectionBacklog);
  34. Task.Run(() => AcceptDefaultEndpointConnectionsAsync(_cancellationTokenSource.Token), _cancellationTokenSource.Token);
  35. }
  36. if (options.TlsEndpointOptions.IsEnabled)
  37. {
  38. if (options.TlsEndpointOptions.Certificate == null)
  39. {
  40. throw new ArgumentException("TLS certificate is not set.");
  41. }
  42. _tlsCertificate = new X509Certificate2(options.TlsEndpointOptions.Certificate);
  43. _tlsEndpointSocket = new Socket(SocketType.Stream, ProtocolType.Tcp);
  44. _tlsEndpointSocket.Bind(new IPEndPoint(IPAddress.Any, options.GetTlsEndpointPort()));
  45. _tlsEndpointSocket.Listen(options.ConnectionBacklog);
  46. Task.Run(() => AcceptTlsEndpointConnectionsAsync(_cancellationTokenSource.Token), _cancellationTokenSource.Token);
  47. }
  48. }
  49. public void Stop()
  50. {
  51. _isRunning = false;
  52. _cancellationTokenSource?.Cancel(false);
  53. _cancellationTokenSource?.Dispose();
  54. _cancellationTokenSource = null;
  55. _defaultEndpointSocket?.Dispose();
  56. _defaultEndpointSocket = null;
  57. _tlsEndpointSocket?.Dispose();
  58. _tlsEndpointSocket = null;
  59. }
  60. public void Dispose()
  61. {
  62. Stop();
  63. }
  64. private async Task AcceptDefaultEndpointConnectionsAsync(CancellationToken cancellationToken)
  65. {
  66. while (!cancellationToken.IsCancellationRequested)
  67. {
  68. try
  69. {
  70. var clientSocket = await Task.Factory.FromAsync(_defaultEndpointSocket.BeginAccept, _defaultEndpointSocket.EndAccept, null).ConfigureAwait(false);
  71. var tcpChannel = new MqttTcpChannel(clientSocket, null);
  72. var clientAdapter = new MqttChannelCommunicationAdapter(tcpChannel, new MqttPacketSerializer());
  73. ClientConnected?.Invoke(this, new MqttClientConnectedEventArgs(clientSocket.RemoteEndPoint.ToString(), clientAdapter));
  74. }
  75. catch (Exception exception) when (!(exception is ObjectDisposedException))
  76. {
  77. MqttTrace.Error(nameof(MqttServerAdapter), exception, "Error while accepting connection at default endpoint.");
  78. //excessive CPU consumed if in endless loop of socket errors
  79. await Task.Delay(TimeSpan.FromSeconds(1), cancellationToken).ConfigureAwait(false);
  80. }
  81. }
  82. }
  83. private async Task AcceptTlsEndpointConnectionsAsync(CancellationToken cancellationToken)
  84. {
  85. while (!cancellationToken.IsCancellationRequested)
  86. {
  87. try
  88. {
  89. var clientSocket = await Task.Factory.FromAsync(_tlsEndpointSocket.BeginAccept, _tlsEndpointSocket.EndAccept, null).ConfigureAwait(false);
  90. var sslStream = new SslStream(new NetworkStream(clientSocket));
  91. await sslStream.AuthenticateAsServerAsync(_tlsCertificate, false, SslProtocols.Tls12, false).ConfigureAwait(false);
  92. var tcpChannel = new MqttTcpChannel(clientSocket, sslStream);
  93. var clientAdapter = new MqttChannelCommunicationAdapter(tcpChannel, new MqttPacketSerializer());
  94. ClientConnected?.Invoke(this, new MqttClientConnectedEventArgs(clientSocket.RemoteEndPoint.ToString(), clientAdapter));
  95. }
  96. catch (Exception exception)
  97. {
  98. MqttTrace.Error(nameof(MqttServerAdapter), exception, "Error while accepting connection at TLS endpoint.");
  99. //excessive CPU consumed if in endless loop of socket errors
  100. await Task.Delay(TimeSpan.FromSeconds(1), cancellationToken).ConfigureAwait(false);
  101. }
  102. }
  103. }
  104. }
  105. }