using System; using System.Collections.Generic; using System.Globalization; using System.Linq; using System.Threading.Tasks; using API.Data.Scanner; using API.DTOs; using API.DTOs.CollectionTags; using API.DTOs.Filtering; using API.DTOs.Metadata; using API.DTOs.ReadingLists; using API.DTOs.Search; using API.Entities; using API.Entities.Enums; using API.Entities.Metadata; using API.Extensions; using API.Helpers; using API.Services.Tasks; using AutoMapper; using AutoMapper.QueryableExtensions; using Kavita.Common.Extensions; using Microsoft.EntityFrameworkCore; namespace API.Data.Repositories; internal class RecentlyAddedSeries { public int LibraryId { get; set; } public LibraryType LibraryType { get; set; } public DateTime Created { get; set; } public int SeriesId { get; set; } public string SeriesName { get; set; } public Series Series { get; set; } public IList Chapters { get; set; } // I don't know if I need this public Chapter Chapter { get; set; } // for Alt implementation public MangaFormat Format { get; set; } public int ChapterId { get; set; } // for Alt implementation public int VolumeId { get; set; } // for Alt implementation public string ChapterNumber { get; set; } public string ChapterRange { get; set; } public string ChapterTitle { get; set; } public bool IsSpecial { get; set; } public int VolumeNumber { get; set; } } public interface ISeriesRepository { void Attach(Series series); void Update(Series series); void Remove(Series series); void Remove(IEnumerable series); Task DoesSeriesNameExistInLibrary(string name, MangaFormat format); /// /// Adds user information like progress, ratings, etc /// /// /// /// /// Task> GetSeriesDtoForLibraryIdAsync(int libraryId, int userId, UserParams userParams, FilterDto filter); /// /// Does not add user information like progress, ratings, etc. /// /// /// /// /// /// Task SearchSeries(int userId, bool isAdmin, int[] libraryIds, string searchQuery); Task> GetSeriesForLibraryIdAsync(int libraryId); Task GetSeriesDtoByIdAsync(int seriesId, int userId); Task DeleteSeriesAsync(int seriesId); Task GetSeriesByIdAsync(int seriesId); Task> GetSeriesByIdsAsync(IList seriesIds); Task GetChapterIdsForSeriesAsync(IList seriesIds); Task>> GetChapterIdWithSeriesIdForSeriesAsync(int[] seriesIds); /// /// Used to add Progress/Rating information to series list. /// /// /// /// Task AddSeriesModifiers(int userId, List series); Task GetSeriesCoverImageAsync(int seriesId); Task> GetOnDeck(int userId, int libraryId, UserParams userParams, FilterDto filter); Task> GetRecentlyAdded(int libraryId, int userId, UserParams userParams, FilterDto filter); // NOTE: Probably put this in LibraryRepo Task GetSeriesMetadata(int seriesId); Task> GetSeriesDtoForCollectionAsync(int collectionId, int userId, UserParams userParams); Task> GetFilesForSeries(int seriesId); Task> GetSeriesDtoForIdsAsync(IEnumerable seriesIds, int userId); Task> GetAllCoverImagesAsync(); Task> GetLockedCoverImagesAsync(); Task> GetFullSeriesForLibraryIdAsync(int libraryId, UserParams userParams); Task GetFullSeriesForSeriesIdAsync(int seriesId); Task GetChunkInfo(int libraryId = 0); Task> GetSeriesMetadataForIdsAsync(IEnumerable seriesIds); Task> GetAllAgeRatingsDtosForLibrariesAsync(List libraryIds); Task> GetAllLanguagesForLibrariesAsync(List libraryIds); Task> GetAllPublicationStatusesDtosForLibrariesAsync(List libraryIds); Task> GetRecentlyUpdatedSeries(int userId); Task> GetRecentlyAddedChapters(int userId); } public class SeriesRepository : ISeriesRepository { private readonly DataContext _context; private readonly IMapper _mapper; public SeriesRepository(DataContext context, IMapper mapper) { _context = context; _mapper = mapper; } public void Attach(Series series) { _context.Series.Attach(series); } public void Update(Series series) { _context.Entry(series).State = EntityState.Modified; } public void Remove(Series series) { _context.Series.Remove(series); } public void Remove(IEnumerable series) { _context.Series.RemoveRange(series); } /// /// Returns if a series name and format exists already in a library /// /// Name of series /// Format of series /// public async Task DoesSeriesNameExistInLibrary(string name, MangaFormat format) { var libraries = _context.Series .AsNoTracking() .Where(x => x.Name.Equals(name) && x.Format == format) .Select(s => s.LibraryId); return await _context.Series .AsNoTracking() .Where(s => libraries.Contains(s.LibraryId) && s.Name.Equals(name) && s.Format == format) .CountAsync() > 1; } public async Task> GetSeriesForLibraryIdAsync(int libraryId) { return await _context.Series .Where(s => s.LibraryId == libraryId) .OrderBy(s => s.SortName) .ToListAsync(); } /// /// Used for to /// /// /// public async Task> GetFullSeriesForLibraryIdAsync(int libraryId, UserParams userParams) { var query = _context.Series .Where(s => s.LibraryId == libraryId) .Include(s => s.Metadata) .ThenInclude(m => m.People) .Include(s => s.Metadata) .ThenInclude(m => m.Genres) .Include(s => s.Metadata) .ThenInclude(m => m.Tags) .Include(s => s.Volumes) .ThenInclude(v => v.Chapters) .ThenInclude(cm => cm.People) .Include(s => s.Volumes) .ThenInclude(v => v.Chapters) .ThenInclude(c => c.Genres) .Include(s => s.Volumes) .ThenInclude(v => v.Chapters) .ThenInclude(c => c.Tags) .Include(s => s.Volumes) .ThenInclude(v => v.Chapters) .ThenInclude(c => c.Files) .AsSplitQuery() .OrderBy(s => s.SortName); return await PagedList.CreateAsync(query, userParams.PageNumber, userParams.PageSize); } /// /// This is a heavy call. Returns all entities down to Files and Library and Series Metadata. /// /// /// public async Task GetFullSeriesForSeriesIdAsync(int seriesId) { return await _context.Series .Where(s => s.Id == seriesId) .Include(s => s.Metadata) .ThenInclude(m => m.People) .Include(s => s.Metadata) .ThenInclude(m => m.Genres) .Include(s => s.Library) .Include(s => s.Volumes) .ThenInclude(v => v.Chapters) .ThenInclude(cm => cm.People) .Include(s => s.Volumes) .ThenInclude(v => v.Chapters) .ThenInclude(c => c.Tags) .Include(s => s.Volumes) .ThenInclude(v => v.Chapters) .ThenInclude(c => c.Genres) .Include(s => s.Metadata) .ThenInclude(m => m.Tags) .Include(s => s.Volumes) .ThenInclude(v => v.Chapters) .ThenInclude(c => c.Files) .AsSplitQuery() .SingleOrDefaultAsync(); } public async Task> GetSeriesDtoForLibraryIdAsync(int libraryId, int userId, UserParams userParams, FilterDto filter) { var query = await CreateFilteredSearchQueryable(userId, libraryId, filter); if (filter.SortOptions == null) { query = query.OrderBy(s => s.SortName); } var retSeries = query .ProjectTo(_mapper.ConfigurationProvider) .AsSplitQuery() .AsNoTracking(); return await PagedList.CreateAsync(retSeries, userParams.PageNumber, userParams.PageSize); } private async Task> GetUserLibraries(int libraryId, int userId) { if (libraryId == 0) { return await _context.Library .Include(l => l.AppUsers) .Where(library => library.AppUsers.Any(user => user.Id == userId)) .AsNoTracking() .Select(library => library.Id) .ToListAsync(); } return new List() { libraryId }; } public async Task SearchSeries(int userId, bool isAdmin, int[] libraryIds, string searchQuery) { var result = new SearchResultGroupDto(); var seriesIds = _context.Series .Where(s => libraryIds.Contains(s.LibraryId)) .Select(s => s.Id) .ToList(); result.Series = await _context.Series .Where(s => libraryIds.Contains(s.LibraryId)) .Where(s => EF.Functions.Like(s.Name, $"%{searchQuery}%") || EF.Functions.Like(s.OriginalName, $"%{searchQuery}%") || EF.Functions.Like(s.LocalizedName, $"%{searchQuery}%")) .Include(s => s.Library) .OrderBy(s => s.SortName) .AsNoTracking() .AsSplitQuery() .ProjectTo(_mapper.ConfigurationProvider) .ToListAsync(); result.ReadingLists = await _context.ReadingList .Where(rl => rl.AppUserId == userId || rl.Promoted) .Where(rl => EF.Functions.Like(rl.Title, $"%{searchQuery}%")) .AsSplitQuery() .ProjectTo(_mapper.ConfigurationProvider) .ToListAsync(); result.Collections = await _context.CollectionTag .Where(s => EF.Functions.Like(s.Title, $"%{searchQuery}%") || EF.Functions.Like(s.NormalizedTitle, $"%{searchQuery}%")) .Where(s => s.Promoted || isAdmin) .OrderBy(s => s.Title) .AsNoTracking() .OrderBy(c => c.NormalizedTitle) .ProjectTo(_mapper.ConfigurationProvider) .ToListAsync(); result.Persons = await _context.SeriesMetadata .Where(sm => seriesIds.Contains(sm.SeriesId)) .SelectMany(sm => sm.People.Where(t => EF.Functions.Like(t.Name, $"%{searchQuery}%"))) .AsSplitQuery() .ProjectTo(_mapper.ConfigurationProvider) .ToListAsync(); result.Genres = await _context.SeriesMetadata .Where(sm => seriesIds.Contains(sm.SeriesId)) .SelectMany(sm => sm.Genres.Where(t => EF.Functions.Like(t.Title, $"%{searchQuery}%"))) .AsSplitQuery() .OrderBy(t => t.Title) .Distinct() .ProjectTo(_mapper.ConfigurationProvider) .ToListAsync(); result.Tags = await _context.SeriesMetadata .Where(sm => seriesIds.Contains(sm.SeriesId)) .SelectMany(sm => sm.Tags.Where(t => EF.Functions.Like(t.Title, $"%{searchQuery}%"))) .AsSplitQuery() .OrderBy(t => t.Title) .Distinct() .ProjectTo(_mapper.ConfigurationProvider) .ToListAsync(); return result; } public async Task GetSeriesDtoByIdAsync(int seriesId, int userId) { var series = await _context.Series.Where(x => x.Id == seriesId) .ProjectTo(_mapper.ConfigurationProvider) .SingleAsync(); var seriesList = new List() {series}; await AddSeriesModifiers(userId, seriesList); return seriesList[0]; } public async Task DeleteSeriesAsync(int seriesId) { var series = await _context.Series.Where(s => s.Id == seriesId).SingleOrDefaultAsync(); if (series != null) _context.Series.Remove(series); return await _context.SaveChangesAsync() > 0; } /// /// Returns Volumes, Metadata, and Collection Tags /// /// /// public async Task GetSeriesByIdAsync(int seriesId) { return await _context.Series .Include(s => s.Volumes) .Include(s => s.Metadata) .ThenInclude(m => m.CollectionTags) .Include(s => s.Metadata) .ThenInclude(m => m.Genres) .Include(s => s.Metadata) .ThenInclude(m => m.People) .Where(s => s.Id == seriesId) .AsSplitQuery() .SingleOrDefaultAsync(); } /// /// Returns Volumes, Metadata, and Collection Tags /// /// /// public async Task> GetSeriesByIdsAsync(IList seriesIds) { return await _context.Series .Include(s => s.Volumes) .Include(s => s.Metadata) .ThenInclude(m => m.CollectionTags) .Where(s => seriesIds.Contains(s.Id)) .AsSplitQuery() .ToListAsync(); } public async Task GetChapterIdsForSeriesAsync(IList seriesIds) { var volumes = await _context.Volume .Where(v => seriesIds.Contains(v.SeriesId)) .Include(v => v.Chapters) .ToListAsync(); IList chapterIds = new List(); foreach (var v in volumes) { foreach (var c in v.Chapters) { chapterIds.Add(c.Id); } } return chapterIds.ToArray(); } /// /// This returns a dictionary mapping seriesId -> list of chapters back for each series id passed /// /// /// public async Task>> GetChapterIdWithSeriesIdForSeriesAsync(int[] seriesIds) { var volumes = await _context.Volume .Where(v => seriesIds.Contains(v.SeriesId)) .Include(v => v.Chapters) .ToListAsync(); var seriesChapters = new Dictionary>(); foreach (var v in volumes) { foreach (var c in v.Chapters) { if (!seriesChapters.ContainsKey(v.SeriesId)) { var list = new List(); seriesChapters.Add(v.SeriesId, list); } seriesChapters[v.SeriesId].Add(c.Id); } } return seriesChapters; } public async Task AddSeriesModifiers(int userId, List series) { var userProgress = await _context.AppUserProgresses .Where(p => p.AppUserId == userId && series.Select(s => s.Id).Contains(p.SeriesId)) .ToListAsync(); var userRatings = await _context.AppUserRating .Where(r => r.AppUserId == userId && series.Select(s => s.Id).Contains(r.SeriesId)) .ToListAsync(); foreach (var s in series) { s.PagesRead = userProgress.Where(p => p.SeriesId == s.Id).Sum(p => p.PagesRead); var rating = userRatings.SingleOrDefault(r => r.SeriesId == s.Id); if (rating == null) continue; s.UserRating = rating.Rating; s.UserReview = rating.Review; } } public async Task GetSeriesCoverImageAsync(int seriesId) { return await _context.Series .Where(s => s.Id == seriesId) .Select(s => s.CoverImage) .AsNoTracking() .SingleOrDefaultAsync(); } /// /// Returns a list of Series that were added, ordered by Created desc /// /// /// Library to restrict to, if 0, will apply to all libraries /// Contains pagination information /// Optional filter on query /// public async Task> GetRecentlyAdded(int libraryId, int userId, UserParams userParams, FilterDto filter) { var query = await CreateFilteredSearchQueryable(userId, libraryId, filter); var retSeries = query .OrderByDescending(s => s.Created) .ProjectTo(_mapper.ConfigurationProvider) .AsSplitQuery() .AsNoTracking(); return await PagedList.CreateAsync(retSeries, userParams.PageNumber, userParams.PageSize); } private IList ExtractFilters(int libraryId, int userId, FilterDto filter, ref List userLibraries, out List allPeopleIds, out bool hasPeopleFilter, out bool hasGenresFilter, out bool hasCollectionTagFilter, out bool hasRatingFilter, out bool hasProgressFilter, out IList seriesIds, out bool hasAgeRating, out bool hasTagsFilter, out bool hasLanguageFilter, out bool hasPublicationFilter) { var formats = filter.GetSqlFilter(); if (filter.Libraries.Count > 0) { userLibraries = userLibraries.Where(l => filter.Libraries.Contains(l)).ToList(); } else if (libraryId > 0) { userLibraries = userLibraries.Where(l => l == libraryId).ToList(); } allPeopleIds = new List(); allPeopleIds.AddRange(filter.Writers); allPeopleIds.AddRange(filter.Character); allPeopleIds.AddRange(filter.Colorist); allPeopleIds.AddRange(filter.Editor); allPeopleIds.AddRange(filter.Inker); allPeopleIds.AddRange(filter.Letterer); allPeopleIds.AddRange(filter.Penciller); allPeopleIds.AddRange(filter.Publisher); allPeopleIds.AddRange(filter.CoverArtist); allPeopleIds.AddRange(filter.Translators); hasPeopleFilter = allPeopleIds.Count > 0; hasGenresFilter = filter.Genres.Count > 0; hasCollectionTagFilter = filter.CollectionTags.Count > 0; hasRatingFilter = filter.Rating > 0; hasProgressFilter = !filter.ReadStatus.Read || !filter.ReadStatus.InProgress || !filter.ReadStatus.NotRead; hasAgeRating = filter.AgeRating.Count > 0; hasTagsFilter = filter.Tags.Count > 0; hasLanguageFilter = filter.Languages.Count > 0; hasPublicationFilter = filter.PublicationStatus.Count > 0; bool ProgressComparison(int pagesRead, int totalPages) { var result = false; if (filter.ReadStatus.NotRead) { result = (pagesRead == 0); } if (filter.ReadStatus.Read) { result = result || (pagesRead == totalPages); } if (filter.ReadStatus.InProgress) { result = result || (pagesRead > 0 && pagesRead < totalPages); } return result; } seriesIds = new List(); if (hasProgressFilter) { seriesIds = _context.Series .Include(s => s.Progress) .Select(s => new { Series = s, PagesRead = s.Progress.Where(p => p.AppUserId == userId).Sum(p => p.PagesRead), }) .AsEnumerable() .Where(s => ProgressComparison(s.PagesRead, s.Series.Pages)) .Select(s => s.Series.Id) .ToList(); } return formats; } /// /// Returns Series that the user has some partial progress on. Sorts based on activity. Sort first by User progress, but if a series /// has been updated recently, bump it to the front. /// /// /// Library to restrict to, if 0, will apply to all libraries /// Pagination information /// Optional (default null) filter on query /// public async Task> GetOnDeck(int userId, int libraryId, UserParams userParams, FilterDto filter) { //var allSeriesWithProgress = await _context.AppUserProgresses.Select(p => p.SeriesId).ToListAsync(); //var allChapters = await GetChapterIdsForSeriesAsync(allSeriesWithProgress); var query = (await CreateFilteredSearchQueryable(userId, libraryId, filter)) .Join(_context.AppUserProgresses, s => s.Id, progress => progress.SeriesId, (s, progress) => new { Series = s, PagesRead = _context.AppUserProgresses.Where(s1 => s1.SeriesId == s.Id && s1.AppUserId == userId) .Sum(s1 => s1.PagesRead), progress.AppUserId, LastReadingProgress = _context.AppUserProgresses.Where(p => p.Id == progress.Id && p.AppUserId == userId) .Max(p => p.LastModified), // This is only taking into account chapters that have progress on them, not all chapters in said series LastChapterCreated = _context.Chapter.Where(c => progress.ChapterId == c.Id).Max(c => c.Created) //LastChapterCreated = _context.Chapter.Where(c => allChapters.Contains(c.Id)).Max(c => c.Created) }); // I think I need another Join statement. The problem is the chapters are still limited to progress var retSeries = query.Where(s => s.AppUserId == userId && s.PagesRead > 0 && s.PagesRead < s.Series.Pages) .OrderByDescending(s => s.LastReadingProgress) .ThenByDescending(s => s.LastChapterCreated) .Select(s => s.Series) .ProjectTo(_mapper.ConfigurationProvider) .AsSplitQuery() .AsNoTracking(); // Pagination does not work for this query as when we pull the data back, we get multiple rows of the same series. See controller for pagination code return await retSeries.ToListAsync(); } private async Task> CreateFilteredSearchQueryable(int userId, int libraryId, FilterDto filter) { var userLibraries = await GetUserLibraries(libraryId, userId); var formats = ExtractFilters(libraryId, userId, filter, ref userLibraries, out var allPeopleIds, out var hasPeopleFilter, out var hasGenresFilter, out var hasCollectionTagFilter, out var hasRatingFilter, out var hasProgressFilter, out var seriesIds, out var hasAgeRating, out var hasTagsFilter, out var hasLanguageFilter, out var hasPublicationFilter); var query = _context.Series .Where(s => userLibraries.Contains(s.LibraryId) && formats.Contains(s.Format) && (!hasGenresFilter || s.Metadata.Genres.Any(g => filter.Genres.Contains(g.Id))) && (!hasPeopleFilter || s.Metadata.People.Any(p => allPeopleIds.Contains(p.Id))) && (!hasCollectionTagFilter || s.Metadata.CollectionTags.Any(t => filter.CollectionTags.Contains(t.Id))) && (!hasRatingFilter || s.Ratings.Any(r => r.Rating >= filter.Rating)) && (!hasProgressFilter || seriesIds.Contains(s.Id)) && (!hasAgeRating || filter.AgeRating.Contains(s.Metadata.AgeRating)) && (!hasTagsFilter || s.Metadata.Tags.Any(t => filter.Tags.Contains(t.Id))) && (!hasLanguageFilter || filter.Languages.Contains(s.Metadata.Language)) && (!hasPublicationFilter || filter.PublicationStatus.Contains(s.Metadata.PublicationStatus)) ) .AsNoTracking(); if (filter.SortOptions != null) { if (filter.SortOptions.IsAscending) { if (filter.SortOptions.SortField == SortField.SortName) { query = query.OrderBy(s => s.SortName); } else if (filter.SortOptions.SortField == SortField.CreatedDate) { query = query.OrderBy(s => s.Created); } else if (filter.SortOptions.SortField == SortField.LastModifiedDate) { query = query.OrderBy(s => s.LastModified); } } else { if (filter.SortOptions.SortField == SortField.SortName) { query = query.OrderByDescending(s => s.SortName); } else if (filter.SortOptions.SortField == SortField.CreatedDate) { query = query.OrderByDescending(s => s.Created); } else if (filter.SortOptions.SortField == SortField.LastModifiedDate) { query = query.OrderByDescending(s => s.LastModified); } } } return query; } public async Task GetSeriesMetadata(int seriesId) { var metadataDto = await _context.SeriesMetadata .Where(metadata => metadata.SeriesId == seriesId) .Include(m => m.Genres) .Include(m => m.Tags) .Include(m => m.People) .AsNoTracking() .ProjectTo(_mapper.ConfigurationProvider) .AsSplitQuery() .SingleOrDefaultAsync(); if (metadataDto != null) { metadataDto.CollectionTags = await _context.CollectionTag .Include(t => t.SeriesMetadatas) .Where(t => t.SeriesMetadatas.Select(s => s.SeriesId).Contains(seriesId)) .ProjectTo(_mapper.ConfigurationProvider) .AsNoTracking() .AsSplitQuery() .ToListAsync(); } return metadataDto; } public async Task> GetSeriesDtoForCollectionAsync(int collectionId, int userId, UserParams userParams) { var userLibraries = _context.Library .Include(l => l.AppUsers) .Where(library => library.AppUsers.Any(user => user.Id == userId)) .AsNoTracking() .Select(library => library.Id) .ToList(); var query = _context.CollectionTag .Where(s => s.Id == collectionId) .Include(c => c.SeriesMetadatas) .ThenInclude(m => m.Series) .SelectMany(c => c.SeriesMetadatas.Select(sm => sm.Series).Where(s => userLibraries.Contains(s.LibraryId))) .OrderBy(s => s.LibraryId) .ThenBy(s => s.SortName) .ProjectTo(_mapper.ConfigurationProvider) .AsSplitQuery() .AsNoTracking(); return await PagedList.CreateAsync(query, userParams.PageNumber, userParams.PageSize); } public async Task> GetFilesForSeries(int seriesId) { return await _context.Volume .Where(v => v.SeriesId == seriesId) .Include(v => v.Chapters) .ThenInclude(c => c.Files) .SelectMany(v => v.Chapters.SelectMany(c => c.Files)) .AsNoTracking() .ToListAsync(); } public async Task> GetSeriesDtoForIdsAsync(IEnumerable seriesIds, int userId) { var allowedLibraries = _context.Library .Include(l => l.AppUsers) .Where(library => library.AppUsers.Any(x => x.Id == userId)) .Select(l => l.Id); return await _context.Series .Where(s => seriesIds.Contains(s.Id) && allowedLibraries.Contains(s.LibraryId)) .OrderBy(s => s.SortName) .ProjectTo(_mapper.ConfigurationProvider) .AsNoTracking() .AsSplitQuery() .ToListAsync(); } public async Task> GetAllCoverImagesAsync() { return await _context.Series .Select(s => s.CoverImage) .Where(t => !string.IsNullOrEmpty(t)) .AsNoTracking() .ToListAsync(); } public async Task> GetLockedCoverImagesAsync() { return await _context.Series .Where(s => s.CoverImageLocked && !string.IsNullOrEmpty(s.CoverImage)) .Select(s => s.CoverImage) .AsNoTracking() .ToListAsync(); } /// /// Returns the number of series for a given library (or all libraries if libraryId is 0) /// /// Defaults to 0, library to restrict count to /// private async Task GetSeriesCount(int libraryId = 0) { if (libraryId > 0) { return await _context.Series .Where(s => s.LibraryId == libraryId) .CountAsync(); } return await _context.Series.CountAsync(); } /// /// Returns the number of series that should be processed in parallel to optimize speed and memory. Minimum of 50 /// /// Defaults to 0 meaning no library /// private async Task> GetChunkSize(int libraryId = 0) { var totalSeries = await GetSeriesCount(libraryId); return new Tuple(totalSeries, 50); } public async Task GetChunkInfo(int libraryId = 0) { var (totalSeries, chunkSize) = await GetChunkSize(libraryId); if (totalSeries == 0) return new Chunk() { TotalChunks = 0, TotalSize = 0, ChunkSize = 0 }; var totalChunks = Math.Max((int) Math.Ceiling((totalSeries * 1.0) / chunkSize), 1); return new Chunk() { TotalSize = totalSeries, ChunkSize = chunkSize, TotalChunks = totalChunks }; } public async Task> GetSeriesMetadataForIdsAsync(IEnumerable seriesIds) { return await _context.SeriesMetadata .Where(sm => seriesIds.Contains(sm.SeriesId)) .Include(sm => sm.CollectionTags) .ToListAsync(); } public async Task> GetAllAgeRatingsDtosForLibrariesAsync(List libraryIds) { return await _context.Series .Where(s => libraryIds.Contains(s.LibraryId)) .Select(s => s.Metadata.AgeRating) .Distinct() .Select(s => new AgeRatingDto() { Value = s, Title = s.ToDescription() }) .ToListAsync(); } public async Task> GetAllLanguagesForLibrariesAsync(List libraryIds) { var ret = await _context.Series .Where(s => libraryIds.Contains(s.LibraryId)) .Select(s => s.Metadata.Language) .AsNoTracking() .Distinct() .ToListAsync(); return ret .Where(s => !string.IsNullOrEmpty(s)) .Select(s => new LanguageDto() { Title = CultureInfo.GetCultureInfo(s).DisplayName, IsoCode = s }) .OrderBy(s => s.Title) .ToList(); } public async Task> GetAllPublicationStatusesDtosForLibrariesAsync(List libraryIds) { return await _context.Series .Where(s => libraryIds.Contains(s.LibraryId)) .Select(s => s.Metadata.PublicationStatus) .Distinct() .Select(s => new PublicationStatusDto() { Value = s, Title = s.ToDescription() }) .OrderBy(s => s.Title) .ToListAsync(); } private static string RecentlyAddedItemTitle(RecentlyAddedSeries item) { switch (item.LibraryType) { case LibraryType.Book: return string.Empty; case LibraryType.Comic: return "Issue"; case LibraryType.Manga: default: return "Chapter"; } } /// /// Show all recently added chapters. Provide some mapping for chapter 0 -> Volume 1 /// /// /// public async Task> GetRecentlyAddedChapters(int userId) { var ret = await GetRecentlyAddedChaptersQuery(userId); var items = new List(); foreach (var item in ret) { var dto = new RecentlyAddedItemDto() { LibraryId = item.LibraryId, LibraryType = item.LibraryType, SeriesId = item.SeriesId, SeriesName = item.SeriesName, Created = item.Created, Id = items.Count, Format = item.Format }; // Add title and Volume/Chapter Id var chapterTitle = RecentlyAddedItemTitle(item); string title; if (item.ChapterNumber.Equals(Parser.Parser.DefaultChapter)) { if ((item.VolumeNumber + string.Empty).Equals(Parser.Parser.DefaultChapter)) { title = item.ChapterTitle; } else { title = "Volume " + item.VolumeNumber; } dto.VolumeId = item.VolumeId; } else { title = item.IsSpecial ? item.ChapterRange : $"{chapterTitle} {item.ChapterRange}"; dto.ChapterId = item.ChapterId; } dto.Title = title; items.Add(dto); } return items; } /// /// Return recently updated series, regardless of read progress, and group the number of volume or chapters added. /// /// Used to ensure user has access to libraries /// public async Task> GetRecentlyUpdatedSeries(int userId) { var ret = await GetRecentlyAddedChaptersQuery(userId); var seriesMap = new Dictionary(); var index = 0; foreach (var item in ret) { if (seriesMap.ContainsKey(item.SeriesName)) { seriesMap[item.SeriesName].Count += 1; } else { seriesMap[item.SeriesName] = new GroupedSeriesDto() { LibraryId = item.LibraryId, LibraryType = item.LibraryType, SeriesId = item.SeriesId, SeriesName = item.SeriesName, Created = item.Created, Id = index, Format = item.Format, Count = 1 }; index += 1; } } return seriesMap.Values.ToList(); } private async Task> GetRecentlyAddedChaptersQuery(int userId) { var libraries = await _context.AppUser .Where(u => u.Id == userId) .SelectMany(u => u.Libraries.Select(l => new {LibraryId = l.Id, LibraryType = l.Type})) .ToListAsync(); var libraryIds = libraries.Select(l => l.LibraryId).ToList(); var withinLastWeek = DateTime.Now - TimeSpan.FromDays(12); var ret = await _context.Chapter .Where(c => c.Created >= withinLastWeek) .AsNoTracking() .Include(c => c.Volume) .ThenInclude(v => v.Series) .ThenInclude(s => s.Library) .OrderByDescending(c => c.Created) .Select(c => new RecentlyAddedSeries() { LibraryId = c.Volume.Series.LibraryId, LibraryType = c.Volume.Series.Library.Type, Created = c.Created, SeriesId = c.Volume.Series.Id, SeriesName = c.Volume.Series.Name, Series = c.Volume.Series, VolumeId = c.VolumeId, ChapterId = c.Id, Format = c.Volume.Series.Format, ChapterNumber = c.Number, ChapterRange = c.Range, IsSpecial = c.IsSpecial, VolumeNumber = c.Volume.Number, ChapterTitle = c.Title }) .Take(50) .Where(c => c.Created >= withinLastWeek && libraryIds.Contains(c.LibraryId)) .ToListAsync(); return ret; } }