mirror of
				https://github.com/Kareadita/Kavita.git
				synced 2025-11-03 19:17:05 -05:00 
			
		
		
		
	Stats V3 (#3443)
This commit is contained in:
		
							parent
							
								
									bcfb4a6172
								
							
						
					
					
						commit
						b24b8686f3
					
				@ -129,15 +129,6 @@ public class ServerController : BaseApiController
 | 
			
		||||
        return Ok();
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// Returns non-sensitive information about the current system
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    /// <returns></returns>
 | 
			
		||||
    [HttpGet("server-info")]
 | 
			
		||||
    public async Task<ActionResult<ServerInfoDto>> GetVersion()
 | 
			
		||||
    {
 | 
			
		||||
        return Ok(await _statsService.GetServerInfo());
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// Returns non-sensitive information about the current system
 | 
			
		||||
@ -145,7 +136,7 @@ public class ServerController : BaseApiController
 | 
			
		||||
    /// <remarks>This is just for the UI and is extremely lightweight</remarks>
 | 
			
		||||
    /// <returns></returns>
 | 
			
		||||
    [HttpGet("server-info-slim")]
 | 
			
		||||
    public async Task<ActionResult<ServerInfoDto>> GetSlimVersion()
 | 
			
		||||
    public async Task<ActionResult<ServerInfoSlimDto>> GetSlimVersion()
 | 
			
		||||
    {
 | 
			
		||||
        return Ok(await _statsService.GetServerInfoSlim());
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
@ -1,15 +0,0 @@
 | 
			
		||||
using API.Entities.Enums;
 | 
			
		||||
 | 
			
		||||
namespace API.DTOs.Stats;
 | 
			
		||||
 | 
			
		||||
public class FileFormatDto
 | 
			
		||||
{
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// The extension with the ., in lowercase
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    public required string Extension { get; set; }
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// Format of extension
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    public required MangaFormat Format { get; set; }
 | 
			
		||||
}
 | 
			
		||||
@ -1,184 +0,0 @@
 | 
			
		||||
using System;
 | 
			
		||||
using System.Collections.Generic;
 | 
			
		||||
using API.Entities.Enums;
 | 
			
		||||
 | 
			
		||||
namespace API.DTOs.Stats;
 | 
			
		||||
#nullable enable
 | 
			
		||||
 | 
			
		||||
/// <summary>
 | 
			
		||||
/// Represents information about a Kavita Installation
 | 
			
		||||
/// </summary>
 | 
			
		||||
public class ServerInfoDto
 | 
			
		||||
{
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// Unique Id that represents a unique install
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    public required string InstallId { get; set; }
 | 
			
		||||
    public required string Os { get; set; }
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// If the Kavita install is using Docker
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    public bool IsDocker { get; set; }
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// Version of .NET instance is running
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    public required string DotnetVersion { get; set; }
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// Version of Kavita
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    public required string KavitaVersion { get; set; }
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// Number of Cores on the instance
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    public int NumOfCores { get; set; }
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// The number of libraries on the instance
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    public int NumberOfLibraries { get; set; }
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// Does any user have bookmarks
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    public bool HasBookmarks { get; set; }
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// The site theme the install is using
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    /// <remarks>Introduced in v0.5.2</remarks>
 | 
			
		||||
    public string? ActiveSiteTheme { get; set; }
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// The reading mode the main user has as a preference
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    /// <remarks>Introduced in v0.5.2</remarks>
 | 
			
		||||
    public ReaderMode MangaReaderMode { get; set; }
 | 
			
		||||
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// Number of users on the install
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    /// <remarks>Introduced in v0.5.2</remarks>
 | 
			
		||||
    public int NumberOfUsers { get; set; }
 | 
			
		||||
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// Number of collections on the install
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    /// <remarks>Introduced in v0.5.2</remarks>
 | 
			
		||||
    public int NumberOfCollections { get; set; }
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// Number of reading lists on the install (Sum of all users)
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    /// <remarks>Introduced in v0.5.2</remarks>
 | 
			
		||||
    public int NumberOfReadingLists { get; set; }
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// Is OPDS enabled
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    /// <remarks>Introduced in v0.5.2</remarks>
 | 
			
		||||
    public bool OPDSEnabled { get; set; }
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// Total number of files in the instance
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    /// <remarks>Introduced in v0.5.2</remarks>
 | 
			
		||||
    public int TotalFiles { get; set; }
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// Total number of Genres in the instance
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    /// <remarks>Introduced in v0.5.4</remarks>
 | 
			
		||||
    public int TotalGenres { get; set; }
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// Total number of People in the instance
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    /// <remarks>Introduced in v0.5.4</remarks>
 | 
			
		||||
    public int TotalPeople { get; set; }
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// Number of users on this instance using Card Layout
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    /// <remarks>Introduced in v0.5.4</remarks>
 | 
			
		||||
    public int UsersOnCardLayout { get; set; }
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// Number of users on this instance using List Layout
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    /// <remarks>Introduced in v0.5.4</remarks>
 | 
			
		||||
    public int UsersOnListLayout { get; set; }
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// Max number of Series for any library on the instance
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    /// <remarks>Introduced in v0.5.4</remarks>
 | 
			
		||||
    public int MaxSeriesInALibrary { get; set; }
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// Max number of Volumes for any library on the instance
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    /// <remarks>Introduced in v0.5.4</remarks>
 | 
			
		||||
    public int MaxVolumesInASeries { get; set; }
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// Max number of Chapters for any library on the instance
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    /// <remarks>Introduced in v0.5.4</remarks>
 | 
			
		||||
    public int MaxChaptersInASeries { get; set; }
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// Does this instance have relationships setup between series
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    /// <remarks>Introduced in v0.5.4</remarks>
 | 
			
		||||
    public bool UsingSeriesRelationships { get; set; }
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// A list of background colors set on the instance
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    /// <remarks>Introduced in v0.6.0</remarks>
 | 
			
		||||
    public required IEnumerable<string> MangaReaderBackgroundColors { get; set; }
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// A list of Page Split defaults being used on the instance
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    /// <remarks>Introduced in v0.6.0</remarks>
 | 
			
		||||
    public required IEnumerable<PageSplitOption> MangaReaderPageSplittingModes { get; set; }
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// A list of Layout Mode defaults being used on the instance
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    /// <remarks>Introduced in v0.6.0</remarks>
 | 
			
		||||
    public required IEnumerable<LayoutMode> MangaReaderLayoutModes { get; set; }
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// A list of file formats existing in the instance
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    /// <remarks>Introduced in v0.6.0</remarks>
 | 
			
		||||
    public required IEnumerable<FileFormatDto> FileFormats { get; set; }
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// If there is at least one user that is using an age restricted profile on the instance
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    /// <remarks>Introduced in v0.6.0</remarks>
 | 
			
		||||
    public bool UsingRestrictedProfiles { get; set; }
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// Number of users using the Emulate Comic Book setting
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    /// <remarks>Introduced in v0.7.0</remarks>
 | 
			
		||||
    public int UsersWithEmulateComicBook { get; set; }
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// Percent (0.0-1.0) of libraries with folder watching enabled
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    /// <remarks>Introduced in v0.7.0</remarks>
 | 
			
		||||
    public float PercentOfLibrariesWithFolderWatchingEnabled { get; set; }
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// Percent (0.0-1.0) of libraries included in Search
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    /// <remarks>Introduced in v0.7.0</remarks>
 | 
			
		||||
    public float PercentOfLibrariesIncludedInSearch { get; set; }
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// Percent (0.0-1.0) of libraries included in Recommended
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    /// <remarks>Introduced in v0.7.0</remarks>
 | 
			
		||||
    public float PercentOfLibrariesIncludedInRecommended { get; set; }
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// Percent (0.0-1.0) of libraries included in Dashboard
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    /// <remarks>Introduced in v0.7.0</remarks>
 | 
			
		||||
    public float PercentOfLibrariesIncludedInDashboard { get; set; }
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// Total reading hours of all users
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    /// <remarks>Introduced in v0.7.0</remarks>
 | 
			
		||||
    public long TotalReadingHours { get; set; }
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// The encoding the server is using to save media
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    /// <remarks>Added in v0.7.3</remarks>
 | 
			
		||||
    public EncodeFormat EncodeMediaAs { get; set; }
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// The last user reading progress on the server (in UTC)
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    /// <remarks>Added in v0.7.4</remarks>
 | 
			
		||||
    public DateTime LastReadTime { get; set; }
 | 
			
		||||
}
 | 
			
		||||
							
								
								
									
										39
									
								
								API/DTOs/Stats/V3/LibraryStatV3.cs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										39
									
								
								API/DTOs/Stats/V3/LibraryStatV3.cs
									
									
									
									
									
										Normal file
									
								
							@ -0,0 +1,39 @@
 | 
			
		||||
using System;
 | 
			
		||||
using System.Collections.Generic;
 | 
			
		||||
using API.Entities.Enums;
 | 
			
		||||
 | 
			
		||||
namespace API.DTOs.Stats.V3;
 | 
			
		||||
 | 
			
		||||
public class LibraryStatV3
 | 
			
		||||
{
 | 
			
		||||
    public bool IncludeInDashboard { get; set; }
 | 
			
		||||
    public bool IncludeInSearch { get; set; }
 | 
			
		||||
    public bool UsingFolderWatching { get; set; }
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// Are any exclude patterns setup
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    public bool UsingExcludePatterns { get; set; }
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// Will this library create collections from ComicInfo
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    public bool CreateCollectionsFromMetadata { get; set; }
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// Will this library create reading lists from ComicInfo
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    public bool CreateReadingListsFromMetadata { get; set; }
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// Type of the Library
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    public LibraryType LibraryType { get; set; }
 | 
			
		||||
    public ICollection<FileTypeGroup> FileTypes { get; set; }
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// Last time library was fully scanned
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    public DateTime LastScanned { get; set; }
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// Number of folders the library has
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    public int NumberOfFolders { get; set; }
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
}
 | 
			
		||||
							
								
								
									
										12
									
								
								API/DTOs/Stats/V3/RelationshipStatV3.cs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										12
									
								
								API/DTOs/Stats/V3/RelationshipStatV3.cs
									
									
									
									
									
										Normal file
									
								
							@ -0,0 +1,12 @@
 | 
			
		||||
using API.Entities.Enums;
 | 
			
		||||
 | 
			
		||||
namespace API.DTOs.Stats.V3;
 | 
			
		||||
 | 
			
		||||
/// <summary>
 | 
			
		||||
/// KavitaStats - Information about Series Relationships
 | 
			
		||||
/// </summary>
 | 
			
		||||
public class RelationshipStatV3
 | 
			
		||||
{
 | 
			
		||||
    public int Count { get; set; }
 | 
			
		||||
    public RelationKind Relationship { get; set; }
 | 
			
		||||
}
 | 
			
		||||
							
								
								
									
										141
									
								
								API/DTOs/Stats/V3/ServerInfoV3Dto.cs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										141
									
								
								API/DTOs/Stats/V3/ServerInfoV3Dto.cs
									
									
									
									
									
										Normal file
									
								
							@ -0,0 +1,141 @@
 | 
			
		||||
using System;
 | 
			
		||||
using System.Collections.Generic;
 | 
			
		||||
using API.Entities.Enums;
 | 
			
		||||
 | 
			
		||||
namespace API.DTOs.Stats.V3;
 | 
			
		||||
 | 
			
		||||
/// <summary>
 | 
			
		||||
/// Represents information about a Kavita Installation for Kavita Stats v3 API
 | 
			
		||||
/// </summary>
 | 
			
		||||
public class ServerInfoV3Dto
 | 
			
		||||
{
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// Unique Id that represents a unique install
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    public required string InstallId { get; set; }
 | 
			
		||||
    public required string Os { get; set; }
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// If the Kavita install is using Docker
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    public bool IsDocker { get; set; }
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// Version of .NET instance is running
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    public required string DotnetVersion { get; set; }
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// Version of Kavita
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    public required string KavitaVersion { get; set; }
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// Version of Kavita on Installation
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    public required string InitialKavitaVersion { get; set; }
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// Date of first Installation
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    public DateTime InitialInstallDate { get; set; }
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// Number of Cores on the instance
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    public int NumOfCores { get; set; }
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// OS locale on the instance
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    public string OsLocale { get; set; }
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// Milliseconds to open a random archive (zip/cbz) for reading
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    public long TimeToOpeCbzMs { get; set; }
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// Number of pages for said archive (zip/cbz)
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    public long TimeToOpenCbzPages { get; set; }
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// Milliseconds to get a response from KavitaStats API
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    /// <remarks>This pings a health check and does not capture any IP Information</remarks>
 | 
			
		||||
    public long TimeToPingKavitaStatsApi { get; set; }
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
    #region Media
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// Number of collections on the install
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    public int NumberOfCollections { get; set; }
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// Number of reading lists on the install (Sum of all users)
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    public int NumberOfReadingLists { get; set; }
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// Total number of files in the instance
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    public int TotalFiles { get; set; }
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// Total number of Genres in the instance
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    public int TotalGenres { get; set; }
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// Total number of Series in the instance
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    public int TotalSeries { get; set; }
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// Total number of Libraries in the instance
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    public int TotalLibraries { get; set; }
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// Total number of People in the instance
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    public int TotalPeople { get; set; }
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// Max number of Series for any library on the instance
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    public int MaxSeriesInALibrary { get; set; }
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// Max number of Volumes for any library on the instance
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    public int MaxVolumesInASeries { get; set; }
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// Max number of Chapters for any library on the instance
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    public int MaxChaptersInASeries { get; set; }
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// Everything about the Libraries on the instance
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    public IList<LibraryStatV3> Libraries { get; set; }
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// Everything around Series Relationships between series
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    public IList<RelationshipStatV3> Relationships { get; set; }
 | 
			
		||||
    #endregion
 | 
			
		||||
 | 
			
		||||
    #region Server
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// Is OPDS enabled
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    public bool OpdsEnabled { get; set; }
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// The encoding the server is using to save media
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    public EncodeFormat EncodeMediaAs { get; set; }
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// The last user reading progress on the server (in UTC)
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    public DateTime LastReadTime { get; set; }
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// Is this server using Kavita+
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    public bool ActiveKavitaPlusSubscription { get; set; }
 | 
			
		||||
    #endregion
 | 
			
		||||
 | 
			
		||||
    #region Users
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// If there is at least one user that is using an age restricted profile on the instance
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    /// <remarks>Introduced in v0.6.0</remarks>
 | 
			
		||||
    public bool UsingRestrictedProfiles { get; set; }
 | 
			
		||||
 | 
			
		||||
    public IList<UserStatV3> Users { get; set; }
 | 
			
		||||
 | 
			
		||||
    #endregion
 | 
			
		||||
}
 | 
			
		||||
							
								
								
									
										81
									
								
								API/DTOs/Stats/V3/UserStatV3.cs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										81
									
								
								API/DTOs/Stats/V3/UserStatV3.cs
									
									
									
									
									
										Normal file
									
								
							@ -0,0 +1,81 @@
 | 
			
		||||
using System;
 | 
			
		||||
using System.Collections.Generic;
 | 
			
		||||
using API.Data.Misc;
 | 
			
		||||
using API.Entities.Enums.Device;
 | 
			
		||||
 | 
			
		||||
namespace API.DTOs.Stats.V3;
 | 
			
		||||
 | 
			
		||||
public class UserStatV3
 | 
			
		||||
{
 | 
			
		||||
    public AgeRestriction AgeRestriction { get; set; }
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// The last reading progress on the server (in UTC)
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    public DateTime LastReadTime { get; set; }
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// The last login on the server (in UTC)
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    public DateTime LastLogin { get; set; }
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// Has the user gone through email confirmation
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    public bool IsEmailConfirmed { get; set; }
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// Is the Email a valid address
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    public bool HasValidEmail { get; set; }
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// Float between 0-1 to showcase how much of the libraries a user has access to
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    public float PercentageOfLibrariesHasAccess { get; set; }
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// Number of reading lists this user created
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    public int ReadingListsCreatedCount { get; set; }
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// Number of collections this user created
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    public int CollectionsCreatedCount { get; set; }
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// Number of series in want to read for this user
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    public int WantToReadSeriesCount { get; set; }
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// Active locale for the user
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    public string Locale { get; set; }
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// Active Theme (name)
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    public string ActiveTheme { get; set; }
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// Number of series with Bookmarks created
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    public int SeriesBookmarksCreatedCount { get; set; }
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// Kavita+ only - Has an AniList Token set
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    public bool HasAniListToken { get; set; }
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// Kavita+ only - Has a MAL Token set
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    public bool HasMALToken { get; set; }
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// Number of Smart Filters a user has created
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    public int SmartFilterCreatedCount { get; set; }
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// Is the user sharing reviews
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    public bool IsSharingReviews { get; set; }
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// The number of devices setup and their platforms
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    public ICollection<DevicePlatform> DevicePlatforms { get; set; }
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// Roles for this user
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    public ICollection<string> Roles { get; set; }
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
}
 | 
			
		||||
@ -39,7 +39,7 @@ public interface ILibraryRepository
 | 
			
		||||
    Task<bool> LibraryExists(string libraryName);
 | 
			
		||||
    Task<Library?> GetLibraryForIdAsync(int libraryId, LibraryIncludes includes = LibraryIncludes.None);
 | 
			
		||||
    IEnumerable<LibraryDto> GetLibraryDtosForUsernameAsync(string userName);
 | 
			
		||||
    Task<IEnumerable<Library>> GetLibrariesAsync(LibraryIncludes includes = LibraryIncludes.None);
 | 
			
		||||
    Task<IEnumerable<Library>> GetLibrariesAsync(LibraryIncludes includes = LibraryIncludes.None, bool track = true);
 | 
			
		||||
    Task<IEnumerable<Library>> GetLibrariesForUserIdAsync(int userId);
 | 
			
		||||
    IEnumerable<int> GetLibraryIdsForUserIdAsync(int userId, QueryContext queryContext = QueryContext.None);
 | 
			
		||||
    Task<LibraryType> GetLibraryTypeAsync(int libraryId);
 | 
			
		||||
@ -104,13 +104,16 @@ public class LibraryRepository : ILibraryRepository
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    /// <param name="includes"></param>
 | 
			
		||||
    /// <returns></returns>
 | 
			
		||||
    public async Task<IEnumerable<Library>> GetLibrariesAsync(LibraryIncludes includes = LibraryIncludes.None)
 | 
			
		||||
    public async Task<IEnumerable<Library>> GetLibrariesAsync(LibraryIncludes includes = LibraryIncludes.None, bool track = true)
 | 
			
		||||
    {
 | 
			
		||||
        return await _context.Library
 | 
			
		||||
        var query = _context.Library
 | 
			
		||||
            .Include(l => l.AppUsers)
 | 
			
		||||
            .Includes(includes)
 | 
			
		||||
            .AsSplitQuery()
 | 
			
		||||
            .ToListAsync();
 | 
			
		||||
            .AsSplitQuery();
 | 
			
		||||
 | 
			
		||||
        if (track) return await query.ToListAsync();
 | 
			
		||||
 | 
			
		||||
        return await query.AsNoTracking().ToListAsync();
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    /// <summary>
 | 
			
		||||
 | 
			
		||||
@ -164,6 +164,7 @@ public interface ISeriesRepository
 | 
			
		||||
    Task ClearOnDeckRemoval(int seriesId, int userId);
 | 
			
		||||
    Task<PagedList<SeriesDto>> GetSeriesDtoForLibraryIdV2Async(int userId, UserParams userParams, FilterV2Dto filterDto, QueryContext queryContext = QueryContext.None);
 | 
			
		||||
    Task<PlusSeriesDto?> GetPlusSeriesDto(int seriesId);
 | 
			
		||||
    Task<int> GetCountAsync();
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
public class SeriesRepository : ISeriesRepository
 | 
			
		||||
@ -726,6 +727,10 @@ public class SeriesRepository : ISeriesRepository
 | 
			
		||||
            .FirstOrDefaultAsync();
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    public async Task<int> GetCountAsync()
 | 
			
		||||
    {
 | 
			
		||||
        return await _context.Series.CountAsync();
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    public async Task AddSeriesModifiers(int userId, IList<SeriesDto> series)
 | 
			
		||||
    {
 | 
			
		||||
 | 
			
		||||
@ -78,7 +78,7 @@ public interface IUserRepository
 | 
			
		||||
    Task<IEnumerable<AppUserPreferences>> GetAllPreferencesByThemeAsync(int themeId);
 | 
			
		||||
    Task<bool> HasAccessToLibrary(int libraryId, int userId);
 | 
			
		||||
    Task<bool> HasAccessToSeries(int userId, int seriesId);
 | 
			
		||||
    Task<IEnumerable<AppUser>> GetAllUsersAsync(AppUserIncludes includeFlags = AppUserIncludes.None);
 | 
			
		||||
    Task<IEnumerable<AppUser>> GetAllUsersAsync(AppUserIncludes includeFlags = AppUserIncludes.None, bool track = true);
 | 
			
		||||
    Task<AppUser?> GetUserByConfirmationToken(string token);
 | 
			
		||||
    Task<AppUser> GetDefaultAdminUser(AppUserIncludes includes = AppUserIncludes.None);
 | 
			
		||||
    Task<IEnumerable<AppUserRating>> GetSeriesWithRatings(int userId);
 | 
			
		||||
@ -283,10 +283,17 @@ public class UserRepository : IUserRepository
 | 
			
		||||
            .AnyAsync(s => s.Id == seriesId);
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    public async Task<IEnumerable<AppUser>> GetAllUsersAsync(AppUserIncludes includeFlags = AppUserIncludes.None)
 | 
			
		||||
    public async Task<IEnumerable<AppUser>> GetAllUsersAsync(AppUserIncludes includeFlags = AppUserIncludes.None, bool track = true)
 | 
			
		||||
    {
 | 
			
		||||
        return await _context.AppUser
 | 
			
		||||
            .Includes(includeFlags)
 | 
			
		||||
        var query = _context.AppUser
 | 
			
		||||
            .Includes(includeFlags);
 | 
			
		||||
        if (track)
 | 
			
		||||
        {
 | 
			
		||||
            return await query.ToListAsync();
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        return await query
 | 
			
		||||
            .AsNoTracking()
 | 
			
		||||
            .ToListAsync();
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
@ -15,5 +15,4 @@ public enum FileTypeGroup
 | 
			
		||||
    Pdf = 3,
 | 
			
		||||
    [Description("Images")]
 | 
			
		||||
    Images = 4
 | 
			
		||||
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
@ -44,9 +44,6 @@ public class Library : IEntityDate, IHasCoverImage
 | 
			
		||||
    public bool AllowScrobbling { get; set; } = true;
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
    public DateTime Created { get; set; }
 | 
			
		||||
    public DateTime LastModified { get; set; }
 | 
			
		||||
    public DateTime CreatedUtc { get; set; }
 | 
			
		||||
 | 
			
		||||
@ -1,19 +1,24 @@
 | 
			
		||||
using System;
 | 
			
		||||
using System.Collections.Generic;
 | 
			
		||||
using System.IO;
 | 
			
		||||
using System.Diagnostics;
 | 
			
		||||
using System.Globalization;
 | 
			
		||||
using System.Linq;
 | 
			
		||||
using System.Net.Http;
 | 
			
		||||
using System.Runtime.InteropServices;
 | 
			
		||||
using System.Threading.Tasks;
 | 
			
		||||
using API.Data;
 | 
			
		||||
using API.Data.Misc;
 | 
			
		||||
using API.Data.Repositories;
 | 
			
		||||
using API.DTOs.Stats;
 | 
			
		||||
using API.DTOs.Stats.V3;
 | 
			
		||||
using API.Entities;
 | 
			
		||||
using API.Entities.Enums;
 | 
			
		||||
using API.Entities.Enums.UserPreferences;
 | 
			
		||||
using API.Services.Plus;
 | 
			
		||||
using Flurl.Http;
 | 
			
		||||
using Kavita.Common.EnvironmentInfo;
 | 
			
		||||
using Kavita.Common.Helpers;
 | 
			
		||||
using Microsoft.AspNetCore.Http;
 | 
			
		||||
using Microsoft.AspNetCore.Identity;
 | 
			
		||||
using Microsoft.EntityFrameworkCore;
 | 
			
		||||
using Microsoft.Extensions.Logging;
 | 
			
		||||
 | 
			
		||||
@ -24,7 +29,6 @@ namespace API.Services.Tasks;
 | 
			
		||||
public interface IStatsService
 | 
			
		||||
{
 | 
			
		||||
    Task Send();
 | 
			
		||||
    Task<ServerInfoDto> GetServerInfo();
 | 
			
		||||
    Task<ServerInfoSlimDto> GetServerInfoSlim();
 | 
			
		||||
    Task SendCancellation();
 | 
			
		||||
}
 | 
			
		||||
@ -36,15 +40,24 @@ public class StatsService : IStatsService
 | 
			
		||||
    private readonly ILogger<StatsService> _logger;
 | 
			
		||||
    private readonly IUnitOfWork _unitOfWork;
 | 
			
		||||
    private readonly DataContext _context;
 | 
			
		||||
    private readonly IStatisticService _statisticService;
 | 
			
		||||
    private readonly ILicenseService _licenseService;
 | 
			
		||||
    private readonly UserManager<AppUser> _userManager;
 | 
			
		||||
    private readonly IEmailService _emailService;
 | 
			
		||||
    private readonly ICacheService _cacheService;
 | 
			
		||||
    private const string ApiUrl = "https://stats.kavitareader.com";
 | 
			
		||||
    private const string ApiKey = "MsnvA2DfQqxSK5jh"; // It's not important this is public, just a way to keep bots from hitting the API willy-nilly
 | 
			
		||||
 | 
			
		||||
    public StatsService(ILogger<StatsService> logger, IUnitOfWork unitOfWork, DataContext context, IStatisticService statisticService)
 | 
			
		||||
    public StatsService(ILogger<StatsService> logger, IUnitOfWork unitOfWork, DataContext context,
 | 
			
		||||
        ILicenseService licenseService, UserManager<AppUser> userManager, IEmailService emailService,
 | 
			
		||||
        ICacheService cacheService)
 | 
			
		||||
    {
 | 
			
		||||
        _logger = logger;
 | 
			
		||||
        _unitOfWork = unitOfWork;
 | 
			
		||||
        _context = context;
 | 
			
		||||
        _statisticService = statisticService;
 | 
			
		||||
        _licenseService = licenseService;
 | 
			
		||||
        _userManager = userManager;
 | 
			
		||||
        _emailService = emailService;
 | 
			
		||||
        _cacheService = cacheService;
 | 
			
		||||
 | 
			
		||||
        FlurlHttp.ConfigureClient(ApiUrl, cli =>
 | 
			
		||||
            cli.Settings.HttpClientFactory = new UntrustedCertClientFactory());
 | 
			
		||||
@ -52,7 +65,7 @@ public class StatsService : IStatsService
 | 
			
		||||
 | 
			
		||||
    /// <summary>
 | 
			
		||||
    /// Due to all instances firing this at the same time, we can DDOS our server. This task when fired will schedule the task to be run
 | 
			
		||||
    /// randomly over a 6 hour spread
 | 
			
		||||
    /// randomly over a six-hour spread
 | 
			
		||||
    /// </summary>
 | 
			
		||||
    public async Task Send()
 | 
			
		||||
    {
 | 
			
		||||
@ -71,21 +84,24 @@ public class StatsService : IStatsService
 | 
			
		||||
    // ReSharper disable once MemberCanBePrivate.Global
 | 
			
		||||
    public async Task SendData()
 | 
			
		||||
    {
 | 
			
		||||
        var data = await GetServerInfo();
 | 
			
		||||
        var sw = Stopwatch.StartNew();
 | 
			
		||||
        var data = await GetStatV3Payload();
 | 
			
		||||
        _logger.LogDebug("Collecting stats took {Time} ms", sw.ElapsedMilliseconds);
 | 
			
		||||
        sw.Stop();
 | 
			
		||||
        await SendDataToStatsServer(data);
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
    private async Task SendDataToStatsServer(ServerInfoDto data)
 | 
			
		||||
    private async Task SendDataToStatsServer(ServerInfoV3Dto data)
 | 
			
		||||
    {
 | 
			
		||||
        var responseContent = string.Empty;
 | 
			
		||||
 | 
			
		||||
        try
 | 
			
		||||
        {
 | 
			
		||||
            var response = await (ApiUrl + "/api/v2/stats")
 | 
			
		||||
            var response = await (ApiUrl + "/api/v3/stats")
 | 
			
		||||
                .WithHeader("Accept", "application/json")
 | 
			
		||||
                .WithHeader("User-Agent", "Kavita")
 | 
			
		||||
                .WithHeader("x-api-key", "MsnvA2DfQqxSK5jh")
 | 
			
		||||
                .WithHeader("x-api-key", ApiKey)
 | 
			
		||||
                .WithHeader("x-kavita-version", BuildInfo.Version)
 | 
			
		||||
                .WithHeader("Content-Type", "application/json")
 | 
			
		||||
                .WithTimeout(TimeSpan.FromSeconds(30))
 | 
			
		||||
@ -112,67 +128,6 @@ public class StatsService : IStatsService
 | 
			
		||||
        }
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    public async Task<ServerInfoDto> GetServerInfo()
 | 
			
		||||
    {
 | 
			
		||||
        var serverSettings = await _unitOfWork.SettingsRepository.GetSettingsDtoAsync();
 | 
			
		||||
 | 
			
		||||
        var serverInfo = new ServerInfoDto
 | 
			
		||||
        {
 | 
			
		||||
            InstallId = serverSettings.InstallId,
 | 
			
		||||
            Os = RuntimeInformation.OSDescription,
 | 
			
		||||
            KavitaVersion = serverSettings.InstallVersion,
 | 
			
		||||
            DotnetVersion = Environment.Version.ToString(),
 | 
			
		||||
            IsDocker = OsInfo.IsDocker,
 | 
			
		||||
            NumOfCores = Math.Max(Environment.ProcessorCount, 1),
 | 
			
		||||
            UsersWithEmulateComicBook = await _context.AppUserPreferences.CountAsync(p => p.EmulateBook),
 | 
			
		||||
            TotalReadingHours = await _statisticService.TimeSpentReadingForUsersAsync(ArraySegment<int>.Empty, ArraySegment<int>.Empty),
 | 
			
		||||
 | 
			
		||||
            PercentOfLibrariesWithFolderWatchingEnabled = await GetPercentageOfLibrariesWithFolderWatchingEnabled(),
 | 
			
		||||
            PercentOfLibrariesIncludedInRecommended = await GetPercentageOfLibrariesIncludedInRecommended(),
 | 
			
		||||
            PercentOfLibrariesIncludedInDashboard = await GetPercentageOfLibrariesIncludedInDashboard(),
 | 
			
		||||
            PercentOfLibrariesIncludedInSearch = await GetPercentageOfLibrariesIncludedInSearch(),
 | 
			
		||||
 | 
			
		||||
            HasBookmarks = (await _unitOfWork.UserRepository.GetAllBookmarksAsync()).Any(),
 | 
			
		||||
            NumberOfLibraries = (await _unitOfWork.LibraryRepository.GetLibrariesAsync()).Count(),
 | 
			
		||||
            NumberOfCollections = (await _unitOfWork.CollectionTagRepository.GetAllCollectionsAsync()).Count(),
 | 
			
		||||
            NumberOfReadingLists = await _unitOfWork.ReadingListRepository.Count(),
 | 
			
		||||
            OPDSEnabled = serverSettings.EnableOpds,
 | 
			
		||||
            NumberOfUsers = (await _unitOfWork.UserRepository.GetAllUsersAsync()).Count(),
 | 
			
		||||
            TotalFiles = await _unitOfWork.LibraryRepository.GetTotalFiles(),
 | 
			
		||||
            TotalGenres = await _unitOfWork.GenreRepository.GetCountAsync(),
 | 
			
		||||
            TotalPeople = await _unitOfWork.PersonRepository.GetCountAsync(),
 | 
			
		||||
            UsingSeriesRelationships = await GetIfUsingSeriesRelationship(),
 | 
			
		||||
            EncodeMediaAs = serverSettings.EncodeMediaAs,
 | 
			
		||||
            MaxSeriesInALibrary = await MaxSeriesInAnyLibrary(),
 | 
			
		||||
            MaxVolumesInASeries = await MaxVolumesInASeries(),
 | 
			
		||||
            MaxChaptersInASeries = await MaxChaptersInASeries(),
 | 
			
		||||
            MangaReaderBackgroundColors = await AllMangaReaderBackgroundColors(),
 | 
			
		||||
            MangaReaderPageSplittingModes = await AllMangaReaderPageSplitting(),
 | 
			
		||||
            MangaReaderLayoutModes = await AllMangaReaderLayoutModes(),
 | 
			
		||||
            FileFormats = AllFormats(),
 | 
			
		||||
            UsingRestrictedProfiles = await GetUsingRestrictedProfiles(),
 | 
			
		||||
            LastReadTime = await _unitOfWork.AppUserProgressRepository.GetLatestProgress()
 | 
			
		||||
        };
 | 
			
		||||
 | 
			
		||||
        var usersWithPref = (await _unitOfWork.UserRepository.GetAllUsersAsync(AppUserIncludes.UserPreferences)).ToList();
 | 
			
		||||
        serverInfo.UsersOnCardLayout =
 | 
			
		||||
            usersWithPref.Count(u => u.UserPreferences.GlobalPageLayoutMode == PageLayoutMode.Cards);
 | 
			
		||||
        serverInfo.UsersOnListLayout =
 | 
			
		||||
            usersWithPref.Count(u => u.UserPreferences.GlobalPageLayoutMode == PageLayoutMode.List);
 | 
			
		||||
 | 
			
		||||
        var firstAdminUser = (await _unitOfWork.UserRepository.GetAdminUsersAsync()).FirstOrDefault();
 | 
			
		||||
 | 
			
		||||
        if (firstAdminUser != null)
 | 
			
		||||
        {
 | 
			
		||||
            var firstAdminUserPref = (await _unitOfWork.UserRepository.GetPreferencesAsync(firstAdminUser.UserName!));
 | 
			
		||||
            var activeTheme = firstAdminUserPref?.Theme ?? Seed.DefaultThemes.First(t => t.IsDefault);
 | 
			
		||||
 | 
			
		||||
            serverInfo.ActiveSiteTheme = activeTheme.Name;
 | 
			
		||||
            if (firstAdminUserPref != null) serverInfo.MangaReaderMode = firstAdminUserPref.ReaderMode;
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        return serverInfo;
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    public async Task<ServerInfoSlimDto> GetServerInfoSlim()
 | 
			
		||||
    {
 | 
			
		||||
@ -199,7 +154,7 @@ public class StatsService : IStatsService
 | 
			
		||||
            var response = await (ApiUrl + "/api/v2/stats/opt-out?installId=" + installId)
 | 
			
		||||
                .WithHeader("Accept", "application/json")
 | 
			
		||||
                .WithHeader("User-Agent", "Kavita")
 | 
			
		||||
                .WithHeader("x-api-key", "MsnvA2DfQqxSK5jh")
 | 
			
		||||
                .WithHeader("x-api-key", ApiKey)
 | 
			
		||||
                .WithHeader("x-kavita-version", BuildInfo.Version)
 | 
			
		||||
                .WithHeader("Content-Type", "application/json")
 | 
			
		||||
                .WithTimeout(TimeSpan.FromSeconds(30))
 | 
			
		||||
@ -220,37 +175,32 @@ public class StatsService : IStatsService
 | 
			
		||||
        }
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    private async Task<float> GetPercentageOfLibrariesWithFolderWatchingEnabled()
 | 
			
		||||
    private static async Task<long> PingStatsApi()
 | 
			
		||||
    {
 | 
			
		||||
        var libraries = (await _unitOfWork.LibraryRepository.GetLibrariesAsync()).ToList();
 | 
			
		||||
        if (libraries.Count == 0) return 0.0f;
 | 
			
		||||
        return libraries.Count(l => l.FolderWatching) / (1.0f * libraries.Count);
 | 
			
		||||
    }
 | 
			
		||||
        try
 | 
			
		||||
        {
 | 
			
		||||
            var sw = Stopwatch.StartNew();
 | 
			
		||||
            var response = await (ApiUrl + "/api/health/")
 | 
			
		||||
                .WithHeader("Accept", "application/json")
 | 
			
		||||
                .WithHeader("User-Agent", "Kavita")
 | 
			
		||||
                .WithHeader("x-api-key", ApiKey)
 | 
			
		||||
                .WithHeader("x-kavita-version", BuildInfo.Version)
 | 
			
		||||
                .WithHeader("Content-Type", "application/json")
 | 
			
		||||
                .WithTimeout(TimeSpan.FromSeconds(30))
 | 
			
		||||
                .GetAsync();
 | 
			
		||||
 | 
			
		||||
    private async Task<float> GetPercentageOfLibrariesIncludedInRecommended()
 | 
			
		||||
    {
 | 
			
		||||
        var libraries = (await _unitOfWork.LibraryRepository.GetLibrariesAsync()).ToList();
 | 
			
		||||
        if (libraries.Count == 0) return 0.0f;
 | 
			
		||||
        return libraries.Count(l => l.IncludeInRecommended) / (1.0f * libraries.Count);
 | 
			
		||||
    }
 | 
			
		||||
            if (response.StatusCode == StatusCodes.Status200OK)
 | 
			
		||||
            {
 | 
			
		||||
                sw.Stop();
 | 
			
		||||
                return sw.ElapsedMilliseconds;
 | 
			
		||||
            }
 | 
			
		||||
        }
 | 
			
		||||
        catch (Exception)
 | 
			
		||||
        {
 | 
			
		||||
            /* Swallow */
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
    private async Task<float> GetPercentageOfLibrariesIncludedInDashboard()
 | 
			
		||||
    {
 | 
			
		||||
        var libraries = (await _unitOfWork.LibraryRepository.GetLibrariesAsync()).ToList();
 | 
			
		||||
        if (libraries.Count == 0) return 0.0f;
 | 
			
		||||
        return libraries.Count(l => l.IncludeInDashboard) / (1.0f * libraries.Count);
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    private async Task<float> GetPercentageOfLibrariesIncludedInSearch()
 | 
			
		||||
    {
 | 
			
		||||
        var libraries = (await _unitOfWork.LibraryRepository.GetLibrariesAsync()).ToList();
 | 
			
		||||
        if (libraries.Count == 0) return 0.0f;
 | 
			
		||||
        return libraries.Count(l => l.IncludeInSearch) / (1.0f * libraries.Count);
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    private Task<bool> GetIfUsingSeriesRelationship()
 | 
			
		||||
    {
 | 
			
		||||
        return _context.SeriesRelation.AnyAsync();
 | 
			
		||||
        return 0;
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    private async Task<int> MaxSeriesInAnyLibrary()
 | 
			
		||||
@ -290,41 +240,178 @@ public class StatsService : IStatsService
 | 
			
		||||
                .Count());
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    private async Task<IEnumerable<string>> AllMangaReaderBackgroundColors()
 | 
			
		||||
    private async Task<ServerInfoV3Dto> GetStatV3Payload()
 | 
			
		||||
    {
 | 
			
		||||
        return await _context.AppUserPreferences.Select(p => p.BackgroundColor).Distinct().ToListAsync();
 | 
			
		||||
    }
 | 
			
		||||
        var serverSettings = await _unitOfWork.SettingsRepository.GetSettingsDtoAsync();
 | 
			
		||||
        var dto = new ServerInfoV3Dto()
 | 
			
		||||
        {
 | 
			
		||||
            InstallId = serverSettings.InstallId,
 | 
			
		||||
            KavitaVersion = serverSettings.InstallVersion,
 | 
			
		||||
            InitialKavitaVersion = serverSettings.FirstInstallVersion,
 | 
			
		||||
            InitialInstallDate = (DateTime)serverSettings.FirstInstallDate!,
 | 
			
		||||
            IsDocker = OsInfo.IsDocker,
 | 
			
		||||
            Os = RuntimeInformation.OSDescription,
 | 
			
		||||
            NumOfCores = Math.Max(Environment.ProcessorCount, 1),
 | 
			
		||||
            DotnetVersion = Environment.Version.ToString(),
 | 
			
		||||
            OpdsEnabled = serverSettings.EnableOpds,
 | 
			
		||||
            EncodeMediaAs = serverSettings.EncodeMediaAs,
 | 
			
		||||
        };
 | 
			
		||||
 | 
			
		||||
    private async Task<IEnumerable<PageSplitOption>> AllMangaReaderPageSplitting()
 | 
			
		||||
    {
 | 
			
		||||
        return await _context.AppUserPreferences.Select(p => p.PageSplitOption).Distinct().ToListAsync();
 | 
			
		||||
    }
 | 
			
		||||
        dto.OsLocale = CultureInfo.CurrentCulture.EnglishName;
 | 
			
		||||
        dto.LastReadTime = await _unitOfWork.AppUserProgressRepository.GetLatestProgress();
 | 
			
		||||
        dto.MaxSeriesInALibrary = await MaxSeriesInAnyLibrary();
 | 
			
		||||
        dto.MaxVolumesInASeries = await MaxVolumesInASeries();
 | 
			
		||||
        dto.MaxChaptersInASeries = await MaxChaptersInASeries();
 | 
			
		||||
        dto.TotalFiles = await _unitOfWork.LibraryRepository.GetTotalFiles();
 | 
			
		||||
        dto.TotalGenres = await _unitOfWork.GenreRepository.GetCountAsync();
 | 
			
		||||
        dto.TotalPeople = await _unitOfWork.PersonRepository.GetCountAsync();
 | 
			
		||||
        dto.TotalSeries = await _unitOfWork.SeriesRepository.GetCountAsync();
 | 
			
		||||
        dto.TotalLibraries = (await _unitOfWork.LibraryRepository.GetLibrariesAsync()).Count();
 | 
			
		||||
        dto.NumberOfCollections = (await _unitOfWork.CollectionTagRepository.GetAllCollectionsAsync()).Count();
 | 
			
		||||
        dto.NumberOfReadingLists = await _unitOfWork.ReadingListRepository.Count();
 | 
			
		||||
 | 
			
		||||
        try
 | 
			
		||||
        {
 | 
			
		||||
            var license = (await _unitOfWork.SettingsRepository.GetSettingAsync(ServerSettingKey.LicenseKey)).Value;
 | 
			
		||||
            dto.ActiveKavitaPlusSubscription = await _licenseService.HasActiveSubscription(license);
 | 
			
		||||
        }
 | 
			
		||||
        catch (Exception)
 | 
			
		||||
        {
 | 
			
		||||
            dto.ActiveKavitaPlusSubscription = false;
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
    private async Task<IEnumerable<LayoutMode>> AllMangaReaderLayoutModes()
 | 
			
		||||
    {
 | 
			
		||||
        return await _context.AppUserPreferences.Select(p => p.LayoutMode).Distinct().ToListAsync();
 | 
			
		||||
    }
 | 
			
		||||
        // Find a random cbz/zip file and open it for reading
 | 
			
		||||
        await OpenRandomFile(dto);
 | 
			
		||||
        dto.TimeToPingKavitaStatsApi = await PingStatsApi();
 | 
			
		||||
 | 
			
		||||
    private IEnumerable<FileFormatDto> AllFormats()
 | 
			
		||||
    {
 | 
			
		||||
        #region Relationships
 | 
			
		||||
 | 
			
		||||
        var results =  _context.MangaFile
 | 
			
		||||
            .AsNoTracking()
 | 
			
		||||
            .AsEnumerable()
 | 
			
		||||
            .Select(m => new FileFormatDto()
 | 
			
		||||
        dto.Relationships = await _context.SeriesRelation
 | 
			
		||||
            .GroupBy(sr => sr.RelationKind)
 | 
			
		||||
            .Select(g => new RelationshipStatV3
 | 
			
		||||
            {
 | 
			
		||||
                Format = m.Format,
 | 
			
		||||
                Extension = m.Extension
 | 
			
		||||
                Relationship = g.Key,
 | 
			
		||||
                Count = g.Count()
 | 
			
		||||
            })
 | 
			
		||||
            .DistinctBy(f => f.Extension)
 | 
			
		||||
            .ToList();
 | 
			
		||||
            .ToListAsync();
 | 
			
		||||
 | 
			
		||||
        return results;
 | 
			
		||||
        #endregion
 | 
			
		||||
 | 
			
		||||
        #region Libraries
 | 
			
		||||
        var allLibraries = (await _unitOfWork.LibraryRepository.GetLibrariesAsync(LibraryIncludes.Folders |
 | 
			
		||||
            LibraryIncludes.FileTypes | LibraryIncludes.ExcludePatterns | LibraryIncludes.AppUser)).ToList();
 | 
			
		||||
        dto.Libraries ??= [];
 | 
			
		||||
        foreach (var library in allLibraries)
 | 
			
		||||
        {
 | 
			
		||||
            var libDto = new LibraryStatV3();
 | 
			
		||||
            libDto.IncludeInDashboard = library.IncludeInDashboard;
 | 
			
		||||
            libDto.IncludeInSearch = library.IncludeInSearch;
 | 
			
		||||
            libDto.LastScanned = library.LastScanned;
 | 
			
		||||
            libDto.NumberOfFolders = library.Folders.Count;
 | 
			
		||||
            libDto.FileTypes = library.LibraryFileTypes.Select(s => s.FileTypeGroup).Distinct().ToList();
 | 
			
		||||
            libDto.UsingExcludePatterns = library.LibraryExcludePatterns.Any(p => !string.IsNullOrEmpty(p.Pattern));
 | 
			
		||||
            libDto.UsingFolderWatching = library.FolderWatching;
 | 
			
		||||
            libDto.CreateCollectionsFromMetadata = library.ManageCollections;
 | 
			
		||||
            libDto.CreateReadingListsFromMetadata = library.ManageReadingLists;
 | 
			
		||||
 | 
			
		||||
            dto.Libraries.Add(libDto);
 | 
			
		||||
        }
 | 
			
		||||
        #endregion
 | 
			
		||||
 | 
			
		||||
        #region Users
 | 
			
		||||
 | 
			
		||||
        // Create a dictionary mapping user IDs to the libraries they have access to
 | 
			
		||||
        var userLibraryAccess = allLibraries
 | 
			
		||||
            .SelectMany(l => l.AppUsers.Select(appUser => new { l, appUser.Id }))
 | 
			
		||||
            .GroupBy(x => x.Id)
 | 
			
		||||
            .ToDictionary(g => g.Key, g => g.Select(x => x.l).ToList());
 | 
			
		||||
        dto.Users ??= [];
 | 
			
		||||
        var allUsers = await _unitOfWork.UserRepository.GetAllUsersAsync(AppUserIncludes.UserPreferences
 | 
			
		||||
                                                                         | AppUserIncludes.ReadingLists | AppUserIncludes.Bookmarks
 | 
			
		||||
                                                                         | AppUserIncludes.Collections | AppUserIncludes.Devices
 | 
			
		||||
                                                                         | AppUserIncludes.Progress | AppUserIncludes.Ratings
 | 
			
		||||
                                                                         | AppUserIncludes.SmartFilters | AppUserIncludes.WantToRead, false);
 | 
			
		||||
        foreach (var user in allUsers)
 | 
			
		||||
        {
 | 
			
		||||
            var userDto = new UserStatV3();
 | 
			
		||||
            userDto.HasMALToken = !string.IsNullOrEmpty(user.MalAccessToken);
 | 
			
		||||
            userDto.HasAniListToken = !string.IsNullOrEmpty(user.AniListAccessToken);
 | 
			
		||||
            userDto.AgeRestriction = new AgeRestriction()
 | 
			
		||||
            {
 | 
			
		||||
                AgeRating = user.AgeRestriction,
 | 
			
		||||
                IncludeUnknowns = user.AgeRestrictionIncludeUnknowns
 | 
			
		||||
            };
 | 
			
		||||
 | 
			
		||||
            userDto.Locale = user.UserPreferences.Locale;
 | 
			
		||||
            userDto.Roles = [.. _userManager.GetRolesAsync(user).Result];
 | 
			
		||||
            userDto.LastLogin = user.LastActiveUtc;
 | 
			
		||||
            userDto.HasValidEmail = user.Email != null && _emailService.IsValidEmail(user.Email);
 | 
			
		||||
            userDto.IsEmailConfirmed = user.EmailConfirmed;
 | 
			
		||||
            userDto.ActiveTheme = user.UserPreferences.Theme.Name;
 | 
			
		||||
            userDto.CollectionsCreatedCount = user.Collections.Count;
 | 
			
		||||
            userDto.ReadingListsCreatedCount = user.ReadingLists.Count;
 | 
			
		||||
            userDto.LastReadTime = user.Progresses
 | 
			
		||||
                .Select(p => p.LastModifiedUtc)
 | 
			
		||||
                .DefaultIfEmpty()
 | 
			
		||||
                .Max();
 | 
			
		||||
            userDto.DevicePlatforms = user.Devices.Select(d => d.Platform).ToList();
 | 
			
		||||
            userDto.SeriesBookmarksCreatedCount = user.Bookmarks.Count;
 | 
			
		||||
            userDto.SmartFilterCreatedCount = user.SmartFilters.Count;
 | 
			
		||||
            userDto.WantToReadSeriesCount = user.WantToRead.Count;
 | 
			
		||||
 | 
			
		||||
            if (allLibraries.Count > 0 && userLibraryAccess.TryGetValue(user.Id, out var accessibleLibraries))
 | 
			
		||||
            {
 | 
			
		||||
                userDto.PercentageOfLibrariesHasAccess = (1f * accessibleLibraries.Count) / allLibraries.Count;
 | 
			
		||||
            }
 | 
			
		||||
            else
 | 
			
		||||
            {
 | 
			
		||||
                userDto.PercentageOfLibrariesHasAccess = 0;
 | 
			
		||||
            }
 | 
			
		||||
 | 
			
		||||
            dto.Users.Add(userDto);
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        #endregion
 | 
			
		||||
 | 
			
		||||
        return dto;
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    private Task<bool> GetUsingRestrictedProfiles()
 | 
			
		||||
    private async Task OpenRandomFile(ServerInfoV3Dto dto)
 | 
			
		||||
    {
 | 
			
		||||
        return _context.Users.AnyAsync(u => u.AgeRestriction > AgeRating.NotApplicable);
 | 
			
		||||
        var random = new Random();
 | 
			
		||||
        List<string> extensions = [".cbz", ".zip"];
 | 
			
		||||
 | 
			
		||||
        // Count the total number of files that match the criteria
 | 
			
		||||
        var count = await _context.MangaFile.AsNoTracking()
 | 
			
		||||
            .Where(r => r.Extension != null && extensions.Contains(r.Extension))
 | 
			
		||||
            .CountAsync();
 | 
			
		||||
 | 
			
		||||
        if (count == 0)
 | 
			
		||||
        {
 | 
			
		||||
            dto.TimeToOpeCbzMs = 0;
 | 
			
		||||
            dto.TimeToOpenCbzPages = 0;
 | 
			
		||||
 | 
			
		||||
            return;
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        // Generate a random skip value
 | 
			
		||||
        var skip = random.Next(count);
 | 
			
		||||
 | 
			
		||||
        // Fetch the random file
 | 
			
		||||
        var randomFile = await _context.MangaFile.AsNoTracking()
 | 
			
		||||
            .Where(r => r.Extension != null && extensions.Contains(r.Extension))
 | 
			
		||||
            .Skip(skip)
 | 
			
		||||
            .Take(1)
 | 
			
		||||
            .FirstAsync();
 | 
			
		||||
 | 
			
		||||
        var sw = Stopwatch.StartNew();
 | 
			
		||||
 | 
			
		||||
        await _cacheService.Ensure(randomFile.ChapterId);
 | 
			
		||||
        var time = sw.ElapsedMilliseconds;
 | 
			
		||||
        sw.Stop();
 | 
			
		||||
 | 
			
		||||
        dto.TimeToOpeCbzMs = time;
 | 
			
		||||
        dto.TimeToOpenCbzPages = randomFile.Pages;
 | 
			
		||||
    }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
		Loading…
	
	
			
			x
			
			
		
	
		Reference in New Issue
	
	Block a user