//
// Client.cs
//
// Copyright (c) František Boháček. All rights reserved.
// Licensed under the MIT license. See LICENSE file in the project root for full license information.
using System.Text.RegularExpressions;
using Microsoft.Extensions.Logging;
using NosSmooth.Core.Client;
using NosSmooth.Core.Commands;
using NosSmooth.Core.Extensions;
using NosSmooth.Core.Packets;
using NosSmooth.Packets;
using NosSmooth.Packets.Errors;
using NosSmooth.PacketSerializer.Abstractions.Attributes;
using Remora.Results;
namespace FileClient;
/// <summary>
/// A NosTale client using stream to read lines.
/// </summary>
public class Client : BaseNostaleClient
{
private const string LineRegex = ".*\\[(Recv|Send)\\]\t(.*)";
private readonly IPacketHandler _packetHandler;
private readonly IPacketSerializer _packetSerializer;
private readonly ILogger<Client> _logger;
private readonly Stream _stream;
/// <summary>
/// Initializes a new instance of the <see cref="Client"/> class.
/// </summary>
/// <param name="stream">The stream with packets.</param>
/// <param name="packetHandler">The packet handler.</param>
/// <param name="commandProcessor">The command processor.</param>
/// <param name="packetSerializer">The packet serializer.</param>
/// <param name="logger">The logger.</param>
public Client(
Stream stream,
IPacketHandler packetHandler,
CommandProcessor commandProcessor,
IPacketSerializer packetSerializer,
ILogger<Client> logger
)
: base(commandProcessor, packetSerializer)
{
_stream = stream;
_packetHandler = packetHandler;
_packetSerializer = packetSerializer;
_logger = logger;
}
/// <inheritdoc />
public override async Task<Result> RunAsync(CancellationToken stopRequested = default)
{
using var reader = new StreamReader(_stream);
var regex = new Regex(LineRegex);
while (!reader.EndOfStream)
{
stopRequested.ThrowIfCancellationRequested();
var line = await reader.ReadLineAsync();
if (line is null)
{
continue;
}
var match = regex.Match(line);
if (!match.Success)
{
_logger.LogError("Could not find match on line {Line}", line);
continue;
}
var type = match.Groups[1].Value;
var packetStr = match.Groups[2].Value;
var source = type == "Recv" ? PacketSource.Server : PacketSource.Client;
var packet = CreatePacket(packetStr, source);
Result result;
if (source == PacketSource.Client)
{
result = await _packetHandler.HandleSentPacketAsync(packet, packetStr, stopRequested);
}
else
{
result = await _packetHandler.HandleReceivedPacketAsync(packet, packetStr, stopRequested);
}
if (!result.IsSuccess)
{
_logger.LogResultError(result);
}
}
return Result.FromSuccess();
}
/// <inheritdoc/>
public override async Task<Result> SendPacketAsync(string packetString, CancellationToken ct = default)
{
return await _packetHandler.HandleReceivedPacketAsync(CreatePacket(packetString, PacketSource.Client), packetString, ct);
}
/// <inheritdoc/>
public override async Task<Result> ReceivePacketAsync(string packetString, CancellationToken ct = default)
{
return await _packetHandler.HandleReceivedPacketAsync(CreatePacket(packetString, PacketSource.Server), packetString, ct);
}
private IPacket CreatePacket(string packetStr, PacketSource source)
{
var packetResult = _packetSerializer.Deserialize(packetStr, source);
if (!packetResult.IsSuccess)
{
if (packetResult.Error is PacketConverterNotFoundError err)
{
return new UnresolvedPacket(err.Header, packetStr);
}
return new ParsingFailedPacket(packetResult, packetStr);
}
return packetResult.Entity;
}
}