47 lines
1.4 KiB
C#
47 lines
1.4 KiB
C#
using Clarity.Server.Data;
|
|
using Microsoft.EntityFrameworkCore;
|
|
using System.Diagnostics;
|
|
|
|
namespace Clarity.MigrationService;
|
|
|
|
public class Worker(
|
|
IServiceProvider serviceProvider,
|
|
IHostApplicationLifetime hostApplicationLifetime) : BackgroundService
|
|
{
|
|
public const string ActivitySourceName = "Migrations";
|
|
private static readonly ActivitySource s_activitySource = new(ActivitySourceName);
|
|
|
|
protected override async Task ExecuteAsync(
|
|
CancellationToken cancellationToken)
|
|
{
|
|
using var activity = s_activitySource.StartActivity(
|
|
"Migrating database", ActivityKind.Client);
|
|
|
|
try
|
|
{
|
|
using var scope = serviceProvider.CreateScope();
|
|
var dbContext = scope.ServiceProvider.GetRequiredService<ApplicationDbContext>();
|
|
|
|
await RunMigrationAsync(dbContext, cancellationToken);
|
|
}
|
|
catch (Exception ex)
|
|
{
|
|
activity?.AddException(ex);
|
|
throw;
|
|
}
|
|
|
|
hostApplicationLifetime.StopApplication();
|
|
}
|
|
|
|
private static async Task RunMigrationAsync(
|
|
ApplicationDbContext dbContext, CancellationToken cancellationToken)
|
|
{
|
|
var strategy = dbContext.Database.CreateExecutionStrategy();
|
|
await strategy.ExecuteAsync(async () =>
|
|
{
|
|
// Run migration in a transaction to avoid partial migration if it fails.
|
|
await dbContext.Database.MigrateAsync(cancellationToken);
|
|
});
|
|
}
|
|
}
|