Stream custom live channels using your own media
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
 
 

148 lines
6.4 KiB

using Dapper;
using ErsatzTV.Core.Domain;
using ErsatzTV.Core.Interfaces.Repositories;
using Microsoft.EntityFrameworkCore;
namespace ErsatzTV.Infrastructure.Data.Repositories;
public class EmbyCollectionRepository : IEmbyCollectionRepository
{
private readonly IDbContextFactory<TvContext> _dbContextFactory;
public EmbyCollectionRepository(IDbContextFactory<TvContext> dbContextFactory) =>
_dbContextFactory = dbContextFactory;
public async Task<List<EmbyCollection>> GetCollections()
{
await using TvContext dbContext = await _dbContextFactory.CreateDbContextAsync();
return await dbContext.EmbyCollections.ToListAsync();
}
public async Task<bool> AddCollection(EmbyCollection collection)
{
await using TvContext dbContext = await _dbContextFactory.CreateDbContextAsync();
await dbContext.AddAsync(collection);
return await dbContext.SaveChangesAsync() > 0;
}
public async Task<bool> RemoveCollection(EmbyCollection collection)
{
await using TvContext dbContext = await _dbContextFactory.CreateDbContextAsync();
dbContext.Remove(collection);
// remove all tags that reference this collection
await dbContext.Connection.ExecuteAsync(
@"DELETE FROM Tag WHERE Name = @Name AND ExternalCollectionId = @ItemId",
new { collection.Name, collection.ItemId });
return await dbContext.SaveChangesAsync() > 0;
}
public async Task<List<int>> RemoveAllTags(EmbyCollection collection)
{
var result = new List<int>();
await using TvContext dbContext = await _dbContextFactory.CreateDbContextAsync();
// movies
result.AddRange(
await dbContext.Connection.QueryAsync<int>(
@"SELECT JM.Id FROM Tag T
INNER JOIN MovieMetadata MM on T.MovieMetadataId = MM.Id
INNER JOIN EmbyMovie JM on JM.Id = MM.MovieId
WHERE T.ExternalCollectionId = @ItemId",
new { collection.ItemId }));
// shows
result.AddRange(
await dbContext.Connection.QueryAsync<int>(
@"SELECT JS.Id FROM Tag T
INNER JOIN ShowMetadata SM on T.ShowMetadataId = SM.Id
INNER JOIN EmbyShow JS on JS.Id = SM.ShowId
WHERE T.ExternalCollectionId = @ItemId",
new { collection.ItemId }));
// seasons
result.AddRange(
await dbContext.Connection.QueryAsync<int>(
@"SELECT JS.Id FROM Tag T
INNER JOIN SeasonMetadata SM on T.SeasonMetadataId = SM.Id
INNER JOIN EmbySeason JS on JS.Id = SM.SeasonId
WHERE T.ExternalCollectionId = @ItemId",
new { collection.ItemId }));
// episodes
result.AddRange(
await dbContext.Connection.QueryAsync<int>(
@"SELECT JE.Id FROM Tag T
INNER JOIN EpisodeMetadata EM on T.EpisodeMetadataId = EM.Id
INNER JOIN EmbyEpisode JE on JE.Id = EM.EpisodeId
WHERE T.ExternalCollectionId = @ItemId",
new { collection.ItemId }));
// delete all tags
await dbContext.Connection.ExecuteAsync(
@"DELETE FROM Tag WHERE Name = @Name AND ExternalCollectionId = @ItemId",
new { collection.Name, collection.ItemId });
return result;
}
public async Task<int> AddTag(MediaItem item, EmbyCollection collection)
{
await using TvContext dbContext = await _dbContextFactory.CreateDbContextAsync();
switch (item)
{
case EmbyMovie movie:
int movieId = await dbContext.Connection.ExecuteScalarAsync<int>(
@"SELECT Id FROM EmbyMovie WHERE ItemId = @ItemId",
new { movie.ItemId });
await dbContext.Connection.ExecuteAsync(
@"INSERT INTO Tag (Name, ExternalCollectionId, MovieMetadataId)
SELECT @Name, @ItemId, Id FROM
(SELECT Id FROM MovieMetadata WHERE MovieId = @MovieId) AS A",
new { collection.Name, collection.ItemId, MovieId = movieId });
return movieId;
case EmbyShow show:
int showId = await dbContext.Connection.ExecuteScalarAsync<int>(
@"SELECT Id FROM EmbyShow WHERE ItemId = @ItemId",
new { show.ItemId });
await dbContext.Connection.ExecuteAsync(
@"INSERT INTO Tag (Name, ExternalCollectionId, ShowMetadataId)
SELECT @Name, @ItemId, Id FROM
(SELECT Id FROM ShowMetadata WHERE ShowId = @ShowId) AS A",
new { collection.Name, collection.ItemId, ShowId = showId });
return showId;
case EmbySeason season:
int seasonId = await dbContext.Connection.ExecuteScalarAsync<int>(
@"SELECT Id FROM EmbySeason WHERE ItemId = @ItemId",
new { season.ItemId });
await dbContext.Connection.ExecuteAsync(
@"INSERT INTO Tag (Name, ExternalCollectionId, SeasonMetadataId)
SELECT @Name, @ItemId, Id FROM
(SELECT Id FROM SeasonMetadata WHERE SeasonId = @SeasonId) AS A",
new { collection.Name, collection.ItemId, SeasonId = seasonId });
return seasonId;
case EmbyEpisode episode:
int episodeId = await dbContext.Connection.ExecuteScalarAsync<int>(
@"SELECT Id FROM EmbyEpisode WHERE ItemId = @ItemId",
new { episode.ItemId });
await dbContext.Connection.ExecuteAsync(
@"INSERT INTO Tag (Name, ExternalCollectionId, EpisodeMetadataId)
SELECT @Name, @ItemId, Id FROM
(SELECT Id FROM EpisodeMetadata WHERE EpisodeId = @EpisodeId) AS A",
new { collection.Name, collection.ItemId, EpisodeId = episodeId });
return episodeId;
default:
return 0;
}
}
public async Task<bool> SetEtag(EmbyCollection collection)
{
await using TvContext dbContext = await _dbContextFactory.CreateDbContextAsync();
return await dbContext.Connection.ExecuteAsync(
@"UPDATE EmbyCollection SET Etag = @Etag WHERE ItemId = @ItemId",
new { collection.Etag, collection.ItemId }) > 0;
}
}