using MediaBrowser.Common.Configuration; using MediaBrowser.Controller.Providers; using MediaBrowser.Model.Logging; using System; using System.Data; using System.IO; using System.Linq; using System.Threading; using System.Threading.Tasks; namespace MediaBrowser.Server.Implementations.Persistence { public class SqliteProviderInfoRepository : IProviderRepository { private IDbConnection _connection; private readonly ILogger _logger; private IDbCommand _saveStatusCommand; private readonly IApplicationPaths _appPaths; public SqliteProviderInfoRepository(IApplicationPaths appPaths, ILogManager logManager) { _appPaths = appPaths; _logger = logManager.GetLogger(GetType().Name); } private SqliteShrinkMemoryTimer _shrinkMemoryTimer; /// /// Gets the name of the repository /// /// The name. public string Name { get { return "SQLite"; } } /// /// Opens the connection to the database /// /// Task. public async Task Initialize() { var dbFile = Path.Combine(_appPaths.DataPath, "refreshinfo.db"); _connection = await SqliteExtensions.ConnectToDb(dbFile, _logger).ConfigureAwait(false); string[] queries = { "create table if not exists MetadataStatus (ItemId GUID PRIMARY KEY, ItemName TEXT, SeriesName TEXT, DateLastMetadataRefresh datetime, DateLastImagesRefresh datetime, LastStatus TEXT, LastErrorMessage TEXT, MetadataProvidersRefreshed TEXT, ImageProvidersRefreshed TEXT)", "create index if not exists idx_MetadataStatus on MetadataStatus(ItemId)", //pragmas "pragma temp_store = memory", "pragma shrink_memory" }; _connection.RunQueries(queries, _logger); PrepareStatements(); _shrinkMemoryTimer = new SqliteShrinkMemoryTimer(_connection, _writeLock, _logger); } private static readonly string[] StatusColumns = { "ItemId", "ItemName", "SeriesName", "DateLastMetadataRefresh", "DateLastImagesRefresh", "LastStatus", "LastErrorMessage", "MetadataProvidersRefreshed", "ImageProvidersRefreshed" }; /// /// The _write lock /// private readonly SemaphoreSlim _writeLock = new SemaphoreSlim(1, 1); /// /// Prepares the statements. /// private void PrepareStatements() { _saveStatusCommand = _connection.CreateCommand(); _saveStatusCommand.CommandText = string.Format("replace into MetadataStatus ({0}) values ({1})", string.Join(",", StatusColumns), string.Join(",", StatusColumns.Select(i => "@" + i).ToArray())); foreach (var col in StatusColumns) { _saveStatusCommand.Parameters.Add(_saveStatusCommand, "@" + col); } } public MetadataStatus GetMetadataStatus(Guid itemId) { if (itemId == Guid.Empty) { throw new ArgumentNullException("itemId"); } using (var cmd = _connection.CreateCommand()) { var cmdText = "select " + string.Join(",", StatusColumns) + " from MetadataStatus where"; cmdText += " ItemId=@ItemId"; cmd.Parameters.Add(cmd, "@ItemId", DbType.Guid).Value = itemId; cmd.CommandText = cmdText; using (var reader = cmd.ExecuteReader(CommandBehavior.SequentialAccess | CommandBehavior.SingleResult | CommandBehavior.SingleRow)) { while (reader.Read()) { return GetStatus(reader); } return null; } } } private MetadataStatus GetStatus(IDataReader reader) { var result = new MetadataStatus { ItemId = reader.GetGuid(0) }; if (!reader.IsDBNull(1)) { result.ItemName = reader.GetString(1); } if (!reader.IsDBNull(2)) { result.SeriesName = reader.GetString(2); } if (!reader.IsDBNull(3)) { result.DateLastMetadataRefresh = reader.GetDateTime(3).ToUniversalTime(); } if (!reader.IsDBNull(4)) { result.DateLastImagesRefresh = reader.GetDateTime(4).ToUniversalTime(); } if (!reader.IsDBNull(5)) { result.LastStatus = (ProviderRefreshStatus)Enum.Parse(typeof(ProviderRefreshStatus), reader.GetString(5), true); } if (!reader.IsDBNull(6)) { result.LastErrorMessage = reader.GetString(6); } if (!reader.IsDBNull(7)) { result.MetadataProvidersRefreshed = reader.GetString(7).Split('|').Where(i => !string.IsNullOrEmpty(i)).Select(i => new Guid(i)).ToList(); } if (!reader.IsDBNull(8)) { result.ImageProvidersRefreshed = reader.GetString(8).Split('|').Where(i => !string.IsNullOrEmpty(i)).Select(i => new Guid(i)).ToList(); } return result; } public async Task SaveMetadataStatus(MetadataStatus status, CancellationToken cancellationToken) { if (status == null) { throw new ArgumentNullException("status"); } cancellationToken.ThrowIfCancellationRequested(); await _writeLock.WaitAsync(cancellationToken).ConfigureAwait(false); IDbTransaction transaction = null; try { transaction = _connection.BeginTransaction(); _saveStatusCommand.GetParameter(0).Value = status.ItemId; _saveStatusCommand.GetParameter(1).Value = status.ItemName; _saveStatusCommand.GetParameter(2).Value = status.SeriesName; _saveStatusCommand.GetParameter(3).Value = status.DateLastMetadataRefresh; _saveStatusCommand.GetParameter(4).Value = status.DateLastImagesRefresh; _saveStatusCommand.GetParameter(5).Value = status.LastStatus.ToString(); _saveStatusCommand.GetParameter(6).Value = status.LastErrorMessage; _saveStatusCommand.GetParameter(7).Value = string.Join("|", status.MetadataProvidersRefreshed.ToArray()); _saveStatusCommand.GetParameter(8).Value = string.Join("|", status.ImageProvidersRefreshed.ToArray()); _saveStatusCommand.Transaction = transaction; _saveStatusCommand.ExecuteNonQuery(); transaction.Commit(); } catch (OperationCanceledException) { if (transaction != null) { transaction.Rollback(); } throw; } catch (Exception e) { _logger.ErrorException("Failed to save provider info:", e); if (transaction != null) { transaction.Rollback(); } throw; } finally { if (transaction != null) { transaction.Dispose(); } _writeLock.Release(); } } /// /// Performs application-defined tasks associated with freeing, releasing, or resetting unmanaged resources. /// public void Dispose() { Dispose(true); GC.SuppressFinalize(this); } private readonly object _disposeLock = new object(); /// /// Releases unmanaged and - optionally - managed resources. /// /// true to release both managed and unmanaged resources; false to release only unmanaged resources. protected virtual void Dispose(bool dispose) { if (dispose) { try { lock (_disposeLock) { if (_shrinkMemoryTimer != null) { _shrinkMemoryTimer.Dispose(); _shrinkMemoryTimer = null; } if (_connection != null) { if (_connection.IsOpen()) { _connection.Close(); } _connection.Dispose(); _connection = null; } } } catch (Exception ex) { _logger.ErrorException("Error disposing database", ex); } } } } }