jellyfin/Jellyfin.Server/Migrations/Routines/MigrateUserDb.cs

222 lines
11 KiB
C#
Raw Normal View History

2020-05-15 23:24:01 +02:00
using System;
2020-05-13 04:25:45 +02:00
using System.IO;
using Emby.Server.Implementations.Data;
using Jellyfin.Data.Entities;
using Jellyfin.Data.Enums;
using Jellyfin.Extensions.Json;
2020-05-13 04:25:45 +02:00
using Jellyfin.Server.Implementations;
2020-05-15 23:24:01 +02:00
using Jellyfin.Server.Implementations.Users;
2020-05-13 04:25:45 +02:00
using MediaBrowser.Controller;
using MediaBrowser.Controller.Entities;
2020-05-13 04:25:45 +02:00
using MediaBrowser.Model.Configuration;
2021-09-25 19:36:29 +02:00
using MediaBrowser.Model.Serialization;
2020-05-13 04:25:45 +02:00
using MediaBrowser.Model.Users;
2023-08-21 15:31:02 +02:00
using Microsoft.Data.Sqlite;
using Microsoft.EntityFrameworkCore;
2020-05-13 04:25:45 +02:00
using Microsoft.Extensions.Logging;
2020-05-15 23:24:01 +02:00
using JsonSerializer = System.Text.Json.JsonSerializer;
2020-05-13 04:25:45 +02:00
2020-05-13 04:10:35 +02:00
namespace Jellyfin.Server.Migrations.Routines
{
2020-05-15 23:24:01 +02:00
/// <summary>
/// The migration routine for migrating the user database to EF Core.
/// </summary>
2020-05-13 04:25:45 +02:00
public class MigrateUserDb : IMigrationRoutine
2020-05-13 04:10:35 +02:00
{
2020-05-15 23:24:01 +02:00
private const string DbFilename = "users.db";
2020-05-13 04:25:45 +02:00
2020-05-15 23:24:01 +02:00
private readonly ILogger<MigrateUserDb> _logger;
2020-05-13 04:25:45 +02:00
private readonly IServerApplicationPaths _paths;
2023-01-16 18:14:44 +01:00
private readonly IDbContextFactory<JellyfinDbContext> _provider;
2021-09-25 19:36:29 +02:00
private readonly IXmlSerializer _xmlSerializer;
2020-05-13 04:25:45 +02:00
2020-05-15 23:24:01 +02:00
/// <summary>
/// Initializes a new instance of the <see cref="MigrateUserDb"/> class.
/// </summary>
/// <param name="logger">The logger.</param>
/// <param name="paths">The server application paths.</param>
/// <param name="provider">The database provider.</param>
/// <param name="xmlSerializer">The xml serializer.</param>
public MigrateUserDb(
ILogger<MigrateUserDb> logger,
IServerApplicationPaths paths,
2023-01-16 18:14:44 +01:00
IDbContextFactory<JellyfinDbContext> provider,
2021-09-25 19:36:29 +02:00
IXmlSerializer xmlSerializer)
2020-05-13 04:25:45 +02:00
{
_logger = logger;
_paths = paths;
_provider = provider;
_xmlSerializer = xmlSerializer;
}
2020-05-15 23:24:01 +02:00
/// <inheritdoc/>
2020-05-13 04:25:45 +02:00
public Guid Id => Guid.Parse("5C4B82A2-F053-4009-BD05-B6FCAD82F14C");
2020-05-15 23:24:01 +02:00
/// <inheritdoc/>
public string Name => "MigrateUserDatabase";
2020-05-13 04:25:45 +02:00
/// <inheritdoc/>
public bool PerformOnNewInstall => false;
2020-05-15 23:24:01 +02:00
/// <inheritdoc/>
2020-05-13 04:25:45 +02:00
public void Perform()
{
var dataPath = _paths.DataPath;
_logger.LogInformation("Migrating the user database may take a while, do not stop Jellyfin.");
2023-08-21 15:31:02 +02:00
using (var connection = new SqliteConnection($"Filename={Path.Combine(dataPath, DbFilename)}"))
2020-05-13 04:25:45 +02:00
{
connection.Open();
var dbContext = _provider.CreateDbContext();
2020-05-13 04:25:45 +02:00
var queryResult = connection.Query("SELECT * FROM LocalUsersv2");
dbContext.RemoveRange(dbContext.Users);
dbContext.SaveChanges();
foreach (var entry in queryResult)
{
2023-08-21 15:31:02 +02:00
UserMockup? mockup = JsonSerializer.Deserialize<UserMockup>(entry.GetStream(2), JsonDefaults.Options);
2022-12-05 15:00:20 +01:00
if (mockup is null)
2020-08-25 15:33:58 +02:00
{
continue;
}
2020-05-15 23:24:01 +02:00
var userDataDir = Path.Combine(_paths.UserConfigurationDirectoryPath, mockup.Name);
var configPath = Path.Combine(userDataDir, "config.xml");
var config = File.Exists(configPath)
? (UserConfiguration?)_xmlSerializer.DeserializeFromFile(typeof(UserConfiguration), configPath) ?? new UserConfiguration()
2020-05-15 23:24:01 +02:00
: new UserConfiguration();
var policyPath = Path.Combine(userDataDir, "policy.xml");
var policy = File.Exists(policyPath)
? (UserPolicy?)_xmlSerializer.DeserializeFromFile(typeof(UserPolicy), policyPath) ?? new UserPolicy()
2020-05-15 23:24:01 +02:00
: new UserPolicy();
policy.AuthenticationProviderId = policy.AuthenticationProviderId?.Replace(
"Emby.Server.Implementations.Library",
"Jellyfin.Server.Implementations.Users",
StringComparison.Ordinal)
?? typeof(DefaultAuthenticationProvider).FullName;
policy.PasswordResetProviderId = typeof(DefaultPasswordResetProvider).FullName;
2020-05-23 22:07:42 +02:00
int? maxLoginAttempts = policy.LoginAttemptsBeforeLockout switch
{
-1 => null,
0 => 3,
_ => policy.LoginAttemptsBeforeLockout
};
2020-05-15 23:24:01 +02:00
var user = new User(mockup.Name, policy.AuthenticationProviderId!, policy.PasswordResetProviderId!)
2020-05-13 04:25:45 +02:00
{
2023-08-21 15:31:02 +02:00
Id = entry.GetGuid(1),
InternalId = entry.GetInt64(0),
2020-05-13 04:25:45 +02:00
MaxParentalAgeRating = policy.MaxParentalRating,
EnableUserPreferenceAccess = policy.EnableUserPreferenceAccess,
RemoteClientBitrateLimit = policy.RemoteClientBitrateLimit,
InvalidLoginAttemptCount = policy.InvalidLoginAttemptCount,
2020-05-23 22:07:42 +02:00
LoginAttemptsBeforeLockout = maxLoginAttempts,
2020-05-13 04:25:45 +02:00
SubtitleMode = config.SubtitleMode,
HidePlayedInLatest = config.HidePlayedInLatest,
EnableLocalPassword = config.EnableLocalPassword,
PlayDefaultAudioTrack = config.PlayDefaultAudioTrack,
DisplayCollectionsView = config.DisplayCollectionsView,
DisplayMissingEpisodes = config.DisplayMissingEpisodes,
AudioLanguagePreference = config.AudioLanguagePreference,
RememberAudioSelections = config.RememberAudioSelections,
EnableNextEpisodeAutoPlay = config.EnableNextEpisodeAutoPlay,
RememberSubtitleSelections = config.RememberSubtitleSelections,
SubtitleLanguagePreference = config.SubtitleLanguagePreference,
2020-05-15 23:24:01 +02:00
Password = mockup.Password,
LastLoginDate = mockup.LastLoginDate,
LastActivityDate = mockup.LastActivityDate
2020-05-13 04:25:45 +02:00
};
if (mockup.ImageInfos.Length > 0)
{
ItemImageInfo info = mockup.ImageInfos[0];
2020-05-20 18:09:52 +02:00
user.ProfileImage = new ImageInfo(info.Path)
{
LastModified = info.DateModified
};
}
2020-05-13 04:25:45 +02:00
user.SetPermission(PermissionKind.IsAdministrator, policy.IsAdministrator);
user.SetPermission(PermissionKind.IsHidden, policy.IsHidden);
user.SetPermission(PermissionKind.IsDisabled, policy.IsDisabled);
user.SetPermission(PermissionKind.EnableSharedDeviceControl, policy.EnableSharedDeviceControl);
user.SetPermission(PermissionKind.EnableRemoteAccess, policy.EnableRemoteAccess);
user.SetPermission(PermissionKind.EnableLiveTvManagement, policy.EnableLiveTvManagement);
user.SetPermission(PermissionKind.EnableLiveTvAccess, policy.EnableLiveTvAccess);
user.SetPermission(PermissionKind.EnableMediaPlayback, policy.EnableMediaPlayback);
user.SetPermission(PermissionKind.EnableAudioPlaybackTranscoding, policy.EnableAudioPlaybackTranscoding);
user.SetPermission(PermissionKind.EnableVideoPlaybackTranscoding, policy.EnableVideoPlaybackTranscoding);
user.SetPermission(PermissionKind.EnableContentDeletion, policy.EnableContentDeletion);
user.SetPermission(PermissionKind.EnableContentDownloading, policy.EnableContentDownloading);
user.SetPermission(PermissionKind.EnableSyncTranscoding, policy.EnableSyncTranscoding);
user.SetPermission(PermissionKind.EnableMediaConversion, policy.EnableMediaConversion);
user.SetPermission(PermissionKind.EnableAllChannels, policy.EnableAllChannels);
user.SetPermission(PermissionKind.EnableAllDevices, policy.EnableAllDevices);
user.SetPermission(PermissionKind.EnableAllFolders, policy.EnableAllFolders);
user.SetPermission(PermissionKind.EnableRemoteControlOfOtherUsers, policy.EnableRemoteControlOfOtherUsers);
user.SetPermission(PermissionKind.EnablePlaybackRemuxing, policy.EnablePlaybackRemuxing);
user.SetPermission(PermissionKind.ForceRemoteSourceTranscoding, policy.ForceRemoteSourceTranscoding);
user.SetPermission(PermissionKind.EnablePublicSharing, policy.EnablePublicSharing);
user.SetPermission(PermissionKind.EnableCollectionManagement, policy.EnableCollectionManagement);
2020-05-15 23:24:01 +02:00
2020-05-13 04:25:45 +02:00
foreach (var policyAccessSchedule in policy.AccessSchedules)
{
user.AccessSchedules.Add(policyAccessSchedule);
}
user.SetPreference(PreferenceKind.BlockedTags, policy.BlockedTags);
2020-12-11 23:00:43 +01:00
user.SetPreference(PreferenceKind.EnabledChannels, policy.EnabledChannels);
2020-05-13 04:25:45 +02:00
user.SetPreference(PreferenceKind.EnabledDevices, policy.EnabledDevices);
2020-12-11 23:00:43 +01:00
user.SetPreference(PreferenceKind.EnabledFolders, policy.EnabledFolders);
2020-05-13 04:25:45 +02:00
user.SetPreference(PreferenceKind.EnableContentDeletionFromFolders, policy.EnableContentDeletionFromFolders);
user.SetPreference(PreferenceKind.OrderedViews, config.OrderedViews);
user.SetPreference(PreferenceKind.GroupedFolders, config.GroupedFolders);
user.SetPreference(PreferenceKind.MyMediaExcludes, config.MyMediaExcludes);
user.SetPreference(PreferenceKind.LatestItemExcludes, config.LatestItemsExcludes);
2020-05-15 23:24:01 +02:00
dbContext.Users.Add(user);
2020-05-13 04:25:45 +02:00
}
dbContext.SaveChanges();
}
try
{
2020-05-15 23:24:01 +02:00
File.Move(Path.Combine(dataPath, DbFilename), Path.Combine(dataPath, DbFilename + ".old"));
var journalPath = Path.Combine(dataPath, DbFilename + "-journal");
if (File.Exists(journalPath))
{
File.Move(journalPath, Path.Combine(dataPath, DbFilename + ".old-journal"));
}
2020-05-13 04:25:45 +02:00
}
catch (IOException e)
{
_logger.LogError(e, "Error renaming legacy user database to 'users.db.old'");
}
}
2020-05-15 23:24:01 +02:00
#nullable disable
internal class UserMockup
{
public string Password { get; set; }
public string EasyPassword { get; set; }
public DateTime? LastLoginDate { get; set; }
2020-05-15 23:24:01 +02:00
public DateTime? LastActivityDate { get; set; }
2020-05-15 23:24:01 +02:00
public string Name { get; set; }
public ItemImageInfo[] ImageInfos { get; set; }
2020-05-15 23:24:01 +02:00
}
2020-05-13 04:10:35 +02:00
}
}