Fixed voice work updater bug. Added integration tests for voice work search updates (Japanese).
All checks were successful
ci / build-test (push) Successful in 2m27s
ci / publish-image (push) Has been skipped

This commit is contained in:
2026-03-05 23:29:29 -05:00
parent 61f2e64972
commit 1d40013837
10 changed files with 171 additions and 5 deletions

View File

@@ -0,0 +1,7 @@
using JSMR.Application.Enums;
namespace JSMR.Application.Scanning.Ports;
public interface IVoiceWorkScannerRepository
{
public IVoiceWorksScanner? GetScanner(Locale locale);
}

View File

@@ -0,0 +1,8 @@
using JSMR.Application.Enums;
namespace JSMR.Application.Scanning.Ports;
public interface IVoiceWorkUpdaterRepository
{
public IVoiceWorkUpdater? GetUpdater(Locale locale);
}

View File

@@ -3,20 +3,20 @@ using JSMR.Application.Integrations.DLSite.Models;
using JSMR.Application.Integrations.Ports;
using JSMR.Application.Scanning.Contracts;
using JSMR.Application.Scanning.Ports;
using Microsoft.Extensions.DependencyInjection;
namespace JSMR.Application.Scanning;
public sealed class ScanVoiceWorksHandler(
IServiceProvider serviceProvider,
IVoiceWorkScannerRepository scannerRepository,
IVoiceWorkUpdaterRepository updaterRepository,
IDLSiteClient dlsiteClient,
ISpamCircleCache spamCircleCache,
IVoiceWorkSearchUpdater searchUpdater)
{
public async Task<ScanVoiceWorksResponse> HandleAsync(ScanVoiceWorksRequest request, CancellationToken cancellationToken)
{
IVoiceWorksScanner? scanner = serviceProvider.GetKeyedService<IVoiceWorksScanner>(request.Locale);
IVoiceWorkUpdater? updater = serviceProvider.GetKeyedService<IVoiceWorkUpdater>(request.Locale);
IVoiceWorksScanner? scanner = scannerRepository.GetScanner(request.Locale);
IVoiceWorkUpdater? updater = updaterRepository.GetUpdater(request.Locale);
if (scanner is null || updater is null)
return new();

View File

@@ -47,9 +47,12 @@ public static class InfrastructureServiceCollectionExtensions
services.AddKeyedScoped<IVoiceWorksScanner, JapaneseVoiceWorksScanner>(Locale.Japanese);
services.AddKeyedScoped<IVoiceWorksScanner, EnglishVoiceWorksScanner>(Locale.English);
services.AddSingleton<IVoiceWorkScannerRepository, VoiceWorkScannerRepository>();
services.AddKeyedScoped<IVoiceWorkUpdater, VoiceWorkUpdater>(Locale.Japanese);
services.AddKeyedScoped<IVoiceWorkUpdater, EnglishVoiceWorkUpdater>(Locale.English);
services.AddSingleton<IVoiceWorkUpdaterRepository, VoiceWorkUpdaterRepository>();
services.AddScoped<IVoiceWorkSearchUpdater, VoiceWorkSearchUpdater>();
//services.AddKeyedScoped<ISupportedLanguage, JapaneseLanguage>(Locale.Japanese);

View File

@@ -35,6 +35,18 @@ public class VoiceWorkUpdater(AppDbContext dbContext, ITimeProvider timeProvider
await dbContext.SaveChangesAsync(cancellationToken);
foreach (VoiceWorkIngest ingest in ingests)
{
VoiceWorkUpsertResult result = upsertContext.Results[ingest.ProductId];
if (result.Status is VoiceWorkUpsertStatus.Skipped)
continue;
VoiceWork voiceWork = upsertContext.VoiceWorks[ingest.ProductId];
result.VoiceWorkId = voiceWork.VoiceWorkId;
}
return [.. upsertContext.Results.Select(x => x.Value)];
}

View File

@@ -0,0 +1,13 @@
using JSMR.Application.Enums;
using JSMR.Application.Scanning.Ports;
using Microsoft.Extensions.DependencyInjection;
namespace JSMR.Infrastructure.Ingestion;
public class VoiceWorkUpdaterRepository(IServiceProvider serviceProvider) : IVoiceWorkUpdaterRepository
{
public IVoiceWorkUpdater? GetUpdater(Locale locale)
{
return serviceProvider.GetKeyedService<IVoiceWorkUpdater>(locale);
}
}

View File

@@ -0,0 +1,13 @@
using JSMR.Application.Enums;
using JSMR.Application.Scanning.Ports;
using Microsoft.Extensions.DependencyInjection;
namespace JSMR.Infrastructure.Scanning;
public class VoiceWorkScannerRepository(IServiceProvider serviceProvider) : IVoiceWorkScannerRepository
{
public IVoiceWorksScanner? GetScanner(Locale locale)
{
return serviceProvider.GetKeyedService<IVoiceWorksScanner>(locale);
}
}

View File

@@ -0,0 +1,32 @@
using JSMR.Application.Scanning.Contracts;
using JSMR.Domain.Enums;
using JSMR.Domain.ValueObjects;
using JSMR.Tests.Ingestion.Search;
namespace JSMR.Tests.Ingestion;
internal class IngestTestFactory
{
public static VoiceWorkIngest Create(SearchRelatedParameters searchRelatedParameters)
{
return new()
{
MakerId = searchRelatedParameters.MakerId,
MakerName = searchRelatedParameters.MakerName,
ProductId = searchRelatedParameters.ProductId,
Title = searchRelatedParameters.Title,
Description = searchRelatedParameters.Description,
Tags = searchRelatedParameters.Tags,
Creators = searchRelatedParameters.Creators,
WishlistCount = 100,
Downloads = 0,
HasTrial = false,
HasDLPlay = false,
AgeRating = AgeRating.AllAges,
HasImage = false,
SupportedLanguages = [SupportedLanguage.Japanese],
SalesDate = null,
ExpectedDate = new DateOnly(2025, 1, 1)
};
}
}

View File

@@ -1,5 +1,7 @@
using JSMR.Application.Scanning.Contracts;
using JSMR.Application.Enums;
using JSMR.Application.Scanning.Contracts;
using JSMR.Application.Scanning.Ports;
using JSMR.Domain.Entities;
using JSMR.Infrastructure.Common.Time;
using JSMR.Infrastructure.Data;
using JSMR.Infrastructure.Ingestion;
@@ -25,6 +27,13 @@ public abstract class JapaneseIngestionTestsBase(MariaDbContainerFixture contain
return await updater.UpsertAsync(ingests, TestContext.Current.CancellationToken);
}
protected static async Task UpdateSearchAsync(AppDbContext dbContext, int[] voiceWorkIds)
{
VoiceWorkSearchUpdater updater = new(dbContext);
await updater.UpdateAsync(voiceWorkIds, TestContext.Current.CancellationToken);
}
protected static DateTime TokyoLocalToUtc(int year, int month, int day, int hour, int minute, int second)
{
var tokyo = TimeZoneInfo.FindSystemTimeZoneById(

View File

@@ -0,0 +1,69 @@
using JSMR.Application.Scanning.Contracts;
using JSMR.Domain.Entities;
using JSMR.Infrastructure.Data;
using JSMR.Tests.Fixtures;
using JSMR.Tests.Ingestion.Japanese;
using Microsoft.EntityFrameworkCore;
using Shouldly;
namespace JSMR.Tests.Ingestion.Search;
internal record SearchRelatedParameters(
string ProductId,
string MakerId,
string Title,
string Description,
string MakerName,
string[] Tags,
string[] Creators
);
public class Insert_Into_Search_Tests(MariaDbContainerFixture container) : JapaneseIngestionTestsBase(container)
{
[Fact]
public async Task Insert_Into_Search_And_Update()
{
await using AppDbContext dbContext = await GetAppDbContextAsync();
// PART 1 - Insert
SearchRelatedParameters parameters = new(
ProductId: "RJ1",
MakerId: "RG1",
Title: "Title",
Description: "Description",
MakerName: "Maker",
Tags: ["Tag 1", "Tag 2"],
Creators: ["Creator 1"]
);
VoiceWorkIngest ingest = IngestTestFactory.Create(parameters);
await UpsertAndVerify(dbContext, ingest, "RJ1 RG1 Title Description Maker Tag 1 Tag 2 Creator 1");
// PART 2 - Update
SearchRelatedParameters updateParameters = parameters with
{
Title = "Updated Title",
Tags = ["Tag 1", "Tag 2", "Tag 3"]
};
VoiceWorkIngest updatedIngest = IngestTestFactory.Create(updateParameters);
await UpsertAndVerify(dbContext, updatedIngest, "RJ1 RG1 Updated Title Description Maker Tag 1 Tag 2 Tag 3 Creator 1");
}
private static async Task UpsertAndVerify(AppDbContext dbContext, VoiceWorkIngest ingest, string searchText)
{
await UpsertAsync(dbContext, TokyoLocalToUtc(2025, 01, 15, 00, 00, 00), [ingest]);
VoiceWork? voiceWork = await dbContext.VoiceWorks.SingleAsync(v => v.ProductId == ingest.ProductId, TestContext.Current.CancellationToken);
voiceWork.ShouldNotBeNull();
await UpdateSearchAsync(dbContext, [voiceWork.VoiceWorkId]);
VoiceWorkSearch voiceWorkSearch = await dbContext.VoiceWorkSearches.SingleAsync(v => v.VoiceWorkId == voiceWork.VoiceWorkId, TestContext.Current.CancellationToken);
voiceWorkSearch.ShouldNotBeNull();
voiceWorkSearch.SearchText.ShouldBe(searchText);
}
}