|
- using System;
- using System.Collections.Generic;
- using System.Linq;
- using System.Threading;
- using System.Threading.Tasks;
- using MQTTnet.Diagnostics;
-
- namespace MQTTnet.Server
- {
- public sealed class MqttRetainedMessagesManager : IDisposable
- {
- private readonly Dictionary<string, MqttApplicationMessage> _retainedMessages = new Dictionary<string, MqttApplicationMessage>();
- private readonly SemaphoreSlim _semaphore = new SemaphoreSlim(1, 1);
- private readonly IMqttNetLogger _logger;
- private readonly IMqttServerOptions _options;
-
- public MqttRetainedMessagesManager(IMqttServerOptions options, IMqttNetLogger logger)
- {
- _logger = logger ?? throw new ArgumentNullException(nameof(logger));
- _options = options ?? throw new ArgumentNullException(nameof(options));
- }
-
- public async Task LoadMessagesAsync()
- {
- if (_options.Storage == null)
- {
- return;
- }
-
- await _semaphore.WaitAsync().ConfigureAwait(false);
- try
- {
- var retainedMessages = await _options.Storage.LoadRetainedMessagesAsync();
-
- _retainedMessages.Clear();
- foreach (var retainedMessage in retainedMessages)
- {
- _retainedMessages[retainedMessage.Topic] = retainedMessage;
- }
- }
- catch (Exception exception)
- {
- _logger.Error<MqttRetainedMessagesManager>(exception, "Unhandled exception while loading retained messages.");
- }
- finally
- {
- _semaphore.Release();
- }
- }
-
- public async Task HandleMessageAsync(string clientId, MqttApplicationMessage applicationMessage)
- {
- if (applicationMessage == null) throw new ArgumentNullException(nameof(applicationMessage));
-
- await _semaphore.WaitAsync().ConfigureAwait(false);
- try
- {
- await HandleMessageInternalAsync(clientId, applicationMessage);
- }
- catch (Exception exception)
- {
- _logger.Error<MqttRetainedMessagesManager>(exception, "Unhandled exception while handling retained messages.");
- }
- finally
- {
- _semaphore.Release();
- }
- }
-
- public async Task<List<MqttApplicationMessage>> GetSubscribedMessagesAsync(ICollection<TopicFilter> topicFilters)
- {
- var retainedMessages = new List<MqttApplicationMessage>();
-
- await _semaphore.WaitAsync().ConfigureAwait(false);
- try
- {
- foreach (var retainedMessage in _retainedMessages.Values)
- {
- foreach (var topicFilter in topicFilters)
- {
- if (!MqttTopicFilterComparer.IsMatch(retainedMessage.Topic, topicFilter.Topic))
- {
- continue;
- }
-
- retainedMessages.Add(retainedMessage);
- break;
- }
- }
- }
- finally
- {
- _semaphore.Release();
- }
-
- return retainedMessages;
- }
-
- public void Dispose()
- {
- _semaphore?.Dispose();
- }
-
- private async Task HandleMessageInternalAsync(string clientId, MqttApplicationMessage applicationMessage)
- {
- var saveIsRequired = false;
-
- if (applicationMessage.Payload?.Any() == false)
- {
- saveIsRequired = _retainedMessages.Remove(applicationMessage.Topic);
- _logger.Info<MqttRetainedMessagesManager>("Client '{0}' cleared retained message for topic '{1}'.", clientId, applicationMessage.Topic);
- }
- else
- {
- if (!_retainedMessages.ContainsKey(applicationMessage.Topic))
- {
- _retainedMessages[applicationMessage.Topic] = applicationMessage;
- saveIsRequired = true;
- }
- else
- {
- var existingMessage = _retainedMessages[applicationMessage.Topic];
- if (existingMessage.QualityOfServiceLevel != applicationMessage.QualityOfServiceLevel || !existingMessage.Payload.SequenceEqual(applicationMessage.Payload ?? new byte[0]))
- {
- _retainedMessages[applicationMessage.Topic] = applicationMessage;
- saveIsRequired = true;
- }
- }
-
- _logger.Info<MqttRetainedMessagesManager>("Client '{0}' set retained message for topic '{1}'.", clientId, applicationMessage.Topic);
- }
-
- if (!saveIsRequired)
- {
- _logger.Verbose<MqttRetainedMessagesManager>("Skipped saving retained messages because no changes were detected.");
- }
-
- if (saveIsRequired && _options.Storage != null)
- {
- await _options.Storage.SaveRetainedMessagesAsync(_retainedMessages.Values.ToList());
- }
- }
- }
- }
|