Added configurations for entity builders. Also added VoiceWorkLocaalization entity.

This commit is contained in:
2025-08-27 00:16:10 -04:00
parent d2201d6f9b
commit 22b8513e34
16 changed files with 299 additions and 0 deletions

View File

@@ -0,0 +1,23 @@
using JSMR.Domain.Entities;
using Microsoft.EntityFrameworkCore;
using Microsoft.EntityFrameworkCore.Metadata.Builders;
namespace JSMR.Infrastructure.Data.Configuration;
public sealed class CircleConfiguration : IEntityTypeConfiguration<Circle>
{
public void Configure(EntityTypeBuilder<Circle> builder)
{
builder.ToTable("circles");
builder.HasKey(x => x.CircleId);
builder.Property(x => x.Name).IsRequired().HasMaxLength(256);
builder.Property(x => x.MakerId).IsRequired().HasMaxLength(16);
builder.HasIndex(x => x.Name);
builder.HasIndex(x => x.MakerId).IsUnique();
builder.HasIndex(x => x.Favorite);
builder.HasIndex(x => x.Blacklisted);
builder.HasIndex(x => x.Spam);
}
}

View File

@@ -0,0 +1,20 @@
using JSMR.Domain.Entities;
using Microsoft.EntityFrameworkCore;
using Microsoft.EntityFrameworkCore.Metadata.Builders;
namespace JSMR.Infrastructure.Data.Configuration;
public sealed class CreatorConfiguration : IEntityTypeConfiguration<Creator>
{
public void Configure(EntityTypeBuilder<Creator> builder)
{
builder.ToTable("creators");
builder.HasKey(x => x.CreatorId);
builder.Property(x => x.Name).IsRequired().HasMaxLength(128);
builder.HasIndex(x => x.Name);
builder.HasIndex(x => x.Favorite);
builder.HasIndex(x => x.Blacklisted);
}
}

View File

@@ -0,0 +1,24 @@
using JSMR.Domain.Entities;
using Microsoft.EntityFrameworkCore;
using Microsoft.EntityFrameworkCore.Metadata.Builders;
namespace JSMR.Infrastructure.Data.Configuration;
public sealed class EnglishTagConfiguration : IEntityTypeConfiguration<EnglishTag>
{
public void Configure(EntityTypeBuilder<EnglishTag> builder)
{
builder.ToTable("english_tags");
builder.HasKey(x => x.EnglishTagId);
builder.Property(x => x.Name).IsRequired().HasMaxLength(128);
builder.HasOne(x => x.Tag)
.WithMany()
.HasForeignKey(x => x.TagId)
.OnDelete(DeleteBehavior.Cascade);
builder.HasIndex(x => x.TagId).IsUnique();
builder.HasIndex(x => x.Name);
}
}

View File

@@ -0,0 +1,25 @@
using JSMR.Domain.Entities;
using Microsoft.EntityFrameworkCore;
using Microsoft.EntityFrameworkCore.Metadata.Builders;
namespace JSMR.Infrastructure.Data.Configuration;
public sealed class EnglishVoiceWorkConfiguration : IEntityTypeConfiguration<EnglishVoiceWork>
{
public void Configure(EntityTypeBuilder<EnglishVoiceWork> builder)
{
builder.ToTable("english_voice_works");
builder.HasKey(x => x.EnglishVoiceWorkId);
builder.Property(x => x.ProductName).HasMaxLength(256);
builder.Property(x => x.Description).HasMaxLength(512);
builder.HasOne(x => x.VoiceWork)
.WithMany(v => v.EnglishVoiceWorks)
.HasForeignKey(x => x.VoiceWorkId)
.OnDelete(DeleteBehavior.Cascade);
builder.HasIndex(x => x.VoiceWorkId).IsUnique();
builder.HasIndex(x => x.IsValid);
}
}

View File

@@ -0,0 +1,20 @@
using JSMR.Domain.Entities;
using Microsoft.EntityFrameworkCore;
using Microsoft.EntityFrameworkCore.Metadata.Builders;
namespace JSMR.Infrastructure.Data.Configuration;
public sealed class TagConfiguration : IEntityTypeConfiguration<Tag>
{
public void Configure(EntityTypeBuilder<Tag> builder)
{
builder.ToTable("tags");
builder.HasKey(x => x.TagId);
builder.Property(x => x.Name).IsRequired().HasMaxLength(128);
builder.HasIndex(x => x.Name).IsUnique();
builder.HasIndex(x => x.Favorite);
builder.HasIndex(x => x.Blacklisted);
}
}

View File

@@ -0,0 +1,51 @@
using JSMR.Domain.Entities;
using Microsoft.EntityFrameworkCore;
using Microsoft.EntityFrameworkCore.Metadata.Builders;
namespace JSMR.Infrastructure.Data.Configuration;
public sealed class VoiceWorkConfiguration : IEntityTypeConfiguration<VoiceWork>
{
public void Configure(EntityTypeBuilder<VoiceWork> builder)
{
builder.ToTable("voice_works");
builder.HasKey(x => x.VoiceWorkId);
builder.Property(x => x.ProductId)
.IsRequired()
.HasMaxLength(16);
builder.Property(x => x.OriginalProductId)
.HasMaxLength(16);
builder.Property(x => x.ProductName)
.IsRequired()
.HasMaxLength(256);
builder.Property(x => x.Description)
.HasMaxLength(512);
// Relationships
builder.HasOne(x => x.Circle)
.WithMany(c => c.VoiceWorks)
.HasForeignKey(x => x.CircleId)
.OnDelete(DeleteBehavior.Restrict);
builder.HasOne(x => x.Series)
.WithMany(s => s.VoiceWorks)
.HasForeignKey(x => x.SeriesId)
.OnDelete(DeleteBehavior.SetNull);
// Indexes for common filters/sorts
builder.HasIndex(x => x.ProductId).IsUnique();
builder.HasIndex(x => x.CircleId);
builder.HasIndex(x => x.SalesDate);
builder.HasIndex(x => x.ExpectedDate);
builder.HasIndex(x => x.Favorite);
builder.HasIndex(x => x.IsValid);
builder.HasIndex(x => x.SubtitleLanguage);
builder.HasIndex(x => x.AIGeneration);
builder.HasIndex(x => x.Downloads);
builder.HasIndex(x => x.WishlistCount);
}
}

View File

@@ -0,0 +1,28 @@
using JSMR.Domain.Entities;
using Microsoft.EntityFrameworkCore;
using Microsoft.EntityFrameworkCore.Metadata.Builders;
namespace JSMR.Infrastructure.Data.Configuration;
public sealed class VoiceWorkCreatorConfiguration : IEntityTypeConfiguration<VoiceWorkCreator>
{
public void Configure(EntityTypeBuilder<VoiceWorkCreator> builder)
{
builder.ToTable("voice_work_creators");
builder.HasKey(x => new { x.VoiceWorkId, x.CreatorId });
builder.HasOne(x => x.VoiceWork)
.WithMany(v => v.VoiceWorkCreators)
.HasForeignKey(x => x.VoiceWorkId)
.OnDelete(DeleteBehavior.Cascade);
builder.HasOne(x => x.Creator)
.WithMany()
.HasForeignKey(x => x.CreatorId)
.OnDelete(DeleteBehavior.Cascade);
builder.HasIndex(x => x.CreatorId);
builder.HasIndex(x => new { x.VoiceWorkId, x.Position });
builder.HasIndex(x => x.IsValid);
}
}

View File

@@ -0,0 +1,27 @@
using JSMR.Domain.Entities;
using Microsoft.EntityFrameworkCore;
using Microsoft.EntityFrameworkCore.Metadata.Builders;
namespace JSMR.Infrastructure.Data.Configuration;
public sealed class VoiceWorkLocalizationConfiguration : IEntityTypeConfiguration<VoiceWorkLocalization>
{
public void Configure(EntityTypeBuilder<VoiceWorkLocalization> builder)
{
builder.ToTable("voice_work_localizations");
builder.HasKey(x => x.VoiceWorkLocalizationId);
builder.Property(x => x.Language).IsRequired().HasMaxLength(10);
builder.Property(x => x.ProductName).HasMaxLength(256);
builder.Property(x => x.Description).HasMaxLength(512);
builder.HasOne(x => x.VoiceWork)
.WithMany(v => v.Localizations)
.HasForeignKey(x => x.VoiceWorkId)
.OnDelete(DeleteBehavior.Cascade);
builder.HasIndex(x => new { x.VoiceWorkId, x.Language }).IsUnique();
builder.HasIndex(x => x.VoiceWorkId);
builder.HasIndex(x => x.ProductName);
}
}

View File

@@ -0,0 +1,27 @@
using JSMR.Domain.Entities;
using Microsoft.EntityFrameworkCore;
using Microsoft.EntityFrameworkCore.Metadata.Builders;
namespace JSMR.Infrastructure.Data.Configuration;
public sealed class VoiceWorkSearchConfiguration : IEntityTypeConfiguration<VoiceWorkSearch>
{
public void Configure(EntityTypeBuilder<VoiceWorkSearch> builder)
{
builder.ToTable("voice_work_searches");
builder.HasKey(x => x.VoiceWorkId); // also the FK
builder.Property(x => x.SearchText).IsRequired(); // TEXT/LONGTEXT
builder.HasOne(x => x.VoiceWork)
.WithOne() // not navigated from VoiceWork in your model
.HasForeignKey<VoiceWorkSearch>(x => x.VoiceWorkId)
.OnDelete(DeleteBehavior.Cascade);
// MariaDB/MySQL (Pomelo) fulltext (BOOLEAN/NATURAL) — create an FT index
// Pomelo supports .HasMethod("FULLTEXT"). If your version doesn't, add it in a migration SQL.
//builder.HasIndex(x => x.SearchText)
// .HasDatabaseName("FT_SearchText")
// .HasMethod("FULLTEXT");
}
}

View File

@@ -0,0 +1,29 @@
using JSMR.Domain.Entities;
using Microsoft.EntityFrameworkCore;
using Microsoft.EntityFrameworkCore.Metadata.Builders;
namespace JSMR.Infrastructure.Data.Configuration;
public sealed class VoiceWorkTagConfiguration : IEntityTypeConfiguration<VoiceWorkTag>
{
public void Configure(EntityTypeBuilder<VoiceWorkTag> builder)
{
builder.ToTable("voice_work_tags");
builder.HasKey(x => new { x.VoiceWorkId, x.TagId });
builder.HasOne(x => x.VoiceWork)
.WithMany(v => v.VoiceWorkTags)
.HasForeignKey(x => x.VoiceWorkId)
.OnDelete(DeleteBehavior.Cascade);
builder.HasOne(x => x.Tag)
.WithMany()
.HasForeignKey(x => x.TagId)
.OnDelete(DeleteBehavior.Cascade);
// Helpful indexes
builder.HasIndex(x => x.TagId);
builder.HasIndex(x => new { x.VoiceWorkId, x.Position });
builder.HasIndex(x => x.IsValid);
}
}