MigrateUserDb.cs 9.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189
  1. using System;
  2. using System.IO;
  3. using Emby.Server.Implementations.Data;
  4. using Emby.Server.Implementations.Serialization;
  5. using Jellyfin.Data.Entities;
  6. using Jellyfin.Data.Enums;
  7. using Jellyfin.Server.Implementations;
  8. using Jellyfin.Server.Implementations.Users;
  9. using MediaBrowser.Controller;
  10. using MediaBrowser.Model.Configuration;
  11. using MediaBrowser.Model.Users;
  12. using Microsoft.Extensions.Logging;
  13. using SQLitePCL.pretty;
  14. using JsonSerializer = System.Text.Json.JsonSerializer;
  15. namespace Jellyfin.Server.Migrations.Routines
  16. {
  17. /// <summary>
  18. /// The migration routine for migrating the user database to EF Core.
  19. /// </summary>
  20. public class MigrateUserDb : IMigrationRoutine
  21. {
  22. private const string DbFilename = "users.db";
  23. private readonly ILogger<MigrateUserDb> _logger;
  24. private readonly IServerApplicationPaths _paths;
  25. private readonly JellyfinDbProvider _provider;
  26. private readonly MyXmlSerializer _xmlSerializer;
  27. /// <summary>
  28. /// Initializes a new instance of the <see cref="MigrateUserDb"/> class.
  29. /// </summary>
  30. /// <param name="logger">The logger.</param>
  31. /// <param name="paths">The server application paths.</param>
  32. /// <param name="provider">The database provider.</param>
  33. /// <param name="xmlSerializer">The xml serializer.</param>
  34. public MigrateUserDb(
  35. ILogger<MigrateUserDb> logger,
  36. IServerApplicationPaths paths,
  37. JellyfinDbProvider provider,
  38. MyXmlSerializer xmlSerializer)
  39. {
  40. _logger = logger;
  41. _paths = paths;
  42. _provider = provider;
  43. _xmlSerializer = xmlSerializer;
  44. }
  45. /// <inheritdoc/>
  46. public Guid Id => Guid.Parse("5C4B82A2-F053-4009-BD05-B6FCAD82F14C");
  47. /// <inheritdoc/>
  48. public string Name => "MigrateUserDatabase";
  49. /// <inheritdoc/>
  50. public void Perform()
  51. {
  52. var dataPath = _paths.DataPath;
  53. _logger.LogInformation("Migrating the user database may take a while, do not stop Jellyfin.");
  54. using (var connection = SQLite3.Open(Path.Combine(dataPath, DbFilename), ConnectionFlags.ReadOnly, null))
  55. {
  56. var dbContext = _provider.CreateContext();
  57. var queryResult = connection.Query("SELECT * FROM LocalUsersv2");
  58. dbContext.RemoveRange(dbContext.Users);
  59. dbContext.SaveChanges();
  60. foreach (var entry in queryResult)
  61. {
  62. UserMockup mockup = JsonSerializer.Deserialize<UserMockup>(entry[2].ToBlob());
  63. var userDataDir = Path.Combine(_paths.UserConfigurationDirectoryPath, mockup.Name);
  64. var config = File.Exists(Path.Combine(userDataDir, "config.xml"))
  65. ? (UserConfiguration)_xmlSerializer.DeserializeFromFile(typeof(UserConfiguration), Path.Combine(userDataDir, "config.xml"))
  66. : new UserConfiguration();
  67. var policy = File.Exists(Path.Combine(userDataDir, "policy.xml"))
  68. ? (UserPolicy)_xmlSerializer.DeserializeFromFile(typeof(UserPolicy), Path.Combine(userDataDir, "policy.xml"))
  69. : new UserPolicy();
  70. policy.AuthenticationProviderId = policy.AuthenticationProviderId?.Replace(
  71. "Emby.Server.Implementations.Library",
  72. "Jellyfin.Server.Implementations.Users",
  73. StringComparison.Ordinal)
  74. ?? typeof(DefaultAuthenticationProvider).FullName;
  75. policy.PasswordResetProviderId ??= typeof(DefaultPasswordResetProvider).FullName;
  76. var user = new User(mockup.Name, policy.AuthenticationProviderId)
  77. {
  78. Id = entry[1].ReadGuidFromBlob(),
  79. InternalId = entry[0].ToInt64(),
  80. MaxParentalAgeRating = policy.MaxParentalRating,
  81. EnableUserPreferenceAccess = policy.EnableUserPreferenceAccess,
  82. RemoteClientBitrateLimit = policy.RemoteClientBitrateLimit,
  83. PasswordResetProviderId = policy.PasswordResetProviderId,
  84. InvalidLoginAttemptCount = policy.InvalidLoginAttemptCount,
  85. LoginAttemptsBeforeLockout = policy.LoginAttemptsBeforeLockout == -1 ? null : new int?(policy.LoginAttemptsBeforeLockout),
  86. SubtitleMode = config.SubtitleMode,
  87. HidePlayedInLatest = config.HidePlayedInLatest,
  88. EnableLocalPassword = config.EnableLocalPassword,
  89. PlayDefaultAudioTrack = config.PlayDefaultAudioTrack,
  90. DisplayCollectionsView = config.DisplayCollectionsView,
  91. DisplayMissingEpisodes = config.DisplayMissingEpisodes,
  92. AudioLanguagePreference = config.AudioLanguagePreference,
  93. RememberAudioSelections = config.RememberAudioSelections,
  94. EnableNextEpisodeAutoPlay = config.EnableNextEpisodeAutoPlay,
  95. RememberSubtitleSelections = config.RememberSubtitleSelections,
  96. SubtitleLanguagePreference = config.SubtitleLanguagePreference,
  97. Password = mockup.Password,
  98. EasyPassword = mockup.EasyPassword,
  99. LastLoginDate = mockup.LastLoginDate ?? DateTime.MinValue,
  100. LastActivityDate = mockup.LastActivityDate ?? DateTime.MinValue
  101. };
  102. user.SetPermission(PermissionKind.IsAdministrator, policy.IsAdministrator);
  103. user.SetPermission(PermissionKind.IsHidden, policy.IsHidden);
  104. user.SetPermission(PermissionKind.IsDisabled, policy.IsDisabled);
  105. user.SetPermission(PermissionKind.EnableSharedDeviceControl, policy.EnableSharedDeviceControl);
  106. user.SetPermission(PermissionKind.EnableRemoteAccess, policy.EnableRemoteAccess);
  107. user.SetPermission(PermissionKind.EnableLiveTvManagement, policy.EnableLiveTvManagement);
  108. user.SetPermission(PermissionKind.EnableLiveTvAccess, policy.EnableLiveTvAccess);
  109. user.SetPermission(PermissionKind.EnableMediaPlayback, policy.EnableMediaPlayback);
  110. user.SetPermission(PermissionKind.EnableAudioPlaybackTranscoding, policy.EnableAudioPlaybackTranscoding);
  111. user.SetPermission(PermissionKind.EnableVideoPlaybackTranscoding, policy.EnableVideoPlaybackTranscoding);
  112. user.SetPermission(PermissionKind.EnableContentDeletion, policy.EnableContentDeletion);
  113. user.SetPermission(PermissionKind.EnableContentDownloading, policy.EnableContentDownloading);
  114. user.SetPermission(PermissionKind.EnableSyncTranscoding, policy.EnableSyncTranscoding);
  115. user.SetPermission(PermissionKind.EnableMediaConversion, policy.EnableMediaConversion);
  116. user.SetPermission(PermissionKind.EnableAllChannels, policy.EnableAllChannels);
  117. user.SetPermission(PermissionKind.EnableAllDevices, policy.EnableAllDevices);
  118. user.SetPermission(PermissionKind.EnableAllFolders, policy.EnableAllFolders);
  119. user.SetPermission(PermissionKind.EnableRemoteControlOfOtherUsers, policy.EnableRemoteControlOfOtherUsers);
  120. user.SetPermission(PermissionKind.EnablePlaybackRemuxing, policy.EnablePlaybackRemuxing);
  121. user.SetPermission(PermissionKind.ForceRemoteSourceTranscoding, policy.ForceRemoteSourceTranscoding);
  122. user.SetPermission(PermissionKind.EnablePublicSharing, policy.EnablePublicSharing);
  123. foreach (var policyAccessSchedule in policy.AccessSchedules)
  124. {
  125. user.AccessSchedules.Add(policyAccessSchedule);
  126. }
  127. user.SetPreference(PreferenceKind.BlockedTags, policy.BlockedTags);
  128. user.SetPreference(PreferenceKind.EnabledChannels, policy.EnabledChannels);
  129. user.SetPreference(PreferenceKind.EnabledDevices, policy.EnabledDevices);
  130. user.SetPreference(PreferenceKind.EnabledFolders, policy.EnabledFolders);
  131. user.SetPreference(PreferenceKind.EnableContentDeletionFromFolders, policy.EnableContentDeletionFromFolders);
  132. user.SetPreference(PreferenceKind.OrderedViews, config.OrderedViews);
  133. user.SetPreference(PreferenceKind.GroupedFolders, config.GroupedFolders);
  134. user.SetPreference(PreferenceKind.MyMediaExcludes, config.MyMediaExcludes);
  135. user.SetPreference(PreferenceKind.LatestItemExcludes, config.LatestItemsExcludes);
  136. dbContext.Users.Add(user);
  137. }
  138. dbContext.SaveChanges();
  139. }
  140. try
  141. {
  142. File.Move(Path.Combine(dataPath, DbFilename), Path.Combine(dataPath, DbFilename + ".old"));
  143. var journalPath = Path.Combine(dataPath, DbFilename + "-journal");
  144. if (File.Exists(journalPath))
  145. {
  146. File.Move(journalPath, Path.Combine(dataPath, DbFilename + ".old-journal"));
  147. }
  148. }
  149. catch (IOException e)
  150. {
  151. _logger.LogError(e, "Error renaming legacy user database to 'users.db.old'");
  152. }
  153. }
  154. #nullable disable
  155. internal class UserMockup
  156. {
  157. public string Password { get; set; }
  158. public string EasyPassword { get; set; }
  159. public DateTime? LastLoginDate { get; set; }
  160. public DateTime? LastActivityDate { get; set; }
  161. public string Name { get; set; }
  162. }
  163. }
  164. }