Refactor Display preference manager (#14056)

This commit is contained in:
JPVenson 2025-09-05 22:39:15 +02:00 committed by GitHub
parent 4849486fa0
commit 20f7ddbf8f
No known key found for this signature in database
GPG Key ID: B5690EEEBB952194
4 changed files with 102 additions and 91 deletions

View File

@ -96,9 +96,6 @@ public class DisplayPreferencesController : BaseJellyfinApiController
dto.CustomPrefs.TryAdd(key, value); dto.CustomPrefs.TryAdd(key, value);
} }
// This will essentially be a noop if no changes have been made, but new prefs must be saved at least.
_displayPreferencesManager.SaveChanges();
return dto; return dto;
} }
@ -210,8 +207,8 @@ public class DisplayPreferencesController : BaseJellyfinApiController
// Set all remaining custom preferences. // Set all remaining custom preferences.
_displayPreferencesManager.SetCustomItemDisplayPreferences(userId.Value, itemId, existingDisplayPreferences.Client, displayPreferences.CustomPrefs); _displayPreferencesManager.SetCustomItemDisplayPreferences(userId.Value, itemId, existingDisplayPreferences.Client, displayPreferences.CustomPrefs);
_displayPreferencesManager.SaveChanges(); _displayPreferencesManager.UpdateItemDisplayPreferences(itemPrefs);
_displayPreferencesManager.UpdateDisplayPreferences(existingDisplayPreferences);
return NoContent(); return NoContent();
} }
} }

View File

@ -1,23 +1,19 @@
#pragma warning disable CA1307
#pragma warning disable CA1309
using System; using System;
using System.Collections.Generic; using System.Collections.Generic;
using System.Linq; using System.Linq;
using System.Threading.Tasks;
using Jellyfin.Database.Implementations; using Jellyfin.Database.Implementations;
using Jellyfin.Database.Implementations.Entities; using Jellyfin.Database.Implementations.Entities;
using MediaBrowser.Controller; using MediaBrowser.Controller;
using Microsoft.EntityFrameworkCore; using Microsoft.EntityFrameworkCore;
namespace Jellyfin.Server.Implementations.Users namespace Jellyfin.Server.Implementations.Users;
/// <summary>
/// Manages the storage and retrieval of display preferences through Entity Framework.
/// </summary>
public sealed class DisplayPreferencesManager : IDisplayPreferencesManager
{ {
/// <summary> private readonly IDbContextFactory<JellyfinDbContext> _dbContextFactory;
/// Manages the storage and retrieval of display preferences through Entity Framework.
/// </summary>
public sealed class DisplayPreferencesManager : IDisplayPreferencesManager, IAsyncDisposable
{
private readonly JellyfinDbContext _dbContext;
/// <summary> /// <summary>
/// Initializes a new instance of the <see cref="DisplayPreferencesManager"/> class. /// Initializes a new instance of the <see cref="DisplayPreferencesManager"/> class.
@ -25,21 +21,23 @@ namespace Jellyfin.Server.Implementations.Users
/// <param name="dbContextFactory">The database context factory.</param> /// <param name="dbContextFactory">The database context factory.</param>
public DisplayPreferencesManager(IDbContextFactory<JellyfinDbContext> dbContextFactory) public DisplayPreferencesManager(IDbContextFactory<JellyfinDbContext> dbContextFactory)
{ {
_dbContext = dbContextFactory.CreateDbContext(); _dbContextFactory = dbContextFactory;
} }
/// <inheritdoc /> /// <inheritdoc />
public DisplayPreferences GetDisplayPreferences(Guid userId, Guid itemId, string client) public DisplayPreferences GetDisplayPreferences(Guid userId, Guid itemId, string client)
{ {
var prefs = _dbContext.DisplayPreferences using var dbContext = _dbContextFactory.CreateDbContext();
var prefs = dbContext.DisplayPreferences
.Include(pref => pref.HomeSections) .Include(pref => pref.HomeSections)
.FirstOrDefault(pref => .FirstOrDefault(pref =>
pref.UserId.Equals(userId) && string.Equals(pref.Client, client) && pref.ItemId.Equals(itemId)); pref.UserId.Equals(userId) && pref.Client == client && pref.ItemId.Equals(itemId));
if (prefs is null) if (prefs is null)
{ {
prefs = new DisplayPreferences(userId, itemId, client); prefs = new DisplayPreferences(userId, itemId, client);
_dbContext.DisplayPreferences.Add(prefs); dbContext.DisplayPreferences.Add(prefs);
dbContext.SaveChanges();
} }
return prefs; return prefs;
@ -48,13 +46,15 @@ namespace Jellyfin.Server.Implementations.Users
/// <inheritdoc /> /// <inheritdoc />
public ItemDisplayPreferences GetItemDisplayPreferences(Guid userId, Guid itemId, string client) public ItemDisplayPreferences GetItemDisplayPreferences(Guid userId, Guid itemId, string client)
{ {
var prefs = _dbContext.ItemDisplayPreferences using var dbContext = _dbContextFactory.CreateDbContext();
.FirstOrDefault(pref => pref.UserId.Equals(userId) && pref.ItemId.Equals(itemId) && string.Equals(pref.Client, client)); var prefs = dbContext.ItemDisplayPreferences
.FirstOrDefault(pref => pref.UserId.Equals(userId) && pref.ItemId.Equals(itemId) && pref.Client == client);
if (prefs is null) if (prefs is null)
{ {
prefs = new ItemDisplayPreferences(userId, Guid.Empty, client); prefs = new ItemDisplayPreferences(userId, Guid.Empty, client);
_dbContext.ItemDisplayPreferences.Add(prefs); dbContext.ItemDisplayPreferences.Add(prefs);
dbContext.SaveChanges();
} }
return prefs; return prefs;
@ -63,47 +63,54 @@ namespace Jellyfin.Server.Implementations.Users
/// <inheritdoc /> /// <inheritdoc />
public IList<ItemDisplayPreferences> ListItemDisplayPreferences(Guid userId, string client) public IList<ItemDisplayPreferences> ListItemDisplayPreferences(Guid userId, string client)
{ {
return _dbContext.ItemDisplayPreferences using var dbContext = _dbContextFactory.CreateDbContext();
.Where(prefs => prefs.UserId.Equals(userId) && !prefs.ItemId.Equals(default) && string.Equals(prefs.Client, client)) return dbContext.ItemDisplayPreferences
.Where(prefs => prefs.UserId.Equals(userId) && !prefs.ItemId.Equals(default) && prefs.Client == client)
.ToList(); .ToList();
} }
/// <inheritdoc /> /// <inheritdoc />
public Dictionary<string, string?> ListCustomItemDisplayPreferences(Guid userId, Guid itemId, string client) public Dictionary<string, string?> ListCustomItemDisplayPreferences(Guid userId, Guid itemId, string client)
{ {
return _dbContext.CustomItemDisplayPreferences using var dbContext = _dbContextFactory.CreateDbContext();
return dbContext.CustomItemDisplayPreferences
.Where(prefs => prefs.UserId.Equals(userId) .Where(prefs => prefs.UserId.Equals(userId)
&& prefs.ItemId.Equals(itemId) && prefs.ItemId.Equals(itemId)
&& string.Equals(prefs.Client, client)) && prefs.Client == client)
.ToDictionary(prefs => prefs.Key, prefs => prefs.Value); .ToDictionary(prefs => prefs.Key, prefs => prefs.Value);
} }
/// <inheritdoc /> /// <inheritdoc />
public void SetCustomItemDisplayPreferences(Guid userId, Guid itemId, string client, Dictionary<string, string?> customPreferences) public void SetCustomItemDisplayPreferences(Guid userId, Guid itemId, string client, Dictionary<string, string?> customPreferences)
{ {
var existingPrefs = _dbContext.CustomItemDisplayPreferences using var dbContext = _dbContextFactory.CreateDbContext();
.Where(prefs => prefs.UserId.Equals(userId) dbContext.CustomItemDisplayPreferences.Where(prefs => prefs.UserId.Equals(userId)
&& prefs.ItemId.Equals(itemId) && prefs.ItemId.Equals(itemId)
&& string.Equals(prefs.Client, client)); && prefs.Client == client)
_dbContext.CustomItemDisplayPreferences.RemoveRange(existingPrefs); .ExecuteDelete();
foreach (var (key, value) in customPreferences) foreach (var (key, value) in customPreferences)
{ {
_dbContext.CustomItemDisplayPreferences dbContext.CustomItemDisplayPreferences
.Add(new CustomItemDisplayPreferences(userId, itemId, client, key, value)); .Add(new CustomItemDisplayPreferences(userId, itemId, client, key, value));
} }
dbContext.SaveChanges();
} }
/// <inheritdoc /> /// <inheritdoc/>
public void SaveChanges() public void UpdateDisplayPreferences(DisplayPreferences displayPreferences)
{ {
_dbContext.SaveChanges(); using var dbContext = _dbContextFactory.CreateDbContext();
dbContext.DisplayPreferences.Attach(displayPreferences).State = EntityState.Modified;
dbContext.SaveChanges();
} }
/// <inheritdoc /> /// <inheritdoc/>
public async ValueTask DisposeAsync() public void UpdateItemDisplayPreferences(ItemDisplayPreferences itemDisplayPreferences)
{ {
await _dbContext.DisposeAsync().ConfigureAwait(false); using var dbContext = _dbContextFactory.CreateDbContext();
} dbContext.ItemDisplayPreferences.Attach(itemDisplayPreferences).State = EntityState.Modified;
dbContext.SaveChanges();
} }
} }

View File

@ -84,7 +84,7 @@ namespace Jellyfin.Server
serviceCollection.AddSingleton<IAuthenticationProvider, DefaultAuthenticationProvider>(); serviceCollection.AddSingleton<IAuthenticationProvider, DefaultAuthenticationProvider>();
serviceCollection.AddSingleton<IAuthenticationProvider, InvalidAuthProvider>(); serviceCollection.AddSingleton<IAuthenticationProvider, InvalidAuthProvider>();
serviceCollection.AddSingleton<IPasswordResetProvider, DefaultPasswordResetProvider>(); serviceCollection.AddSingleton<IPasswordResetProvider, DefaultPasswordResetProvider>();
serviceCollection.AddScoped<IDisplayPreferencesManager, DisplayPreferencesManager>(); serviceCollection.AddSingleton<IDisplayPreferencesManager, DisplayPreferencesManager>();
serviceCollection.AddSingleton<IDeviceManager, DeviceManager>(); serviceCollection.AddSingleton<IDeviceManager, DeviceManager>();
serviceCollection.AddSingleton<ITrickplayManager, TrickplayManager>(); serviceCollection.AddSingleton<ITrickplayManager, TrickplayManager>();

View File

@ -60,8 +60,15 @@ namespace MediaBrowser.Controller
void SetCustomItemDisplayPreferences(Guid userId, Guid itemId, string client, Dictionary<string, string?> customPreferences); void SetCustomItemDisplayPreferences(Guid userId, Guid itemId, string client, Dictionary<string, string?> customPreferences);
/// <summary> /// <summary>
/// Saves changes made to the database. /// Updates or Creates the display preferences.
/// </summary> /// </summary>
void SaveChanges(); /// <param name="displayPreferences">The entity to update or create.</param>
void UpdateDisplayPreferences(DisplayPreferences displayPreferences);
/// <summary>
/// Updates or Creates the display preferences for the given item.
/// </summary>
/// <param name="itemDisplayPreferences">The entity to update or create.</param>
void UpdateItemDisplayPreferences(ItemDisplayPreferences itemDisplayPreferences);
} }
} }