Merge branch 'develop' of https://github.com/Kareadita/Kavita into feature/magazine
This commit is contained in:
commit
08a32a26bc
283 changed files with 8056 additions and 2679 deletions
|
@ -28,6 +28,7 @@
|
|||
|
||||
<ItemGroup>
|
||||
<Folder Include="Services\Test Data\ArchiveService\ComicInfos" />
|
||||
<Folder Include="Services\Test Data\CoverDbService\" />
|
||||
<Folder Include="Services\Test Data\ImageService\Covers\" />
|
||||
</ItemGroup>
|
||||
|
||||
|
|
|
@ -20,10 +20,11 @@ namespace API.Tests;
|
|||
|
||||
public abstract class AbstractDbTest : AbstractFsTest , IDisposable
|
||||
{
|
||||
protected readonly DbConnection _connection;
|
||||
protected readonly DataContext _context;
|
||||
protected readonly IUnitOfWork _unitOfWork;
|
||||
protected readonly IMapper _mapper;
|
||||
protected readonly DataContext Context;
|
||||
protected readonly IUnitOfWork UnitOfWork;
|
||||
protected readonly IMapper Mapper;
|
||||
private readonly DbConnection _connection;
|
||||
private bool _disposed;
|
||||
|
||||
protected AbstractDbTest()
|
||||
{
|
||||
|
@ -34,17 +35,17 @@ public abstract class AbstractDbTest : AbstractFsTest , IDisposable
|
|||
|
||||
_connection = RelationalOptionsExtension.Extract(contextOptions).Connection;
|
||||
|
||||
_context = new DataContext(contextOptions);
|
||||
Context = new DataContext(contextOptions);
|
||||
|
||||
_context.Database.EnsureCreated(); // Ensure DB schema is created
|
||||
Context.Database.EnsureCreated(); // Ensure DB schema is created
|
||||
|
||||
Task.Run(SeedDb).GetAwaiter().GetResult();
|
||||
|
||||
var config = new MapperConfiguration(cfg => cfg.AddProfile<AutoMapperProfiles>());
|
||||
_mapper = config.CreateMapper();
|
||||
Mapper = config.CreateMapper();
|
||||
|
||||
GlobalConfiguration.Configuration.UseInMemoryStorage();
|
||||
_unitOfWork = new UnitOfWork(_context, _mapper, null);
|
||||
UnitOfWork = new UnitOfWork(Context, Mapper, null);
|
||||
}
|
||||
|
||||
private static DbConnection CreateInMemoryDatabase()
|
||||
|
@ -59,34 +60,34 @@ public abstract class AbstractDbTest : AbstractFsTest , IDisposable
|
|||
{
|
||||
try
|
||||
{
|
||||
await _context.Database.EnsureCreatedAsync();
|
||||
await Context.Database.EnsureCreatedAsync();
|
||||
var filesystem = CreateFileSystem();
|
||||
|
||||
await Seed.SeedSettings(_context, new DirectoryService(Substitute.For<ILogger<DirectoryService>>(), filesystem));
|
||||
await Seed.SeedSettings(Context, new DirectoryService(Substitute.For<ILogger<DirectoryService>>(), filesystem));
|
||||
|
||||
var setting = await _context.ServerSetting.Where(s => s.Key == ServerSettingKey.CacheDirectory).SingleAsync();
|
||||
var setting = await Context.ServerSetting.Where(s => s.Key == ServerSettingKey.CacheDirectory).SingleAsync();
|
||||
setting.Value = CacheDirectory;
|
||||
|
||||
setting = await _context.ServerSetting.Where(s => s.Key == ServerSettingKey.BackupDirectory).SingleAsync();
|
||||
setting = await Context.ServerSetting.Where(s => s.Key == ServerSettingKey.BackupDirectory).SingleAsync();
|
||||
setting.Value = BackupDirectory;
|
||||
|
||||
setting = await _context.ServerSetting.Where(s => s.Key == ServerSettingKey.BookmarkDirectory).SingleAsync();
|
||||
setting = await Context.ServerSetting.Where(s => s.Key == ServerSettingKey.BookmarkDirectory).SingleAsync();
|
||||
setting.Value = BookmarkDirectory;
|
||||
|
||||
setting = await _context.ServerSetting.Where(s => s.Key == ServerSettingKey.TotalLogs).SingleAsync();
|
||||
setting = await Context.ServerSetting.Where(s => s.Key == ServerSettingKey.TotalLogs).SingleAsync();
|
||||
setting.Value = "10";
|
||||
|
||||
_context.ServerSetting.Update(setting);
|
||||
Context.ServerSetting.Update(setting);
|
||||
|
||||
|
||||
_context.Library.Add(new LibraryBuilder("Manga")
|
||||
Context.Library.Add(new LibraryBuilder("Manga")
|
||||
.WithAllowMetadataMatching(true)
|
||||
.WithFolderPath(new FolderPathBuilder(DataDirectory).Build())
|
||||
.Build());
|
||||
|
||||
await _context.SaveChangesAsync();
|
||||
await Context.SaveChangesAsync();
|
||||
|
||||
await Seed.SeedMetadataSettings(_context);
|
||||
await Seed.SeedMetadataSettings(Context);
|
||||
|
||||
return true;
|
||||
}
|
||||
|
@ -101,8 +102,21 @@ public abstract class AbstractDbTest : AbstractFsTest , IDisposable
|
|||
|
||||
public void Dispose()
|
||||
{
|
||||
_context.Dispose();
|
||||
_connection.Dispose();
|
||||
Dispose(true);
|
||||
GC.SuppressFinalize(this);
|
||||
}
|
||||
|
||||
protected virtual void Dispose(bool disposing)
|
||||
{
|
||||
if (_disposed) return;
|
||||
|
||||
if (disposing)
|
||||
{
|
||||
Context?.Dispose();
|
||||
_connection?.Dispose();
|
||||
}
|
||||
|
||||
_disposed = true;
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
|
@ -114,9 +128,9 @@ public abstract class AbstractDbTest : AbstractFsTest , IDisposable
|
|||
{
|
||||
var role = new AppRole { Id = userId, Name = roleName, NormalizedName = roleName.ToUpper() };
|
||||
|
||||
await _context.Roles.AddAsync(role);
|
||||
await _context.UserRoles.AddAsync(new AppUserRole { UserId = userId, RoleId = userId });
|
||||
await Context.Roles.AddAsync(role);
|
||||
await Context.UserRoles.AddAsync(new AppUserRole { UserId = userId, RoleId = userId });
|
||||
|
||||
await _context.SaveChangesAsync();
|
||||
await Context.SaveChangesAsync();
|
||||
}
|
||||
}
|
||||
|
|
|
@ -1,6 +1,7 @@
|
|||
|
||||
|
||||
using System.IO;
|
||||
using System.IO.Abstractions;
|
||||
using System.IO.Abstractions.TestingHelpers;
|
||||
using API.Services.Tasks.Scanner.Parser;
|
||||
|
||||
|
|
|
@ -142,7 +142,7 @@ public class ChapterListExtensionsTests
|
|||
CreateChapter("darker than black", "1", CreateFile("/manga/darker than black.cbz", MangaFormat.Archive), false),
|
||||
};
|
||||
|
||||
Assert.Equal(chapterList.First(), chapterList.GetFirstChapterWithFiles());
|
||||
Assert.Equal(chapterList[0], chapterList.GetFirstChapterWithFiles());
|
||||
}
|
||||
|
||||
[Fact]
|
||||
|
@ -150,13 +150,13 @@ public class ChapterListExtensionsTests
|
|||
{
|
||||
var chapterList = new List<Chapter>()
|
||||
{
|
||||
CreateChapter("darker than black", API.Services.Tasks.Scanner.Parser.Parser.DefaultChapter, CreateFile("/manga/darker than black.cbz", MangaFormat.Archive), true),
|
||||
CreateChapter("darker than black", Parser.DefaultChapter, CreateFile("/manga/darker than black.cbz", MangaFormat.Archive), true),
|
||||
CreateChapter("darker than black", "1", CreateFile("/manga/darker than black.cbz", MangaFormat.Archive), false),
|
||||
};
|
||||
|
||||
chapterList.First().Files = new List<MangaFile>();
|
||||
chapterList[0].Files = new List<MangaFile>();
|
||||
|
||||
Assert.Equal(chapterList.Last(), chapterList.GetFirstChapterWithFiles());
|
||||
Assert.Equal(chapterList[^1], chapterList.GetFirstChapterWithFiles());
|
||||
}
|
||||
|
||||
|
||||
|
@ -181,7 +181,7 @@ public class ChapterListExtensionsTests
|
|||
CreateChapter("detective comics", API.Services.Tasks.Scanner.Parser.Parser.DefaultChapter, CreateFile("/manga/detective comics #001.cbz", MangaFormat.Archive), true)
|
||||
};
|
||||
|
||||
chapterList[0].ReleaseDate = new DateTime(10, 1, 1);
|
||||
chapterList[0].ReleaseDate = new DateTime(10, 1, 1, 0, 0, 0, DateTimeKind.Utc);
|
||||
chapterList[1].ReleaseDate = DateTime.MinValue;
|
||||
|
||||
Assert.Equal(0, chapterList.MinimumReleaseYear());
|
||||
|
@ -196,8 +196,8 @@ public class ChapterListExtensionsTests
|
|||
CreateChapter("detective comics", API.Services.Tasks.Scanner.Parser.Parser.DefaultChapter, CreateFile("/manga/detective comics #001.cbz", MangaFormat.Archive), true)
|
||||
};
|
||||
|
||||
chapterList[0].ReleaseDate = new DateTime(2002, 1, 1);
|
||||
chapterList[1].ReleaseDate = new DateTime(2012, 2, 1);
|
||||
chapterList[0].ReleaseDate = new DateTime(2002, 1, 1, 0, 0, 0, DateTimeKind.Utc);
|
||||
chapterList[1].ReleaseDate = new DateTime(2012, 2, 1, 0, 0, 0, DateTimeKind.Utc);
|
||||
|
||||
Assert.Equal(2002, chapterList.MinimumReleaseYear());
|
||||
}
|
||||
|
|
|
@ -24,9 +24,9 @@ public class SeriesFilterTests : AbstractDbTest
|
|||
{
|
||||
protected override async Task ResetDb()
|
||||
{
|
||||
_context.Series.RemoveRange(_context.Series);
|
||||
_context.AppUser.RemoveRange(_context.AppUser);
|
||||
await _context.SaveChangesAsync();
|
||||
Context.Series.RemoveRange(Context.Series);
|
||||
Context.AppUser.RemoveRange(Context.AppUser);
|
||||
await Context.SaveChangesAsync();
|
||||
}
|
||||
|
||||
#region HasProgress
|
||||
|
@ -54,18 +54,18 @@ public class SeriesFilterTests : AbstractDbTest
|
|||
.WithLibrary(library)
|
||||
.Build();
|
||||
|
||||
_context.Users.Add(user);
|
||||
_context.Library.Add(library);
|
||||
await _context.SaveChangesAsync();
|
||||
Context.Users.Add(user);
|
||||
Context.Library.Add(library);
|
||||
await Context.SaveChangesAsync();
|
||||
|
||||
|
||||
// Create read progress on Partial and Full
|
||||
var readerService = new ReaderService(_unitOfWork, Substitute.For<ILogger<ReaderService>>(),
|
||||
var readerService = new ReaderService(UnitOfWork, Substitute.For<ILogger<ReaderService>>(),
|
||||
Substitute.For<IEventHub>(), Substitute.For<IImageService>(),
|
||||
Substitute.For<IDirectoryService>(), Substitute.For<IScrobblingService>());
|
||||
|
||||
// Select Partial and set pages read to 5 on first chapter
|
||||
var partialSeries = await _unitOfWork.SeriesRepository.GetSeriesByIdAsync(2);
|
||||
var partialSeries = await UnitOfWork.SeriesRepository.GetSeriesByIdAsync(2);
|
||||
var partialChapter = partialSeries.Volumes.First().Chapters.First();
|
||||
|
||||
Assert.True(await readerService.SaveReadingProgress(new ProgressDto()
|
||||
|
@ -78,7 +78,7 @@ public class SeriesFilterTests : AbstractDbTest
|
|||
}, user.Id));
|
||||
|
||||
// Select Full and set pages read to 10 on first chapter
|
||||
var fullSeries = await _unitOfWork.SeriesRepository.GetSeriesByIdAsync(3);
|
||||
var fullSeries = await UnitOfWork.SeriesRepository.GetSeriesByIdAsync(3);
|
||||
var fullChapter = fullSeries.Volumes.First().Chapters.First();
|
||||
|
||||
Assert.True(await readerService.SaveReadingProgress(new ProgressDto()
|
||||
|
@ -98,7 +98,7 @@ public class SeriesFilterTests : AbstractDbTest
|
|||
{
|
||||
var user = await SetupHasProgress();
|
||||
|
||||
var queryResult = await _context.Series.HasReadingProgress(true, FilterComparison.LessThan, 50, user.Id)
|
||||
var queryResult = await Context.Series.HasReadingProgress(true, FilterComparison.LessThan, 50, user.Id)
|
||||
.ToListAsync();
|
||||
|
||||
Assert.Single(queryResult);
|
||||
|
@ -111,7 +111,7 @@ public class SeriesFilterTests : AbstractDbTest
|
|||
var user = await SetupHasProgress();
|
||||
|
||||
// Query series with progress <= 50%
|
||||
var queryResult = await _context.Series.HasReadingProgress(true, FilterComparison.LessThanEqual, 50, user.Id)
|
||||
var queryResult = await Context.Series.HasReadingProgress(true, FilterComparison.LessThanEqual, 50, user.Id)
|
||||
.ToListAsync();
|
||||
|
||||
Assert.Equal(2, queryResult.Count);
|
||||
|
@ -125,7 +125,7 @@ public class SeriesFilterTests : AbstractDbTest
|
|||
var user = await SetupHasProgress();
|
||||
|
||||
// Query series with progress > 50%
|
||||
var queryResult = await _context.Series.HasReadingProgress(true, FilterComparison.GreaterThan, 50, user.Id)
|
||||
var queryResult = await Context.Series.HasReadingProgress(true, FilterComparison.GreaterThan, 50, user.Id)
|
||||
.ToListAsync();
|
||||
|
||||
Assert.Single(queryResult);
|
||||
|
@ -138,7 +138,7 @@ public class SeriesFilterTests : AbstractDbTest
|
|||
var user = await SetupHasProgress();
|
||||
|
||||
// Query series with progress == 100%
|
||||
var queryResult = await _context.Series.HasReadingProgress(true, FilterComparison.Equal, 100, user.Id)
|
||||
var queryResult = await Context.Series.HasReadingProgress(true, FilterComparison.Equal, 100, user.Id)
|
||||
.ToListAsync();
|
||||
|
||||
Assert.Single(queryResult);
|
||||
|
@ -151,7 +151,7 @@ public class SeriesFilterTests : AbstractDbTest
|
|||
var user = await SetupHasProgress();
|
||||
|
||||
// Query series with progress < 100%
|
||||
var queryResult = await _context.Series.HasReadingProgress(true, FilterComparison.LessThan, 100, user.Id)
|
||||
var queryResult = await Context.Series.HasReadingProgress(true, FilterComparison.LessThan, 100, user.Id)
|
||||
.ToListAsync();
|
||||
|
||||
Assert.Equal(2, queryResult.Count);
|
||||
|
@ -165,7 +165,7 @@ public class SeriesFilterTests : AbstractDbTest
|
|||
var user = await SetupHasProgress();
|
||||
|
||||
// Query series with progress <= 100%
|
||||
var queryResult = await _context.Series.HasReadingProgress(true, FilterComparison.LessThanEqual, 100, user.Id)
|
||||
var queryResult = await Context.Series.HasReadingProgress(true, FilterComparison.LessThanEqual, 100, user.Id)
|
||||
.ToListAsync();
|
||||
|
||||
Assert.Equal(3, queryResult.Count);
|
||||
|
@ -188,16 +188,16 @@ public class SeriesFilterTests : AbstractDbTest
|
|||
.WithLibrary(library)
|
||||
.Build();
|
||||
|
||||
_context.Users.Add(user);
|
||||
_context.Library.Add(library);
|
||||
await _context.SaveChangesAsync();
|
||||
Context.Users.Add(user);
|
||||
Context.Library.Add(library);
|
||||
await Context.SaveChangesAsync();
|
||||
|
||||
var readerService = new ReaderService(_unitOfWork, Substitute.For<ILogger<ReaderService>>(),
|
||||
var readerService = new ReaderService(UnitOfWork, Substitute.For<ILogger<ReaderService>>(),
|
||||
Substitute.For<IEventHub>(), Substitute.For<IImageService>(),
|
||||
Substitute.For<IDirectoryService>(), Substitute.For<IScrobblingService>());
|
||||
|
||||
// Set progress to 99.99% (99/100 pages read)
|
||||
var series = await _unitOfWork.SeriesRepository.GetSeriesByIdAsync(1);
|
||||
var series = await UnitOfWork.SeriesRepository.GetSeriesByIdAsync(1);
|
||||
var chapter = series.Volumes.First().Chapters.First();
|
||||
|
||||
Assert.True(await readerService.SaveReadingProgress(new ProgressDto()
|
||||
|
@ -210,7 +210,7 @@ public class SeriesFilterTests : AbstractDbTest
|
|||
}, user.Id));
|
||||
|
||||
// Query series with progress < 100%
|
||||
var queryResult = await _context.Series.HasReadingProgress(true, FilterComparison.LessThan, 100, user.Id)
|
||||
var queryResult = await Context.Series.HasReadingProgress(true, FilterComparison.LessThan, 100, user.Id)
|
||||
.ToListAsync();
|
||||
|
||||
Assert.Single(queryResult);
|
||||
|
@ -246,9 +246,9 @@ public class SeriesFilterTests : AbstractDbTest
|
|||
.WithLibrary(library)
|
||||
.Build();
|
||||
|
||||
_context.Users.Add(user);
|
||||
_context.Library.Add(library);
|
||||
await _context.SaveChangesAsync();
|
||||
Context.Users.Add(user);
|
||||
Context.Library.Add(library);
|
||||
await Context.SaveChangesAsync();
|
||||
|
||||
return user;
|
||||
}
|
||||
|
@ -258,7 +258,7 @@ public class SeriesFilterTests : AbstractDbTest
|
|||
{
|
||||
await SetupHasLanguage();
|
||||
|
||||
var foundSeries = await _context.Series.HasLanguage(true, FilterComparison.Equal, ["en"]).ToListAsync();
|
||||
var foundSeries = await Context.Series.HasLanguage(true, FilterComparison.Equal, ["en"]).ToListAsync();
|
||||
Assert.Single(foundSeries);
|
||||
Assert.Equal("en", foundSeries[0].Metadata.Language);
|
||||
}
|
||||
|
@ -268,7 +268,7 @@ public class SeriesFilterTests : AbstractDbTest
|
|||
{
|
||||
await SetupHasLanguage();
|
||||
|
||||
var foundSeries = await _context.Series.HasLanguage(true, FilterComparison.NotEqual, ["en"]).ToListAsync();
|
||||
var foundSeries = await Context.Series.HasLanguage(true, FilterComparison.NotEqual, ["en"]).ToListAsync();
|
||||
Assert.Equal(2, foundSeries.Count);
|
||||
Assert.DoesNotContain(foundSeries, s => s.Metadata.Language == "en");
|
||||
}
|
||||
|
@ -278,7 +278,7 @@ public class SeriesFilterTests : AbstractDbTest
|
|||
{
|
||||
await SetupHasLanguage();
|
||||
|
||||
var foundSeries = await _context.Series.HasLanguage(true, FilterComparison.Contains, ["en", "fr"]).ToListAsync();
|
||||
var foundSeries = await Context.Series.HasLanguage(true, FilterComparison.Contains, ["en", "fr"]).ToListAsync();
|
||||
Assert.Equal(2, foundSeries.Count);
|
||||
Assert.Contains(foundSeries, s => s.Metadata.Language == "en");
|
||||
Assert.Contains(foundSeries, s => s.Metadata.Language == "fr");
|
||||
|
@ -289,7 +289,7 @@ public class SeriesFilterTests : AbstractDbTest
|
|||
{
|
||||
await SetupHasLanguage();
|
||||
|
||||
var foundSeries = await _context.Series.HasLanguage(true, FilterComparison.NotContains, ["en", "fr"]).ToListAsync();
|
||||
var foundSeries = await Context.Series.HasLanguage(true, FilterComparison.NotContains, ["en", "fr"]).ToListAsync();
|
||||
Assert.Single(foundSeries);
|
||||
Assert.Equal("es", foundSeries[0].Metadata.Language);
|
||||
}
|
||||
|
@ -300,11 +300,11 @@ public class SeriesFilterTests : AbstractDbTest
|
|||
await SetupHasLanguage();
|
||||
|
||||
// Since "MustContains" matches all the provided languages, no series should match in this case.
|
||||
var foundSeries = await _context.Series.HasLanguage(true, FilterComparison.MustContains, ["en", "fr"]).ToListAsync();
|
||||
var foundSeries = await Context.Series.HasLanguage(true, FilterComparison.MustContains, ["en", "fr"]).ToListAsync();
|
||||
Assert.Empty(foundSeries);
|
||||
|
||||
// Single language should work.
|
||||
foundSeries = await _context.Series.HasLanguage(true, FilterComparison.MustContains, ["en"]).ToListAsync();
|
||||
foundSeries = await Context.Series.HasLanguage(true, FilterComparison.MustContains, ["en"]).ToListAsync();
|
||||
Assert.Single(foundSeries);
|
||||
Assert.Equal("en", foundSeries[0].Metadata.Language);
|
||||
}
|
||||
|
@ -314,7 +314,7 @@ public class SeriesFilterTests : AbstractDbTest
|
|||
{
|
||||
await SetupHasLanguage();
|
||||
|
||||
var foundSeries = await _context.Series.HasLanguage(true, FilterComparison.Matches, ["e"]).ToListAsync();
|
||||
var foundSeries = await Context.Series.HasLanguage(true, FilterComparison.Matches, ["e"]).ToListAsync();
|
||||
Assert.Equal(2, foundSeries.Count);
|
||||
Assert.Contains("en", foundSeries.Select(s => s.Metadata.Language));
|
||||
Assert.Contains("es", foundSeries.Select(s => s.Metadata.Language));
|
||||
|
@ -325,7 +325,7 @@ public class SeriesFilterTests : AbstractDbTest
|
|||
{
|
||||
await SetupHasLanguage();
|
||||
|
||||
var foundSeries = await _context.Series.HasLanguage(false, FilterComparison.Equal, ["en"]).ToListAsync();
|
||||
var foundSeries = await Context.Series.HasLanguage(false, FilterComparison.Equal, ["en"]).ToListAsync();
|
||||
Assert.Equal(3, foundSeries.Count);
|
||||
}
|
||||
|
||||
|
@ -334,7 +334,7 @@ public class SeriesFilterTests : AbstractDbTest
|
|||
{
|
||||
await SetupHasLanguage();
|
||||
|
||||
var foundSeries = await _context.Series.HasLanguage(true, FilterComparison.Equal, new List<string>()).ToListAsync();
|
||||
var foundSeries = await Context.Series.HasLanguage(true, FilterComparison.Equal, new List<string>()).ToListAsync();
|
||||
Assert.Equal(3, foundSeries.Count);
|
||||
}
|
||||
|
||||
|
@ -345,7 +345,7 @@ public class SeriesFilterTests : AbstractDbTest
|
|||
|
||||
await Assert.ThrowsAsync<ArgumentOutOfRangeException>(async () =>
|
||||
{
|
||||
await _context.Series.HasLanguage(true, FilterComparison.GreaterThan, ["en"]).ToListAsync();
|
||||
await Context.Series.HasLanguage(true, FilterComparison.GreaterThan, ["en"]).ToListAsync();
|
||||
});
|
||||
}
|
||||
|
||||
|
@ -379,9 +379,9 @@ public class SeriesFilterTests : AbstractDbTest
|
|||
.WithLibrary(library)
|
||||
.Build();
|
||||
|
||||
_context.Users.Add(user);
|
||||
_context.Library.Add(library);
|
||||
await _context.SaveChangesAsync();
|
||||
Context.Users.Add(user);
|
||||
Context.Library.Add(library);
|
||||
await Context.SaveChangesAsync();
|
||||
|
||||
return user;
|
||||
}
|
||||
|
@ -391,7 +391,7 @@ public class SeriesFilterTests : AbstractDbTest
|
|||
{
|
||||
await SetupHasAverageRating();
|
||||
|
||||
var series = await _context.Series.HasAverageRating(true, FilterComparison.Equal, 100).ToListAsync();
|
||||
var series = await Context.Series.HasAverageRating(true, FilterComparison.Equal, 100).ToListAsync();
|
||||
Assert.Single(series);
|
||||
Assert.Equal("Full", series[0].Name);
|
||||
}
|
||||
|
@ -401,7 +401,7 @@ public class SeriesFilterTests : AbstractDbTest
|
|||
{
|
||||
await SetupHasAverageRating();
|
||||
|
||||
var series = await _context.Series.HasAverageRating(true, FilterComparison.GreaterThan, 50).ToListAsync();
|
||||
var series = await Context.Series.HasAverageRating(true, FilterComparison.GreaterThan, 50).ToListAsync();
|
||||
Assert.Single(series);
|
||||
Assert.Equal("Full", series[0].Name);
|
||||
}
|
||||
|
@ -411,7 +411,7 @@ public class SeriesFilterTests : AbstractDbTest
|
|||
{
|
||||
await SetupHasAverageRating();
|
||||
|
||||
var series = await _context.Series.HasAverageRating(true, FilterComparison.GreaterThanEqual, 50).ToListAsync();
|
||||
var series = await Context.Series.HasAverageRating(true, FilterComparison.GreaterThanEqual, 50).ToListAsync();
|
||||
Assert.Equal(2, series.Count);
|
||||
Assert.Contains(series, s => s.Name == "Partial");
|
||||
Assert.Contains(series, s => s.Name == "Full");
|
||||
|
@ -422,7 +422,7 @@ public class SeriesFilterTests : AbstractDbTest
|
|||
{
|
||||
await SetupHasAverageRating();
|
||||
|
||||
var series = await _context.Series.HasAverageRating(true, FilterComparison.LessThan, 50).ToListAsync();
|
||||
var series = await Context.Series.HasAverageRating(true, FilterComparison.LessThan, 50).ToListAsync();
|
||||
Assert.Single(series);
|
||||
Assert.Equal("None", series[0].Name);
|
||||
}
|
||||
|
@ -432,7 +432,7 @@ public class SeriesFilterTests : AbstractDbTest
|
|||
{
|
||||
await SetupHasAverageRating();
|
||||
|
||||
var series = await _context.Series.HasAverageRating(true, FilterComparison.LessThanEqual, 50).ToListAsync();
|
||||
var series = await Context.Series.HasAverageRating(true, FilterComparison.LessThanEqual, 50).ToListAsync();
|
||||
Assert.Equal(2, series.Count);
|
||||
Assert.Contains(series, s => s.Name == "None");
|
||||
Assert.Contains(series, s => s.Name == "Partial");
|
||||
|
@ -443,7 +443,7 @@ public class SeriesFilterTests : AbstractDbTest
|
|||
{
|
||||
await SetupHasAverageRating();
|
||||
|
||||
var series = await _context.Series.HasAverageRating(true, FilterComparison.NotEqual, 100).ToListAsync();
|
||||
var series = await Context.Series.HasAverageRating(true, FilterComparison.NotEqual, 100).ToListAsync();
|
||||
Assert.Equal(2, series.Count);
|
||||
Assert.DoesNotContain(series, s => s.Name == "Full");
|
||||
}
|
||||
|
@ -453,7 +453,7 @@ public class SeriesFilterTests : AbstractDbTest
|
|||
{
|
||||
await SetupHasAverageRating();
|
||||
|
||||
var series = await _context.Series.HasAverageRating(false, FilterComparison.Equal, 100).ToListAsync();
|
||||
var series = await Context.Series.HasAverageRating(false, FilterComparison.Equal, 100).ToListAsync();
|
||||
Assert.Equal(3, series.Count);
|
||||
}
|
||||
|
||||
|
@ -462,7 +462,7 @@ public class SeriesFilterTests : AbstractDbTest
|
|||
{
|
||||
await SetupHasAverageRating();
|
||||
|
||||
var series = await _context.Series.HasAverageRating(true, FilterComparison.Equal, -1).ToListAsync();
|
||||
var series = await Context.Series.HasAverageRating(true, FilterComparison.Equal, -1).ToListAsync();
|
||||
Assert.Single(series);
|
||||
Assert.Equal("None", series[0].Name);
|
||||
}
|
||||
|
@ -474,7 +474,7 @@ public class SeriesFilterTests : AbstractDbTest
|
|||
|
||||
await Assert.ThrowsAsync<KavitaException>(async () =>
|
||||
{
|
||||
await _context.Series.HasAverageRating(true, FilterComparison.Contains, 50).ToListAsync();
|
||||
await Context.Series.HasAverageRating(true, FilterComparison.Contains, 50).ToListAsync();
|
||||
});
|
||||
}
|
||||
|
||||
|
@ -485,7 +485,7 @@ public class SeriesFilterTests : AbstractDbTest
|
|||
|
||||
await Assert.ThrowsAsync<ArgumentOutOfRangeException>(async () =>
|
||||
{
|
||||
await _context.Series.HasAverageRating(true, (FilterComparison)999, 50).ToListAsync();
|
||||
await Context.Series.HasAverageRating(true, (FilterComparison)999, 50).ToListAsync();
|
||||
});
|
||||
}
|
||||
|
||||
|
@ -519,9 +519,9 @@ public class SeriesFilterTests : AbstractDbTest
|
|||
.WithLibrary(library)
|
||||
.Build();
|
||||
|
||||
_context.Users.Add(user);
|
||||
_context.Library.Add(library);
|
||||
await _context.SaveChangesAsync();
|
||||
Context.Users.Add(user);
|
||||
Context.Library.Add(library);
|
||||
await Context.SaveChangesAsync();
|
||||
|
||||
return user;
|
||||
}
|
||||
|
@ -531,7 +531,7 @@ public class SeriesFilterTests : AbstractDbTest
|
|||
{
|
||||
await SetupHasPublicationStatus();
|
||||
|
||||
var foundSeries = await _context.Series.HasPublicationStatus(true, FilterComparison.Equal, new List<PublicationStatus> { PublicationStatus.Cancelled }).ToListAsync();
|
||||
var foundSeries = await Context.Series.HasPublicationStatus(true, FilterComparison.Equal, new List<PublicationStatus> { PublicationStatus.Cancelled }).ToListAsync();
|
||||
Assert.Single(foundSeries);
|
||||
Assert.Equal("Cancelled", foundSeries[0].Name);
|
||||
}
|
||||
|
@ -541,7 +541,7 @@ public class SeriesFilterTests : AbstractDbTest
|
|||
{
|
||||
await SetupHasPublicationStatus();
|
||||
|
||||
var foundSeries = await _context.Series.HasPublicationStatus(true, FilterComparison.Contains, new List<PublicationStatus> { PublicationStatus.Cancelled, PublicationStatus.Completed }).ToListAsync();
|
||||
var foundSeries = await Context.Series.HasPublicationStatus(true, FilterComparison.Contains, new List<PublicationStatus> { PublicationStatus.Cancelled, PublicationStatus.Completed }).ToListAsync();
|
||||
Assert.Equal(2, foundSeries.Count);
|
||||
Assert.Contains(foundSeries, s => s.Name == "Cancelled");
|
||||
Assert.Contains(foundSeries, s => s.Name == "Completed");
|
||||
|
@ -552,7 +552,7 @@ public class SeriesFilterTests : AbstractDbTest
|
|||
{
|
||||
await SetupHasPublicationStatus();
|
||||
|
||||
var foundSeries = await _context.Series.HasPublicationStatus(true, FilterComparison.NotContains, new List<PublicationStatus> { PublicationStatus.Cancelled }).ToListAsync();
|
||||
var foundSeries = await Context.Series.HasPublicationStatus(true, FilterComparison.NotContains, new List<PublicationStatus> { PublicationStatus.Cancelled }).ToListAsync();
|
||||
Assert.Equal(2, foundSeries.Count);
|
||||
Assert.Contains(foundSeries, s => s.Name == "OnGoing");
|
||||
Assert.Contains(foundSeries, s => s.Name == "Completed");
|
||||
|
@ -563,7 +563,7 @@ public class SeriesFilterTests : AbstractDbTest
|
|||
{
|
||||
await SetupHasPublicationStatus();
|
||||
|
||||
var foundSeries = await _context.Series.HasPublicationStatus(true, FilterComparison.NotEqual, new List<PublicationStatus> { PublicationStatus.OnGoing }).ToListAsync();
|
||||
var foundSeries = await Context.Series.HasPublicationStatus(true, FilterComparison.NotEqual, new List<PublicationStatus> { PublicationStatus.OnGoing }).ToListAsync();
|
||||
Assert.Equal(2, foundSeries.Count);
|
||||
Assert.Contains(foundSeries, s => s.Name == "Cancelled");
|
||||
Assert.Contains(foundSeries, s => s.Name == "Completed");
|
||||
|
@ -574,7 +574,7 @@ public class SeriesFilterTests : AbstractDbTest
|
|||
{
|
||||
await SetupHasPublicationStatus();
|
||||
|
||||
var foundSeries = await _context.Series.HasPublicationStatus(false, FilterComparison.Equal, new List<PublicationStatus> { PublicationStatus.Cancelled }).ToListAsync();
|
||||
var foundSeries = await Context.Series.HasPublicationStatus(false, FilterComparison.Equal, new List<PublicationStatus> { PublicationStatus.Cancelled }).ToListAsync();
|
||||
Assert.Equal(3, foundSeries.Count);
|
||||
}
|
||||
|
||||
|
@ -583,7 +583,7 @@ public class SeriesFilterTests : AbstractDbTest
|
|||
{
|
||||
await SetupHasPublicationStatus();
|
||||
|
||||
var foundSeries = await _context.Series.HasPublicationStatus(true, FilterComparison.Equal, new List<PublicationStatus>()).ToListAsync();
|
||||
var foundSeries = await Context.Series.HasPublicationStatus(true, FilterComparison.Equal, new List<PublicationStatus>()).ToListAsync();
|
||||
Assert.Equal(3, foundSeries.Count);
|
||||
}
|
||||
|
||||
|
@ -594,7 +594,7 @@ public class SeriesFilterTests : AbstractDbTest
|
|||
|
||||
await Assert.ThrowsAsync<KavitaException>(async () =>
|
||||
{
|
||||
await _context.Series.HasPublicationStatus(true, FilterComparison.BeginsWith, new List<PublicationStatus> { PublicationStatus.Cancelled }).ToListAsync();
|
||||
await Context.Series.HasPublicationStatus(true, FilterComparison.BeginsWith, new List<PublicationStatus> { PublicationStatus.Cancelled }).ToListAsync();
|
||||
});
|
||||
}
|
||||
|
||||
|
@ -605,7 +605,7 @@ public class SeriesFilterTests : AbstractDbTest
|
|||
|
||||
await Assert.ThrowsAsync<ArgumentOutOfRangeException>(async () =>
|
||||
{
|
||||
await _context.Series.HasPublicationStatus(true, (FilterComparison)999, new List<PublicationStatus> { PublicationStatus.Cancelled }).ToListAsync();
|
||||
await Context.Series.HasPublicationStatus(true, (FilterComparison)999, new List<PublicationStatus> { PublicationStatus.Cancelled }).ToListAsync();
|
||||
});
|
||||
}
|
||||
#endregion
|
||||
|
@ -637,9 +637,9 @@ public class SeriesFilterTests : AbstractDbTest
|
|||
.WithLibrary(library)
|
||||
.Build();
|
||||
|
||||
_context.Users.Add(user);
|
||||
_context.Library.Add(library);
|
||||
await _context.SaveChangesAsync();
|
||||
Context.Users.Add(user);
|
||||
Context.Library.Add(library);
|
||||
await Context.SaveChangesAsync();
|
||||
|
||||
return user;
|
||||
}
|
||||
|
@ -649,7 +649,7 @@ public class SeriesFilterTests : AbstractDbTest
|
|||
{
|
||||
await SetupHasAgeRating();
|
||||
|
||||
var foundSeries = await _context.Series.HasAgeRating(true, FilterComparison.Equal, [AgeRating.G]).ToListAsync();
|
||||
var foundSeries = await Context.Series.HasAgeRating(true, FilterComparison.Equal, [AgeRating.G]).ToListAsync();
|
||||
Assert.Single(foundSeries);
|
||||
Assert.Equal("G", foundSeries[0].Name);
|
||||
}
|
||||
|
@ -659,7 +659,7 @@ public class SeriesFilterTests : AbstractDbTest
|
|||
{
|
||||
await SetupHasAgeRating();
|
||||
|
||||
var foundSeries = await _context.Series.HasAgeRating(true, FilterComparison.Contains, new List<AgeRating> { AgeRating.G, AgeRating.Mature }).ToListAsync();
|
||||
var foundSeries = await Context.Series.HasAgeRating(true, FilterComparison.Contains, new List<AgeRating> { AgeRating.G, AgeRating.Mature }).ToListAsync();
|
||||
Assert.Equal(2, foundSeries.Count);
|
||||
Assert.Contains(foundSeries, s => s.Name == "G");
|
||||
Assert.Contains(foundSeries, s => s.Name == "Mature");
|
||||
|
@ -670,7 +670,7 @@ public class SeriesFilterTests : AbstractDbTest
|
|||
{
|
||||
await SetupHasAgeRating();
|
||||
|
||||
var foundSeries = await _context.Series.HasAgeRating(true, FilterComparison.NotContains, new List<AgeRating> { AgeRating.Unknown }).ToListAsync();
|
||||
var foundSeries = await Context.Series.HasAgeRating(true, FilterComparison.NotContains, new List<AgeRating> { AgeRating.Unknown }).ToListAsync();
|
||||
Assert.Equal(2, foundSeries.Count);
|
||||
Assert.Contains(foundSeries, s => s.Name == "G");
|
||||
Assert.Contains(foundSeries, s => s.Name == "Mature");
|
||||
|
@ -681,7 +681,7 @@ public class SeriesFilterTests : AbstractDbTest
|
|||
{
|
||||
await SetupHasAgeRating();
|
||||
|
||||
var foundSeries = await _context.Series.HasAgeRating(true, FilterComparison.NotEqual, new List<AgeRating> { AgeRating.G }).ToListAsync();
|
||||
var foundSeries = await Context.Series.HasAgeRating(true, FilterComparison.NotEqual, new List<AgeRating> { AgeRating.G }).ToListAsync();
|
||||
Assert.Equal(2, foundSeries.Count);
|
||||
Assert.Contains(foundSeries, s => s.Name == "Unknown");
|
||||
Assert.Contains(foundSeries, s => s.Name == "Mature");
|
||||
|
@ -692,7 +692,7 @@ public class SeriesFilterTests : AbstractDbTest
|
|||
{
|
||||
await SetupHasAgeRating();
|
||||
|
||||
var foundSeries = await _context.Series.HasAgeRating(true, FilterComparison.GreaterThan, new List<AgeRating> { AgeRating.Unknown }).ToListAsync();
|
||||
var foundSeries = await Context.Series.HasAgeRating(true, FilterComparison.GreaterThan, new List<AgeRating> { AgeRating.Unknown }).ToListAsync();
|
||||
Assert.Equal(2, foundSeries.Count);
|
||||
Assert.Contains(foundSeries, s => s.Name == "G");
|
||||
Assert.Contains(foundSeries, s => s.Name == "Mature");
|
||||
|
@ -703,7 +703,7 @@ public class SeriesFilterTests : AbstractDbTest
|
|||
{
|
||||
await SetupHasAgeRating();
|
||||
|
||||
var foundSeries = await _context.Series.HasAgeRating(true, FilterComparison.GreaterThanEqual, new List<AgeRating> { AgeRating.G }).ToListAsync();
|
||||
var foundSeries = await Context.Series.HasAgeRating(true, FilterComparison.GreaterThanEqual, new List<AgeRating> { AgeRating.G }).ToListAsync();
|
||||
Assert.Equal(2, foundSeries.Count);
|
||||
Assert.Contains(foundSeries, s => s.Name == "G");
|
||||
Assert.Contains(foundSeries, s => s.Name == "Mature");
|
||||
|
@ -714,7 +714,7 @@ public class SeriesFilterTests : AbstractDbTest
|
|||
{
|
||||
await SetupHasAgeRating();
|
||||
|
||||
var foundSeries = await _context.Series.HasAgeRating(true, FilterComparison.LessThan, new List<AgeRating> { AgeRating.Mature }).ToListAsync();
|
||||
var foundSeries = await Context.Series.HasAgeRating(true, FilterComparison.LessThan, new List<AgeRating> { AgeRating.Mature }).ToListAsync();
|
||||
Assert.Equal(2, foundSeries.Count);
|
||||
Assert.Contains(foundSeries, s => s.Name == "Unknown");
|
||||
Assert.Contains(foundSeries, s => s.Name == "G");
|
||||
|
@ -725,7 +725,7 @@ public class SeriesFilterTests : AbstractDbTest
|
|||
{
|
||||
await SetupHasAgeRating();
|
||||
|
||||
var foundSeries = await _context.Series.HasAgeRating(true, FilterComparison.LessThanEqual, new List<AgeRating> { AgeRating.G }).ToListAsync();
|
||||
var foundSeries = await Context.Series.HasAgeRating(true, FilterComparison.LessThanEqual, new List<AgeRating> { AgeRating.G }).ToListAsync();
|
||||
Assert.Equal(2, foundSeries.Count);
|
||||
Assert.Contains(foundSeries, s => s.Name == "Unknown");
|
||||
Assert.Contains(foundSeries, s => s.Name == "G");
|
||||
|
@ -736,7 +736,7 @@ public class SeriesFilterTests : AbstractDbTest
|
|||
{
|
||||
await SetupHasAgeRating();
|
||||
|
||||
var foundSeries = await _context.Series.HasAgeRating(false, FilterComparison.Equal, new List<AgeRating> { AgeRating.G }).ToListAsync();
|
||||
var foundSeries = await Context.Series.HasAgeRating(false, FilterComparison.Equal, new List<AgeRating> { AgeRating.G }).ToListAsync();
|
||||
Assert.Equal(3, foundSeries.Count);
|
||||
}
|
||||
|
||||
|
@ -745,7 +745,7 @@ public class SeriesFilterTests : AbstractDbTest
|
|||
{
|
||||
await SetupHasAgeRating();
|
||||
|
||||
var foundSeries = await _context.Series.HasAgeRating(true, FilterComparison.Equal, new List<AgeRating>()).ToListAsync();
|
||||
var foundSeries = await Context.Series.HasAgeRating(true, FilterComparison.Equal, new List<AgeRating>()).ToListAsync();
|
||||
Assert.Equal(3, foundSeries.Count);
|
||||
}
|
||||
|
||||
|
@ -756,7 +756,7 @@ public class SeriesFilterTests : AbstractDbTest
|
|||
|
||||
await Assert.ThrowsAsync<KavitaException>(async () =>
|
||||
{
|
||||
await _context.Series.HasAgeRating(true, FilterComparison.BeginsWith, new List<AgeRating> { AgeRating.G }).ToListAsync();
|
||||
await Context.Series.HasAgeRating(true, FilterComparison.BeginsWith, new List<AgeRating> { AgeRating.G }).ToListAsync();
|
||||
});
|
||||
}
|
||||
|
||||
|
@ -767,7 +767,7 @@ public class SeriesFilterTests : AbstractDbTest
|
|||
|
||||
await Assert.ThrowsAsync<ArgumentOutOfRangeException>(async () =>
|
||||
{
|
||||
await _context.Series.HasAgeRating(true, (FilterComparison)999, new List<AgeRating> { AgeRating.G }).ToListAsync();
|
||||
await Context.Series.HasAgeRating(true, (FilterComparison)999, new List<AgeRating> { AgeRating.G }).ToListAsync();
|
||||
});
|
||||
}
|
||||
|
||||
|
@ -801,9 +801,9 @@ public class SeriesFilterTests : AbstractDbTest
|
|||
.WithLibrary(library)
|
||||
.Build();
|
||||
|
||||
_context.Users.Add(user);
|
||||
_context.Library.Add(library);
|
||||
await _context.SaveChangesAsync();
|
||||
Context.Users.Add(user);
|
||||
Context.Library.Add(library);
|
||||
await Context.SaveChangesAsync();
|
||||
|
||||
return user;
|
||||
}
|
||||
|
@ -813,7 +813,7 @@ public class SeriesFilterTests : AbstractDbTest
|
|||
{
|
||||
await SetupHasReleaseYear();
|
||||
|
||||
var foundSeries = await _context.Series.HasReleaseYear(true, FilterComparison.Equal, 2020).ToListAsync();
|
||||
var foundSeries = await Context.Series.HasReleaseYear(true, FilterComparison.Equal, 2020).ToListAsync();
|
||||
Assert.Single(foundSeries);
|
||||
Assert.Equal("2020", foundSeries[0].Name);
|
||||
}
|
||||
|
@ -823,7 +823,7 @@ public class SeriesFilterTests : AbstractDbTest
|
|||
{
|
||||
await SetupHasReleaseYear();
|
||||
|
||||
var foundSeries = await _context.Series.HasReleaseYear(true, FilterComparison.GreaterThan, 2000).ToListAsync();
|
||||
var foundSeries = await Context.Series.HasReleaseYear(true, FilterComparison.GreaterThan, 2000).ToListAsync();
|
||||
Assert.Equal(2, foundSeries.Count);
|
||||
Assert.Contains(foundSeries, s => s.Name == "2020");
|
||||
Assert.Contains(foundSeries, s => s.Name == "2025");
|
||||
|
@ -834,7 +834,7 @@ public class SeriesFilterTests : AbstractDbTest
|
|||
{
|
||||
await SetupHasReleaseYear();
|
||||
|
||||
var foundSeries = await _context.Series.HasReleaseYear(true, FilterComparison.LessThan, 2025).ToListAsync();
|
||||
var foundSeries = await Context.Series.HasReleaseYear(true, FilterComparison.LessThan, 2025).ToListAsync();
|
||||
Assert.Equal(2, foundSeries.Count);
|
||||
Assert.Contains(foundSeries, s => s.Name == "2000");
|
||||
Assert.Contains(foundSeries, s => s.Name == "2020");
|
||||
|
@ -845,7 +845,7 @@ public class SeriesFilterTests : AbstractDbTest
|
|||
{
|
||||
await SetupHasReleaseYear();
|
||||
|
||||
var foundSeries = await _context.Series.HasReleaseYear(true, FilterComparison.IsInLast, 5).ToListAsync();
|
||||
var foundSeries = await Context.Series.HasReleaseYear(true, FilterComparison.IsInLast, 5).ToListAsync();
|
||||
Assert.Equal(2, foundSeries.Count);
|
||||
}
|
||||
|
||||
|
@ -854,7 +854,7 @@ public class SeriesFilterTests : AbstractDbTest
|
|||
{
|
||||
await SetupHasReleaseYear();
|
||||
|
||||
var foundSeries = await _context.Series.HasReleaseYear(true, FilterComparison.IsNotInLast, 5).ToListAsync();
|
||||
var foundSeries = await Context.Series.HasReleaseYear(true, FilterComparison.IsNotInLast, 5).ToListAsync();
|
||||
Assert.Single(foundSeries);
|
||||
Assert.Contains(foundSeries, s => s.Name == "2000");
|
||||
}
|
||||
|
@ -864,7 +864,7 @@ public class SeriesFilterTests : AbstractDbTest
|
|||
{
|
||||
await SetupHasReleaseYear();
|
||||
|
||||
var foundSeries = await _context.Series.HasReleaseYear(false, FilterComparison.Equal, 2020).ToListAsync();
|
||||
var foundSeries = await Context.Series.HasReleaseYear(false, FilterComparison.Equal, 2020).ToListAsync();
|
||||
Assert.Equal(3, foundSeries.Count);
|
||||
}
|
||||
|
||||
|
@ -873,7 +873,7 @@ public class SeriesFilterTests : AbstractDbTest
|
|||
{
|
||||
await SetupHasReleaseYear();
|
||||
|
||||
var foundSeries = await _context.Series.HasReleaseYear(true, FilterComparison.Equal, null).ToListAsync();
|
||||
var foundSeries = await Context.Series.HasReleaseYear(true, FilterComparison.Equal, null).ToListAsync();
|
||||
Assert.Equal(3, foundSeries.Count);
|
||||
}
|
||||
|
||||
|
@ -889,10 +889,10 @@ public class SeriesFilterTests : AbstractDbTest
|
|||
.Build())
|
||||
.Build();
|
||||
|
||||
_context.Library.Add(library);
|
||||
await _context.SaveChangesAsync();
|
||||
Context.Library.Add(library);
|
||||
await Context.SaveChangesAsync();
|
||||
|
||||
var foundSeries = await _context.Series.HasReleaseYear(true, FilterComparison.IsEmpty, 0).ToListAsync();
|
||||
var foundSeries = await Context.Series.HasReleaseYear(true, FilterComparison.IsEmpty, 0).ToListAsync();
|
||||
Assert.Single(foundSeries);
|
||||
Assert.Equal("EmptyYear", foundSeries[0].Name);
|
||||
}
|
||||
|
@ -925,30 +925,26 @@ public class SeriesFilterTests : AbstractDbTest
|
|||
.WithLibrary(library)
|
||||
.Build();
|
||||
|
||||
_context.Users.Add(user);
|
||||
_context.Library.Add(library);
|
||||
await _context.SaveChangesAsync();
|
||||
Context.Users.Add(user);
|
||||
Context.Library.Add(library);
|
||||
await Context.SaveChangesAsync();
|
||||
|
||||
|
||||
var seriesService = new SeriesService(_unitOfWork, Substitute.For<IEventHub>(),
|
||||
Substitute.For<ITaskScheduler>(), Substitute.For<ILogger<SeriesService>>(),
|
||||
Substitute.For<IScrobblingService>(), Substitute.For<ILocalizationService>(),
|
||||
Substitute.For<IReadingListService>());
|
||||
var ratingService = new RatingService(UnitOfWork, Substitute.For<IScrobblingService>(), Substitute.For<ILogger<RatingService>>());
|
||||
|
||||
// Select 0 Rating
|
||||
var zeroRating = await _unitOfWork.SeriesRepository.GetSeriesByIdAsync(2);
|
||||
var zeroRating = await UnitOfWork.SeriesRepository.GetSeriesByIdAsync(2);
|
||||
Assert.NotNull(zeroRating);
|
||||
|
||||
Assert.True(await seriesService.UpdateRating(user, new UpdateSeriesRatingDto()
|
||||
Assert.True(await ratingService.UpdateSeriesRating(user, new UpdateRatingDto()
|
||||
{
|
||||
SeriesId = zeroRating.Id,
|
||||
UserRating = 0
|
||||
}));
|
||||
|
||||
// Select 4.5 Rating
|
||||
var partialRating = await _unitOfWork.SeriesRepository.GetSeriesByIdAsync(3);
|
||||
var partialRating = await UnitOfWork.SeriesRepository.GetSeriesByIdAsync(3);
|
||||
|
||||
Assert.True(await seriesService.UpdateRating(user, new UpdateSeriesRatingDto()
|
||||
Assert.True(await ratingService.UpdateSeriesRating(user, new UpdateRatingDto()
|
||||
{
|
||||
SeriesId = partialRating.Id,
|
||||
UserRating = 4.5f
|
||||
|
@ -962,7 +958,7 @@ public class SeriesFilterTests : AbstractDbTest
|
|||
{
|
||||
var user = await SetupHasRating();
|
||||
|
||||
var foundSeries = await _context.Series
|
||||
var foundSeries = await Context.Series
|
||||
.HasRating(true, FilterComparison.Equal, 4.5f, user.Id)
|
||||
.ToListAsync();
|
||||
|
||||
|
@ -975,7 +971,7 @@ public class SeriesFilterTests : AbstractDbTest
|
|||
{
|
||||
var user = await SetupHasRating();
|
||||
|
||||
var foundSeries = await _context.Series
|
||||
var foundSeries = await Context.Series
|
||||
.HasRating(true, FilterComparison.GreaterThan, 0, user.Id)
|
||||
.ToListAsync();
|
||||
|
||||
|
@ -988,7 +984,7 @@ public class SeriesFilterTests : AbstractDbTest
|
|||
{
|
||||
var user = await SetupHasRating();
|
||||
|
||||
var foundSeries = await _context.Series
|
||||
var foundSeries = await Context.Series
|
||||
.HasRating(true, FilterComparison.LessThan, 4.5f, user.Id)
|
||||
.ToListAsync();
|
||||
|
||||
|
@ -1001,7 +997,7 @@ public class SeriesFilterTests : AbstractDbTest
|
|||
{
|
||||
var user = await SetupHasRating();
|
||||
|
||||
var foundSeries = await _context.Series
|
||||
var foundSeries = await Context.Series
|
||||
.HasRating(true, FilterComparison.IsEmpty, 0, user.Id)
|
||||
.ToListAsync();
|
||||
|
||||
|
@ -1014,7 +1010,7 @@ public class SeriesFilterTests : AbstractDbTest
|
|||
{
|
||||
var user = await SetupHasRating();
|
||||
|
||||
var foundSeries = await _context.Series
|
||||
var foundSeries = await Context.Series
|
||||
.HasRating(true, FilterComparison.GreaterThanEqual, 4.5f, user.Id)
|
||||
.ToListAsync();
|
||||
|
||||
|
@ -1027,7 +1023,7 @@ public class SeriesFilterTests : AbstractDbTest
|
|||
{
|
||||
var user = await SetupHasRating();
|
||||
|
||||
var foundSeries = await _context.Series
|
||||
var foundSeries = await Context.Series
|
||||
.HasRating(true, FilterComparison.LessThanEqual, 0, user.Id)
|
||||
.ToListAsync();
|
||||
|
||||
|
@ -1105,9 +1101,9 @@ public class SeriesFilterTests : AbstractDbTest
|
|||
.WithLibrary(library)
|
||||
.Build();
|
||||
|
||||
_context.Users.Add(user);
|
||||
_context.Library.Add(library);
|
||||
await _context.SaveChangesAsync();
|
||||
Context.Users.Add(user);
|
||||
Context.Library.Add(library);
|
||||
await Context.SaveChangesAsync();
|
||||
|
||||
return user;
|
||||
}
|
||||
|
@ -1117,7 +1113,7 @@ public class SeriesFilterTests : AbstractDbTest
|
|||
{
|
||||
await SetupHasName();
|
||||
|
||||
var foundSeries = await _context.Series
|
||||
var foundSeries = await Context.Series
|
||||
.HasName(true, FilterComparison.Equal, "My Dress-Up Darling")
|
||||
.ToListAsync();
|
||||
|
||||
|
@ -1130,7 +1126,7 @@ public class SeriesFilterTests : AbstractDbTest
|
|||
{
|
||||
await SetupHasName();
|
||||
|
||||
var foundSeries = await _context.Series
|
||||
var foundSeries = await Context.Series
|
||||
.HasName(true, FilterComparison.Equal, "Ijiranaide, Nagatoro-san")
|
||||
.ToListAsync();
|
||||
|
||||
|
@ -1143,7 +1139,7 @@ public class SeriesFilterTests : AbstractDbTest
|
|||
{
|
||||
await SetupHasName();
|
||||
|
||||
var foundSeries = await _context.Series
|
||||
var foundSeries = await Context.Series
|
||||
.HasName(true, FilterComparison.BeginsWith, "My Dress")
|
||||
.ToListAsync();
|
||||
|
||||
|
@ -1156,7 +1152,7 @@ public class SeriesFilterTests : AbstractDbTest
|
|||
{
|
||||
await SetupHasName();
|
||||
|
||||
var foundSeries = await _context.Series
|
||||
var foundSeries = await Context.Series
|
||||
.HasName(true, FilterComparison.BeginsWith, "Sono Bisque")
|
||||
.ToListAsync();
|
||||
|
||||
|
@ -1169,7 +1165,7 @@ public class SeriesFilterTests : AbstractDbTest
|
|||
{
|
||||
await SetupHasName();
|
||||
|
||||
var foundSeries = await _context.Series
|
||||
var foundSeries = await Context.Series
|
||||
.HasName(true, FilterComparison.EndsWith, "Nagatoro")
|
||||
.ToListAsync();
|
||||
|
||||
|
@ -1182,7 +1178,7 @@ public class SeriesFilterTests : AbstractDbTest
|
|||
{
|
||||
await SetupHasName();
|
||||
|
||||
var foundSeries = await _context.Series
|
||||
var foundSeries = await Context.Series
|
||||
.HasName(true, FilterComparison.Matches, "Toy With Me")
|
||||
.ToListAsync();
|
||||
|
||||
|
@ -1195,7 +1191,7 @@ public class SeriesFilterTests : AbstractDbTest
|
|||
{
|
||||
await SetupHasName();
|
||||
|
||||
var foundSeries = await _context.Series
|
||||
var foundSeries = await Context.Series
|
||||
.HasName(true, FilterComparison.NotEqual, "My Dress-Up Darling")
|
||||
.ToListAsync();
|
||||
|
||||
|
@ -1239,9 +1235,9 @@ public class SeriesFilterTests : AbstractDbTest
|
|||
.WithLibrary(library)
|
||||
.Build();
|
||||
|
||||
_context.Users.Add(user);
|
||||
_context.Library.Add(library);
|
||||
await _context.SaveChangesAsync();
|
||||
Context.Users.Add(user);
|
||||
Context.Library.Add(library);
|
||||
await Context.SaveChangesAsync();
|
||||
|
||||
return user;
|
||||
}
|
||||
|
@ -1251,7 +1247,7 @@ public class SeriesFilterTests : AbstractDbTest
|
|||
{
|
||||
await SetupHasSummary();
|
||||
|
||||
var foundSeries = await _context.Series
|
||||
var foundSeries = await Context.Series
|
||||
.HasSummary(true, FilterComparison.Equal, "I like hippos")
|
||||
.ToListAsync();
|
||||
|
||||
|
@ -1264,7 +1260,7 @@ public class SeriesFilterTests : AbstractDbTest
|
|||
{
|
||||
await SetupHasSummary();
|
||||
|
||||
var foundSeries = await _context.Series
|
||||
var foundSeries = await Context.Series
|
||||
.HasSummary(true, FilterComparison.BeginsWith, "I like h")
|
||||
.ToListAsync();
|
||||
|
||||
|
@ -1277,7 +1273,7 @@ public class SeriesFilterTests : AbstractDbTest
|
|||
{
|
||||
await SetupHasSummary();
|
||||
|
||||
var foundSeries = await _context.Series
|
||||
var foundSeries = await Context.Series
|
||||
.HasSummary(true, FilterComparison.EndsWith, "apples")
|
||||
.ToListAsync();
|
||||
|
||||
|
@ -1290,7 +1286,7 @@ public class SeriesFilterTests : AbstractDbTest
|
|||
{
|
||||
await SetupHasSummary();
|
||||
|
||||
var foundSeries = await _context.Series
|
||||
var foundSeries = await Context.Series
|
||||
.HasSummary(true, FilterComparison.Matches, "like ducks")
|
||||
.ToListAsync();
|
||||
|
||||
|
@ -1303,7 +1299,7 @@ public class SeriesFilterTests : AbstractDbTest
|
|||
{
|
||||
await SetupHasSummary();
|
||||
|
||||
var foundSeries = await _context.Series
|
||||
var foundSeries = await Context.Series
|
||||
.HasSummary(true, FilterComparison.NotEqual, "I like ducks")
|
||||
.ToListAsync();
|
||||
|
||||
|
@ -1316,7 +1312,7 @@ public class SeriesFilterTests : AbstractDbTest
|
|||
{
|
||||
await SetupHasSummary();
|
||||
|
||||
var foundSeries = await _context.Series
|
||||
var foundSeries = await Context.Series
|
||||
.HasSummary(true, FilterComparison.IsEmpty, string.Empty)
|
||||
.ToListAsync();
|
||||
|
||||
|
|
|
@ -7,8 +7,8 @@ public class PersonHelperTests : AbstractDbTest
|
|||
{
|
||||
protected override async Task ResetDb()
|
||||
{
|
||||
_context.Series.RemoveRange(_context.Series.ToList());
|
||||
await _context.SaveChangesAsync();
|
||||
Context.Series.RemoveRange(Context.Series.ToList());
|
||||
await Context.SaveChangesAsync();
|
||||
}
|
||||
//
|
||||
// // 1. Test adding new people and keeping existing ones
|
||||
|
|
|
@ -29,11 +29,11 @@ public class CleanupServiceTests : AbstractDbTest
|
|||
|
||||
public CleanupServiceTests() : base()
|
||||
{
|
||||
_context.Library.Add(new LibraryBuilder("Manga")
|
||||
Context.Library.Add(new LibraryBuilder("Manga")
|
||||
.WithFolderPath(new FolderPathBuilder(Root + "data/").Build())
|
||||
.Build());
|
||||
|
||||
_readerService = new ReaderService(_unitOfWork, Substitute.For<ILogger<ReaderService>>(), Substitute.For<IEventHub>(),
|
||||
_readerService = new ReaderService(UnitOfWork, Substitute.For<ILogger<ReaderService>>(), Substitute.For<IEventHub>(),
|
||||
Substitute.For<IImageService>(),
|
||||
new DirectoryService(Substitute.For<ILogger<DirectoryService>>(), new MockFileSystem()), Substitute.For<IScrobblingService>());
|
||||
}
|
||||
|
@ -43,11 +43,11 @@ public class CleanupServiceTests : AbstractDbTest
|
|||
|
||||
protected override async Task ResetDb()
|
||||
{
|
||||
_context.Series.RemoveRange(_context.Series.ToList());
|
||||
_context.Users.RemoveRange(_context.Users.ToList());
|
||||
_context.AppUserBookmark.RemoveRange(_context.AppUserBookmark.ToList());
|
||||
Context.Series.RemoveRange(Context.Series.ToList());
|
||||
Context.Users.RemoveRange(Context.Users.ToList());
|
||||
Context.AppUserBookmark.RemoveRange(Context.AppUserBookmark.ToList());
|
||||
|
||||
await _context.SaveChangesAsync();
|
||||
await Context.SaveChangesAsync();
|
||||
}
|
||||
|
||||
#endregion
|
||||
|
@ -68,18 +68,18 @@ public class CleanupServiceTests : AbstractDbTest
|
|||
var s = new SeriesBuilder("Test 1").Build();
|
||||
s.CoverImage = $"{ImageService.GetSeriesFormat(1)}.jpg";
|
||||
s.LibraryId = 1;
|
||||
_context.Series.Add(s);
|
||||
Context.Series.Add(s);
|
||||
s = new SeriesBuilder("Test 2").Build();
|
||||
s.CoverImage = $"{ImageService.GetSeriesFormat(3)}.jpg";
|
||||
s.LibraryId = 1;
|
||||
_context.Series.Add(s);
|
||||
Context.Series.Add(s);
|
||||
s = new SeriesBuilder("Test 3").Build();
|
||||
s.CoverImage = $"{ImageService.GetSeriesFormat(1000)}.jpg";
|
||||
s.LibraryId = 1;
|
||||
_context.Series.Add(s);
|
||||
Context.Series.Add(s);
|
||||
|
||||
var ds = new DirectoryService(Substitute.For<ILogger<DirectoryService>>(), filesystem);
|
||||
var cleanupService = new CleanupService(_logger, _unitOfWork, _messageHub,
|
||||
var cleanupService = new CleanupService(_logger, UnitOfWork, _messageHub,
|
||||
ds);
|
||||
|
||||
await cleanupService.DeleteSeriesCoverImages();
|
||||
|
@ -102,16 +102,16 @@ public class CleanupServiceTests : AbstractDbTest
|
|||
var s = new SeriesBuilder("Test 1").Build();
|
||||
s.CoverImage = $"{ImageService.GetSeriesFormat(1)}.jpg";
|
||||
s.LibraryId = 1;
|
||||
_context.Series.Add(s);
|
||||
Context.Series.Add(s);
|
||||
s = new SeriesBuilder("Test 2").Build();
|
||||
s.CoverImage = $"{ImageService.GetSeriesFormat(3)}.jpg";
|
||||
s.LibraryId = 1;
|
||||
_context.Series.Add(s);
|
||||
Context.Series.Add(s);
|
||||
|
||||
|
||||
await _context.SaveChangesAsync();
|
||||
await Context.SaveChangesAsync();
|
||||
var ds = new DirectoryService(Substitute.For<ILogger<DirectoryService>>(), filesystem);
|
||||
var cleanupService = new CleanupService(_logger, _unitOfWork, _messageHub,
|
||||
var cleanupService = new CleanupService(_logger, UnitOfWork, _messageHub,
|
||||
ds);
|
||||
|
||||
await cleanupService.DeleteSeriesCoverImages();
|
||||
|
@ -133,7 +133,7 @@ public class CleanupServiceTests : AbstractDbTest
|
|||
await ResetDb();
|
||||
|
||||
// Add 2 series with cover images
|
||||
_context.Series.Add(new SeriesBuilder("Test 1")
|
||||
Context.Series.Add(new SeriesBuilder("Test 1")
|
||||
.WithVolume(new VolumeBuilder("1")
|
||||
.WithChapter(new ChapterBuilder(API.Services.Tasks.Scanner.Parser.Parser.DefaultChapter).WithCoverImage("v01_c01.jpg").Build())
|
||||
.WithCoverImage("v01_c01.jpg")
|
||||
|
@ -142,7 +142,7 @@ public class CleanupServiceTests : AbstractDbTest
|
|||
.WithLibraryId(1)
|
||||
.Build());
|
||||
|
||||
_context.Series.Add(new SeriesBuilder("Test 2")
|
||||
Context.Series.Add(new SeriesBuilder("Test 2")
|
||||
.WithVolume(new VolumeBuilder("1")
|
||||
.WithChapter(new ChapterBuilder(API.Services.Tasks.Scanner.Parser.Parser.DefaultChapter).WithCoverImage("v01_c03.jpg").Build())
|
||||
.WithCoverImage("v01_c03.jpg")
|
||||
|
@ -152,9 +152,9 @@ public class CleanupServiceTests : AbstractDbTest
|
|||
.Build());
|
||||
|
||||
|
||||
await _context.SaveChangesAsync();
|
||||
await Context.SaveChangesAsync();
|
||||
var ds = new DirectoryService(Substitute.For<ILogger<DirectoryService>>(), filesystem);
|
||||
var cleanupService = new CleanupService(_logger, _unitOfWork, _messageHub,
|
||||
var cleanupService = new CleanupService(_logger, UnitOfWork, _messageHub,
|
||||
ds);
|
||||
|
||||
await cleanupService.DeleteChapterCoverImages();
|
||||
|
@ -223,7 +223,7 @@ public class CleanupServiceTests : AbstractDbTest
|
|||
// Delete all Series to reset state
|
||||
await ResetDb();
|
||||
|
||||
_context.Users.Add(new AppUser()
|
||||
Context.Users.Add(new AppUser()
|
||||
{
|
||||
UserName = "Joe",
|
||||
ReadingLists = new List<ReadingList>()
|
||||
|
@ -239,9 +239,9 @@ public class CleanupServiceTests : AbstractDbTest
|
|||
}
|
||||
});
|
||||
|
||||
await _context.SaveChangesAsync();
|
||||
await Context.SaveChangesAsync();
|
||||
var ds = new DirectoryService(Substitute.For<ILogger<DirectoryService>>(), filesystem);
|
||||
var cleanupService = new CleanupService(_logger, _unitOfWork, _messageHub,
|
||||
var cleanupService = new CleanupService(_logger, UnitOfWork, _messageHub,
|
||||
ds);
|
||||
|
||||
await cleanupService.DeleteReadingListCoverImages();
|
||||
|
@ -260,7 +260,7 @@ public class CleanupServiceTests : AbstractDbTest
|
|||
filesystem.AddFile($"{CacheDirectory}02.jpg", new MockFileData(""));
|
||||
|
||||
var ds = new DirectoryService(Substitute.For<ILogger<DirectoryService>>(), filesystem);
|
||||
var cleanupService = new CleanupService(_logger, _unitOfWork, _messageHub,
|
||||
var cleanupService = new CleanupService(_logger, UnitOfWork, _messageHub,
|
||||
ds);
|
||||
cleanupService.CleanupCacheAndTempDirectories();
|
||||
Assert.Empty(ds.GetFiles(CacheDirectory, searchOption: SearchOption.AllDirectories));
|
||||
|
@ -274,7 +274,7 @@ public class CleanupServiceTests : AbstractDbTest
|
|||
filesystem.AddFile($"{CacheDirectory}subdir/02.jpg", new MockFileData(""));
|
||||
|
||||
var ds = new DirectoryService(Substitute.For<ILogger<DirectoryService>>(), filesystem);
|
||||
var cleanupService = new CleanupService(_logger, _unitOfWork, _messageHub,
|
||||
var cleanupService = new CleanupService(_logger, UnitOfWork, _messageHub,
|
||||
ds);
|
||||
cleanupService.CleanupCacheAndTempDirectories();
|
||||
Assert.Empty(ds.GetFiles(CacheDirectory, searchOption: SearchOption.AllDirectories));
|
||||
|
@ -297,7 +297,7 @@ public class CleanupServiceTests : AbstractDbTest
|
|||
filesystem.AddFile($"{BackupDirectory}randomfile.zip", filesystemFile);
|
||||
|
||||
var ds = new DirectoryService(Substitute.For<ILogger<DirectoryService>>(), filesystem);
|
||||
var cleanupService = new CleanupService(_logger, _unitOfWork, _messageHub,
|
||||
var cleanupService = new CleanupService(_logger, UnitOfWork, _messageHub,
|
||||
ds);
|
||||
await cleanupService.CleanupBackups();
|
||||
Assert.Single(ds.GetFiles(BackupDirectory, searchOption: SearchOption.AllDirectories));
|
||||
|
@ -319,7 +319,7 @@ public class CleanupServiceTests : AbstractDbTest
|
|||
});
|
||||
|
||||
var ds = new DirectoryService(Substitute.For<ILogger<DirectoryService>>(), filesystem);
|
||||
var cleanupService = new CleanupService(_logger, _unitOfWork, _messageHub,
|
||||
var cleanupService = new CleanupService(_logger, UnitOfWork, _messageHub,
|
||||
ds);
|
||||
await cleanupService.CleanupBackups();
|
||||
Assert.True(filesystem.File.Exists($"{BackupDirectory}randomfile.zip"));
|
||||
|
@ -343,7 +343,7 @@ public class CleanupServiceTests : AbstractDbTest
|
|||
}
|
||||
|
||||
var ds = new DirectoryService(Substitute.For<ILogger<DirectoryService>>(), filesystem);
|
||||
var cleanupService = new CleanupService(_logger, _unitOfWork, _messageHub,
|
||||
var cleanupService = new CleanupService(_logger, UnitOfWork, _messageHub,
|
||||
ds);
|
||||
await cleanupService.CleanupLogs();
|
||||
Assert.Single(ds.GetFiles(LogDirectory, searchOption: SearchOption.AllDirectories));
|
||||
|
@ -372,7 +372,7 @@ public class CleanupServiceTests : AbstractDbTest
|
|||
|
||||
|
||||
var ds = new DirectoryService(Substitute.For<ILogger<DirectoryService>>(), filesystem);
|
||||
var cleanupService = new CleanupService(_logger, _unitOfWork, _messageHub,
|
||||
var cleanupService = new CleanupService(_logger, UnitOfWork, _messageHub,
|
||||
ds);
|
||||
await cleanupService.CleanupLogs();
|
||||
Assert.True(filesystem.File.Exists($"{LogDirectory}kavita20200911.log"));
|
||||
|
@ -396,36 +396,36 @@ public class CleanupServiceTests : AbstractDbTest
|
|||
.Build();
|
||||
series.Library = new LibraryBuilder("Test LIb").Build();
|
||||
|
||||
_context.Series.Add(series);
|
||||
Context.Series.Add(series);
|
||||
|
||||
|
||||
_context.AppUser.Add(new AppUser()
|
||||
Context.AppUser.Add(new AppUser()
|
||||
{
|
||||
UserName = "majora2007"
|
||||
});
|
||||
|
||||
await _context.SaveChangesAsync();
|
||||
await Context.SaveChangesAsync();
|
||||
|
||||
var user = await _unitOfWork.UserRepository.GetUserByUsernameAsync("majora2007", AppUserIncludes.Progress);
|
||||
var user = await UnitOfWork.UserRepository.GetUserByUsernameAsync("majora2007", AppUserIncludes.Progress);
|
||||
await _readerService.MarkChaptersUntilAsRead(user, 1, 5);
|
||||
await _context.SaveChangesAsync();
|
||||
await Context.SaveChangesAsync();
|
||||
|
||||
// Validate correct chapters have read status
|
||||
Assert.Equal(1, (await _unitOfWork.AppUserProgressRepository.GetUserProgressAsync(1, 1)).PagesRead);
|
||||
Assert.Equal(1, (await UnitOfWork.AppUserProgressRepository.GetUserProgressAsync(1, 1)).PagesRead);
|
||||
|
||||
var cleanupService = new CleanupService(Substitute.For<ILogger<CleanupService>>(), _unitOfWork,
|
||||
var cleanupService = new CleanupService(Substitute.For<ILogger<CleanupService>>(), UnitOfWork,
|
||||
Substitute.For<IEventHub>(),
|
||||
new DirectoryService(Substitute.For<ILogger<DirectoryService>>(), new MockFileSystem()));
|
||||
|
||||
// Delete the Chapter
|
||||
_context.Chapter.Remove(c);
|
||||
await _unitOfWork.CommitAsync();
|
||||
Assert.Empty(await _unitOfWork.AppUserProgressRepository.GetUserProgressForSeriesAsync(1, 1));
|
||||
Context.Chapter.Remove(c);
|
||||
await UnitOfWork.CommitAsync();
|
||||
Assert.Empty(await UnitOfWork.AppUserProgressRepository.GetUserProgressForSeriesAsync(1, 1));
|
||||
|
||||
// NOTE: This may not be needed, the underlying DB structure seems fixed as of v0.7
|
||||
await cleanupService.CleanupDbEntries();
|
||||
|
||||
Assert.Empty(await _unitOfWork.AppUserProgressRepository.GetUserProgressForSeriesAsync(1, 1));
|
||||
Assert.Empty(await UnitOfWork.AppUserProgressRepository.GetUserProgressForSeriesAsync(1, 1));
|
||||
}
|
||||
|
||||
[Fact]
|
||||
|
@ -436,7 +436,7 @@ public class CleanupServiceTests : AbstractDbTest
|
|||
.WithMetadata(new SeriesMetadataBuilder().Build())
|
||||
.Build();
|
||||
s.Library = new LibraryBuilder("Test LIb").Build();
|
||||
_context.Series.Add(s);
|
||||
Context.Series.Add(s);
|
||||
|
||||
var c = new AppUserCollection()
|
||||
{
|
||||
|
@ -446,24 +446,24 @@ public class CleanupServiceTests : AbstractDbTest
|
|||
Items = new List<Series>() {s}
|
||||
};
|
||||
|
||||
_context.AppUser.Add(new AppUser()
|
||||
Context.AppUser.Add(new AppUser()
|
||||
{
|
||||
UserName = "majora2007",
|
||||
Collections = new List<AppUserCollection>() {c}
|
||||
});
|
||||
await _context.SaveChangesAsync();
|
||||
await Context.SaveChangesAsync();
|
||||
|
||||
var cleanupService = new CleanupService(Substitute.For<ILogger<CleanupService>>(), _unitOfWork,
|
||||
var cleanupService = new CleanupService(Substitute.For<ILogger<CleanupService>>(), UnitOfWork,
|
||||
Substitute.For<IEventHub>(),
|
||||
new DirectoryService(Substitute.For<ILogger<DirectoryService>>(), new MockFileSystem()));
|
||||
|
||||
// Delete the Chapter
|
||||
_context.Series.Remove(s);
|
||||
await _unitOfWork.CommitAsync();
|
||||
Context.Series.Remove(s);
|
||||
await UnitOfWork.CommitAsync();
|
||||
|
||||
await cleanupService.CleanupDbEntries();
|
||||
|
||||
Assert.Empty(await _unitOfWork.CollectionTagRepository.GetAllCollectionsAsync());
|
||||
Assert.Empty(await UnitOfWork.CollectionTagRepository.GetAllCollectionsAsync());
|
||||
}
|
||||
|
||||
#endregion
|
||||
|
@ -480,15 +480,15 @@ public class CleanupServiceTests : AbstractDbTest
|
|||
.Build();
|
||||
|
||||
s.Library = new LibraryBuilder("Test LIb").Build();
|
||||
_context.Series.Add(s);
|
||||
Context.Series.Add(s);
|
||||
|
||||
var user = new AppUser()
|
||||
{
|
||||
UserName = "CleanupWantToRead_ShouldRemoveFullyReadSeries",
|
||||
};
|
||||
_context.AppUser.Add(user);
|
||||
Context.AppUser.Add(user);
|
||||
|
||||
await _unitOfWork.CommitAsync();
|
||||
await UnitOfWork.CommitAsync();
|
||||
|
||||
// Add want to read
|
||||
user.WantToRead = new List<AppUserWantToRead>()
|
||||
|
@ -498,12 +498,12 @@ public class CleanupServiceTests : AbstractDbTest
|
|||
SeriesId = s.Id
|
||||
}
|
||||
};
|
||||
await _unitOfWork.CommitAsync();
|
||||
await UnitOfWork.CommitAsync();
|
||||
|
||||
await _readerService.MarkSeriesAsRead(user, s.Id);
|
||||
await _unitOfWork.CommitAsync();
|
||||
await UnitOfWork.CommitAsync();
|
||||
|
||||
var cleanupService = new CleanupService(Substitute.For<ILogger<CleanupService>>(), _unitOfWork,
|
||||
var cleanupService = new CleanupService(Substitute.For<ILogger<CleanupService>>(), UnitOfWork,
|
||||
Substitute.For<IEventHub>(),
|
||||
new DirectoryService(Substitute.For<ILogger<DirectoryService>>(), new MockFileSystem()));
|
||||
|
||||
|
@ -511,7 +511,7 @@ public class CleanupServiceTests : AbstractDbTest
|
|||
await cleanupService.CleanupWantToRead();
|
||||
|
||||
var wantToRead =
|
||||
await _unitOfWork.SeriesRepository.GetWantToReadForUserAsync(user.Id, new UserParams(), new FilterDto());
|
||||
await UnitOfWork.SeriesRepository.GetWantToReadForUserAsync(user.Id, new UserParams(), new FilterDto());
|
||||
|
||||
Assert.Equal(0, wantToRead.TotalCount);
|
||||
}
|
||||
|
@ -533,15 +533,15 @@ public class CleanupServiceTests : AbstractDbTest
|
|||
.Build();
|
||||
|
||||
s.Library = new LibraryBuilder("Test Lib").Build();
|
||||
_context.Series.Add(s);
|
||||
Context.Series.Add(s);
|
||||
|
||||
var user = new AppUser()
|
||||
{
|
||||
UserName = "ConsolidateProgress_ShouldRemoveDuplicates",
|
||||
};
|
||||
_context.AppUser.Add(user);
|
||||
Context.AppUser.Add(user);
|
||||
|
||||
await _unitOfWork.CommitAsync();
|
||||
await UnitOfWork.CommitAsync();
|
||||
|
||||
// Add 2 progress events
|
||||
user.Progresses ??= [];
|
||||
|
@ -553,7 +553,7 @@ public class CleanupServiceTests : AbstractDbTest
|
|||
LibraryId = s.LibraryId,
|
||||
PagesRead = 1,
|
||||
});
|
||||
await _unitOfWork.CommitAsync();
|
||||
await UnitOfWork.CommitAsync();
|
||||
|
||||
// Add a duplicate with higher page number
|
||||
user.Progresses.Add(new AppUserProgress()
|
||||
|
@ -564,18 +564,18 @@ public class CleanupServiceTests : AbstractDbTest
|
|||
LibraryId = s.LibraryId,
|
||||
PagesRead = 3,
|
||||
});
|
||||
await _unitOfWork.CommitAsync();
|
||||
await UnitOfWork.CommitAsync();
|
||||
|
||||
Assert.Equal(2, (await _unitOfWork.AppUserProgressRepository.GetAllProgress()).Count());
|
||||
Assert.Equal(2, (await UnitOfWork.AppUserProgressRepository.GetAllProgress()).Count());
|
||||
|
||||
var cleanupService = new CleanupService(Substitute.For<ILogger<CleanupService>>(), _unitOfWork,
|
||||
var cleanupService = new CleanupService(Substitute.For<ILogger<CleanupService>>(), UnitOfWork,
|
||||
Substitute.For<IEventHub>(),
|
||||
new DirectoryService(Substitute.For<ILogger<DirectoryService>>(), new MockFileSystem()));
|
||||
|
||||
|
||||
await cleanupService.ConsolidateProgress();
|
||||
|
||||
var progress = await _unitOfWork.AppUserProgressRepository.GetAllProgress();
|
||||
var progress = await UnitOfWork.AppUserProgressRepository.GetAllProgress();
|
||||
|
||||
Assert.Single(progress);
|
||||
Assert.True(progress.First().PagesRead == 3);
|
||||
|
@ -601,50 +601,50 @@ public class CleanupServiceTests : AbstractDbTest
|
|||
{
|
||||
new VolumeBuilder(API.Services.Tasks.Scanner.Parser.Parser.LooseLeafVolume).WithChapter(c).Build()
|
||||
};
|
||||
_context.Series.Add(s);
|
||||
Context.Series.Add(s);
|
||||
|
||||
var user = new AppUser()
|
||||
{
|
||||
UserName = "EnsureChapterProgressIsCapped",
|
||||
Progresses = new List<AppUserProgress>()
|
||||
};
|
||||
_context.AppUser.Add(user);
|
||||
Context.AppUser.Add(user);
|
||||
|
||||
await _unitOfWork.CommitAsync();
|
||||
await UnitOfWork.CommitAsync();
|
||||
|
||||
await _readerService.MarkChaptersAsRead(user, s.Id, new List<Chapter>() {c});
|
||||
await _unitOfWork.CommitAsync();
|
||||
await UnitOfWork.CommitAsync();
|
||||
|
||||
var chapter = await _unitOfWork.ChapterRepository.GetChapterDtoAsync(c.Id);
|
||||
await _unitOfWork.ChapterRepository.AddChapterModifiers(user.Id, chapter);
|
||||
var chapter = await UnitOfWork.ChapterRepository.GetChapterDtoAsync(c.Id);
|
||||
await UnitOfWork.ChapterRepository.AddChapterModifiers(user.Id, chapter);
|
||||
|
||||
Assert.NotNull(chapter);
|
||||
Assert.Equal(2, chapter.PagesRead);
|
||||
|
||||
// Update chapter to have 1 page
|
||||
c.Pages = 1;
|
||||
_unitOfWork.ChapterRepository.Update(c);
|
||||
await _unitOfWork.CommitAsync();
|
||||
UnitOfWork.ChapterRepository.Update(c);
|
||||
await UnitOfWork.CommitAsync();
|
||||
|
||||
chapter = await _unitOfWork.ChapterRepository.GetChapterDtoAsync(c.Id);
|
||||
await _unitOfWork.ChapterRepository.AddChapterModifiers(user.Id, chapter);
|
||||
chapter = await UnitOfWork.ChapterRepository.GetChapterDtoAsync(c.Id);
|
||||
await UnitOfWork.ChapterRepository.AddChapterModifiers(user.Id, chapter);
|
||||
Assert.NotNull(chapter);
|
||||
Assert.Equal(2, chapter.PagesRead);
|
||||
Assert.Equal(1, chapter.Pages);
|
||||
|
||||
var cleanupService = new CleanupService(Substitute.For<ILogger<CleanupService>>(), _unitOfWork,
|
||||
var cleanupService = new CleanupService(Substitute.For<ILogger<CleanupService>>(), UnitOfWork,
|
||||
Substitute.For<IEventHub>(),
|
||||
new DirectoryService(Substitute.For<ILogger<DirectoryService>>(), new MockFileSystem()));
|
||||
|
||||
await cleanupService.EnsureChapterProgressIsCapped();
|
||||
chapter = await _unitOfWork.ChapterRepository.GetChapterDtoAsync(c.Id);
|
||||
await _unitOfWork.ChapterRepository.AddChapterModifiers(user.Id, chapter);
|
||||
chapter = await UnitOfWork.ChapterRepository.GetChapterDtoAsync(c.Id);
|
||||
await UnitOfWork.ChapterRepository.AddChapterModifiers(user.Id, chapter);
|
||||
|
||||
Assert.NotNull(chapter);
|
||||
Assert.Equal(1, chapter.PagesRead);
|
||||
|
||||
_context.AppUser.Remove(user);
|
||||
await _unitOfWork.CommitAsync();
|
||||
Context.AppUser.Remove(user);
|
||||
await UnitOfWork.CommitAsync();
|
||||
}
|
||||
#endregion
|
||||
|
||||
|
|
|
@ -23,24 +23,24 @@ public class CollectionTagServiceTests : AbstractDbTest
|
|||
private readonly ICollectionTagService _service;
|
||||
public CollectionTagServiceTests()
|
||||
{
|
||||
_service = new CollectionTagService(_unitOfWork, Substitute.For<IEventHub>());
|
||||
_service = new CollectionTagService(UnitOfWork, Substitute.For<IEventHub>());
|
||||
}
|
||||
|
||||
protected override async Task ResetDb()
|
||||
{
|
||||
_context.AppUserCollection.RemoveRange(_context.AppUserCollection.ToList());
|
||||
_context.Library.RemoveRange(_context.Library.ToList());
|
||||
Context.AppUserCollection.RemoveRange(Context.AppUserCollection.ToList());
|
||||
Context.Library.RemoveRange(Context.Library.ToList());
|
||||
|
||||
await _unitOfWork.CommitAsync();
|
||||
await UnitOfWork.CommitAsync();
|
||||
}
|
||||
|
||||
private async Task SeedSeries()
|
||||
{
|
||||
if (_context.AppUserCollection.Any()) return;
|
||||
if (Context.AppUserCollection.Any()) return;
|
||||
|
||||
var s1 = new SeriesBuilder("Series 1").WithMetadata(new SeriesMetadataBuilder().WithAgeRating(AgeRating.Mature).Build()).Build();
|
||||
var s2 = new SeriesBuilder("Series 2").WithMetadata(new SeriesMetadataBuilder().WithAgeRating(AgeRating.G).Build()).Build();
|
||||
_context.Library.Add(new LibraryBuilder("Library 2", LibraryType.Manga)
|
||||
Context.Library.Add(new LibraryBuilder("Library 2", LibraryType.Manga)
|
||||
.WithSeries(s1)
|
||||
.WithSeries(s2)
|
||||
.Build());
|
||||
|
@ -51,9 +51,9 @@ public class CollectionTagServiceTests : AbstractDbTest
|
|||
new AppUserCollectionBuilder("Tag 1").WithItems(new []{s1}).Build(),
|
||||
new AppUserCollectionBuilder("Tag 2").WithItems(new []{s1, s2}).WithIsPromoted(true).Build()
|
||||
};
|
||||
_unitOfWork.UserRepository.Add(user);
|
||||
UnitOfWork.UserRepository.Add(user);
|
||||
|
||||
await _unitOfWork.CommitAsync();
|
||||
await UnitOfWork.CommitAsync();
|
||||
}
|
||||
|
||||
#region DeleteTag
|
||||
|
@ -64,7 +64,7 @@ public class CollectionTagServiceTests : AbstractDbTest
|
|||
// Arrange
|
||||
await SeedSeries();
|
||||
|
||||
var user = await _unitOfWork.UserRepository.GetUserByIdAsync(1, AppUserIncludes.Collections);
|
||||
var user = await UnitOfWork.UserRepository.GetUserByIdAsync(1, AppUserIncludes.Collections);
|
||||
Assert.NotNull(user);
|
||||
|
||||
// Act
|
||||
|
@ -72,7 +72,7 @@ public class CollectionTagServiceTests : AbstractDbTest
|
|||
|
||||
// Assert
|
||||
Assert.True(result);
|
||||
var deletedTag = await _unitOfWork.CollectionTagRepository.GetCollectionAsync(1);
|
||||
var deletedTag = await UnitOfWork.CollectionTagRepository.GetCollectionAsync(1);
|
||||
Assert.Null(deletedTag);
|
||||
Assert.Single(user.Collections); // Only one collection should remain
|
||||
}
|
||||
|
@ -82,7 +82,7 @@ public class CollectionTagServiceTests : AbstractDbTest
|
|||
{
|
||||
// Arrange
|
||||
await SeedSeries();
|
||||
var user = await _unitOfWork.UserRepository.GetUserByIdAsync(1, AppUserIncludes.Collections);
|
||||
var user = await UnitOfWork.UserRepository.GetUserByIdAsync(1, AppUserIncludes.Collections);
|
||||
Assert.NotNull(user);
|
||||
|
||||
// Act - Try to delete a non-existent tag
|
||||
|
@ -98,7 +98,7 @@ public class CollectionTagServiceTests : AbstractDbTest
|
|||
{
|
||||
// Arrange
|
||||
await SeedSeries();
|
||||
var user = await _unitOfWork.UserRepository.GetUserByIdAsync(1, AppUserIncludes.Collections);
|
||||
var user = await UnitOfWork.UserRepository.GetUserByIdAsync(1, AppUserIncludes.Collections);
|
||||
Assert.NotNull(user);
|
||||
|
||||
// Act
|
||||
|
@ -106,7 +106,7 @@ public class CollectionTagServiceTests : AbstractDbTest
|
|||
|
||||
// Assert
|
||||
Assert.True(result);
|
||||
var remainingTag = await _unitOfWork.CollectionTagRepository.GetCollectionAsync(2);
|
||||
var remainingTag = await UnitOfWork.CollectionTagRepository.GetCollectionAsync(2);
|
||||
Assert.NotNull(remainingTag);
|
||||
Assert.Equal("Tag 2", remainingTag.Title);
|
||||
Assert.True(remainingTag.Promoted);
|
||||
|
@ -121,12 +121,12 @@ public class CollectionTagServiceTests : AbstractDbTest
|
|||
{
|
||||
await SeedSeries();
|
||||
|
||||
var user = await _unitOfWork.UserRepository.GetUserByIdAsync(1, AppUserIncludes.Collections);
|
||||
var user = await UnitOfWork.UserRepository.GetUserByIdAsync(1, AppUserIncludes.Collections);
|
||||
Assert.NotNull(user);
|
||||
|
||||
user.Collections.Add(new AppUserCollectionBuilder("UpdateTag_ShouldUpdateFields").WithIsPromoted(true).Build());
|
||||
_unitOfWork.UserRepository.Update(user);
|
||||
await _unitOfWork.CommitAsync();
|
||||
UnitOfWork.UserRepository.Update(user);
|
||||
await UnitOfWork.CommitAsync();
|
||||
|
||||
await _service.UpdateTag(new AppUserCollectionDto()
|
||||
{
|
||||
|
@ -137,7 +137,7 @@ public class CollectionTagServiceTests : AbstractDbTest
|
|||
AgeRating = AgeRating.Unknown
|
||||
}, 1);
|
||||
|
||||
var tag = await _unitOfWork.CollectionTagRepository.GetCollectionAsync(3);
|
||||
var tag = await UnitOfWork.CollectionTagRepository.GetCollectionAsync(3);
|
||||
Assert.NotNull(tag);
|
||||
Assert.True(tag.Promoted);
|
||||
Assert.False(string.IsNullOrEmpty(tag.Summary));
|
||||
|
@ -151,12 +151,12 @@ public class CollectionTagServiceTests : AbstractDbTest
|
|||
{
|
||||
await SeedSeries();
|
||||
|
||||
var user = await _unitOfWork.UserRepository.GetUserByIdAsync(1, AppUserIncludes.Collections);
|
||||
var user = await UnitOfWork.UserRepository.GetUserByIdAsync(1, AppUserIncludes.Collections);
|
||||
Assert.NotNull(user);
|
||||
|
||||
user.Collections.Add(new AppUserCollectionBuilder("UpdateTag_ShouldNotChangeTitle_WhenNotKavitaSource").WithSource(ScrobbleProvider.Mal).Build());
|
||||
_unitOfWork.UserRepository.Update(user);
|
||||
await _unitOfWork.CommitAsync();
|
||||
UnitOfWork.UserRepository.Update(user);
|
||||
await UnitOfWork.CommitAsync();
|
||||
|
||||
await _service.UpdateTag(new AppUserCollectionDto()
|
||||
{
|
||||
|
@ -167,7 +167,7 @@ public class CollectionTagServiceTests : AbstractDbTest
|
|||
AgeRating = AgeRating.Unknown
|
||||
}, 1);
|
||||
|
||||
var tag = await _unitOfWork.CollectionTagRepository.GetCollectionAsync(3);
|
||||
var tag = await UnitOfWork.CollectionTagRepository.GetCollectionAsync(3);
|
||||
Assert.NotNull(tag);
|
||||
Assert.Equal("UpdateTag_ShouldNotChangeTitle_WhenNotKavitaSource", tag.Title);
|
||||
Assert.False(string.IsNullOrEmpty(tag.Summary));
|
||||
|
@ -198,8 +198,8 @@ public class CollectionTagServiceTests : AbstractDbTest
|
|||
|
||||
// Create a second user
|
||||
var user2 = new AppUserBuilder("user2", "user2", Seed.DefaultThemes.First()).Build();
|
||||
_unitOfWork.UserRepository.Add(user2);
|
||||
await _unitOfWork.CommitAsync();
|
||||
UnitOfWork.UserRepository.Add(user2);
|
||||
await UnitOfWork.CommitAsync();
|
||||
|
||||
// Act & Assert
|
||||
var exception = await Assert.ThrowsAsync<KavitaException>(() => _service.UpdateTag(new AppUserCollectionDto()
|
||||
|
@ -261,7 +261,7 @@ public class CollectionTagServiceTests : AbstractDbTest
|
|||
}, 1);
|
||||
|
||||
// Assert
|
||||
var tag = await _unitOfWork.CollectionTagRepository.GetCollectionAsync(1);
|
||||
var tag = await UnitOfWork.CollectionTagRepository.GetCollectionAsync(1);
|
||||
Assert.NotNull(tag);
|
||||
Assert.True(tag.CoverImageLocked);
|
||||
|
||||
|
@ -273,7 +273,7 @@ public class CollectionTagServiceTests : AbstractDbTest
|
|||
CoverImageLocked = false
|
||||
}, 1);
|
||||
|
||||
tag = await _unitOfWork.CollectionTagRepository.GetCollectionAsync(1);
|
||||
tag = await UnitOfWork.CollectionTagRepository.GetCollectionAsync(1);
|
||||
Assert.NotNull(tag);
|
||||
Assert.False(tag.CoverImageLocked);
|
||||
Assert.Equal(string.Empty, tag.CoverImage);
|
||||
|
@ -286,7 +286,7 @@ public class CollectionTagServiceTests : AbstractDbTest
|
|||
await SeedSeries();
|
||||
|
||||
// Setup a user with admin role
|
||||
var user = await _unitOfWork.UserRepository.GetUserByIdAsync(1, AppUserIncludes.Collections);
|
||||
var user = await UnitOfWork.UserRepository.GetUserByIdAsync(1, AppUserIncludes.Collections);
|
||||
Assert.NotNull(user);
|
||||
await AddUserWithRole(user.Id, PolicyConstants.AdminRole);
|
||||
|
||||
|
@ -300,7 +300,7 @@ public class CollectionTagServiceTests : AbstractDbTest
|
|||
}, 1);
|
||||
|
||||
// Assert
|
||||
var tag = await _unitOfWork.CollectionTagRepository.GetCollectionAsync(1);
|
||||
var tag = await UnitOfWork.CollectionTagRepository.GetCollectionAsync(1);
|
||||
Assert.NotNull(tag);
|
||||
Assert.True(tag.Promoted);
|
||||
}
|
||||
|
@ -312,7 +312,7 @@ public class CollectionTagServiceTests : AbstractDbTest
|
|||
await SeedSeries();
|
||||
|
||||
// Setup a user with promote role
|
||||
var user = await _unitOfWork.UserRepository.GetUserByIdAsync(1, AppUserIncludes.Collections);
|
||||
var user = await UnitOfWork.UserRepository.GetUserByIdAsync(1, AppUserIncludes.Collections);
|
||||
Assert.NotNull(user);
|
||||
|
||||
// Mock to return promote role for the user
|
||||
|
@ -327,7 +327,7 @@ public class CollectionTagServiceTests : AbstractDbTest
|
|||
}, 1);
|
||||
|
||||
// Assert
|
||||
var tag = await _unitOfWork.CollectionTagRepository.GetCollectionAsync(1);
|
||||
var tag = await UnitOfWork.CollectionTagRepository.GetCollectionAsync(1);
|
||||
Assert.NotNull(tag);
|
||||
Assert.True(tag.Promoted);
|
||||
}
|
||||
|
@ -339,7 +339,7 @@ public class CollectionTagServiceTests : AbstractDbTest
|
|||
await SeedSeries();
|
||||
|
||||
// Setup a user with no special roles
|
||||
var user = await _unitOfWork.UserRepository.GetUserByIdAsync(1, AppUserIncludes.Collections);
|
||||
var user = await UnitOfWork.UserRepository.GetUserByIdAsync(1, AppUserIncludes.Collections);
|
||||
Assert.NotNull(user);
|
||||
|
||||
// Act - Try to promote a tag without proper role
|
||||
|
@ -351,7 +351,7 @@ public class CollectionTagServiceTests : AbstractDbTest
|
|||
}, 1);
|
||||
|
||||
// Assert
|
||||
var tag = await _unitOfWork.CollectionTagRepository.GetCollectionAsync(1);
|
||||
var tag = await UnitOfWork.CollectionTagRepository.GetCollectionAsync(1);
|
||||
Assert.NotNull(tag);
|
||||
Assert.False(tag.Promoted); // Should remain unpromoted
|
||||
}
|
||||
|
@ -365,15 +365,15 @@ public class CollectionTagServiceTests : AbstractDbTest
|
|||
{
|
||||
await SeedSeries();
|
||||
|
||||
var user = await _unitOfWork.UserRepository.GetUserByIdAsync(1, AppUserIncludes.Collections);
|
||||
var user = await UnitOfWork.UserRepository.GetUserByIdAsync(1, AppUserIncludes.Collections);
|
||||
Assert.NotNull(user);
|
||||
|
||||
// Tag 2 has 2 series
|
||||
var tag = await _unitOfWork.CollectionTagRepository.GetCollectionAsync(2);
|
||||
var tag = await UnitOfWork.CollectionTagRepository.GetCollectionAsync(2);
|
||||
Assert.NotNull(tag);
|
||||
|
||||
await _service.RemoveTagFromSeries(tag, new[] {1});
|
||||
var userCollections = await _unitOfWork.UserRepository.GetUserByIdAsync(1, AppUserIncludes.Collections);
|
||||
var userCollections = await UnitOfWork.UserRepository.GetUserByIdAsync(1, AppUserIncludes.Collections);
|
||||
Assert.Equal(2, userCollections!.Collections.Count);
|
||||
Assert.Single(tag.Items);
|
||||
Assert.Equal(2, tag.Items.First().Id);
|
||||
|
@ -387,11 +387,11 @@ public class CollectionTagServiceTests : AbstractDbTest
|
|||
{
|
||||
await SeedSeries();
|
||||
|
||||
var user = await _unitOfWork.UserRepository.GetUserByIdAsync(1, AppUserIncludes.Collections);
|
||||
var user = await UnitOfWork.UserRepository.GetUserByIdAsync(1, AppUserIncludes.Collections);
|
||||
Assert.NotNull(user);
|
||||
|
||||
// Tag 2 has 2 series
|
||||
var tag = await _unitOfWork.CollectionTagRepository.GetCollectionAsync(2);
|
||||
var tag = await UnitOfWork.CollectionTagRepository.GetCollectionAsync(2);
|
||||
Assert.NotNull(tag);
|
||||
|
||||
await _service.RemoveTagFromSeries(tag, new[] {1});
|
||||
|
@ -407,15 +407,15 @@ public class CollectionTagServiceTests : AbstractDbTest
|
|||
{
|
||||
await SeedSeries();
|
||||
|
||||
var user = await _unitOfWork.UserRepository.GetUserByIdAsync(1, AppUserIncludes.Collections);
|
||||
var user = await UnitOfWork.UserRepository.GetUserByIdAsync(1, AppUserIncludes.Collections);
|
||||
Assert.NotNull(user);
|
||||
|
||||
// Tag 1 has 1 series
|
||||
var tag = await _unitOfWork.CollectionTagRepository.GetCollectionAsync(1);
|
||||
var tag = await UnitOfWork.CollectionTagRepository.GetCollectionAsync(1);
|
||||
Assert.NotNull(tag);
|
||||
|
||||
await _service.RemoveTagFromSeries(tag, new[] {1});
|
||||
var tag2 = await _unitOfWork.CollectionTagRepository.GetCollectionAsync(1);
|
||||
var tag2 = await UnitOfWork.CollectionTagRepository.GetCollectionAsync(1);
|
||||
Assert.Null(tag2);
|
||||
}
|
||||
|
||||
|
@ -435,7 +435,7 @@ public class CollectionTagServiceTests : AbstractDbTest
|
|||
// Arrange
|
||||
await SeedSeries();
|
||||
|
||||
var tag = await _unitOfWork.CollectionTagRepository.GetCollectionAsync(1);
|
||||
var tag = await UnitOfWork.CollectionTagRepository.GetCollectionAsync(1);
|
||||
Assert.NotNull(tag);
|
||||
var initialItemCount = tag.Items.Count;
|
||||
|
||||
|
@ -444,7 +444,7 @@ public class CollectionTagServiceTests : AbstractDbTest
|
|||
|
||||
// Assert
|
||||
Assert.True(result);
|
||||
tag = await _unitOfWork.CollectionTagRepository.GetCollectionAsync(1);
|
||||
tag = await UnitOfWork.CollectionTagRepository.GetCollectionAsync(1);
|
||||
Assert.NotNull(tag);
|
||||
Assert.Equal(initialItemCount, tag.Items.Count); // No items should be removed
|
||||
}
|
||||
|
@ -455,7 +455,7 @@ public class CollectionTagServiceTests : AbstractDbTest
|
|||
// Arrange
|
||||
await SeedSeries();
|
||||
|
||||
var tag = await _unitOfWork.CollectionTagRepository.GetCollectionAsync(1);
|
||||
var tag = await UnitOfWork.CollectionTagRepository.GetCollectionAsync(1);
|
||||
Assert.NotNull(tag);
|
||||
var initialItemCount = tag.Items.Count;
|
||||
|
||||
|
@ -464,7 +464,7 @@ public class CollectionTagServiceTests : AbstractDbTest
|
|||
|
||||
// Assert
|
||||
Assert.True(result);
|
||||
tag = await _unitOfWork.CollectionTagRepository.GetCollectionAsync(1);
|
||||
tag = await UnitOfWork.CollectionTagRepository.GetCollectionAsync(1);
|
||||
Assert.NotNull(tag);
|
||||
Assert.Equal(initialItemCount, tag.Items.Count); // No items should be removed
|
||||
}
|
||||
|
@ -475,13 +475,13 @@ public class CollectionTagServiceTests : AbstractDbTest
|
|||
// Arrange
|
||||
await SeedSeries();
|
||||
|
||||
var tag = await _unitOfWork.CollectionTagRepository.GetCollectionAsync(1);
|
||||
var tag = await UnitOfWork.CollectionTagRepository.GetCollectionAsync(1);
|
||||
Assert.NotNull(tag);
|
||||
|
||||
// Force null items list
|
||||
tag.Items = null;
|
||||
_unitOfWork.CollectionTagRepository.Update(tag);
|
||||
await _unitOfWork.CommitAsync();
|
||||
UnitOfWork.CollectionTagRepository.Update(tag);
|
||||
await UnitOfWork.CommitAsync();
|
||||
|
||||
// Act
|
||||
var result = await _service.RemoveTagFromSeries(tag, [1]);
|
||||
|
@ -489,7 +489,7 @@ public class CollectionTagServiceTests : AbstractDbTest
|
|||
// Assert
|
||||
Assert.True(result);
|
||||
// The tag should not be removed since the items list was null, not empty
|
||||
var tagAfter = await _unitOfWork.CollectionTagRepository.GetCollectionAsync(1);
|
||||
var tagAfter = await UnitOfWork.CollectionTagRepository.GetCollectionAsync(1);
|
||||
Assert.Null(tagAfter);
|
||||
}
|
||||
|
||||
|
@ -501,21 +501,21 @@ public class CollectionTagServiceTests : AbstractDbTest
|
|||
|
||||
// Add a third series with a different age rating
|
||||
var s3 = new SeriesBuilder("Series 3").WithMetadata(new SeriesMetadataBuilder().WithAgeRating(AgeRating.PG).Build()).Build();
|
||||
_context.Library.First().Series.Add(s3);
|
||||
await _unitOfWork.CommitAsync();
|
||||
Context.Library.First().Series.Add(s3);
|
||||
await UnitOfWork.CommitAsync();
|
||||
|
||||
// Add series 3 to tag 2
|
||||
var tag = await _unitOfWork.CollectionTagRepository.GetCollectionAsync(2);
|
||||
var tag = await UnitOfWork.CollectionTagRepository.GetCollectionAsync(2);
|
||||
Assert.NotNull(tag);
|
||||
tag.Items.Add(s3);
|
||||
_unitOfWork.CollectionTagRepository.Update(tag);
|
||||
await _unitOfWork.CommitAsync();
|
||||
UnitOfWork.CollectionTagRepository.Update(tag);
|
||||
await UnitOfWork.CommitAsync();
|
||||
|
||||
// Act - Remove the series with Mature rating
|
||||
await _service.RemoveTagFromSeries(tag, new[] {1});
|
||||
|
||||
// Assert
|
||||
tag = await _unitOfWork.CollectionTagRepository.GetCollectionAsync(2);
|
||||
tag = await UnitOfWork.CollectionTagRepository.GetCollectionAsync(2);
|
||||
Assert.NotNull(tag);
|
||||
Assert.Equal(2, tag.Items.Count);
|
||||
|
||||
|
|
117
API.Tests/Services/CoverDbServiceTests.cs
Normal file
117
API.Tests/Services/CoverDbServiceTests.cs
Normal file
|
@ -0,0 +1,117 @@
|
|||
using System.IO;
|
||||
using System.IO.Abstractions;
|
||||
using System.Reflection;
|
||||
using System.Threading.Tasks;
|
||||
using API.Constants;
|
||||
using API.Entities.Enums;
|
||||
using API.Extensions;
|
||||
using API.Services;
|
||||
using API.Services.Tasks.Metadata;
|
||||
using API.SignalR;
|
||||
using EasyCaching.Core;
|
||||
using Kavita.Common;
|
||||
using Microsoft.Extensions.Hosting;
|
||||
using Microsoft.Extensions.Logging;
|
||||
using NSubstitute;
|
||||
using Xunit;
|
||||
|
||||
namespace API.Tests.Services;
|
||||
|
||||
public class CoverDbServiceTests : AbstractDbTest
|
||||
{
|
||||
private readonly DirectoryService _directoryService;
|
||||
private readonly IEasyCachingProviderFactory _cacheFactory = Substitute.For<IEasyCachingProviderFactory>();
|
||||
private readonly ICoverDbService _coverDbService;
|
||||
|
||||
private static readonly string FaviconPath = Path.Join(Directory.GetCurrentDirectory(),
|
||||
"../../../Services/Test Data/CoverDbService/Favicons");
|
||||
/// <summary>
|
||||
/// Path to download files temp to. Should be empty after each test.
|
||||
/// </summary>
|
||||
private static readonly string TempPath = Path.Join(Directory.GetCurrentDirectory(),
|
||||
"../../../Services/Test Data/CoverDbService/Temp");
|
||||
|
||||
public CoverDbServiceTests()
|
||||
{
|
||||
_directoryService = new DirectoryService(Substitute.For<ILogger<DirectoryService>>(), CreateFileSystem());
|
||||
var imageService = new ImageService(Substitute.For<ILogger<ImageService>>(), _directoryService);
|
||||
|
||||
_coverDbService = new CoverDbService(Substitute.For<ILogger<CoverDbService>>(), _directoryService, _cacheFactory,
|
||||
Substitute.For<IHostEnvironment>(), imageService, UnitOfWork, Substitute.For<IEventHub>());
|
||||
}
|
||||
|
||||
protected override Task ResetDb()
|
||||
{
|
||||
throw new System.NotImplementedException();
|
||||
}
|
||||
|
||||
|
||||
#region Download Favicon
|
||||
|
||||
/// <summary>
|
||||
/// I cannot figure out how to test this code due to the reliance on the _directoryService.FaviconDirectory and not being
|
||||
/// able to redirect it to the real filesystem.
|
||||
/// </summary>
|
||||
public async Task DownloadFaviconAsync_ShouldDownloadAndMatchExpectedFavicon()
|
||||
{
|
||||
// Arrange
|
||||
var testUrl = "https://anilist.co/anime/6205/Kmpfer/";
|
||||
var encodeFormat = EncodeFormat.WEBP;
|
||||
var expectedFaviconPath = Path.Combine(FaviconPath, "anilist.co.webp");
|
||||
|
||||
// Ensure TempPath exists
|
||||
_directoryService.ExistOrCreate(TempPath);
|
||||
|
||||
var baseUrl = "https://anilist.co";
|
||||
|
||||
// Ensure there is no cache result for this URL
|
||||
var provider = Substitute.For<IEasyCachingProvider>();
|
||||
provider.GetAsync<string>(baseUrl).Returns(new CacheValue<string>(null, false));
|
||||
_cacheFactory.GetCachingProvider(EasyCacheProfiles.Favicon).Returns(provider);
|
||||
|
||||
|
||||
// // Replace favicon directory with TempPath
|
||||
// var directoryService = (DirectoryService)_directoryService;
|
||||
// directoryService.FaviconDirectory = TempPath;
|
||||
|
||||
// Hack: Swap FaviconDirectory with TempPath for ability to download real files
|
||||
typeof(DirectoryService)
|
||||
.GetField("FaviconDirectory", BindingFlags.NonPublic | BindingFlags.Instance)
|
||||
?.SetValue(_directoryService, TempPath);
|
||||
|
||||
|
||||
// Act
|
||||
var resultFilename = await _coverDbService.DownloadFaviconAsync(testUrl, encodeFormat);
|
||||
var actualFaviconPath = Path.Combine(TempPath, resultFilename);
|
||||
|
||||
// Assert file exists
|
||||
Assert.True(File.Exists(actualFaviconPath), "Downloaded favicon does not exist in temp path");
|
||||
|
||||
// Load and compare similarity
|
||||
|
||||
var similarity = expectedFaviconPath.CalculateSimilarity(actualFaviconPath); // Assuming you have this extension
|
||||
Assert.True(similarity > 0.9f, $"Image similarity too low: {similarity}");
|
||||
}
|
||||
|
||||
[Fact]
|
||||
public async Task DownloadFaviconAsync_ShouldThrowKavitaException_WhenPreviouslyFailedUrlExistsInCache()
|
||||
{
|
||||
// Arrange
|
||||
var testUrl = "https://example.com";
|
||||
var encodeFormat = EncodeFormat.WEBP;
|
||||
|
||||
var provider = Substitute.For<IEasyCachingProvider>();
|
||||
provider.GetAsync<string>(Arg.Any<string>())
|
||||
.Returns(new CacheValue<string>(string.Empty, true)); // Simulate previous failure
|
||||
|
||||
_cacheFactory.GetCachingProvider(EasyCacheProfiles.Favicon).Returns(provider);
|
||||
|
||||
// Act & Assert
|
||||
await Assert.ThrowsAsync<KavitaException>(() =>
|
||||
_coverDbService.DownloadFaviconAsync(testUrl, encodeFormat));
|
||||
}
|
||||
|
||||
#endregion
|
||||
|
||||
|
||||
}
|
|
@ -18,13 +18,13 @@ public class DeviceServiceDbTests : AbstractDbTest
|
|||
|
||||
public DeviceServiceDbTests() : base()
|
||||
{
|
||||
_deviceService = new DeviceService(_unitOfWork, _logger, Substitute.For<IEmailService>());
|
||||
_deviceService = new DeviceService(UnitOfWork, _logger, Substitute.For<IEmailService>());
|
||||
}
|
||||
|
||||
protected override async Task ResetDb()
|
||||
{
|
||||
_context.Users.RemoveRange(_context.Users.ToList());
|
||||
await _unitOfWork.CommitAsync();
|
||||
Context.Users.RemoveRange(Context.Users.ToList());
|
||||
await UnitOfWork.CommitAsync();
|
||||
}
|
||||
|
||||
|
||||
|
@ -39,8 +39,8 @@ public class DeviceServiceDbTests : AbstractDbTest
|
|||
Devices = new List<Device>()
|
||||
};
|
||||
|
||||
_context.Users.Add(user);
|
||||
await _unitOfWork.CommitAsync();
|
||||
Context.Users.Add(user);
|
||||
await UnitOfWork.CommitAsync();
|
||||
|
||||
var device = await _deviceService.Create(new CreateDeviceDto()
|
||||
{
|
||||
|
@ -62,8 +62,8 @@ public class DeviceServiceDbTests : AbstractDbTest
|
|||
Devices = new List<Device>()
|
||||
};
|
||||
|
||||
_context.Users.Add(user);
|
||||
await _unitOfWork.CommitAsync();
|
||||
Context.Users.Add(user);
|
||||
await UnitOfWork.CommitAsync();
|
||||
|
||||
var device = await _deviceService.Create(new CreateDeviceDto()
|
||||
{
|
||||
|
|
File diff suppressed because it is too large
Load diff
|
@ -99,14 +99,14 @@ public class ParseScannedFilesTests : AbstractDbTest
|
|||
{
|
||||
// Since ProcessFile relies on _readingItemService, we can implement our own versions of _readingItemService so we have control over how the calls work
|
||||
GlobalConfiguration.Configuration.UseInMemoryStorage();
|
||||
_scannerHelper = new ScannerHelper(_unitOfWork, testOutputHelper);
|
||||
_scannerHelper = new ScannerHelper(UnitOfWork, testOutputHelper);
|
||||
}
|
||||
|
||||
protected override async Task ResetDb()
|
||||
{
|
||||
_context.Series.RemoveRange(_context.Series.ToList());
|
||||
Context.Series.RemoveRange(Context.Series.ToList());
|
||||
|
||||
await _context.SaveChangesAsync();
|
||||
await Context.SaveChangesAsync();
|
||||
}
|
||||
|
||||
#region MergeName
|
||||
|
@ -206,13 +206,13 @@ public class ParseScannedFilesTests : AbstractDbTest
|
|||
|
||||
|
||||
var library =
|
||||
await _unitOfWork.LibraryRepository.GetLibraryForIdAsync(1,
|
||||
await UnitOfWork.LibraryRepository.GetLibraryForIdAsync(1,
|
||||
LibraryIncludes.Folders | LibraryIncludes.FileTypes);
|
||||
Assert.NotNull(library);
|
||||
|
||||
library.Type = LibraryType.Manga;
|
||||
var parsedSeries = await psf.ScanLibrariesForSeries(library, new List<string>() {Root + "Data/"}, false,
|
||||
await _unitOfWork.SeriesRepository.GetFolderPathMap(1));
|
||||
await UnitOfWork.SeriesRepository.GetFolderPathMap(1));
|
||||
|
||||
|
||||
// Assert.Equal(3, parsedSeries.Values.Count);
|
||||
|
@ -251,9 +251,9 @@ public class ParseScannedFilesTests : AbstractDbTest
|
|||
new MockReadingItemService(ds, Substitute.For<IBookService>()), Substitute.For<IEventHub>());
|
||||
|
||||
var directoriesSeen = new HashSet<string>();
|
||||
var library = await _unitOfWork.LibraryRepository.GetLibraryForIdAsync(1,
|
||||
var library = await UnitOfWork.LibraryRepository.GetLibraryForIdAsync(1,
|
||||
LibraryIncludes.Folders | LibraryIncludes.FileTypes);
|
||||
var scanResults = await psf.ScanFiles("C:/Data/", true, await _unitOfWork.SeriesRepository.GetFolderPathMap(1), library);
|
||||
var scanResults = await psf.ScanFiles("C:/Data/", true, await UnitOfWork.SeriesRepository.GetFolderPathMap(1), library);
|
||||
foreach (var scanResult in scanResults)
|
||||
{
|
||||
directoriesSeen.Add(scanResult.Folder);
|
||||
|
@ -270,13 +270,13 @@ public class ParseScannedFilesTests : AbstractDbTest
|
|||
var psf = new ParseScannedFiles(Substitute.For<ILogger<ParseScannedFiles>>(), ds,
|
||||
new MockReadingItemService(ds, Substitute.For<IBookService>()), Substitute.For<IEventHub>());
|
||||
|
||||
var library = await _unitOfWork.LibraryRepository.GetLibraryForIdAsync(1,
|
||||
var library = await UnitOfWork.LibraryRepository.GetLibraryForIdAsync(1,
|
||||
LibraryIncludes.Folders | LibraryIncludes.FileTypes);
|
||||
Assert.NotNull(library);
|
||||
|
||||
var directoriesSeen = new HashSet<string>();
|
||||
var scanResults = await psf.ScanFiles("C:/Data/", false,
|
||||
await _unitOfWork.SeriesRepository.GetFolderPathMap(1), library);
|
||||
await UnitOfWork.SeriesRepository.GetFolderPathMap(1), library);
|
||||
|
||||
foreach (var scanResult in scanResults)
|
||||
{
|
||||
|
@ -305,10 +305,10 @@ public class ParseScannedFilesTests : AbstractDbTest
|
|||
var psf = new ParseScannedFiles(Substitute.For<ILogger<ParseScannedFiles>>(), ds,
|
||||
new MockReadingItemService(ds, Substitute.For<IBookService>()), Substitute.For<IEventHub>());
|
||||
|
||||
var library = await _unitOfWork.LibraryRepository.GetLibraryForIdAsync(1,
|
||||
var library = await UnitOfWork.LibraryRepository.GetLibraryForIdAsync(1,
|
||||
LibraryIncludes.Folders | LibraryIncludes.FileTypes);
|
||||
Assert.NotNull(library);
|
||||
var scanResults = await psf.ScanFiles("C:/Data", true, await _unitOfWork.SeriesRepository.GetFolderPathMap(1), library);
|
||||
var scanResults = await psf.ScanFiles("C:/Data", true, await UnitOfWork.SeriesRepository.GetFolderPathMap(1), library);
|
||||
|
||||
Assert.Equal(2, scanResults.Count);
|
||||
}
|
||||
|
@ -334,11 +334,11 @@ public class ParseScannedFilesTests : AbstractDbTest
|
|||
var psf = new ParseScannedFiles(Substitute.For<ILogger<ParseScannedFiles>>(), ds,
|
||||
new MockReadingItemService(ds, Substitute.For<IBookService>()), Substitute.For<IEventHub>());
|
||||
|
||||
var library = await _unitOfWork.LibraryRepository.GetLibraryForIdAsync(1,
|
||||
var library = await UnitOfWork.LibraryRepository.GetLibraryForIdAsync(1,
|
||||
LibraryIncludes.Folders | LibraryIncludes.FileTypes);
|
||||
Assert.NotNull(library);
|
||||
var scanResults = await psf.ScanFiles("C:/Data", false,
|
||||
await _unitOfWork.SeriesRepository.GetFolderPathMap(1), library);
|
||||
await UnitOfWork.SeriesRepository.GetFolderPathMap(1), library);
|
||||
|
||||
Assert.Single(scanResults);
|
||||
}
|
||||
|
@ -357,8 +357,8 @@ public class ParseScannedFilesTests : AbstractDbTest
|
|||
var library = await _scannerHelper.GenerateScannerData(testcase, infos);
|
||||
var testDirectoryPath = library.Folders.First().Path;
|
||||
|
||||
_unitOfWork.LibraryRepository.Update(library);
|
||||
await _unitOfWork.CommitAsync();
|
||||
UnitOfWork.LibraryRepository.Update(library);
|
||||
await UnitOfWork.CommitAsync();
|
||||
|
||||
var fs = new FileSystem();
|
||||
var ds = new DirectoryService(Substitute.For<ILogger<DirectoryService>>(), fs);
|
||||
|
@ -368,7 +368,7 @@ public class ParseScannedFilesTests : AbstractDbTest
|
|||
var scanner = _scannerHelper.CreateServices(ds, fs);
|
||||
await scanner.ScanLibrary(library.Id);
|
||||
|
||||
var postLib = await _unitOfWork.LibraryRepository.GetLibraryForIdAsync(library.Id, LibraryIncludes.Series);
|
||||
var postLib = await UnitOfWork.LibraryRepository.GetLibraryForIdAsync(library.Id, LibraryIncludes.Series);
|
||||
Assert.NotNull(postLib);
|
||||
Assert.Equal(4, postLib.Series.Count);
|
||||
|
||||
|
@ -391,7 +391,7 @@ public class ParseScannedFilesTests : AbstractDbTest
|
|||
Path.Join(executionerCopyDir, "The Executioner and Her Way of Life Vol. 1 Ch. 0002.cbz"));
|
||||
|
||||
// 4 series, of which 2 have volumes as directories
|
||||
var folderMap = await _unitOfWork.SeriesRepository.GetFolderPathMap(postLib.Id);
|
||||
var folderMap = await UnitOfWork.SeriesRepository.GetFolderPathMap(postLib.Id);
|
||||
Assert.Equal(6, folderMap.Count);
|
||||
|
||||
var res = await psf.ScanFiles(testDirectoryPath, true, folderMap, postLib);
|
||||
|
@ -409,8 +409,8 @@ public class ParseScannedFilesTests : AbstractDbTest
|
|||
var library = await _scannerHelper.GenerateScannerData(testcase, infos);
|
||||
var testDirectoryPath = library.Folders.First().Path;
|
||||
|
||||
_unitOfWork.LibraryRepository.Update(library);
|
||||
await _unitOfWork.CommitAsync();
|
||||
UnitOfWork.LibraryRepository.Update(library);
|
||||
await UnitOfWork.CommitAsync();
|
||||
|
||||
var fs = new FileSystem();
|
||||
var ds = new DirectoryService(Substitute.For<ILogger<DirectoryService>>(), fs);
|
||||
|
@ -420,7 +420,7 @@ public class ParseScannedFilesTests : AbstractDbTest
|
|||
var scanner = _scannerHelper.CreateServices(ds, fs);
|
||||
await scanner.ScanLibrary(library.Id);
|
||||
|
||||
var postLib = await _unitOfWork.LibraryRepository.GetLibraryForIdAsync(library.Id, LibraryIncludes.Series);
|
||||
var postLib = await UnitOfWork.LibraryRepository.GetLibraryForIdAsync(library.Id, LibraryIncludes.Series);
|
||||
Assert.NotNull(postLib);
|
||||
Assert.Equal(4, postLib.Series.Count);
|
||||
|
||||
|
@ -438,7 +438,7 @@ public class ParseScannedFilesTests : AbstractDbTest
|
|||
Path.Join(executionerCopyDir, "The Executioner and Her Way of Life Vol. 2.cbz"));
|
||||
|
||||
var res = await psf.ScanFiles(testDirectoryPath, true,
|
||||
await _unitOfWork.SeriesRepository.GetFolderPathMap(postLib.Id), postLib);
|
||||
await UnitOfWork.SeriesRepository.GetFolderPathMap(postLib.Id), postLib);
|
||||
var changes = res.Count(sc => sc.HasChanged);
|
||||
Assert.Equal(1, changes);
|
||||
}
|
||||
|
@ -452,8 +452,8 @@ public class ParseScannedFilesTests : AbstractDbTest
|
|||
var library = await _scannerHelper.GenerateScannerData(testcase, infos);
|
||||
var testDirectoryPath = library.Folders.First().Path;
|
||||
|
||||
_unitOfWork.LibraryRepository.Update(library);
|
||||
await _unitOfWork.CommitAsync();
|
||||
UnitOfWork.LibraryRepository.Update(library);
|
||||
await UnitOfWork.CommitAsync();
|
||||
|
||||
var fs = new FileSystem();
|
||||
var ds = new DirectoryService(Substitute.For<ILogger<DirectoryService>>(), fs);
|
||||
|
@ -463,7 +463,7 @@ public class ParseScannedFilesTests : AbstractDbTest
|
|||
var scanner = _scannerHelper.CreateServices(ds, fs);
|
||||
await scanner.ScanLibrary(library.Id);
|
||||
|
||||
var postLib = await _unitOfWork.LibraryRepository.GetLibraryForIdAsync(library.Id, LibraryIncludes.Series);
|
||||
var postLib = await UnitOfWork.LibraryRepository.GetLibraryForIdAsync(library.Id, LibraryIncludes.Series);
|
||||
Assert.NotNull(postLib);
|
||||
Assert.Single(postLib.Series);
|
||||
|
||||
|
@ -476,7 +476,7 @@ public class ParseScannedFilesTests : AbstractDbTest
|
|||
await Task.Delay(1100); // Ensure at least one second has passed since library scan
|
||||
|
||||
var res = await psf.ScanFiles(testDirectoryPath, true,
|
||||
await _unitOfWork.SeriesRepository.GetFolderPathMap(postLib.Id), postLib);
|
||||
await UnitOfWork.SeriesRepository.GetFolderPathMap(postLib.Id), postLib);
|
||||
Assert.DoesNotContain(res, sc => sc.HasChanged);
|
||||
}
|
||||
|
||||
|
@ -488,8 +488,8 @@ public class ParseScannedFilesTests : AbstractDbTest
|
|||
var library = await _scannerHelper.GenerateScannerData(testcase, infos);
|
||||
var testDirectoryPath = library.Folders.First().Path;
|
||||
|
||||
_unitOfWork.LibraryRepository.Update(library);
|
||||
await _unitOfWork.CommitAsync();
|
||||
UnitOfWork.LibraryRepository.Update(library);
|
||||
await UnitOfWork.CommitAsync();
|
||||
|
||||
var fs = new FileSystem();
|
||||
var ds = new DirectoryService(Substitute.For<ILogger<DirectoryService>>(), fs);
|
||||
|
@ -499,7 +499,7 @@ public class ParseScannedFilesTests : AbstractDbTest
|
|||
var scanner = _scannerHelper.CreateServices(ds, fs);
|
||||
await scanner.ScanLibrary(library.Id);
|
||||
|
||||
var postLib = await _unitOfWork.LibraryRepository.GetLibraryForIdAsync(library.Id, LibraryIncludes.Series);
|
||||
var postLib = await UnitOfWork.LibraryRepository.GetLibraryForIdAsync(library.Id, LibraryIncludes.Series);
|
||||
Assert.NotNull(postLib);
|
||||
Assert.Single(postLib.Series);
|
||||
|
||||
|
@ -508,8 +508,8 @@ public class ParseScannedFilesTests : AbstractDbTest
|
|||
Assert.Equal(4, spiceAndWolf.Volumes.Sum(v => v.Chapters.Count));
|
||||
|
||||
spiceAndWolf.LastFolderScanned = DateTime.Now.Subtract(TimeSpan.FromMinutes(2));
|
||||
_context.Series.Update(spiceAndWolf);
|
||||
await _context.SaveChangesAsync();
|
||||
Context.Series.Update(spiceAndWolf);
|
||||
await Context.SaveChangesAsync();
|
||||
|
||||
// Add file at series root
|
||||
var spiceAndWolfDir = Path.Join(testDirectoryPath, "Spice and Wolf");
|
||||
|
@ -517,7 +517,7 @@ public class ParseScannedFilesTests : AbstractDbTest
|
|||
Path.Join(spiceAndWolfDir, "Spice and Wolf Vol. 4.cbz"));
|
||||
|
||||
var res = await psf.ScanFiles(testDirectoryPath, true,
|
||||
await _unitOfWork.SeriesRepository.GetFolderPathMap(postLib.Id), postLib);
|
||||
await UnitOfWork.SeriesRepository.GetFolderPathMap(postLib.Id), postLib);
|
||||
var changes = res.Count(sc => sc.HasChanged);
|
||||
Assert.Equal(2, changes);
|
||||
}
|
||||
|
@ -530,8 +530,8 @@ public class ParseScannedFilesTests : AbstractDbTest
|
|||
var library = await _scannerHelper.GenerateScannerData(testcase, infos);
|
||||
var testDirectoryPath = library.Folders.First().Path;
|
||||
|
||||
_unitOfWork.LibraryRepository.Update(library);
|
||||
await _unitOfWork.CommitAsync();
|
||||
UnitOfWork.LibraryRepository.Update(library);
|
||||
await UnitOfWork.CommitAsync();
|
||||
|
||||
var fs = new FileSystem();
|
||||
var ds = new DirectoryService(Substitute.For<ILogger<DirectoryService>>(), fs);
|
||||
|
@ -541,7 +541,7 @@ public class ParseScannedFilesTests : AbstractDbTest
|
|||
var scanner = _scannerHelper.CreateServices(ds, fs);
|
||||
await scanner.ScanLibrary(library.Id);
|
||||
|
||||
var postLib = await _unitOfWork.LibraryRepository.GetLibraryForIdAsync(library.Id, LibraryIncludes.Series);
|
||||
var postLib = await UnitOfWork.LibraryRepository.GetLibraryForIdAsync(library.Id, LibraryIncludes.Series);
|
||||
Assert.NotNull(postLib);
|
||||
Assert.Single(postLib.Series);
|
||||
|
||||
|
@ -550,8 +550,8 @@ public class ParseScannedFilesTests : AbstractDbTest
|
|||
Assert.Equal(4, spiceAndWolf.Volumes.Sum(v => v.Chapters.Count));
|
||||
|
||||
spiceAndWolf.LastFolderScanned = DateTime.Now.Subtract(TimeSpan.FromMinutes(2));
|
||||
_context.Series.Update(spiceAndWolf);
|
||||
await _context.SaveChangesAsync();
|
||||
Context.Series.Update(spiceAndWolf);
|
||||
await Context.SaveChangesAsync();
|
||||
|
||||
// Add file in subfolder
|
||||
var spiceAndWolfDir = Path.Join(Path.Join(testDirectoryPath, "Spice and Wolf"), "Spice and Wolf Vol. 3");
|
||||
|
@ -559,7 +559,7 @@ public class ParseScannedFilesTests : AbstractDbTest
|
|||
Path.Join(spiceAndWolfDir, "Spice and Wolf Vol. 3 Ch. 0013.cbz"));
|
||||
|
||||
var res = await psf.ScanFiles(testDirectoryPath, true,
|
||||
await _unitOfWork.SeriesRepository.GetFolderPathMap(postLib.Id), postLib);
|
||||
await UnitOfWork.SeriesRepository.GetFolderPathMap(postLib.Id), postLib);
|
||||
var changes = res.Count(sc => sc.HasChanged);
|
||||
Assert.Equal(2, changes);
|
||||
}
|
||||
|
|
189
API.Tests/Services/RatingServiceTests.cs
Normal file
189
API.Tests/Services/RatingServiceTests.cs
Normal file
|
@ -0,0 +1,189 @@
|
|||
using System.Linq;
|
||||
using System.Threading.Tasks;
|
||||
using API.Data.Repositories;
|
||||
using API.DTOs;
|
||||
using API.Entities.Enums;
|
||||
using API.Helpers.Builders;
|
||||
using API.Services;
|
||||
using API.Services.Plus;
|
||||
using Hangfire;
|
||||
using Hangfire.InMemory;
|
||||
using Microsoft.Extensions.Logging;
|
||||
using NSubstitute;
|
||||
using Xunit;
|
||||
|
||||
namespace API.Tests.Services;
|
||||
|
||||
public class RatingServiceTests: AbstractDbTest
|
||||
{
|
||||
private readonly RatingService _ratingService;
|
||||
|
||||
public RatingServiceTests()
|
||||
{
|
||||
_ratingService = new RatingService(UnitOfWork, Substitute.For<IScrobblingService>(), Substitute.For<ILogger<RatingService>>());
|
||||
}
|
||||
|
||||
[Fact]
|
||||
public async Task UpdateRating_ShouldSetRating()
|
||||
{
|
||||
await ResetDb();
|
||||
|
||||
Context.Library.Add(new LibraryBuilder("Test LIb")
|
||||
.WithAppUser(new AppUserBuilder("majora2007", string.Empty).Build())
|
||||
.WithSeries(new SeriesBuilder("Test")
|
||||
|
||||
.WithVolume(new VolumeBuilder("1")
|
||||
.WithChapter(new ChapterBuilder("1").WithPages(1).Build())
|
||||
.Build())
|
||||
.Build())
|
||||
.Build());
|
||||
|
||||
|
||||
await Context.SaveChangesAsync();
|
||||
|
||||
|
||||
var user = await UnitOfWork.UserRepository.GetUserByUsernameAsync("majora2007", AppUserIncludes.Ratings);
|
||||
|
||||
JobStorage.Current = new InMemoryStorage();
|
||||
var result = await _ratingService.UpdateSeriesRating(user, new UpdateRatingDto
|
||||
{
|
||||
SeriesId = 1,
|
||||
UserRating = 3,
|
||||
});
|
||||
|
||||
Assert.True(result);
|
||||
|
||||
var ratings = (await UnitOfWork.UserRepository.GetUserByUsernameAsync("majora2007", AppUserIncludes.Ratings))!
|
||||
.Ratings;
|
||||
Assert.NotEmpty(ratings);
|
||||
Assert.Equal(3, ratings.First().Rating);
|
||||
}
|
||||
|
||||
[Fact]
|
||||
public async Task UpdateRating_ShouldUpdateExistingRating()
|
||||
{
|
||||
await ResetDb();
|
||||
|
||||
Context.Library.Add(new LibraryBuilder("Test LIb")
|
||||
.WithAppUser(new AppUserBuilder("majora2007", string.Empty).Build())
|
||||
.WithSeries(new SeriesBuilder("Test")
|
||||
|
||||
.WithVolume(new VolumeBuilder("1")
|
||||
.WithChapter(new ChapterBuilder("1").WithPages(1).Build())
|
||||
.Build())
|
||||
.Build())
|
||||
.Build());
|
||||
|
||||
|
||||
await Context.SaveChangesAsync();
|
||||
|
||||
var user = await UnitOfWork.UserRepository.GetUserByUsernameAsync("majora2007", AppUserIncludes.Ratings);
|
||||
|
||||
var result = await _ratingService.UpdateSeriesRating(user, new UpdateRatingDto
|
||||
{
|
||||
SeriesId = 1,
|
||||
UserRating = 3,
|
||||
});
|
||||
|
||||
Assert.True(result);
|
||||
|
||||
JobStorage.Current = new InMemoryStorage();
|
||||
var ratings = (await UnitOfWork.UserRepository.GetUserByUsernameAsync("majora2007", AppUserIncludes.Ratings))
|
||||
.Ratings;
|
||||
Assert.NotEmpty(ratings);
|
||||
Assert.Equal(3, ratings.First().Rating);
|
||||
|
||||
// Update the DB again
|
||||
|
||||
var result2 = await _ratingService.UpdateSeriesRating(user, new UpdateRatingDto
|
||||
{
|
||||
SeriesId = 1,
|
||||
UserRating = 5,
|
||||
});
|
||||
|
||||
Assert.True(result2);
|
||||
|
||||
var ratings2 = (await UnitOfWork.UserRepository.GetUserByUsernameAsync("majora2007", AppUserIncludes.Ratings))
|
||||
.Ratings;
|
||||
Assert.NotEmpty(ratings2);
|
||||
Assert.True(ratings2.Count == 1);
|
||||
Assert.Equal(5, ratings2.First().Rating);
|
||||
}
|
||||
|
||||
[Fact]
|
||||
public async Task UpdateRating_ShouldClampRatingAt5()
|
||||
{
|
||||
await ResetDb();
|
||||
|
||||
Context.Library.Add(new LibraryBuilder("Test LIb")
|
||||
.WithAppUser(new AppUserBuilder("majora2007", string.Empty).Build())
|
||||
.WithSeries(new SeriesBuilder("Test")
|
||||
|
||||
.WithVolume(new VolumeBuilder("1")
|
||||
.WithChapter(new ChapterBuilder("1").WithPages(1).Build())
|
||||
.Build())
|
||||
.Build())
|
||||
.Build());
|
||||
|
||||
await Context.SaveChangesAsync();
|
||||
|
||||
var user = await UnitOfWork.UserRepository.GetUserByUsernameAsync("majora2007", AppUserIncludes.Ratings);
|
||||
|
||||
var result = await _ratingService.UpdateSeriesRating(user, new UpdateRatingDto
|
||||
{
|
||||
SeriesId = 1,
|
||||
UserRating = 10,
|
||||
});
|
||||
|
||||
Assert.True(result);
|
||||
|
||||
JobStorage.Current = new InMemoryStorage();
|
||||
var ratings = (await UnitOfWork.UserRepository.GetUserByUsernameAsync("majora2007",
|
||||
AppUserIncludes.Ratings)!)
|
||||
.Ratings;
|
||||
Assert.NotEmpty(ratings);
|
||||
Assert.Equal(5, ratings.First().Rating);
|
||||
}
|
||||
|
||||
[Fact]
|
||||
public async Task UpdateRating_ShouldReturnFalseWhenSeriesDoesntExist()
|
||||
{
|
||||
await ResetDb();
|
||||
|
||||
Context.Library.Add(new LibraryBuilder("Test LIb", LibraryType.Book)
|
||||
.WithAppUser(new AppUserBuilder("majora2007", string.Empty).Build())
|
||||
.WithSeries(new SeriesBuilder("Test")
|
||||
|
||||
.WithVolume(new VolumeBuilder("1")
|
||||
.WithChapter(new ChapterBuilder("1").WithPages(1).Build())
|
||||
.Build())
|
||||
.Build())
|
||||
.Build());
|
||||
|
||||
await Context.SaveChangesAsync();
|
||||
|
||||
var user = await UnitOfWork.UserRepository.GetUserByUsernameAsync("majora2007", AppUserIncludes.Ratings);
|
||||
|
||||
var result = await _ratingService.UpdateSeriesRating(user, new UpdateRatingDto
|
||||
{
|
||||
SeriesId = 2,
|
||||
UserRating = 5,
|
||||
});
|
||||
|
||||
Assert.False(result);
|
||||
|
||||
var ratings = user.Ratings;
|
||||
Assert.Empty(ratings);
|
||||
}
|
||||
protected override async Task ResetDb()
|
||||
{
|
||||
Context.Series.RemoveRange(Context.Series.ToList());
|
||||
Context.AppUserRating.RemoveRange(Context.AppUserRating.ToList());
|
||||
Context.Genre.RemoveRange(Context.Genre.ToList());
|
||||
Context.CollectionTag.RemoveRange(Context.CollectionTag.ToList());
|
||||
Context.Person.RemoveRange(Context.Person.ToList());
|
||||
Context.Library.RemoveRange(Context.Library.ToList());
|
||||
|
||||
await Context.SaveChangesAsync();
|
||||
}
|
||||
}
|
File diff suppressed because it is too large
Load diff
|
@ -28,13 +28,13 @@ public class ScannerServiceTests : AbstractDbTest
|
|||
|
||||
// Set up Hangfire to use in-memory storage for testing
|
||||
GlobalConfiguration.Configuration.UseInMemoryStorage();
|
||||
_scannerHelper = new ScannerHelper(_unitOfWork, testOutputHelper);
|
||||
_scannerHelper = new ScannerHelper(UnitOfWork, testOutputHelper);
|
||||
}
|
||||
|
||||
protected override async Task ResetDb()
|
||||
{
|
||||
_context.Library.RemoveRange(_context.Library);
|
||||
await _context.SaveChangesAsync();
|
||||
Context.Library.RemoveRange(Context.Library);
|
||||
await Context.SaveChangesAsync();
|
||||
}
|
||||
|
||||
|
||||
|
@ -44,18 +44,18 @@ public class ScannerServiceTests : AbstractDbTest
|
|||
{
|
||||
await SetLastScannedInThePast(series, duration, false);
|
||||
}
|
||||
await _context.SaveChangesAsync();
|
||||
await Context.SaveChangesAsync();
|
||||
}
|
||||
|
||||
protected async Task SetLastScannedInThePast(Series series, TimeSpan? duration = null, bool save = true)
|
||||
{
|
||||
duration ??= TimeSpan.FromMinutes(2);
|
||||
series.LastFolderScanned = DateTime.Now.Subtract(duration.Value);
|
||||
_context.Series.Update(series);
|
||||
Context.Series.Update(series);
|
||||
|
||||
if (save)
|
||||
{
|
||||
await _context.SaveChangesAsync();
|
||||
await Context.SaveChangesAsync();
|
||||
}
|
||||
}
|
||||
|
||||
|
@ -66,7 +66,7 @@ public class ScannerServiceTests : AbstractDbTest
|
|||
var library = await _scannerHelper.GenerateScannerData(testcase);
|
||||
var scanner = _scannerHelper.CreateServices();
|
||||
await scanner.ScanLibrary(library.Id);
|
||||
var postLib = await _unitOfWork.LibraryRepository.GetLibraryForIdAsync(library.Id, LibraryIncludes.Series);
|
||||
var postLib = await UnitOfWork.LibraryRepository.GetLibraryForIdAsync(library.Id, LibraryIncludes.Series);
|
||||
|
||||
Assert.NotNull(postLib);
|
||||
Assert.Equal(4, postLib.Series.Count);
|
||||
|
@ -79,7 +79,7 @@ public class ScannerServiceTests : AbstractDbTest
|
|||
var library = await _scannerHelper.GenerateScannerData(testcase);
|
||||
var scanner = _scannerHelper.CreateServices();
|
||||
await scanner.ScanLibrary(library.Id);
|
||||
var postLib = await _unitOfWork.LibraryRepository.GetLibraryForIdAsync(library.Id, LibraryIncludes.Series);
|
||||
var postLib = await UnitOfWork.LibraryRepository.GetLibraryForIdAsync(library.Id, LibraryIncludes.Series);
|
||||
|
||||
Assert.NotNull(postLib);
|
||||
Assert.Single(postLib.Series);
|
||||
|
@ -94,7 +94,7 @@ public class ScannerServiceTests : AbstractDbTest
|
|||
var library = await _scannerHelper.GenerateScannerData(testcase);
|
||||
var scanner = _scannerHelper.CreateServices();
|
||||
await scanner.ScanLibrary(library.Id);
|
||||
var postLib = await _unitOfWork.LibraryRepository.GetLibraryForIdAsync(library.Id, LibraryIncludes.Series);
|
||||
var postLib = await UnitOfWork.LibraryRepository.GetLibraryForIdAsync(library.Id, LibraryIncludes.Series);
|
||||
|
||||
Assert.NotNull(postLib);
|
||||
Assert.Single(postLib.Series);
|
||||
|
@ -110,7 +110,7 @@ public class ScannerServiceTests : AbstractDbTest
|
|||
var library = await _scannerHelper.GenerateScannerData(testcase);
|
||||
var scanner = _scannerHelper.CreateServices();
|
||||
await scanner.ScanLibrary(library.Id);
|
||||
var postLib = await _unitOfWork.LibraryRepository.GetLibraryForIdAsync(library.Id, LibraryIncludes.Series);
|
||||
var postLib = await UnitOfWork.LibraryRepository.GetLibraryForIdAsync(library.Id, LibraryIncludes.Series);
|
||||
|
||||
Assert.NotNull(postLib);
|
||||
Assert.Single(postLib.Series);
|
||||
|
@ -125,7 +125,7 @@ public class ScannerServiceTests : AbstractDbTest
|
|||
var library = await _scannerHelper.GenerateScannerData(testcase);
|
||||
var scanner = _scannerHelper.CreateServices();
|
||||
await scanner.ScanLibrary(library.Id);
|
||||
var postLib = await _unitOfWork.LibraryRepository.GetLibraryForIdAsync(library.Id, LibraryIncludes.Series);
|
||||
var postLib = await UnitOfWork.LibraryRepository.GetLibraryForIdAsync(library.Id, LibraryIncludes.Series);
|
||||
|
||||
Assert.NotNull(postLib);
|
||||
Assert.Single(postLib.Series);
|
||||
|
@ -141,7 +141,7 @@ public class ScannerServiceTests : AbstractDbTest
|
|||
var library = await _scannerHelper.GenerateScannerData(testcase);
|
||||
var scanner = _scannerHelper.CreateServices();
|
||||
await scanner.ScanLibrary(library.Id);
|
||||
var postLib = await _unitOfWork.LibraryRepository.GetLibraryForIdAsync(library.Id, LibraryIncludes.Series);
|
||||
var postLib = await UnitOfWork.LibraryRepository.GetLibraryForIdAsync(library.Id, LibraryIncludes.Series);
|
||||
|
||||
Assert.NotNull(postLib);
|
||||
Assert.Single(postLib.Series);
|
||||
|
@ -157,7 +157,7 @@ public class ScannerServiceTests : AbstractDbTest
|
|||
var library = await _scannerHelper.GenerateScannerData(testcase);
|
||||
var scanner = _scannerHelper.CreateServices();
|
||||
await scanner.ScanLibrary(library.Id);
|
||||
var postLib = await _unitOfWork.LibraryRepository.GetLibraryForIdAsync(library.Id, LibraryIncludes.Series);
|
||||
var postLib = await UnitOfWork.LibraryRepository.GetLibraryForIdAsync(library.Id, LibraryIncludes.Series);
|
||||
|
||||
Assert.NotNull(postLib);
|
||||
Assert.Single(postLib.Series);
|
||||
|
@ -188,7 +188,7 @@ public class ScannerServiceTests : AbstractDbTest
|
|||
|
||||
var scanner = _scannerHelper.CreateServices();
|
||||
await scanner.ScanLibrary(library.Id);
|
||||
var postLib = await _unitOfWork.LibraryRepository.GetLibraryForIdAsync(library.Id, LibraryIncludes.Series);
|
||||
var postLib = await UnitOfWork.LibraryRepository.GetLibraryForIdAsync(library.Id, LibraryIncludes.Series);
|
||||
|
||||
Assert.NotNull(postLib);
|
||||
Assert.Single(postLib.Series);
|
||||
|
@ -213,7 +213,7 @@ public class ScannerServiceTests : AbstractDbTest
|
|||
|
||||
var scanner = _scannerHelper.CreateServices();
|
||||
await scanner.ScanLibrary(library.Id);
|
||||
var postLib = await _unitOfWork.LibraryRepository.GetLibraryForIdAsync(library.Id, LibraryIncludes.Series);
|
||||
var postLib = await UnitOfWork.LibraryRepository.GetLibraryForIdAsync(library.Id, LibraryIncludes.Series);
|
||||
|
||||
Assert.NotNull(postLib);
|
||||
Assert.Single(postLib.Series);
|
||||
|
@ -244,7 +244,7 @@ public class ScannerServiceTests : AbstractDbTest
|
|||
|
||||
var scanner = _scannerHelper.CreateServices();
|
||||
await scanner.ScanLibrary(library.Id);
|
||||
var postLib = await _unitOfWork.LibraryRepository.GetLibraryForIdAsync(library.Id, LibraryIncludes.Series);
|
||||
var postLib = await UnitOfWork.LibraryRepository.GetLibraryForIdAsync(library.Id, LibraryIncludes.Series);
|
||||
|
||||
Assert.NotNull(postLib);
|
||||
Assert.Single(postLib.Series);
|
||||
|
@ -268,7 +268,7 @@ public class ScannerServiceTests : AbstractDbTest
|
|||
|
||||
var scanner = _scannerHelper.CreateServices();
|
||||
await scanner.ScanLibrary(library.Id);
|
||||
var postLib = await _unitOfWork.LibraryRepository.GetLibraryForIdAsync(library.Id, LibraryIncludes.Series);
|
||||
var postLib = await UnitOfWork.LibraryRepository.GetLibraryForIdAsync(library.Id, LibraryIncludes.Series);
|
||||
|
||||
Assert.NotNull(postLib);
|
||||
Assert.Single(postLib.Series);
|
||||
|
@ -288,7 +288,7 @@ public class ScannerServiceTests : AbstractDbTest
|
|||
|
||||
var scanner = _scannerHelper.CreateServices();
|
||||
await scanner.ScanLibrary(library.Id);
|
||||
var postLib = await _unitOfWork.LibraryRepository.GetLibraryForIdAsync(library.Id, LibraryIncludes.Series);
|
||||
var postLib = await UnitOfWork.LibraryRepository.GetLibraryForIdAsync(library.Id, LibraryIncludes.Series);
|
||||
|
||||
Assert.NotNull(postLib);
|
||||
Assert.Single(postLib.Series);
|
||||
|
@ -326,7 +326,7 @@ public class ScannerServiceTests : AbstractDbTest
|
|||
|
||||
var scanner = _scannerHelper.CreateServices();
|
||||
await scanner.ScanLibrary(library.Id);
|
||||
var postLib = await _unitOfWork.LibraryRepository.GetLibraryForIdAsync(library.Id, LibraryIncludes.Series);
|
||||
var postLib = await UnitOfWork.LibraryRepository.GetLibraryForIdAsync(library.Id, LibraryIncludes.Series);
|
||||
|
||||
Assert.NotNull(postLib);
|
||||
Assert.Single(postLib.Series);
|
||||
|
@ -350,7 +350,7 @@ public class ScannerServiceTests : AbstractDbTest
|
|||
|
||||
var scanner = _scannerHelper.CreateServices();
|
||||
await scanner.ScanLibrary(library.Id);
|
||||
var postLib = await _unitOfWork.LibraryRepository.GetLibraryForIdAsync(library.Id, LibraryIncludes.Series);
|
||||
var postLib = await UnitOfWork.LibraryRepository.GetLibraryForIdAsync(library.Id, LibraryIncludes.Series);
|
||||
|
||||
Assert.NotNull(postLib);
|
||||
Assert.Single(postLib.Series);
|
||||
|
@ -369,7 +369,7 @@ public class ScannerServiceTests : AbstractDbTest
|
|||
|
||||
var scanner = _scannerHelper.CreateServices();
|
||||
await scanner.ScanLibrary(library.Id);
|
||||
var postLib = await _unitOfWork.LibraryRepository.GetLibraryForIdAsync(library.Id, LibraryIncludes.Series);
|
||||
var postLib = await UnitOfWork.LibraryRepository.GetLibraryForIdAsync(library.Id, LibraryIncludes.Series);
|
||||
|
||||
Assert.NotNull(postLib);
|
||||
Assert.Single(postLib.Series);
|
||||
|
@ -399,7 +399,7 @@ public class ScannerServiceTests : AbstractDbTest
|
|||
|
||||
var scanner = _scannerHelper.CreateServices();
|
||||
await scanner.ScanLibrary(library.Id);
|
||||
var postLib = await _unitOfWork.LibraryRepository.GetLibraryForIdAsync(library.Id, LibraryIncludes.Series);
|
||||
var postLib = await UnitOfWork.LibraryRepository.GetLibraryForIdAsync(library.Id, LibraryIncludes.Series);
|
||||
|
||||
Assert.NotNull(postLib);
|
||||
Assert.Single(postLib.Series);
|
||||
|
@ -436,7 +436,7 @@ public class ScannerServiceTests : AbstractDbTest
|
|||
|
||||
var scanner = _scannerHelper.CreateServices();
|
||||
await scanner.ScanLibrary(library.Id);
|
||||
var postLib = await _unitOfWork.LibraryRepository.GetLibraryForIdAsync(library.Id, LibraryIncludes.Series);
|
||||
var postLib = await UnitOfWork.LibraryRepository.GetLibraryForIdAsync(library.Id, LibraryIncludes.Series);
|
||||
|
||||
Assert.NotNull(postLib);
|
||||
Assert.Single(postLib.Series);
|
||||
|
@ -464,7 +464,7 @@ public class ScannerServiceTests : AbstractDbTest
|
|||
|
||||
var scanner = _scannerHelper.CreateServices();
|
||||
await scanner.ScanLibrary(library.Id);
|
||||
var postLib = await _unitOfWork.LibraryRepository.GetLibraryForIdAsync(library.Id, LibraryIncludes.Series);
|
||||
var postLib = await UnitOfWork.LibraryRepository.GetLibraryForIdAsync(library.Id, LibraryIncludes.Series);
|
||||
|
||||
Assert.NotNull(postLib);
|
||||
Assert.Single(postLib.Series);
|
||||
|
@ -484,13 +484,13 @@ public class ScannerServiceTests : AbstractDbTest
|
|||
var library = await _scannerHelper.GenerateScannerData(testcase, infos);
|
||||
|
||||
library.LibraryExcludePatterns = [new LibraryExcludePattern() {Pattern = "**/Extra/*"}];
|
||||
_unitOfWork.LibraryRepository.Update(library);
|
||||
await _unitOfWork.CommitAsync();
|
||||
UnitOfWork.LibraryRepository.Update(library);
|
||||
await UnitOfWork.CommitAsync();
|
||||
|
||||
|
||||
var scanner = _scannerHelper.CreateServices();
|
||||
await scanner.ScanLibrary(library.Id);
|
||||
var postLib = await _unitOfWork.LibraryRepository.GetLibraryForIdAsync(library.Id, LibraryIncludes.Series);
|
||||
var postLib = await UnitOfWork.LibraryRepository.GetLibraryForIdAsync(library.Id, LibraryIncludes.Series);
|
||||
|
||||
Assert.NotNull(postLib);
|
||||
Assert.Single(postLib.Series);
|
||||
|
@ -508,13 +508,13 @@ public class ScannerServiceTests : AbstractDbTest
|
|||
var library = await _scannerHelper.GenerateScannerData(testcase, infos);
|
||||
|
||||
library.LibraryExcludePatterns = [new LibraryExcludePattern() {Pattern = "**\\Extra\\*"}];
|
||||
_unitOfWork.LibraryRepository.Update(library);
|
||||
await _unitOfWork.CommitAsync();
|
||||
UnitOfWork.LibraryRepository.Update(library);
|
||||
await UnitOfWork.CommitAsync();
|
||||
|
||||
|
||||
var scanner = _scannerHelper.CreateServices();
|
||||
await scanner.ScanLibrary(library.Id);
|
||||
var postLib = await _unitOfWork.LibraryRepository.GetLibraryForIdAsync(library.Id, LibraryIncludes.Series);
|
||||
var postLib = await UnitOfWork.LibraryRepository.GetLibraryForIdAsync(library.Id, LibraryIncludes.Series);
|
||||
|
||||
Assert.NotNull(postLib);
|
||||
Assert.Single(postLib.Series);
|
||||
|
@ -541,13 +541,13 @@ public class ScannerServiceTests : AbstractDbTest
|
|||
new FolderPath() {Path = Path.Join(testDirectoryPath, "Root 2")}
|
||||
];
|
||||
|
||||
_unitOfWork.LibraryRepository.Update(library);
|
||||
await _unitOfWork.CommitAsync();
|
||||
UnitOfWork.LibraryRepository.Update(library);
|
||||
await UnitOfWork.CommitAsync();
|
||||
|
||||
|
||||
var scanner = _scannerHelper.CreateServices();
|
||||
await scanner.ScanLibrary(library.Id);
|
||||
var postLib = await _unitOfWork.LibraryRepository.GetLibraryForIdAsync(library.Id, LibraryIncludes.Series);
|
||||
var postLib = await UnitOfWork.LibraryRepository.GetLibraryForIdAsync(library.Id, LibraryIncludes.Series);
|
||||
|
||||
Assert.NotNull(postLib);
|
||||
Assert.Equal(2, postLib.Series.Count);
|
||||
|
@ -563,7 +563,7 @@ public class ScannerServiceTests : AbstractDbTest
|
|||
// Rescan to ensure nothing changes yet again
|
||||
await scanner.ScanLibrary(library.Id, true);
|
||||
|
||||
postLib = await _unitOfWork.LibraryRepository.GetLibraryForIdAsync(library.Id, LibraryIncludes.Series);
|
||||
postLib = await UnitOfWork.LibraryRepository.GetLibraryForIdAsync(library.Id, LibraryIncludes.Series);
|
||||
Assert.Equal(2, postLib.Series.Count);
|
||||
s = postLib.Series.First(s => s.Name == "Plush");
|
||||
Assert.Equal(3, s.Volumes.Count);
|
||||
|
@ -594,13 +594,13 @@ public class ScannerServiceTests : AbstractDbTest
|
|||
new FolderPath() {Path = Path.Join(testDirectoryPath, "Root 2")}
|
||||
];
|
||||
|
||||
_unitOfWork.LibraryRepository.Update(library);
|
||||
await _unitOfWork.CommitAsync();
|
||||
UnitOfWork.LibraryRepository.Update(library);
|
||||
await UnitOfWork.CommitAsync();
|
||||
|
||||
|
||||
var scanner = _scannerHelper.CreateServices();
|
||||
await scanner.ScanLibrary(library.Id);
|
||||
var postLib = await _unitOfWork.LibraryRepository.GetLibraryForIdAsync(library.Id, LibraryIncludes.Series);
|
||||
var postLib = await UnitOfWork.LibraryRepository.GetLibraryForIdAsync(library.Id, LibraryIncludes.Series);
|
||||
|
||||
Assert.NotNull(postLib);
|
||||
Assert.Equal(2, postLib.Series.Count);
|
||||
|
@ -619,7 +619,7 @@ public class ScannerServiceTests : AbstractDbTest
|
|||
// Rescan to ensure nothing changes yet again
|
||||
await scanner.ScanLibrary(library.Id, false);
|
||||
|
||||
postLib = await _unitOfWork.LibraryRepository.GetLibraryForIdAsync(library.Id, LibraryIncludes.Series);
|
||||
postLib = await UnitOfWork.LibraryRepository.GetLibraryForIdAsync(library.Id, LibraryIncludes.Series);
|
||||
Assert.Equal(2, postLib.Series.Count);
|
||||
s = postLib.Series.First(s => s.Name == "Plush");
|
||||
Assert.Equal(3, s.Volumes.Count);
|
||||
|
@ -647,14 +647,14 @@ public class ScannerServiceTests : AbstractDbTest
|
|||
new FolderPath() { Path = Path.Combine(testDirectoryPath, "Root 2") }
|
||||
];
|
||||
|
||||
_unitOfWork.LibraryRepository.Update(library);
|
||||
await _unitOfWork.CommitAsync();
|
||||
UnitOfWork.LibraryRepository.Update(library);
|
||||
await UnitOfWork.CommitAsync();
|
||||
|
||||
var scanner = _scannerHelper.CreateServices();
|
||||
|
||||
// First Scan: Everything should be added
|
||||
await scanner.ScanLibrary(library.Id);
|
||||
var postLib = await _unitOfWork.LibraryRepository.GetLibraryForIdAsync(library.Id, LibraryIncludes.Series);
|
||||
var postLib = await UnitOfWork.LibraryRepository.GetLibraryForIdAsync(library.Id, LibraryIncludes.Series);
|
||||
|
||||
Assert.NotNull(postLib);
|
||||
Assert.Contains(postLib.Series, s => s.Name == "Accel");
|
||||
|
@ -662,19 +662,19 @@ public class ScannerServiceTests : AbstractDbTest
|
|||
|
||||
// Second Scan: Remove Root 2, expect Accel to be removed
|
||||
library.Folders = [new FolderPath() { Path = Path.Combine(testDirectoryPath, "Root 1") }];
|
||||
_unitOfWork.LibraryRepository.Update(library);
|
||||
await _unitOfWork.CommitAsync();
|
||||
UnitOfWork.LibraryRepository.Update(library);
|
||||
await UnitOfWork.CommitAsync();
|
||||
|
||||
// Emulate time passage by updating lastFolderScan to be a min in the past
|
||||
foreach (var s in postLib.Series)
|
||||
{
|
||||
s.LastFolderScanned = DateTime.Now.Subtract(TimeSpan.FromMinutes(1));
|
||||
_context.Series.Update(s);
|
||||
Context.Series.Update(s);
|
||||
}
|
||||
await _context.SaveChangesAsync();
|
||||
await Context.SaveChangesAsync();
|
||||
|
||||
await scanner.ScanLibrary(library.Id);
|
||||
postLib = await _unitOfWork.LibraryRepository.GetLibraryForIdAsync(library.Id, LibraryIncludes.Series);
|
||||
postLib = await UnitOfWork.LibraryRepository.GetLibraryForIdAsync(library.Id, LibraryIncludes.Series);
|
||||
|
||||
Assert.DoesNotContain(postLib.Series, s => s.Name == "Accel"); // Ensure Accel is gone
|
||||
Assert.Contains(postLib.Series, s => s.Name == "Plush");
|
||||
|
@ -685,19 +685,19 @@ public class ScannerServiceTests : AbstractDbTest
|
|||
new FolderPath() { Path = Path.Combine(testDirectoryPath, "Root 1") },
|
||||
new FolderPath() { Path = Path.Combine(testDirectoryPath, "Root 2") }
|
||||
];
|
||||
_unitOfWork.LibraryRepository.Update(library);
|
||||
await _unitOfWork.CommitAsync();
|
||||
UnitOfWork.LibraryRepository.Update(library);
|
||||
await UnitOfWork.CommitAsync();
|
||||
|
||||
// Emulate time passage by updating lastFolderScan to be a min in the past
|
||||
foreach (var s in postLib.Series)
|
||||
{
|
||||
s.LastFolderScanned = DateTime.Now.Subtract(TimeSpan.FromMinutes(1));
|
||||
_context.Series.Update(s);
|
||||
Context.Series.Update(s);
|
||||
}
|
||||
await _context.SaveChangesAsync();
|
||||
await Context.SaveChangesAsync();
|
||||
|
||||
await scanner.ScanLibrary(library.Id);
|
||||
postLib = await _unitOfWork.LibraryRepository.GetLibraryForIdAsync(library.Id, LibraryIncludes.Series);
|
||||
postLib = await UnitOfWork.LibraryRepository.GetLibraryForIdAsync(library.Id, LibraryIncludes.Series);
|
||||
|
||||
Assert.Contains(postLib.Series, s => s.Name == "Accel"); // Accel should be back
|
||||
Assert.Contains(postLib.Series, s => s.Name == "Plush");
|
||||
|
@ -707,7 +707,7 @@ public class ScannerServiceTests : AbstractDbTest
|
|||
|
||||
// Fourth Scan: Run again to check stability (should not remove Accel)
|
||||
await scanner.ScanLibrary(library.Id);
|
||||
postLib = await _unitOfWork.LibraryRepository.GetLibraryForIdAsync(library.Id, LibraryIncludes.Series);
|
||||
postLib = await UnitOfWork.LibraryRepository.GetLibraryForIdAsync(library.Id, LibraryIncludes.Series);
|
||||
|
||||
Assert.Contains(postLib.Series, s => s.Name == "Accel");
|
||||
Assert.Contains(postLib.Series, s => s.Name == "Plush");
|
||||
|
@ -732,14 +732,14 @@ public class ScannerServiceTests : AbstractDbTest
|
|||
new FolderPath() { Path = Path.Combine(testDirectoryPath, "Root 2") }
|
||||
];
|
||||
|
||||
_unitOfWork.LibraryRepository.Update(library);
|
||||
await _unitOfWork.CommitAsync();
|
||||
UnitOfWork.LibraryRepository.Update(library);
|
||||
await UnitOfWork.CommitAsync();
|
||||
|
||||
var scanner = _scannerHelper.CreateServices();
|
||||
|
||||
// First Scan: Everything should be added
|
||||
await scanner.ScanLibrary(library.Id);
|
||||
var postLib = await _unitOfWork.LibraryRepository.GetLibraryForIdAsync(library.Id, LibraryIncludes.Series);
|
||||
var postLib = await UnitOfWork.LibraryRepository.GetLibraryForIdAsync(library.Id, LibraryIncludes.Series);
|
||||
|
||||
Assert.NotNull(postLib);
|
||||
Assert.Contains(postLib.Series, s => s.Name == "Accel");
|
||||
|
@ -747,14 +747,14 @@ public class ScannerServiceTests : AbstractDbTest
|
|||
|
||||
// Second Scan: Delete the Series
|
||||
library.Series = [];
|
||||
await _unitOfWork.CommitAsync();
|
||||
await UnitOfWork.CommitAsync();
|
||||
|
||||
postLib = await _unitOfWork.LibraryRepository.GetLibraryForIdAsync(library.Id, LibraryIncludes.Series);
|
||||
postLib = await UnitOfWork.LibraryRepository.GetLibraryForIdAsync(library.Id, LibraryIncludes.Series);
|
||||
Assert.NotNull(postLib);
|
||||
Assert.Empty(postLib.Series);
|
||||
|
||||
await scanner.ScanLibrary(library.Id);
|
||||
postLib = await _unitOfWork.LibraryRepository.GetLibraryForIdAsync(library.Id, LibraryIncludes.Series);
|
||||
postLib = await UnitOfWork.LibraryRepository.GetLibraryForIdAsync(library.Id, LibraryIncludes.Series);
|
||||
|
||||
Assert.Contains(postLib.Series, s => s.Name == "Accel"); // Ensure Accel is gone
|
||||
Assert.Contains(postLib.Series, s => s.Name == "Plush");
|
||||
|
@ -768,13 +768,13 @@ public class ScannerServiceTests : AbstractDbTest
|
|||
var library = await _scannerHelper.GenerateScannerData(testcase, infos);
|
||||
var testDirectoryPath = library.Folders.First().Path;
|
||||
|
||||
_unitOfWork.LibraryRepository.Update(library);
|
||||
await _unitOfWork.CommitAsync();
|
||||
UnitOfWork.LibraryRepository.Update(library);
|
||||
await UnitOfWork.CommitAsync();
|
||||
|
||||
var scanner = _scannerHelper.CreateServices();
|
||||
await scanner.ScanLibrary(library.Id);
|
||||
|
||||
var postLib = await _unitOfWork.LibraryRepository.GetLibraryForIdAsync(library.Id, LibraryIncludes.Series);
|
||||
var postLib = await UnitOfWork.LibraryRepository.GetLibraryForIdAsync(library.Id, LibraryIncludes.Series);
|
||||
Assert.NotNull(postLib);
|
||||
Assert.Equal(4, postLib.Series.Count);
|
||||
|
||||
|
@ -800,9 +800,9 @@ public class ScannerServiceTests : AbstractDbTest
|
|||
Path.Join(executionerCopyDir, "The Executioner and Her Way of Life Vol. 1 Ch. 0002.cbz"));
|
||||
|
||||
await scanner.ScanLibrary(library.Id);
|
||||
await _unitOfWork.CommitAsync();
|
||||
await UnitOfWork.CommitAsync();
|
||||
|
||||
postLib = await _unitOfWork.LibraryRepository.GetLibraryForIdAsync(library.Id, LibraryIncludes.Series);
|
||||
postLib = await UnitOfWork.LibraryRepository.GetLibraryForIdAsync(library.Id, LibraryIncludes.Series);
|
||||
Assert.NotNull(postLib);
|
||||
Assert.Equal(4, postLib.Series.Count);
|
||||
|
||||
|
@ -827,13 +827,13 @@ public class ScannerServiceTests : AbstractDbTest
|
|||
var library = await _scannerHelper.GenerateScannerData(testcase, infos);
|
||||
var testDirectoryPath = library.Folders.First().Path;
|
||||
|
||||
_unitOfWork.LibraryRepository.Update(library);
|
||||
await _unitOfWork.CommitAsync();
|
||||
UnitOfWork.LibraryRepository.Update(library);
|
||||
await UnitOfWork.CommitAsync();
|
||||
|
||||
var scanner = _scannerHelper.CreateServices();
|
||||
await scanner.ScanLibrary(library.Id);
|
||||
|
||||
var postLib = await _unitOfWork.LibraryRepository.GetLibraryForIdAsync(library.Id, LibraryIncludes.Series);
|
||||
var postLib = await UnitOfWork.LibraryRepository.GetLibraryForIdAsync(library.Id, LibraryIncludes.Series);
|
||||
Assert.NotNull(postLib);
|
||||
Assert.Equal(2, postLib.Series.Count);
|
||||
|
||||
|
@ -841,9 +841,9 @@ public class ScannerServiceTests : AbstractDbTest
|
|||
Directory.Delete(executionerCopyDir, true);
|
||||
|
||||
await scanner.ScanLibrary(library.Id);
|
||||
await _unitOfWork.CommitAsync();
|
||||
await UnitOfWork.CommitAsync();
|
||||
|
||||
postLib = await _unitOfWork.LibraryRepository.GetLibraryForIdAsync(library.Id, LibraryIncludes.Series);
|
||||
postLib = await UnitOfWork.LibraryRepository.GetLibraryForIdAsync(library.Id, LibraryIncludes.Series);
|
||||
Assert.NotNull(postLib);
|
||||
Assert.Single(postLib.Series);
|
||||
Assert.Single(postLib.Series, s => s.Name == "Spice and Wolf");
|
||||
|
@ -859,13 +859,13 @@ public class ScannerServiceTests : AbstractDbTest
|
|||
var library = await _scannerHelper.GenerateScannerData(testcase, infos);
|
||||
var testDirectoryPath = library.Folders.First().Path;
|
||||
|
||||
_unitOfWork.LibraryRepository.Update(library);
|
||||
await _unitOfWork.CommitAsync();
|
||||
UnitOfWork.LibraryRepository.Update(library);
|
||||
await UnitOfWork.CommitAsync();
|
||||
|
||||
var scanner = _scannerHelper.CreateServices();
|
||||
await scanner.ScanLibrary(library.Id);
|
||||
|
||||
var postLib = await _unitOfWork.LibraryRepository.GetLibraryForIdAsync(library.Id, LibraryIncludes.Series);
|
||||
var postLib = await UnitOfWork.LibraryRepository.GetLibraryForIdAsync(library.Id, LibraryIncludes.Series);
|
||||
Assert.NotNull(postLib);
|
||||
Assert.Single(postLib.Series);
|
||||
|
||||
|
@ -882,7 +882,7 @@ public class ScannerServiceTests : AbstractDbTest
|
|||
|
||||
await scanner.ScanLibrary(library.Id);
|
||||
|
||||
postLib = await _unitOfWork.LibraryRepository.GetLibraryForIdAsync(library.Id, LibraryIncludes.Series);
|
||||
postLib = await UnitOfWork.LibraryRepository.GetLibraryForIdAsync(library.Id, LibraryIncludes.Series);
|
||||
Assert.NotNull(postLib);
|
||||
Assert.Single(postLib.Series);
|
||||
|
||||
|
@ -899,7 +899,7 @@ public class ScannerServiceTests : AbstractDbTest
|
|||
|
||||
await scanner.ScanLibrary(library.Id);
|
||||
|
||||
postLib = await _unitOfWork.LibraryRepository.GetLibraryForIdAsync(library.Id, LibraryIncludes.Series);
|
||||
postLib = await UnitOfWork.LibraryRepository.GetLibraryForIdAsync(library.Id, LibraryIncludes.Series);
|
||||
Assert.NotNull(postLib);
|
||||
Assert.Single(postLib.Series);
|
||||
|
||||
|
@ -923,7 +923,7 @@ public class ScannerServiceTests : AbstractDbTest
|
|||
|
||||
var scanner = _scannerHelper.CreateServices();
|
||||
await scanner.ScanLibrary(library.Id);
|
||||
var postLib = await _unitOfWork.LibraryRepository.GetLibraryForIdAsync(library.Id, LibraryIncludes.Series);
|
||||
var postLib = await UnitOfWork.LibraryRepository.GetLibraryForIdAsync(library.Id, LibraryIncludes.Series);
|
||||
Assert.NotNull(postLib);
|
||||
|
||||
// Get the loose leaf volume and confirm each chapter aligns with expectation of Sort Order
|
||||
|
|
|
@ -28,17 +28,17 @@ public class ScrobblingServiceTests : AbstractDbTest
|
|||
_logger = Substitute.For<ILogger<ScrobblingService>>();
|
||||
_emailService = Substitute.For<IEmailService>();
|
||||
|
||||
_service = new ScrobblingService(_unitOfWork, Substitute.For<IEventHub>(), _logger, _licenseService, _localizationService, _emailService);
|
||||
_service = new ScrobblingService(UnitOfWork, Substitute.For<IEventHub>(), _logger, _licenseService, _localizationService, _emailService);
|
||||
}
|
||||
|
||||
protected override async Task ResetDb()
|
||||
{
|
||||
_context.ScrobbleEvent.RemoveRange(_context.ScrobbleEvent.ToList());
|
||||
_context.Series.RemoveRange(_context.Series.ToList());
|
||||
_context.Library.RemoveRange(_context.Library.ToList());
|
||||
_context.AppUser.RemoveRange(_context.AppUser.ToList());
|
||||
Context.ScrobbleEvent.RemoveRange(Context.ScrobbleEvent.ToList());
|
||||
Context.Series.RemoveRange(Context.Series.ToList());
|
||||
Context.Library.RemoveRange(Context.Library.ToList());
|
||||
Context.AppUser.RemoveRange(Context.AppUser.ToList());
|
||||
|
||||
await _unitOfWork.CommitAsync();
|
||||
await UnitOfWork.CommitAsync();
|
||||
}
|
||||
|
||||
private async Task SeedData()
|
||||
|
@ -54,7 +54,7 @@ public class ScrobblingServiceTests : AbstractDbTest
|
|||
.Build();
|
||||
|
||||
|
||||
_context.Library.Add(library);
|
||||
Context.Library.Add(library);
|
||||
|
||||
var user = new AppUserBuilder("testuser", "testuser")
|
||||
//.WithPreferences(new UserPreferencesBuilder().WithAniListScrobblingEnabled(true).Build())
|
||||
|
@ -62,9 +62,9 @@ public class ScrobblingServiceTests : AbstractDbTest
|
|||
|
||||
user.UserPreferences.AniListScrobblingEnabled = true;
|
||||
|
||||
_unitOfWork.UserRepository.Add(user);
|
||||
UnitOfWork.UserRepository.Add(user);
|
||||
|
||||
await _unitOfWork.CommitAsync();
|
||||
await UnitOfWork.CommitAsync();
|
||||
}
|
||||
|
||||
#region ScrobbleWantToReadUpdate Tests
|
||||
|
@ -83,7 +83,7 @@ public class ScrobblingServiceTests : AbstractDbTest
|
|||
await _service.ScrobbleWantToReadUpdate(userId, seriesId, true);
|
||||
|
||||
// Assert
|
||||
var events = await _unitOfWork.ScrobbleRepository.GetAllEventsForSeries(seriesId);
|
||||
var events = await UnitOfWork.ScrobbleRepository.GetAllEventsForSeries(seriesId);
|
||||
Assert.Single(events);
|
||||
Assert.Equal(ScrobbleEventType.AddWantToRead, events[0].ScrobbleEventType);
|
||||
Assert.Equal(userId, events[0].AppUserId);
|
||||
|
@ -103,7 +103,7 @@ public class ScrobblingServiceTests : AbstractDbTest
|
|||
await _service.ScrobbleWantToReadUpdate(userId, seriesId, false);
|
||||
|
||||
// Assert
|
||||
var events = await _unitOfWork.ScrobbleRepository.GetAllEventsForSeries(seriesId);
|
||||
var events = await UnitOfWork.ScrobbleRepository.GetAllEventsForSeries(seriesId);
|
||||
Assert.Single(events);
|
||||
Assert.Equal(ScrobbleEventType.RemoveWantToRead, events[0].ScrobbleEventType);
|
||||
Assert.Equal(userId, events[0].AppUserId);
|
||||
|
@ -126,7 +126,7 @@ public class ScrobblingServiceTests : AbstractDbTest
|
|||
await _service.ScrobbleWantToReadUpdate(userId, seriesId, true);
|
||||
|
||||
// Assert
|
||||
var events = await _unitOfWork.ScrobbleRepository.GetAllEventsForSeries(seriesId);
|
||||
var events = await UnitOfWork.ScrobbleRepository.GetAllEventsForSeries(seriesId);
|
||||
|
||||
Assert.Single(events);
|
||||
Assert.All(events, e => Assert.Equal(ScrobbleEventType.AddWantToRead, e.ScrobbleEventType));
|
||||
|
@ -149,7 +149,7 @@ public class ScrobblingServiceTests : AbstractDbTest
|
|||
await _service.ScrobbleWantToReadUpdate(userId, seriesId, false);
|
||||
|
||||
// Assert
|
||||
var events = await _unitOfWork.ScrobbleRepository.GetAllEventsForSeries(seriesId);
|
||||
var events = await UnitOfWork.ScrobbleRepository.GetAllEventsForSeries(seriesId);
|
||||
|
||||
Assert.Single(events);
|
||||
Assert.Contains(events, e => e.ScrobbleEventType == ScrobbleEventType.RemoveWantToRead);
|
||||
|
@ -172,7 +172,7 @@ public class ScrobblingServiceTests : AbstractDbTest
|
|||
await _service.ScrobbleWantToReadUpdate(userId, seriesId, false);
|
||||
|
||||
// Assert
|
||||
var events = await _unitOfWork.ScrobbleRepository.GetAllEventsForSeries(seriesId);
|
||||
var events = await UnitOfWork.ScrobbleRepository.GetAllEventsForSeries(seriesId);
|
||||
|
||||
Assert.Single(events);
|
||||
Assert.All(events, e => Assert.Equal(ScrobbleEventType.RemoveWantToRead, e.ScrobbleEventType));
|
||||
|
@ -195,7 +195,7 @@ public class ScrobblingServiceTests : AbstractDbTest
|
|||
await _service.ScrobbleWantToReadUpdate(userId, seriesId, true);
|
||||
|
||||
// Assert
|
||||
var events = await _unitOfWork.ScrobbleRepository.GetAllEventsForSeries(seriesId);
|
||||
var events = await UnitOfWork.ScrobbleRepository.GetAllEventsForSeries(seriesId);
|
||||
|
||||
Assert.Single(events);
|
||||
Assert.Contains(events, e => e.ScrobbleEventType == ScrobbleEventType.AddWantToRead);
|
||||
|
|
File diff suppressed because it is too large
Load diff
|
@ -31,24 +31,24 @@ public abstract class SiteThemeServiceTest : AbstractDbTest
|
|||
|
||||
protected override async Task ResetDb()
|
||||
{
|
||||
_context.SiteTheme.RemoveRange(_context.SiteTheme);
|
||||
await _context.SaveChangesAsync();
|
||||
Context.SiteTheme.RemoveRange(Context.SiteTheme);
|
||||
await Context.SaveChangesAsync();
|
||||
// Recreate defaults
|
||||
await Seed.SeedThemes(_context);
|
||||
await Seed.SeedThemes(Context);
|
||||
}
|
||||
|
||||
[Fact]
|
||||
public async Task UpdateDefault_ShouldThrowOnInvalidId()
|
||||
{
|
||||
await ResetDb();
|
||||
_testOutputHelper.WriteLine($"[UpdateDefault_ShouldThrowOnInvalidId] All Themes: {(await _unitOfWork.SiteThemeRepository.GetThemes()).Count(t => t.IsDefault)}");
|
||||
_testOutputHelper.WriteLine($"[UpdateDefault_ShouldThrowOnInvalidId] All Themes: {(await UnitOfWork.SiteThemeRepository.GetThemes()).Count(t => t.IsDefault)}");
|
||||
var filesystem = CreateFileSystem();
|
||||
filesystem.AddFile($"{SiteThemeDirectory}custom.css", new MockFileData("123"));
|
||||
var ds = new DirectoryService(Substitute.For<ILogger<DirectoryService>>(), filesystem);
|
||||
var siteThemeService = new ThemeService(ds, _unitOfWork, _messageHub, Substitute.For<IFileService>(),
|
||||
var siteThemeService = new ThemeService(ds, UnitOfWork, _messageHub, Substitute.For<IFileService>(),
|
||||
Substitute.For<ILogger<ThemeService>>(), Substitute.For<IMemoryCache>());
|
||||
|
||||
_context.SiteTheme.Add(new SiteTheme()
|
||||
Context.SiteTheme.Add(new SiteTheme()
|
||||
{
|
||||
Name = "Custom",
|
||||
NormalizedName = "Custom".ToNormalized(),
|
||||
|
@ -56,7 +56,7 @@ public abstract class SiteThemeServiceTest : AbstractDbTest
|
|||
FileName = "custom.css",
|
||||
IsDefault = false
|
||||
});
|
||||
await _context.SaveChangesAsync();
|
||||
await Context.SaveChangesAsync();
|
||||
|
||||
var ex = await Assert.ThrowsAsync<KavitaException>(() => siteThemeService.UpdateDefault(10));
|
||||
Assert.Equal("Theme file missing or invalid", ex.Message);
|
||||
|
@ -68,14 +68,14 @@ public abstract class SiteThemeServiceTest : AbstractDbTest
|
|||
public async Task GetContent_ShouldReturnContent()
|
||||
{
|
||||
await ResetDb();
|
||||
_testOutputHelper.WriteLine($"[GetContent_ShouldReturnContent] All Themes: {(await _unitOfWork.SiteThemeRepository.GetThemes()).Count(t => t.IsDefault)}");
|
||||
_testOutputHelper.WriteLine($"[GetContent_ShouldReturnContent] All Themes: {(await UnitOfWork.SiteThemeRepository.GetThemes()).Count(t => t.IsDefault)}");
|
||||
var filesystem = CreateFileSystem();
|
||||
filesystem.AddFile($"{SiteThemeDirectory}custom.css", new MockFileData("123"));
|
||||
var ds = new DirectoryService(Substitute.For<ILogger<DirectoryService>>(), filesystem);
|
||||
var siteThemeService = new ThemeService(ds, _unitOfWork, _messageHub, Substitute.For<IFileService>(),
|
||||
var siteThemeService = new ThemeService(ds, UnitOfWork, _messageHub, Substitute.For<IFileService>(),
|
||||
Substitute.For<ILogger<ThemeService>>(), Substitute.For<IMemoryCache>());
|
||||
|
||||
_context.SiteTheme.Add(new SiteTheme()
|
||||
Context.SiteTheme.Add(new SiteTheme()
|
||||
{
|
||||
Name = "Custom",
|
||||
NormalizedName = "Custom".ToNormalized(),
|
||||
|
@ -83,9 +83,9 @@ public abstract class SiteThemeServiceTest : AbstractDbTest
|
|||
FileName = "custom.css",
|
||||
IsDefault = false
|
||||
});
|
||||
await _context.SaveChangesAsync();
|
||||
await Context.SaveChangesAsync();
|
||||
|
||||
var content = await siteThemeService.GetContent((await _unitOfWork.SiteThemeRepository.GetThemeDtoByName("Custom")).Id);
|
||||
var content = await siteThemeService.GetContent((await UnitOfWork.SiteThemeRepository.GetThemeDtoByName("Custom")).Id);
|
||||
Assert.NotNull(content);
|
||||
Assert.NotEmpty(content);
|
||||
Assert.Equal("123", content);
|
||||
|
@ -95,14 +95,14 @@ public abstract class SiteThemeServiceTest : AbstractDbTest
|
|||
public async Task UpdateDefault_ShouldHaveOneDefault()
|
||||
{
|
||||
await ResetDb();
|
||||
_testOutputHelper.WriteLine($"[UpdateDefault_ShouldHaveOneDefault] All Themes: {(await _unitOfWork.SiteThemeRepository.GetThemes()).Count(t => t.IsDefault)}");
|
||||
_testOutputHelper.WriteLine($"[UpdateDefault_ShouldHaveOneDefault] All Themes: {(await UnitOfWork.SiteThemeRepository.GetThemes()).Count(t => t.IsDefault)}");
|
||||
var filesystem = CreateFileSystem();
|
||||
filesystem.AddFile($"{SiteThemeDirectory}custom.css", new MockFileData("123"));
|
||||
var ds = new DirectoryService(Substitute.For<ILogger<DirectoryService>>(), filesystem);
|
||||
var siteThemeService = new ThemeService(ds, _unitOfWork, _messageHub, Substitute.For<IFileService>(),
|
||||
var siteThemeService = new ThemeService(ds, UnitOfWork, _messageHub, Substitute.For<IFileService>(),
|
||||
Substitute.For<ILogger<ThemeService>>(), Substitute.For<IMemoryCache>());
|
||||
|
||||
_context.SiteTheme.Add(new SiteTheme()
|
||||
Context.SiteTheme.Add(new SiteTheme()
|
||||
{
|
||||
Name = "Custom",
|
||||
NormalizedName = "Custom".ToNormalized(),
|
||||
|
@ -110,16 +110,16 @@ public abstract class SiteThemeServiceTest : AbstractDbTest
|
|||
FileName = "custom.css",
|
||||
IsDefault = false
|
||||
});
|
||||
await _context.SaveChangesAsync();
|
||||
await Context.SaveChangesAsync();
|
||||
|
||||
var customTheme = (await _unitOfWork.SiteThemeRepository.GetThemeDtoByName("Custom"));
|
||||
var customTheme = (await UnitOfWork.SiteThemeRepository.GetThemeDtoByName("Custom"));
|
||||
|
||||
Assert.NotNull(customTheme);
|
||||
await siteThemeService.UpdateDefault(customTheme.Id);
|
||||
|
||||
|
||||
|
||||
Assert.Equal(customTheme.Id, (await _unitOfWork.SiteThemeRepository.GetDefaultTheme()).Id);
|
||||
Assert.Equal(customTheme.Id, (await UnitOfWork.SiteThemeRepository.GetDefaultTheme()).Id);
|
||||
}
|
||||
|
||||
}
|
||||
|
|
BIN
API.Tests/Services/Test Data/CoverDbService/Existing/01.webp
Normal file
BIN
API.Tests/Services/Test Data/CoverDbService/Existing/01.webp
Normal file
Binary file not shown.
After Width: | Height: | Size: 17 KiB |
Binary file not shown.
After Width: | Height: | Size: 678 B |
|
@ -16,19 +16,15 @@ namespace API.Tests.Services;
|
|||
|
||||
public class VersionUpdaterServiceTests : IDisposable
|
||||
{
|
||||
private readonly ILogger<VersionUpdaterService> _logger;
|
||||
private readonly IEventHub _eventHub;
|
||||
private readonly IDirectoryService _directoryService;
|
||||
private readonly ILogger<VersionUpdaterService> _logger = Substitute.For<ILogger<VersionUpdaterService>>();
|
||||
private readonly IEventHub _eventHub = Substitute.For<IEventHub>();
|
||||
private readonly IDirectoryService _directoryService = Substitute.For<IDirectoryService>();
|
||||
private readonly VersionUpdaterService _service;
|
||||
private readonly string _tempPath;
|
||||
private readonly HttpTest _httpTest;
|
||||
|
||||
public VersionUpdaterServiceTests()
|
||||
{
|
||||
_logger = Substitute.For<ILogger<VersionUpdaterService>>();
|
||||
_eventHub = Substitute.For<IEventHub>();
|
||||
_directoryService = Substitute.For<IDirectoryService>();
|
||||
|
||||
// Create temp directory for cache
|
||||
_tempPath = Path.Combine(Path.GetTempPath(), Guid.NewGuid().ToString());
|
||||
Directory.CreateDirectory(_tempPath);
|
||||
|
@ -55,6 +51,7 @@ public class VersionUpdaterServiceTests : IDisposable
|
|||
|
||||
// Reset BuildInfo.Version
|
||||
typeof(BuildInfo).GetProperty(nameof(BuildInfo.Version))?.SetValue(null, null);
|
||||
GC.SuppressFinalize(this);
|
||||
}
|
||||
|
||||
[Fact]
|
||||
|
@ -302,7 +299,7 @@ public class VersionUpdaterServiceTests : IDisposable
|
|||
var result = await _service.GetAllReleases();
|
||||
|
||||
|
||||
Assert.Equal(1, result.Count);
|
||||
Assert.Single(result);
|
||||
Assert.Equal("0.7.0.0", result[0].UpdateVersion);
|
||||
Assert.NotEmpty(_httpTest.CallLog); // HTTP call was made
|
||||
}
|
||||
|
|
|
@ -26,9 +26,10 @@ public class WordCountAnalysisTests : AbstractDbTest
|
|||
private const long MinHoursToRead = 1;
|
||||
private const float AvgHoursToRead = 1.66954792f;
|
||||
private const long MaxHoursToRead = 3;
|
||||
public WordCountAnalysisTests() : base()
|
||||
|
||||
public WordCountAnalysisTests()
|
||||
{
|
||||
_readerService = new ReaderService(_unitOfWork, Substitute.For<ILogger<ReaderService>>(),
|
||||
_readerService = new ReaderService(UnitOfWork, Substitute.For<ILogger<ReaderService>>(),
|
||||
Substitute.For<IEventHub>(), Substitute.For<IImageService>(),
|
||||
new DirectoryService(Substitute.For<ILogger<DirectoryService>>(), new MockFileSystem()),
|
||||
Substitute.For<IScrobblingService>());
|
||||
|
@ -36,9 +37,9 @@ public class WordCountAnalysisTests : AbstractDbTest
|
|||
|
||||
protected override async Task ResetDb()
|
||||
{
|
||||
_context.Series.RemoveRange(_context.Series.ToList());
|
||||
Context.Series.RemoveRange(Context.Series.ToList());
|
||||
|
||||
await _context.SaveChangesAsync();
|
||||
await Context.SaveChangesAsync();
|
||||
}
|
||||
|
||||
[Fact]
|
||||
|
@ -56,7 +57,7 @@ public class WordCountAnalysisTests : AbstractDbTest
|
|||
MangaFormat.Epub).Build())
|
||||
.Build();
|
||||
|
||||
_context.Library.Add(new LibraryBuilder("Test LIb", LibraryType.Book)
|
||||
Context.Library.Add(new LibraryBuilder("Test LIb", LibraryType.Book)
|
||||
.WithSeries(series)
|
||||
.Build());
|
||||
|
||||
|
@ -67,11 +68,11 @@ public class WordCountAnalysisTests : AbstractDbTest
|
|||
.Build(),
|
||||
};
|
||||
|
||||
await _context.SaveChangesAsync();
|
||||
await Context.SaveChangesAsync();
|
||||
|
||||
|
||||
var cacheService = new CacheHelper(new FileService());
|
||||
var service = new WordCountAnalyzerService(Substitute.For<ILogger<WordCountAnalyzerService>>(), _unitOfWork,
|
||||
var service = new WordCountAnalyzerService(Substitute.For<ILogger<WordCountAnalyzerService>>(), UnitOfWork,
|
||||
Substitute.For<IEventHub>(), cacheService, _readerService, Substitute.For<IMediaErrorService>());
|
||||
|
||||
|
||||
|
@ -83,7 +84,7 @@ public class WordCountAnalysisTests : AbstractDbTest
|
|||
Assert.Equal(MaxHoursToRead, series.MaxHoursToRead);
|
||||
|
||||
// Validate the Chapter gets updated correctly
|
||||
var volume = series.Volumes.First();
|
||||
var volume = series.Volumes[0];
|
||||
Assert.Equal(WordCount, volume.WordCount);
|
||||
Assert.Equal(MinHoursToRead, volume.MinHoursToRead);
|
||||
Assert.Equal(AvgHoursToRead, volume.AvgHoursToRead);
|
||||
|
@ -114,16 +115,16 @@ public class WordCountAnalysisTests : AbstractDbTest
|
|||
.Build())
|
||||
.Build();
|
||||
|
||||
_context.Library.Add(new LibraryBuilder("Test", LibraryType.Book)
|
||||
Context.Library.Add(new LibraryBuilder("Test", LibraryType.Book)
|
||||
.WithSeries(series)
|
||||
.Build());
|
||||
|
||||
|
||||
await _context.SaveChangesAsync();
|
||||
await Context.SaveChangesAsync();
|
||||
|
||||
|
||||
var cacheService = new CacheHelper(new FileService());
|
||||
var service = new WordCountAnalyzerService(Substitute.For<ILogger<WordCountAnalyzerService>>(), _unitOfWork,
|
||||
var service = new WordCountAnalyzerService(Substitute.For<ILogger<WordCountAnalyzerService>>(), UnitOfWork,
|
||||
Substitute.For<IEventHub>(), cacheService, _readerService, Substitute.For<IMediaErrorService>());
|
||||
await service.ScanSeries(1, 1);
|
||||
|
||||
|
@ -139,21 +140,21 @@ public class WordCountAnalysisTests : AbstractDbTest
|
|||
.WithChapter(chapter2)
|
||||
.Build());
|
||||
|
||||
series.Volumes.First().Chapters.Add(chapter2);
|
||||
await _unitOfWork.CommitAsync();
|
||||
series.Volumes[0].Chapters.Add(chapter2);
|
||||
await UnitOfWork.CommitAsync();
|
||||
|
||||
await service.ScanSeries(1, 1);
|
||||
|
||||
Assert.Equal(WordCount * 2L, series.WordCount);
|
||||
Assert.Equal(MinHoursToRead * 2, series.MinHoursToRead);
|
||||
|
||||
var firstVolume = series.Volumes.ElementAt(0);
|
||||
var firstVolume = series.Volumes[0];
|
||||
Assert.Equal(WordCount, firstVolume.WordCount);
|
||||
Assert.Equal(MinHoursToRead, firstVolume.MinHoursToRead);
|
||||
Assert.True(series.AvgHoursToRead.Is(AvgHoursToRead * 2));
|
||||
Assert.Equal(MaxHoursToRead, firstVolume.MaxHoursToRead);
|
||||
|
||||
var secondVolume = series.Volumes.ElementAt(1);
|
||||
var secondVolume = series.Volumes[1];
|
||||
Assert.Equal(WordCount, secondVolume.WordCount);
|
||||
Assert.Equal(MinHoursToRead, secondVolume.MinHoursToRead);
|
||||
Assert.Equal(AvgHoursToRead, secondVolume.AvgHoursToRead);
|
||||
|
|
|
@ -8,6 +8,10 @@ public static class EasyCacheProfiles
|
|||
public const string RevokedJwt = "revokedJWT";
|
||||
public const string Favicon = "favicon";
|
||||
/// <summary>
|
||||
/// Images for Publishers
|
||||
/// </summary>
|
||||
public const string Publisher = "publisherImages";
|
||||
/// <summary>
|
||||
/// If a user's license is valid
|
||||
/// </summary>
|
||||
public const string License = "license";
|
||||
|
|
|
@ -6,6 +6,7 @@ using API.Constants;
|
|||
using API.Data;
|
||||
using API.Data.Repositories;
|
||||
using API.DTOs;
|
||||
using API.DTOs.SeriesDetail;
|
||||
using API.Entities;
|
||||
using API.Entities.Enums;
|
||||
using API.Entities.Person;
|
||||
|
@ -14,8 +15,10 @@ using API.Helpers;
|
|||
using API.Services;
|
||||
using API.Services.Tasks.Scanner.Parser;
|
||||
using API.SignalR;
|
||||
using AutoMapper;
|
||||
using Microsoft.AspNetCore.Authorization;
|
||||
using Microsoft.AspNetCore.Mvc;
|
||||
using Microsoft.EntityFrameworkCore;
|
||||
using Microsoft.Extensions.Logging;
|
||||
using Nager.ArticleNumber;
|
||||
|
||||
|
@ -27,13 +30,16 @@ public class ChapterController : BaseApiController
|
|||
private readonly ILocalizationService _localizationService;
|
||||
private readonly IEventHub _eventHub;
|
||||
private readonly ILogger<ChapterController> _logger;
|
||||
private readonly IMapper _mapper;
|
||||
|
||||
public ChapterController(IUnitOfWork unitOfWork, ILocalizationService localizationService, IEventHub eventHub, ILogger<ChapterController> logger)
|
||||
public ChapterController(IUnitOfWork unitOfWork, ILocalizationService localizationService, IEventHub eventHub, ILogger<ChapterController> logger,
|
||||
IMapper mapper)
|
||||
{
|
||||
_unitOfWork = unitOfWork;
|
||||
_localizationService = localizationService;
|
||||
_eventHub = eventHub;
|
||||
_logger = logger;
|
||||
_mapper = mapper;
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
|
@ -62,7 +68,8 @@ public class ChapterController : BaseApiController
|
|||
{
|
||||
if (User.IsInRole(PolicyConstants.ReadOnlyRole)) return BadRequest(await _localizationService.Translate(User.GetUserId(), "permission-denied"));
|
||||
|
||||
var chapter = await _unitOfWork.ChapterRepository.GetChapterAsync(chapterId);
|
||||
var chapter = await _unitOfWork.ChapterRepository.GetChapterAsync(chapterId,
|
||||
ChapterIncludes.Files | ChapterIncludes.ExternalReviews | ChapterIncludes.ExternalRatings);
|
||||
if (chapter == null)
|
||||
return BadRequest(_localizationService.Translate(User.GetUserId(), "chapter-doesnt-exist"));
|
||||
|
||||
|
@ -80,6 +87,15 @@ public class ChapterController : BaseApiController
|
|||
_unitOfWork.ChapterRepository.Remove(chapter);
|
||||
}
|
||||
|
||||
// If we removed the volume, do an additional check if we need to delete the actual series as well or not
|
||||
var series = await _unitOfWork.SeriesRepository.GetSeriesByIdAsync(vol.SeriesId, SeriesIncludes.ExternalData | SeriesIncludes.Volumes);
|
||||
var needToRemoveSeries = needToRemoveVolume && series != null && series.Volumes.Count <= 1;
|
||||
if (needToRemoveSeries)
|
||||
{
|
||||
_unitOfWork.SeriesRepository.Remove(series!);
|
||||
}
|
||||
|
||||
|
||||
|
||||
if (!await _unitOfWork.CommitAsync()) return Ok(false);
|
||||
|
||||
|
@ -89,6 +105,12 @@ public class ChapterController : BaseApiController
|
|||
await _eventHub.SendMessageAsync(MessageFactory.VolumeRemoved, MessageFactory.VolumeRemovedEvent(chapter.VolumeId, vol.SeriesId), false);
|
||||
}
|
||||
|
||||
if (needToRemoveSeries)
|
||||
{
|
||||
await _eventHub.SendMessageAsync(MessageFactory.SeriesRemoved,
|
||||
MessageFactory.SeriesRemovedEvent(series!.Id, series.Name, series.LibraryId), false);
|
||||
}
|
||||
|
||||
return Ok(true);
|
||||
}
|
||||
|
||||
|
@ -391,6 +413,39 @@ public class ChapterController : BaseApiController
|
|||
return Ok();
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Returns Ratings and Reviews for an individual Chapter
|
||||
/// </summary>
|
||||
/// <param name="chapterId"></param>
|
||||
/// <returns></returns>
|
||||
[HttpGet("chapter-detail-plus")]
|
||||
public async Task<ActionResult<ChapterDetailPlusDto>> ChapterDetailPlus([FromQuery] int chapterId)
|
||||
{
|
||||
var ret = new ChapterDetailPlusDto();
|
||||
|
||||
var userReviews = (await _unitOfWork.UserRepository.GetUserRatingDtosForChapterAsync(chapterId, User.GetUserId()))
|
||||
.Where(r => !string.IsNullOrEmpty(r.Body))
|
||||
.OrderByDescending(review => review.Username.Equals(User.GetUsername()) ? 1 : 0)
|
||||
.ToList();
|
||||
|
||||
var ownRating = await _unitOfWork.UserRepository.GetUserChapterRatingAsync(User.GetUserId(), chapterId);
|
||||
if (ownRating != null)
|
||||
{
|
||||
ret.Rating = ownRating.Rating;
|
||||
ret.HasBeenRated = ownRating.HasBeenRated;
|
||||
}
|
||||
|
||||
var externalReviews = await _unitOfWork.ChapterRepository.GetExternalChapterReviewDtos(chapterId);
|
||||
if (externalReviews.Count > 0)
|
||||
{
|
||||
userReviews.AddRange(ReviewHelper.SelectSpectrumOfReviews(externalReviews));
|
||||
}
|
||||
|
||||
ret.Reviews = userReviews;
|
||||
|
||||
ret.Ratings = await _unitOfWork.ChapterRepository.GetExternalChapterRatingDtos(chapterId);
|
||||
|
||||
return Ok(ret);
|
||||
}
|
||||
|
||||
}
|
||||
|
|
|
@ -221,7 +221,7 @@ public class MetadataController(IUnitOfWork unitOfWork, ILocalizationService loc
|
|||
return Ok(ret);
|
||||
}
|
||||
|
||||
private async Task PrepareSeriesDetail(List<UserReviewDto> userReviews, SeriesDetailPlusDto ret)
|
||||
private async Task PrepareSeriesDetail(List<UserReviewDto> userReviews, SeriesDetailPlusDto? ret)
|
||||
{
|
||||
var isAdmin = User.IsInRole(PolicyConstants.AdminRole);
|
||||
var user = await unitOfWork.UserRepository.GetUserByIdAsync(User.GetUserId())!;
|
||||
|
@ -235,12 +235,12 @@ public class MetadataController(IUnitOfWork unitOfWork, ILocalizationService loc
|
|||
ret.Recommendations.OwnedSeries =
|
||||
await unitOfWork.SeriesRepository.GetSeriesDtoByIdsAsync(
|
||||
ret.Recommendations.OwnedSeries.Select(s => s.Id), user);
|
||||
ret.Recommendations.ExternalSeries = new List<ExternalSeriesDto>();
|
||||
ret.Recommendations.ExternalSeries = [];
|
||||
}
|
||||
|
||||
if (ret.Recommendations != null && user != null)
|
||||
{
|
||||
ret.Recommendations.OwnedSeries ??= new List<SeriesDto>();
|
||||
ret.Recommendations.OwnedSeries ??= [];
|
||||
await unitOfWork.SeriesRepository.AddSeriesModifiers(user.Id, ret.Recommendations.OwnedSeries);
|
||||
}
|
||||
}
|
||||
|
|
|
@ -30,7 +30,7 @@ public class PluginController(IUnitOfWork unitOfWork, ITokenService tokenService
|
|||
public async Task<ActionResult<UserDto>> Authenticate([Required] string apiKey, [Required] string pluginName)
|
||||
{
|
||||
// NOTE: In order to log information about plugins, we need some Plugin Description information for each request
|
||||
// Should log into access table so we can tell the user
|
||||
// Should log into the access table so we can tell the user
|
||||
var ipAddress = HttpContext.Connection.RemoteIpAddress?.ToString();
|
||||
var userAgent = HttpContext.Request.Headers.UserAgent;
|
||||
var userId = await unitOfWork.UserRepository.GetUserIdByApiKeyAsync(apiKey);
|
||||
|
|
|
@ -1,15 +1,12 @@
|
|||
using System;
|
||||
using System.Collections.Generic;
|
||||
using System.Linq;
|
||||
using System.Threading.Tasks;
|
||||
using API.Constants;
|
||||
using API.Data;
|
||||
using API.Data.Repositories;
|
||||
using API.DTOs;
|
||||
using API.Extensions;
|
||||
using API.Services;
|
||||
using API.Services.Plus;
|
||||
using EasyCaching.Core;
|
||||
using Microsoft.AspNetCore.Mvc;
|
||||
using Microsoft.Extensions.Logging;
|
||||
|
||||
namespace API.Controllers;
|
||||
|
||||
|
@ -21,21 +18,85 @@ namespace API.Controllers;
|
|||
public class RatingController : BaseApiController
|
||||
{
|
||||
private readonly IUnitOfWork _unitOfWork;
|
||||
private readonly IRatingService _ratingService;
|
||||
private readonly ILocalizationService _localizationService;
|
||||
|
||||
public RatingController(IUnitOfWork unitOfWork)
|
||||
public RatingController(IUnitOfWork unitOfWork, IRatingService ratingService, ILocalizationService localizationService)
|
||||
{
|
||||
_unitOfWork = unitOfWork;
|
||||
|
||||
_ratingService = ratingService;
|
||||
_localizationService = localizationService;
|
||||
}
|
||||
|
||||
[HttpGet("overall")]
|
||||
public async Task<ActionResult<RatingDto>> GetOverallRating(int seriesId)
|
||||
/// <summary>
|
||||
/// Update the users' rating of the given series
|
||||
/// </summary>
|
||||
/// <param name="updateRating"></param>
|
||||
/// <returns></returns>
|
||||
/// <exception cref="UnauthorizedAccessException"></exception>
|
||||
[HttpPost("series")]
|
||||
public async Task<ActionResult> UpdateSeriesRating(UpdateRatingDto updateRating)
|
||||
{
|
||||
var user = await _unitOfWork.UserRepository.GetUserByIdAsync(User.GetUserId(), AppUserIncludes.Ratings | AppUserIncludes.ChapterRatings);
|
||||
if (user == null) throw new UnauthorizedAccessException();
|
||||
|
||||
if (await _ratingService.UpdateSeriesRating(user, updateRating))
|
||||
{
|
||||
return Ok();
|
||||
}
|
||||
|
||||
return BadRequest(await _localizationService.Translate(User.GetUserId(), "generic-error"));
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Update the users' rating of the given chapter
|
||||
/// </summary>
|
||||
/// <param name="updateRating">chapterId must be set</param>
|
||||
/// <returns></returns>
|
||||
/// <exception cref="UnauthorizedAccessException"></exception>
|
||||
[HttpPost("chapter")]
|
||||
public async Task<ActionResult> UpdateChapterRating(UpdateRatingDto updateRating)
|
||||
{
|
||||
var user = await _unitOfWork.UserRepository.GetUserByIdAsync(User.GetUserId(), AppUserIncludes.Ratings | AppUserIncludes.ChapterRatings);
|
||||
if (user == null) throw new UnauthorizedAccessException();
|
||||
|
||||
if (await _ratingService.UpdateChapterRating(user, updateRating))
|
||||
{
|
||||
return Ok();
|
||||
}
|
||||
|
||||
return BadRequest(await _localizationService.Translate(User.GetUserId(), "generic-error"));
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Overall rating from all Kavita users for a given Series
|
||||
/// </summary>
|
||||
/// <param name="seriesId"></param>
|
||||
/// <returns></returns>
|
||||
[HttpGet("overall-series")]
|
||||
public async Task<ActionResult<RatingDto>> GetOverallSeriesRating(int seriesId)
|
||||
{
|
||||
return Ok(new RatingDto()
|
||||
{
|
||||
Provider = ScrobbleProvider.Kavita,
|
||||
AverageScore = await _unitOfWork.SeriesRepository.GetAverageUserRating(seriesId, User.GetUserId()),
|
||||
FavoriteCount = 0
|
||||
FavoriteCount = 0,
|
||||
});
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Overall rating from all Kavita users for a given Chapter
|
||||
/// </summary>
|
||||
/// <param name="chapterId"></param>
|
||||
/// <returns></returns>
|
||||
[HttpGet("overall-chapter")]
|
||||
public async Task<ActionResult<RatingDto>> GetOverallChapterRating(int chapterId)
|
||||
{
|
||||
return Ok(new RatingDto()
|
||||
{
|
||||
Provider = ScrobbleProvider.Kavita,
|
||||
AverageScore = await _unitOfWork.ChapterRepository.GetAverageUserRating(chapterId, User.GetUserId()),
|
||||
FavoriteCount = 0,
|
||||
});
|
||||
}
|
||||
}
|
||||
|
|
|
@ -1,8 +1,11 @@
|
|||
using System.Linq;
|
||||
using System.Collections.Generic;
|
||||
using System.Linq;
|
||||
using System.Threading.Tasks;
|
||||
using API.Data;
|
||||
using API.Data.Repositories;
|
||||
using API.DTOs.SeriesDetail;
|
||||
using API.Entities;
|
||||
using API.Entities.Enums;
|
||||
using API.Extensions;
|
||||
using API.Helpers.Builders;
|
||||
using API.Services.Plus;
|
||||
|
@ -30,17 +33,17 @@ public class ReviewController : BaseApiController
|
|||
|
||||
|
||||
/// <summary>
|
||||
/// Updates the review for a given series
|
||||
/// Updates the user's review for a given series
|
||||
/// </summary>
|
||||
/// <param name="dto"></param>
|
||||
/// <returns></returns>
|
||||
[HttpPost]
|
||||
public async Task<ActionResult<UserReviewDto>> UpdateReview(UpdateUserReviewDto dto)
|
||||
[HttpPost("series")]
|
||||
public async Task<ActionResult<UserReviewDto>> UpdateSeriesReview(UpdateUserReviewDto dto)
|
||||
{
|
||||
var user = await _unitOfWork.UserRepository.GetUserByIdAsync(User.GetUserId(), AppUserIncludes.Ratings);
|
||||
if (user == null) return Unauthorized();
|
||||
|
||||
var ratingBuilder = new RatingBuilder(user.Ratings.FirstOrDefault(r => r.SeriesId == dto.SeriesId));
|
||||
var ratingBuilder = new RatingBuilder(await _unitOfWork.UserRepository.GetUserRatingAsync(dto.SeriesId, user.Id));
|
||||
|
||||
var rating = ratingBuilder
|
||||
.WithBody(dto.Body)
|
||||
|
@ -52,22 +55,58 @@ public class ReviewController : BaseApiController
|
|||
{
|
||||
user.Ratings.Add(rating);
|
||||
}
|
||||
|
||||
_unitOfWork.UserRepository.Update(user);
|
||||
|
||||
await _unitOfWork.CommitAsync();
|
||||
|
||||
|
||||
BackgroundJob.Enqueue(() =>
|
||||
_scrobblingService.ScrobbleReviewUpdate(user.Id, dto.SeriesId, string.Empty, dto.Body));
|
||||
return Ok(_mapper.Map<UserReviewDto>(rating));
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Update the user's review for a given chapter
|
||||
/// </summary>
|
||||
/// <param name="dto">chapterId must be set</param>
|
||||
/// <returns></returns>
|
||||
[HttpPost("chapter")]
|
||||
public async Task<ActionResult<UserReviewDto>> UpdateChapterReview(UpdateUserReviewDto dto)
|
||||
{
|
||||
var user = await _unitOfWork.UserRepository.GetUserByIdAsync(User.GetUserId(), AppUserIncludes.ChapterRatings);
|
||||
if (user == null) return Unauthorized();
|
||||
|
||||
if (dto.ChapterId == null) return BadRequest();
|
||||
|
||||
int chapterId = dto.ChapterId.Value;
|
||||
|
||||
var ratingBuilder = new ChapterRatingBuilder(await _unitOfWork.UserRepository.GetUserChapterRatingAsync(user.Id, chapterId));
|
||||
|
||||
var rating = ratingBuilder
|
||||
.WithBody(dto.Body)
|
||||
.WithSeriesId(dto.SeriesId)
|
||||
.WithChapterId(chapterId)
|
||||
.Build();
|
||||
|
||||
if (rating.Id == 0)
|
||||
{
|
||||
user.ChapterRatings.Add(rating);
|
||||
}
|
||||
|
||||
_unitOfWork.UserRepository.Update(user);
|
||||
|
||||
await _unitOfWork.CommitAsync();
|
||||
|
||||
return Ok(_mapper.Map<UserReviewDto>(rating));
|
||||
}
|
||||
|
||||
|
||||
/// <summary>
|
||||
/// Deletes the user's review for the given series
|
||||
/// </summary>
|
||||
/// <returns></returns>
|
||||
[HttpDelete]
|
||||
public async Task<ActionResult> DeleteReview(int seriesId)
|
||||
[HttpDelete("series")]
|
||||
public async Task<ActionResult> DeleteSeriesReview([FromQuery] int seriesId)
|
||||
{
|
||||
var user = await _unitOfWork.UserRepository.GetUserByIdAsync(User.GetUserId(), AppUserIncludes.Ratings);
|
||||
if (user == null) return Unauthorized();
|
||||
|
@ -80,4 +119,23 @@ public class ReviewController : BaseApiController
|
|||
|
||||
return Ok();
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Deletes the user's review for the given chapter
|
||||
/// </summary>
|
||||
/// <returns></returns>
|
||||
[HttpDelete("chapter")]
|
||||
public async Task<ActionResult> DeleteChapterReview([FromQuery] int chapterId)
|
||||
{
|
||||
var user = await _unitOfWork.UserRepository.GetUserByIdAsync(User.GetUserId(), AppUserIncludes.ChapterRatings);
|
||||
if (user == null) return Unauthorized();
|
||||
|
||||
user.ChapterRatings = user.ChapterRatings.Where(r => r.ChapterId != chapterId).ToList();
|
||||
|
||||
_unitOfWork.UserRepository.Update(user);
|
||||
|
||||
await _unitOfWork.CommitAsync();
|
||||
|
||||
return Ok();
|
||||
}
|
||||
}
|
||||
|
|
|
@ -191,21 +191,6 @@ public class SeriesController : BaseApiController
|
|||
return Ok(await _unitOfWork.ChapterRepository.GetChapterMetadataDtoAsync(chapterId));
|
||||
}
|
||||
|
||||
|
||||
/// <summary>
|
||||
/// Update the user rating for the given series
|
||||
/// </summary>
|
||||
/// <param name="updateSeriesRatingDto"></param>
|
||||
/// <returns></returns>
|
||||
[HttpPost("update-rating")]
|
||||
public async Task<ActionResult> UpdateSeriesRating(UpdateSeriesRatingDto updateSeriesRatingDto)
|
||||
{
|
||||
var user = await _unitOfWork.UserRepository.GetUserByUsernameAsync(User.GetUsername(), AppUserIncludes.Ratings);
|
||||
if (!await _seriesService.UpdateRating(user!, updateSeriesRatingDto))
|
||||
return BadRequest(await _localizationService.Translate(User.GetUserId(), "generic-error"));
|
||||
return Ok();
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Updates the Series
|
||||
/// </summary>
|
||||
|
|
|
@ -2,15 +2,15 @@
|
|||
|
||||
namespace API.DTOs.Account;
|
||||
|
||||
public class AgeRestrictionDto
|
||||
public sealed record AgeRestrictionDto
|
||||
{
|
||||
/// <summary>
|
||||
/// The maximum age rating a user has access to. -1 if not applicable
|
||||
/// </summary>
|
||||
public required AgeRating AgeRating { get; set; } = AgeRating.NotApplicable;
|
||||
public required AgeRating AgeRating { get; init; } = AgeRating.NotApplicable;
|
||||
/// <summary>
|
||||
/// Are Unknowns explicitly allowed against age rating
|
||||
/// </summary>
|
||||
/// <remarks>Unknown is always lowest and default age rating. Setting this to false will ensure Teen age rating applies and unknowns are still filtered</remarks>
|
||||
public required bool IncludeUnknowns { get; set; } = false;
|
||||
public required bool IncludeUnknowns { get; init; } = false;
|
||||
}
|
||||
|
|
|
@ -2,7 +2,7 @@
|
|||
|
||||
namespace API.DTOs.Account;
|
||||
|
||||
public class ConfirmEmailDto
|
||||
public sealed record ConfirmEmailDto
|
||||
{
|
||||
[Required]
|
||||
public string Email { get; set; } = default!;
|
||||
|
|
|
@ -2,7 +2,7 @@
|
|||
|
||||
namespace API.DTOs.Account;
|
||||
|
||||
public class ConfirmEmailUpdateDto
|
||||
public sealed record ConfirmEmailUpdateDto
|
||||
{
|
||||
[Required]
|
||||
public string Email { get; set; } = default!;
|
||||
|
|
|
@ -1,6 +1,6 @@
|
|||
namespace API.DTOs.Account;
|
||||
|
||||
public class ConfirmMigrationEmailDto
|
||||
public sealed record ConfirmMigrationEmailDto
|
||||
{
|
||||
public string Email { get; set; } = default!;
|
||||
public string Token { get; set; } = default!;
|
||||
|
|
|
@ -2,7 +2,7 @@
|
|||
|
||||
namespace API.DTOs.Account;
|
||||
|
||||
public class ConfirmPasswordResetDto
|
||||
public sealed record ConfirmPasswordResetDto
|
||||
{
|
||||
[Required]
|
||||
public string Email { get; set; } = default!;
|
||||
|
|
|
@ -3,7 +3,7 @@ using System.ComponentModel.DataAnnotations;
|
|||
|
||||
namespace API.DTOs.Account;
|
||||
|
||||
public class InviteUserDto
|
||||
public sealed record InviteUserDto
|
||||
{
|
||||
[Required]
|
||||
public string Email { get; set; } = default!;
|
||||
|
|
|
@ -1,6 +1,6 @@
|
|||
namespace API.DTOs.Account;
|
||||
|
||||
public class InviteUserResponse
|
||||
public sealed record InviteUserResponse
|
||||
{
|
||||
/// <summary>
|
||||
/// Email link used to setup the user account
|
||||
|
|
|
@ -1,7 +1,7 @@
|
|||
namespace API.DTOs.Account;
|
||||
#nullable enable
|
||||
|
||||
public class LoginDto
|
||||
public sealed record LoginDto
|
||||
{
|
||||
public string Username { get; init; } = default!;
|
||||
public string Password { get; set; } = default!;
|
||||
|
|
|
@ -1,6 +1,6 @@
|
|||
namespace API.DTOs.Account;
|
||||
|
||||
public class MigrateUserEmailDto
|
||||
public sealed record MigrateUserEmailDto
|
||||
{
|
||||
public string Email { get; set; } = default!;
|
||||
public string Username { get; set; } = default!;
|
||||
|
|
|
@ -2,7 +2,7 @@
|
|||
|
||||
namespace API.DTOs.Account;
|
||||
|
||||
public class ResetPasswordDto
|
||||
public sealed record ResetPasswordDto
|
||||
{
|
||||
/// <summary>
|
||||
/// The Username of the User
|
||||
|
|
|
@ -1,6 +1,6 @@
|
|||
namespace API.DTOs.Account;
|
||||
|
||||
public class TokenRequestDto
|
||||
public sealed record TokenRequestDto
|
||||
{
|
||||
public string Token { get; init; } = default!;
|
||||
public string RefreshToken { get; init; } = default!;
|
||||
|
|
|
@ -3,7 +3,7 @@ using API.Entities.Enums;
|
|||
|
||||
namespace API.DTOs.Account;
|
||||
|
||||
public class UpdateAgeRestrictionDto
|
||||
public sealed record UpdateAgeRestrictionDto
|
||||
{
|
||||
[Required]
|
||||
public AgeRating AgeRating { get; set; }
|
||||
|
|
|
@ -1,6 +1,6 @@
|
|||
namespace API.DTOs.Account;
|
||||
|
||||
public class UpdateEmailDto
|
||||
public sealed record UpdateEmailDto
|
||||
{
|
||||
public string Email { get; set; } = default!;
|
||||
public string Password { get; set; } = default!;
|
||||
|
|
|
@ -4,12 +4,16 @@ using System.ComponentModel.DataAnnotations;
|
|||
namespace API.DTOs.Account;
|
||||
#nullable enable
|
||||
|
||||
public record UpdateUserDto
|
||||
public sealed record UpdateUserDto
|
||||
{
|
||||
/// <inheritdoc cref="API.Entities.AppUser.Id"/>
|
||||
public int UserId { get; set; }
|
||||
/// <inheritdoc cref="API.Entities.AppUser.UserName"/>
|
||||
public string Username { get; set; } = default!;
|
||||
/// <summary>
|
||||
/// List of Roles to assign to user. If admin not present, Pleb will be applied.
|
||||
/// If admin present, all libraries will be granted access and will ignore those from DTO.
|
||||
/// </summary>
|
||||
public IList<string> Roles { get; init; } = default!;
|
||||
/// <summary>
|
||||
/// A list of libraries to grant access to
|
||||
|
@ -19,8 +23,6 @@ public record UpdateUserDto
|
|||
/// An Age Rating which will limit the account to seeing everything equal to or below said rating.
|
||||
/// </summary>
|
||||
public AgeRestrictionDto AgeRestriction { get; init; } = default!;
|
||||
/// <summary>
|
||||
/// Email of the user
|
||||
/// </summary>
|
||||
/// <inheritdoc cref="API.Entities.AppUser.Email"/>
|
||||
public string? Email { get; set; } = default!;
|
||||
}
|
||||
|
|
|
@ -2,7 +2,7 @@
|
|||
|
||||
namespace API.DTOs;
|
||||
|
||||
public class BulkActionDto
|
||||
public sealed record BulkActionDto
|
||||
{
|
||||
public List<int> Ids { get; set; }
|
||||
/**
|
||||
|
|
14
API/DTOs/ChapterDetailPlusDto.cs
Normal file
14
API/DTOs/ChapterDetailPlusDto.cs
Normal file
|
@ -0,0 +1,14 @@
|
|||
#nullable enable
|
||||
using System.Collections.Generic;
|
||||
using API.DTOs.SeriesDetail;
|
||||
|
||||
namespace API.DTOs;
|
||||
|
||||
public sealed record ChapterDetailPlusDto
|
||||
{
|
||||
public float Rating { get; set; }
|
||||
public bool HasBeenRated { get; set; }
|
||||
|
||||
public IList<UserReviewDto> Reviews { get; set; } = [];
|
||||
public IList<RatingDto> Ratings { get; set; } = [];
|
||||
}
|
|
@ -13,37 +13,24 @@ namespace API.DTOs;
|
|||
/// </summary>
|
||||
public class ChapterDto : IHasReadTimeEstimate, IHasCoverImage
|
||||
{
|
||||
/// <inheritdoc cref="API.Entities.Chapter.Id"/>
|
||||
public int Id { get; init; }
|
||||
/// <summary>
|
||||
/// Range of chapters. Chapter 2-4 -> "2-4". Chapter 2 -> "2". If special, will be special name.
|
||||
/// </summary>
|
||||
/// <remarks>This can be something like 19.HU or Alpha as some comics are like this</remarks>
|
||||
/// <inheritdoc cref="API.Entities.Chapter.Range"/>
|
||||
public string Range { get; init; } = default!;
|
||||
/// <summary>
|
||||
/// Smallest number of the Range.
|
||||
/// </summary>
|
||||
/// <inheritdoc cref="API.Entities.Chapter.Number"/>
|
||||
[Obsolete("Use MinNumber and MaxNumber instead")]
|
||||
public string Number { get; init; } = default!;
|
||||
/// <summary>
|
||||
/// This may be 0 under the circumstance that the Issue is "Alpha" or other non-standard numbers.
|
||||
/// </summary>
|
||||
/// <inheritdoc cref="API.Entities.Chapter.MinNumber"/>
|
||||
public float MinNumber { get; init; }
|
||||
/// <inheritdoc cref="API.Entities.Chapter.MaxNumber"/>
|
||||
public float MaxNumber { get; init; }
|
||||
/// <summary>
|
||||
/// The sorting order of the Chapter. Inherits from MinNumber, but can be overridden.
|
||||
/// </summary>
|
||||
/// <inheritdoc cref="API.Entities.Chapter.SortOrder"/>
|
||||
public float SortOrder { get; set; }
|
||||
/// <summary>
|
||||
/// Total number of pages in all MangaFiles
|
||||
/// </summary>
|
||||
/// <inheritdoc cref="API.Entities.Chapter.Pages"/>
|
||||
public int Pages { get; init; }
|
||||
/// <summary>
|
||||
/// If this Chapter contains files that could only be identified as Series or has Special Identifier from filename
|
||||
/// </summary>
|
||||
/// <inheritdoc cref="API.Entities.Chapter.IsSpecial"/>
|
||||
public bool IsSpecial { get; init; }
|
||||
/// <summary>
|
||||
/// Used for books/specials to display custom title. For non-specials/books, will be set to <see cref="Range"/>
|
||||
/// </summary>
|
||||
/// <inheritdoc cref="API.Entities.Chapter.Title"/>
|
||||
public string Title { get; set; } = default!;
|
||||
/// <summary>
|
||||
/// The files that represent this Chapter
|
||||
|
@ -61,46 +48,25 @@ public class ChapterDto : IHasReadTimeEstimate, IHasCoverImage
|
|||
/// The last time a chapter was read by current authenticated user
|
||||
/// </summary>
|
||||
public DateTime LastReadingProgress { get; set; }
|
||||
/// <summary>
|
||||
/// If the Cover Image is locked for this entity
|
||||
/// </summary>
|
||||
/// <inheritdoc cref="API.Entities.Chapter.CoverImageLocked"/>
|
||||
public bool CoverImageLocked { get; set; }
|
||||
/// <summary>
|
||||
/// Volume Id this Chapter belongs to
|
||||
/// </summary>
|
||||
/// <inheritdoc cref="API.Entities.Chapter.VolumeId"/>
|
||||
public int VolumeId { get; init; }
|
||||
/// <summary>
|
||||
/// When chapter was created
|
||||
/// </summary>
|
||||
/// <inheritdoc cref="API.Entities.Chapter.CreatedUtc"/>
|
||||
public DateTime CreatedUtc { get; set; }
|
||||
/// <inheritdoc cref="API.Entities.Chapter.LastModifiedUtc"/>
|
||||
public DateTime LastModifiedUtc { get; set; }
|
||||
/// <summary>
|
||||
/// When chapter was created in local server time
|
||||
/// </summary>
|
||||
/// <remarks>This is required for Tachiyomi Extension</remarks>
|
||||
/// <inheritdoc cref="API.Entities.Chapter.Created"/>
|
||||
public DateTime Created { get; set; }
|
||||
/// <summary>
|
||||
/// When the chapter was released.
|
||||
/// </summary>
|
||||
/// <remarks>Metadata field</remarks>
|
||||
/// <inheritdoc cref="API.Entities.Chapter.ReleaseDate"/>
|
||||
public DateTime ReleaseDate { get; init; }
|
||||
/// <summary>
|
||||
/// Title of the Chapter/Issue
|
||||
/// </summary>
|
||||
/// <remarks>Metadata field</remarks>
|
||||
/// <inheritdoc cref="API.Entities.Chapter.TitleName"/>
|
||||
public string TitleName { get; set; } = default!;
|
||||
/// <summary>
|
||||
/// Summary of the Chapter
|
||||
/// </summary>
|
||||
/// <remarks>This is not set normally, only for Series Detail</remarks>
|
||||
/// <inheritdoc cref="API.Entities.Chapter.Summary"/>
|
||||
public string Summary { get; init; } = default!;
|
||||
/// <summary>
|
||||
/// Age Rating for the issue/chapter
|
||||
/// </summary>
|
||||
/// <inheritdoc cref="API.Entities.Chapter.AgeRating"/>
|
||||
public AgeRating AgeRating { get; init; }
|
||||
/// <summary>
|
||||
/// Total words in a Chapter (books only)
|
||||
/// </summary>
|
||||
/// <inheritdoc cref="API.Entities.Chapter.WordCount"/>
|
||||
public long WordCount { get; set; } = 0L;
|
||||
/// <summary>
|
||||
/// Formatted Volume title ie) Volume 2.
|
||||
|
@ -113,14 +79,9 @@ public class ChapterDto : IHasReadTimeEstimate, IHasCoverImage
|
|||
public int MaxHoursToRead { get; set; }
|
||||
/// <inheritdoc cref="IHasReadTimeEstimate.AvgHoursToRead"/>
|
||||
public float AvgHoursToRead { get; set; }
|
||||
/// <summary>
|
||||
/// Comma-separated link of urls to external services that have some relation to the Chapter
|
||||
/// </summary>
|
||||
/// <inheritdoc cref="API.Entities.Chapter.WebLinks"/>
|
||||
public string WebLinks { get; set; }
|
||||
/// <summary>
|
||||
/// ISBN-13 (usually) of the Chapter
|
||||
/// </summary>
|
||||
/// <remarks>This is guaranteed to be Valid</remarks>
|
||||
/// <inheritdoc cref="API.Entities.Chapter.ISBN"/>
|
||||
public string ISBN { get; set; }
|
||||
|
||||
#region Metadata
|
||||
|
@ -146,51 +107,60 @@ public class ChapterDto : IHasReadTimeEstimate, IHasCoverImage
|
|||
/// </summary>
|
||||
public ICollection<TagDto> Tags { get; set; } = new List<TagDto>();
|
||||
public PublicationStatus PublicationStatus { get; set; }
|
||||
/// <summary>
|
||||
/// Language for the Chapter/Issue
|
||||
/// </summary>
|
||||
/// <inheritdoc cref="API.Entities.Chapter.Language"/>
|
||||
public string? Language { get; set; }
|
||||
/// <summary>
|
||||
/// Number in the TotalCount of issues
|
||||
/// </summary>
|
||||
/// <inheritdoc cref="API.Entities.Chapter.Count"/>
|
||||
public int Count { get; set; }
|
||||
/// <summary>
|
||||
/// Total number of issues for the series
|
||||
/// </summary>
|
||||
/// <inheritdoc cref="API.Entities.Chapter.TotalCount"/>
|
||||
public int TotalCount { get; set; }
|
||||
|
||||
/// <inheritdoc cref="API.Entities.Chapter.LanguageLocked"/>
|
||||
public bool LanguageLocked { get; set; }
|
||||
/// <inheritdoc cref="API.Entities.Chapter.SummaryLocked"/>
|
||||
public bool SummaryLocked { get; set; }
|
||||
/// <summary>
|
||||
/// Locked by user so metadata updates from scan loop will not override AgeRating
|
||||
/// </summary>
|
||||
/// <inheritdoc cref="API.Entities.Chapter.AgeRatingLocked"/>
|
||||
public bool AgeRatingLocked { get; set; }
|
||||
/// <summary>
|
||||
/// Locked by user so metadata updates from scan loop will not override PublicationStatus
|
||||
/// </summary>
|
||||
public bool PublicationStatusLocked { get; set; }
|
||||
/// <inheritdoc cref="API.Entities.Chapter.GenresLocked"/>
|
||||
public bool GenresLocked { get; set; }
|
||||
/// <inheritdoc cref="API.Entities.Chapter.TagsLocked"/>
|
||||
public bool TagsLocked { get; set; }
|
||||
/// <inheritdoc cref="API.Entities.Chapter.WriterLocked"/>
|
||||
public bool WriterLocked { get; set; }
|
||||
/// <inheritdoc cref="API.Entities.Chapter.CharacterLocked"/>
|
||||
public bool CharacterLocked { get; set; }
|
||||
/// <inheritdoc cref="API.Entities.Chapter.ColoristLocked"/>
|
||||
public bool ColoristLocked { get; set; }
|
||||
/// <inheritdoc cref="API.Entities.Chapter.EditorLocked"/>
|
||||
public bool EditorLocked { get; set; }
|
||||
/// <inheritdoc cref="API.Entities.Chapter.InkerLocked"/>
|
||||
public bool InkerLocked { get; set; }
|
||||
/// <inheritdoc cref="API.Entities.Chapter.ImprintLocked"/>
|
||||
public bool ImprintLocked { get; set; }
|
||||
/// <inheritdoc cref="API.Entities.Chapter.LettererLocked"/>
|
||||
public bool LettererLocked { get; set; }
|
||||
/// <inheritdoc cref="API.Entities.Chapter.PencillerLocked"/>
|
||||
public bool PencillerLocked { get; set; }
|
||||
/// <inheritdoc cref="API.Entities.Chapter.PublisherLocked"/>
|
||||
public bool PublisherLocked { get; set; }
|
||||
/// <inheritdoc cref="API.Entities.Chapter.TranslatorLocked"/>
|
||||
public bool TranslatorLocked { get; set; }
|
||||
/// <inheritdoc cref="API.Entities.Chapter.TeamLocked"/>
|
||||
public bool TeamLocked { get; set; }
|
||||
/// <inheritdoc cref="API.Entities.Chapter.LocationLocked"/>
|
||||
public bool LocationLocked { get; set; }
|
||||
/// <inheritdoc cref="API.Entities.Chapter.CoverArtistLocked"/>
|
||||
public bool CoverArtistLocked { get; set; }
|
||||
public bool ReleaseYearLocked { get; set; }
|
||||
|
||||
#endregion
|
||||
|
||||
public string CoverImage { get; set; }
|
||||
public string PrimaryColor { get; set; } = string.Empty;
|
||||
public string SecondaryColor { get; set; } = string.Empty;
|
||||
/// <inheritdoc cref="API.Entities.Chapter.CoverImage"/>
|
||||
public string? CoverImage { get; set; }
|
||||
/// <inheritdoc cref="API.Entities.Chapter.PrimaryColor"/>
|
||||
public string? PrimaryColor { get; set; } = string.Empty;
|
||||
/// <inheritdoc cref="API.Entities.Chapter.SecondaryColor"/>
|
||||
public string? SecondaryColor { get; set; } = string.Empty;
|
||||
|
||||
public void ResetColorScape()
|
||||
{
|
||||
|
|
|
@ -6,52 +6,52 @@ using API.Services.Plus;
|
|||
namespace API.DTOs.Collection;
|
||||
#nullable enable
|
||||
|
||||
public class AppUserCollectionDto : IHasCoverImage
|
||||
public sealed record AppUserCollectionDto : IHasCoverImage
|
||||
{
|
||||
public int Id { get; init; }
|
||||
public string Title { get; set; } = default!;
|
||||
public string? Summary { get; set; } = default!;
|
||||
public bool Promoted { get; set; }
|
||||
public AgeRating AgeRating { get; set; }
|
||||
public string Title { get; init; } = default!;
|
||||
public string? Summary { get; init; } = default!;
|
||||
public bool Promoted { get; init; }
|
||||
public AgeRating AgeRating { get; init; }
|
||||
|
||||
/// <summary>
|
||||
/// This is used to tell the UI if it should request a Cover Image or not. If null or empty, it has not been set.
|
||||
/// </summary>
|
||||
public string? CoverImage { get; set; } = string.Empty;
|
||||
|
||||
public string PrimaryColor { get; set; } = string.Empty;
|
||||
public string SecondaryColor { get; set; } = string.Empty;
|
||||
public bool CoverImageLocked { get; set; }
|
||||
public string? PrimaryColor { get; set; } = string.Empty;
|
||||
public string? SecondaryColor { get; set; } = string.Empty;
|
||||
public bool CoverImageLocked { get; init; }
|
||||
|
||||
/// <summary>
|
||||
/// Number of Series in the Collection
|
||||
/// </summary>
|
||||
public int ItemCount { get; set; }
|
||||
public int ItemCount { get; init; }
|
||||
|
||||
/// <summary>
|
||||
/// Owner of the Collection
|
||||
/// </summary>
|
||||
public string? Owner { get; set; }
|
||||
public string? Owner { get; init; }
|
||||
/// <summary>
|
||||
/// Last time Kavita Synced the Collection with an upstream source (for non Kavita sourced collections)
|
||||
/// </summary>
|
||||
public DateTime LastSyncUtc { get; set; }
|
||||
public DateTime LastSyncUtc { get; init; }
|
||||
/// <summary>
|
||||
/// Who created/manages the list. Non-Kavita lists are not editable by the user, except to promote
|
||||
/// </summary>
|
||||
public ScrobbleProvider Source { get; set; } = ScrobbleProvider.Kavita;
|
||||
public ScrobbleProvider Source { get; init; } = ScrobbleProvider.Kavita;
|
||||
/// <summary>
|
||||
/// For Non-Kavita sourced collections, the url to sync from
|
||||
/// </summary>
|
||||
public string? SourceUrl { get; set; }
|
||||
public string? SourceUrl { get; init; }
|
||||
/// <summary>
|
||||
/// Total number of items as of the last sync. Not applicable for Kavita managed collections.
|
||||
/// </summary>
|
||||
public int TotalSourceCount { get; set; }
|
||||
public int TotalSourceCount { get; init; }
|
||||
/// <summary>
|
||||
/// A <br/> separated string of all missing series
|
||||
/// </summary>
|
||||
public string? MissingSeriesFromSource { get; set; }
|
||||
public string? MissingSeriesFromSource { get; init; }
|
||||
|
||||
public void ResetColorScape()
|
||||
{
|
||||
|
|
|
@ -2,7 +2,7 @@
|
|||
|
||||
namespace API.DTOs.CollectionTags;
|
||||
|
||||
public class CollectionTagBulkAddDto
|
||||
public sealed record CollectionTagBulkAddDto
|
||||
{
|
||||
/// <summary>
|
||||
/// Collection Tag Id
|
||||
|
|
|
@ -3,15 +3,21 @@
|
|||
namespace API.DTOs.CollectionTags;
|
||||
|
||||
[Obsolete("Use AppUserCollectionDto")]
|
||||
public class CollectionTagDto
|
||||
public sealed record CollectionTagDto
|
||||
{
|
||||
/// <inheritdoc cref="API.Entities.CollectionTag.Id"/>
|
||||
public int Id { get; set; }
|
||||
/// <inheritdoc cref="API.Entities.CollectionTag.Title"/>
|
||||
public string Title { get; set; } = default!;
|
||||
/// <inheritdoc cref="API.Entities.CollectionTag.Summary"/>
|
||||
public string Summary { get; set; } = default!;
|
||||
/// <inheritdoc cref="API.Entities.CollectionTag.Promoted"/>
|
||||
public bool Promoted { get; set; }
|
||||
/// <summary>
|
||||
/// The cover image string. This is used on Frontend to show or hide the Cover Image
|
||||
/// </summary>
|
||||
/// <inheritdoc cref="API.Entities.CollectionTag.CoverImage"/>
|
||||
public string CoverImage { get; set; } = default!;
|
||||
/// <inheritdoc cref="API.Entities.CollectionTag.CoverImageLocked"/>
|
||||
public bool CoverImageLocked { get; set; }
|
||||
}
|
||||
|
|
|
@ -4,7 +4,7 @@ using API.DTOs.Collection;
|
|||
|
||||
namespace API.DTOs.CollectionTags;
|
||||
|
||||
public class UpdateSeriesForTagDto
|
||||
public sealed record UpdateSeriesForTagDto
|
||||
{
|
||||
public AppUserCollectionDto Tag { get; init; } = default!;
|
||||
public IEnumerable<int> SeriesIdsToRemove { get; init; } = default!;
|
||||
|
|
|
@ -4,7 +4,7 @@
|
|||
/// <summary>
|
||||
/// A primary and secondary color
|
||||
/// </summary>
|
||||
public class ColorScape
|
||||
public sealed record ColorScape
|
||||
{
|
||||
public required string? Primary { get; set; }
|
||||
public required string? Secondary { get; set; }
|
||||
|
|
|
@ -2,7 +2,7 @@
|
|||
|
||||
namespace API.DTOs;
|
||||
|
||||
public class CopySettingsFromLibraryDto
|
||||
public sealed record CopySettingsFromLibraryDto
|
||||
{
|
||||
public int SourceLibraryId { get; set; }
|
||||
public List<int> TargetLibraryIds { get; set; }
|
||||
|
|
|
@ -3,7 +3,7 @@ using YamlDotNet.Serialization;
|
|||
|
||||
namespace API.DTOs.CoverDb;
|
||||
|
||||
public class CoverDbAuthor
|
||||
public sealed record CoverDbAuthor
|
||||
{
|
||||
[YamlMember(Alias = "name", ApplyNamingConventions = false)]
|
||||
public string Name { get; set; }
|
||||
|
|
|
@ -3,7 +3,7 @@ using YamlDotNet.Serialization;
|
|||
|
||||
namespace API.DTOs.CoverDb;
|
||||
|
||||
public class CoverDbPeople
|
||||
public sealed record CoverDbPeople
|
||||
{
|
||||
[YamlMember(Alias = "people", ApplyNamingConventions = false)]
|
||||
public List<CoverDbAuthor> People { get; set; } = new List<CoverDbAuthor>();
|
||||
|
|
|
@ -3,7 +3,7 @@
|
|||
namespace API.DTOs.CoverDb;
|
||||
#nullable enable
|
||||
|
||||
public class CoverDbPersonIds
|
||||
public sealed record CoverDbPersonIds
|
||||
{
|
||||
[YamlMember(Alias = "hardcover_id", ApplyNamingConventions = false)]
|
||||
public string? HardcoverId { get; set; } = null;
|
||||
|
|
|
@ -4,7 +4,7 @@ using API.Entities.Enums;
|
|||
|
||||
namespace API.DTOs.Dashboard;
|
||||
|
||||
public class DashboardStreamDto
|
||||
public sealed record DashboardStreamDto
|
||||
{
|
||||
public int Id { get; set; }
|
||||
public required string Name { get; set; }
|
||||
|
|
|
@ -5,7 +5,7 @@ namespace API.DTOs.Dashboard;
|
|||
/// <summary>
|
||||
/// This is a representation of a Series with some amount of underlying files within it. This is used for Recently Updated Series section
|
||||
/// </summary>
|
||||
public class GroupedSeriesDto
|
||||
public sealed record GroupedSeriesDto
|
||||
{
|
||||
public string SeriesName { get; set; } = default!;
|
||||
public int SeriesId { get; set; }
|
||||
|
|
|
@ -6,7 +6,7 @@ namespace API.DTOs.Dashboard;
|
|||
/// <summary>
|
||||
/// A mesh of data for Recently added volume/chapters
|
||||
/// </summary>
|
||||
public class RecentlyAddedItemDto
|
||||
public sealed record RecentlyAddedItemDto
|
||||
{
|
||||
public string SeriesName { get; set; } = default!;
|
||||
public int SeriesId { get; set; }
|
||||
|
|
|
@ -2,7 +2,7 @@
|
|||
|
||||
namespace API.DTOs.Dashboard;
|
||||
|
||||
public class SmartFilterDto
|
||||
public sealed record SmartFilterDto
|
||||
{
|
||||
public int Id { get; set; }
|
||||
public required string Name { get; set; }
|
||||
|
|
|
@ -1,6 +1,6 @@
|
|||
namespace API.DTOs.Dashboard;
|
||||
|
||||
public class UpdateDashboardStreamPositionDto
|
||||
public sealed record UpdateDashboardStreamPositionDto
|
||||
{
|
||||
public int FromPosition { get; set; }
|
||||
public int ToPosition { get; set; }
|
||||
|
|
|
@ -1,6 +1,6 @@
|
|||
namespace API.DTOs.Dashboard;
|
||||
|
||||
public class UpdateStreamPositionDto
|
||||
public sealed record UpdateStreamPositionDto
|
||||
{
|
||||
public int FromPosition { get; set; }
|
||||
public int ToPosition { get; set; }
|
||||
|
|
|
@ -2,7 +2,7 @@
|
|||
|
||||
namespace API.DTOs;
|
||||
|
||||
public class DeleteChaptersDto
|
||||
public sealed record DeleteChaptersDto
|
||||
{
|
||||
public IList<int> ChapterIds { get; set; } = default!;
|
||||
}
|
||||
|
|
|
@ -2,7 +2,7 @@
|
|||
|
||||
namespace API.DTOs;
|
||||
|
||||
public class DeleteSeriesDto
|
||||
public sealed record DeleteSeriesDto
|
||||
{
|
||||
public IList<int> SeriesIds { get; set; } = default!;
|
||||
}
|
||||
|
|
|
@ -3,7 +3,7 @@ using API.Entities.Enums.Device;
|
|||
|
||||
namespace API.DTOs.Device;
|
||||
|
||||
public class CreateDeviceDto
|
||||
public sealed record CreateDeviceDto
|
||||
{
|
||||
[Required]
|
||||
public string Name { get; set; } = default!;
|
||||
|
|
|
@ -6,7 +6,7 @@ namespace API.DTOs.Device;
|
|||
/// <summary>
|
||||
/// A Device is an entity that can receive data from Kavita (kindle)
|
||||
/// </summary>
|
||||
public class DeviceDto
|
||||
public sealed record DeviceDto
|
||||
{
|
||||
/// <summary>
|
||||
/// The device Id
|
||||
|
|
|
@ -1,6 +1,6 @@
|
|||
namespace API.DTOs.Device;
|
||||
|
||||
public class SendSeriesToDeviceDto
|
||||
public sealed record SendSeriesToDeviceDto
|
||||
{
|
||||
public int DeviceId { get; set; }
|
||||
public int SeriesId { get; set; }
|
||||
|
|
|
@ -2,7 +2,7 @@
|
|||
|
||||
namespace API.DTOs.Device;
|
||||
|
||||
public class SendToDeviceDto
|
||||
public sealed record SendToDeviceDto
|
||||
{
|
||||
public int DeviceId { get; set; }
|
||||
public IReadOnlyList<int> ChapterIds { get; set; } = default!;
|
||||
|
|
|
@ -3,7 +3,7 @@ using API.Entities.Enums.Device;
|
|||
|
||||
namespace API.DTOs.Device;
|
||||
|
||||
public class UpdateDeviceDto
|
||||
public sealed record UpdateDeviceDto
|
||||
{
|
||||
[Required]
|
||||
public int Id { get; set; }
|
||||
|
|
|
@ -4,7 +4,7 @@ using API.DTOs.Reader;
|
|||
|
||||
namespace API.DTOs.Downloads;
|
||||
|
||||
public class DownloadBookmarkDto
|
||||
public sealed record DownloadBookmarkDto
|
||||
{
|
||||
[Required]
|
||||
public IEnumerable<BookmarkDto> Bookmarks { get; set; } = default!;
|
||||
|
|
|
@ -1,6 +1,6 @@
|
|||
namespace API.DTOs.Email;
|
||||
|
||||
public class ConfirmationEmailDto
|
||||
public sealed record ConfirmationEmailDto
|
||||
{
|
||||
public string InvitingUser { get; init; } = default!;
|
||||
public string EmailAddress { get; init; } = default!;
|
||||
|
|
|
@ -2,7 +2,7 @@
|
|||
|
||||
namespace API.DTOs.Email;
|
||||
|
||||
public class EmailHistoryDto
|
||||
public sealed record EmailHistoryDto
|
||||
{
|
||||
public long Id { get; set; }
|
||||
public bool Sent { get; set; }
|
||||
|
|
|
@ -1,6 +1,6 @@
|
|||
namespace API.DTOs.Email;
|
||||
|
||||
public class EmailMigrationDto
|
||||
public sealed record EmailMigrationDto
|
||||
{
|
||||
public string EmailAddress { get; init; } = default!;
|
||||
public string Username { get; init; } = default!;
|
||||
|
|
|
@ -3,7 +3,7 @@
|
|||
/// <summary>
|
||||
/// Represents if Test Email Service URL was successful or not and if any error occured
|
||||
/// </summary>
|
||||
public class EmailTestResultDto
|
||||
public sealed record EmailTestResultDto
|
||||
{
|
||||
public bool Successful { get; set; }
|
||||
public string ErrorMessage { get; set; } = default!;
|
||||
|
|
|
@ -1,6 +1,6 @@
|
|||
namespace API.DTOs.Email;
|
||||
|
||||
public class PasswordResetEmailDto
|
||||
public sealed record PasswordResetEmailDto
|
||||
{
|
||||
public string EmailAddress { get; init; } = default!;
|
||||
public string ServerConfirmationLink { get; init; } = default!;
|
||||
|
|
|
@ -2,7 +2,7 @@
|
|||
|
||||
namespace API.DTOs.Email;
|
||||
|
||||
public class SendToDto
|
||||
public sealed record SendToDto
|
||||
{
|
||||
public string DestinationEmail { get; set; } = default!;
|
||||
public IEnumerable<string> FilePaths { get; set; } = default!;
|
||||
|
|
|
@ -1,6 +1,6 @@
|
|||
namespace API.DTOs.Email;
|
||||
|
||||
public class TestEmailDto
|
||||
public sealed record TestEmailDto
|
||||
{
|
||||
public string Url { get; set; } = default!;
|
||||
}
|
||||
|
|
|
@ -5,7 +5,7 @@ using API.Entities.Enums;
|
|||
namespace API.DTOs.Filtering;
|
||||
#nullable enable
|
||||
|
||||
public class FilterDto
|
||||
public sealed record FilterDto
|
||||
{
|
||||
/// <summary>
|
||||
/// The type of Formats you want to be returned. An empty list will return all formats back
|
||||
|
|
|
@ -1,6 +1,6 @@
|
|||
namespace API.DTOs.Filtering;
|
||||
|
||||
public class LanguageDto
|
||||
public sealed record LanguageDto
|
||||
{
|
||||
public required string IsoCode { get; set; }
|
||||
public required string Title { get; set; }
|
||||
|
|
|
@ -4,7 +4,7 @@
|
|||
/// <summary>
|
||||
/// Represents a range between two int/float/double
|
||||
/// </summary>
|
||||
public class Range<T>
|
||||
public sealed record Range<T>
|
||||
{
|
||||
public T? Min { get; init; }
|
||||
public T? Max { get; init; }
|
||||
|
|
|
@ -3,7 +3,7 @@
|
|||
/// <summary>
|
||||
/// Represents the Reading Status. This is a flag and allows multiple statues
|
||||
/// </summary>
|
||||
public class ReadStatus
|
||||
public sealed record ReadStatus
|
||||
{
|
||||
public bool NotRead { get; set; } = true;
|
||||
public bool InProgress { get; set; } = true;
|
||||
|
|
|
@ -3,7 +3,7 @@
|
|||
/// <summary>
|
||||
/// Sorting Options for a query
|
||||
/// </summary>
|
||||
public class SortOptions
|
||||
public sealed record SortOptions
|
||||
{
|
||||
public SortField SortField { get; set; }
|
||||
public bool IsAscending { get; set; } = true;
|
||||
|
|
|
@ -3,7 +3,7 @@
|
|||
/// <summary>
|
||||
/// For requesting an encoded filter to be decoded
|
||||
/// </summary>
|
||||
public class DecodeFilterDto
|
||||
public sealed record DecodeFilterDto
|
||||
{
|
||||
public string EncodedFilter { get; set; }
|
||||
}
|
||||
|
|
|
@ -1,6 +1,6 @@
|
|||
namespace API.DTOs.Filtering.v2;
|
||||
|
||||
public class FilterStatementDto
|
||||
public sealed record FilterStatementDto
|
||||
{
|
||||
public FilterComparison Comparison { get; set; }
|
||||
public FilterField Field { get; set; }
|
||||
|
|
|
@ -6,7 +6,7 @@ namespace API.DTOs.Filtering.v2;
|
|||
/// <summary>
|
||||
/// Metadata filtering for v2 API only
|
||||
/// </summary>
|
||||
public class FilterV2Dto
|
||||
public sealed record FilterV2Dto
|
||||
{
|
||||
/// <summary>
|
||||
/// Not used in the UI.
|
||||
|
|
|
@ -2,7 +2,7 @@
|
|||
|
||||
namespace API.DTOs.Jobs;
|
||||
|
||||
public class JobDto
|
||||
public sealed record JobDto
|
||||
{
|
||||
/// <summary>
|
||||
/// Job Id
|
||||
|
|
|
@ -3,7 +3,7 @@
|
|||
/// <summary>
|
||||
/// Represents an individual button in a Jump Bar
|
||||
/// </summary>
|
||||
public class JumpKeyDto
|
||||
public sealed record JumpKeyDto
|
||||
{
|
||||
/// <summary>
|
||||
/// Number of items in this Key
|
||||
|
|
|
@ -1,6 +1,6 @@
|
|||
namespace API.DTOs;
|
||||
|
||||
public class KavitaLocale
|
||||
public sealed record KavitaLocale
|
||||
{
|
||||
public string FileName { get; set; } // Key
|
||||
public string RenderName { get; set; }
|
||||
|
|
|
@ -1,6 +1,6 @@
|
|||
namespace API.DTOs.KavitaPlus.Account;
|
||||
|
||||
public class AniListUpdateDto
|
||||
public sealed record AniListUpdateDto
|
||||
{
|
||||
public string Token { get; set; }
|
||||
}
|
||||
|
|
|
@ -5,7 +5,7 @@ namespace API.DTOs.KavitaPlus.Account;
|
|||
/// <summary>
|
||||
/// Represents information around a user's tokens and their status
|
||||
/// </summary>
|
||||
public class UserTokenInfo
|
||||
public sealed record UserTokenInfo
|
||||
{
|
||||
public int UserId { get; set; }
|
||||
public string Username { get; set; }
|
||||
|
|
|
@ -6,7 +6,7 @@ namespace API.DTOs.KavitaPlus.ExternalMetadata;
|
|||
/// <summary>
|
||||
/// Used for matching and fetching metadata on a series
|
||||
/// </summary>
|
||||
internal class ExternalMetadataIdsDto
|
||||
internal sealed record ExternalMetadataIdsDto
|
||||
{
|
||||
public long? MalId { get; set; }
|
||||
public int? AniListId { get; set; }
|
||||
|
|
|
@ -4,7 +4,7 @@ using API.DTOs.Scrobbling;
|
|||
namespace API.DTOs.KavitaPlus.ExternalMetadata;
|
||||
#nullable enable
|
||||
|
||||
internal class MatchSeriesRequestDto
|
||||
internal sealed record MatchSeriesRequestDto
|
||||
{
|
||||
public string SeriesName { get; set; }
|
||||
public ICollection<string> AlternativeNames { get; set; }
|
||||
|
|
|
@ -1,11 +1,12 @@
|
|||
using System.Collections.Generic;
|
||||
using API.DTOs.KavitaPlus.Metadata;
|
||||
using API.DTOs.Recommendation;
|
||||
using API.DTOs.Scrobbling;
|
||||
using API.DTOs.SeriesDetail;
|
||||
|
||||
namespace API.DTOs.KavitaPlus.ExternalMetadata;
|
||||
|
||||
internal class SeriesDetailPlusApiDto
|
||||
internal sealed record SeriesDetailPlusApiDto
|
||||
{
|
||||
public IEnumerable<MediaRecommendationDto> Recommendations { get; set; }
|
||||
public IEnumerable<UserReviewDto> Reviews { get; set; }
|
||||
|
|
|
@ -1,7 +1,7 @@
|
|||
namespace API.DTOs.KavitaPlus.License;
|
||||
#nullable enable
|
||||
|
||||
public class EncryptLicenseDto
|
||||
public sealed record EncryptLicenseDto
|
||||
{
|
||||
public required string License { get; set; }
|
||||
public required string InstallId { get; set; }
|
||||
|
|
|
@ -2,7 +2,7 @@
|
|||
|
||||
namespace API.DTOs.KavitaPlus.License;
|
||||
|
||||
public class LicenseInfoDto
|
||||
public sealed record LicenseInfoDto
|
||||
{
|
||||
/// <summary>
|
||||
/// If cancelled, will represent cancellation date. If not, will represent repayment date
|
||||
|
|
|
@ -1,6 +1,6 @@
|
|||
namespace API.DTOs.KavitaPlus.License;
|
||||
|
||||
public class LicenseValidDto
|
||||
public sealed record LicenseValidDto
|
||||
{
|
||||
public required string License { get; set; }
|
||||
public required string InstallId { get; set; }
|
||||
|
|
|
@ -1,6 +1,6 @@
|
|||
namespace API.DTOs.KavitaPlus.License;
|
||||
|
||||
public class ResetLicenseDto
|
||||
public sealed record ResetLicenseDto
|
||||
{
|
||||
public required string License { get; set; }
|
||||
public required string InstallId { get; set; }
|
||||
|
|
|
@ -1,7 +1,7 @@
|
|||
namespace API.DTOs.KavitaPlus.License;
|
||||
#nullable enable
|
||||
|
||||
public class UpdateLicenseDto
|
||||
public sealed record UpdateLicenseDto
|
||||
{
|
||||
/// <summary>
|
||||
/// License Key received from Kavita+
|
||||
|
|
|
@ -12,7 +12,7 @@ public enum MatchStateOption
|
|||
DontMatch = 4
|
||||
}
|
||||
|
||||
public class ManageMatchFilterDto
|
||||
public sealed record ManageMatchFilterDto
|
||||
{
|
||||
public MatchStateOption MatchStateOption { get; set; } = MatchStateOption.All;
|
||||
public string SearchTerm { get; set; } = string.Empty;
|
||||
|
|
|
@ -2,7 +2,7 @@
|
|||
|
||||
namespace API.DTOs.KavitaPlus.Manage;
|
||||
|
||||
public class ManageMatchSeriesDto
|
||||
public sealed record ManageMatchSeriesDto
|
||||
{
|
||||
public SeriesDto Series { get; set; }
|
||||
public bool IsMatched { get; set; }
|
||||
|
|
|
@ -7,7 +7,7 @@ namespace API.DTOs.KavitaPlus.Metadata;
|
|||
/// <summary>
|
||||
/// Information about an individual issue/chapter/book from Kavita+
|
||||
/// </summary>
|
||||
public class ExternalChapterDto
|
||||
public sealed record ExternalChapterDto
|
||||
{
|
||||
public string Title { get; set; }
|
||||
|
||||
|
|
Some files were not shown because too many files have changed in this diff Show more
Loading…
Add table
Add a link
Reference in a new issue