mirror of
https://github.com/Kareadita/Kavita.git
synced 2026-03-10 03:55:45 -04:00
Co-authored-by: Joseph Milazzo <joseph.v.milazzo@gmail.com> Co-authored-by: Joe Milazzo <josephmajora@gmail.com>
139 lines
5.7 KiB
C#
139 lines
5.7 KiB
C#
using System;
|
|
using System.Threading.Tasks;
|
|
using System.Collections.Immutable;
|
|
using System.Collections.Generic;
|
|
using System.Globalization;
|
|
using System.Linq;
|
|
using System.Threading;
|
|
using AutoMapper;
|
|
using Kavita.API.Database;
|
|
using Kavita.API.Services;
|
|
using Kavita.API.Services.Reading;
|
|
using Kavita.Common.Extensions;
|
|
using Kavita.Models.DTOs;
|
|
using Kavita.Models.Entities.Progress;
|
|
using Kavita.Models.Entities.User;
|
|
using Kavita.Services.Comparators;
|
|
using Kavita.Services.Extensions;
|
|
using Kavita.Services.Scanner;
|
|
using Microsoft.Extensions.Logging;
|
|
|
|
namespace Kavita.Services;
|
|
|
|
/// <summary>
|
|
/// All APIs are for Tachiyomi extension and app. They have hacks for our implementation and should not be used for any
|
|
/// other purposes.
|
|
/// </summary>
|
|
public class TachiyomiService(
|
|
IUnitOfWork unitOfWork,
|
|
IMapper mapper,
|
|
ILogger<TachiyomiService> logger,
|
|
IReaderService readerService)
|
|
: ITachiyomiService
|
|
{
|
|
private static readonly CultureInfo EnglishCulture = CultureInfo.CreateSpecificCulture("en-US");
|
|
|
|
public async Task<TachiyomiChapterDto?> GetLatestChapter(int seriesId, int userId, CancellationToken ct = default)
|
|
{
|
|
var currentChapter = await readerService.GetContinuePoint(seriesId, userId);
|
|
|
|
var prevChapterId =
|
|
await readerService.GetPrevChapterIdAsync(seriesId, currentChapter.VolumeId, currentChapter.Id, userId);
|
|
|
|
// If prevChapterId is -1, this means either nothing is read or everything is read.
|
|
if (prevChapterId == -1)
|
|
{
|
|
var series = await unitOfWork.SeriesRepository.GetSeriesDtoByIdAsync(seriesId, userId, ct);
|
|
var userHasProgress = series.PagesRead != 0 && series.PagesRead <= series.Pages;
|
|
|
|
// If the user doesn't have progress, then return null, which the extension will catch as 204 (no content) and report nothing as read
|
|
if (!userHasProgress) return null;
|
|
|
|
// Else return the max chapter to Tachiyomi so it can consider everything read
|
|
var volumes = (await unitOfWork.VolumeRepository.GetVolumes(seriesId, ct)).ToImmutableList();
|
|
var looseLeafChapterVolume = volumes.GetLooseLeafVolumeOrDefault();
|
|
if (looseLeafChapterVolume == null)
|
|
{
|
|
var volumeChapter = mapper.Map<ChapterDto>(volumes
|
|
[^1].Chapters
|
|
.OrderBy(c => c.MinNumber, ChapterSortComparerDefaultFirst.Default)
|
|
.Last());
|
|
|
|
if (volumeChapter.MinNumber.Is(Parser.LooseLeafVolumeNumber))
|
|
{
|
|
var volume = volumes.First(v => v.Id == volumeChapter.VolumeId);
|
|
return CreateTachiyomiChapterDto(volume.MinNumber);
|
|
}
|
|
|
|
return CreateTachiyomiChapterDto(volumeChapter.MinNumber);
|
|
}
|
|
|
|
var lastChapter = looseLeafChapterVolume.Chapters
|
|
.OrderBy(c => c.MinNumber, ChapterSortComparerDefaultLast.Default)
|
|
.Last();
|
|
|
|
return mapper.Map<TachiyomiChapterDto>(lastChapter);
|
|
}
|
|
|
|
// There is progress, we now need to figure out the highest volume or chapter and return that.
|
|
var prevChapter = (await unitOfWork.ChapterRepository.GetChapterDtoAsync(prevChapterId, userId, ct))!;
|
|
|
|
var volumeWithProgress = (await unitOfWork.VolumeRepository.GetVolumeDtoAsync(prevChapter.VolumeId, userId, ct))!;
|
|
// We only encode for single-file volumes
|
|
if (!volumeWithProgress.IsLooseLeaf() && volumeWithProgress.Chapters.Count == 1)
|
|
{
|
|
// The progress is on a volume, encode it as a fake chapterDTO
|
|
return CreateTachiyomiChapterDto(volumeWithProgress.MinNumber);
|
|
}
|
|
|
|
// Progress is just on a chapter, return as is
|
|
return mapper.Map<TachiyomiChapterDto>(prevChapter);
|
|
}
|
|
|
|
private static TachiyomiChapterDto CreateTachiyomiChapterDto(float number)
|
|
{
|
|
return new TachiyomiChapterDto()
|
|
{
|
|
// Use R to ensure that localization of underlying system doesn't affect the stringification
|
|
// https://docs.microsoft.com/en-us/globalization/locale/number-formatting-in-dotnet-framework
|
|
Number = (number / 10_000f).ToString("R", EnglishCulture)
|
|
};
|
|
}
|
|
|
|
public async Task<bool> MarkChaptersUntilAsRead(AppUser userWithProgress, int seriesId, float chapterNumber,
|
|
CancellationToken ct = default)
|
|
{
|
|
userWithProgress.Progresses ??= [];
|
|
|
|
switch (chapterNumber)
|
|
{
|
|
// When Tachiyomi sync's progress, if there is no current progress in Tachiyomi, 0.0f is sent.
|
|
// Due to the encoding for volumes, this marks all chapters in volume 0 (loose chapters) as read.
|
|
// Hence we catch and return early, so we ignore the request.
|
|
case 0.0f:
|
|
return true;
|
|
case < 1.0f:
|
|
{
|
|
// This is a hack to track volume number. We need to map it back by x10,000
|
|
var volumeNumber = int.Parse($"{(int)(chapterNumber * 10_000)}", EnglishCulture);
|
|
await readerService.MarkVolumesUntilAsRead(userWithProgress, seriesId, volumeNumber);
|
|
break;
|
|
}
|
|
default:
|
|
await readerService.MarkChaptersUntilAsRead(userWithProgress, seriesId, chapterNumber);
|
|
break;
|
|
}
|
|
|
|
try {
|
|
unitOfWork.UserRepository.Update(userWithProgress);
|
|
|
|
if (!unitOfWork.HasChanges()) return true;
|
|
if (await unitOfWork.CommitAsync(ct)) return true;
|
|
} catch (Exception ex) {
|
|
logger.LogError(ex, "There was an error saving progress from tachiyomi");
|
|
await unitOfWork.RollbackAsync(ct);
|
|
}
|
|
return false;
|
|
}
|
|
}
|