|
123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620 |
- using MQTTnet.Client;
- using MQTTnet.Client.Connecting;
- using MQTTnet.Client.Disconnecting;
- using MQTTnet.Client.Publishing;
- using MQTTnet.Client.Receiving;
- using MQTTnet.Diagnostics;
- using MQTTnet.Exceptions;
- using MQTTnet.Internal;
- using MQTTnet.Protocol;
- using MQTTnet.Server;
- using System;
- using System.Collections.Generic;
- using System.Linq;
- using System.Threading;
- using System.Threading.Tasks;
-
- namespace MQTTnet.Extensions.ManagedClient
- {
- public class ManagedMqttClient : Disposable, IManagedMqttClient
- {
- readonly BlockingQueue<ManagedMqttApplicationMessage> _messageQueue = new BlockingQueue<ManagedMqttApplicationMessage>();
-
- /// <summary>
- /// The subscriptions are managed in 2 separate buckets:
- /// <see cref="_subscriptions"/> and <see cref="_unsubscriptions"/> are processed during normal operation
- /// and are moved to the <see cref="_reconnectSubscriptions"/> when they get processed. They can be accessed by
- /// any thread and are therefore mutex'ed. <see cref="_reconnectSubscriptions"/> get sent to the broker
- /// at reconnect and are solely owned by <see cref="MaintainConnectionAsync"/>.
- /// </summary>
- readonly Dictionary<string, MqttQualityOfServiceLevel> _reconnectSubscriptions = new Dictionary<string, MqttQualityOfServiceLevel>();
- readonly Dictionary<string, MqttQualityOfServiceLevel> _subscriptions = new Dictionary<string, MqttQualityOfServiceLevel>();
- readonly HashSet<string> _unsubscriptions = new HashSet<string>();
- readonly SemaphoreSlim _subscriptionsQueuedSignal = new SemaphoreSlim(0);
-
- readonly IMqttNetScopedLogger _logger;
-
- readonly AsyncLock _messageQueueLock = new AsyncLock();
-
- CancellationTokenSource _connectionCancellationToken;
- CancellationTokenSource _publishingCancellationToken;
- Task _maintainConnectionTask;
-
- ManagedMqttClientStorageManager _storageManager;
-
- public ManagedMqttClient(IMqttClient mqttClient, IMqttNetLogger logger)
- {
- InternalClient = mqttClient ?? throw new ArgumentNullException(nameof(mqttClient));
-
- if (logger == null) throw new ArgumentNullException(nameof(logger));
- _logger = logger.CreateScopedLogger(nameof(ManagedMqttClient));
- }
-
- public bool IsConnected => InternalClient.IsConnected;
-
- public bool IsStarted => _connectionCancellationToken != null;
-
- public IMqttClient InternalClient { get; }
-
- public int PendingApplicationMessagesCount => _messageQueue.Count;
-
- public IManagedMqttClientOptions Options { get; private set; }
-
- public IMqttClientConnectedHandler ConnectedHandler
- {
- get => InternalClient.ConnectedHandler;
- set => InternalClient.ConnectedHandler = value;
- }
-
- public IMqttClientDisconnectedHandler DisconnectedHandler
- {
- get => InternalClient.DisconnectedHandler;
- set => InternalClient.DisconnectedHandler = value;
- }
-
- public IMqttApplicationMessageReceivedHandler ApplicationMessageReceivedHandler
- {
- get => InternalClient.ApplicationMessageReceivedHandler;
- set => InternalClient.ApplicationMessageReceivedHandler = value;
- }
-
- public IApplicationMessageProcessedHandler ApplicationMessageProcessedHandler { get; set; }
-
- public IApplicationMessageSkippedHandler ApplicationMessageSkippedHandler { get; set; }
-
- public IConnectingFailedHandler ConnectingFailedHandler { get; set; }
-
- public ISynchronizingSubscriptionsFailedHandler SynchronizingSubscriptionsFailedHandler { get; set; }
-
- public async Task StartAsync(IManagedMqttClientOptions options)
- {
- ThrowIfDisposed();
-
- if (options == null) throw new ArgumentNullException(nameof(options));
- if (options.ClientOptions == null) throw new ArgumentException("The client options are not set.", nameof(options));
-
- if (!_maintainConnectionTask?.IsCompleted ?? false) throw new InvalidOperationException("The managed client is already started.");
-
- Options = options;
-
- if (options.Storage != null)
- {
- _storageManager = new ManagedMqttClientStorageManager(options.Storage);
- var messages = await _storageManager.LoadQueuedMessagesAsync().ConfigureAwait(false);
-
- foreach (var message in messages)
- {
- _messageQueue.Enqueue(message);
- }
- }
-
- var cancellationTokenSource = new CancellationTokenSource();
- var cancellationToken = cancellationTokenSource.Token;
- _connectionCancellationToken = cancellationTokenSource;
-
- _maintainConnectionTask = Task.Run(() => MaintainConnectionAsync(cancellationToken), cancellationToken);
- _maintainConnectionTask.RunInBackground(_logger);
-
- _logger.Info("Started");
- }
-
- public async Task StopAsync()
- {
- ThrowIfDisposed();
-
- StopPublishing();
- StopMaintainingConnection();
-
- _messageQueue.Clear();
-
- if (_maintainConnectionTask != null)
- {
- await Task.WhenAny(_maintainConnectionTask);
- _maintainConnectionTask = null;
- }
- }
-
- public Task PingAsync(CancellationToken cancellationToken)
- {
- return InternalClient.PingAsync(cancellationToken);
- }
-
- public async Task<MqttClientPublishResult> PublishAsync(MqttApplicationMessage applicationMessage, CancellationToken cancellationToken)
- {
- ThrowIfDisposed();
-
- if (applicationMessage == null) throw new ArgumentNullException(nameof(applicationMessage));
-
- await PublishAsync(new ManagedMqttApplicationMessageBuilder().WithApplicationMessage(applicationMessage).Build()).ConfigureAwait(false);
- return new MqttClientPublishResult();
- }
-
- public async Task PublishAsync(ManagedMqttApplicationMessage applicationMessage)
- {
- ThrowIfDisposed();
-
- if (applicationMessage == null) throw new ArgumentNullException(nameof(applicationMessage));
- if (Options == null) throw new InvalidOperationException("call StartAsync before publishing messages");
-
- MqttTopicValidator.ThrowIfInvalid(applicationMessage.ApplicationMessage.Topic);
-
- ManagedMqttApplicationMessage removedMessage = null;
- ApplicationMessageSkippedEventArgs applicationMessageSkippedEventArgs = null;
-
- try
- {
- using (await _messageQueueLock.WaitAsync(CancellationToken.None).ConfigureAwait(false))
- {
- if (_messageQueue.Count >= Options.MaxPendingMessages)
- {
- if (Options.PendingMessagesOverflowStrategy == MqttPendingMessagesOverflowStrategy.DropNewMessage)
- {
- _logger.Verbose("Skipping publish of new application message because internal queue is full.");
- applicationMessageSkippedEventArgs = new ApplicationMessageSkippedEventArgs(applicationMessage);
- return;
- }
-
- if (Options.PendingMessagesOverflowStrategy == MqttPendingMessagesOverflowStrategy.DropOldestQueuedMessage)
- {
- removedMessage = _messageQueue.RemoveFirst();
- _logger.Verbose("Removed oldest application message from internal queue because it is full.");
- applicationMessageSkippedEventArgs = new ApplicationMessageSkippedEventArgs(removedMessage);
- }
- }
-
- _messageQueue.Enqueue(applicationMessage);
-
- if (_storageManager != null)
- {
- if (removedMessage != null)
- {
- await _storageManager.RemoveAsync(removedMessage).ConfigureAwait(false);
- }
-
- await _storageManager.AddAsync(applicationMessage).ConfigureAwait(false);
- }
- }
- }
- finally
- {
- if (applicationMessageSkippedEventArgs != null)
- {
- var applicationMessageSkippedHandler = ApplicationMessageSkippedHandler;
- if (applicationMessageSkippedHandler != null)
- {
- await applicationMessageSkippedHandler.HandleApplicationMessageSkippedAsync(applicationMessageSkippedEventArgs).ConfigureAwait(false);
- }
- }
-
- }
- }
-
- public Task SubscribeAsync(IEnumerable<MqttTopicFilter> topicFilters)
- {
- ThrowIfDisposed();
-
- if (topicFilters == null) throw new ArgumentNullException(nameof(topicFilters));
-
- lock (_subscriptions)
- {
- foreach (var topicFilter in topicFilters)
- {
- _subscriptions[topicFilter.Topic] = topicFilter.QualityOfServiceLevel;
- _unsubscriptions.Remove(topicFilter.Topic);
- }
- }
-
- _subscriptionsQueuedSignal.Release();
-
- return Task.FromResult(0);
- }
-
- public Task UnsubscribeAsync(IEnumerable<string> topics)
- {
- ThrowIfDisposed();
-
- if (topics == null) throw new ArgumentNullException(nameof(topics));
-
- lock (_subscriptions)
- {
- foreach (var topic in topics)
- {
- _subscriptions.Remove(topic);
- _unsubscriptions.Add(topic);
- }
- }
- _subscriptionsQueuedSignal.Release();
-
- return Task.FromResult(0);
- }
-
- protected override void Dispose(bool disposing)
- {
- if (disposing)
- {
- StopPublishing();
- StopMaintainingConnection();
-
- if (_maintainConnectionTask != null)
- {
- _maintainConnectionTask.GetAwaiter().GetResult();
- _maintainConnectionTask = null;
- }
-
- _messageQueue.Dispose();
- _messageQueueLock.Dispose();
- InternalClient.Dispose();
- _subscriptionsQueuedSignal.Dispose();
- }
-
- base.Dispose(disposing);
- }
-
- async Task MaintainConnectionAsync(CancellationToken cancellationToken)
- {
- try
- {
- while (!cancellationToken.IsCancellationRequested)
- {
- await TryMaintainConnectionAsync(cancellationToken).ConfigureAwait(false);
- }
- }
- catch (OperationCanceledException)
- {
- }
- catch (Exception exception)
- {
- _logger.Error(exception, "Error exception while maintaining connection.");
- }
- finally
- {
- if (!IsDisposed)
- {
- try
- {
- using (var disconnectTimeout = new CancellationTokenSource(Options.ClientOptions.CommunicationTimeout))
- {
- await InternalClient.DisconnectAsync(disconnectTimeout.Token).ConfigureAwait(false);
- }
- }
- catch (OperationCanceledException)
- {
- _logger.Warning("Timeout while sending DISCONNECT packet.");
- }
- catch (Exception exception)
- {
- _logger.Error(exception, "Error while disconnecting.");
- }
-
- _logger.Info("Stopped");
- }
-
- _reconnectSubscriptions.Clear();
-
- lock (_subscriptions)
- {
- _subscriptions.Clear();
- _unsubscriptions.Clear();
- }
- }
- }
-
- async Task TryMaintainConnectionAsync(CancellationToken cancellationToken)
- {
- try
- {
- var connectionState = await ReconnectIfRequiredAsync(cancellationToken).ConfigureAwait(false);
- if (connectionState == ReconnectionResult.NotConnected)
- {
- StopPublishing();
- await Task.Delay(Options.AutoReconnectDelay, cancellationToken).ConfigureAwait(false);
- return;
- }
-
- if (connectionState == ReconnectionResult.Reconnected)
- {
- await PublishReconnectSubscriptionsAsync().ConfigureAwait(false);
- StartPublishing();
- return;
- }
-
- if (connectionState == ReconnectionResult.Recovered)
- {
- StartPublishing();
- return;
- }
-
- if (connectionState == ReconnectionResult.StillConnected)
- {
- await PublishSubscriptionsAsync(Options.ConnectionCheckInterval, cancellationToken).ConfigureAwait(false);
- }
- }
- catch (OperationCanceledException)
- {
- }
- catch (MqttCommunicationException exception)
- {
- _logger.Warning(exception, "Communication error while maintaining connection.");
- }
- catch (Exception exception)
- {
- _logger.Error(exception, "Error exception while maintaining connection.");
- }
- }
-
- async Task PublishQueuedMessagesAsync(CancellationToken cancellationToken)
- {
- try
- {
- while (!cancellationToken.IsCancellationRequested && InternalClient.IsConnected)
- {
- // Peek at the message without dequeueing in order to prevent the
- // possibility of the queue growing beyond the configured cap.
- // Previously, messages could be re-enqueued if there was an
- // exception, and this re-enqueueing did not honor the cap.
- // Furthermore, because re-enqueueing would shuffle the order
- // of the messages, the DropOldestQueuedMessage strategy would
- // be unable to know which message is actually the oldest and would
- // instead drop the first item in the queue.
- var message = _messageQueue.PeekAndWait(cancellationToken);
- if (message == null)
- {
- continue;
- }
-
- cancellationToken.ThrowIfCancellationRequested();
-
- await TryPublishQueuedMessageAsync(message).ConfigureAwait(false);
- }
- }
- catch (OperationCanceledException)
- {
- }
- catch (Exception exception)
- {
- _logger.Error(exception, "Error while publishing queued application messages.");
- }
- finally
- {
- _logger.Verbose("Stopped publishing messages.");
- }
- }
-
- async Task TryPublishQueuedMessageAsync(ManagedMqttApplicationMessage message)
- {
- Exception transmitException = null;
- try
- {
- await InternalClient.PublishAsync(message.ApplicationMessage).ConfigureAwait(false);
-
- using (await _messageQueueLock.WaitAsync(CancellationToken.None).ConfigureAwait(false)) //lock to avoid conflict with this.PublishAsync
- {
- // While publishing this message, this.PublishAsync could have booted this
- // message off the queue to make room for another (when using a cap
- // with the DropOldestQueuedMessage strategy). If the first item
- // in the queue is equal to this message, then it's safe to remove
- // it from the queue. If not, that means this.PublishAsync has already
- // removed it, in which case we don't want to do anything.
- _messageQueue.RemoveFirst(i => i.Id.Equals(message.Id));
-
- if (_storageManager != null)
- {
- await _storageManager.RemoveAsync(message).ConfigureAwait(false);
- }
- }
- }
- catch (MqttCommunicationException exception)
- {
- transmitException = exception;
-
- _logger.Warning(exception, "Publishing application message ({0}) failed.", message.Id);
-
- if (message.ApplicationMessage.QualityOfServiceLevel == MqttQualityOfServiceLevel.AtMostOnce)
- {
- //If QoS 0, we don't want this message to stay on the queue.
- //If QoS 1 or 2, it's possible that, when using a cap, this message
- //has been booted off the queue by this.PublishAsync, in which case this
- //thread will not continue to try to publish it. While this does
- //contradict the expected behavior of QoS 1 and 2, that's also true
- //for the usage of a message queue cap, so it's still consistent
- //with prior behavior in that way.
- using (await _messageQueueLock.WaitAsync(CancellationToken.None).ConfigureAwait(false)) //lock to avoid conflict with this.PublishAsync
- {
- _messageQueue.RemoveFirst(i => i.Id.Equals(message.Id));
-
- if (_storageManager != null)
- {
- await _storageManager.RemoveAsync(message).ConfigureAwait(false);
- }
- }
- }
- }
- catch (Exception exception)
- {
- transmitException = exception;
- _logger.Error(exception, "Error while publishing application message ({0}).", message.Id);
- }
- finally
- {
- var eventHandler = ApplicationMessageProcessedHandler;
- if (eventHandler != null)
- {
- var eventArguments = new ApplicationMessageProcessedEventArgs(message, transmitException);
- await eventHandler.HandleApplicationMessageProcessedAsync(eventArguments).ConfigureAwait(false);
- }
- }
- }
-
- async Task PublishSubscriptionsAsync(TimeSpan timeout, CancellationToken cancellationToken)
- {
- var endTime = DateTime.UtcNow + timeout;
-
- while (await _subscriptionsQueuedSignal.WaitAsync(GetRemainingTime(endTime), cancellationToken).ConfigureAwait(false))
- {
- List<MqttTopicFilter> subscriptions;
- HashSet<string> unsubscriptions;
-
- lock (_subscriptions)
- {
- subscriptions = _subscriptions.Select(i => new MqttTopicFilter { Topic = i.Key, QualityOfServiceLevel = i.Value }).ToList();
- _subscriptions.Clear();
- unsubscriptions = new HashSet<string>(_unsubscriptions);
- _unsubscriptions.Clear();
- }
-
- if (!subscriptions.Any() && !unsubscriptions.Any())
- {
- continue;
- }
-
- _logger.Verbose("Publishing {0} subscriptions and {1} unsubscriptions)", subscriptions.Count, unsubscriptions.Count);
-
- foreach (var unsubscription in unsubscriptions)
- {
- _reconnectSubscriptions.Remove(unsubscription);
- }
-
- foreach (var subscription in subscriptions)
- {
- _reconnectSubscriptions[subscription.Topic] = subscription.QualityOfServiceLevel;
- }
-
- try
- {
- if (unsubscriptions.Any())
- {
- await InternalClient.UnsubscribeAsync(unsubscriptions.ToArray()).ConfigureAwait(false);
- }
-
- if (subscriptions.Any())
- {
- await InternalClient.SubscribeAsync(subscriptions.ToArray()).ConfigureAwait(false);
- }
- }
- catch (Exception exception)
- {
- await HandleSubscriptionExceptionAsync(exception).ConfigureAwait(false);
- }
- }
- }
-
- async Task PublishReconnectSubscriptionsAsync()
- {
- _logger.Info("Publishing subscriptions at reconnect");
-
- try
- {
- if (_reconnectSubscriptions.Any())
- {
- var subscriptions = _reconnectSubscriptions.Select(i => new MqttTopicFilter { Topic = i.Key, QualityOfServiceLevel = i.Value });
- await InternalClient.SubscribeAsync(subscriptions.ToArray()).ConfigureAwait(false);
- }
- }
- catch (Exception exception)
- {
- await HandleSubscriptionExceptionAsync(exception).ConfigureAwait(false);
- }
- }
-
- async Task HandleSubscriptionExceptionAsync(Exception exception)
- {
- _logger.Warning(exception, "Synchronizing subscriptions failed.");
-
- var synchronizingSubscriptionsFailedHandler = SynchronizingSubscriptionsFailedHandler;
- if (SynchronizingSubscriptionsFailedHandler != null)
- {
- await synchronizingSubscriptionsFailedHandler.HandleSynchronizingSubscriptionsFailedAsync(new ManagedProcessFailedEventArgs(exception)).ConfigureAwait(false);
- }
- }
-
- async Task<ReconnectionResult> ReconnectIfRequiredAsync(CancellationToken cancellationToken)
- {
- if (InternalClient.IsConnected)
- {
- return ReconnectionResult.StillConnected;
- }
-
- try
- {
- var result = await InternalClient.ConnectAsync(Options.ClientOptions, cancellationToken).ConfigureAwait(false);
- return result.IsSessionPresent ? ReconnectionResult.Recovered : ReconnectionResult.Reconnected;
- }
- catch (Exception exception)
- {
- var connectingFailedHandler = ConnectingFailedHandler;
- if (connectingFailedHandler != null)
- {
- await connectingFailedHandler.HandleConnectingFailedAsync(new ManagedProcessFailedEventArgs(exception)).ConfigureAwait(false);
- }
-
- return ReconnectionResult.NotConnected;
- }
- }
-
- void StartPublishing()
- {
- if (_publishingCancellationToken != null)
- {
- StopPublishing();
- }
-
- var cancellationTokenSource = new CancellationTokenSource();
- var cancellationToken = cancellationTokenSource.Token;
- _publishingCancellationToken = cancellationTokenSource;
-
- Task.Run(() => PublishQueuedMessagesAsync(cancellationToken), cancellationToken).RunInBackground(_logger);
- }
-
- void StopPublishing()
- {
- try
- {
- _publishingCancellationToken?.Cancel(false);
- }
- finally
- {
- _publishingCancellationToken?.Dispose();
- _publishingCancellationToken = null;
- }
- }
-
- void StopMaintainingConnection()
- {
- try
- {
- _connectionCancellationToken?.Cancel(false);
- }
- finally
- {
- _connectionCancellationToken?.Dispose();
- _connectionCancellationToken = null;
- }
- }
-
- static TimeSpan GetRemainingTime(DateTime endTime)
- {
- var remainingTime = endTime - DateTime.UtcNow;
- return remainingTime < TimeSpan.Zero ? TimeSpan.Zero : remainingTime;
- }
- }
- }
|