mirror of
				https://github.com/Kareadita/Kavita.git
				synced 2025-10-26 08:12:28 -04:00 
			
		
		
		
	* Refactored ResponseCache profiles into consts * Refactored code to use an extension method for getting user library ids. * Started server statistics, added a charting library, and added a table sort column (not finished) * Refactored code and have a fully working example of sortable headers. Still doesn't work with default sorting state, will work on that later. * Implemented file size, but it's too expensive, so commented out. * Added a migration to provide extension and length/size information in the DB to allow for faster stat apis. * Added the ability to force a library scan from library settings. * Refactored some apis to provide more of a file breakdown rather than just file size. * Working on visualization of file breakdown * Fixed the file breakdown visual * Fixed up 2 visualizations * Added back an api for member names, started work on top reads * Hooked up the other library types and username/days. * Preparing to remove top reads and refactor into Top users * Added LibraryId to AppUserProgress to help with complex lookups. * Added the new libraryId hook into some stats methods * Updated api methods to use libraryId for progress * More places where LibraryId is needed * Added some high level server stats * Got a ton done on server stats * Updated default theme (dark) to be the default root variables. This will allow user themes to override just what they want, rather than maintain their own css variables. * Implemented a monster query for top users by reading time. It's very slow and can be cleaned up likely. * Hooked up top reads. Code needs a big refactor. Handing off for Robbie treatment and I'll switch to User stats. * Implemented last 5 recently read series (broken) and added some basic css * Fixed recently read query * Cleanup the css a bit, Robbie we need you * More css love * Cleaned up DTOs that aren't needed anymore * Fixed top readers query * When calculating top readers, don't include read events where nothing is read (0 pages) * Hooked up the date into GetTopUsers * Hooked top readers up with days and refactored and cleaned up componets not used * Fixed up query * Started on a day by day breakdown, but going to take a break from stats. * Added a temp task to run some migration manually for stats to work * Ensure OPDS-PS uses new libraryId for progress reporting * Fixed a code smell * Adding some styling * adding more styles * Removed some debug stuff from user stats * Bump qs from 6.5.2 to 6.5.3 in /UI/Web Bumps [qs](https://github.com/ljharb/qs) from 6.5.2 to 6.5.3. - [Release notes](https://github.com/ljharb/qs/releases) - [Changelog](https://github.com/ljharb/qs/blob/main/CHANGELOG.md) - [Commits](https://github.com/ljharb/qs/compare/v6.5.2...v6.5.3) --- updated-dependencies: - dependency-name: qs dependency-type: indirect ... Signed-off-by: dependabot[bot] <support@github.com> * Tweaked some code for bad data cases * Refactored a chapter lookup to remove un-needed Volume join in 5 places across the code. * API push Signed-off-by: dependabot[bot] <support@github.com> Co-authored-by: Robbie Davis <robbie@therobbiedavis.com> Co-authored-by: dependabot[bot] <49699333+dependabot[bot]@users.noreply.github.com>
		
			
				
	
	
		
			124 lines
		
	
	
		
			4.3 KiB
		
	
	
	
		
			C#
		
	
	
	
	
	
			
		
		
	
	
			124 lines
		
	
	
		
			4.3 KiB
		
	
	
	
		
			C#
		
	
	
	
	
	
| using System.Collections.Generic;
 | |
| using System.Linq;
 | |
| using System.Threading.Tasks;
 | |
| using API.Entities;
 | |
| using API.Entities.Enums;
 | |
| using Microsoft.EntityFrameworkCore;
 | |
| 
 | |
| namespace API.Data.Repositories;
 | |
| 
 | |
| public interface IAppUserProgressRepository
 | |
| {
 | |
|     void Update(AppUserProgress userProgress);
 | |
|     Task<int> CleanupAbandonedChapters();
 | |
|     Task<bool> UserHasProgress(LibraryType libraryType, int userId);
 | |
|     Task<AppUserProgress> GetUserProgressAsync(int chapterId, int userId);
 | |
|     Task<bool> HasAnyProgressOnSeriesAsync(int seriesId, int userId);
 | |
|     /// <summary>
 | |
|     /// This is built exclusively for <see cref="MigrateUserProgressLibraryId"/>
 | |
|     /// </summary>
 | |
|     /// <returns></returns>
 | |
|     Task<AppUserProgress> GetAnyProgress();
 | |
|     Task<IEnumerable<AppUserProgress>> GetUserProgressForSeriesAsync(int seriesId, int userId);
 | |
|     Task<IEnumerable<AppUserProgress>> GetAllProgress();
 | |
| }
 | |
| 
 | |
| public class AppUserProgressRepository : IAppUserProgressRepository
 | |
| {
 | |
|     private readonly DataContext _context;
 | |
| 
 | |
|     public AppUserProgressRepository(DataContext context)
 | |
|     {
 | |
|         _context = context;
 | |
|     }
 | |
| 
 | |
|     public void Update(AppUserProgress userProgress)
 | |
|     {
 | |
|         _context.Entry(userProgress).State = EntityState.Modified;
 | |
|     }
 | |
| 
 | |
|     /// <summary>
 | |
|     /// This will remove any entries that have chapterIds that no longer exists. This will execute the save as well.
 | |
|     /// </summary>
 | |
|     public async Task<int> CleanupAbandonedChapters()
 | |
|     {
 | |
|         var chapterIds = _context.Chapter.Select(c => c.Id);
 | |
| 
 | |
|         var rowsToRemove = await _context.AppUserProgresses
 | |
|             .Where(progress => !chapterIds.Contains(progress.ChapterId))
 | |
|             .ToListAsync();
 | |
| 
 | |
|         var rowsToRemoveBookmarks = await _context.AppUserBookmark
 | |
|             .Where(progress => !chapterIds.Contains(progress.ChapterId))
 | |
|             .ToListAsync();
 | |
| 
 | |
|         var rowsToRemoveReadingLists = await _context.ReadingListItem
 | |
|             .Where(item => !chapterIds.Contains(item.ChapterId))
 | |
|             .ToListAsync();
 | |
| 
 | |
|         _context.RemoveRange(rowsToRemove);
 | |
|         _context.RemoveRange(rowsToRemoveBookmarks);
 | |
|         _context.RemoveRange(rowsToRemoveReadingLists);
 | |
|         return await _context.SaveChangesAsync() > 0 ? rowsToRemove.Count : 0;
 | |
|     }
 | |
| 
 | |
|     /// <summary>
 | |
|     /// Checks if user has any progress against a library of passed type
 | |
|     /// </summary>
 | |
|     /// <param name="libraryType"></param>
 | |
|     /// <param name="userId"></param>
 | |
|     /// <returns></returns>
 | |
|     public async Task<bool> UserHasProgress(LibraryType libraryType, int userId)
 | |
|     {
 | |
|         var seriesIds = await _context.AppUserProgresses
 | |
|             .Where(aup => aup.PagesRead > 0 && aup.AppUserId == userId)
 | |
|             .AsNoTracking()
 | |
|             .Select(aup => aup.SeriesId)
 | |
|             .ToListAsync();
 | |
| 
 | |
|         if (seriesIds.Count == 0) return false;
 | |
| 
 | |
|         return await _context.Series
 | |
|             .Include(s => s.Library)
 | |
|             .Where(s => seriesIds.Contains(s.Id) && s.Library.Type == libraryType)
 | |
|             .AsNoTracking()
 | |
|             .AnyAsync();
 | |
|     }
 | |
| 
 | |
|     public async Task<bool> HasAnyProgressOnSeriesAsync(int seriesId, int userId)
 | |
|     {
 | |
|         return await _context.AppUserProgresses
 | |
|             .AnyAsync(aup => aup.PagesRead > 0 && aup.AppUserId == userId && aup.SeriesId == seriesId);
 | |
|     }
 | |
| 
 | |
|     public async Task<AppUserProgress> GetAnyProgress()
 | |
|     {
 | |
|         return await _context.AppUserProgresses.FirstOrDefaultAsync();
 | |
|     }
 | |
| 
 | |
|     /// <summary>
 | |
|     /// This will return any user progress. This filters out progress rows that have no pages read.
 | |
|     /// </summary>
 | |
|     /// <param name="seriesId"></param>
 | |
|     /// <param name="userId"></param>
 | |
|     /// <returns></returns>
 | |
|     public async Task<IEnumerable<AppUserProgress>> GetUserProgressForSeriesAsync(int seriesId, int userId)
 | |
|     {
 | |
|         return await _context.AppUserProgresses
 | |
|             .Where(p => p.SeriesId == seriesId && p.AppUserId == userId && p.PagesRead > 0)
 | |
|             .ToListAsync();
 | |
|     }
 | |
| 
 | |
|     public async Task<IEnumerable<AppUserProgress>> GetAllProgress()
 | |
|     {
 | |
|         return await _context.AppUserProgresses.ToListAsync();
 | |
|     }
 | |
| 
 | |
|     public async Task<AppUserProgress> GetUserProgressAsync(int chapterId, int userId)
 | |
|     {
 | |
|         return await _context.AppUserProgresses
 | |
|             .Where(p => p.ChapterId == chapterId && p.AppUserId == userId)
 | |
|             .FirstOrDefaultAsync();
 | |
|     }
 | |
| }
 |