using System.Threading.Tasks; using API.Data; using API.DTOs.Koreader; using API.DTOs.Progress; using API.Extensions; using API.Helpers; using API.Helpers.Builders; using Kavita.Common; using Microsoft.Extensions.Logging; namespace API.Services; #nullable enable public interface IKoreaderService { Task SaveProgress(KoreaderBookDto koreaderBookDto, int userId); Task GetProgress(string bookHash, int userId); } public class KoreaderService : IKoreaderService { private readonly IReaderService _readerService; private readonly IUnitOfWork _unitOfWork; private readonly ILocalizationService _localizationService; private readonly ILogger _logger; public KoreaderService(IReaderService readerService, IUnitOfWork unitOfWork, ILocalizationService localizationService, ILogger logger) { _readerService = readerService; _unitOfWork = unitOfWork; _localizationService = localizationService; _logger = logger; } /// /// Given a Koreader hash, locate the underlying file and generate/update a progress event. /// /// /// public async Task SaveProgress(KoreaderBookDto koreaderBookDto, int userId) { _logger.LogDebug("Saving Koreader progress for User ({UserId}): {KoreaderProgress}", userId, koreaderBookDto.progress.Sanitize()); var file = await _unitOfWork.MangaFileRepository.GetByKoreaderHash(koreaderBookDto.document); if (file == null) throw new KavitaException(await _localizationService.Translate(userId, "file-missing")); var userProgressDto = await _unitOfWork.AppUserProgressRepository.GetUserProgressDtoAsync(file.ChapterId, userId); if (userProgressDto == null) { var chapterDto = await _unitOfWork.ChapterRepository.GetChapterDtoAsync(file.ChapterId); if (chapterDto == null) throw new KavitaException(await _localizationService.Translate(userId, "chapter-doesnt-exist")); var volumeDto = await _unitOfWork.VolumeRepository.GetVolumeByIdAsync(chapterDto.VolumeId); if (volumeDto == null) throw new KavitaException(await _localizationService.Translate(userId, "volume-doesnt-exist")); userProgressDto = new ProgressDto() { ChapterId = file.ChapterId, VolumeId = chapterDto.VolumeId, SeriesId = volumeDto.SeriesId, }; } // Update the bookScrollId if possible KoreaderHelper.UpdateProgressDto(userProgressDto, koreaderBookDto.progress); await _readerService.SaveReadingProgress(userProgressDto, userId); } /// /// Returns a Koreader Dto representing current book and the progress within /// /// /// /// public async Task GetProgress(string bookHash, int userId) { var settingsDto = await _unitOfWork.SettingsRepository.GetSettingsDtoAsync(); var file = await _unitOfWork.MangaFileRepository.GetByKoreaderHash(bookHash); if (file == null) throw new KavitaException(await _localizationService.Translate(userId, "file-missing")); var progressDto = await _unitOfWork.AppUserProgressRepository.GetUserProgressDtoAsync(file.ChapterId, userId); var koreaderProgress = KoreaderHelper.GetKoreaderPosition(progressDto); return new KoreaderBookDtoBuilder(bookHash).WithProgress(koreaderProgress) .WithPercentage(progressDto?.PageNum, file.Pages) .WithDeviceId(settingsDto.InstallId, userId) .WithTimestamp(progressDto?.LastModifiedUtc) .Build(); } }