//
//  PacketHandler.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;
using System.Collections.Generic;
using System.Linq;
using System.Reflection;
using System.Threading;
using System.Threading.Tasks;
using Microsoft.Extensions.DependencyInjection;
using NosCore.Packets.Interfaces;
using Remora.Results;
namespace NosSmooth.Core.Packets;
/// 
public class PacketHandler : IPacketHandler
{
    private readonly IServiceProvider _provider;
    /// 
    /// Initializes a new instance of the  class.
    /// 
    /// The dependency injection provider.
    public PacketHandler(IServiceProvider provider)
    {
        _provider = provider;
    }
    /// 
    public Task HandleReceivedPacketAsync(IPacket packet, string packetString, CancellationToken ct)
        => HandlePacketAsync(PacketType.Received, packet, packetString, ct);
    /// 
    public Task HandleSentPacketAsync(IPacket packet, string packetString, CancellationToken ct)
        => HandlePacketAsync(PacketType.Sent, packet, packetString, ct);
    private Task HandlePacketAsync
    (
        PacketType packetType,
        IPacket packet,
        string packetString,
        CancellationToken ct = default
    )
    {
        var processMethod = GetType().GetMethod
        (
            nameof(DispatchResponder),
            BindingFlags.NonPublic | BindingFlags.Instance
        );
        if (processMethod is null)
        {
            throw new InvalidOperationException("Could not find process command generic method in command processor.");
        }
        var boundProcessMethod = processMethod.MakeGenericMethod(packet.GetType());
        return (Task)boundProcessMethod.Invoke(this, new object[]
        {
            packetType,
            packet,
            packetString,
            ct
        })!;
    }
    private async Task DispatchResponder(
        PacketType packetType,
        TPacket packet,
        string packetString,
        CancellationToken ct
    )
        where TPacket : class, IPacket
    {
        using var scope = _provider.CreateScope();
        var packetResponders = scope.ServiceProvider.GetServices>();
        var genericPacketResponders = scope.ServiceProvider.GetServices();
        var packetEventArgs = new PacketEventArgs(packetType, packet, packetString);
        var tasks = packetResponders.Select(responder => responder.Respond(packetEventArgs, ct)).ToList();
        tasks.AddRange(genericPacketResponders.Select(responder => responder.Respond(packetEventArgs, ct)));
        var results = await Task.WhenAll(tasks);
        var errors = new List();
        foreach (var result in results)
        {
            if (!result.IsSuccess)
            {
                errors.Add(result);
            }
        }
        return errors.Count switch
        {
            0 => Result.FromSuccess(),
            1 => errors[0],
            _ => new AggregateError(errors.Cast().ToArray())
        };
    }
}