|
- using System;
- using System.Collections.Concurrent;
- using System.Threading;
- using System.Threading.Tasks;
- using MQTTnet.Core.Adapter;
- using MQTTnet.Core.Exceptions;
- using MQTTnet.Core.Packets;
- using MQTTnet.Core.Protocol;
- using Microsoft.Extensions.Logging;
-
- namespace MQTTnet.Core.Server
- {
- public sealed class MqttClientPendingMessagesQueue
- {
- private readonly BlockingCollection<MqttPublishPacket> _pendingPublishPackets = new BlockingCollection<MqttPublishPacket>();
- private readonly MqttServerOptions _options;
- private readonly MqttClientSession _session;
- private readonly ILogger<MqttClientPendingMessagesQueue> _logger;
-
- public MqttClientPendingMessagesQueue(MqttServerOptions options, MqttClientSession session, ILogger<MqttClientPendingMessagesQueue> logger)
- {
- _logger = logger ?? throw new ArgumentNullException(nameof(logger));
- _session = session ?? throw new ArgumentNullException(nameof(session));
- _options = options ?? throw new ArgumentNullException(nameof(options));
- }
-
- public void Start(IMqttCommunicationAdapter adapter, CancellationToken cancellationToken)
- {
- if (adapter == null) throw new ArgumentNullException(nameof(adapter));
-
- Task.Run(async () => await SendPendingPublishPacketsAsync(adapter, cancellationToken), cancellationToken).ConfigureAwait(false);
- }
-
- public void Enqueue(MqttPublishPacket publishPacket)
- {
- if (publishPacket == null) throw new ArgumentNullException(nameof(publishPacket));
-
- _pendingPublishPackets.Add(publishPacket);
- _logger.LogTrace("Enqueued packet (ClientId: {0}).", _session.ClientId);
- }
-
- private async Task SendPendingPublishPacketsAsync(IMqttCommunicationAdapter adapter, CancellationToken cancellationToken)
- {
- try
- {
- while (!cancellationToken.IsCancellationRequested)
- {
- await SendPendingPublishPacketAsync(adapter, cancellationToken);
- }
- }
- catch (OperationCanceledException)
- {
- }
- catch (Exception exception)
- {
- _logger.LogError(new EventId(), exception, "Unhandled exception while sending enqueued packet (ClientId: {0}).", _session.ClientId);
- }
- }
-
- private async Task SendPendingPublishPacketAsync(IMqttCommunicationAdapter adapter, CancellationToken cancellationToken)
- {
- MqttPublishPacket packet = null;
- try
- {
- packet = _pendingPublishPackets.Take(cancellationToken);
- await adapter.SendPacketsAsync(_options.DefaultCommunicationTimeout, cancellationToken, packet).ConfigureAwait(false);
-
- _logger.LogTrace("Enqueued packet sent (ClientId: {0}).", _session.ClientId);
- }
- catch (Exception exception)
- {
- if (exception is MqttCommunicationTimedOutException)
- {
- _logger.LogWarning(new EventId(), exception, "Sending publish packet failed due to timeout (ClientId: {0}).", _session.ClientId);
- }
- else if (exception is MqttCommunicationException)
- {
- _logger.LogWarning(new EventId(), exception, "Sending publish packet failed due to communication exception (ClientId: {0}).", _session.ClientId);
- }
- else if (exception is OperationCanceledException)
- {
- }
- else
- {
- _logger.LogError(new EventId(), exception, "Sending publish packet failed (ClientId: {0}).", _session.ClientId);
- }
-
- if (packet != null && packet.QualityOfServiceLevel > MqttQualityOfServiceLevel.AtMostOnce)
- {
- packet.Dup = true;
- _pendingPublishPackets.Add(packet, CancellationToken.None);
- }
-
- _session.Stop();
- }
- }
- }
- }
|