You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
 
 
 
 

125 line
5.0 KiB

  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 (_isRunning) throw new InvalidOperationException("Server is already started.");
  26. _isRunning = true;
  27. _cancellationTokenSource = new CancellationTokenSource();
  28. if (options.DefaultEndpointOptions.IsEnabled)
  29. {
  30. _defaultEndpointSocket = new Socket(SocketType.Stream, ProtocolType.Tcp);
  31. _defaultEndpointSocket.Bind(new IPEndPoint(IPAddress.Any, options.GetDefaultEndpointPort()));
  32. _defaultEndpointSocket.Listen(options.ConnectionBacklog);
  33. Task.Run(() => AcceptDefaultEndpointConnectionsAsync(_cancellationTokenSource.Token), _cancellationTokenSource.Token);
  34. }
  35. if (options.TlsEndpointOptions.IsEnabled)
  36. {
  37. if (options.TlsEndpointOptions.Certificate == null)
  38. {
  39. throw new ArgumentException("TLS certificate is not set.");
  40. }
  41. _tlsCertificate = new X509Certificate2(options.TlsEndpointOptions.Certificate);
  42. _tlsEndpointSocket = new Socket(SocketType.Stream, ProtocolType.Tcp);
  43. _tlsEndpointSocket.Bind(new IPEndPoint(IPAddress.Any, options.GetTlsEndpointPort()));
  44. _tlsEndpointSocket.Listen(options.ConnectionBacklog);
  45. Task.Run(() => AcceptTlsEndpointConnectionsAsync(_cancellationTokenSource.Token), _cancellationTokenSource.Token);
  46. }
  47. }
  48. public void Stop()
  49. {
  50. _isRunning = false;
  51. _cancellationTokenSource?.Cancel(false);
  52. _cancellationTokenSource?.Dispose();
  53. _cancellationTokenSource = null;
  54. _defaultEndpointSocket?.Dispose();
  55. _defaultEndpointSocket = null;
  56. _tlsEndpointSocket?.Dispose();
  57. _tlsEndpointSocket = null;
  58. }
  59. public void Dispose()
  60. {
  61. Stop();
  62. }
  63. private async Task AcceptDefaultEndpointConnectionsAsync(CancellationToken cancellationToken)
  64. {
  65. while (!cancellationToken.IsCancellationRequested)
  66. {
  67. try
  68. {
  69. var clientSocket = await _defaultEndpointSocket.AcceptAsync().ConfigureAwait(false);
  70. var clientAdapter = new MqttChannelCommunicationAdapter(new MqttTcpChannel(clientSocket, null), new MqttPacketSerializer());
  71. ClientConnected?.Invoke(this, new MqttClientConnectedEventArgs(clientSocket.RemoteEndPoint.ToString(), clientAdapter));
  72. }
  73. catch (Exception exception)
  74. {
  75. MqttTrace.Error(nameof(MqttServerAdapter), exception, "Error while accepting connection at default endpoint.");
  76. //excessive CPU consumed if in endless loop of socket errors
  77. await Task.Delay(TimeSpan.FromSeconds(1), cancellationToken).ConfigureAwait(false);
  78. }
  79. }
  80. }
  81. private async Task AcceptTlsEndpointConnectionsAsync(CancellationToken cancellationToken)
  82. {
  83. while (!cancellationToken.IsCancellationRequested)
  84. {
  85. try
  86. {
  87. var clientSocket = await _tlsEndpointSocket.AcceptAsync().ConfigureAwait(false);
  88. var sslStream = new SslStream(new NetworkStream(clientSocket));
  89. await sslStream.AuthenticateAsServerAsync(_tlsCertificate, false, SslProtocols.Tls12, false).ConfigureAwait(false);
  90. var clientAdapter = new MqttChannelCommunicationAdapter(new MqttTcpChannel(clientSocket, sslStream), new MqttPacketSerializer());
  91. ClientConnected?.Invoke(this, new MqttClientConnectedEventArgs(clientSocket.RemoteEndPoint.ToString(), clientAdapter));
  92. }
  93. catch (Exception exception)
  94. {
  95. MqttTrace.Error(nameof(MqttServerAdapter), exception, "Error while accepting connection at TLS endpoint.");
  96. //excessive CPU consumed if in endless loop of socket errors
  97. await Task.Delay(TimeSpan.FromSeconds(1), cancellationToken).ConfigureAwait(false);
  98. }
  99. }
  100. }
  101. }
  102. }