mirror of
				https://github.com/Kareadita/Kavita.git
				synced 2025-10-26 16:22:28 -04:00 
			
		
		
		
	* Don't allow Comic libraries to do any scrobbling as there aren't any Comic scrobbling providers yet. * Fixed a bug where if you have multiple libraries pointing the same folder (for whatever reason), the Scan Folder api could be rejected. * Handle if publication from an epub is empty to avoid a bad parse error * Cleaned up some hardcoded default strings. * Fixed up some defaulting code for the cache size. * Changed how moving something back to on deck works after it's been removed. Now any progress will trigger it, as epubs don't have chapters. * Ignore .caltrash, which is a Calibre managed folder, when scanning. * Added the ability to see Volume Last Read Date (or individual chapter) in details drawer. Hover over the clock for the full timestamp.
		
			
				
	
	
		
			268 lines
		
	
	
		
			9.0 KiB
		
	
	
	
		
			C#
		
	
	
	
	
	
			
		
		
	
	
			268 lines
		
	
	
		
			9.0 KiB
		
	
	
	
		
			C#
		
	
	
	
	
	
| using System;
 | |
| using System.Collections.Generic;
 | |
| using System.Linq;
 | |
| using System.Threading.Tasks;
 | |
| using API.DTOs;
 | |
| using API.DTOs.Metadata;
 | |
| using API.DTOs.Reader;
 | |
| using API.Entities;
 | |
| using API.Entities.Enums;
 | |
| using API.Extensions;
 | |
| using API.Extensions.QueryExtensions;
 | |
| using AutoMapper;
 | |
| using AutoMapper.QueryableExtensions;
 | |
| using Microsoft.EntityFrameworkCore;
 | |
| 
 | |
| namespace API.Data.Repositories;
 | |
| 
 | |
| [Flags]
 | |
| public enum ChapterIncludes
 | |
| {
 | |
|     None = 1,
 | |
|     Volumes = 2,
 | |
|     Files = 4,
 | |
| }
 | |
| 
 | |
| public interface IChapterRepository
 | |
| {
 | |
|     void Update(Chapter chapter);
 | |
|     Task<IEnumerable<Chapter>> GetChaptersByIdsAsync(IList<int> chapterIds, ChapterIncludes includes = ChapterIncludes.None);
 | |
|     Task<IChapterInfoDto?> GetChapterInfoDtoAsync(int chapterId);
 | |
|     Task<int> GetChapterTotalPagesAsync(int chapterId);
 | |
|     Task<Chapter?> GetChapterAsync(int chapterId, ChapterIncludes includes = ChapterIncludes.Files);
 | |
|     Task<ChapterDto?> GetChapterDtoAsync(int chapterId, ChapterIncludes includes = ChapterIncludes.Files);
 | |
|     Task<ChapterMetadataDto?> GetChapterMetadataDtoAsync(int chapterId, ChapterIncludes includes = ChapterIncludes.Files);
 | |
|     Task<IList<MangaFile>> GetFilesForChapterAsync(int chapterId);
 | |
|     Task<IList<Chapter>> GetChaptersAsync(int volumeId);
 | |
|     Task<IList<MangaFile>> GetFilesForChaptersAsync(IReadOnlyList<int> chapterIds);
 | |
|     Task<string?> GetChapterCoverImageAsync(int chapterId);
 | |
|     Task<IList<string>> GetAllCoverImagesAsync();
 | |
|     Task<IList<Chapter>> GetAllChaptersWithCoversInDifferentEncoding(EncodeFormat format);
 | |
|     Task<IEnumerable<string>> GetCoverImagesForLockedChaptersAsync();
 | |
|     Task<ChapterDto> AddChapterModifiers(int userId, ChapterDto chapter);
 | |
| }
 | |
| public class ChapterRepository : IChapterRepository
 | |
| {
 | |
|     private readonly DataContext _context;
 | |
|     private readonly IMapper _mapper;
 | |
| 
 | |
|     public ChapterRepository(DataContext context, IMapper mapper)
 | |
|     {
 | |
|         _context = context;
 | |
|         _mapper = mapper;
 | |
|     }
 | |
| 
 | |
|     public void Update(Chapter chapter)
 | |
|     {
 | |
|         _context.Entry(chapter).State = EntityState.Modified;
 | |
|     }
 | |
| 
 | |
|     public async Task<IEnumerable<Chapter>> GetChaptersByIdsAsync(IList<int> chapterIds, ChapterIncludes includes = ChapterIncludes.None)
 | |
|     {
 | |
|         return await _context.Chapter
 | |
|             .Where(c => chapterIds.Contains(c.Id))
 | |
|             .Includes(includes)
 | |
|             .AsSplitQuery()
 | |
|             .ToListAsync();
 | |
|     }
 | |
| 
 | |
|     /// <summary>
 | |
|     /// Populates a partial IChapterInfoDto
 | |
|     /// </summary>
 | |
|     /// <returns></returns>
 | |
|     public async Task<IChapterInfoDto?> GetChapterInfoDtoAsync(int chapterId)
 | |
|     {
 | |
|         var chapterInfo = await _context.Chapter
 | |
|             .Where(c => c.Id == chapterId)
 | |
|             .Join(_context.Volume, c => c.VolumeId, v => v.Id, (chapter, volume) => new
 | |
|             {
 | |
|                 ChapterNumber = chapter.Range,
 | |
|                 VolumeNumber = volume.Name,
 | |
|                 VolumeId = volume.Id,
 | |
|                 chapter.IsSpecial,
 | |
|                 chapter.TitleName,
 | |
|                 volume.SeriesId,
 | |
|                 chapter.Pages,
 | |
|             })
 | |
|             .Join(_context.Series, data => data.SeriesId, series => series.Id, (data, series) => new
 | |
|             {
 | |
|                 data.ChapterNumber,
 | |
|                 data.VolumeNumber,
 | |
|                 data.VolumeId,
 | |
|                 data.IsSpecial,
 | |
|                 data.SeriesId,
 | |
|                 data.Pages,
 | |
|                 data.TitleName,
 | |
|                 SeriesFormat = series.Format,
 | |
|                 SeriesName = series.Name,
 | |
|                 series.LibraryId,
 | |
|                 LibraryType = series.Library.Type
 | |
|             })
 | |
|             .Select(data => new ChapterInfoDto()
 | |
|             {
 | |
|                 ChapterNumber = data.ChapterNumber,
 | |
|                 VolumeNumber = data.VolumeNumber + string.Empty,
 | |
|                 VolumeId = data.VolumeId,
 | |
|                 IsSpecial = data.IsSpecial,
 | |
|                 SeriesId = data.SeriesId,
 | |
|                 SeriesFormat = data.SeriesFormat,
 | |
|                 SeriesName = data.SeriesName,
 | |
|                 LibraryId = data.LibraryId,
 | |
|                 Pages = data.Pages,
 | |
|                 ChapterTitle = data.TitleName,
 | |
|                 LibraryType = data.LibraryType
 | |
|             })
 | |
|             .AsNoTracking()
 | |
|             .AsSplitQuery()
 | |
|             .SingleOrDefaultAsync();
 | |
| 
 | |
|         return chapterInfo;
 | |
|     }
 | |
| 
 | |
|     public Task<int> GetChapterTotalPagesAsync(int chapterId)
 | |
|     {
 | |
|         return _context.Chapter
 | |
|             .Where(c => c.Id == chapterId)
 | |
|             .Select(c => c.Pages)
 | |
|             .FirstOrDefaultAsync();
 | |
|     }
 | |
|     public async Task<ChapterDto?> GetChapterDtoAsync(int chapterId, ChapterIncludes includes = ChapterIncludes.Files)
 | |
|     {
 | |
|         var chapter = await _context.Chapter
 | |
|             .Includes(includes)
 | |
|             .ProjectTo<ChapterDto>(_mapper.ConfigurationProvider)
 | |
|             .AsNoTracking()
 | |
|             .AsSplitQuery()
 | |
|             .FirstOrDefaultAsync(c => c.Id == chapterId);
 | |
| 
 | |
|         return chapter;
 | |
|     }
 | |
| 
 | |
|     public async Task<ChapterMetadataDto?> GetChapterMetadataDtoAsync(int chapterId, ChapterIncludes includes = ChapterIncludes.Files)
 | |
|     {
 | |
|         var chapter = await _context.Chapter
 | |
|             .Includes(includes)
 | |
|             .ProjectTo<ChapterMetadataDto>(_mapper.ConfigurationProvider)
 | |
|             .AsNoTracking()
 | |
|             .AsSplitQuery()
 | |
|             .SingleOrDefaultAsync(c => c.Id == chapterId);
 | |
| 
 | |
|         return chapter;
 | |
|     }
 | |
| 
 | |
|     /// <summary>
 | |
|     /// Returns non-tracked files for a given chapterId
 | |
|     /// </summary>
 | |
|     /// <param name="chapterId"></param>
 | |
|     /// <returns></returns>
 | |
|     public async Task<IList<MangaFile>> GetFilesForChapterAsync(int chapterId)
 | |
|     {
 | |
|         return await _context.MangaFile
 | |
|             .Where(c => chapterId == c.ChapterId)
 | |
|             .AsNoTracking()
 | |
|             .ToListAsync();
 | |
|     }
 | |
| 
 | |
|     /// <summary>
 | |
|     /// Returns a Chapter for an Id. Includes linked <see cref="MangaFile"/>s.
 | |
|     /// </summary>
 | |
|     /// <param name="chapterId"></param>
 | |
|     /// <param name="includes"></param>
 | |
|     /// <returns></returns>
 | |
|     public async Task<Chapter?> GetChapterAsync(int chapterId, ChapterIncludes includes = ChapterIncludes.Files)
 | |
|     {
 | |
|         return await _context.Chapter
 | |
|             .Includes(includes)
 | |
|             .FirstOrDefaultAsync(c => c.Id == chapterId);
 | |
|     }
 | |
| 
 | |
|     /// <summary>
 | |
|     /// Returns Chapters for a volume id.
 | |
|     /// </summary>
 | |
|     /// <param name="volumeId"></param>
 | |
|     /// <returns></returns>
 | |
|     public async Task<IList<Chapter>> GetChaptersAsync(int volumeId)
 | |
|     {
 | |
|         return await _context.Chapter
 | |
|             .Where(c => c.VolumeId == volumeId)
 | |
|             .ToListAsync();
 | |
|     }
 | |
| 
 | |
|     /// <summary>
 | |
|     /// Returns the cover image for a chapter id.
 | |
|     /// </summary>
 | |
|     /// <param name="chapterId"></param>
 | |
|     /// <returns></returns>
 | |
|     public async Task<string?> GetChapterCoverImageAsync(int chapterId)
 | |
|     {
 | |
|         return await _context.Chapter
 | |
|             .Where(c => c.Id == chapterId)
 | |
|             .Select(c => c.CoverImage)
 | |
|             .SingleOrDefaultAsync();
 | |
|     }
 | |
| 
 | |
|     public async Task<IList<string>> GetAllCoverImagesAsync()
 | |
|     {
 | |
|         return (await _context.Chapter
 | |
|             .Select(c => c.CoverImage)
 | |
|             .Where(t => !string.IsNullOrEmpty(t))
 | |
|             .ToListAsync())!;
 | |
|     }
 | |
| 
 | |
|     public async Task<IList<Chapter>> GetAllChaptersWithCoversInDifferentEncoding(EncodeFormat format)
 | |
|     {
 | |
|         var extension = format.GetExtension();
 | |
|         return await _context.Chapter
 | |
|             .Where(c => !string.IsNullOrEmpty(c.CoverImage)  && !c.CoverImage.EndsWith(extension))
 | |
|             .ToListAsync();
 | |
|     }
 | |
| 
 | |
|     /// <summary>
 | |
|     /// Returns cover images for locked chapters
 | |
|     /// </summary>
 | |
|     /// <returns></returns>
 | |
|     public async Task<IEnumerable<string>> GetCoverImagesForLockedChaptersAsync()
 | |
|     {
 | |
|         return (await _context.Chapter
 | |
|             .Where(c => c.CoverImageLocked)
 | |
|             .Select(c => c.CoverImage)
 | |
|             .Where(t => !string.IsNullOrEmpty(t))
 | |
|             .ToListAsync())!;
 | |
|     }
 | |
| 
 | |
|     /// <summary>
 | |
|     /// Returns non-tracked files for a set of <paramref name="chapterIds"/>
 | |
|     /// </summary>
 | |
|     /// <param name="chapterIds">List of chapter Ids</param>
 | |
|     /// <returns></returns>
 | |
|     public async Task<IList<MangaFile>> GetFilesForChaptersAsync(IReadOnlyList<int> chapterIds)
 | |
|     {
 | |
|         return await _context.MangaFile
 | |
|             .Where(c => chapterIds.Contains(c.ChapterId))
 | |
|             .AsNoTracking()
 | |
|             .ToListAsync();
 | |
|     }
 | |
| 
 | |
|     public async Task<ChapterDto> AddChapterModifiers(int userId, ChapterDto chapter)
 | |
|     {
 | |
|         var progress = await _context.AppUserProgresses.Where(x =>
 | |
|                 x.AppUserId == userId && x.ChapterId == chapter.Id)
 | |
|             .AsNoTracking()
 | |
|             .FirstOrDefaultAsync();
 | |
|         if (progress != null)
 | |
|         {
 | |
|             chapter.PagesRead = progress.PagesRead ;
 | |
|             chapter.LastReadingProgressUtc = progress.LastModifiedUtc;
 | |
|             chapter.LastReadingProgress = progress.LastModified;
 | |
|         }
 | |
|         else
 | |
|         {
 | |
|             chapter.PagesRead = 0;
 | |
|             chapter.LastReadingProgressUtc = DateTime.MinValue;
 | |
|             chapter.LastReadingProgress = DateTime.MinValue;
 | |
|         }
 | |
| 
 | |
|         return chapter;
 | |
|     }
 | |
| }
 |