175 lines
8.3 KiB
C#
175 lines
8.3 KiB
C#
using Microsoft.EntityFrameworkCore;
|
|
using Microsoft.Extensions.DependencyInjection;
|
|
using Microsoft.Extensions.Options;
|
|
using Serilog;
|
|
using Umbraco.Cms.Core;
|
|
using Umbraco.Cms.Core.Configuration.Models;
|
|
using Umbraco.Cms.Core.DistributedLocking;
|
|
using Umbraco.Cms.Persistence.EFCore.Locking;
|
|
using Umbraco.Cms.Persistence.EFCore.Scoping;
|
|
|
|
namespace Umbraco.Extensions;
|
|
|
|
public static class UmbracoEFCoreServiceCollectionExtensions
|
|
{
|
|
public delegate void DefaultEFCoreOptionsAction(DbContextOptionsBuilder options, string? providerName, string? connectionString);
|
|
|
|
[Obsolete("Use AddUmbracoDbContext<T>(this IServiceCollection services, Action<DbContextOptionsBuilder>? optionsAction = null) instead.")]
|
|
public static IServiceCollection AddUmbracoEFCoreContext<T>(this IServiceCollection services, DefaultEFCoreOptionsAction? defaultEFCoreOptionsAction = null)
|
|
where T : DbContext
|
|
{
|
|
services.AddPooledDbContextFactory<T>((provider, builder) => SetupDbContext(defaultEFCoreOptionsAction, provider, builder));
|
|
services.AddTransient(services => services.GetRequiredService<IDbContextFactory<T>>().CreateDbContext());
|
|
|
|
services.AddUnique<IAmbientEFCoreScopeStack<T>, AmbientEFCoreScopeStack<T>>();
|
|
services.AddUnique<IEFCoreScopeAccessor<T>, EFCoreScopeAccessor<T>>();
|
|
services.AddUnique<IEFCoreScopeProvider<T>, EFCoreScopeProvider<T>>();
|
|
services.AddSingleton<IDistributedLockingMechanism, SqliteEFCoreDistributedLockingMechanism<T>>();
|
|
services.AddSingleton<IDistributedLockingMechanism, SqlServerEFCoreDistributedLockingMechanism<T>>();
|
|
|
|
return services;
|
|
}
|
|
|
|
[Obsolete("Use AddUmbracoDbContext<T>(this IServiceCollection services, Action<DbContextOptionsBuilder>? optionsAction = null) instead.")]
|
|
public static IServiceCollection AddUmbracoEFCoreContext<T>(this IServiceCollection services, string connectionString, string providerName, DefaultEFCoreOptionsAction? defaultEFCoreOptionsAction = null)
|
|
where T : DbContext
|
|
{
|
|
// Replace data directory
|
|
string? dataDirectory = AppDomain.CurrentDomain.GetData(Constants.System.DataDirectoryName)?.ToString();
|
|
if (string.IsNullOrEmpty(dataDirectory) is false)
|
|
{
|
|
connectionString = connectionString.Replace(Constants.System.DataDirectoryPlaceholder, dataDirectory);
|
|
}
|
|
|
|
services.AddPooledDbContextFactory<T>(options => defaultEFCoreOptionsAction?.Invoke(options, providerName, connectionString));
|
|
services.AddTransient(services => services.GetRequiredService<IDbContextFactory<T>>().CreateDbContext());
|
|
|
|
services.AddUnique<IAmbientEFCoreScopeStack<T>, AmbientEFCoreScopeStack<T>>();
|
|
services.AddUnique<IEFCoreScopeAccessor<T>, EFCoreScopeAccessor<T>>();
|
|
services.AddUnique<IEFCoreScopeProvider<T>, EFCoreScopeProvider<T>>();
|
|
services.AddSingleton<IDistributedLockingMechanism, SqliteEFCoreDistributedLockingMechanism<T>>();
|
|
services.AddSingleton<IDistributedLockingMechanism, SqlServerEFCoreDistributedLockingMechanism<T>>();
|
|
|
|
return services;
|
|
}
|
|
|
|
/// <summary>
|
|
/// Adds a EFCore DbContext with all the services needed to integrate with Umbraco scopes.
|
|
/// </summary>
|
|
/// <typeparam name="T"></typeparam>
|
|
/// <param name="services"></param>
|
|
/// <param name="optionsAction"></param>
|
|
/// <returns></returns>
|
|
public static IServiceCollection AddUmbracoDbContext<T>(this IServiceCollection services, Action<DbContextOptionsBuilder>? optionsAction = null)
|
|
where T : DbContext
|
|
{
|
|
return AddUmbracoDbContext<T>(services, (IServiceProvider _, DbContextOptionsBuilder options) =>
|
|
{
|
|
optionsAction?.Invoke(options);
|
|
});
|
|
}
|
|
|
|
/// <summary>
|
|
/// Adds a EFCore DbContext with all the services needed to integrate with Umbraco scopes.
|
|
/// </summary>
|
|
/// <typeparam name="T"></typeparam>
|
|
/// <param name="services"></param>
|
|
/// <param name="optionsAction"></param>
|
|
/// <returns></returns>
|
|
public static IServiceCollection AddUmbracoDbContext<T>(this IServiceCollection services, Action<IServiceProvider, DbContextOptionsBuilder>? optionsAction = null)
|
|
where T : DbContext
|
|
{
|
|
optionsAction ??= (sp, options) => { };
|
|
|
|
services.AddPooledDbContextFactory<T>(optionsAction);
|
|
services.AddTransient(services => services.GetRequiredService<IDbContextFactory<T>>().CreateDbContext());
|
|
|
|
services.AddUnique<IAmbientEFCoreScopeStack<T>, AmbientEFCoreScopeStack<T>>();
|
|
services.AddUnique<IEFCoreScopeAccessor<T>, EFCoreScopeAccessor<T>>();
|
|
services.AddUnique<IEFCoreScopeProvider<T>, EFCoreScopeProvider<T>>();
|
|
services.AddSingleton<IDistributedLockingMechanism, SqliteEFCoreDistributedLockingMechanism<T>>();
|
|
services.AddSingleton<IDistributedLockingMechanism, SqlServerEFCoreDistributedLockingMechanism<T>>();
|
|
|
|
return services;
|
|
}
|
|
|
|
/// <summary>
|
|
/// Sets the database provider. I.E UseSqlite or UseSqlServer based on the provider name.
|
|
/// </summary>
|
|
/// <param name="builder"></param>
|
|
/// <param name="providerName"></param>
|
|
/// <param name="connectionString"></param>
|
|
/// <exception cref="InvalidDataException"></exception>
|
|
/// <remarks>
|
|
/// Only supports the databases normally supported in Umbraco.
|
|
/// </remarks>
|
|
public static void UseDatabaseProvider(this DbContextOptionsBuilder builder, string providerName, string connectionString)
|
|
{
|
|
switch (providerName)
|
|
{
|
|
case Constants.ProviderNames.SQLServer:
|
|
builder.UseSqlServer(connectionString);
|
|
break;
|
|
case Constants.ProviderNames.SQLLite:
|
|
case "Microsoft.Data.SQLite":
|
|
builder.UseSqlite(connectionString);
|
|
break;
|
|
default:
|
|
throw new InvalidDataException($"The provider {providerName} is not supported. Manually add the add the UseXXX statement to the options. I.E UseNpgsql()");
|
|
}
|
|
}
|
|
|
|
/// <summary>
|
|
/// Sets the database provider to use based on the Umbraco connection string.
|
|
/// </summary>
|
|
/// <param name="builder"></param>
|
|
/// <param name="serviceProvider"></param>
|
|
public static void UseUmbracoDatabaseProvider(this DbContextOptionsBuilder builder, IServiceProvider serviceProvider)
|
|
{
|
|
ConnectionStrings connectionStrings = serviceProvider.GetRequiredService<IOptionsMonitor<ConnectionStrings>>().CurrentValue;
|
|
|
|
// Replace data directory
|
|
string? dataDirectory = AppDomain.CurrentDomain.GetData(Constants.System.DataDirectoryName)?.ToString();
|
|
if (string.IsNullOrEmpty(dataDirectory) is false)
|
|
{
|
|
connectionStrings.ConnectionString = connectionStrings.ConnectionString?.Replace(Constants.System.DataDirectoryPlaceholder, dataDirectory);
|
|
}
|
|
|
|
if (string.IsNullOrEmpty(connectionStrings.ProviderName))
|
|
{
|
|
Log.Warning("No database provider was set. ProviderName is null");
|
|
return;
|
|
}
|
|
|
|
if (string.IsNullOrEmpty(connectionStrings.ConnectionString))
|
|
{
|
|
Log.Warning("No database provider was set. Connection string is null");
|
|
return;
|
|
}
|
|
|
|
builder.UseDatabaseProvider(connectionStrings.ProviderName, connectionStrings.ConnectionString);
|
|
}
|
|
|
|
[Obsolete]
|
|
private static void SetupDbContext(DefaultEFCoreOptionsAction? defaultEFCoreOptionsAction, IServiceProvider provider, DbContextOptionsBuilder builder)
|
|
{
|
|
ConnectionStrings connectionStrings = GetConnectionStringAndProviderName(provider);
|
|
defaultEFCoreOptionsAction?.Invoke(builder, connectionStrings.ConnectionString, connectionStrings.ProviderName);
|
|
}
|
|
|
|
[Obsolete]
|
|
private static ConnectionStrings GetConnectionStringAndProviderName(IServiceProvider serviceProvider)
|
|
{
|
|
ConnectionStrings connectionStrings = serviceProvider.GetRequiredService<IOptionsMonitor<ConnectionStrings>>().CurrentValue;
|
|
|
|
// Replace data directory
|
|
string? dataDirectory = AppDomain.CurrentDomain.GetData(Constants.System.DataDirectoryName)?.ToString();
|
|
if (string.IsNullOrEmpty(dataDirectory) is false)
|
|
{
|
|
connectionStrings.ConnectionString = connectionStrings.ConnectionString?.Replace(Constants.System.DataDirectoryPlaceholder, dataDirectory);
|
|
}
|
|
|
|
return connectionStrings;
|
|
}
|
|
}
|