| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233 | using System;using System.IO;using Emby.Server.Implementations.Data;using Jellyfin.Data;using Jellyfin.Database.Implementations;using Jellyfin.Database.Implementations.Entities;using Jellyfin.Database.Implementations.Enums;using Jellyfin.Extensions.Json;using Jellyfin.Server.Implementations.Users;using MediaBrowser.Controller;using MediaBrowser.Controller.Entities;using MediaBrowser.Model.Configuration;using MediaBrowser.Model.Serialization;using MediaBrowser.Model.Users;using Microsoft.Data.Sqlite;using Microsoft.EntityFrameworkCore;using Microsoft.Extensions.Logging;using JsonSerializer = System.Text.Json.JsonSerializer;namespace Jellyfin.Server.Migrations.Routines;/// <summary>/// The migration routine for migrating the user database to EF Core./// </summary>#pragma warning disable CS0618 // Type or member is obsolete[JellyfinMigration("2025-04-20T10:00:00", nameof(MigrateUserDb), "5C4B82A2-F053-4009-BD05-B6FCAD82F14C")]public class MigrateUserDb : IMigrationRoutine#pragma warning restore CS0618 // Type or member is obsolete{    private const string DbFilename = "users.db";    private readonly ILogger<MigrateUserDb> _logger;    private readonly IServerApplicationPaths _paths;    private readonly IDbContextFactory<JellyfinDbContext> _provider;    private readonly IXmlSerializer _xmlSerializer;    /// <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,        IDbContextFactory<JellyfinDbContext> provider,        IXmlSerializer xmlSerializer)    {        _logger = logger;        _paths = paths;        _provider = provider;        _xmlSerializer = xmlSerializer;    }    /// <inheritdoc/>    public void Perform()    {        var dataPath = _paths.DataPath;        var userDbPath = Path.Combine(dataPath, DbFilename);        if (!File.Exists(userDbPath))        {            _logger.LogWarning("{UserDbPath} doesn't exist, nothing to migrate", userDbPath);            return;        }        _logger.LogInformation("Migrating the user database may take a while, do not stop Jellyfin.");        using (var connection = new SqliteConnection($"Filename={userDbPath}"))        {            connection.Open();            var tableQuery = connection.Query("SELECT count(*) FROM sqlite_master WHERE type='table' AND name='LocalUsersv2';");            foreach (var row in tableQuery)            {                if (row.GetInt32(0) == 0)                {                    _logger.LogWarning("Table 'LocalUsersv2' doesn't exist in {UserDbPath}, nothing to migrate", userDbPath);                    break;                }            }            using var dbContext = _provider.CreateDbContext();            var queryResult = connection.Query("SELECT * FROM LocalUsersv2");            dbContext.RemoveRange(dbContext.Users);            dbContext.SaveChanges();            foreach (var entry in queryResult)            {                UserMockup? mockup = JsonSerializer.Deserialize<UserMockup>(entry.GetStream(2), JsonDefaults.Options);                if (mockup is null)                {                    continue;                }                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()                    : new UserConfiguration();                var policyPath = Path.Combine(userDataDir, "policy.xml");                var policy = File.Exists(policyPath)                    ? (UserPolicy?)_xmlSerializer.DeserializeFromFile(typeof(UserPolicy), policyPath) ?? new UserPolicy()                    : 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;                int? maxLoginAttempts = policy.LoginAttemptsBeforeLockout switch                {                    -1 => null,                    0 => 3,                    _ => policy.LoginAttemptsBeforeLockout                };                var user = new User(mockup.Name, policy.AuthenticationProviderId!, policy.PasswordResetProviderId!)                {                    Id = entry.GetGuid(1),                    InternalId = entry.GetInt64(0),                    MaxParentalRatingScore = policy.MaxParentalRating,                    MaxParentalRatingSubScore = null,                    EnableUserPreferenceAccess = policy.EnableUserPreferenceAccess,                    RemoteClientBitrateLimit = policy.RemoteClientBitrateLimit,                    InvalidLoginAttemptCount = policy.InvalidLoginAttemptCount,                    LoginAttemptsBeforeLockout = maxLoginAttempts,                    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,                    Password = mockup.Password,                    LastLoginDate = mockup.LastLoginDate,                    LastActivityDate = mockup.LastActivityDate                };                if (mockup.ImageInfos.Length > 0)                {                    ItemImageInfo info = mockup.ImageInfos[0];                    user.ProfileImage = new ImageInfo(info.Path)                    {                        LastModified = info.DateModified                    };                }                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);                foreach (var policyAccessSchedule in policy.AccessSchedules)                {                    user.AccessSchedules.Add(policyAccessSchedule);                }                user.SetPreference(PreferenceKind.BlockedTags, policy.BlockedTags);                user.SetPreference(PreferenceKind.EnabledChannels, policy.EnabledChannels);                user.SetPreference(PreferenceKind.EnabledDevices, policy.EnabledDevices);                user.SetPreference(PreferenceKind.EnabledFolders, policy.EnabledFolders);                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);                dbContext.Users.Add(user);            }            dbContext.SaveChanges();        }        try        {            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"));            }        }        catch (IOException e)        {            _logger.LogError(e, "Error renaming legacy user database to 'users.db.old'");        }    }#nullable disable    internal class UserMockup    {        public string Password { get; set; }        public string EasyPassword { get; set; }        public DateTime? LastLoginDate { get; set; }        public DateTime? LastActivityDate { get; set; }        public string Name { get; set; }        public ItemImageInfo[] ImageInfos { get; set; }    }}
 |