mirror of
https://github.com/Kareadita/Kavita.git
synced 2025-05-31 04:04:19 -04:00
798 lines
35 KiB
C#
798 lines
35 KiB
C#
using System;
|
|
using System.Collections.Generic;
|
|
using System.Globalization;
|
|
using System.IO;
|
|
using System.Linq;
|
|
using System.Threading.Tasks;
|
|
using API.Comparators;
|
|
using API.Data;
|
|
using API.Data.Repositories;
|
|
using API.DTOs;
|
|
using API.DTOs.CollectionTags;
|
|
using API.DTOs.SeriesDetail;
|
|
using API.Entities;
|
|
using API.Entities.Enums;
|
|
using API.Entities.Metadata;
|
|
using API.Helpers;
|
|
using API.Helpers.Builders;
|
|
using API.Services.Plus;
|
|
using API.SignalR;
|
|
using Hangfire;
|
|
using Kavita.Common;
|
|
using Microsoft.Extensions.Logging;
|
|
|
|
namespace API.Services;
|
|
|
|
|
|
public interface ISeriesService
|
|
{
|
|
Task<SeriesDetailDto> GetSeriesDetail(int seriesId, int userId);
|
|
Task<bool> UpdateSeriesMetadata(UpdateSeriesMetadataDto updateSeriesMetadataDto);
|
|
Task<bool> UpdateRating(AppUser user, UpdateSeriesRatingDto updateSeriesRatingDto);
|
|
Task<bool> DeleteMultipleSeries(IList<int> seriesIds);
|
|
Task<bool> UpdateRelatedSeries(UpdateRelatedSeriesDto dto);
|
|
Task<RelatedSeriesDto> GetRelatedSeries(int userId, int seriesId);
|
|
Task<string> FormatChapterTitle(int userId, ChapterDto chapter, LibraryType libraryType, bool withHash = true);
|
|
Task<string> FormatChapterTitle(int userId, Chapter chapter, LibraryType libraryType, bool withHash = true);
|
|
|
|
Task<string> FormatChapterTitle(int userId, bool isSpecial, LibraryType libraryType, string? chapterTitle,
|
|
bool withHash);
|
|
Task<string> FormatChapterName(int userId, LibraryType libraryType, bool withHash = false);
|
|
Task<NextExpectedChapterDto> GetEstimatedChapterCreationDate(int seriesId, int userId);
|
|
}
|
|
|
|
public class SeriesService : ISeriesService
|
|
{
|
|
private readonly IUnitOfWork _unitOfWork;
|
|
private readonly IEventHub _eventHub;
|
|
private readonly ITaskScheduler _taskScheduler;
|
|
private readonly ILogger<SeriesService> _logger;
|
|
private readonly IScrobblingService _scrobblingService;
|
|
private readonly ILocalizationService _localizationService;
|
|
|
|
private readonly NextExpectedChapterDto _emptyExpectedChapter = new NextExpectedChapterDto()
|
|
{
|
|
ExpectedDate = null,
|
|
ChapterNumber = 0,
|
|
VolumeNumber = 0
|
|
};
|
|
|
|
public SeriesService(IUnitOfWork unitOfWork, IEventHub eventHub, ITaskScheduler taskScheduler,
|
|
ILogger<SeriesService> logger, IScrobblingService scrobblingService, ILocalizationService localizationService)
|
|
{
|
|
_unitOfWork = unitOfWork;
|
|
_eventHub = eventHub;
|
|
_taskScheduler = taskScheduler;
|
|
_logger = logger;
|
|
_scrobblingService = scrobblingService;
|
|
_localizationService = localizationService;
|
|
|
|
}
|
|
|
|
/// <summary>
|
|
/// Returns the first chapter for a series to extract metadata from (ie Summary, etc)
|
|
/// </summary>
|
|
/// <param name="series">The full series with all volumes and chapters on it</param>
|
|
/// <returns></returns>
|
|
public static Chapter? GetFirstChapterForMetadata(Series series)
|
|
{
|
|
var sortedVolumes = series.Volumes
|
|
.Where(v => float.TryParse(v.Name, out var parsedValue) && parsedValue != 0.0f)
|
|
.OrderBy(v => float.TryParse(v.Name, out var parsedValue) ? parsedValue : float.MaxValue);
|
|
var minVolumeNumber = sortedVolumes
|
|
.MinBy(v => float.Parse(v.Name));
|
|
|
|
|
|
var allChapters = series.Volumes
|
|
.SelectMany(v => v.Chapters.OrderBy(c => float.Parse(c.Number), ChapterSortComparer.Default)).ToList();
|
|
var minChapter = allChapters
|
|
.FirstOrDefault();
|
|
|
|
if (minVolumeNumber != null && minChapter != null && float.TryParse(minChapter.Number, out var chapNum) && chapNum >= minVolumeNumber.Number)
|
|
{
|
|
return minVolumeNumber.Chapters.MinBy(c => float.Parse(c.Number), ChapterSortComparer.Default);
|
|
}
|
|
|
|
return minChapter;
|
|
}
|
|
|
|
public async Task<bool> UpdateSeriesMetadata(UpdateSeriesMetadataDto updateSeriesMetadataDto)
|
|
{
|
|
try
|
|
{
|
|
var seriesId = updateSeriesMetadataDto.SeriesMetadata.SeriesId;
|
|
var series = await _unitOfWork.SeriesRepository.GetSeriesByIdAsync(seriesId);
|
|
if (series == null) return false;
|
|
var allCollectionTags = (await _unitOfWork.CollectionTagRepository.GetAllTagsAsync()).ToList();
|
|
var allGenres = (await _unitOfWork.GenreRepository.GetAllGenresAsync()).ToList();
|
|
var allPeople = (await _unitOfWork.PersonRepository.GetAllPeople()).ToList();
|
|
var allTags = (await _unitOfWork.TagRepository.GetAllTagsAsync()).ToList();
|
|
|
|
series.Metadata ??= new SeriesMetadataBuilder()
|
|
.WithCollectionTags(updateSeriesMetadataDto.CollectionTags.Select(dto =>
|
|
new CollectionTagBuilder(dto.Title)
|
|
.WithId(dto.Id)
|
|
.WithSummary(dto.Summary)
|
|
.WithIsPromoted(dto.Promoted)
|
|
.Build()).ToList())
|
|
.Build();
|
|
|
|
if (series.Metadata.AgeRating != updateSeriesMetadataDto.SeriesMetadata.AgeRating)
|
|
{
|
|
series.Metadata.AgeRating = updateSeriesMetadataDto.SeriesMetadata.AgeRating;
|
|
series.Metadata.AgeRatingLocked = true;
|
|
}
|
|
|
|
if (NumberHelper.IsValidYear(updateSeriesMetadataDto.SeriesMetadata.ReleaseYear) && series.Metadata.ReleaseYear != updateSeriesMetadataDto.SeriesMetadata.ReleaseYear)
|
|
{
|
|
series.Metadata.ReleaseYear = updateSeriesMetadataDto.SeriesMetadata.ReleaseYear;
|
|
series.Metadata.ReleaseYearLocked = true;
|
|
}
|
|
|
|
if (series.Metadata.PublicationStatus != updateSeriesMetadataDto.SeriesMetadata.PublicationStatus)
|
|
{
|
|
series.Metadata.PublicationStatus = updateSeriesMetadataDto.SeriesMetadata.PublicationStatus;
|
|
series.Metadata.PublicationStatusLocked = true;
|
|
}
|
|
|
|
if (string.IsNullOrEmpty(updateSeriesMetadataDto.SeriesMetadata.Summary))
|
|
{
|
|
updateSeriesMetadataDto.SeriesMetadata.Summary = string.Empty;
|
|
}
|
|
|
|
if (series.Metadata.Summary != updateSeriesMetadataDto.SeriesMetadata.Summary.Trim())
|
|
{
|
|
series.Metadata.Summary = updateSeriesMetadataDto.SeriesMetadata?.Summary.Trim() ?? string.Empty;
|
|
series.Metadata.SummaryLocked = true;
|
|
}
|
|
|
|
if (series.Metadata.Language != updateSeriesMetadataDto.SeriesMetadata?.Language)
|
|
{
|
|
series.Metadata.Language = updateSeriesMetadataDto.SeriesMetadata?.Language ?? string.Empty;
|
|
series.Metadata.LanguageLocked = true;
|
|
}
|
|
|
|
if (string.IsNullOrEmpty(updateSeriesMetadataDto.SeriesMetadata?.WebLinks))
|
|
{
|
|
series.Metadata.WebLinks = string.Empty;
|
|
} else
|
|
{
|
|
series.Metadata.WebLinks = string.Join(",", updateSeriesMetadataDto.SeriesMetadata?.WebLinks
|
|
.Split(",")
|
|
.Where(s => !string.IsNullOrEmpty(s))
|
|
.Select(s => s.Trim())!
|
|
);
|
|
}
|
|
|
|
|
|
series.Metadata.CollectionTags ??= new List<CollectionTag>();
|
|
UpdateCollectionsList(updateSeriesMetadataDto.CollectionTags, series, allCollectionTags, (tag) =>
|
|
{
|
|
series.Metadata.CollectionTags.Add(tag);
|
|
});
|
|
|
|
series.Metadata.Genres ??= new List<Genre>();
|
|
GenreHelper.UpdateGenreList(updateSeriesMetadataDto.SeriesMetadata?.Genres, series, allGenres, (genre) =>
|
|
{
|
|
series.Metadata.Genres.Add(genre);
|
|
}, () => series.Metadata.GenresLocked = true);
|
|
|
|
series.Metadata.Tags ??= new List<Tag>();
|
|
TagHelper.UpdateTagList(updateSeriesMetadataDto.SeriesMetadata?.Tags, series, allTags, (tag) =>
|
|
{
|
|
series.Metadata.Tags.Add(tag);
|
|
}, () => series.Metadata.TagsLocked = true);
|
|
|
|
void HandleAddPerson(Person person)
|
|
{
|
|
PersonHelper.AddPersonIfNotExists(series.Metadata.People, person);
|
|
allPeople.Add(person);
|
|
}
|
|
|
|
series.Metadata.People ??= new List<Person>();
|
|
PersonHelper.UpdatePeopleList(PersonRole.Writer, updateSeriesMetadataDto.SeriesMetadata!.Writers, series, allPeople,
|
|
HandleAddPerson, () => series.Metadata.WriterLocked = true);
|
|
PersonHelper.UpdatePeopleList(PersonRole.Character, updateSeriesMetadataDto.SeriesMetadata.Characters, series, allPeople,
|
|
HandleAddPerson, () => series.Metadata.CharacterLocked = true);
|
|
PersonHelper.UpdatePeopleList(PersonRole.Colorist, updateSeriesMetadataDto.SeriesMetadata.Colorists, series, allPeople,
|
|
HandleAddPerson, () => series.Metadata.ColoristLocked = true);
|
|
PersonHelper.UpdatePeopleList(PersonRole.Editor, updateSeriesMetadataDto.SeriesMetadata.Editors, series, allPeople,
|
|
HandleAddPerson, () => series.Metadata.EditorLocked = true);
|
|
PersonHelper.UpdatePeopleList(PersonRole.Inker, updateSeriesMetadataDto.SeriesMetadata.Inkers, series, allPeople,
|
|
HandleAddPerson, () => series.Metadata.InkerLocked = true);
|
|
PersonHelper.UpdatePeopleList(PersonRole.Letterer, updateSeriesMetadataDto.SeriesMetadata.Letterers, series, allPeople,
|
|
HandleAddPerson, () => series.Metadata.LettererLocked = true);
|
|
PersonHelper.UpdatePeopleList(PersonRole.Penciller, updateSeriesMetadataDto.SeriesMetadata.Pencillers, series, allPeople,
|
|
HandleAddPerson, () => series.Metadata.PencillerLocked = true);
|
|
PersonHelper.UpdatePeopleList(PersonRole.Publisher, updateSeriesMetadataDto.SeriesMetadata.Publishers, series, allPeople,
|
|
HandleAddPerson, () => series.Metadata.PublisherLocked = true);
|
|
PersonHelper.UpdatePeopleList(PersonRole.Translator, updateSeriesMetadataDto.SeriesMetadata.Translators, series, allPeople,
|
|
HandleAddPerson, () => series.Metadata.TranslatorLocked = true);
|
|
PersonHelper.UpdatePeopleList(PersonRole.CoverArtist, updateSeriesMetadataDto.SeriesMetadata.CoverArtists, series, allPeople,
|
|
HandleAddPerson, () => series.Metadata.CoverArtistLocked = true);
|
|
|
|
series.Metadata.AgeRatingLocked = updateSeriesMetadataDto.SeriesMetadata.AgeRatingLocked;
|
|
series.Metadata.PublicationStatusLocked = updateSeriesMetadataDto.SeriesMetadata.PublicationStatusLocked;
|
|
series.Metadata.LanguageLocked = updateSeriesMetadataDto.SeriesMetadata.LanguageLocked;
|
|
series.Metadata.GenresLocked = updateSeriesMetadataDto.SeriesMetadata.GenresLocked;
|
|
series.Metadata.TagsLocked = updateSeriesMetadataDto.SeriesMetadata.TagsLocked;
|
|
series.Metadata.CharacterLocked = updateSeriesMetadataDto.SeriesMetadata.CharactersLocked;
|
|
series.Metadata.ColoristLocked = updateSeriesMetadataDto.SeriesMetadata.ColoristsLocked;
|
|
series.Metadata.EditorLocked = updateSeriesMetadataDto.SeriesMetadata.EditorsLocked;
|
|
series.Metadata.InkerLocked = updateSeriesMetadataDto.SeriesMetadata.InkersLocked;
|
|
series.Metadata.LettererLocked = updateSeriesMetadataDto.SeriesMetadata.LetterersLocked;
|
|
series.Metadata.PencillerLocked = updateSeriesMetadataDto.SeriesMetadata.PencillersLocked;
|
|
series.Metadata.PublisherLocked = updateSeriesMetadataDto.SeriesMetadata.PublishersLocked;
|
|
series.Metadata.TranslatorLocked = updateSeriesMetadataDto.SeriesMetadata.TranslatorsLocked;
|
|
series.Metadata.CoverArtistLocked = updateSeriesMetadataDto.SeriesMetadata.CoverArtistsLocked;
|
|
series.Metadata.WriterLocked = updateSeriesMetadataDto.SeriesMetadata.WritersLocked;
|
|
series.Metadata.SummaryLocked = updateSeriesMetadataDto.SeriesMetadata.SummaryLocked;
|
|
series.Metadata.ReleaseYearLocked = updateSeriesMetadataDto.SeriesMetadata.ReleaseYearLocked;
|
|
|
|
if (!_unitOfWork.HasChanges())
|
|
{
|
|
return true;
|
|
}
|
|
|
|
await _unitOfWork.CommitAsync();
|
|
|
|
// Trigger code to cleanup tags, collections, people, etc
|
|
try
|
|
{
|
|
await _taskScheduler.CleanupDbEntries();
|
|
}
|
|
catch (Exception ex)
|
|
{
|
|
_logger.LogError(ex, "There was an issue cleaning up DB entries. This may happen if Komf is spamming updates. Nightly cleanup will work");
|
|
}
|
|
|
|
|
|
if (updateSeriesMetadataDto.CollectionTags == null) return true;
|
|
foreach (var tag in updateSeriesMetadataDto.CollectionTags)
|
|
{
|
|
await _eventHub.SendMessageAsync(MessageFactory.SeriesAddedToCollection,
|
|
MessageFactory.SeriesAddedToCollectionEvent(tag.Id,
|
|
updateSeriesMetadataDto.SeriesMetadata.SeriesId), false);
|
|
}
|
|
return true;
|
|
}
|
|
catch (Exception ex)
|
|
{
|
|
_logger.LogError(ex, "There was an exception when updating metadata");
|
|
await _unitOfWork.RollbackAsync();
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
|
|
public static void UpdateCollectionsList(ICollection<CollectionTagDto>? tags, Series series, IReadOnlyCollection<CollectionTag> allTags,
|
|
Action<CollectionTag> handleAdd)
|
|
{
|
|
// TODO: Move UpdateCollectionsList to a helper so we can easily test
|
|
if (tags == null) return;
|
|
// I want a union of these 2 lists. Return only elements that are in both lists, but the list types are different
|
|
var existingTags = series.Metadata.CollectionTags.ToList();
|
|
foreach (var existing in existingTags)
|
|
{
|
|
if (tags.SingleOrDefault(t => t.Id == existing.Id) == null)
|
|
{
|
|
// Remove tag
|
|
series.Metadata.CollectionTags.Remove(existing);
|
|
}
|
|
}
|
|
|
|
// At this point, all tags that aren't in dto have been removed.
|
|
foreach (var tag in tags)
|
|
{
|
|
var existingTag = allTags.SingleOrDefault(t => t.Title == tag.Title);
|
|
if (existingTag != null)
|
|
{
|
|
if (series.Metadata.CollectionTags.All(t => t.Title != tag.Title))
|
|
{
|
|
handleAdd(existingTag);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
// Add new tag
|
|
handleAdd(new CollectionTagBuilder(tag.Title)
|
|
.WithId(tag.Id)
|
|
.WithSummary(tag.Summary)
|
|
.WithIsPromoted(tag.Promoted)
|
|
.Build());
|
|
}
|
|
}
|
|
}
|
|
|
|
/// <summary>
|
|
///
|
|
/// </summary>
|
|
/// <param name="user">User with Ratings includes</param>
|
|
/// <param name="updateSeriesRatingDto"></param>
|
|
/// <returns></returns>
|
|
public async Task<bool> UpdateRating(AppUser? user, UpdateSeriesRatingDto updateSeriesRatingDto)
|
|
{
|
|
if (user == null)
|
|
{
|
|
_logger.LogError("Cannot update rating of null user");
|
|
return false;
|
|
}
|
|
|
|
var userRating =
|
|
await _unitOfWork.UserRepository.GetUserRatingAsync(updateSeriesRatingDto.SeriesId, user.Id) ??
|
|
new AppUserRating();
|
|
try
|
|
{
|
|
userRating.Rating = Math.Clamp(updateSeriesRatingDto.UserRating, 0f, 5f);
|
|
userRating.HasBeenRated = true;
|
|
userRating.SeriesId = updateSeriesRatingDto.SeriesId;
|
|
|
|
if (userRating.Id == 0)
|
|
{
|
|
user.Ratings ??= new List<AppUserRating>();
|
|
user.Ratings.Add(userRating);
|
|
}
|
|
|
|
_unitOfWork.UserRepository.Update(user);
|
|
|
|
if (!_unitOfWork.HasChanges() || await _unitOfWork.CommitAsync())
|
|
{
|
|
BackgroundJob.Enqueue(() =>
|
|
_scrobblingService.ScrobbleRatingUpdate(user.Id, updateSeriesRatingDto.SeriesId,
|
|
userRating.Rating));
|
|
return true;
|
|
}
|
|
}
|
|
catch (Exception ex)
|
|
{
|
|
_logger.LogError(ex, "There was an exception saving rating");
|
|
}
|
|
|
|
await _unitOfWork.RollbackAsync();
|
|
user.Ratings?.Remove(userRating);
|
|
|
|
return false;
|
|
}
|
|
|
|
public async Task<bool> DeleteMultipleSeries(IList<int> seriesIds)
|
|
{
|
|
try
|
|
{
|
|
var chapterMappings =
|
|
await _unitOfWork.SeriesRepository.GetChapterIdWithSeriesIdForSeriesAsync(seriesIds.ToArray());
|
|
|
|
var allChapterIds = new List<int>();
|
|
foreach (var mapping in chapterMappings)
|
|
{
|
|
allChapterIds.AddRange(mapping.Value);
|
|
}
|
|
|
|
var series = await _unitOfWork.SeriesRepository.GetSeriesByIdsAsync(seriesIds);
|
|
var libraryIds = series.Select(s => s.LibraryId);
|
|
var libraries = await _unitOfWork.LibraryRepository.GetLibraryForIdsAsync(libraryIds);
|
|
foreach (var library in libraries)
|
|
{
|
|
library.UpdateLastModified();
|
|
_unitOfWork.LibraryRepository.Update(library);
|
|
}
|
|
|
|
_unitOfWork.SeriesRepository.Remove(series);
|
|
|
|
|
|
if (!_unitOfWork.HasChanges() || !await _unitOfWork.CommitAsync()) return true;
|
|
|
|
foreach (var s in series)
|
|
{
|
|
await _eventHub.SendMessageAsync(MessageFactory.SeriesRemoved,
|
|
MessageFactory.SeriesRemovedEvent(s.Id, s.Name, s.LibraryId), false);
|
|
}
|
|
|
|
await _unitOfWork.AppUserProgressRepository.CleanupAbandonedChapters();
|
|
await _unitOfWork.CollectionTagRepository.RemoveTagsWithoutSeries();
|
|
_taskScheduler.CleanupChapters(allChapterIds.ToArray());
|
|
}
|
|
catch (Exception ex)
|
|
{
|
|
_logger.LogError(ex, "There was an issue when trying to delete multiple series");
|
|
return false;
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
/// <summary>
|
|
/// This generates all the arrays needed by the Series Detail page in the UI. It is a specialized API for the unique layout constraints.
|
|
/// </summary>
|
|
/// <param name="seriesId"></param>
|
|
/// <param name="userId"></param>
|
|
/// <returns></returns>
|
|
public async Task<SeriesDetailDto> GetSeriesDetail(int seriesId, int userId)
|
|
{
|
|
var series = await _unitOfWork.SeriesRepository.GetSeriesDtoByIdAsync(seriesId, userId);
|
|
if (series == null) throw new KavitaException(await _localizationService.Translate(userId, "series-doesnt-exist"));
|
|
var libraryIds = _unitOfWork.LibraryRepository.GetLibraryIdsForUserIdAsync(userId);
|
|
if (!libraryIds.Contains(series.LibraryId))
|
|
throw new UnauthorizedAccessException("user-no-access-library-from-series");
|
|
|
|
var user = await _unitOfWork.UserRepository.GetUserByIdAsync(userId);
|
|
if (user!.AgeRestriction != AgeRating.NotApplicable)
|
|
{
|
|
var seriesMetadata = await _unitOfWork.SeriesRepository.GetSeriesMetadata(seriesId);
|
|
if (seriesMetadata!.AgeRating > user.AgeRestriction)
|
|
throw new UnauthorizedAccessException("series-restricted-age-restriction");
|
|
}
|
|
|
|
|
|
var libraryType = await _unitOfWork.LibraryRepository.GetLibraryTypeAsync(series.LibraryId);
|
|
var volumes = (await _unitOfWork.VolumeRepository.GetVolumesDtoAsync(seriesId, userId))
|
|
.OrderBy(v => Tasks.Scanner.Parser.Parser.MinNumberFromRange(v.Name))
|
|
.ToList();
|
|
|
|
// For books, the Name of the Volume is remapped to the actual name of the book, rather than Volume number.
|
|
var processedVolumes = new List<VolumeDto>();
|
|
if (libraryType == LibraryType.Book)
|
|
{
|
|
var volumeLabel = await _localizationService.Translate(userId, "volume-num", string.Empty);
|
|
foreach (var volume in volumes)
|
|
{
|
|
volume.Chapters = volume.Chapters.OrderBy(d => double.Parse(d.Number), ChapterSortComparer.Default).ToList();
|
|
var firstChapter = volume.Chapters.First();
|
|
// On Books, skip volumes that are specials, since these will be shown
|
|
if (firstChapter.IsSpecial) continue;
|
|
RenameVolumeName(firstChapter, volume, libraryType, volumeLabel);
|
|
processedVolumes.Add(volume);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
processedVolumes = volumes.Where(v => v.Number > 0).ToList();
|
|
processedVolumes.ForEach(v =>
|
|
{
|
|
v.Name = $"Volume {v.Name}";
|
|
v.Chapters = v.Chapters.OrderBy(d => double.Parse(d.Number), ChapterSortComparer.Default).ToList();
|
|
});
|
|
}
|
|
|
|
var specials = new List<ChapterDto>();
|
|
var chapters = volumes.SelectMany(v => v.Chapters.Select(c =>
|
|
{
|
|
if (v.Number == 0) return c;
|
|
c.VolumeTitle = v.Name;
|
|
return c;
|
|
}).OrderBy(c => float.Parse(c.Number), ChapterSortComparer.Default)).ToList();
|
|
|
|
foreach (var chapter in chapters)
|
|
{
|
|
chapter.Title = await FormatChapterTitle(userId, chapter, libraryType);
|
|
if (!chapter.IsSpecial) continue;
|
|
|
|
if (!string.IsNullOrEmpty(chapter.TitleName)) chapter.Title = chapter.TitleName;
|
|
specials.Add(chapter);
|
|
}
|
|
|
|
// Don't show chapter 0 (aka single volume chapters) in the Chapters tab or books that are just single numbers (they show as volumes)
|
|
IEnumerable<ChapterDto> retChapters;
|
|
if (libraryType == LibraryType.Book)
|
|
{
|
|
retChapters = Array.Empty<ChapterDto>();
|
|
} else
|
|
{
|
|
retChapters = chapters
|
|
.Where(ShouldIncludeChapter);
|
|
}
|
|
|
|
var storylineChapters = volumes
|
|
.Where(v => v.Number == 0)
|
|
.SelectMany(v => v.Chapters.Where(c => !c.IsSpecial))
|
|
.OrderBy(c => float.Parse(c.Number), ChapterSortComparer.Default)
|
|
.ToList();
|
|
|
|
// When there's chapters without a volume number revert to chapter sorting only as opposed to volume then chapter
|
|
if (storylineChapters.Any()) {
|
|
retChapters = retChapters.OrderBy(c => float.Parse(c.Number), ChapterSortComparer.Default);
|
|
}
|
|
|
|
return new SeriesDetailDto()
|
|
{
|
|
Specials = specials,
|
|
Chapters = retChapters,
|
|
Volumes = processedVolumes,
|
|
StorylineChapters = storylineChapters,
|
|
TotalCount = chapters.Count,
|
|
UnreadCount = chapters.Count(c => c.Pages > 0 && c.PagesRead < c.Pages),
|
|
};
|
|
}
|
|
|
|
/// <summary>
|
|
/// Should we show the given chapter on the UI. We only show non-specials and non-zero chapters.
|
|
/// </summary>
|
|
/// <param name="chapter"></param>
|
|
/// <returns></returns>
|
|
private static bool ShouldIncludeChapter(ChapterDto chapter)
|
|
{
|
|
return !chapter.IsSpecial && !chapter.Number.Equals(Tasks.Scanner.Parser.Parser.DefaultChapter);
|
|
}
|
|
|
|
public static void RenameVolumeName(ChapterDto firstChapter, VolumeDto volume, LibraryType libraryType, string volumeLabel = "Volume")
|
|
{
|
|
if (libraryType == LibraryType.Book)
|
|
{
|
|
if (string.IsNullOrEmpty(firstChapter.TitleName))
|
|
{
|
|
if (firstChapter.Range.Equals(Tasks.Scanner.Parser.Parser.DefaultVolume)) return;
|
|
var title = Path.GetFileNameWithoutExtension(firstChapter.Range);
|
|
if (string.IsNullOrEmpty(title)) return;
|
|
volume.Name += $" - {title}";
|
|
}
|
|
else if (volume.Name != "0")
|
|
{
|
|
volume.Name += $" - {firstChapter.TitleName}";
|
|
}
|
|
// else
|
|
// {
|
|
// volume.Name += $"";
|
|
// }
|
|
|
|
return;
|
|
}
|
|
|
|
volume.Name = $"{volumeLabel} {volume.Name}".Trim();
|
|
}
|
|
|
|
|
|
public async Task<string> FormatChapterTitle(int userId, bool isSpecial, LibraryType libraryType, string? chapterTitle, bool withHash)
|
|
{
|
|
if (string.IsNullOrEmpty(chapterTitle)) throw new ArgumentException("Chapter Title cannot be null");
|
|
|
|
if (isSpecial)
|
|
{
|
|
return Tasks.Scanner.Parser.Parser.CleanSpecialTitle(chapterTitle);
|
|
}
|
|
|
|
var hashSpot = withHash ? "#" : string.Empty;
|
|
return libraryType switch
|
|
{
|
|
LibraryType.Book => await _localizationService.Translate(userId, "book-num", chapterTitle),
|
|
LibraryType.Comic => await _localizationService.Translate(userId, "issue-num", hashSpot, chapterTitle),
|
|
LibraryType.Manga => await _localizationService.Translate(userId, "chapter-num", chapterTitle),
|
|
_ => await _localizationService.Translate(userId, "chapter-num", ' ')
|
|
};
|
|
}
|
|
|
|
public async Task<string> FormatChapterTitle(int userId, ChapterDto chapter, LibraryType libraryType, bool withHash = true)
|
|
{
|
|
return await FormatChapterTitle(userId, chapter.IsSpecial, libraryType, chapter.Title, withHash);
|
|
}
|
|
|
|
public async Task<string> FormatChapterTitle(int userId, Chapter chapter, LibraryType libraryType, bool withHash = true)
|
|
{
|
|
return await FormatChapterTitle(userId, chapter.IsSpecial, libraryType, chapter.Title, withHash);
|
|
}
|
|
|
|
public async Task<string> FormatChapterName(int userId, LibraryType libraryType, bool withHash = false)
|
|
{
|
|
var hashSpot = withHash ? "#" : string.Empty;
|
|
return (libraryType switch
|
|
{
|
|
LibraryType.Book => await _localizationService.Translate(userId, "book-num", string.Empty),
|
|
LibraryType.Comic => await _localizationService.Translate(userId, "issue-num", hashSpot, string.Empty),
|
|
LibraryType.Manga => await _localizationService.Translate(userId, "chapter-num", string.Empty),
|
|
_ => await _localizationService.Translate(userId, "chapter-num", ' ')
|
|
}).Trim();
|
|
}
|
|
|
|
/// <summary>
|
|
/// Returns all related series against the passed series Id
|
|
/// </summary>
|
|
/// <param name="userId"></param>
|
|
/// <param name="seriesId"></param>
|
|
/// <returns></returns>
|
|
public async Task<RelatedSeriesDto> GetRelatedSeries(int userId, int seriesId)
|
|
{
|
|
return await _unitOfWork.SeriesRepository.GetRelatedSeries(userId, seriesId);
|
|
}
|
|
|
|
/// <summary>
|
|
/// Update the relations attached to the Series. Does not generate associated Sequel/Prequel pairs on target series.
|
|
/// </summary>
|
|
/// <param name="dto"></param>
|
|
/// <returns></returns>
|
|
public async Task<bool> UpdateRelatedSeries(UpdateRelatedSeriesDto dto)
|
|
{
|
|
var series = await _unitOfWork.SeriesRepository.GetSeriesByIdAsync(dto.SeriesId, SeriesIncludes.Related);
|
|
if (series == null) return false;
|
|
|
|
UpdateRelationForKind(dto.Adaptations, series.Relations.Where(r => r.RelationKind == RelationKind.Adaptation).ToList(), series, RelationKind.Adaptation);
|
|
UpdateRelationForKind(dto.Characters, series.Relations.Where(r => r.RelationKind == RelationKind.Character).ToList(), series, RelationKind.Character);
|
|
UpdateRelationForKind(dto.Contains, series.Relations.Where(r => r.RelationKind == RelationKind.Contains).ToList(), series, RelationKind.Contains);
|
|
UpdateRelationForKind(dto.Others, series.Relations.Where(r => r.RelationKind == RelationKind.Other).ToList(), series, RelationKind.Other);
|
|
UpdateRelationForKind(dto.SideStories, series.Relations.Where(r => r.RelationKind == RelationKind.SideStory).ToList(), series, RelationKind.SideStory);
|
|
UpdateRelationForKind(dto.SpinOffs, series.Relations.Where(r => r.RelationKind == RelationKind.SpinOff).ToList(), series, RelationKind.SpinOff);
|
|
UpdateRelationForKind(dto.AlternativeSettings, series.Relations.Where(r => r.RelationKind == RelationKind.AlternativeSetting).ToList(), series, RelationKind.AlternativeSetting);
|
|
UpdateRelationForKind(dto.AlternativeVersions, series.Relations.Where(r => r.RelationKind == RelationKind.AlternativeVersion).ToList(), series, RelationKind.AlternativeVersion);
|
|
UpdateRelationForKind(dto.Doujinshis, series.Relations.Where(r => r.RelationKind == RelationKind.Doujinshi).ToList(), series, RelationKind.Doujinshi);
|
|
UpdateRelationForKind(dto.Prequels, series.Relations.Where(r => r.RelationKind == RelationKind.Prequel).ToList(), series, RelationKind.Prequel);
|
|
UpdateRelationForKind(dto.Sequels, series.Relations.Where(r => r.RelationKind == RelationKind.Sequel).ToList(), series, RelationKind.Sequel);
|
|
UpdateRelationForKind(dto.Editions, series.Relations.Where(r => r.RelationKind == RelationKind.Edition).ToList(), series, RelationKind.Edition);
|
|
|
|
if (!_unitOfWork.HasChanges()) return true;
|
|
return await _unitOfWork.CommitAsync();
|
|
}
|
|
|
|
|
|
/// <summary>
|
|
/// Applies the provided list to the series. Adds new relations and removes deleted relations.
|
|
/// </summary>
|
|
/// <param name="dtoTargetSeriesIds"></param>
|
|
/// <param name="adaptations"></param>
|
|
/// <param name="series"></param>
|
|
/// <param name="kind"></param>
|
|
private void UpdateRelationForKind(ICollection<int> dtoTargetSeriesIds, IEnumerable<SeriesRelation> adaptations, Series series, RelationKind kind)
|
|
{
|
|
foreach (var adaptation in adaptations.Where(adaptation => !dtoTargetSeriesIds.Contains(adaptation.TargetSeriesId)))
|
|
{
|
|
// If the seriesId isn't in dto, it means we've removed or reclassified
|
|
series.Relations.Remove(adaptation);
|
|
}
|
|
|
|
// At this point, we only have things to add
|
|
foreach (var targetSeriesId in dtoTargetSeriesIds)
|
|
{
|
|
// This ensures we don't allow any duplicates to be added
|
|
if (series.Relations.SingleOrDefault(r =>
|
|
r.RelationKind == kind && r.TargetSeriesId == targetSeriesId) !=
|
|
null) continue;
|
|
|
|
series.Relations.Add(new SeriesRelation()
|
|
{
|
|
Series = series,
|
|
SeriesId = series.Id,
|
|
TargetSeriesId = targetSeriesId,
|
|
RelationKind = kind
|
|
});
|
|
_unitOfWork.SeriesRepository.Update(series);
|
|
}
|
|
}
|
|
|
|
public async Task<NextExpectedChapterDto> GetEstimatedChapterCreationDate(int seriesId, int userId)
|
|
{
|
|
var series = await _unitOfWork.SeriesRepository.GetSeriesByIdAsync(seriesId, SeriesIncludes.Metadata | SeriesIncludes.Library);
|
|
if (series == null) throw new KavitaException(await _localizationService.Translate(userId, "series-doesnt-exist"));
|
|
if (!(await _unitOfWork.UserRepository.HasAccessToSeries(userId, seriesId)))
|
|
{
|
|
throw new UnauthorizedAccessException("user-no-access-library-from-series");
|
|
}
|
|
if (series?.Metadata.PublicationStatus is not (PublicationStatus.OnGoing or PublicationStatus.Ended) || series.Library.Type == LibraryType.Book)
|
|
{
|
|
return _emptyExpectedChapter;
|
|
}
|
|
|
|
var chapters = _unitOfWork.ChapterRepository.GetChaptersForSeries(seriesId)
|
|
.Where(c => !c.IsSpecial)
|
|
.OrderBy(c => c.CreatedUtc)
|
|
.ToList();
|
|
|
|
// Calculate the time differences between consecutive chapters
|
|
var timeDifferences = new List<TimeSpan>();
|
|
DateTime? previousChapterTime = null;
|
|
foreach (var chapter in chapters)
|
|
{
|
|
if (previousChapterTime.HasValue && (chapter.CreatedUtc - previousChapterTime.Value) <= TimeSpan.FromHours(1))
|
|
{
|
|
continue; // Skip this chapter if it's within an hour of the previous one
|
|
}
|
|
|
|
if ((chapter.CreatedUtc - previousChapterTime ?? TimeSpan.Zero) != TimeSpan.Zero)
|
|
{
|
|
timeDifferences.Add(chapter.CreatedUtc - previousChapterTime ?? TimeSpan.Zero);
|
|
}
|
|
|
|
previousChapterTime = chapter.CreatedUtc;
|
|
}
|
|
|
|
|
|
if (timeDifferences.Count < 3)
|
|
{
|
|
return _emptyExpectedChapter;
|
|
}
|
|
|
|
var historicalTimeDifferences = timeDifferences.Select(td => td.TotalDays).ToList();
|
|
|
|
if (historicalTimeDifferences.Count < 3)
|
|
{
|
|
return _emptyExpectedChapter;
|
|
}
|
|
|
|
const double alpha = 0.2; // A smaller alpha will give more weight to recent data, while a larger alpha will smooth the data more.
|
|
var forecastedTimeDifference = ExponentialSmoothing(historicalTimeDifferences, alpha);
|
|
|
|
if (forecastedTimeDifference <= 0)
|
|
{
|
|
return _emptyExpectedChapter;
|
|
}
|
|
|
|
// Calculate the forecast for when the next chapter is expected
|
|
var nextChapterExpected = chapters.Any()
|
|
? chapters.Max(c => c.CreatedUtc) + TimeSpan.FromDays(forecastedTimeDifference)
|
|
: (DateTime?)null;
|
|
|
|
// if (nextChapterExpected != null && nextChapterExpected < DateTime.UtcNow)
|
|
// {
|
|
// nextChapterExpected = DateTime.UtcNow + TimeSpan.FromDays(forecastedTimeDifference);
|
|
// }
|
|
//
|
|
// var averageTimeDifference = timeDifferences
|
|
// .Average(td => td.TotalDays);
|
|
//
|
|
//
|
|
// if (averageTimeDifference == 0)
|
|
// {
|
|
// return _emptyExpectedChapter;
|
|
// }
|
|
//
|
|
//
|
|
// // Calculate the forecast for when the next chapter is expected
|
|
// var nextChapterExpected = chapters.Any()
|
|
// ? chapters.Max(c => c.CreatedUtc) + TimeSpan.FromDays(averageTimeDifference)
|
|
// : (DateTime?) null;
|
|
//
|
|
// if (nextChapterExpected != null && nextChapterExpected < DateTime.UtcNow)
|
|
// {
|
|
// nextChapterExpected = DateTime.UtcNow + TimeSpan.FromDays(averageTimeDifference);
|
|
// }
|
|
|
|
// For number and volume number, we need the highest chapter, not the latest created
|
|
var lastChapter = chapters.MaxBy(c => float.Parse(c.Number))!;
|
|
float.TryParse(lastChapter.Number, NumberStyles.Number, CultureInfo.InvariantCulture,
|
|
out var lastChapterNumber);
|
|
|
|
var lastVolumeNum = chapters.Select(c => c.Volume.Number).Max();
|
|
|
|
var result = new NextExpectedChapterDto()
|
|
{
|
|
ChapterNumber = 0,
|
|
VolumeNumber = 0,
|
|
ExpectedDate = nextChapterExpected,
|
|
Title = string.Empty
|
|
};
|
|
|
|
if (lastChapterNumber > 0)
|
|
{
|
|
result.ChapterNumber = lastChapterNumber + 1;
|
|
result.VolumeNumber = lastChapter.Volume.Number;
|
|
result.Title = series.Library.Type switch
|
|
{
|
|
LibraryType.Manga => await _localizationService.Translate(userId, "chapter-num",
|
|
new object[] {result.ChapterNumber}),
|
|
LibraryType.Comic => await _localizationService.Translate(userId, "issue-num",
|
|
new object[] {"#", result.ChapterNumber}),
|
|
LibraryType.Book => await _localizationService.Translate(userId, "book-num",
|
|
new object[] {result.ChapterNumber}),
|
|
_ => "Chapter " + result.ChapterNumber
|
|
};
|
|
}
|
|
else
|
|
{
|
|
result.VolumeNumber = lastVolumeNum + 1;
|
|
result.Title = await _localizationService.Translate(userId, "volume-num",
|
|
new object[] {result.VolumeNumber});
|
|
}
|
|
|
|
|
|
return result;
|
|
}
|
|
|
|
private double ExponentialSmoothing(IEnumerable<double> data, double alpha)
|
|
{
|
|
double forecast = data.First();
|
|
|
|
foreach (var value in data)
|
|
{
|
|
forecast = alpha * value + (1 - alpha) * forecast;
|
|
}
|
|
|
|
return forecast;
|
|
}
|
|
}
|