#nullable disable #pragma warning disable CS1591 using System; using System.Collections.Generic; using System.Threading; using Jellyfin.Extensions; using Microsoft.Data.Sqlite; using Microsoft.Extensions.Logging; namespace Emby.Server.Implementations.Data { public abstract class BaseSqliteRepository : IDisposable { private bool _disposed = false; private SemaphoreSlim _writeLock = new SemaphoreSlim(1, 1); private SqliteConnection _writeConnection; /// /// Initializes a new instance of the class. /// /// The logger. protected BaseSqliteRepository(ILogger logger) { Logger = logger; } /// /// Gets or sets the path to the DB file. /// protected string DbFilePath { get; set; } /// /// Gets the logger. /// /// The logger. protected ILogger Logger { get; } /// /// Gets the cache size. /// /// The cache size or null. protected virtual int? CacheSize => null; /// /// Gets the locking mode. . /// protected virtual string LockingMode => "NORMAL"; /// /// Gets the journal mode. . /// /// The journal mode. protected virtual string JournalMode => "WAL"; /// /// Gets the journal size limit. . /// The default (-1) is overridden to prevent unconstrained WAL size, as reported by users. /// /// The journal size limit. protected virtual int? JournalSizeLimit => 134_217_728; // 128MiB /// /// Gets the page size. /// /// The page size or null. protected virtual int? PageSize => null; /// /// Gets the temp store mode. /// /// The temp store mode. /// protected virtual TempStoreMode TempStore => TempStoreMode.Memory; /// /// Gets the synchronous mode. /// /// The synchronous mode or null. /// protected virtual SynchronousMode? Synchronous => SynchronousMode.Normal; public virtual void Initialize() { // Configuration and pragmas can affect VACUUM so it needs to be last. using (var connection = GetConnection()) { connection.Execute("VACUUM"); } } protected ManagedConnection GetConnection(bool readOnly = false) { if (!readOnly) { _writeLock.Wait(); if (_writeConnection is not null) { return new ManagedConnection(_writeConnection, _writeLock); } var writeConnection = new SqliteConnection($"Filename={DbFilePath};Pooling=False"); writeConnection.Open(); if (CacheSize.HasValue) { writeConnection.Execute("PRAGMA cache_size=" + CacheSize.Value); } if (!string.IsNullOrWhiteSpace(LockingMode)) { writeConnection.Execute("PRAGMA locking_mode=" + LockingMode); } if (!string.IsNullOrWhiteSpace(JournalMode)) { writeConnection.Execute("PRAGMA journal_mode=" + JournalMode); } if (JournalSizeLimit.HasValue) { writeConnection.Execute("PRAGMA journal_size_limit=" + JournalSizeLimit.Value); } if (Synchronous.HasValue) { writeConnection.Execute("PRAGMA synchronous=" + (int)Synchronous.Value); } if (PageSize.HasValue) { writeConnection.Execute("PRAGMA page_size=" + PageSize.Value); } writeConnection.Execute("PRAGMA temp_store=" + (int)TempStore); return new ManagedConnection(_writeConnection = writeConnection, _writeLock); } var connection = new SqliteConnection($"Filename={DbFilePath};Mode=ReadOnly"); connection.Open(); if (CacheSize.HasValue) { connection.Execute("PRAGMA cache_size=" + CacheSize.Value); } if (!string.IsNullOrWhiteSpace(LockingMode)) { connection.Execute("PRAGMA locking_mode=" + LockingMode); } if (!string.IsNullOrWhiteSpace(JournalMode)) { connection.Execute("PRAGMA journal_mode=" + JournalMode); } if (JournalSizeLimit.HasValue) { connection.Execute("PRAGMA journal_size_limit=" + JournalSizeLimit.Value); } if (Synchronous.HasValue) { connection.Execute("PRAGMA synchronous=" + (int)Synchronous.Value); } if (PageSize.HasValue) { connection.Execute("PRAGMA page_size=" + PageSize.Value); } connection.Execute("PRAGMA temp_store=" + (int)TempStore); return new ManagedConnection(connection, null); } public SqliteCommand PrepareStatement(ManagedConnection connection, string sql) { var command = connection.CreateCommand(); command.CommandText = sql; return command; } protected bool TableExists(ManagedConnection connection, string name) { using var statement = PrepareStatement(connection, "select DISTINCT tbl_name from sqlite_master"); foreach (var row in statement.ExecuteQuery()) { if (string.Equals(name, row.GetString(0), StringComparison.OrdinalIgnoreCase)) { return true; } } return false; } protected List GetColumnNames(ManagedConnection connection, string table) { var columnNames = new List(); foreach (var row in connection.Query("PRAGMA table_info(" + table + ")")) { if (row.TryGetString(1, out var columnName)) { columnNames.Add(columnName); } } return columnNames; } protected void AddColumn(ManagedConnection connection, string table, string columnName, string type, List existingColumnNames) { if (existingColumnNames.Contains(columnName, StringComparison.OrdinalIgnoreCase)) { return; } connection.Execute("alter table " + table + " add column " + columnName + " " + type + " NULL"); } protected void CheckDisposed() { ObjectDisposedException.ThrowIf(_disposed, this); } /// public void Dispose() { Dispose(true); GC.SuppressFinalize(this); } /// /// 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 (_disposed) { return; } if (dispose) { _writeLock.Wait(); try { _writeConnection.Dispose(); } finally { _writeLock.Release(); } _writeLock.Dispose(); } _writeConnection = null; _writeLock = null; _disposed = true; } } }