SqliteDisplayPreferencesRepository.cs 9.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256
  1. using System;
  2. using System.Collections.Generic;
  3. using System.IO;
  4. using System.Threading;
  5. using MediaBrowser.Common.Configuration;
  6. using MediaBrowser.Common.Extensions;
  7. using MediaBrowser.Controller.Persistence;
  8. using MediaBrowser.Model.Entities;
  9. using MediaBrowser.Model.IO;
  10. using MediaBrowser.Model.Logging;
  11. using MediaBrowser.Model.Serialization;
  12. using SQLitePCL.pretty;
  13. namespace Emby.Server.Implementations.Data
  14. {
  15. /// <summary>
  16. /// Class SQLiteDisplayPreferencesRepository
  17. /// </summary>
  18. public class SqliteDisplayPreferencesRepository : BaseSqliteRepository, IDisplayPreferencesRepository
  19. {
  20. private readonly IMemoryStreamFactory _memoryStreamProvider;
  21. protected IFileSystem FileSystem { get; private set; }
  22. public SqliteDisplayPreferencesRepository(ILogger logger, IJsonSerializer jsonSerializer, IApplicationPaths appPaths, IMemoryStreamFactory memoryStreamProvider, IFileSystem fileSystem)
  23. : base(logger)
  24. {
  25. _jsonSerializer = jsonSerializer;
  26. _memoryStreamProvider = memoryStreamProvider;
  27. FileSystem = fileSystem;
  28. DbFilePath = Path.Combine(appPaths.DataPath, "displaypreferences.db");
  29. }
  30. /// <summary>
  31. /// Gets the name of the repository
  32. /// </summary>
  33. /// <value>The name.</value>
  34. public string Name
  35. {
  36. get
  37. {
  38. return "SQLite";
  39. }
  40. }
  41. /// <summary>
  42. /// The _json serializer
  43. /// </summary>
  44. private readonly IJsonSerializer _jsonSerializer;
  45. public void Initialize()
  46. {
  47. try
  48. {
  49. InitializeInternal();
  50. }
  51. catch (Exception ex)
  52. {
  53. Logger.ErrorException("Error loading database file. Will reset and retry.", ex);
  54. FileSystem.DeleteFile(DbFilePath);
  55. InitializeInternal();
  56. }
  57. }
  58. /// <summary>
  59. /// Opens the connection to the database
  60. /// </summary>
  61. /// <returns>Task.</returns>
  62. private void InitializeInternal()
  63. {
  64. using (var connection = CreateConnection())
  65. {
  66. RunDefaultInitialization(connection);
  67. string[] queries = {
  68. "create table if not exists userdisplaypreferences (id GUID NOT NULL, userId GUID NOT NULL, client text NOT NULL, data BLOB NOT NULL)",
  69. "create unique index if not exists userdisplaypreferencesindex on userdisplaypreferences (id, userId, client)"
  70. };
  71. connection.RunQueries(queries);
  72. }
  73. }
  74. /// <summary>
  75. /// Save the display preferences associated with an item in the repo
  76. /// </summary>
  77. /// <param name="displayPreferences">The display preferences.</param>
  78. /// <param name="userId">The user id.</param>
  79. /// <param name="client">The client.</param>
  80. /// <param name="cancellationToken">The cancellation token.</param>
  81. /// <returns>Task.</returns>
  82. /// <exception cref="System.ArgumentNullException">item</exception>
  83. public void SaveDisplayPreferences(DisplayPreferences displayPreferences, Guid userId, string client, CancellationToken cancellationToken)
  84. {
  85. if (displayPreferences == null)
  86. {
  87. throw new ArgumentNullException("displayPreferences");
  88. }
  89. if (string.IsNullOrWhiteSpace(displayPreferences.Id))
  90. {
  91. throw new ArgumentNullException("displayPreferences.Id");
  92. }
  93. cancellationToken.ThrowIfCancellationRequested();
  94. using (WriteLock.Write())
  95. {
  96. using (var connection = CreateConnection())
  97. {
  98. connection.RunInTransaction(db =>
  99. {
  100. SaveDisplayPreferences(displayPreferences, userId, client, db);
  101. }, TransactionMode);
  102. }
  103. }
  104. }
  105. private void SaveDisplayPreferences(DisplayPreferences displayPreferences, Guid userId, string client, IDatabaseConnection connection)
  106. {
  107. var serialized = _jsonSerializer.SerializeToBytes(displayPreferences, _memoryStreamProvider);
  108. using (var statement = connection.PrepareStatement("replace into userdisplaypreferences (id, userid, client, data) values (@id, @userId, @client, @data)"))
  109. {
  110. statement.TryBind("@id", displayPreferences.Id.ToGuidBlob());
  111. statement.TryBind("@userId", userId.ToGuidBlob());
  112. statement.TryBind("@client", client);
  113. statement.TryBind("@data", serialized);
  114. statement.MoveNext();
  115. }
  116. }
  117. /// <summary>
  118. /// Save all display preferences associated with a user in the repo
  119. /// </summary>
  120. /// <param name="displayPreferences">The display preferences.</param>
  121. /// <param name="userId">The user id.</param>
  122. /// <param name="cancellationToken">The cancellation token.</param>
  123. /// <returns>Task.</returns>
  124. /// <exception cref="System.ArgumentNullException">item</exception>
  125. public void SaveAllDisplayPreferences(IEnumerable<DisplayPreferences> displayPreferences, Guid userId, CancellationToken cancellationToken)
  126. {
  127. if (displayPreferences == null)
  128. {
  129. throw new ArgumentNullException("displayPreferences");
  130. }
  131. cancellationToken.ThrowIfCancellationRequested();
  132. using (WriteLock.Write())
  133. {
  134. using (var connection = CreateConnection())
  135. {
  136. connection.RunInTransaction(db =>
  137. {
  138. foreach (var displayPreference in displayPreferences)
  139. {
  140. SaveDisplayPreferences(displayPreference, userId, displayPreference.Client, db);
  141. }
  142. }, TransactionMode);
  143. }
  144. }
  145. }
  146. /// <summary>
  147. /// Gets the display preferences.
  148. /// </summary>
  149. /// <param name="displayPreferencesId">The display preferences id.</param>
  150. /// <param name="userId">The user id.</param>
  151. /// <param name="client">The client.</param>
  152. /// <returns>Task{DisplayPreferences}.</returns>
  153. /// <exception cref="System.ArgumentNullException">item</exception>
  154. public DisplayPreferences GetDisplayPreferences(string displayPreferencesId, Guid userId, string client)
  155. {
  156. if (string.IsNullOrWhiteSpace(displayPreferencesId))
  157. {
  158. throw new ArgumentNullException("displayPreferencesId");
  159. }
  160. var guidId = displayPreferencesId.GetMD5();
  161. using (WriteLock.Read())
  162. {
  163. using (var connection = CreateConnection(true))
  164. {
  165. using (var statement = connection.PrepareStatement("select data from userdisplaypreferences where id = @id and userId=@userId and client=@client"))
  166. {
  167. statement.TryBind("@id", guidId.ToGuidBlob());
  168. statement.TryBind("@userId", userId.ToGuidBlob());
  169. statement.TryBind("@client", client);
  170. foreach (var row in statement.ExecuteQuery())
  171. {
  172. return Get(row);
  173. }
  174. }
  175. return new DisplayPreferences
  176. {
  177. Id = guidId.ToString("N")
  178. };
  179. }
  180. }
  181. }
  182. /// <summary>
  183. /// Gets all display preferences for the given user.
  184. /// </summary>
  185. /// <param name="userId">The user id.</param>
  186. /// <returns>Task{DisplayPreferences}.</returns>
  187. /// <exception cref="System.ArgumentNullException">item</exception>
  188. public IEnumerable<DisplayPreferences> GetAllDisplayPreferences(Guid userId)
  189. {
  190. var list = new List<DisplayPreferences>();
  191. using (WriteLock.Read())
  192. {
  193. using (var connection = CreateConnection(true))
  194. {
  195. using (var statement = connection.PrepareStatement("select data from userdisplaypreferences where userId=@userId"))
  196. {
  197. statement.TryBind("@userId", userId.ToGuidBlob());
  198. foreach (var row in statement.ExecuteQuery())
  199. {
  200. list.Add(Get(row));
  201. }
  202. }
  203. }
  204. }
  205. return list;
  206. }
  207. private DisplayPreferences Get(IReadOnlyList<IResultSetValue> row)
  208. {
  209. using (var stream = _memoryStreamProvider.CreateNew(row[0].ToBlob()))
  210. {
  211. stream.Position = 0;
  212. return _jsonSerializer.DeserializeFromStream<DisplayPreferences>(stream);
  213. }
  214. }
  215. public void SaveDisplayPreferences(DisplayPreferences displayPreferences, string userId, string client, CancellationToken cancellationToken)
  216. {
  217. SaveDisplayPreferences(displayPreferences, new Guid(userId), client, cancellationToken);
  218. }
  219. public DisplayPreferences GetDisplayPreferences(string displayPreferencesId, string userId, string client)
  220. {
  221. return GetDisplayPreferences(displayPreferencesId, new Guid(userId), client);
  222. }
  223. }
  224. }