< Summary - Jellyfin

Information
Class: Jellyfin.Database.Providers.Sqlite.SqliteDatabaseProvider
Assembly: Jellyfin.Database.Providers.Sqlite
File(s): /srv/git/jellyfin/src/Jellyfin.Database/Jellyfin.Database.Providers.Sqlite/SqliteDatabaseProvider.cs
Line coverage
42%
Covered lines: 11
Uncovered lines: 15
Coverable lines: 26
Total lines: 121
Line coverage: 42.3%
Branch coverage
0%
Covered branches: 0
Total branches: 4
Branch coverage: 0%
Method coverage

Feature is only available for sponsors

Upgrade to PRO version

Coverage history

Coverage history 0 25 50 75 100

Metrics

MethodBranch coverage Crap Score Cyclomatic complexity Line coverage
.ctor(...)100%11100%
Initialise(...)100%11100%
OnModelCreating(...)100%11100%
ConfigureConventions(...)100%11100%
MigrationBackupFast(...)0%620%
RestoreBackupFast(...)0%620%

File(s)

/srv/git/jellyfin/src/Jellyfin.Database/Jellyfin.Database.Providers.Sqlite/SqliteDatabaseProvider.cs

#LineLine coverage
 1using System;
 2using System.Globalization;
 3using System.IO;
 4using System.Threading;
 5using System.Threading.Tasks;
 6using Jellyfin.Database.Implementations;
 7using MediaBrowser.Common.Configuration;
 8using Microsoft.Data.Sqlite;
 9using Microsoft.EntityFrameworkCore;
 10using Microsoft.Extensions.Logging;
 11
 12namespace Jellyfin.Database.Providers.Sqlite;
 13
 14/// <summary>
 15/// Configures jellyfin to use an SQLite database.
 16/// </summary>
 17[JellyfinDatabaseProviderKey("Jellyfin-SQLite")]
 18public sealed class SqliteDatabaseProvider : IJellyfinDatabaseProvider
 19{
 20    private const string BackupFolderName = "SQLiteBackups";
 21    private readonly IApplicationPaths _applicationPaths;
 22    private readonly ILogger<SqliteDatabaseProvider> _logger;
 23
 24    /// <summary>
 25    /// Initializes a new instance of the <see cref="SqliteDatabaseProvider"/> class.
 26    /// </summary>
 27    /// <param name="applicationPaths">Service to construct the fallback when the old data path configuration is used.</
 28    /// <param name="logger">A logger.</param>
 29    public SqliteDatabaseProvider(IApplicationPaths applicationPaths, ILogger<SqliteDatabaseProvider> logger)
 30    {
 2231        _applicationPaths = applicationPaths;
 2232        _logger = logger;
 2233    }
 34
 35    /// <inheritdoc/>
 36    public IDbContextFactory<JellyfinDbContext>? DbContextFactory { get; set; }
 37
 38    /// <inheritdoc/>
 39    public void Initialise(DbContextOptionsBuilder options)
 40    {
 2141        options.UseSqlite(
 2142            $"Filename={Path.Combine(_applicationPaths.DataPath, "jellyfin.db")};Pooling=false",
 2143            sqLiteOptions => sqLiteOptions.MigrationsAssembly(GetType().Assembly));
 2144    }
 45
 46    /// <inheritdoc/>
 47    public async Task RunScheduledOptimisation(CancellationToken cancellationToken)
 48    {
 49        var context = await DbContextFactory!.CreateDbContextAsync(cancellationToken).ConfigureAwait(false);
 50        await using (context.ConfigureAwait(false))
 51        {
 52            if (context.Database.IsSqlite())
 53            {
 54                await context.Database.ExecuteSqlRawAsync("PRAGMA optimize", cancellationToken).ConfigureAwait(false);
 55                await context.Database.ExecuteSqlRawAsync("VACUUM", cancellationToken).ConfigureAwait(false);
 56                _logger.LogInformation("jellyfin.db optimized successfully!");
 57            }
 58            else
 59            {
 60                _logger.LogInformation("This database doesn't support optimization");
 61            }
 62        }
 63    }
 64
 65    /// <inheritdoc/>
 66    public void OnModelCreating(ModelBuilder modelBuilder)
 67    {
 268        modelBuilder.SetDefaultDateTimeKind(DateTimeKind.Utc);
 269    }
 70
 71    /// <inheritdoc/>
 72    public async Task RunShutdownTask(CancellationToken cancellationToken)
 73    {
 74        // Run before disposing the application
 75        var context = await DbContextFactory!.CreateDbContextAsync(cancellationToken).ConfigureAwait(false);
 76        await using (context.ConfigureAwait(false))
 77        {
 78            await context.Database.ExecuteSqlRawAsync("PRAGMA optimize", cancellationToken).ConfigureAwait(false);
 79        }
 80
 81        SqliteConnection.ClearAllPools();
 82    }
 83
 84    /// <inheritdoc/>
 85    public void ConfigureConventions(ModelConfigurationBuilder configurationBuilder)
 86    {
 287        configurationBuilder.Conventions.Add(_ => new DoNotUseReturningClauseConvention());
 288    }
 89
 90    /// <inheritdoc />
 91    public Task<string> MigrationBackupFast(CancellationToken cancellationToken)
 92    {
 093        var key = DateTime.UtcNow.ToString("yyyyMMddhhmmss", CultureInfo.InvariantCulture);
 094        var path = Path.Combine(_applicationPaths.DataPath, "jellyfin.db");
 095        var backupFile = Path.Combine(_applicationPaths.DataPath, BackupFolderName);
 096        if (!Directory.Exists(backupFile))
 97        {
 098            Directory.CreateDirectory(backupFile);
 99        }
 100
 0101        backupFile = Path.Combine(_applicationPaths.DataPath, $"{key}_jellyfin.db");
 0102        File.Copy(path, backupFile);
 0103        return Task.FromResult(key);
 104    }
 105
 106    /// <inheritdoc />
 107    public Task RestoreBackupFast(string key, CancellationToken cancellationToken)
 108    {
 0109        var path = Path.Combine(_applicationPaths.DataPath, "jellyfin.db");
 0110        var backupFile = Path.Combine(_applicationPaths.DataPath, BackupFolderName, $"{key}_jellyfin.db");
 111
 0112        if (!File.Exists(backupFile))
 113        {
 0114            _logger.LogCritical("Tried to restore a backup that does not exist.");
 0115            return Task.CompletedTask;
 116        }
 117
 0118        File.Copy(backupFile, path, true);
 0119        return Task.CompletedTask;
 120    }
 121}