< Summary

Information
Class: NostrSure.Infrastructure.Client.Implementation.MessageSender
Assembly: NostrSure.Infrastructure
File(s): /home/runner/work/NostrSure/NostrSure/NostrSure.Infrastructure/Client/Implementation/MessageSender.cs
Line coverage
48%
Covered lines: 17
Uncovered lines: 18
Coverable lines: 35
Total lines: 63
Line coverage: 48.5%
Branch coverage
27%
Covered branches: 5
Total branches: 18
Branch coverage: 27.7%
Method coverage

Feature is only available for sponsors

Upgrade to PRO version

Metrics

MethodBranch coverage Crap Score Cyclomatic complexity Line coverage
.ctor(...)50%66100%
SendAsync()16.66%731225%

File(s)

/home/runner/work/NostrSure/NostrSure/NostrSure.Infrastructure/Client/Implementation/MessageSender.cs

#LineLine coverage
 1using System.Net.WebSockets;
 2using System.Text;
 3using Microsoft.Extensions.Logging;
 4using NostrSure.Infrastructure.Client.Abstractions;
 5
 6namespace NostrSure.Infrastructure.Client.Implementation;
 7
 8/// <summary>
 9/// Handles WebSocket message transmission
 10/// </summary>
 11public sealed class MessageSender : IMessageSender
 12{
 13    private readonly ClientWebSocket _webSocket;
 14    private readonly IConnectionErrorHandler _errorHandler;
 15    private readonly IConnectionStateManager _stateManager;
 16    private readonly ILogger<MessageSender>? _logger;
 17
 318    public MessageSender(
 319        ClientWebSocket webSocket,
 320        IConnectionErrorHandler errorHandler,
 321        IConnectionStateManager stateManager,
 322        ILogger<MessageSender>? logger = null)
 323    {
 324        _webSocket = webSocket ?? throw new ArgumentNullException(nameof(webSocket));
 325        _errorHandler = errorHandler ?? throw new ArgumentNullException(nameof(errorHandler));
 326        _stateManager = stateManager ?? throw new ArgumentNullException(nameof(stateManager));
 327        _logger = logger;
 328    }
 29
 30    public async Task SendAsync(string message, CancellationToken cancellationToken = default)
 231    {
 32        // Check actual WebSocket state as primary source of truth
 233        if (_webSocket.State != WebSocketState.Open)
 234        {
 235            _logger?.LogError("WebSocket send failed - WebSocket.State: {WebSocketState}, StateManager.IsConnected: {Sta
 236                             _webSocket.State, _stateManager.IsConnected);
 237            throw new InvalidOperationException($"WebSocket is not connected. Current state: {_webSocket.State}");
 38        }
 39
 040        if (string.IsNullOrEmpty(message))
 041            throw new ArgumentException("Message cannot be null or empty", nameof(message));
 42
 43        try
 044        {
 045            _logger?.LogDebug("Sending message: {MessageLength} characters", message.Length);
 46
 047            var buffer = Encoding.UTF8.GetBytes(message);
 048            await _webSocket.SendAsync(
 049                new ArraySegment<byte>(buffer),
 050                WebSocketMessageType.Text,
 051                endOfMessage: true,
 052                cancellationToken);
 53
 054            _logger?.LogDebug("Message sent successfully");
 055        }
 056        catch (Exception ex)
 057        {
 058            _logger?.LogError(ex, "Failed to send message");
 059            await _errorHandler.HandleErrorAsync(ex, nameof(SendAsync));
 060            throw;
 61        }
 062    }
 63}