From 6c075698cf4d3038b7016ee264bbd9bcba7404be Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Kriszti=C3=A1n=20N=C3=A9meth?= Date: Thu, 29 Aug 2024 16:38:51 +0200 Subject: [PATCH 01/15] Add elasticsearch delete maintenance and middleware --- .../Constants/FeatureNames.cs | 1 + .../Lombiq.Hosting.Tenants.Maintenance.csproj | 1 + ...eElasticsearchIndexesMaintenanceOptions.cs | 6 ++ ...ElasticsearchIndexesMaintenanceProvider.cs | 40 +++++++++ .../DeleteElasticsearchIndexesMiddleware.cs | 89 +++++++++++++++++++ ...teElasticsearchIndexesMiddlewareOptions.cs | 6 ++ .../DeleteElasticsearchIndexes/Startup.cs | 46 ++++++++++ .../Manifest.cs | 9 ++ 8 files changed, 198 insertions(+) create mode 100644 Lombiq.Hosting.Tenants.Maintenance/Maintenance/DeleteElasticsearchIndexes/DeleteElasticsearchIndexesMaintenanceOptions.cs create mode 100644 Lombiq.Hosting.Tenants.Maintenance/Maintenance/DeleteElasticsearchIndexes/DeleteElasticsearchIndexesMaintenanceProvider.cs create mode 100644 Lombiq.Hosting.Tenants.Maintenance/Maintenance/DeleteElasticsearchIndexes/DeleteElasticsearchIndexesMiddleware.cs create mode 100644 Lombiq.Hosting.Tenants.Maintenance/Maintenance/DeleteElasticsearchIndexes/DeleteElasticsearchIndexesMiddlewareOptions.cs create mode 100644 Lombiq.Hosting.Tenants.Maintenance/Maintenance/DeleteElasticsearchIndexes/Startup.cs diff --git a/Lombiq.Hosting.Tenants.Maintenance/Constants/FeatureNames.cs b/Lombiq.Hosting.Tenants.Maintenance/Constants/FeatureNames.cs index 3f6f508b..4862dafa 100644 --- a/Lombiq.Hosting.Tenants.Maintenance/Constants/FeatureNames.cs +++ b/Lombiq.Hosting.Tenants.Maintenance/Constants/FeatureNames.cs @@ -9,4 +9,5 @@ public static class FeatureNames public const string AddSiteOwnerPermissionToRole = Maintenance + "." + nameof(AddSiteOwnerPermissionToRole); public const string RemoveUsers = Maintenance + "." + nameof(RemoveUsers); public const string ChangeUserSensitiveContent = Maintenance + "." + nameof(ChangeUserSensitiveContent); + public const string DeleteElasticsearchIndexes = Maintenance + "." + nameof(DeleteElasticsearchIndexes); } diff --git a/Lombiq.Hosting.Tenants.Maintenance/Lombiq.Hosting.Tenants.Maintenance.csproj b/Lombiq.Hosting.Tenants.Maintenance/Lombiq.Hosting.Tenants.Maintenance.csproj index 8a10ff65..f57ca18b 100644 --- a/Lombiq.Hosting.Tenants.Maintenance/Lombiq.Hosting.Tenants.Maintenance.csproj +++ b/Lombiq.Hosting.Tenants.Maintenance/Lombiq.Hosting.Tenants.Maintenance.csproj @@ -27,6 +27,7 @@ + diff --git a/Lombiq.Hosting.Tenants.Maintenance/Maintenance/DeleteElasticsearchIndexes/DeleteElasticsearchIndexesMaintenanceOptions.cs b/Lombiq.Hosting.Tenants.Maintenance/Maintenance/DeleteElasticsearchIndexes/DeleteElasticsearchIndexesMaintenanceOptions.cs new file mode 100644 index 00000000..2ee0217d --- /dev/null +++ b/Lombiq.Hosting.Tenants.Maintenance/Maintenance/DeleteElasticsearchIndexes/DeleteElasticsearchIndexesMaintenanceOptions.cs @@ -0,0 +1,6 @@ +namespace Lombiq.Hosting.Tenants.Maintenance.Maintenance.DeleteElasticsearchIndexes; + +public class DeleteElasticsearchIndexesMaintenanceOptions +{ + public bool IsEnabled { get; set; } +} diff --git a/Lombiq.Hosting.Tenants.Maintenance/Maintenance/DeleteElasticsearchIndexes/DeleteElasticsearchIndexesMaintenanceProvider.cs b/Lombiq.Hosting.Tenants.Maintenance/Maintenance/DeleteElasticsearchIndexes/DeleteElasticsearchIndexesMaintenanceProvider.cs new file mode 100644 index 00000000..00178c39 --- /dev/null +++ b/Lombiq.Hosting.Tenants.Maintenance/Maintenance/DeleteElasticsearchIndexes/DeleteElasticsearchIndexesMaintenanceProvider.cs @@ -0,0 +1,40 @@ +using Lombiq.Hosting.Tenants.Maintenance.Extensions; +using Lombiq.Hosting.Tenants.Maintenance.Models; +using Lombiq.Hosting.Tenants.Maintenance.Services; +using Microsoft.Extensions.Options; +using OrchardCore.Search.Elasticsearch.Core.Services; +using System.Linq; +using System.Threading.Tasks; + +namespace Lombiq.Hosting.Tenants.Maintenance.Maintenance.DeleteElasticsearchIndexes; + +public class DeleteElasticsearchIndexesMaintenanceProvider : MaintenanceProviderBase +{ + private readonly IOptions _options; + private readonly ElasticIndexManager _elasticIndexManager; + private readonly ElasticIndexSettingsService _elasticIndexSettingsService; + + public DeleteElasticsearchIndexesMaintenanceProvider( + IOptions options, + ElasticIndexManager elasticIndexManager, + ElasticIndexSettingsService elasticIndexSettingsService) + { + _options = options; + _elasticIndexManager = elasticIndexManager; + _elasticIndexSettingsService = elasticIndexSettingsService; + } + + public override Task ShouldExecuteAsync(MaintenanceTaskExecutionContext context) => + Task.FromResult( + _options.Value.IsEnabled && + !context.WasLatestExecutionSuccessful()); + + public override async Task ExecuteAsync(MaintenanceTaskExecutionContext context) + { + var indexes = (await _elasticIndexSettingsService.GetSettingsAsync()) + .Select(index => index.IndexName) + .ToList(); + + await _elasticIndexManager.DeleteIndex("*"); + } +} diff --git a/Lombiq.Hosting.Tenants.Maintenance/Maintenance/DeleteElasticsearchIndexes/DeleteElasticsearchIndexesMiddleware.cs b/Lombiq.Hosting.Tenants.Maintenance/Maintenance/DeleteElasticsearchIndexes/DeleteElasticsearchIndexesMiddleware.cs new file mode 100644 index 00000000..45e624db --- /dev/null +++ b/Lombiq.Hosting.Tenants.Maintenance/Maintenance/DeleteElasticsearchIndexes/DeleteElasticsearchIndexesMiddleware.cs @@ -0,0 +1,89 @@ +using Microsoft.AspNetCore.Http; +using Microsoft.Extensions.DependencyInjection; +using OrchardCore.Environment.Shell; +using OrchardCore.Locking.Distributed; +using OrchardCore.Search.Elasticsearch.Core.Services; +using System; +using System.Threading.Tasks; + +namespace Lombiq.Hosting.Tenants.Maintenance.Maintenance.DeleteElasticsearchIndexes; + +public class DeleteElasticsearchIndexesMiddleware +{ + private readonly RequestDelegate _next; + + private readonly IShellHost _shellHost; + + private readonly ShellSettings _shellSettings; + + private readonly IShellSettingsManager _shellSettingsManager; + + private readonly IDistributedLock _distributedLock; + + public DeleteElasticsearchIndexesMiddleware( + RequestDelegate next, + IShellHost shellHost, + ShellSettings shellSettings, + IShellSettingsManager shellSettingsManager, + IDistributedLock distributedLock) + { + _next = next; + _shellHost = shellHost; + _shellSettings = shellSettings; + _shellSettingsManager = shellSettingsManager; + _distributedLock = distributedLock; + } + + public async Task InvokeAsync(HttpContext httpContext) + { + if (!_shellSettings.IsUninitialized()) + { + await _next.Invoke(httpContext); + return; + } + + // Try to acquire a lock before starting installation + var (locker, locked) = await _distributedLock.TryAcquireLockAsync( + "ELASTICSERACH_INDICES_DELETION_LOCK", + TimeSpan.FromMinutes(1), + TimeSpan.FromMinutes(1)); + + if (!locked) + { + throw new TimeoutException($"Fails to acquire an elasticsearch indices deletion lock for the tenant: {_shellSettings.Name}"); + } + + await using var acquiredLock = locker; + + // Check if the tenant was installed by another instance. + if (!_shellSettings.IsUninitialized()) + { + await _next.Invoke(httpContext); + return; + } + + var pathBase = httpContext.Request.PathBase; + if (!pathBase.HasValue) + { + pathBase = "/"; + } + + using var settings = (await _shellSettingsManager + .LoadSettingsAsync(_shellSettings.Name)) + .AsDisposable(); + + // If the tenant was initialized by another instance, reload the shell context and redirect to the path base. + if (!settings.IsUninitialized()) + { + await _shellHost.ReloadShellContextAsync(_shellSettings, eventSource: false); + httpContext.Response.Redirect(pathBase); + + return; + } + + var elasticIndexManager = httpContext.RequestServices.GetRequiredService(); + await elasticIndexManager.DeleteIndex("*"); + + await _next.Invoke(httpContext); + } +} diff --git a/Lombiq.Hosting.Tenants.Maintenance/Maintenance/DeleteElasticsearchIndexes/DeleteElasticsearchIndexesMiddlewareOptions.cs b/Lombiq.Hosting.Tenants.Maintenance/Maintenance/DeleteElasticsearchIndexes/DeleteElasticsearchIndexesMiddlewareOptions.cs new file mode 100644 index 00000000..4dd32b1b --- /dev/null +++ b/Lombiq.Hosting.Tenants.Maintenance/Maintenance/DeleteElasticsearchIndexes/DeleteElasticsearchIndexesMiddlewareOptions.cs @@ -0,0 +1,6 @@ +namespace Lombiq.Hosting.Tenants.Maintenance.Maintenance.DeleteElasticsearchIndexes; + +public class DeleteElasticsearchIndexesMiddlewareOptions +{ + public bool IsEnabled { get; set; } +} diff --git a/Lombiq.Hosting.Tenants.Maintenance/Maintenance/DeleteElasticsearchIndexes/Startup.cs b/Lombiq.Hosting.Tenants.Maintenance/Maintenance/DeleteElasticsearchIndexes/Startup.cs new file mode 100644 index 00000000..ca2c09bc --- /dev/null +++ b/Lombiq.Hosting.Tenants.Maintenance/Maintenance/DeleteElasticsearchIndexes/Startup.cs @@ -0,0 +1,46 @@ +using Lombiq.HelpfulLibraries.OrchardCore.Mvc; +using Lombiq.Hosting.Tenants.Maintenance.Constants; +using Lombiq.Hosting.Tenants.Maintenance.Services; +using Microsoft.AspNetCore.Builder; +using Microsoft.AspNetCore.Routing; +using Microsoft.Extensions.DependencyInjection; +using Microsoft.Extensions.Options; +using OrchardCore.Environment.Shell; +using OrchardCore.Environment.Shell.Configuration; +using OrchardCore.Modules; +using System; + +namespace Lombiq.Hosting.Tenants.Maintenance.Maintenance.DeleteElasticsearchIndexes; + +[Feature(FeatureNames.DeleteElasticsearchIndexes)] +public class Startup : StartupBase +{ + private readonly IShellConfiguration _shellConfiguration; + private readonly ShellSettings _shellSettings; + + public Startup(IShellConfiguration shellConfiguration, ShellSettings shellSettings) + { + _shellConfiguration = shellConfiguration; + _shellSettings = shellSettings; + } + + public override void ConfigureServices(IServiceCollection services) + { + services.BindAndConfigureSection( + _shellConfiguration, + "Lombiq_Hosting_Tenants_Maintenance:DeleteElasticsearchIndexes"); + + services.AddScoped(); + } + + public override void Configure(IApplicationBuilder app, IEndpointRouteBuilder routes, IServiceProvider serviceProvider) + { + if (!_shellSettings.IsUninitialized()) return; + + var options = serviceProvider.GetRequiredService>().Value; + if (options.IsEnabled) + { + app.UseMiddleware(); + } + } +} diff --git a/Lombiq.Hosting.Tenants.Maintenance/Manifest.cs b/Lombiq.Hosting.Tenants.Maintenance/Manifest.cs index 6883a7bb..3d7991f5 100644 --- a/Lombiq.Hosting.Tenants.Maintenance/Manifest.cs +++ b/Lombiq.Hosting.Tenants.Maintenance/Manifest.cs @@ -60,3 +60,12 @@ DefaultTenantOnly = true, Dependencies = [Maintenance] )] + +[assembly: Feature( + Id = DeleteElasticsearchIndexes, + Name = "Lombiq Hosting - Tenants Maintenance Delete Elasticsearch Indexes", + Description = "Deletes Elasticsearch indexes.", + Category = "Maintenance", + DefaultTenantOnly = true, + Dependencies = [Maintenance, "OrchardCore.Search.Elasticsearch"] +)] From 0ed5c7db0bba0e4fc0712a380079764fd967d8c7 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Kriszti=C3=A1n=20N=C3=A9meth?= Date: Thu, 29 Aug 2024 16:39:00 +0200 Subject: [PATCH 02/15] Use extension --- .../AddSiteOwnerPermissionToRole/Startup.cs | 9 ++++----- .../Maintenance/ChangeUserSensitiveContent/Startup.cs | 10 ++++------ .../Maintenance/RemoveUsers/Startup.cs | 9 ++++----- .../Maintenance/UpdateShellRequestUrl/Startup.cs | 9 ++++----- .../Maintenance/UpdateSiteUrl/Startup.cs | 9 ++++----- 5 files changed, 20 insertions(+), 26 deletions(-) diff --git a/Lombiq.Hosting.Tenants.Maintenance/Maintenance/AddSiteOwnerPermissionToRole/Startup.cs b/Lombiq.Hosting.Tenants.Maintenance/Maintenance/AddSiteOwnerPermissionToRole/Startup.cs index 4722717d..5619ae37 100644 --- a/Lombiq.Hosting.Tenants.Maintenance/Maintenance/AddSiteOwnerPermissionToRole/Startup.cs +++ b/Lombiq.Hosting.Tenants.Maintenance/Maintenance/AddSiteOwnerPermissionToRole/Startup.cs @@ -1,6 +1,6 @@ +using Lombiq.HelpfulLibraries.OrchardCore.Mvc; using Lombiq.Hosting.Tenants.Maintenance.Constants; using Lombiq.Hosting.Tenants.Maintenance.Services; -using Microsoft.Extensions.Configuration; using Microsoft.Extensions.DependencyInjection; using OrchardCore.Environment.Shell.Configuration; using OrchardCore.Modules; @@ -17,10 +17,9 @@ public Startup(IShellConfiguration shellConfiguration) => public override void ConfigureServices(IServiceCollection services) { - var options = new AddSiteOwnerPermissionToRoleMaintenanceOptions(); - var configSection = _shellConfiguration.GetSection("Lombiq_Hosting_Tenants_Maintenance:AddSiteOwnerPermissionToRole"); - configSection.Bind(options); - services.Configure(configSection); + services.BindAndConfigureSection( + _shellConfiguration, + "Lombiq_Hosting_Tenants_Maintenance:AddSiteOwnerPermissionToRole"); services.AddScoped(); } diff --git a/Lombiq.Hosting.Tenants.Maintenance/Maintenance/ChangeUserSensitiveContent/Startup.cs b/Lombiq.Hosting.Tenants.Maintenance/Maintenance/ChangeUserSensitiveContent/Startup.cs index 23e7e97e..95b46fe2 100644 --- a/Lombiq.Hosting.Tenants.Maintenance/Maintenance/ChangeUserSensitiveContent/Startup.cs +++ b/Lombiq.Hosting.Tenants.Maintenance/Maintenance/ChangeUserSensitiveContent/Startup.cs @@ -1,6 +1,6 @@ +using Lombiq.HelpfulLibraries.OrchardCore.Mvc; using Lombiq.Hosting.Tenants.Maintenance.Constants; using Lombiq.Hosting.Tenants.Maintenance.Services; -using Microsoft.Extensions.Configuration; using Microsoft.Extensions.DependencyInjection; using OrchardCore.Environment.Shell.Configuration; using OrchardCore.Modules; @@ -17,11 +17,9 @@ public Startup(IShellConfiguration shellConfiguration) => public override void ConfigureServices(IServiceCollection services) { - var options = new ChangeUserSensitiveContentMaintenanceOptions(); - var configSection = _shellConfiguration - .GetSection("Lombiq_Hosting_Tenants_Maintenance:ChangeUserSensitiveContent"); - configSection.Bind(options); - services.Configure(configSection); + services.BindAndConfigureSection( + _shellConfiguration, + "Lombiq_Hosting_Tenants_Maintenance:ChangeUserSensitiveContent"); services.AddScoped(); } diff --git a/Lombiq.Hosting.Tenants.Maintenance/Maintenance/RemoveUsers/Startup.cs b/Lombiq.Hosting.Tenants.Maintenance/Maintenance/RemoveUsers/Startup.cs index 58fa4760..a73e4781 100644 --- a/Lombiq.Hosting.Tenants.Maintenance/Maintenance/RemoveUsers/Startup.cs +++ b/Lombiq.Hosting.Tenants.Maintenance/Maintenance/RemoveUsers/Startup.cs @@ -1,6 +1,6 @@ +using Lombiq.HelpfulLibraries.OrchardCore.Mvc; using Lombiq.Hosting.Tenants.Maintenance.Constants; using Lombiq.Hosting.Tenants.Maintenance.Services; -using Microsoft.Extensions.Configuration; using Microsoft.Extensions.DependencyInjection; using OrchardCore.Environment.Shell.Configuration; using OrchardCore.Modules; @@ -17,10 +17,9 @@ public Startup(IShellConfiguration shellConfiguration) => public override void ConfigureServices(IServiceCollection services) { - var options = new RemoveUsersMaintenanceOptions(); - var configSection = _shellConfiguration.GetSection("Lombiq_Hosting_Tenants_Maintenance:RemoveUsers"); - configSection.Bind(options); - services.Configure(configSection); + services.BindAndConfigureSection( + _shellConfiguration, + "Lombiq_Hosting_Tenants_Maintenance:RemoveUsers"); services.AddScoped(); } diff --git a/Lombiq.Hosting.Tenants.Maintenance/Maintenance/UpdateShellRequestUrl/Startup.cs b/Lombiq.Hosting.Tenants.Maintenance/Maintenance/UpdateShellRequestUrl/Startup.cs index 2b21fa1e..b303b0d9 100644 --- a/Lombiq.Hosting.Tenants.Maintenance/Maintenance/UpdateShellRequestUrl/Startup.cs +++ b/Lombiq.Hosting.Tenants.Maintenance/Maintenance/UpdateShellRequestUrl/Startup.cs @@ -1,6 +1,6 @@ +using Lombiq.HelpfulLibraries.OrchardCore.Mvc; using Lombiq.Hosting.Tenants.Maintenance.Constants; using Lombiq.Hosting.Tenants.Maintenance.Services; -using Microsoft.Extensions.Configuration; using Microsoft.Extensions.DependencyInjection; using OrchardCore.Environment.Shell.Configuration; using OrchardCore.Modules; @@ -17,10 +17,9 @@ public Startup(IShellConfiguration shellConfiguration) => public override void ConfigureServices(IServiceCollection services) { - var options = new UpdateShellRequestUrlMaintenanceOptions(); - var configSection = _shellConfiguration.GetSection("Lombiq_Hosting_Tenants_Maintenance:UpdateShellRequestUrl"); - configSection.Bind(options); - services.Configure(configSection); + services.BindAndConfigureSection( + _shellConfiguration, + "Lombiq_Hosting_Tenants_Maintenance:UpdateShellRequestUrl"); services.AddScoped(); } diff --git a/Lombiq.Hosting.Tenants.Maintenance/Maintenance/UpdateSiteUrl/Startup.cs b/Lombiq.Hosting.Tenants.Maintenance/Maintenance/UpdateSiteUrl/Startup.cs index ff86c21c..aeba6fa5 100644 --- a/Lombiq.Hosting.Tenants.Maintenance/Maintenance/UpdateSiteUrl/Startup.cs +++ b/Lombiq.Hosting.Tenants.Maintenance/Maintenance/UpdateSiteUrl/Startup.cs @@ -1,6 +1,6 @@ +using Lombiq.HelpfulLibraries.OrchardCore.Mvc; using Lombiq.Hosting.Tenants.Maintenance.Constants; using Lombiq.Hosting.Tenants.Maintenance.Services; -using Microsoft.Extensions.Configuration; using Microsoft.Extensions.DependencyInjection; using OrchardCore.Environment.Shell.Configuration; using OrchardCore.Modules; @@ -17,10 +17,9 @@ public Startup(IShellConfiguration shellConfiguration) => public override void ConfigureServices(IServiceCollection services) { - var options = new UpdateSiteUrlMaintenanceOptions(); - var configSection = _shellConfiguration.GetSection("Lombiq_Hosting_Tenants_Maintenance:UpdateSiteUrl"); - configSection.Bind(options); - services.Configure(configSection); + services.BindAndConfigureSection( + _shellConfiguration, + "Lombiq_Hosting_Tenants_Maintenance:UpdateSiteUrl"); services.AddScoped(); } From d7e3f224a96c929a0fbafac346a1bc06611769cf Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Kriszti=C3=A1n=20N=C3=A9meth?= Date: Thu, 29 Aug 2024 17:00:34 +0200 Subject: [PATCH 03/15] Renaming --- .../Constants/FeatureNames.cs | 2 +- ...ElasticsearchIndexesMaintenanceProvider.cs | 40 -------------- ...ElasticsearchIndicesMaintenanceOptions.cs} | 4 +- ...ElasticsearchIndicesMaintenanceProvider.cs | 31 +++++++++++ .../DeleteElasticsearchIndicesMiddleware.cs} | 55 +++++++------------ ...eElasticsearchIndicesMiddlewareOptions.cs} | 4 +- .../Startup.cs | 14 ++--- .../Manifest.cs | 2 +- 8 files changed, 65 insertions(+), 87 deletions(-) delete mode 100644 Lombiq.Hosting.Tenants.Maintenance/Maintenance/DeleteElasticsearchIndexes/DeleteElasticsearchIndexesMaintenanceProvider.cs rename Lombiq.Hosting.Tenants.Maintenance/Maintenance/{DeleteElasticsearchIndexes/DeleteElasticsearchIndexesMaintenanceOptions.cs => DeleteElasticsearchIndices/DeleteElasticsearchIndicesMaintenanceOptions.cs} (57%) create mode 100644 Lombiq.Hosting.Tenants.Maintenance/Maintenance/DeleteElasticsearchIndices/DeleteElasticsearchIndicesMaintenanceProvider.cs rename Lombiq.Hosting.Tenants.Maintenance/Maintenance/{DeleteElasticsearchIndexes/DeleteElasticsearchIndexesMiddleware.cs => DeleteElasticsearchIndices/DeleteElasticsearchIndicesMiddleware.cs} (61%) rename Lombiq.Hosting.Tenants.Maintenance/Maintenance/{DeleteElasticsearchIndexes/DeleteElasticsearchIndexesMiddlewareOptions.cs => DeleteElasticsearchIndices/DeleteElasticsearchIndicesMiddlewareOptions.cs} (58%) rename Lombiq.Hosting.Tenants.Maintenance/Maintenance/{DeleteElasticsearchIndexes => DeleteElasticsearchIndices}/Startup.cs (80%) diff --git a/Lombiq.Hosting.Tenants.Maintenance/Constants/FeatureNames.cs b/Lombiq.Hosting.Tenants.Maintenance/Constants/FeatureNames.cs index 4862dafa..c37c77f0 100644 --- a/Lombiq.Hosting.Tenants.Maintenance/Constants/FeatureNames.cs +++ b/Lombiq.Hosting.Tenants.Maintenance/Constants/FeatureNames.cs @@ -9,5 +9,5 @@ public static class FeatureNames public const string AddSiteOwnerPermissionToRole = Maintenance + "." + nameof(AddSiteOwnerPermissionToRole); public const string RemoveUsers = Maintenance + "." + nameof(RemoveUsers); public const string ChangeUserSensitiveContent = Maintenance + "." + nameof(ChangeUserSensitiveContent); - public const string DeleteElasticsearchIndexes = Maintenance + "." + nameof(DeleteElasticsearchIndexes); + public const string DeleteElasticsearchIndices = Maintenance + "." + nameof(DeleteElasticsearchIndices); } diff --git a/Lombiq.Hosting.Tenants.Maintenance/Maintenance/DeleteElasticsearchIndexes/DeleteElasticsearchIndexesMaintenanceProvider.cs b/Lombiq.Hosting.Tenants.Maintenance/Maintenance/DeleteElasticsearchIndexes/DeleteElasticsearchIndexesMaintenanceProvider.cs deleted file mode 100644 index 00178c39..00000000 --- a/Lombiq.Hosting.Tenants.Maintenance/Maintenance/DeleteElasticsearchIndexes/DeleteElasticsearchIndexesMaintenanceProvider.cs +++ /dev/null @@ -1,40 +0,0 @@ -using Lombiq.Hosting.Tenants.Maintenance.Extensions; -using Lombiq.Hosting.Tenants.Maintenance.Models; -using Lombiq.Hosting.Tenants.Maintenance.Services; -using Microsoft.Extensions.Options; -using OrchardCore.Search.Elasticsearch.Core.Services; -using System.Linq; -using System.Threading.Tasks; - -namespace Lombiq.Hosting.Tenants.Maintenance.Maintenance.DeleteElasticsearchIndexes; - -public class DeleteElasticsearchIndexesMaintenanceProvider : MaintenanceProviderBase -{ - private readonly IOptions _options; - private readonly ElasticIndexManager _elasticIndexManager; - private readonly ElasticIndexSettingsService _elasticIndexSettingsService; - - public DeleteElasticsearchIndexesMaintenanceProvider( - IOptions options, - ElasticIndexManager elasticIndexManager, - ElasticIndexSettingsService elasticIndexSettingsService) - { - _options = options; - _elasticIndexManager = elasticIndexManager; - _elasticIndexSettingsService = elasticIndexSettingsService; - } - - public override Task ShouldExecuteAsync(MaintenanceTaskExecutionContext context) => - Task.FromResult( - _options.Value.IsEnabled && - !context.WasLatestExecutionSuccessful()); - - public override async Task ExecuteAsync(MaintenanceTaskExecutionContext context) - { - var indexes = (await _elasticIndexSettingsService.GetSettingsAsync()) - .Select(index => index.IndexName) - .ToList(); - - await _elasticIndexManager.DeleteIndex("*"); - } -} diff --git a/Lombiq.Hosting.Tenants.Maintenance/Maintenance/DeleteElasticsearchIndexes/DeleteElasticsearchIndexesMaintenanceOptions.cs b/Lombiq.Hosting.Tenants.Maintenance/Maintenance/DeleteElasticsearchIndices/DeleteElasticsearchIndicesMaintenanceOptions.cs similarity index 57% rename from Lombiq.Hosting.Tenants.Maintenance/Maintenance/DeleteElasticsearchIndexes/DeleteElasticsearchIndexesMaintenanceOptions.cs rename to Lombiq.Hosting.Tenants.Maintenance/Maintenance/DeleteElasticsearchIndices/DeleteElasticsearchIndicesMaintenanceOptions.cs index 2ee0217d..c230a7df 100644 --- a/Lombiq.Hosting.Tenants.Maintenance/Maintenance/DeleteElasticsearchIndexes/DeleteElasticsearchIndexesMaintenanceOptions.cs +++ b/Lombiq.Hosting.Tenants.Maintenance/Maintenance/DeleteElasticsearchIndices/DeleteElasticsearchIndicesMaintenanceOptions.cs @@ -1,6 +1,6 @@ -namespace Lombiq.Hosting.Tenants.Maintenance.Maintenance.DeleteElasticsearchIndexes; +namespace Lombiq.Hosting.Tenants.Maintenance.Maintenance.DeleteElasticsearchIndices; -public class DeleteElasticsearchIndexesMaintenanceOptions +public class DeleteElasticsearchIndicesMaintenanceOptions { public bool IsEnabled { get; set; } } diff --git a/Lombiq.Hosting.Tenants.Maintenance/Maintenance/DeleteElasticsearchIndices/DeleteElasticsearchIndicesMaintenanceProvider.cs b/Lombiq.Hosting.Tenants.Maintenance/Maintenance/DeleteElasticsearchIndices/DeleteElasticsearchIndicesMaintenanceProvider.cs new file mode 100644 index 00000000..1c10e3f0 --- /dev/null +++ b/Lombiq.Hosting.Tenants.Maintenance/Maintenance/DeleteElasticsearchIndices/DeleteElasticsearchIndicesMaintenanceProvider.cs @@ -0,0 +1,31 @@ +using Lombiq.Hosting.Tenants.Maintenance.Extensions; +using Lombiq.Hosting.Tenants.Maintenance.Models; +using Lombiq.Hosting.Tenants.Maintenance.Services; +using Microsoft.Extensions.Options; +using OrchardCore.Search.Elasticsearch.Core.Services; +using System.Threading.Tasks; + +namespace Lombiq.Hosting.Tenants.Maintenance.Maintenance.DeleteElasticsearchIndices; + +public class DeleteElasticsearchIndicesMaintenanceProvider : MaintenanceProviderBase +{ + private readonly IOptions _options; + private readonly ElasticIndexManager _elasticIndexManager; + + public DeleteElasticsearchIndicesMaintenanceProvider( + IOptions options, + ElasticIndexManager elasticIndexManager) + { + _options = options; + _elasticIndexManager = elasticIndexManager; + } + + public override Task ShouldExecuteAsync(MaintenanceTaskExecutionContext context) => + Task.FromResult( + _options.Value.IsEnabled && + !context.WasLatestExecutionSuccessful()); + + public override Task ExecuteAsync(MaintenanceTaskExecutionContext context) => + // Delete all tenant specific indexes in Elasticsearch. + _elasticIndexManager.DeleteIndex("*"); +} diff --git a/Lombiq.Hosting.Tenants.Maintenance/Maintenance/DeleteElasticsearchIndexes/DeleteElasticsearchIndexesMiddleware.cs b/Lombiq.Hosting.Tenants.Maintenance/Maintenance/DeleteElasticsearchIndices/DeleteElasticsearchIndicesMiddleware.cs similarity index 61% rename from Lombiq.Hosting.Tenants.Maintenance/Maintenance/DeleteElasticsearchIndexes/DeleteElasticsearchIndexesMiddleware.cs rename to Lombiq.Hosting.Tenants.Maintenance/Maintenance/DeleteElasticsearchIndices/DeleteElasticsearchIndicesMiddleware.cs index 45e624db..304031b0 100644 --- a/Lombiq.Hosting.Tenants.Maintenance/Maintenance/DeleteElasticsearchIndexes/DeleteElasticsearchIndexesMiddleware.cs +++ b/Lombiq.Hosting.Tenants.Maintenance/Maintenance/DeleteElasticsearchIndices/DeleteElasticsearchIndicesMiddleware.cs @@ -6,29 +6,25 @@ using System; using System.Threading.Tasks; -namespace Lombiq.Hosting.Tenants.Maintenance.Maintenance.DeleteElasticsearchIndexes; +namespace Lombiq.Hosting.Tenants.Maintenance.Maintenance.DeleteElasticsearchIndices; -public class DeleteElasticsearchIndexesMiddleware +public class DeleteElasticsearchIndicesMiddleware { private readonly RequestDelegate _next; - private readonly IShellHost _shellHost; - private readonly ShellSettings _shellSettings; private readonly IShellSettingsManager _shellSettingsManager; private readonly IDistributedLock _distributedLock; - public DeleteElasticsearchIndexesMiddleware( + public DeleteElasticsearchIndicesMiddleware( RequestDelegate next, - IShellHost shellHost, ShellSettings shellSettings, IShellSettingsManager shellSettingsManager, IDistributedLock distributedLock) { _next = next; - _shellHost = shellHost; _shellSettings = shellSettings; _shellSettingsManager = shellSettingsManager; _distributedLock = distributedLock; @@ -36,11 +32,7 @@ public DeleteElasticsearchIndexesMiddleware( public async Task InvokeAsync(HttpContext httpContext) { - if (!_shellSettings.IsUninitialized()) - { - await _next.Invoke(httpContext); - return; - } + if (await InvokeNextIfUninitializedAsync(_shellSettings, httpContext)) return; // Try to acquire a lock before starting installation var (locker, locked) = await _distributedLock.TryAcquireLockAsync( @@ -56,34 +48,29 @@ public async Task InvokeAsync(HttpContext httpContext) await using var acquiredLock = locker; // Check if the tenant was installed by another instance. - if (!_shellSettings.IsUninitialized()) - { - await _next.Invoke(httpContext); - return; - } + if (await InvokeNextIfUninitializedAsync(_shellSettings, httpContext)) return; - var pathBase = httpContext.Request.PathBase; - if (!pathBase.HasValue) - { - pathBase = "/"; - } + using var settings = (await _shellSettingsManager.LoadSettingsAsync(_shellSettings.Name)).AsDisposable(); - using var settings = (await _shellSettingsManager - .LoadSettingsAsync(_shellSettings.Name)) - .AsDisposable(); - - // If the tenant was initialized by another instance, reload the shell context and redirect to the path base. - if (!settings.IsUninitialized()) - { - await _shellHost.ReloadShellContextAsync(_shellSettings, eventSource: false); - httpContext.Response.Redirect(pathBase); - - return; - } + // If the tenant was initialized by another instance, then skip again. + if (await InvokeNextIfUninitializedAsync(settings, httpContext)) return; var elasticIndexManager = httpContext.RequestServices.GetRequiredService(); + + // Delete all tenant specific indexes in Elasticsearch. await elasticIndexManager.DeleteIndex("*"); await _next.Invoke(httpContext); } + + private async Task InvokeNextIfUninitializedAsync(ShellSettings shellSettings, HttpContext httpContext) + { + if (shellSettings.IsUninitialized()) + { + return false; + } + + await _next.Invoke(httpContext); + return true; + } } diff --git a/Lombiq.Hosting.Tenants.Maintenance/Maintenance/DeleteElasticsearchIndexes/DeleteElasticsearchIndexesMiddlewareOptions.cs b/Lombiq.Hosting.Tenants.Maintenance/Maintenance/DeleteElasticsearchIndices/DeleteElasticsearchIndicesMiddlewareOptions.cs similarity index 58% rename from Lombiq.Hosting.Tenants.Maintenance/Maintenance/DeleteElasticsearchIndexes/DeleteElasticsearchIndexesMiddlewareOptions.cs rename to Lombiq.Hosting.Tenants.Maintenance/Maintenance/DeleteElasticsearchIndices/DeleteElasticsearchIndicesMiddlewareOptions.cs index 4dd32b1b..19af09e5 100644 --- a/Lombiq.Hosting.Tenants.Maintenance/Maintenance/DeleteElasticsearchIndexes/DeleteElasticsearchIndexesMiddlewareOptions.cs +++ b/Lombiq.Hosting.Tenants.Maintenance/Maintenance/DeleteElasticsearchIndices/DeleteElasticsearchIndicesMiddlewareOptions.cs @@ -1,6 +1,6 @@ -namespace Lombiq.Hosting.Tenants.Maintenance.Maintenance.DeleteElasticsearchIndexes; +namespace Lombiq.Hosting.Tenants.Maintenance.Maintenance.DeleteElasticsearchIndices; -public class DeleteElasticsearchIndexesMiddlewareOptions +public class DeleteElasticsearchIndicesMiddlewareOptions { public bool IsEnabled { get; set; } } diff --git a/Lombiq.Hosting.Tenants.Maintenance/Maintenance/DeleteElasticsearchIndexes/Startup.cs b/Lombiq.Hosting.Tenants.Maintenance/Maintenance/DeleteElasticsearchIndices/Startup.cs similarity index 80% rename from Lombiq.Hosting.Tenants.Maintenance/Maintenance/DeleteElasticsearchIndexes/Startup.cs rename to Lombiq.Hosting.Tenants.Maintenance/Maintenance/DeleteElasticsearchIndices/Startup.cs index ca2c09bc..84374945 100644 --- a/Lombiq.Hosting.Tenants.Maintenance/Maintenance/DeleteElasticsearchIndexes/Startup.cs +++ b/Lombiq.Hosting.Tenants.Maintenance/Maintenance/DeleteElasticsearchIndices/Startup.cs @@ -10,9 +10,9 @@ using OrchardCore.Modules; using System; -namespace Lombiq.Hosting.Tenants.Maintenance.Maintenance.DeleteElasticsearchIndexes; +namespace Lombiq.Hosting.Tenants.Maintenance.Maintenance.DeleteElasticsearchIndices; -[Feature(FeatureNames.DeleteElasticsearchIndexes)] +[Feature(FeatureNames.DeleteElasticsearchIndices)] public class Startup : StartupBase { private readonly IShellConfiguration _shellConfiguration; @@ -26,21 +26,21 @@ public Startup(IShellConfiguration shellConfiguration, ShellSettings shellSettin public override void ConfigureServices(IServiceCollection services) { - services.BindAndConfigureSection( + services.BindAndConfigureSection( _shellConfiguration, - "Lombiq_Hosting_Tenants_Maintenance:DeleteElasticsearchIndexes"); + "Lombiq_Hosting_Tenants_Maintenance:DeleteElasticsearchIndices"); - services.AddScoped(); + services.AddScoped(); } public override void Configure(IApplicationBuilder app, IEndpointRouteBuilder routes, IServiceProvider serviceProvider) { if (!_shellSettings.IsUninitialized()) return; - var options = serviceProvider.GetRequiredService>().Value; + var options = serviceProvider.GetRequiredService>().Value; if (options.IsEnabled) { - app.UseMiddleware(); + app.UseMiddleware(); } } } diff --git a/Lombiq.Hosting.Tenants.Maintenance/Manifest.cs b/Lombiq.Hosting.Tenants.Maintenance/Manifest.cs index 3d7991f5..eb484e51 100644 --- a/Lombiq.Hosting.Tenants.Maintenance/Manifest.cs +++ b/Lombiq.Hosting.Tenants.Maintenance/Manifest.cs @@ -62,7 +62,7 @@ )] [assembly: Feature( - Id = DeleteElasticsearchIndexes, + Id = DeleteElasticsearchIndices, Name = "Lombiq Hosting - Tenants Maintenance Delete Elasticsearch Indexes", Description = "Deletes Elasticsearch indexes.", Category = "Maintenance", From 99c3b1c9c833e89201d5371a925769c88f133651 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Kriszti=C3=A1n=20N=C3=A9meth?= Date: Fri, 30 Aug 2024 14:04:27 +0200 Subject: [PATCH 04/15] Cleanup --- .../DeleteElasticsearchIndicesMaintenanceOptions.cs | 3 ++- .../DeleteElasticsearchIndicesMaintenanceProvider.cs | 2 +- .../DeleteElasticsearchIndicesMiddleware.cs | 2 +- .../DeleteElasticsearchIndicesMiddlewareOptions.cs | 6 ------ .../Maintenance/DeleteElasticsearchIndices/Startup.cs | 2 +- 5 files changed, 5 insertions(+), 10 deletions(-) delete mode 100644 Lombiq.Hosting.Tenants.Maintenance/Maintenance/DeleteElasticsearchIndices/DeleteElasticsearchIndicesMiddlewareOptions.cs diff --git a/Lombiq.Hosting.Tenants.Maintenance/Maintenance/DeleteElasticsearchIndices/DeleteElasticsearchIndicesMaintenanceOptions.cs b/Lombiq.Hosting.Tenants.Maintenance/Maintenance/DeleteElasticsearchIndices/DeleteElasticsearchIndicesMaintenanceOptions.cs index c230a7df..9bf9ae7c 100644 --- a/Lombiq.Hosting.Tenants.Maintenance/Maintenance/DeleteElasticsearchIndices/DeleteElasticsearchIndicesMaintenanceOptions.cs +++ b/Lombiq.Hosting.Tenants.Maintenance/Maintenance/DeleteElasticsearchIndices/DeleteElasticsearchIndicesMaintenanceOptions.cs @@ -2,5 +2,6 @@ namespace Lombiq.Hosting.Tenants.Maintenance.Maintenance.DeleteElasticsearchIndi public class DeleteElasticsearchIndicesMaintenanceOptions { - public bool IsEnabled { get; set; } + public bool MaintenanceIsEnabled { get; set; } + public bool MiddlewareIsEnabled { get; set; } } diff --git a/Lombiq.Hosting.Tenants.Maintenance/Maintenance/DeleteElasticsearchIndices/DeleteElasticsearchIndicesMaintenanceProvider.cs b/Lombiq.Hosting.Tenants.Maintenance/Maintenance/DeleteElasticsearchIndices/DeleteElasticsearchIndicesMaintenanceProvider.cs index 1c10e3f0..199888f0 100644 --- a/Lombiq.Hosting.Tenants.Maintenance/Maintenance/DeleteElasticsearchIndices/DeleteElasticsearchIndicesMaintenanceProvider.cs +++ b/Lombiq.Hosting.Tenants.Maintenance/Maintenance/DeleteElasticsearchIndices/DeleteElasticsearchIndicesMaintenanceProvider.cs @@ -22,7 +22,7 @@ public DeleteElasticsearchIndicesMaintenanceProvider( public override Task ShouldExecuteAsync(MaintenanceTaskExecutionContext context) => Task.FromResult( - _options.Value.IsEnabled && + _options.Value.MaintenanceIsEnabled && !context.WasLatestExecutionSuccessful()); public override Task ExecuteAsync(MaintenanceTaskExecutionContext context) => diff --git a/Lombiq.Hosting.Tenants.Maintenance/Maintenance/DeleteElasticsearchIndices/DeleteElasticsearchIndicesMiddleware.cs b/Lombiq.Hosting.Tenants.Maintenance/Maintenance/DeleteElasticsearchIndices/DeleteElasticsearchIndicesMiddleware.cs index 304031b0..90c1c774 100644 --- a/Lombiq.Hosting.Tenants.Maintenance/Maintenance/DeleteElasticsearchIndices/DeleteElasticsearchIndicesMiddleware.cs +++ b/Lombiq.Hosting.Tenants.Maintenance/Maintenance/DeleteElasticsearchIndices/DeleteElasticsearchIndicesMiddleware.cs @@ -36,7 +36,7 @@ public async Task InvokeAsync(HttpContext httpContext) // Try to acquire a lock before starting installation var (locker, locked) = await _distributedLock.TryAcquireLockAsync( - "ELASTICSERACH_INDICES_DELETION_LOCK", + "ELASTICSEARCH_INDICES_DELETION", TimeSpan.FromMinutes(1), TimeSpan.FromMinutes(1)); diff --git a/Lombiq.Hosting.Tenants.Maintenance/Maintenance/DeleteElasticsearchIndices/DeleteElasticsearchIndicesMiddlewareOptions.cs b/Lombiq.Hosting.Tenants.Maintenance/Maintenance/DeleteElasticsearchIndices/DeleteElasticsearchIndicesMiddlewareOptions.cs deleted file mode 100644 index 19af09e5..00000000 --- a/Lombiq.Hosting.Tenants.Maintenance/Maintenance/DeleteElasticsearchIndices/DeleteElasticsearchIndicesMiddlewareOptions.cs +++ /dev/null @@ -1,6 +0,0 @@ -namespace Lombiq.Hosting.Tenants.Maintenance.Maintenance.DeleteElasticsearchIndices; - -public class DeleteElasticsearchIndicesMiddlewareOptions -{ - public bool IsEnabled { get; set; } -} diff --git a/Lombiq.Hosting.Tenants.Maintenance/Maintenance/DeleteElasticsearchIndices/Startup.cs b/Lombiq.Hosting.Tenants.Maintenance/Maintenance/DeleteElasticsearchIndices/Startup.cs index 84374945..f436295e 100644 --- a/Lombiq.Hosting.Tenants.Maintenance/Maintenance/DeleteElasticsearchIndices/Startup.cs +++ b/Lombiq.Hosting.Tenants.Maintenance/Maintenance/DeleteElasticsearchIndices/Startup.cs @@ -38,7 +38,7 @@ public override void Configure(IApplicationBuilder app, IEndpointRouteBuilder ro if (!_shellSettings.IsUninitialized()) return; var options = serviceProvider.GetRequiredService>().Value; - if (options.IsEnabled) + if (options.MiddlewareIsEnabled) { app.UseMiddleware(); } From a3820368de78782e8db2e9471ad3446ead18186b Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Kriszti=C3=A1n=20N=C3=A9meth?= Date: Fri, 30 Aug 2024 14:06:23 +0200 Subject: [PATCH 05/15] Adding docs --- Lombiq.Hosting.Tenants.Maintenance/Readme.md | 21 ++++++++++++++++++++ 1 file changed, 21 insertions(+) diff --git a/Lombiq.Hosting.Tenants.Maintenance/Readme.md b/Lombiq.Hosting.Tenants.Maintenance/Readme.md index 07c0e8a7..5036981d 100644 --- a/Lombiq.Hosting.Tenants.Maintenance/Readme.md +++ b/Lombiq.Hosting.Tenants.Maintenance/Readme.md @@ -154,3 +154,24 @@ The following configuration should be used to allow the maintenance to run: ``` Any user accounts with an e-mail matching the `EmailExcludePattern` regex will not be depersonalized. + +### `Lombiq.Hosting.Tenants.Maintenance.DeleteElasitcsearchIndices` + +This contains a maintenance task that deletes all Elasticsearch indices related to the tenant that is being activated. + +It also contains a middleware that deletes all Elasticsearch indices related to the tenant, but it does that before the tenant setup. This is useful when you want to delete the indices before the tenant setup, so you can create indices from a recipe during setup. To be able to use the middleware before setup this feature must be added as a setup feature. You can do this with `OrchardCoreBuilder.AddSetupFeatures(Lombiq.Hosting.Tenants.Maintenance.Constants.FeatureNames.DeleteElasticsearchIndices);` + +The following configuration should be used to allow the maintenance to run and for the middleware to be added: + +```json +{ + "OrchardCore": { + "Lombiq_Hosting_Tenants_Maintenance": { + "DeleteElasticsearchIndices": { + "MaintenanceIsEnabled": true, + "MiddlewareIsEnabled": true + } + } + } +} +``` From 80cb103d93b2c5cb6873f17428d44910b028e52f Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Kriszti=C3=A1n=20N=C3=A9meth?= Date: Fri, 30 Aug 2024 14:23:55 +0200 Subject: [PATCH 06/15] Typo --- Lombiq.Hosting.Tenants.Maintenance/Readme.md | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/Lombiq.Hosting.Tenants.Maintenance/Readme.md b/Lombiq.Hosting.Tenants.Maintenance/Readme.md index 5036981d..2b789bd6 100644 --- a/Lombiq.Hosting.Tenants.Maintenance/Readme.md +++ b/Lombiq.Hosting.Tenants.Maintenance/Readme.md @@ -155,7 +155,7 @@ The following configuration should be used to allow the maintenance to run: Any user accounts with an e-mail matching the `EmailExcludePattern` regex will not be depersonalized. -### `Lombiq.Hosting.Tenants.Maintenance.DeleteElasitcsearchIndices` +### `Lombiq.Hosting.Tenants.Maintenance.DeleteElasticsearchIndices` This contains a maintenance task that deletes all Elasticsearch indices related to the tenant that is being activated. From 6b1c8d754e4d18acd4e38579c50eda3d55ee556c Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Kriszti=C3=A1n=20N=C3=A9meth?= Date: Thu, 5 Sep 2024 13:54:32 +0200 Subject: [PATCH 07/15] Modifing structure --- .../Constants/FeatureNames.cs | 3 +- ...eElasticsearchIndicesMaintenanceOptions.cs | 7 - ...ElasticsearchIndicesMaintenanceProvider.cs | 6 +- .../DeleteElasticsearchIndicesMiddleware.cs | 84 +++++++++++ .../ElasticsearchIndicesMaintenanceOptions.cs | 8 ++ ...ElasticsearchIndicesMaintenanceProvider.cs | 50 +++++++ .../DeleteElasticsearchIndices/Startup.cs | 132 ++++++++++++++++-- .../Manifest.cs | 13 +- Lombiq.Hosting.Tenants.Maintenance/Readme.md | 7 +- 9 files changed, 285 insertions(+), 25 deletions(-) delete mode 100644 Lombiq.Hosting.Tenants.Maintenance/Maintenance/DeleteElasticsearchIndices/DeleteElasticsearchIndicesMaintenanceOptions.cs create mode 100644 Lombiq.Hosting.Tenants.Maintenance/Maintenance/DeleteElasticsearchIndices/ElasticsearchIndicesMaintenanceOptions.cs create mode 100644 Lombiq.Hosting.Tenants.Maintenance/Maintenance/DeleteElasticsearchIndices/RebuildElasticsearchIndicesMaintenanceProvider.cs diff --git a/Lombiq.Hosting.Tenants.Maintenance/Constants/FeatureNames.cs b/Lombiq.Hosting.Tenants.Maintenance/Constants/FeatureNames.cs index c37c77f0..a62185d5 100644 --- a/Lombiq.Hosting.Tenants.Maintenance/Constants/FeatureNames.cs +++ b/Lombiq.Hosting.Tenants.Maintenance/Constants/FeatureNames.cs @@ -9,5 +9,6 @@ public static class FeatureNames public const string AddSiteOwnerPermissionToRole = Maintenance + "." + nameof(AddSiteOwnerPermissionToRole); public const string RemoveUsers = Maintenance + "." + nameof(RemoveUsers); public const string ChangeUserSensitiveContent = Maintenance + "." + nameof(ChangeUserSensitiveContent); - public const string DeleteElasticsearchIndices = Maintenance + "." + nameof(DeleteElasticsearchIndices); + public const string DeleteOrRebuildElasticsearchIndices = Maintenance + "." + nameof(DeleteOrRebuildElasticsearchIndices); + public const string DeleteElasticsearchIndicesBeforeSetup = Maintenance + "." + nameof(DeleteElasticsearchIndicesBeforeSetup); } diff --git a/Lombiq.Hosting.Tenants.Maintenance/Maintenance/DeleteElasticsearchIndices/DeleteElasticsearchIndicesMaintenanceOptions.cs b/Lombiq.Hosting.Tenants.Maintenance/Maintenance/DeleteElasticsearchIndices/DeleteElasticsearchIndicesMaintenanceOptions.cs deleted file mode 100644 index 9bf9ae7c..00000000 --- a/Lombiq.Hosting.Tenants.Maintenance/Maintenance/DeleteElasticsearchIndices/DeleteElasticsearchIndicesMaintenanceOptions.cs +++ /dev/null @@ -1,7 +0,0 @@ -namespace Lombiq.Hosting.Tenants.Maintenance.Maintenance.DeleteElasticsearchIndices; - -public class DeleteElasticsearchIndicesMaintenanceOptions -{ - public bool MaintenanceIsEnabled { get; set; } - public bool MiddlewareIsEnabled { get; set; } -} diff --git a/Lombiq.Hosting.Tenants.Maintenance/Maintenance/DeleteElasticsearchIndices/DeleteElasticsearchIndicesMaintenanceProvider.cs b/Lombiq.Hosting.Tenants.Maintenance/Maintenance/DeleteElasticsearchIndices/DeleteElasticsearchIndicesMaintenanceProvider.cs index 199888f0..9f654c3a 100644 --- a/Lombiq.Hosting.Tenants.Maintenance/Maintenance/DeleteElasticsearchIndices/DeleteElasticsearchIndicesMaintenanceProvider.cs +++ b/Lombiq.Hosting.Tenants.Maintenance/Maintenance/DeleteElasticsearchIndices/DeleteElasticsearchIndicesMaintenanceProvider.cs @@ -9,11 +9,11 @@ namespace Lombiq.Hosting.Tenants.Maintenance.Maintenance.DeleteElasticsearchIndi public class DeleteElasticsearchIndicesMaintenanceProvider : MaintenanceProviderBase { - private readonly IOptions _options; + private readonly IOptions _options; private readonly ElasticIndexManager _elasticIndexManager; public DeleteElasticsearchIndicesMaintenanceProvider( - IOptions options, + IOptions options, ElasticIndexManager elasticIndexManager) { _options = options; @@ -22,7 +22,7 @@ public DeleteElasticsearchIndicesMaintenanceProvider( public override Task ShouldExecuteAsync(MaintenanceTaskExecutionContext context) => Task.FromResult( - _options.Value.MaintenanceIsEnabled && + _options.Value.DeleteMaintenanceIsEnabled && !context.WasLatestExecutionSuccessful()); public override Task ExecuteAsync(MaintenanceTaskExecutionContext context) => diff --git a/Lombiq.Hosting.Tenants.Maintenance/Maintenance/DeleteElasticsearchIndices/DeleteElasticsearchIndicesMiddleware.cs b/Lombiq.Hosting.Tenants.Maintenance/Maintenance/DeleteElasticsearchIndices/DeleteElasticsearchIndicesMiddleware.cs index 90c1c774..1b746e7a 100644 --- a/Lombiq.Hosting.Tenants.Maintenance/Maintenance/DeleteElasticsearchIndices/DeleteElasticsearchIndicesMiddleware.cs +++ b/Lombiq.Hosting.Tenants.Maintenance/Maintenance/DeleteElasticsearchIndices/DeleteElasticsearchIndicesMiddleware.cs @@ -1,9 +1,14 @@ +using Elasticsearch.Net; using Microsoft.AspNetCore.Http; using Microsoft.Extensions.DependencyInjection; +using Nest; using OrchardCore.Environment.Shell; using OrchardCore.Locking.Distributed; +using OrchardCore.Search.Elasticsearch.Core.Models; using OrchardCore.Search.Elasticsearch.Core.Services; using System; +using System.Linq; +using System.Net; using System.Threading.Tasks; namespace Lombiq.Hosting.Tenants.Maintenance.Maintenance.DeleteElasticsearchIndices; @@ -32,6 +37,12 @@ public DeleteElasticsearchIndicesMiddleware( public async Task InvokeAsync(HttpContext httpContext) { + if (httpContext.Request.Method != WebRequestMethods.Http.Get) + { + await _next.Invoke(httpContext); + return; + } + if (await InvokeNextIfUninitializedAsync(_shellSettings, httpContext)) return; // Try to acquire a lock before starting installation @@ -73,4 +84,77 @@ private async Task InvokeNextIfUninitializedAsync(ShellSettings shellSetti await _next.Invoke(httpContext); return true; } + + private static ConnectionSettings GetConnectionSettings(ElasticConnectionOptions elasticConfiguration) + { + // This is a copy of the OC Elasticsearch module's OrchardCore.Search.Elasticsearch.Startup.GetConnectionSettings method. +#pragma warning disable CA2000 // Call System. IDisposable. Dispose on object created by + // 'GetConnectionPool(elasticConfiguration)' before all references to it are out of scope + var pool = GetConnectionPool(elasticConfiguration); +#pragma warning restore CA2000 + + var settings = new ConnectionSettings(pool); + + if (elasticConfiguration.ConnectionType != "CloudConnectionPool" && + !string.IsNullOrWhiteSpace(elasticConfiguration.Username) && + !string.IsNullOrWhiteSpace(elasticConfiguration.Password)) + { + settings.BasicAuthentication(elasticConfiguration.Username, elasticConfiguration.Password); + } + + if (!string.IsNullOrWhiteSpace(elasticConfiguration.CertificateFingerprint)) + { + settings.CertificateFingerprint(elasticConfiguration.CertificateFingerprint); + } + + if (elasticConfiguration.EnableApiVersioningHeader) + { + settings.EnableApiVersioningHeader(); + } + + return settings; + } + + private static IConnectionPool GetConnectionPool(ElasticConnectionOptions elasticConfiguration) + { + var uris = elasticConfiguration.Ports.Select(port => new Uri($"{elasticConfiguration.Url}:{port.ToTechnicalString()}")).Distinct(); + IConnectionPool pool = null; + switch (elasticConfiguration.ConnectionType) + { + case "SingleNodeConnectionPool": + pool = new SingleNodeConnectionPool(uris.First()); + break; + + case "CloudConnectionPool": + if (!string.IsNullOrWhiteSpace(elasticConfiguration.Username) && + !string.IsNullOrWhiteSpace(elasticConfiguration.Password) && + !string.IsNullOrWhiteSpace(elasticConfiguration.CloudId)) + { + using var credentials = new BasicAuthenticationCredentials( + elasticConfiguration.Username, + elasticConfiguration.Password); + pool = new CloudConnectionPool(elasticConfiguration.CloudId, credentials); + } + + break; + + case "StaticConnectionPool": + pool = new StaticConnectionPool(uris); + break; + + case "SniffingConnectionPool": + pool = new SniffingConnectionPool(uris); + break; + + case "StickyConnectionPool": + pool = new StickyConnectionPool(uris); + break; + + default: + pool = new SingleNodeConnectionPool(uris.First()); + break; + } + + return pool; + } } diff --git a/Lombiq.Hosting.Tenants.Maintenance/Maintenance/DeleteElasticsearchIndices/ElasticsearchIndicesMaintenanceOptions.cs b/Lombiq.Hosting.Tenants.Maintenance/Maintenance/DeleteElasticsearchIndices/ElasticsearchIndicesMaintenanceOptions.cs new file mode 100644 index 00000000..e996aca7 --- /dev/null +++ b/Lombiq.Hosting.Tenants.Maintenance/Maintenance/DeleteElasticsearchIndices/ElasticsearchIndicesMaintenanceOptions.cs @@ -0,0 +1,8 @@ +namespace Lombiq.Hosting.Tenants.Maintenance.Maintenance.DeleteElasticsearchIndices; + +public class ElasticsearchIndicesMaintenanceOptions +{ + public bool DeleteMaintenanceIsEnabled { get; set; } + public bool RebuildMaintenanceIsEnabled { get; set; } + public bool BeforeSetupMiddlewareIsEnabled { get; set; } +} diff --git a/Lombiq.Hosting.Tenants.Maintenance/Maintenance/DeleteElasticsearchIndices/RebuildElasticsearchIndicesMaintenanceProvider.cs b/Lombiq.Hosting.Tenants.Maintenance/Maintenance/DeleteElasticsearchIndices/RebuildElasticsearchIndicesMaintenanceProvider.cs new file mode 100644 index 00000000..98e5d8c8 --- /dev/null +++ b/Lombiq.Hosting.Tenants.Maintenance/Maintenance/DeleteElasticsearchIndices/RebuildElasticsearchIndicesMaintenanceProvider.cs @@ -0,0 +1,50 @@ +using Lombiq.Hosting.Tenants.Maintenance.Extensions; +using Lombiq.Hosting.Tenants.Maintenance.Models; +using Lombiq.Hosting.Tenants.Maintenance.Services; +using Microsoft.Extensions.Options; +using OrchardCore.Search.Elasticsearch.Core.Services; +using System.Threading.Tasks; + +namespace Lombiq.Hosting.Tenants.Maintenance.Maintenance.DeleteElasticsearchIndices; + +public class RebuildElasticsearchIndicesMaintenanceProvider : MaintenanceProviderBase +{ + private readonly IOptions _options; + private readonly ElasticIndexingService _elasticIndexingService; + private readonly ElasticIndexSettingsService _elasticIndexSettingsService; + + public RebuildElasticsearchIndicesMaintenanceProvider( + IOptions options, + ElasticIndexingService elasticIndexingService, + ElasticIndexSettingsService elasticIndexSettingsService) + { + _options = options; + _elasticIndexingService = elasticIndexingService; + _elasticIndexSettingsService = elasticIndexSettingsService; + } + + public override Task ShouldExecuteAsync(MaintenanceTaskExecutionContext context) => + Task.FromResult( + _options.Value.RebuildMaintenanceIsEnabled && + !context.WasLatestExecutionSuccessful()); + + public override async Task ExecuteAsync(MaintenanceTaskExecutionContext context) + { + var settings = await _elasticIndexSettingsService.GetSettingsAsync(); + foreach (var setting in settings) + { + await _elasticIndexingService.RebuildIndexAsync(setting); + + if (setting.QueryAnalyzerName != setting.AnalyzerName) + { + // Query Analyzer may be different until the index in rebuilt. + // Since the index is rebuilt, lets make sure we query using the same analyzer. + setting.QueryAnalyzerName = setting.AnalyzerName; + + await _elasticIndexSettingsService.UpdateIndexAsync(setting); + } + + await _elasticIndexingService.ProcessContentItemsAsync(setting.IndexName); + } + } +} diff --git a/Lombiq.Hosting.Tenants.Maintenance/Maintenance/DeleteElasticsearchIndices/Startup.cs b/Lombiq.Hosting.Tenants.Maintenance/Maintenance/DeleteElasticsearchIndices/Startup.cs index f436295e..1ff75ef9 100644 --- a/Lombiq.Hosting.Tenants.Maintenance/Maintenance/DeleteElasticsearchIndices/Startup.cs +++ b/Lombiq.Hosting.Tenants.Maintenance/Maintenance/DeleteElasticsearchIndices/Startup.cs @@ -1,24 +1,52 @@ +using Elasticsearch.Net; using Lombiq.HelpfulLibraries.OrchardCore.Mvc; using Lombiq.Hosting.Tenants.Maintenance.Constants; using Lombiq.Hosting.Tenants.Maintenance.Services; using Microsoft.AspNetCore.Builder; using Microsoft.AspNetCore.Routing; +using Microsoft.Extensions.Configuration; using Microsoft.Extensions.DependencyInjection; using Microsoft.Extensions.Options; +using Nest; using OrchardCore.Environment.Shell; using OrchardCore.Environment.Shell.Configuration; using OrchardCore.Modules; +using OrchardCore.Search.Elasticsearch.Core.Models; +using OrchardCore.Search.Elasticsearch.Core.Services; using System; +using System.Linq; namespace Lombiq.Hosting.Tenants.Maintenance.Maintenance.DeleteElasticsearchIndices; -[Feature(FeatureNames.DeleteElasticsearchIndices)] -public class Startup : StartupBase +[Feature(FeatureNames.DeleteOrRebuildElasticsearchIndices)] +public class DeleteElasticsearchIndicesStartup : StartupBase +{ + private readonly IShellConfiguration _shellConfiguration; + + public override int Order => int.MaxValue; + + public DeleteElasticsearchIndicesStartup(IShellConfiguration shellConfiguration) => _shellConfiguration = shellConfiguration; + + public override void ConfigureServices(IServiceCollection services) + { + services.BindAndConfigureSection( + _shellConfiguration, + "Lombiq_Hosting_Tenants_Maintenance:ElasticsearchIndicesOptions"); + + services.AddScoped(); + services.AddScoped(); + } +} + +[Feature(FeatureNames.DeleteElasticsearchIndicesBeforeSetup)] +public class DeleteElasticsearchIndicesBeforeSetupStartup : StartupBase { private readonly IShellConfiguration _shellConfiguration; private readonly ShellSettings _shellSettings; - public Startup(IShellConfiguration shellConfiguration, ShellSettings shellSettings) + public override int Order => int.MaxValue; + + public DeleteElasticsearchIndicesBeforeSetupStartup(IShellConfiguration shellConfiguration, ShellSettings shellSettings) { _shellConfiguration = shellConfiguration; _shellSettings = shellSettings; @@ -26,21 +54,109 @@ public Startup(IShellConfiguration shellConfiguration, ShellSettings shellSettin public override void ConfigureServices(IServiceCollection services) { - services.BindAndConfigureSection( + services.BindAndConfigureSection( _shellConfiguration, - "Lombiq_Hosting_Tenants_Maintenance:DeleteElasticsearchIndices"); + "Lombiq_Hosting_Tenants_Maintenance:ElasticsearchIndicesOptions"); - services.AddScoped(); + // After the setup, the Elasticsearch module can be loaded the regular way. + if (!_shellSettings.IsUninitialized()) return; + + // This is necessary because the Elasticsearch module can't be enabled the regular way if this module is added + // as a setup feature, otherwise you get a ContentsAdminList shape missing exception on the admin dashboard. For + // more info see: + var configuration = _shellConfiguration.GetSection("OrchardCore_Elasticsearch"); + var elasticConfiguration = configuration.Get(); + services.Configure(o => o.ConfigurationExists = true); + // Otherwise the ElasticClient won't work. Copied all this from the OC Elasticsearch module. +#pragma warning disable CA2000 // Call System. IDisposable. Dispose on object created by + // 'GetConnectionSettings(elasticConfiguration)' before all references to it are out of scope + var settings = GetConnectionSettings(elasticConfiguration); +#pragma warning restore CA2000 + services.AddSingleton(new ElasticClient(settings)); + services.AddSingleton(); } public override void Configure(IApplicationBuilder app, IEndpointRouteBuilder routes, IServiceProvider serviceProvider) { if (!_shellSettings.IsUninitialized()) return; - var options = serviceProvider.GetRequiredService>().Value; - if (options.MiddlewareIsEnabled) + var options = serviceProvider.GetRequiredService>().Value; + if (options.BeforeSetupMiddlewareIsEnabled) { app.UseMiddleware(); } } + + private static ConnectionSettings GetConnectionSettings(ElasticConnectionOptions elasticConfiguration) + { + // This is a copy of the OC Elasticsearch module's OrchardCore.Search.Elasticsearch.Startup.GetConnectionSettings method. +#pragma warning disable CA2000 // Call System. IDisposable. Dispose on object created by + // 'GetConnectionPool(elasticConfiguration)' before all references to it are out of scope + var pool = GetConnectionPool(elasticConfiguration); +#pragma warning restore CA2000 + + var settings = new ConnectionSettings(pool); + + if (elasticConfiguration.ConnectionType != "CloudConnectionPool" && + !string.IsNullOrWhiteSpace(elasticConfiguration.Username) && + !string.IsNullOrWhiteSpace(elasticConfiguration.Password)) + { + settings.BasicAuthentication(elasticConfiguration.Username, elasticConfiguration.Password); + } + + if (!string.IsNullOrWhiteSpace(elasticConfiguration.CertificateFingerprint)) + { + settings.CertificateFingerprint(elasticConfiguration.CertificateFingerprint); + } + + if (elasticConfiguration.EnableApiVersioningHeader) + { + settings.EnableApiVersioningHeader(); + } + + return settings; + } + + private static IConnectionPool GetConnectionPool(ElasticConnectionOptions elasticConfiguration) + { + var uris = elasticConfiguration.Ports.Select(port => new Uri($"{elasticConfiguration.Url}:{port.ToTechnicalString()}")).Distinct(); + IConnectionPool pool = null; + switch (elasticConfiguration.ConnectionType) + { + case "SingleNodeConnectionPool": + pool = new SingleNodeConnectionPool(uris.First()); + break; + + case "CloudConnectionPool": + if (!string.IsNullOrWhiteSpace(elasticConfiguration.Username) && + !string.IsNullOrWhiteSpace(elasticConfiguration.Password) && + !string.IsNullOrWhiteSpace(elasticConfiguration.CloudId)) + { + using var credentials = new BasicAuthenticationCredentials( + elasticConfiguration.Username, + elasticConfiguration.Password); + pool = new CloudConnectionPool(elasticConfiguration.CloudId, credentials); + } + + break; + + case "StaticConnectionPool": + pool = new StaticConnectionPool(uris); + break; + + case "SniffingConnectionPool": + pool = new SniffingConnectionPool(uris); + break; + + case "StickyConnectionPool": + pool = new StickyConnectionPool(uris); + break; + + default: + pool = new SingleNodeConnectionPool(uris.First()); + break; + } + + return pool; + } } diff --git a/Lombiq.Hosting.Tenants.Maintenance/Manifest.cs b/Lombiq.Hosting.Tenants.Maintenance/Manifest.cs index eb484e51..89c1abfd 100644 --- a/Lombiq.Hosting.Tenants.Maintenance/Manifest.cs +++ b/Lombiq.Hosting.Tenants.Maintenance/Manifest.cs @@ -62,10 +62,17 @@ )] [assembly: Feature( - Id = DeleteElasticsearchIndices, + Id = DeleteOrRebuildElasticsearchIndices, Name = "Lombiq Hosting - Tenants Maintenance Delete Elasticsearch Indexes", Description = "Deletes Elasticsearch indexes.", Category = "Maintenance", - DefaultTenantOnly = true, - Dependencies = [Maintenance, "OrchardCore.Search.Elasticsearch"] + Dependencies = [Maintenance] +)] + +[assembly: Feature( + Id = DeleteElasticsearchIndicesBeforeSetup, + Name = "Lombiq Hosting - Tenants Maintenance Delete Elasticsearch Indexes Before Setup", + Description = "Deletes Elasticsearch indexes before setup.", + Category = "Maintenance", + Dependencies = [] )] diff --git a/Lombiq.Hosting.Tenants.Maintenance/Readme.md b/Lombiq.Hosting.Tenants.Maintenance/Readme.md index 2b789bd6..61e78eaa 100644 --- a/Lombiq.Hosting.Tenants.Maintenance/Readme.md +++ b/Lombiq.Hosting.Tenants.Maintenance/Readme.md @@ -167,9 +167,10 @@ The following configuration should be used to allow the maintenance to run and f { "OrchardCore": { "Lombiq_Hosting_Tenants_Maintenance": { - "DeleteElasticsearchIndices": { - "MaintenanceIsEnabled": true, - "MiddlewareIsEnabled": true + "ElasticsearchIndicesOptions": { + "DeleteMaintenanceIsEnabled": true, + "RebuildMaintenanceIsEnabled": true, + "BeforeSetupMiddlewareIsEnabled": true } } } From 6452ba388b17c7f83095d8f8c58287367ca4c88a Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Kriszti=C3=A1n=20N=C3=A9meth?= Date: Thu, 5 Sep 2024 13:56:02 +0200 Subject: [PATCH 08/15] Deleting not needed stuff --- .../DeleteElasticsearchIndicesMiddleware.cs | 77 ------------------- 1 file changed, 77 deletions(-) diff --git a/Lombiq.Hosting.Tenants.Maintenance/Maintenance/DeleteElasticsearchIndices/DeleteElasticsearchIndicesMiddleware.cs b/Lombiq.Hosting.Tenants.Maintenance/Maintenance/DeleteElasticsearchIndices/DeleteElasticsearchIndicesMiddleware.cs index 1b746e7a..68e23b46 100644 --- a/Lombiq.Hosting.Tenants.Maintenance/Maintenance/DeleteElasticsearchIndices/DeleteElasticsearchIndicesMiddleware.cs +++ b/Lombiq.Hosting.Tenants.Maintenance/Maintenance/DeleteElasticsearchIndices/DeleteElasticsearchIndicesMiddleware.cs @@ -1,13 +1,9 @@ -using Elasticsearch.Net; using Microsoft.AspNetCore.Http; using Microsoft.Extensions.DependencyInjection; -using Nest; using OrchardCore.Environment.Shell; using OrchardCore.Locking.Distributed; -using OrchardCore.Search.Elasticsearch.Core.Models; using OrchardCore.Search.Elasticsearch.Core.Services; using System; -using System.Linq; using System.Net; using System.Threading.Tasks; @@ -84,77 +80,4 @@ private async Task InvokeNextIfUninitializedAsync(ShellSettings shellSetti await _next.Invoke(httpContext); return true; } - - private static ConnectionSettings GetConnectionSettings(ElasticConnectionOptions elasticConfiguration) - { - // This is a copy of the OC Elasticsearch module's OrchardCore.Search.Elasticsearch.Startup.GetConnectionSettings method. -#pragma warning disable CA2000 // Call System. IDisposable. Dispose on object created by - // 'GetConnectionPool(elasticConfiguration)' before all references to it are out of scope - var pool = GetConnectionPool(elasticConfiguration); -#pragma warning restore CA2000 - - var settings = new ConnectionSettings(pool); - - if (elasticConfiguration.ConnectionType != "CloudConnectionPool" && - !string.IsNullOrWhiteSpace(elasticConfiguration.Username) && - !string.IsNullOrWhiteSpace(elasticConfiguration.Password)) - { - settings.BasicAuthentication(elasticConfiguration.Username, elasticConfiguration.Password); - } - - if (!string.IsNullOrWhiteSpace(elasticConfiguration.CertificateFingerprint)) - { - settings.CertificateFingerprint(elasticConfiguration.CertificateFingerprint); - } - - if (elasticConfiguration.EnableApiVersioningHeader) - { - settings.EnableApiVersioningHeader(); - } - - return settings; - } - - private static IConnectionPool GetConnectionPool(ElasticConnectionOptions elasticConfiguration) - { - var uris = elasticConfiguration.Ports.Select(port => new Uri($"{elasticConfiguration.Url}:{port.ToTechnicalString()}")).Distinct(); - IConnectionPool pool = null; - switch (elasticConfiguration.ConnectionType) - { - case "SingleNodeConnectionPool": - pool = new SingleNodeConnectionPool(uris.First()); - break; - - case "CloudConnectionPool": - if (!string.IsNullOrWhiteSpace(elasticConfiguration.Username) && - !string.IsNullOrWhiteSpace(elasticConfiguration.Password) && - !string.IsNullOrWhiteSpace(elasticConfiguration.CloudId)) - { - using var credentials = new BasicAuthenticationCredentials( - elasticConfiguration.Username, - elasticConfiguration.Password); - pool = new CloudConnectionPool(elasticConfiguration.CloudId, credentials); - } - - break; - - case "StaticConnectionPool": - pool = new StaticConnectionPool(uris); - break; - - case "SniffingConnectionPool": - pool = new SniffingConnectionPool(uris); - break; - - case "StickyConnectionPool": - pool = new StickyConnectionPool(uris); - break; - - default: - pool = new SingleNodeConnectionPool(uris.First()); - break; - } - - return pool; - } } From ae19beab7abaffa58a01b2f98ee807bb9c1ce791 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Kriszti=C3=A1n=20N=C3=A9meth?= Date: Mon, 9 Sep 2024 18:08:27 +0200 Subject: [PATCH 09/15] Renaming --- .../DeleteElasticsearchIndicesMaintenanceProvider.cs | 2 +- .../DeleteElasticsearchIndicesMiddleware.cs | 2 +- .../ElasticsearchIndicesMaintenanceOptions.cs | 2 +- .../RebuildElasticsearchIndicesMaintenanceProvider.cs | 2 +- .../Startup.cs | 6 +----- Lombiq.Hosting.Tenants.Maintenance/Readme.md | 4 ++-- 6 files changed, 7 insertions(+), 11 deletions(-) rename Lombiq.Hosting.Tenants.Maintenance/Maintenance/{DeleteElasticsearchIndices => ElasticsearchIndices}/DeleteElasticsearchIndicesMaintenanceProvider.cs (93%) rename Lombiq.Hosting.Tenants.Maintenance/Maintenance/{DeleteElasticsearchIndices => ElasticsearchIndices}/DeleteElasticsearchIndicesMiddleware.cs (96%) rename Lombiq.Hosting.Tenants.Maintenance/Maintenance/{DeleteElasticsearchIndices => ElasticsearchIndices}/ElasticsearchIndicesMaintenanceOptions.cs (73%) rename Lombiq.Hosting.Tenants.Maintenance/Maintenance/{DeleteElasticsearchIndices => ElasticsearchIndices}/RebuildElasticsearchIndicesMaintenanceProvider.cs (96%) rename Lombiq.Hosting.Tenants.Maintenance/Maintenance/{DeleteElasticsearchIndices => ElasticsearchIndices}/Startup.cs (97%) diff --git a/Lombiq.Hosting.Tenants.Maintenance/Maintenance/DeleteElasticsearchIndices/DeleteElasticsearchIndicesMaintenanceProvider.cs b/Lombiq.Hosting.Tenants.Maintenance/Maintenance/ElasticsearchIndices/DeleteElasticsearchIndicesMaintenanceProvider.cs similarity index 93% rename from Lombiq.Hosting.Tenants.Maintenance/Maintenance/DeleteElasticsearchIndices/DeleteElasticsearchIndicesMaintenanceProvider.cs rename to Lombiq.Hosting.Tenants.Maintenance/Maintenance/ElasticsearchIndices/DeleteElasticsearchIndicesMaintenanceProvider.cs index 9f654c3a..4a150ea0 100644 --- a/Lombiq.Hosting.Tenants.Maintenance/Maintenance/DeleteElasticsearchIndices/DeleteElasticsearchIndicesMaintenanceProvider.cs +++ b/Lombiq.Hosting.Tenants.Maintenance/Maintenance/ElasticsearchIndices/DeleteElasticsearchIndicesMaintenanceProvider.cs @@ -5,7 +5,7 @@ using OrchardCore.Search.Elasticsearch.Core.Services; using System.Threading.Tasks; -namespace Lombiq.Hosting.Tenants.Maintenance.Maintenance.DeleteElasticsearchIndices; +namespace Lombiq.Hosting.Tenants.Maintenance.Maintenance.ElasticsearchIndices; public class DeleteElasticsearchIndicesMaintenanceProvider : MaintenanceProviderBase { diff --git a/Lombiq.Hosting.Tenants.Maintenance/Maintenance/DeleteElasticsearchIndices/DeleteElasticsearchIndicesMiddleware.cs b/Lombiq.Hosting.Tenants.Maintenance/Maintenance/ElasticsearchIndices/DeleteElasticsearchIndicesMiddleware.cs similarity index 96% rename from Lombiq.Hosting.Tenants.Maintenance/Maintenance/DeleteElasticsearchIndices/DeleteElasticsearchIndicesMiddleware.cs rename to Lombiq.Hosting.Tenants.Maintenance/Maintenance/ElasticsearchIndices/DeleteElasticsearchIndicesMiddleware.cs index 68e23b46..3d31ef79 100644 --- a/Lombiq.Hosting.Tenants.Maintenance/Maintenance/DeleteElasticsearchIndices/DeleteElasticsearchIndicesMiddleware.cs +++ b/Lombiq.Hosting.Tenants.Maintenance/Maintenance/ElasticsearchIndices/DeleteElasticsearchIndicesMiddleware.cs @@ -7,7 +7,7 @@ using System.Net; using System.Threading.Tasks; -namespace Lombiq.Hosting.Tenants.Maintenance.Maintenance.DeleteElasticsearchIndices; +namespace Lombiq.Hosting.Tenants.Maintenance.Maintenance.ElasticsearchIndices; public class DeleteElasticsearchIndicesMiddleware { diff --git a/Lombiq.Hosting.Tenants.Maintenance/Maintenance/DeleteElasticsearchIndices/ElasticsearchIndicesMaintenanceOptions.cs b/Lombiq.Hosting.Tenants.Maintenance/Maintenance/ElasticsearchIndices/ElasticsearchIndicesMaintenanceOptions.cs similarity index 73% rename from Lombiq.Hosting.Tenants.Maintenance/Maintenance/DeleteElasticsearchIndices/ElasticsearchIndicesMaintenanceOptions.cs rename to Lombiq.Hosting.Tenants.Maintenance/Maintenance/ElasticsearchIndices/ElasticsearchIndicesMaintenanceOptions.cs index e996aca7..078e299e 100644 --- a/Lombiq.Hosting.Tenants.Maintenance/Maintenance/DeleteElasticsearchIndices/ElasticsearchIndicesMaintenanceOptions.cs +++ b/Lombiq.Hosting.Tenants.Maintenance/Maintenance/ElasticsearchIndices/ElasticsearchIndicesMaintenanceOptions.cs @@ -1,4 +1,4 @@ -namespace Lombiq.Hosting.Tenants.Maintenance.Maintenance.DeleteElasticsearchIndices; +namespace Lombiq.Hosting.Tenants.Maintenance.Maintenance.ElasticsearchIndices; public class ElasticsearchIndicesMaintenanceOptions { diff --git a/Lombiq.Hosting.Tenants.Maintenance/Maintenance/DeleteElasticsearchIndices/RebuildElasticsearchIndicesMaintenanceProvider.cs b/Lombiq.Hosting.Tenants.Maintenance/Maintenance/ElasticsearchIndices/RebuildElasticsearchIndicesMaintenanceProvider.cs similarity index 96% rename from Lombiq.Hosting.Tenants.Maintenance/Maintenance/DeleteElasticsearchIndices/RebuildElasticsearchIndicesMaintenanceProvider.cs rename to Lombiq.Hosting.Tenants.Maintenance/Maintenance/ElasticsearchIndices/RebuildElasticsearchIndicesMaintenanceProvider.cs index 98e5d8c8..a3586017 100644 --- a/Lombiq.Hosting.Tenants.Maintenance/Maintenance/DeleteElasticsearchIndices/RebuildElasticsearchIndicesMaintenanceProvider.cs +++ b/Lombiq.Hosting.Tenants.Maintenance/Maintenance/ElasticsearchIndices/RebuildElasticsearchIndicesMaintenanceProvider.cs @@ -5,7 +5,7 @@ using OrchardCore.Search.Elasticsearch.Core.Services; using System.Threading.Tasks; -namespace Lombiq.Hosting.Tenants.Maintenance.Maintenance.DeleteElasticsearchIndices; +namespace Lombiq.Hosting.Tenants.Maintenance.Maintenance.ElasticsearchIndices; public class RebuildElasticsearchIndicesMaintenanceProvider : MaintenanceProviderBase { diff --git a/Lombiq.Hosting.Tenants.Maintenance/Maintenance/DeleteElasticsearchIndices/Startup.cs b/Lombiq.Hosting.Tenants.Maintenance/Maintenance/ElasticsearchIndices/Startup.cs similarity index 97% rename from Lombiq.Hosting.Tenants.Maintenance/Maintenance/DeleteElasticsearchIndices/Startup.cs rename to Lombiq.Hosting.Tenants.Maintenance/Maintenance/ElasticsearchIndices/Startup.cs index 1ff75ef9..5ce26256 100644 --- a/Lombiq.Hosting.Tenants.Maintenance/Maintenance/DeleteElasticsearchIndices/Startup.cs +++ b/Lombiq.Hosting.Tenants.Maintenance/Maintenance/ElasticsearchIndices/Startup.cs @@ -16,15 +16,13 @@ using System; using System.Linq; -namespace Lombiq.Hosting.Tenants.Maintenance.Maintenance.DeleteElasticsearchIndices; +namespace Lombiq.Hosting.Tenants.Maintenance.Maintenance.ElasticsearchIndices; [Feature(FeatureNames.DeleteOrRebuildElasticsearchIndices)] public class DeleteElasticsearchIndicesStartup : StartupBase { private readonly IShellConfiguration _shellConfiguration; - public override int Order => int.MaxValue; - public DeleteElasticsearchIndicesStartup(IShellConfiguration shellConfiguration) => _shellConfiguration = shellConfiguration; public override void ConfigureServices(IServiceCollection services) @@ -44,8 +42,6 @@ public class DeleteElasticsearchIndicesBeforeSetupStartup : StartupBase private readonly IShellConfiguration _shellConfiguration; private readonly ShellSettings _shellSettings; - public override int Order => int.MaxValue; - public DeleteElasticsearchIndicesBeforeSetupStartup(IShellConfiguration shellConfiguration, ShellSettings shellSettings) { _shellConfiguration = shellConfiguration; diff --git a/Lombiq.Hosting.Tenants.Maintenance/Readme.md b/Lombiq.Hosting.Tenants.Maintenance/Readme.md index 61e78eaa..781b880f 100644 --- a/Lombiq.Hosting.Tenants.Maintenance/Readme.md +++ b/Lombiq.Hosting.Tenants.Maintenance/Readme.md @@ -157,9 +157,9 @@ Any user accounts with an e-mail matching the `EmailExcludePattern` regex will n ### `Lombiq.Hosting.Tenants.Maintenance.DeleteElasticsearchIndices` -This contains a maintenance task that deletes all Elasticsearch indices related to the tenant that is being activated. +This contains a maintenance task that deletes all Elasticsearch indices related to the tenant that is being activated, and another one that rebuilds them. -It also contains a middleware that deletes all Elasticsearch indices related to the tenant, but it does that before the tenant setup. This is useful when you want to delete the indices before the tenant setup, so you can create indices from a recipe during setup. To be able to use the middleware before setup this feature must be added as a setup feature. You can do this with `OrchardCoreBuilder.AddSetupFeatures(Lombiq.Hosting.Tenants.Maintenance.Constants.FeatureNames.DeleteElasticsearchIndices);` +It also contains a middleware that deletes all Elasticsearch indices related to the tenant, but it does that before the tenant setup. This is useful when you want to delete the indices before the tenant setup, so you can create indices from a recipe during setup. To be able to use the middleware before setup this feature must be added as a setup feature. You can do this with `OrchardCoreBuilder.AddSetupFeatures(Lombiq.Hosting.Tenants.Maintenance.Constants.FeatureNames.DeleteElasticsearchIndicesBeforeSetup);` The following configuration should be used to allow the maintenance to run and for the middleware to be added: From ed0c3e6019ada16c35db6329cad9f7b69fdcf9d5 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Kriszti=C3=A1n=20N=C3=A9meth?= Date: Wed, 11 Sep 2024 11:21:04 +0200 Subject: [PATCH 10/15] Delete using --- .../Maintenance/ElasticsearchIndices/Startup.cs | 6 +++++- 1 file changed, 5 insertions(+), 1 deletion(-) diff --git a/Lombiq.Hosting.Tenants.Maintenance/Maintenance/ElasticsearchIndices/Startup.cs b/Lombiq.Hosting.Tenants.Maintenance/Maintenance/ElasticsearchIndices/Startup.cs index 5ce26256..f2526d37 100644 --- a/Lombiq.Hosting.Tenants.Maintenance/Maintenance/ElasticsearchIndices/Startup.cs +++ b/Lombiq.Hosting.Tenants.Maintenance/Maintenance/ElasticsearchIndices/Startup.cs @@ -128,9 +128,13 @@ private static IConnectionPool GetConnectionPool(ElasticConnectionOptions elasti !string.IsNullOrWhiteSpace(elasticConfiguration.Password) && !string.IsNullOrWhiteSpace(elasticConfiguration.CloudId)) { - using var credentials = new BasicAuthenticationCredentials( + // This is a copy of the OC Elasticsearch module's OrchardCore.Search.Elasticsearch.Startup.GetConnectionPool method. +#pragma warning disable CA2000 // CA2000: Call System. IDisposable. Dispose on object created by + // 'new BasicAuthenticationCredentials(' before all references to it are out of scope + var credentials = new BasicAuthenticationCredentials( elasticConfiguration.Username, elasticConfiguration.Password); +#pragma warning restore CA2000 pool = new CloudConnectionPool(elasticConfiguration.CloudId, credentials); } From 5f26590f47dc149eb394a9df21c02f7398029368 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Kriszti=C3=A1n=20N=C3=A9meth?= Date: Thu, 12 Sep 2024 13:14:45 +0200 Subject: [PATCH 11/15] Update Lombiq.Hosting.Tenants.Maintenance/Maintenance/ElasticsearchIndices/DeleteElasticsearchIndicesMiddleware.cs MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit Co-authored-by: Gábor Pór <51411356+porgabi@users.noreply.github.com> --- .../DeleteElasticsearchIndicesMiddleware.cs | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/Lombiq.Hosting.Tenants.Maintenance/Maintenance/ElasticsearchIndices/DeleteElasticsearchIndicesMiddleware.cs b/Lombiq.Hosting.Tenants.Maintenance/Maintenance/ElasticsearchIndices/DeleteElasticsearchIndicesMiddleware.cs index 3d31ef79..f2cfa8a8 100644 --- a/Lombiq.Hosting.Tenants.Maintenance/Maintenance/ElasticsearchIndices/DeleteElasticsearchIndicesMiddleware.cs +++ b/Lombiq.Hosting.Tenants.Maintenance/Maintenance/ElasticsearchIndices/DeleteElasticsearchIndicesMiddleware.cs @@ -49,7 +49,7 @@ public async Task InvokeAsync(HttpContext httpContext) if (!locked) { - throw new TimeoutException($"Fails to acquire an elasticsearch indices deletion lock for the tenant: {_shellSettings.Name}"); + throw new TimeoutException($"Failed to acquire an Elasticsearch indices deletion lock for the tenant: {_shellSettings.Name}"); } await using var acquiredLock = locker; From 1f178ecb2fb293cef0b256106a95e2c9b4dc6a29 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Kriszti=C3=A1n=20N=C3=A9meth?= Date: Thu, 12 Sep 2024 13:15:00 +0200 Subject: [PATCH 12/15] Update Lombiq.Hosting.Tenants.Maintenance/Maintenance/ElasticsearchIndices/RebuildElasticsearchIndicesMaintenanceProvider.cs MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit Co-authored-by: Gábor Pór <51411356+porgabi@users.noreply.github.com> --- .../RebuildElasticsearchIndicesMaintenanceProvider.cs | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/Lombiq.Hosting.Tenants.Maintenance/Maintenance/ElasticsearchIndices/RebuildElasticsearchIndicesMaintenanceProvider.cs b/Lombiq.Hosting.Tenants.Maintenance/Maintenance/ElasticsearchIndices/RebuildElasticsearchIndicesMaintenanceProvider.cs index a3586017..c39436eb 100644 --- a/Lombiq.Hosting.Tenants.Maintenance/Maintenance/ElasticsearchIndices/RebuildElasticsearchIndicesMaintenanceProvider.cs +++ b/Lombiq.Hosting.Tenants.Maintenance/Maintenance/ElasticsearchIndices/RebuildElasticsearchIndicesMaintenanceProvider.cs @@ -37,7 +37,7 @@ public override async Task ExecuteAsync(MaintenanceTaskExecutionContext context) if (setting.QueryAnalyzerName != setting.AnalyzerName) { - // Query Analyzer may be different until the index in rebuilt. + // Query Analyzer may be different until the index is rebuilt. // Since the index is rebuilt, lets make sure we query using the same analyzer. setting.QueryAnalyzerName = setting.AnalyzerName; From 400c418968255c435b6b408c42b6afbd9caef5c9 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Kriszti=C3=A1n=20N=C3=A9meth?= Date: Thu, 12 Sep 2024 13:15:32 +0200 Subject: [PATCH 13/15] Update Lombiq.Hosting.Tenants.Maintenance/Maintenance/ElasticsearchIndices/Startup.cs MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit Co-authored-by: Gábor Pór <51411356+porgabi@users.noreply.github.com> --- .../Maintenance/ElasticsearchIndices/Startup.cs | 2 ++ 1 file changed, 2 insertions(+) diff --git a/Lombiq.Hosting.Tenants.Maintenance/Maintenance/ElasticsearchIndices/Startup.cs b/Lombiq.Hosting.Tenants.Maintenance/Maintenance/ElasticsearchIndices/Startup.cs index f2526d37..9f23715d 100644 --- a/Lombiq.Hosting.Tenants.Maintenance/Maintenance/ElasticsearchIndices/Startup.cs +++ b/Lombiq.Hosting.Tenants.Maintenance/Maintenance/ElasticsearchIndices/Startup.cs @@ -63,11 +63,13 @@ public override void ConfigureServices(IServiceCollection services) var configuration = _shellConfiguration.GetSection("OrchardCore_Elasticsearch"); var elasticConfiguration = configuration.Get(); services.Configure(o => o.ConfigurationExists = true); + // Otherwise the ElasticClient won't work. Copied all this from the OC Elasticsearch module. #pragma warning disable CA2000 // Call System. IDisposable. Dispose on object created by // 'GetConnectionSettings(elasticConfiguration)' before all references to it are out of scope var settings = GetConnectionSettings(elasticConfiguration); #pragma warning restore CA2000 + services.AddSingleton(new ElasticClient(settings)); services.AddSingleton(); } From 745d1f56bb4da579502141088037f1bef0c8b4f4 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Kriszti=C3=A1n=20N=C3=A9meth?= Date: Thu, 12 Sep 2024 13:23:52 +0200 Subject: [PATCH 14/15] Better docs --- .../Maintenance/ElasticsearchIndices/Startup.cs | 6 +++--- 1 file changed, 3 insertions(+), 3 deletions(-) diff --git a/Lombiq.Hosting.Tenants.Maintenance/Maintenance/ElasticsearchIndices/Startup.cs b/Lombiq.Hosting.Tenants.Maintenance/Maintenance/ElasticsearchIndices/Startup.cs index 9f23715d..6b657316 100644 --- a/Lombiq.Hosting.Tenants.Maintenance/Maintenance/ElasticsearchIndices/Startup.cs +++ b/Lombiq.Hosting.Tenants.Maintenance/Maintenance/ElasticsearchIndices/Startup.cs @@ -57,9 +57,9 @@ public override void ConfigureServices(IServiceCollection services) // After the setup, the Elasticsearch module can be loaded the regular way. if (!_shellSettings.IsUninitialized()) return; - // This is necessary because the Elasticsearch module can't be enabled the regular way if this module is added - // as a setup feature, otherwise you get a ContentsAdminList shape missing exception on the admin dashboard. For - // more info see: + // This is necessary to initialize Elasticsearch here like this, instead of using the Elasticsearch module from + // OC. Because the Elasticsearch module can't be enabled the regular way if this module is added + // as a setup feature, otherwise you get a ContentsAdminList shape missing exception on the admin dashboard. var configuration = _shellConfiguration.GetSection("OrchardCore_Elasticsearch"); var elasticConfiguration = configuration.Get(); services.Configure(o => o.ConfigurationExists = true); From 23448b547aca3f6217a16296498a1c31ab2f9792 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Kriszti=C3=A1n=20N=C3=A9meth?= Date: Thu, 12 Sep 2024 13:24:01 +0200 Subject: [PATCH 15/15] Deleting whitespace --- .../Maintenance/ElasticsearchIndices/Startup.cs | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/Lombiq.Hosting.Tenants.Maintenance/Maintenance/ElasticsearchIndices/Startup.cs b/Lombiq.Hosting.Tenants.Maintenance/Maintenance/ElasticsearchIndices/Startup.cs index 6b657316..332635a0 100644 --- a/Lombiq.Hosting.Tenants.Maintenance/Maintenance/ElasticsearchIndices/Startup.cs +++ b/Lombiq.Hosting.Tenants.Maintenance/Maintenance/ElasticsearchIndices/Startup.cs @@ -63,13 +63,13 @@ public override void ConfigureServices(IServiceCollection services) var configuration = _shellConfiguration.GetSection("OrchardCore_Elasticsearch"); var elasticConfiguration = configuration.Get(); services.Configure(o => o.ConfigurationExists = true); - + // Otherwise the ElasticClient won't work. Copied all this from the OC Elasticsearch module. #pragma warning disable CA2000 // Call System. IDisposable. Dispose on object created by // 'GetConnectionSettings(elasticConfiguration)' before all references to it are out of scope var settings = GetConnectionSettings(elasticConfiguration); #pragma warning restore CA2000 - + services.AddSingleton(new ElasticClient(settings)); services.AddSingleton(); }