Migration Safety (#967)
* Updated EF version * When we perform a migration, backup the database to temp/migration/VERSION and do it only once in case a migration fails. * When a migration fails, we will now restore what was corrupted.
This commit is contained in:
parent
d54e43edb1
commit
6c2731071d
2 changed files with 43 additions and 5 deletions
|
|
@ -62,7 +62,15 @@ namespace API
|
|||
if (pendingMigrations.Any())
|
||||
{
|
||||
logger.LogInformation("Performing backup as migrations are needed. Backup will be kavita.db in temp folder");
|
||||
directoryService.CopyFileToDirectory(directoryService.FileSystem.Path.Join(directoryService.ConfigDirectory, "kavita.db"), directoryService.TempDirectory);
|
||||
var migrationDirectory = await GetMigrationDirectory(context, directoryService);
|
||||
directoryService.ExistOrCreate(migrationDirectory);
|
||||
|
||||
if (!directoryService.FileSystem.File.Exists(
|
||||
directoryService.FileSystem.Path.Join(migrationDirectory, "kavita.db")))
|
||||
{
|
||||
directoryService.CopyFileToDirectory(directoryService.FileSystem.Path.Join(directoryService.ConfigDirectory, "kavita.db"), migrationDirectory);
|
||||
logger.LogInformation("Database backed up to {MigrationDirectory}", migrationDirectory);
|
||||
}
|
||||
}
|
||||
|
||||
await context.Database.MigrateAsync();
|
||||
|
|
@ -82,12 +90,42 @@ namespace API
|
|||
catch (Exception ex)
|
||||
{
|
||||
var logger = services.GetRequiredService<ILogger<Program>>();
|
||||
logger.LogCritical(ex, "An error occurred during migration");
|
||||
var context = services.GetRequiredService<DataContext>();
|
||||
var migrationDirectory = await GetMigrationDirectory(context, directoryService);
|
||||
|
||||
logger.LogCritical(ex, "A migration failed during startup. Restoring backup from {MigrationDirectory} and exiting", migrationDirectory);
|
||||
directoryService.CopyFileToDirectory(directoryService.FileSystem.Path.Join(migrationDirectory, "kavita.db"), directoryService.ConfigDirectory);
|
||||
|
||||
return;
|
||||
}
|
||||
|
||||
await host.RunAsync();
|
||||
}
|
||||
|
||||
private static async Task<string> GetMigrationDirectory(DataContext context, IDirectoryService directoryService)
|
||||
{
|
||||
string currentVersion = null;
|
||||
try
|
||||
{
|
||||
currentVersion =
|
||||
(await context.ServerSetting.SingleOrDefaultAsync(s =>
|
||||
s.Key == ServerSettingKey.InstallVersion))?.Value;
|
||||
}
|
||||
catch
|
||||
{
|
||||
// ignored
|
||||
}
|
||||
|
||||
if (string.IsNullOrEmpty(currentVersion))
|
||||
{
|
||||
currentVersion = "vUnknown";
|
||||
}
|
||||
|
||||
var migrationDirectory = directoryService.FileSystem.Path.Join(directoryService.TempDirectory,
|
||||
"migration", currentVersion);
|
||||
return migrationDirectory;
|
||||
}
|
||||
|
||||
private static IHostBuilder CreateHostBuilder(string[] args) =>
|
||||
Host.CreateDefaultBuilder(args)
|
||||
.ConfigureAppConfiguration((hostingContext, config) =>
|
||||
|
|
|
|||
Loading…
Add table
Add a link
Reference in a new issue