using System;
using System.Collections.Generic;
using System.Linq;
using System.Threading;
using System.Threading.Tasks;
using Jellyfin.Database.Implementations;
using Jellyfin.Database.Implementations.Entities;
using MediaBrowser.Controller.Persistence;
using Microsoft.EntityFrameworkCore;
namespace Jellyfin.Server.Implementations.Item;
/// 
/// Repository for obtaining Keyframe data.
/// 
public class KeyframeRepository : IKeyframeRepository
{
    private readonly IDbContextFactory _dbProvider;
    /// 
    /// Initializes a new instance of the  class.
    /// 
    /// The EFCore db factory.
    public KeyframeRepository(IDbContextFactory dbProvider)
    {
        _dbProvider = dbProvider;
    }
    private static MediaEncoding.Keyframes.KeyframeData Map(KeyframeData entity)
    {
        return new MediaEncoding.Keyframes.KeyframeData(
            entity.TotalDuration,
            (entity.KeyframeTicks ?? []).ToList());
    }
    private KeyframeData Map(MediaEncoding.Keyframes.KeyframeData dto, Guid itemId)
    {
        return new()
        {
            ItemId = itemId,
            TotalDuration = dto.TotalDuration,
            KeyframeTicks = dto.KeyframeTicks.ToList()
        };
    }
    /// 
    public IReadOnlyList GetKeyframeData(Guid itemId)
    {
        using var context = _dbProvider.CreateDbContext();
        return context.KeyframeData.AsNoTracking().Where(e => e.ItemId.Equals(itemId)).Select(e => Map(e)).ToList();
    }
    /// 
    public async Task SaveKeyframeDataAsync(Guid itemId, MediaEncoding.Keyframes.KeyframeData data, CancellationToken cancellationToken)
    {
        using var context = _dbProvider.CreateDbContext();
        var transaction = await context.Database.BeginTransactionAsync(cancellationToken).ConfigureAwait(false);
        await using (transaction.ConfigureAwait(false))
        {
            await context.KeyframeData.Where(e => e.ItemId.Equals(itemId)).ExecuteDeleteAsync(cancellationToken).ConfigureAwait(false);
            await context.KeyframeData.AddAsync(Map(data, itemId), cancellationToken).ConfigureAwait(false);
            await context.SaveChangesAsync(cancellationToken).ConfigureAwait(false);
            await transaction.CommitAsync(cancellationToken).ConfigureAwait(false);
        }
    }
    /// 
    public async Task DeleteKeyframeDataAsync(Guid itemId, CancellationToken cancellationToken)
    {
        using var context = _dbProvider.CreateDbContext();
        await context.KeyframeData.Where(e => e.ItemId.Equals(itemId)).ExecuteDeleteAsync(cancellationToken).ConfigureAwait(false);
        await context.SaveChangesAsync(cancellationToken).ConfigureAwait(false);
    }
}