mirror of
https://github.com/zoriya/Kyoo.git
synced 2025-11-02 18:47:11 -05:00
203 lines
6.1 KiB
C#
203 lines
6.1 KiB
C#
using System;
|
|
using System.Collections.Generic;
|
|
using System.Linq;
|
|
using System.Linq.Expressions;
|
|
using System.Threading.Tasks;
|
|
using Kyoo.Models;
|
|
using Microsoft.EntityFrameworkCore;
|
|
using Microsoft.Extensions.DependencyInjection;
|
|
|
|
namespace Kyoo.Controllers
|
|
{
|
|
public class ShowRepository : LocalRepository<Show>, IShowRepository
|
|
{
|
|
private bool _disposed;
|
|
private readonly DatabaseContext _database;
|
|
private readonly IStudioRepository _studios;
|
|
private readonly IPeopleRepository _people;
|
|
private readonly IGenreRepository _genres;
|
|
private readonly IProviderRepository _providers;
|
|
private readonly Lazy<ISeasonRepository> _seasons;
|
|
private readonly Lazy<IEpisodeRepository> _episodes;
|
|
protected override Expression<Func<Show, object>> DefaultSort => x => x.Title;
|
|
|
|
public ShowRepository(DatabaseContext database,
|
|
IStudioRepository studios,
|
|
IPeopleRepository people,
|
|
IGenreRepository genres,
|
|
IProviderRepository providers,
|
|
IServiceProvider services)
|
|
: base(database)
|
|
{
|
|
_database = database;
|
|
_studios = studios;
|
|
_people = people;
|
|
_genres = genres;
|
|
_providers = providers;
|
|
_seasons = new Lazy<ISeasonRepository>(services.GetRequiredService<ISeasonRepository>);
|
|
_episodes = new Lazy<IEpisodeRepository>(services.GetRequiredService<IEpisodeRepository>);
|
|
}
|
|
|
|
public override void Dispose()
|
|
{
|
|
if (_disposed)
|
|
return;
|
|
_disposed = true;
|
|
_database.Dispose();
|
|
_studios.Dispose();
|
|
_people.Dispose();
|
|
_genres.Dispose();
|
|
_providers.Dispose();
|
|
if (_seasons.IsValueCreated)
|
|
_seasons.Value.Dispose();
|
|
if (_episodes.IsValueCreated)
|
|
_episodes.Value.Dispose();
|
|
GC.SuppressFinalize(this);
|
|
}
|
|
|
|
public override async ValueTask DisposeAsync()
|
|
{
|
|
if (_disposed)
|
|
return;
|
|
_disposed = true;
|
|
await _database.DisposeAsync();
|
|
await _studios.DisposeAsync();
|
|
await _people.DisposeAsync();
|
|
await _genres.DisposeAsync();
|
|
await _providers.DisposeAsync();
|
|
if (_seasons.IsValueCreated)
|
|
await _seasons.Value.DisposeAsync();
|
|
if (_episodes.IsValueCreated)
|
|
await _episodes.Value.DisposeAsync();
|
|
}
|
|
|
|
public override async Task<ICollection<Show>> Search(string query)
|
|
{
|
|
query = $"%{query}%";
|
|
return await _database.Shows
|
|
.Where(x => EF.Functions.ILike(x.Title, query)
|
|
|| EF.Functions.ILike(x.Slug, query)
|
|
/*|| x.Aliases.Any(y => EF.Functions.ILike(y, query))*/) // NOT TRANSLATABLE.
|
|
.OrderBy(DefaultSort)
|
|
.Take(20)
|
|
.ToListAsync();
|
|
}
|
|
|
|
public override async Task<Show> Create(Show obj)
|
|
{
|
|
await base.Create(obj);
|
|
_database.Entry(obj).State = EntityState.Added;
|
|
obj.GenreLinks.ForEach(x => _database.Entry(x).State = EntityState.Added);
|
|
obj.People.ForEach(x => _database.Entry(x).State = EntityState.Added);
|
|
obj.ExternalIDs.ForEach(x => _database.Entry(x).State = EntityState.Added);
|
|
await _database.SaveChangesAsync($"Trying to insert a duplicated show (slug {obj.Slug} already exists).");
|
|
return obj;
|
|
}
|
|
|
|
protected override async Task Validate(Show resource)
|
|
{
|
|
await base.Validate(resource);
|
|
if (resource.Studio != null)
|
|
resource.Studio = await _studios.CreateIfNotExists(resource.Studio, true);
|
|
resource.Genres = await resource.Genres
|
|
.SelectAsync(x => _genres.CreateIfNotExists(x, true))
|
|
.ToListAsync();
|
|
resource.GenreLinks = resource.Genres?
|
|
.Select(x => Link.UCreate(resource, x))
|
|
.ToList();
|
|
await resource.ExternalIDs.ForEachAsync(async id =>
|
|
{
|
|
id.Provider = await _providers.CreateIfNotExists(id.Provider, true);
|
|
id.ProviderID = id.Provider.ID;
|
|
_database.Entry(id.Provider).State = EntityState.Detached;
|
|
});
|
|
await resource.People.ForEachAsync(async role =>
|
|
{
|
|
role.People = await _people.CreateIfNotExists(role.People, true);
|
|
role.PeopleID = role.People.ID;
|
|
_database.Entry(role.People).State = EntityState.Detached;
|
|
});
|
|
}
|
|
|
|
protected override async Task EditRelations(Show resource, Show changed, bool resetOld)
|
|
{
|
|
await Validate(changed);
|
|
|
|
if (changed.Aliases != null || resetOld)
|
|
resource.Aliases = changed.Aliases;
|
|
|
|
if (changed.Genres != null || resetOld)
|
|
{
|
|
await Database.Entry(resource).Collection(x => x.GenreLinks).LoadAsync();
|
|
resource.GenreLinks = changed.Genres?.Select(x => Link.UCreate(resource, x)).ToList();
|
|
}
|
|
|
|
if (changed.People != null || resetOld)
|
|
{
|
|
await Database.Entry(resource).Collection(x => x.People).LoadAsync();
|
|
resource.People = changed.People;
|
|
}
|
|
|
|
if (changed.ExternalIDs != null || resetOld)
|
|
{
|
|
await Database.Entry(resource).Collection(x => x.ExternalIDs).LoadAsync();
|
|
resource.ExternalIDs = changed.ExternalIDs;
|
|
}
|
|
}
|
|
|
|
public async Task AddShowLink(int showID, int? libraryID, int? collectionID)
|
|
{
|
|
if (collectionID != null)
|
|
{
|
|
await _database.Links<Collection, Show>()
|
|
.AddAsync(new Link<Collection, Show>(collectionID.Value, showID));
|
|
await _database.SaveIfNoDuplicates();
|
|
|
|
if (libraryID != null)
|
|
{
|
|
await _database.Links<Library, Collection>()
|
|
.AddAsync(new Link<Library, Collection>(libraryID.Value, collectionID.Value));
|
|
await _database.SaveIfNoDuplicates();
|
|
}
|
|
}
|
|
if (libraryID != null)
|
|
{
|
|
await _database.Links<Library, Show>()
|
|
.AddAsync(new Link<Library, Show>(libraryID.Value, showID));
|
|
await _database.SaveIfNoDuplicates();
|
|
}
|
|
}
|
|
|
|
public Task<string> GetSlug(int showID)
|
|
{
|
|
return _database.Shows.Where(x => x.ID == showID)
|
|
.Select(x => x.Slug)
|
|
.FirstOrDefaultAsync();
|
|
}
|
|
|
|
public override async Task Delete(Show obj)
|
|
{
|
|
if (obj == null)
|
|
throw new ArgumentNullException(nameof(obj));
|
|
|
|
_database.Entry(obj).State = EntityState.Deleted;
|
|
|
|
|
|
if (obj.People != null)
|
|
foreach (PeopleRole entry in obj.People)
|
|
_database.Entry(entry).State = EntityState.Deleted;
|
|
|
|
if (obj.ExternalIDs != null)
|
|
foreach (MetadataID entry in obj.ExternalIDs)
|
|
_database.Entry(entry).State = EntityState.Deleted;
|
|
|
|
await _database.SaveChangesAsync();
|
|
|
|
if (obj.Seasons != null)
|
|
await _seasons.Value.DeleteRange(obj.Seasons);
|
|
|
|
if (obj.Episodes != null)
|
|
await _episodes.Value.DeleteRange(obj.Episodes);
|
|
}
|
|
}
|
|
} |