diff --git a/src/administration/Administration.Service/DependencyInjection/PartnerRegistrationSettings.cs b/src/administration/Administration.Service/DependencyInjection/PartnerRegistrationSettings.cs index 8b7cac7fbb..5d113b8732 100644 --- a/src/administration/Administration.Service/DependencyInjection/PartnerRegistrationSettings.cs +++ b/src/administration/Administration.Service/DependencyInjection/PartnerRegistrationSettings.cs @@ -28,6 +28,8 @@ public class PartnerRegistrationSettings [Required] [DistinctValues("x => x.ClientId")] public IEnumerable InitialRoles { get; set; } = null!; + + public int ApplicationsMaxPageSize { get; set; } } public static class PartnerRegistrationSettingsExtensions diff --git a/src/administration/Administration.Service/Extensions/CompanyApplicationStatusFilterExtensions.cs b/src/administration/Administration.Service/Extensions/CompanyApplicationStatusFilterExtensions.cs new file mode 100644 index 0000000000..0c20d93009 --- /dev/null +++ b/src/administration/Administration.Service/Extensions/CompanyApplicationStatusFilterExtensions.cs @@ -0,0 +1,22 @@ +using Org.Eclipse.TractusX.Portal.Backend.Administration.Service.Models; +using Org.Eclipse.TractusX.Portal.Backend.PortalBackend.PortalEntities.Enums; + +namespace Org.Eclipse.TractusX.Portal.Backend.Administration.Service.Extensions; + +public static class CompanyApplicationStatusFilterExtensions +{ + public static IEnumerable GetCompanyApplicationStatusIds(this CompanyApplicationStatusFilter? companyApplicationStatusFilter) => + companyApplicationStatusFilter switch + { + CompanyApplicationStatusFilter.Closed => + [ + CompanyApplicationStatusId.CONFIRMED, CompanyApplicationStatusId.DECLINED + ], + CompanyApplicationStatusFilter.InReview => [CompanyApplicationStatusId.SUBMITTED], + _ => + [ + CompanyApplicationStatusId.SUBMITTED, CompanyApplicationStatusId.CONFIRMED, + CompanyApplicationStatusId.DECLINED + ] + }; +} diff --git a/src/administration/Administration.Service/appsettings.json b/src/administration/Administration.Service/appsettings.json index e5736f627a..448d58b569 100644 --- a/src/administration/Administration.Service/appsettings.json +++ b/src/administration/Administration.Service/appsettings.json @@ -305,7 +305,8 @@ "Scope": "", "TokenAddress": "", "BaseAddress": "", - "UseDimWallet": false + "UseDimWallet": false, + "RetriggerEndClearinghouseIntervalInDays": 30 }, "SdFactory": { "Username": "", diff --git a/src/externalsystems/Clearinghouse.Library/BusinessLogic/ClearinghouseBusinessLogic.cs b/src/externalsystems/Clearinghouse.Library/BusinessLogic/ClearinghouseBusinessLogic.cs index 0c4d79cfa2..7d38fd2ee5 100644 --- a/src/externalsystems/Clearinghouse.Library/BusinessLogic/ClearinghouseBusinessLogic.cs +++ b/src/externalsystems/Clearinghouse.Library/BusinessLogic/ClearinghouseBusinessLogic.cs @@ -20,6 +20,7 @@ using Microsoft.Extensions.Options; using Org.Eclipse.TractusX.Portal.Backend.Clearinghouse.Library.Models; using Org.Eclipse.TractusX.Portal.Backend.Custodian.Library.BusinessLogic; +using Org.Eclipse.TractusX.Portal.Backend.Framework.DateTimeProvider; using Org.Eclipse.TractusX.Portal.Backend.Framework.ErrorHandling; using Org.Eclipse.TractusX.Portal.Backend.PortalBackend.DBAccess; using Org.Eclipse.TractusX.Portal.Backend.PortalBackend.DBAccess.Repositories; @@ -33,6 +34,7 @@ public class ClearinghouseBusinessLogic( IClearinghouseService clearinghouseService, ICustodianBusinessLogic custodianBusinessLogic, IApplicationChecklistService checklistService, + IDateTimeProvider dateTimeProvider, IOptions options) : IClearinghouseBusinessLogic { @@ -136,4 +138,39 @@ public async Task ProcessEndClearinghouse(Guid applicationId, ClearinghouseRespo ? [ProcessStepTypeId.TRIGGER_OVERRIDE_CLEARING_HOUSE] : [ProcessStepTypeId.START_SELF_DESCRIPTION_LP]); } + + public async Task CheckEndClearinghouseProcesses(CancellationToken stoppingToken) + { + var applicationIds = await portalRepositories.GetInstance() + .GetApplicationsForClearinghouseRetrigger(dateTimeProvider.OffsetNow.AddDays(-_settings.RetriggerEndClearinghouseIntervalInDays)) + .ToListAsync(stoppingToken).ConfigureAwait(false); + + var hasChanges = false; + foreach (var applicationId in applicationIds) + { + var context = await checklistService + .VerifyChecklistEntryAndProcessSteps( + applicationId, + ApplicationChecklistEntryTypeId.CLEARING_HOUSE, + [ApplicationChecklistEntryStatusId.IN_PROGRESS], + ProcessStepTypeId.END_CLEARING_HOUSE) + .ConfigureAwait(ConfigureAwaitOptions.None); + + checklistService.FinalizeChecklistEntryAndProcessSteps( + context, + null, + item => + { + item.ApplicationChecklistEntryStatusId = ApplicationChecklistEntryStatusId.TO_DO; + item.Comment = "Reset to retrigger clearinghouse"; + }, + [ProcessStepTypeId.START_OVERRIDE_CLEARING_HOUSE]); + hasChanges = true; + } + + if (hasChanges) + { + await portalRepositories.SaveAsync().ConfigureAwait(ConfigureAwaitOptions.None); + } + } } diff --git a/src/externalsystems/Clearinghouse.Library/BusinessLogic/IClearinghouseBusinessLogic.cs b/src/externalsystems/Clearinghouse.Library/BusinessLogic/IClearinghouseBusinessLogic.cs index a0b024b0f7..bc880a38fa 100644 --- a/src/externalsystems/Clearinghouse.Library/BusinessLogic/IClearinghouseBusinessLogic.cs +++ b/src/externalsystems/Clearinghouse.Library/BusinessLogic/IClearinghouseBusinessLogic.cs @@ -27,4 +27,5 @@ public interface IClearinghouseBusinessLogic { Task HandleClearinghouse(IApplicationChecklistService.WorkerChecklistProcessStepData context, CancellationToken cancellationToken); Task ProcessEndClearinghouse(Guid applicationId, ClearinghouseResponseData data, CancellationToken cancellationToken); + Task CheckEndClearinghouseProcesses(CancellationToken stoppingToken); } diff --git a/src/externalsystems/Clearinghouse.Library/ClearinghouseSettings.cs b/src/externalsystems/Clearinghouse.Library/ClearinghouseSettings.cs index de4cfca5c0..3eae16b761 100644 --- a/src/externalsystems/Clearinghouse.Library/ClearinghouseSettings.cs +++ b/src/externalsystems/Clearinghouse.Library/ClearinghouseSettings.cs @@ -35,4 +35,7 @@ public class ClearinghouseSettings : KeyVaultAuthSettings public string CallbackUrl { get; set; } = null!; public bool UseDimWallet { get; set; } + + [Required] + public int RetriggerEndClearinghouseIntervalInDays { get; set; } } diff --git a/src/maintenance/Maintenance.App/BatchDeleteService.cs b/src/maintenance/Maintenance.App/BatchDeleteService.cs deleted file mode 100644 index 71738aa79b..0000000000 --- a/src/maintenance/Maintenance.App/BatchDeleteService.cs +++ /dev/null @@ -1,100 +0,0 @@ -/******************************************************************************** - * Copyright (c) 2022 BMW Group AG - * Copyright (c) 2022 Contributors to the Eclipse Foundation - * - * See the NOTICE file(s) distributed with this work for additional - * information regarding copyright ownership. - * - * This program and the accompanying materials are made available under the - * terms of the Apache License, Version 2.0 which is available at - * https://www.apache.org/licenses/LICENSE-2.0. - * - * Unless required by applicable law or agreed to in writing, software - * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT - * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the - * License for the specific language governing permissions and limitations - * under the License. - * - * SPDX-License-Identifier: Apache-2.0 - ********************************************************************************/ - -using Microsoft.EntityFrameworkCore; -using Org.Eclipse.TractusX.Portal.Backend.PortalBackend.PortalEntities; -using Org.Eclipse.TractusX.Portal.Backend.PortalBackend.PortalEntities.Entities; -using Org.Eclipse.TractusX.Portal.Backend.PortalBackend.PortalEntities.Enums; - -namespace Org.Eclipse.TractusX.Portal.Backend.Maintenance.App; - -/// -/// Service to delete the pending and inactive documents as well as the depending consents from the database -/// -public class BatchDeleteService : BackgroundService -{ - private readonly IHostApplicationLifetime _applicationLifetime; - private readonly IServiceScopeFactory _serviceScopeFactory; - private readonly ILogger _logger; - private readonly int _days; - - /// - /// Creates a new instance of - /// - /// Application lifetime - /// access to the services - /// the logger - /// the apps configuration - public BatchDeleteService( - IHostApplicationLifetime applicationLifetime, - IServiceScopeFactory serviceScopeFactory, - ILogger logger, - IConfiguration config) - { - _applicationLifetime = applicationLifetime; - _serviceScopeFactory = serviceScopeFactory; - _logger = logger; - _days = config.GetValue("DeleteIntervalInDays"); - } - - /// - protected override async Task ExecuteAsync(CancellationToken stoppingToken) - { - using var scope = _serviceScopeFactory.CreateScope(); - var dbContext = scope.ServiceProvider.GetRequiredService(); - - if (!stoppingToken.IsCancellationRequested) - { - try - { - _logger.LogInformation("Getting documents and assignments older {Days} days", _days); - List<(Guid DocumentId, IEnumerable AgreementIds, IEnumerable OfferIds)> documentData = await dbContext.Documents.Where(x => - x.DateCreated < DateTimeOffset.UtcNow.AddDays(-_days) && - !x.Companies.Any() && - x.Connector == null && - !x.Consents.Any() && - x.DocumentStatusId == DocumentStatusId.INACTIVE) - .Select(doc => new ValueTuple, IEnumerable>( - doc.Id, - doc.Agreements.Select(x => x.Id), - doc.Offers.Select(x => x.Id) - )) - .ToListAsync(stoppingToken) - .ConfigureAwait(ConfigureAwaitOptions.None); - _logger.LogInformation("Cleaning up {DocumentCount} Documents and {OfferIdCount} OfferAssignedDocuments", documentData.Count, documentData.SelectMany(x => x.OfferIds).Count()); - - var agreementsToDeleteDocumentId = documentData.SelectMany(data => data.AgreementIds.Select(agreementId => new Agreement(agreementId, default, null!, default, default, default) { DocumentId = data.DocumentId })).ToList(); - dbContext.Agreements.AttachRange(agreementsToDeleteDocumentId); - agreementsToDeleteDocumentId.ForEach(agreement => agreement.DocumentId = null); - dbContext.OfferAssignedDocuments.RemoveRange(documentData.SelectMany(data => data.OfferIds.Select(offerId => new OfferAssignedDocument(offerId, data.DocumentId)))); - dbContext.Documents.RemoveRange(documentData.Select(x => new Document(x.DocumentId, null!, null!, null!, default, default, default, default))); - await dbContext.SaveChangesAsync(stoppingToken).ConfigureAwait(ConfigureAwaitOptions.None); - _logger.LogInformation("Documents older than {Days} days and depending consents successfully cleaned up", _days); - } - catch (Exception ex) - { - Environment.ExitCode = 1; - _logger.LogError("Database clean up failed with error: {Errors}", ex.Message); - } - } - - _applicationLifetime.StopApplication(); - } -} diff --git a/src/maintenance/Maintenance.App/DependencyInjection/BatchDeleteServiceSettings.cs b/src/maintenance/Maintenance.App/DependencyInjection/BatchDeleteServiceSettings.cs new file mode 100644 index 0000000000..7fca4fa931 --- /dev/null +++ b/src/maintenance/Maintenance.App/DependencyInjection/BatchDeleteServiceSettings.cs @@ -0,0 +1,36 @@ +using Org.Eclipse.TractusX.Portal.Backend.Maintenance.App.Services; +using System.ComponentModel.DataAnnotations; + +namespace Org.Eclipse.TractusX.Portal.Backend.Maintenance.App.DependencyInjection; + +/// +/// Settings for the +/// +public class BatchDeleteServiceSettings +{ + /// + /// Documents older than this configured value will be deleted + /// + [Required] + public int DeleteIntervalInDays { get; set; } +} + +/// +/// Extensions for the +/// +public static class BatchDeleteServiceExtensions +{ + /// + /// Adds the to the service collection + /// + /// The service collection used for di + /// The configuration section to get the settings from + /// The enhanced service collection + public static IServiceCollection AddBatchDelete(this IServiceCollection services, IConfigurationSection section) + { + services.AddOptions().Bind(section); + services + .AddTransient(); + return services; + } +} diff --git a/src/maintenance/Maintenance.App/DependencyInjection/MaintenanceServiceExtensions.cs b/src/maintenance/Maintenance.App/DependencyInjection/MaintenanceServiceExtensions.cs new file mode 100644 index 0000000000..3e0e3ad8fe --- /dev/null +++ b/src/maintenance/Maintenance.App/DependencyInjection/MaintenanceServiceExtensions.cs @@ -0,0 +1,20 @@ +using Org.Eclipse.TractusX.Portal.Backend.Framework.DateTimeProvider; +using Org.Eclipse.TractusX.Portal.Backend.Maintenance.App.Services; + +namespace Org.Eclipse.TractusX.Portal.Backend.Maintenance.App.DependencyInjection; + +/// +/// Extension methods to register the necessary services for the maintenance job +/// +public static class MaintenanceServiceExtensions +{ + /// + /// Adds the dependencies for the maintenance service + /// + /// The service collection + /// The enhanced service collection + public static IServiceCollection AddMaintenanceService(this IServiceCollection services) => + services + .AddTransient() + .AddTransient(); +} diff --git a/src/maintenance/Maintenance.App/Maintenance.App.csproj b/src/maintenance/Maintenance.App/Maintenance.App.csproj index 40d65b3fbc..2c5b5a2ec1 100644 --- a/src/maintenance/Maintenance.App/Maintenance.App.csproj +++ b/src/maintenance/Maintenance.App/Maintenance.App.csproj @@ -46,6 +46,7 @@ + diff --git a/src/maintenance/Maintenance.App/Program.cs b/src/maintenance/Maintenance.App/Program.cs index 8b3e261e30..91d7cacdc7 100644 --- a/src/maintenance/Maintenance.App/Program.cs +++ b/src/maintenance/Maintenance.App/Program.cs @@ -20,8 +20,10 @@ using Laraue.EfCoreTriggers.PostgreSql.Extensions; using Microsoft.EntityFrameworkCore; +using Org.Eclipse.TractusX.Portal.Backend.Clearinghouse.Library; using Org.Eclipse.TractusX.Portal.Backend.Framework.Logging; -using Org.Eclipse.TractusX.Portal.Backend.Maintenance.App; +using Org.Eclipse.TractusX.Portal.Backend.Maintenance.App.DependencyInjection; +using Org.Eclipse.TractusX.Portal.Backend.Maintenance.App.Services; using Org.Eclipse.TractusX.Portal.Backend.PortalBackend.PortalEntities; using Org.Eclipse.TractusX.Portal.Backend.PortalBackend.PortalEntities.Auditing; using Org.Eclipse.TractusX.Portal.Backend.Processes.ProcessIdentity.DependencyInjection; @@ -31,22 +33,36 @@ Log.Information("Building service"); try { - var host = Host.CreateDefaultBuilder(args) - .UseSystemd() + var host = Host + .CreateDefaultBuilder(args) .ConfigureServices((hostContext, services) => { services + .AddMaintenanceService() .AddConfigurationProcessIdentityIdDetermination(hostContext.Configuration.GetSection("ProcessIdentity")) + .AddBatchDelete(hostContext.Configuration.GetSection("BatchDelete")) + .AddClearinghouseService(hostContext.Configuration.GetSection("Clearinghouse")) .AddDbAuditing() .AddDbContext(o => o.UseNpgsql(hostContext.Configuration.GetConnectionString("PortalDb")) - .UsePostgreSqlTriggers()); - services.AddHostedService(); + .UsePostgreSqlTriggers()); }) .AddLogging() .Build(); + Log.Information("Building worker completed"); - await host.RunAsync().ConfigureAwait(ConfigureAwaitOptions.None); + using var tokenSource = new CancellationTokenSource(); + Console.CancelKeyPress += (s, e) => + { + Log.Information("Canceling..."); + tokenSource.Cancel(); + e.Cancel = true; + }; + + Log.Information("Start processing"); + var workerInstance = host.Services.GetRequiredService(); + await workerInstance.ExecuteAsync(tokenSource.Token).ConfigureAwait(ConfigureAwaitOptions.None); + Log.Information("Execution finished shutting down"); } catch (Exception ex) when (!ex.GetType().Name.Equals("StopTheHostException", StringComparison.Ordinal)) { diff --git a/src/maintenance/Maintenance.App/Services/BatchDeleteService.cs b/src/maintenance/Maintenance.App/Services/BatchDeleteService.cs new file mode 100644 index 0000000000..97895877e8 --- /dev/null +++ b/src/maintenance/Maintenance.App/Services/BatchDeleteService.cs @@ -0,0 +1,69 @@ +/******************************************************************************** + * Copyright (c) 2024 Contributors to the Eclipse Foundation + * + * See the NOTICE file(s) distributed with this work for additional + * information regarding copyright ownership. + * + * This program and the accompanying materials are made available under the + * terms of the Apache License, Version 2.0 which is available at + * https://www.apache.org/licenses/LICENSE-2.0. + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT + * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the + * License for the specific language governing permissions and limitations + * under the License. + * + * SPDX-License-Identifier: Apache-2.0 + ********************************************************************************/ + +using Microsoft.Extensions.Options; +using Org.Eclipse.TractusX.Portal.Backend.Framework.DateTimeProvider; +using Org.Eclipse.TractusX.Portal.Backend.Maintenance.App.DependencyInjection; +using Org.Eclipse.TractusX.Portal.Backend.PortalBackend.DBAccess; +using Org.Eclipse.TractusX.Portal.Backend.PortalBackend.DBAccess.Repositories; +using Org.Eclipse.TractusX.Portal.Backend.PortalBackend.PortalEntities.Entities; + +namespace Org.Eclipse.TractusX.Portal.Backend.Maintenance.App.Services; + +/// +public class BatchDeleteService( + ILogger logger, + IOptions options, + IPortalRepositories portalRepositories, + IDateTimeProvider dateTimeProvider) : IBatchDeleteService +{ + private readonly BatchDeleteServiceSettings _settings = options.Value; + + /// + public async Task CleanupDocuments(CancellationToken cancellationToken) + { + try + { + logger.LogInformation("Getting documents and assignments older {Days} days", _settings.DeleteIntervalInDays); + var documentRepository = portalRepositories.GetInstance(); + var documentData = await documentRepository + .GetDocumentDataForCleanup(dateTimeProvider.OffsetNow.AddDays(-_settings.DeleteIntervalInDays)) + .ConfigureAwait(ConfigureAwaitOptions.None); + if (documentData.Count == 0) + { + logger.LogInformation("No documents to cleanup"); + return; + } + + logger.LogInformation("Cleaning up {DocumentCount} Documents and {OfferIdCount} OfferAssignedDocuments", documentData.Count, documentData.SelectMany(x => x.OfferIds).Count()); + + var agreementsToDeleteDocumentId = documentData.SelectMany(data => data.AgreementIds.Select(agreementId => new Agreement(agreementId, default, null!, default, default, default) { DocumentId = data.DocumentId })).ToList(); + portalRepositories.AttachRange(agreementsToDeleteDocumentId); + agreementsToDeleteDocumentId.ForEach(agreement => agreement.DocumentId = null); + documentRepository.RemoveOfferAssignedDocuments(documentData.SelectMany(data => data.OfferIds.Select(offerId => new OfferAssignedDocument(offerId, data.DocumentId)))); + documentRepository.RemoveDocuments(documentData.Select(x => x.DocumentId)); + await portalRepositories.SaveAsync().ConfigureAwait(ConfigureAwaitOptions.None); + logger.LogInformation("Documents older than {Days} days and depending consents successfully cleaned up", _settings.DeleteIntervalInDays); + } + catch (Exception ex) + { + logger.LogError("Database clean up failed with error: {Errors}", ex.Message); + } + } +} diff --git a/src/maintenance/Maintenance.App/Services/IBatchDeleteService.cs b/src/maintenance/Maintenance.App/Services/IBatchDeleteService.cs new file mode 100644 index 0000000000..f680226bec --- /dev/null +++ b/src/maintenance/Maintenance.App/Services/IBatchDeleteService.cs @@ -0,0 +1,32 @@ +/******************************************************************************** + * Copyright (c) 2024 Contributors to the Eclipse Foundation + * + * See the NOTICE file(s) distributed with this work for additional + * information regarding copyright ownership. + * + * This program and the accompanying materials are made available under the + * terms of the Apache License, Version 2.0 which is available at + * https://www.apache.org/licenses/LICENSE-2.0. + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT + * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the + * License for the specific language governing permissions and limitations + * under the License. + * + * SPDX-License-Identifier: Apache-2.0 + ********************************************************************************/ + +namespace Org.Eclipse.TractusX.Portal.Backend.Maintenance.App.Services; + +/// +/// Service to delete the pending and inactive documents as well as the depending on consents from the database +/// +public interface IBatchDeleteService +{ + /// + /// Cleans up the documents and related entries from the database + /// + /// The cancellation token + Task CleanupDocuments(CancellationToken cancellationToken); +} diff --git a/src/maintenance/Maintenance.App/Services/MaintenanceService.cs b/src/maintenance/Maintenance.App/Services/MaintenanceService.cs new file mode 100644 index 0000000000..7fd7262bec --- /dev/null +++ b/src/maintenance/Maintenance.App/Services/MaintenanceService.cs @@ -0,0 +1,54 @@ +/******************************************************************************** + * Copyright (c) 2022 BMW Group AG + * Copyright (c) 2022 Contributors to the Eclipse Foundation + * + * See the NOTICE file(s) distributed with this work for additional + * information regarding copyright ownership. + * + * This program and the accompanying materials are made available under the + * terms of the Apache License, Version 2.0 which is available at + * https://www.apache.org/licenses/LICENSE-2.0. + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT + * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the + * License for the specific language governing permissions and limitations + * under the License. + * + * SPDX-License-Identifier: Apache-2.0 + ********************************************************************************/ + +using Org.Eclipse.TractusX.Portal.Backend.Clearinghouse.Library.BusinessLogic; +using Org.Eclipse.TractusX.Portal.Backend.Processes.ProcessIdentity; + +namespace Org.Eclipse.TractusX.Portal.Backend.Maintenance.App.Services; + +/// +/// Service to +/// 1. delete the pending and inactive documents as well as the depending on consents from the database +/// 2. schedule clearinghouse process steps where the END_CLEARINGHOUSE is in todo for a specific duration +/// +public class MaintenanceService(IServiceScopeFactory serviceScopeFactory) +{ + /// + /// executes the logic + /// + /// The cancellation token + public async Task ExecuteAsync(CancellationToken cancellationToken) + { + using var scope = serviceScopeFactory.CreateScope(); + + var processIdentityDataDetermination = scope.ServiceProvider.GetRequiredService(); + //call processIdentityDataDetermination.GetIdentityData() once to initialize IdentityService IdentityData for synchronous use: + await processIdentityDataDetermination.GetIdentityData().ConfigureAwait(ConfigureAwaitOptions.None); + + var batchDeleteBusinessLogic = scope.ServiceProvider.GetRequiredService(); + var clearinghouseBusinessLogic = scope.ServiceProvider.GetRequiredService(); + + if (!cancellationToken.IsCancellationRequested) + { + await batchDeleteBusinessLogic.CleanupDocuments(cancellationToken).ConfigureAwait(ConfigureAwaitOptions.None); + await clearinghouseBusinessLogic.CheckEndClearinghouseProcesses(cancellationToken).ConfigureAwait(ConfigureAwaitOptions.None); + } + } +} diff --git a/src/maintenance/Maintenance.App/appsettings.json b/src/maintenance/Maintenance.App/appsettings.json index c68407236b..f340ce15e3 100644 --- a/src/maintenance/Maintenance.App/appsettings.json +++ b/src/maintenance/Maintenance.App/appsettings.json @@ -23,7 +23,21 @@ "ConnectionStrings": { "PortalDb": "Server=placeholder;Database=placeholder;Port=5432;User Id=placeholder;Password=placeholder;Ssl Mode=Disable;" }, - "DeleteIntervalInDays": 80, + "BatchDelete": { + "DeleteIntervalInDays": 80 + }, + "Clearinghouse": { + "Username": "", + "Password": "", + "ClientId": "", + "GrantType": "", + "ClientSecret": "", + "Scope": "", + "TokenAddress": "", + "BaseAddress": "", + "UseDimWallet": false, + "RetriggerEndClearinghouseIntervalInDays": 30 + }, "ProcessIdentity": { "ProcessUserId": "" } diff --git a/src/portalbackend/PortalBackend.DBAccess/Repositories/ApplicationChecklistRepository.cs b/src/portalbackend/PortalBackend.DBAccess/Repositories/ApplicationChecklistRepository.cs index 468736470f..51fe492b00 100644 --- a/src/portalbackend/PortalBackend.DBAccess/Repositories/ApplicationChecklistRepository.cs +++ b/src/portalbackend/PortalBackend.DBAccess/Repositories/ApplicationChecklistRepository.cs @@ -26,19 +26,9 @@ namespace Org.Eclipse.TractusX.Portal.Backend.PortalBackend.DBAccess.Repositories; -public class ApplicationChecklistRepository : IApplicationChecklistRepository +public class ApplicationChecklistRepository(PortalDbContext dbContext) + : IApplicationChecklistRepository { - private readonly PortalDbContext _portalDbContext; - - /// - /// Creates a new instance of - /// - /// The portal db context - public ApplicationChecklistRepository(PortalDbContext portalDbContext) - { - _portalDbContext = portalDbContext; - } - /// public void CreateChecklistForApplication(Guid applicationId, IEnumerable<(ApplicationChecklistEntryTypeId TypeId, ApplicationChecklistEntryStatusId StatusId)> checklistEntries) { @@ -48,7 +38,7 @@ public void CreateChecklistForApplication(Guid applicationId, IEnumerable<(Appli x.TypeId, x.StatusId, DateTimeOffset.UtcNow)); - _portalDbContext.ApplicationChecklist.AddRange(entries); + dbContext.ApplicationChecklist.AddRange(entries); } /// @@ -56,14 +46,14 @@ public ApplicationChecklistEntry AttachAndModifyApplicationChecklist(Guid applic { var entity = new ApplicationChecklistEntry(applicationId, applicationChecklistTypeId, default, default); initialize?.Invoke(entity); - _portalDbContext.ApplicationChecklist.Attach(entity); + dbContext.ApplicationChecklist.Attach(entity); entity.DateLastChanged = DateTimeOffset.UtcNow; setFields.Invoke(entity); return entity; } public Task<(bool IsValidProcessId, Guid ApplicationId, CompanyApplicationStatusId ApplicationStatusId, IEnumerable<(ApplicationChecklistEntryTypeId EntryTypeId, ApplicationChecklistEntryStatusId EntryStatusId)> Checklist)> GetChecklistData(Guid processId) => - _portalDbContext.Processes + dbContext.Processes .AsNoTracking() .Where(process => process.Id == processId) .Select(process => @@ -79,7 +69,7 @@ public ApplicationChecklistEntry AttachAndModifyApplicationChecklist(Guid applic .SingleOrDefaultAsync(); public Task GetChecklistProcessStepData(Guid applicationId, IEnumerable entryTypeIds, IEnumerable processStepTypeIds) => - _portalDbContext.CompanyApplications + dbContext.CompanyApplications .AsNoTracking() .AsSplitQuery() .Where(application => application.Id == applicationId) @@ -106,4 +96,19 @@ public ApplicationChecklistEntry AttachAndModifyApplicationChecklist(Guid applic step.ProcessStepStatusId == ProcessStepStatusId.TODO) : null)) .SingleOrDefaultAsync(); + + public IAsyncEnumerable GetApplicationsForClearinghouseRetrigger(DateTimeOffset dateCreatedThreshold) => + dbContext.CompanyApplications + .Where(ca => + ca.ApplicationChecklistEntries.Any(ce => + ce.ApplicationChecklistEntryTypeId == ApplicationChecklistEntryTypeId.CLEARING_HOUSE && + ce.ApplicationChecklistEntryStatusId == ApplicationChecklistEntryStatusId.IN_PROGRESS && + ce.DateCreated < dateCreatedThreshold) && + ca.ChecklistProcess!.ProcessTypeId == ProcessTypeId.APPLICATION_CHECKLIST && + ca.ChecklistProcess!.ProcessSteps.Any(ps => + ps.ProcessStepTypeId == ProcessStepTypeId.END_CLEARING_HOUSE && + ps.ProcessStepStatusId == ProcessStepStatusId.TODO && + ps.DateCreated < dateCreatedThreshold)) + .Select(x => x.Id) + .ToAsyncEnumerable(); } diff --git a/src/portalbackend/PortalBackend.DBAccess/Repositories/DocumentRepository.cs b/src/portalbackend/PortalBackend.DBAccess/Repositories/DocumentRepository.cs index a071201172..e6a7e9a861 100644 --- a/src/portalbackend/PortalBackend.DBAccess/Repositories/DocumentRepository.cs +++ b/src/portalbackend/PortalBackend.DBAccess/Repositories/DocumentRepository.cs @@ -26,19 +26,8 @@ namespace Org.Eclipse.TractusX.Portal.Backend.PortalBackend.DBAccess.Repositories; /// Implementation of accessing database with EF Core. -public class DocumentRepository : IDocumentRepository +public class DocumentRepository(PortalDbContext dbContext) : IDocumentRepository { - private readonly PortalDbContext _dbContext; - - /// - /// Constructor. - /// - /// PortalDb context. - public DocumentRepository(PortalDbContext dbContext) - { - _dbContext = dbContext; - } - /// public Document CreateDocument(string documentName, byte[] documentContent, byte[] hash, MediaTypeId mediaTypeId, DocumentTypeId documentTypeId, Action? setupOptionalFields) { @@ -53,12 +42,12 @@ public Document CreateDocument(string documentName, byte[] documentContent, byte documentTypeId); setupOptionalFields?.Invoke(document); - return _dbContext.Documents.Add(document).Entity; + return dbContext.Documents.Add(document).Entity; } /// public Task<(Guid DocumentId, DocumentStatusId DocumentStatusId, IEnumerable ConsentIds, bool IsSameUser)> GetDocumentDetailsForIdUntrackedAsync(Guid documentId, Guid companyUserId) => - _dbContext.Documents + dbContext.Documents .AsNoTracking() .Where(x => x.Id == documentId) .Select(document => @@ -70,7 +59,7 @@ public Document CreateDocument(string documentName, byte[] documentContent, byte .SingleOrDefaultAsync(); public Task<(bool IsApplicationAssignedUser, IEnumerable Documents)> GetUploadedDocumentsAsync(Guid applicationId, DocumentTypeId documentTypeId, Guid companyUserId) => - _dbContext.CompanyApplications + dbContext.CompanyApplications .AsNoTracking() .Where(application => application.Id == applicationId) .Select(application => new @@ -92,14 +81,14 @@ public Document CreateDocument(string documentName, byte[] documentContent, byte /// public Task<(Guid DocumentId, bool IsSameUser)> GetDocumentIdWithCompanyUserCheckAsync(Guid documentId, Guid companyUserId) => - _dbContext.Documents + dbContext.Documents .Where(x => x.Id == documentId) .Select(x => new ValueTuple(x.Id, x.CompanyUserId == companyUserId)) .SingleOrDefaultAsync(); /// public Task<(byte[]? Content, string FileName, MediaTypeId MediaTypeId, bool IsUserInCompany)> GetDocumentDataAndIsCompanyUserAsync(Guid documentId, Guid userCompanyId) => - _dbContext.Documents + dbContext.Documents .Where(x => x.Id == documentId) .Select(x => new { @@ -115,22 +104,22 @@ public Document CreateDocument(string documentName, byte[] documentContent, byte /// public Task<(byte[] Content, string FileName, MediaTypeId MediaTypeId)> GetDocumentDataByIdAndTypeAsync(Guid documentId, DocumentTypeId documentTypeId) => - _dbContext.Documents + dbContext.Documents .Where(x => x.Id == documentId && x.DocumentTypeId == documentTypeId) .Select(x => new ValueTuple(x.DocumentContent, x.DocumentName, x.MediaTypeId)) .SingleOrDefaultAsync(); /// public void RemoveDocument(Guid documentId) => - _dbContext.Documents.Remove(new Document(documentId, null!, null!, null!, default, default, default, default)); + dbContext.Documents.Remove(new Document(documentId, null!, null!, null!, default, default, default, default)); /// public Task GetDocumentByIdAsync(Guid documentId) => - _dbContext.Documents.SingleOrDefaultAsync(x => x.Id == documentId); + dbContext.Documents.SingleOrDefaultAsync(x => x.Id == documentId); /// public Task<(Guid DocumentId, DocumentStatusId DocumentStatusId, bool IsSameApplicationUser, DocumentTypeId documentTypeId, bool IsQueriedApplicationStatus, IEnumerable applicationId)> GetDocumentDetailsForApplicationUntrackedAsync(Guid documentId, Guid userCompanyId, IEnumerable applicationStatusIds) => - _dbContext.Documents + dbContext.Documents .AsNoTracking() .Where(x => x.Id == documentId) .Select(document => new @@ -152,7 +141,7 @@ public void AttachAndModifyDocument(Guid documentId, Action? initializ { var document = new Document(documentId, null!, null!, null!, default, default, default, default); initialize?.Invoke(document); - _dbContext.Attach(document); + dbContext.Attach(document); modify(document); } @@ -165,13 +154,13 @@ public void AttachAndModifyDocuments(IEnumerable<(Guid DocumentId, Action x.Document)); + dbContext.AttachRange(initial.Select(x => x.Document)); initial.ForEach(x => x.Modify(x.Document)); } /// public Task GetDocumentSeedDataByIdAsync(Guid documentId) => - _dbContext.Documents + dbContext.Documents .AsNoTracking() .Where(x => x.Id == documentId) .Select(doc => new DocumentSeedData( @@ -187,7 +176,7 @@ public void AttachAndModifyDocuments(IEnumerable<(Guid DocumentId, Action public Task GetOfferDocumentContentAsync(Guid offerId, Guid documentId, IEnumerable documentTypeIds, OfferTypeId offerTypeId, CancellationToken cancellationToken) => - _dbContext.Documents + dbContext.Documents .Where(document => document.Id == documentId) .Select(document => new { @@ -215,7 +204,7 @@ public void AttachAndModifyDocuments(IEnumerable<(Guid DocumentId, Action public Task<(IEnumerable<(OfferStatusId OfferStatusId, Guid OfferId, bool IsOfferType)> OfferData, bool IsDocumentTypeMatch, DocumentStatusId DocumentStatusId, bool IsProviderCompanyUser)> GetOfferDocumentsAsync(Guid documentId, Guid userCompanyId, IEnumerable documentTypeIds, OfferTypeId offerTypeId) => - _dbContext.Documents + dbContext.Documents .Where(document => document.Id == documentId) .Select(document => new { @@ -240,12 +229,28 @@ public void AttachAndModifyDocuments(IEnumerable<(Guid DocumentId, Action public void RemoveDocuments(IEnumerable documentIds) => - _dbContext.Documents.RemoveRange(documentIds.Select(documentId => new Document(documentId, null!, null!, null!, default, default, default, default))); + dbContext.Documents.RemoveRange(documentIds.Select(documentId => new Document(documentId, null!, null!, null!, default, default, default, default))); + + public void RemoveOfferAssignedDocuments(IEnumerable offerAssignedDocuments) => + dbContext.OfferAssignedDocuments.RemoveRange(offerAssignedDocuments); public Task<(byte[] Content, string FileName, bool IsDocumentTypeMatch, MediaTypeId MediaTypeId)> GetDocumentAsync(Guid documentId, IEnumerable documentTypeIds) => - _dbContext.Documents + dbContext.Documents .Where(x => x.Id == documentId) .Select(x => new ValueTuple(x.DocumentContent, x.DocumentName, documentTypeIds.Contains(x.DocumentTypeId), x.MediaTypeId)) .SingleOrDefaultAsync(); + public Task AgreementIds, IEnumerable OfferIds)>> GetDocumentDataForCleanup(DateTimeOffset dateCreated) => + dbContext.Documents.Where(x => + x.DateCreated < dateCreated && + !x.Companies.Any() && + x.Connector == null && + !x.Consents.Any() && + x.DocumentStatusId == DocumentStatusId.INACTIVE) + .Select(doc => new ValueTuple, IEnumerable>( + doc.Id, + doc.Agreements.Select(x => x.Id), + doc.Offers.Select(x => x.Id) + )) + .ToListAsync(); } diff --git a/src/portalbackend/PortalBackend.DBAccess/Repositories/IApplicationChecklistRepository.cs b/src/portalbackend/PortalBackend.DBAccess/Repositories/IApplicationChecklistRepository.cs index e7d0323ac6..c632346ebb 100644 --- a/src/portalbackend/PortalBackend.DBAccess/Repositories/IApplicationChecklistRepository.cs +++ b/src/portalbackend/PortalBackend.DBAccess/Repositories/IApplicationChecklistRepository.cs @@ -46,4 +46,5 @@ public interface IApplicationChecklistRepository Task<(bool IsValidProcessId, Guid ApplicationId, CompanyApplicationStatusId ApplicationStatusId, IEnumerable<(ApplicationChecklistEntryTypeId EntryTypeId, ApplicationChecklistEntryStatusId EntryStatusId)> Checklist)> GetChecklistData(Guid processId); Task GetChecklistProcessStepData(Guid applicationId, IEnumerable entryTypeIds, IEnumerable processStepTypeIds); + IAsyncEnumerable GetApplicationsForClearinghouseRetrigger(DateTimeOffset dateCreatedThreshold); } diff --git a/src/portalbackend/PortalBackend.DBAccess/Repositories/IDocumentRepository.cs b/src/portalbackend/PortalBackend.DBAccess/Repositories/IDocumentRepository.cs index 88b36d717d..e3e2c84485 100644 --- a/src/portalbackend/PortalBackend.DBAccess/Repositories/IDocumentRepository.cs +++ b/src/portalbackend/PortalBackend.DBAccess/Repositories/IDocumentRepository.cs @@ -146,6 +146,12 @@ public interface IDocumentRepository /// void RemoveDocuments(IEnumerable documentIds); + /// + /// Delete List Of Document + /// + /// + void RemoveOfferAssignedDocuments(IEnumerable offerAssignedDocuments); + /// /// Gets the registration document with the given id /// @@ -153,4 +159,6 @@ public interface IDocumentRepository /// the document types /// Task<(byte[] Content, string FileName, bool IsDocumentTypeMatch, MediaTypeId MediaTypeId)> GetDocumentAsync(Guid documentId, IEnumerable documentTypeIds); + + Task AgreementIds, IEnumerable OfferIds)>> GetDocumentDataForCleanup(DateTimeOffset dateCreated); } diff --git a/src/portalbackend/PortalBackend.DBAccess/Repositories/ProcessStepRepository.cs b/src/portalbackend/PortalBackend.DBAccess/Repositories/ProcessStepRepository.cs index af4077983e..7c53f6d24d 100644 --- a/src/portalbackend/PortalBackend.DBAccess/Repositories/ProcessStepRepository.cs +++ b/src/portalbackend/PortalBackend.DBAccess/Repositories/ProcessStepRepository.cs @@ -26,36 +26,25 @@ namespace Org.Eclipse.TractusX.Portal.Backend.PortalBackend.DBAccess.Repositories; -public class ProcessStepRepository : IProcessStepRepository +public class ProcessStepRepository(PortalDbContext dbContext) : IProcessStepRepository { - private readonly PortalDbContext _context; - - /// - /// Constructor - /// - /// PortalDb context. - public ProcessStepRepository(PortalDbContext portalDbContext) - { - _context = portalDbContext; - } - public Process CreateProcess(ProcessTypeId processTypeId) => - _context.Add(new Process(Guid.NewGuid(), processTypeId, Guid.NewGuid())).Entity; + dbContext.Add(new Process(Guid.NewGuid(), processTypeId, Guid.NewGuid())).Entity; public IEnumerable CreateProcessRange(IEnumerable processTypeIds) { var processes = processTypeIds.Select(x => new Process(Guid.NewGuid(), x, Guid.NewGuid())).ToImmutableList(); - _context.AddRange(processes); + dbContext.AddRange(processes); return processes; } public ProcessStep CreateProcessStep(ProcessStepTypeId processStepTypeId, ProcessStepStatusId processStepStatusId, Guid processId) => - _context.Add(new ProcessStep(Guid.NewGuid(), processStepTypeId, processStepStatusId, processId, DateTimeOffset.UtcNow)).Entity; + dbContext.Add(new ProcessStep(Guid.NewGuid(), processStepTypeId, processStepStatusId, processId, DateTimeOffset.UtcNow)).Entity; public IEnumerable CreateProcessStepRange(IEnumerable<(ProcessStepTypeId ProcessStepTypeId, ProcessStepStatusId ProcessStepStatusId, Guid ProcessId)> processStepTypeStatus) { var processSteps = processStepTypeStatus.Select(x => new ProcessStep(Guid.NewGuid(), x.ProcessStepTypeId, x.ProcessStepStatusId, x.ProcessId, DateTimeOffset.UtcNow)).ToImmutableList(); - _context.AddRange(processSteps); + dbContext.AddRange(processSteps); return processSteps; } @@ -63,7 +52,7 @@ public void AttachAndModifyProcessStep(Guid processStepId, Action? { var step = new ProcessStep(processStepId, default, default, Guid.Empty, default); initialize?.Invoke(step); - _context.Attach(step); + dbContext.Attach(step); step.DateLastChanged = DateTimeOffset.UtcNow; modify(step); } @@ -76,7 +65,7 @@ public void AttachAndModifyProcessSteps(IEnumerable<(Guid ProcessStepId, Action< data.Initialize?.Invoke(step); return (Step: step, data.Modify); }).ToImmutableList(); - _context.AttachRange(stepModifyData.Select(data => data.Step)); + dbContext.AttachRange(stepModifyData.Select(data => data.Step)); stepModifyData.ForEach(data => { data.Step.DateLastChanged = DateTimeOffset.UtcNow; @@ -85,7 +74,7 @@ public void AttachAndModifyProcessSteps(IEnumerable<(Guid ProcessStepId, Action< } public IAsyncEnumerable GetActiveProcesses(IEnumerable processTypeIds, IEnumerable processStepTypeIds, DateTimeOffset lockExpiryDate) => - _context.Processes + dbContext.Processes .AsNoTracking() .Where(process => processTypeIds.Contains(process.ProcessTypeId) && @@ -94,8 +83,7 @@ public IAsyncEnumerable GetActiveProcesses(IEnumerable p .AsAsyncEnumerable(); public IAsyncEnumerable<(Guid ProcessStepId, ProcessStepTypeId ProcessStepTypeId)> GetProcessStepData(Guid processId) => - _context.ProcessSteps - .AsNoTracking() + dbContext.ProcessSteps .Where(step => step.ProcessId == processId && step.ProcessStepStatusId == ProcessStepStatusId.TODO) @@ -107,8 +95,7 @@ public IAsyncEnumerable GetActiveProcesses(IEnumerable p .AsAsyncEnumerable(); public Task<(bool ProcessExists, VerifyProcessData ProcessData)> IsValidProcess(Guid processId, ProcessTypeId processTypeId, IEnumerable processStepTypeIds) => - _context.Processes - .AsNoTracking() + dbContext.Processes .Where(x => x.Id == processId && x.ProcessTypeId == processTypeId) .Select(x => new ValueTuple( true, @@ -122,8 +109,7 @@ public IAsyncEnumerable GetActiveProcesses(IEnumerable p .SingleOrDefaultAsync(); public Task<(ProcessTypeId ProcessTypeId, VerifyProcessData ProcessData, Guid? ServiceAccountId, Guid? ServiceAccountVersion)> GetProcessDataForServiceAccountCallback(Guid processId, IEnumerable processStepTypeIds) => - _context.Processes - .AsNoTracking() + dbContext.Processes .Where(x => x.Id == processId) .Select(x => new ValueTuple( x.ProcessTypeId, @@ -139,8 +125,7 @@ public IAsyncEnumerable GetActiveProcesses(IEnumerable p .SingleOrDefaultAsync(); public Task<(ProcessTypeId ProcessTypeId, VerifyProcessData ProcessData, Guid? ServiceAccountId)> GetProcessDataForServiceAccountDeletionCallback(Guid processId, IEnumerable? processStepTypeIds) => - _context.Processes - .AsNoTracking() + dbContext.Processes .Where(x => x.Id == processId && x.ProcessTypeId == ProcessTypeId.DIM_TECHNICAL_USER) .Select(x => new ValueTuple( x.ProcessTypeId, diff --git a/src/processes/Processes.Worker/appsettings.json b/src/processes/Processes.Worker/appsettings.json index 5e3fa7f8db..e09d4daebc 100644 --- a/src/processes/Processes.Worker/appsettings.json +++ b/src/processes/Processes.Worker/appsettings.json @@ -344,7 +344,8 @@ "Scope": "", "TokenAddress": "", "BaseAddress": "", - "UseDimWallet": false + "UseDimWallet": false, + "RetriggerEndClearinghouseIntervalInDays": 30 }, "Dim": { "Username": "", diff --git a/tests/administration/Administration.Service.Tests/BusinessLogic/RegistrationBusinessLogicTest.cs b/tests/administration/Administration.Service.Tests/BusinessLogic/RegistrationBusinessLogicTest.cs index fa6e6a68e2..d49c8cbc22 100644 --- a/tests/administration/Administration.Service.Tests/BusinessLogic/RegistrationBusinessLogicTest.cs +++ b/tests/administration/Administration.Service.Tests/BusinessLogic/RegistrationBusinessLogicTest.cs @@ -205,7 +205,7 @@ public async Task GetCompanyApplicationDetailsAsync_WithClosedRequest_GetsExpect public async Task GetOspCompanyApplicationDetailsAsync_WithDefaultRequest_GetsExpectedEntries(CompanyApplicationStatusFilter? statusFilter, DateCreatedOrderFilter? dateCreatedOrderFilter) { // Arrange - var companyName = _fixture.Create(); + var companyName = "TestCompany"; var externalId = _fixture.Create(); var data = _fixture.CreateMany<(Guid Id, Guid CompanyId, CompanyApplicationStatusId CompanyApplicationStatusId, DateTimeOffset Created)>(10) .Select(x => new CompanyApplication(x.Id, x.CompanyId, x.CompanyApplicationStatusId, CompanyApplicationTypeId.EXTERNAL, x.Created) diff --git a/tests/administration/Administration.Service.Tests/appsettings.IntegrationTests.json b/tests/administration/Administration.Service.Tests/appsettings.IntegrationTests.json index 632ec19dfa..1b975d194e 100644 --- a/tests/administration/Administration.Service.Tests/appsettings.IntegrationTests.json +++ b/tests/administration/Administration.Service.Tests/appsettings.IntegrationTests.json @@ -313,7 +313,9 @@ "ClientSecret": "test", "Scope": "test", "TokenAddress": "test", - "BaseAddress": "test" + "BaseAddress": "test", + "UseDimWallet": false, + "RetriggerEndClearinghouseIntervalInDays": 30 }, "SdFactory":{ "Username": "test", diff --git a/tests/externalsystems/Clearinghouse.Library.Tests/ClearinghouseBusinessLogicTests.cs b/tests/externalsystems/Clearinghouse.Library.Tests/ClearinghouseBusinessLogicTests.cs index 1d89729957..6f55798b1d 100644 --- a/tests/externalsystems/Clearinghouse.Library.Tests/ClearinghouseBusinessLogicTests.cs +++ b/tests/externalsystems/Clearinghouse.Library.Tests/ClearinghouseBusinessLogicTests.cs @@ -22,6 +22,7 @@ using Org.Eclipse.TractusX.Portal.Backend.Clearinghouse.Library.Models; using Org.Eclipse.TractusX.Portal.Backend.Custodian.Library.BusinessLogic; using Org.Eclipse.TractusX.Portal.Backend.Custodian.Library.Models; +using Org.Eclipse.TractusX.Portal.Backend.Framework.DateTimeProvider; using Org.Eclipse.TractusX.Portal.Backend.Framework.ErrorHandling; using Org.Eclipse.TractusX.Portal.Backend.PortalBackend.DBAccess; using Org.Eclipse.TractusX.Portal.Backend.PortalBackend.DBAccess.Models; @@ -52,6 +53,8 @@ public class ClearinghouseBusinessLogicTests private readonly IClearinghouseService _clearinghouseService; private readonly IApplicationChecklistService _checklistService; private readonly ICustodianBusinessLogic _custodianBusinessLogic; + private readonly IDateTimeProvider _dateTimeProvider; + private readonly IApplicationChecklistRepository _applicationChecklistRepository; public ClearinghouseBusinessLogicTests() { @@ -61,14 +64,19 @@ public ClearinghouseBusinessLogicTests() _fixture.Behaviors.Add(new OmitOnRecursionBehavior()); _applicationRepository = A.Fake(); + _applicationChecklistRepository = A.Fake(); _portalRepositories = A.Fake(); _clearinghouseService = A.Fake(); _custodianBusinessLogic = A.Fake(); _checklistService = A.Fake(); A.CallTo(() => _portalRepositories.GetInstance()).Returns(_applicationRepository); + A.CallTo(() => _portalRepositories.GetInstance()).Returns(_applicationChecklistRepository); - _logic = new ClearinghouseBusinessLogic(_portalRepositories, _clearinghouseService, _custodianBusinessLogic, _checklistService, Options.Create(new ClearinghouseSettings + _dateTimeProvider = A.Fake(); + A.CallTo(() => _dateTimeProvider.OffsetNow).Returns(DateTimeOffset.UtcNow); + + _logic = new ClearinghouseBusinessLogic(_portalRepositories, _clearinghouseService, _custodianBusinessLogic, _checklistService, _dateTimeProvider, Options.Create(new ClearinghouseSettings { CallbackUrl = "https://api.com", UseDimWallet = false @@ -226,7 +234,7 @@ public async Task HandleStartClearingHouse_WithDimActiveAndNonExistingApplicatio var context = new IApplicationChecklistService.WorkerChecklistProcessStepData(IdWithBpn, ProcessStepTypeId.START_CLEARING_HOUSE, checklist, Enumerable.Empty()); A.CallTo(() => _applicationRepository.GetDidForApplicationId(A._)) .Returns<(bool, string?)>(default); - var logic = new ClearinghouseBusinessLogic(_portalRepositories, _clearinghouseService, _custodianBusinessLogic, _checklistService, Options.Create(new ClearinghouseSettings + var logic = new ClearinghouseBusinessLogic(_portalRepositories, _clearinghouseService, _custodianBusinessLogic, _checklistService, _dateTimeProvider, Options.Create(new ClearinghouseSettings { CallbackUrl = "https://api.com", UseDimWallet = true @@ -257,7 +265,7 @@ public async Task HandleStartClearingHouse_WithDimActiveAndDidNotSet_ThrowsConfl var context = new IApplicationChecklistService.WorkerChecklistProcessStepData(IdWithBpn, ProcessStepTypeId.START_CLEARING_HOUSE, checklist, Enumerable.Empty()); A.CallTo(() => _applicationRepository.GetDidForApplicationId(A._)) .Returns((true, null)); - var logic = new ClearinghouseBusinessLogic(_portalRepositories, _clearinghouseService, _custodianBusinessLogic, _checklistService, Options.Create(new ClearinghouseSettings + var logic = new ClearinghouseBusinessLogic(_portalRepositories, _clearinghouseService, _custodianBusinessLogic, _checklistService, _dateTimeProvider, Options.Create(new ClearinghouseSettings { CallbackUrl = "https://api.com", UseDimWallet = true @@ -290,7 +298,7 @@ public async Task HandleStartClearingHouse_WithDimActive_CallsExpected() A.CallTo(() => _applicationRepository.GetDidForApplicationId(A._)) .Returns((true, "did:web:test123456")); SetupForHandleStartClearingHouse(); - var logic = new ClearinghouseBusinessLogic(_portalRepositories, _clearinghouseService, _custodianBusinessLogic, _checklistService, Options.Create(new ClearinghouseSettings + var logic = new ClearinghouseBusinessLogic(_portalRepositories, _clearinghouseService, _custodianBusinessLogic, _checklistService, _dateTimeProvider, Options.Create(new ClearinghouseSettings { CallbackUrl = "https://api.com", UseDimWallet = true @@ -361,6 +369,29 @@ public async Task ProcessClearinghouseResponseAsync_WithDecline_UpdatesEntry() #endregion + #region CheckEndClearinghouseProcesses + + [Fact] + public async Task CheckEndClearinghouseProcesses_WithEntry_CreatesProcessStep() + { + // Arrange + var entry = new ApplicationChecklistEntry(IdWithBpn, ApplicationChecklistEntryTypeId.IDENTITY_WALLET, ApplicationChecklistEntryStatusId.TO_DO, DateTimeOffset.UtcNow); + A.CallTo(() => _applicationChecklistRepository.GetApplicationsForClearinghouseRetrigger(A._)) + .Returns(Enumerable.Repeat(IdWithBpn, 1).ToAsyncEnumerable()); + SetupForCheckEndClearinghouseProcesses(Enumerable.Repeat(entry, 1)); + + // Act + await _logic.CheckEndClearinghouseProcesses(CancellationToken.None); + + // Assert + A.CallTo(() => _checklistService.FinalizeChecklistEntryAndProcessSteps(A._, A>._, A>._, A>.That.Matches(x => x.Count(y => y == ProcessStepTypeId.START_OVERRIDE_CLEARING_HOUSE) == 1))).MustHaveHappenedOnceExactly(); + A.CallTo(() => _portalRepositories.SaveAsync()).MustHaveHappenedOnceExactly(); + entry.Comment.Should().Be("Reset to retrigger clearinghouse"); + entry.ApplicationChecklistEntryStatusId.Should().Be(ApplicationChecklistEntryStatusId.TO_DO); + } + + #endregion + #region Setup private void SetupForHandleStartClearingHouse() @@ -419,5 +450,39 @@ private void SetupForProcessClearinghouseResponse(ApplicationChecklistEntry appl .Returns(new IApplicationChecklistService.ManualChecklistProcessStepData(Guid.Empty, new Process(Guid.NewGuid(), ProcessTypeId.APPLICATION_CHECKLIST, Guid.NewGuid()), Guid.Empty, ApplicationChecklistEntryTypeId.CLEARING_HOUSE, ImmutableDictionary.Empty, new List())); } + private void SetupForCheckEndClearinghouseProcesses(IEnumerable applicationChecklistEntries) + { + A.CallTo(() => _checklistService.FinalizeChecklistEntryAndProcessSteps(A._, A>._, A>._, A>._)) + .Invokes((IApplicationChecklistService.ManualChecklistProcessStepData data, Action? _, Action modifyApplicationChecklistEntry, IEnumerable _) => + { + var entry = applicationChecklistEntries.SingleOrDefault(x => x.ApplicationId == data.ApplicationId); + if (entry == null) + return; + + entry.DateLastChanged = DateTimeOffset.UtcNow; + modifyApplicationChecklistEntry(entry); + }); + + A.CallTo(() => _checklistService.VerifyChecklistEntryAndProcessSteps( + A._, + ApplicationChecklistEntryTypeId.CLEARING_HOUSE, + A>._, + ProcessStepTypeId.END_CLEARING_HOUSE, + A?>._, + A?>._)) + .ReturnsLazily((Guid id, + ApplicationChecklistEntryTypeId _, + IEnumerable _, + ProcessStepTypeId _, + IEnumerable? _, + IEnumerable? _) => new IApplicationChecklistService.ManualChecklistProcessStepData( + id, + new Process(Guid.NewGuid(), ProcessTypeId.APPLICATION_CHECKLIST, Guid.NewGuid()), + Guid.Empty, + ApplicationChecklistEntryTypeId.CLEARING_HOUSE, + ImmutableDictionary.Empty, + new List())); + } + #endregion } diff --git a/tests/maintenance/Maintenance.App.Tests/BatchDeleteServiceTests.cs b/tests/maintenance/Maintenance.App.Tests/BatchDeleteServiceTests.cs index fb3502cc37..6837e18a41 100644 --- a/tests/maintenance/Maintenance.App.Tests/BatchDeleteServiceTests.cs +++ b/tests/maintenance/Maintenance.App.Tests/BatchDeleteServiceTests.cs @@ -1,86 +1,110 @@ -/******************************************************************************** - * Copyright (c) 2022 Contributors to the Eclipse Foundation - * - * See the NOTICE file(s) distributed with this work for additional - * information regarding copyright ownership. - * - * This program and the accompanying materials are made available under the - * terms of the Apache License, Version 2.0 which is available at - * https://www.apache.org/licenses/LICENSE-2.0. - * - * Unless required by applicable law or agreed to in writing, software - * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT - * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the - * License for the specific language governing permissions and limitations - * under the License. - * - * SPDX-License-Identifier: Apache-2.0 - ********************************************************************************/ - -using FluentAssertions; -using Microsoft.Extensions.Configuration; -using Microsoft.Extensions.DependencyInjection; -using Microsoft.Extensions.Hosting; using Microsoft.Extensions.Logging; -using Org.Eclipse.TractusX.Portal.Backend.Maintenance.App; -using Org.Eclipse.TractusX.Portal.Backend.PortalBackend.Maintenance.App.Tests.Setup; -using Org.Eclipse.TractusX.Portal.Backend.PortalBackend.PortalEntities; -using Xunit.Extensions.AssemblyFixture; +using Microsoft.Extensions.Options; +using Org.Eclipse.TractusX.Portal.Backend.Framework.DateTimeProvider; +using Org.Eclipse.TractusX.Portal.Backend.Framework.Tests.Shared; +using Org.Eclipse.TractusX.Portal.Backend.Maintenance.App.DependencyInjection; +using Org.Eclipse.TractusX.Portal.Backend.Maintenance.App.Services; +using Org.Eclipse.TractusX.Portal.Backend.PortalBackend.DBAccess; +using Org.Eclipse.TractusX.Portal.Backend.PortalBackend.DBAccess.Repositories; +using Org.Eclipse.TractusX.Portal.Backend.PortalBackend.PortalEntities.Entities; namespace Org.Eclipse.TractusX.Portal.Backend.PortalBackend.Maintenance.App.Tests; -/// -/// Tests the functionality of the -/// -public class BatchDeleteServiceTests : IAssemblyFixture +public class BatchDeleteServiceTests { - private readonly TestDbFixture _dbTestDbFixture; - private readonly IFixture _fixture; + private readonly IPortalRepositories _portalRepositories; + private readonly IDocumentRepository _documentRepository; + private readonly BatchDeleteService _sut; + private readonly IMockLogger _mockLogger; -#pragma warning disable xUnit1041 - public BatchDeleteServiceTests(TestDbFixture testDbFixture) -#pragma warning restore xUnit1041 + public BatchDeleteServiceTests() { - _fixture = new Fixture().Customize(new AutoFakeItEasyCustomization { ConfigureMembers = true }); - _fixture.Behaviors.OfType().ToList() - .ForEach(b => _fixture.Behaviors.Remove(b)); + var fixture = new Fixture().Customize(new AutoFakeItEasyCustomization { ConfigureMembers = true }); + fixture.Behaviors.OfType().ToList() + .ForEach(b => fixture.Behaviors.Remove(b)); + fixture.Behaviors.Add(new OmitOnRecursionBehavior()); - _fixture.Behaviors.Add(new OmitOnRecursionBehavior()); - _dbTestDbFixture = testDbFixture; + _mockLogger = A.Fake>(); + ILogger logger = new MockLogger(_mockLogger); + + _portalRepositories = A.Fake(); + _documentRepository = A.Fake(); + + A.CallTo(() => _portalRepositories.GetInstance()).Returns(_documentRepository); + + var dateTimeProvider = A.Fake(); + A.CallTo(() => dateTimeProvider.OffsetNow).Returns(DateTimeOffset.UtcNow); + + var options = Options.Create(new BatchDeleteServiceSettings { DeleteIntervalInDays = 5 }); + _sut = new BatchDeleteService(logger, options, _portalRepositories, dateTimeProvider); } [Fact] - public async Task ExecuteAsync_WithOldDocumentsAndAssigned_Removes() + public async Task CleanupDocuments_WithoutMatchingDocuments_DoesNothing() { // Arrange - var sut = await CreateSut(); + A.CallTo(() => _documentRepository.GetDocumentDataForCleanup(A._)) + .Returns([]); // Act - await sut.StartAsync(CancellationToken.None); + await _sut.CleanupDocuments(CancellationToken.None); // Assert - true.Should().BeTrue(); + A.CallTo(() => _portalRepositories.SaveAsync()) + .MustNotHaveHappened(); + A.CallTo(() => _portalRepositories.AttachRange(A>._)) + .MustNotHaveHappened(); + A.CallTo(() => _documentRepository.RemoveDocuments(A>._)) + .MustNotHaveHappened(); + A.CallTo(() => _documentRepository.RemoveOfferAssignedDocuments(A>._)) + .MustNotHaveHappened(); } - private async Task CreateSut() + [Fact] + public async Task CleanupDocuments_WithMatchingDocuments_RemovesExpected() { - var hostApplicationLifetime = _fixture.Create(); - var inMemorySettings = new[] - { - KeyValuePair.Create("DeleteIntervalInDays", "5") - }; - var config = new ConfigurationBuilder() - .AddInMemoryCollection(inMemorySettings) - .Build(); - var context = await _dbTestDbFixture.GetPortalDbContext(); - - var serviceProvider = _fixture.Create(); - A.CallTo(() => serviceProvider.GetService(typeof(PortalDbContext))).Returns(context); - var serviceScope = _fixture.Create(); - A.CallTo(() => serviceScope.ServiceProvider).Returns(serviceProvider); - var serviceScopeFactory = _fixture.Create(); - A.CallTo(() => serviceScopeFactory.CreateScope()).Returns(serviceScope); - - return new BatchDeleteService(hostApplicationLifetime, serviceScopeFactory, _fixture.Create>(), config); + // Arrange + var documentId = Guid.NewGuid(); + var agreementId1 = Guid.NewGuid(); + var agreementId2 = Guid.NewGuid(); + var offerId1 = Guid.NewGuid(); + var offerId2 = Guid.NewGuid(); + A.CallTo(() => _documentRepository.GetDocumentDataForCleanup(A._)) + .Returns([new(documentId, new[] { agreementId1, agreementId2 }, new[] { offerId1, offerId2 })]); + + // Act + await _sut.CleanupDocuments(CancellationToken.None); + + // Assert + A.CallTo(() => _portalRepositories.SaveAsync()) + .MustHaveHappenedOnceExactly(); + A.CallTo(() => _portalRepositories.AttachRange(A>.That.Matches(x => x.Count(a => a.Id == agreementId1) == 1 && x.Count(a => a.Id == agreementId2) == 1))) + .MustHaveHappenedOnceExactly(); + A.CallTo(() => _documentRepository.RemoveDocuments(A>.That.Matches(x => x.Single() == documentId))) + .MustHaveHappenedOnceExactly(); + A.CallTo(() => _documentRepository.RemoveOfferAssignedDocuments(A>.That.Matches(x => x.Count(a => a.OfferId == offerId1 && a.DocumentId == documentId) == 1 && x.Count(a => a.OfferId == offerId2 && a.DocumentId == documentId) == 1))) + .MustHaveHappenedOnceExactly(); + } + + [Fact] + public async Task CleanupDocuments_WithException_LogsException() + { + // Arrange + A.CallTo(() => _documentRepository.GetDocumentDataForCleanup(A._)) + .Throws(new DataMisalignedException("Test message")); + + // Act + await _sut.CleanupDocuments(CancellationToken.None); + + // Assert + A.CallTo(() => _portalRepositories.SaveAsync()) + .MustNotHaveHappened(); + A.CallTo(() => _portalRepositories.AttachRange(A>._)) + .MustNotHaveHappened(); + A.CallTo(() => _documentRepository.RemoveDocuments(A>._)) + .MustNotHaveHappened(); + A.CallTo(() => _documentRepository.RemoveOfferAssignedDocuments(A>._)) + .MustNotHaveHappened(); + A.CallTo(() => _mockLogger.Log(A.That.IsEqualTo(LogLevel.Error), A._, A._)).MustHaveHappenedOnceExactly(); } } diff --git a/tests/maintenance/Maintenance.App.Tests/Maintenance.App.Tests.csproj b/tests/maintenance/Maintenance.App.Tests/Maintenance.App.Tests.csproj index d877c564ba..b5be8c3945 100644 --- a/tests/maintenance/Maintenance.App.Tests/Maintenance.App.Tests.csproj +++ b/tests/maintenance/Maintenance.App.Tests/Maintenance.App.Tests.csproj @@ -21,6 +21,7 @@ Org.Eclipse.TractusX.Portal.Backend.PortalBackend.Maintenance.App.Tests + Org.Eclipse.TractusX.Portal.Backend.PortalBackend.Maintenance.App.Tests net8.0 enable enable @@ -54,4 +55,12 @@ + + + true + PreserveNewest + PreserveNewest + + + diff --git a/tests/maintenance/Maintenance.App.Tests/MaintenanceServiceTests.cs b/tests/maintenance/Maintenance.App.Tests/MaintenanceServiceTests.cs new file mode 100644 index 0000000000..3265313e87 --- /dev/null +++ b/tests/maintenance/Maintenance.App.Tests/MaintenanceServiceTests.cs @@ -0,0 +1,72 @@ +/******************************************************************************** + * Copyright (c) 2022 Contributors to the Eclipse Foundation + * + * See the NOTICE file(s) distributed with this work for additional + * information regarding copyright ownership. + * + * This program and the accompanying materials are made available under the + * terms of the Apache License, Version 2.0 which is available at + * https://www.apache.org/licenses/LICENSE-2.0. + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT + * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the + * License for the specific language governing permissions and limitations + * under the License. + * + * SPDX-License-Identifier: Apache-2.0 + ********************************************************************************/ + +using Microsoft.Extensions.DependencyInjection; +using Org.Eclipse.TractusX.Portal.Backend.Clearinghouse.Library.BusinessLogic; +using Org.Eclipse.TractusX.Portal.Backend.Maintenance.App.Services; +using Org.Eclipse.TractusX.Portal.Backend.Processes.ProcessIdentity; + +namespace Org.Eclipse.TractusX.Portal.Backend.PortalBackend.Maintenance.App.Tests; + +public class MaintenanceServiceTests +{ + private readonly IBatchDeleteService _batchDeleteService; + private readonly IClearinghouseBusinessLogic _clearinghouseBusinessLogic; + private readonly MaintenanceService _service; + private readonly IProcessIdentityDataDetermination _processIdentityDataDetermination; + + public MaintenanceServiceTests() + { + var fixture = new Fixture().Customize(new AutoFakeItEasyCustomization { ConfigureMembers = true }); + fixture.Behaviors.OfType().ToList() + .ForEach(b => fixture.Behaviors.Remove(b)); + fixture.Behaviors.Add(new OmitOnRecursionBehavior()); + + _batchDeleteService = A.Fake(); + _clearinghouseBusinessLogic = A.Fake(); + _processIdentityDataDetermination = A.Fake(); + + var serviceProvider = A.Fake(); + A.CallTo(() => serviceProvider.GetService(typeof(IBatchDeleteService))).Returns(_batchDeleteService); + A.CallTo(() => serviceProvider.GetService(typeof(IClearinghouseBusinessLogic))).Returns(_clearinghouseBusinessLogic); + A.CallTo(() => serviceProvider.GetService(typeof(IProcessIdentityDataDetermination))).Returns(_processIdentityDataDetermination); + var serviceScope = A.Fake(); + A.CallTo(() => serviceScope.ServiceProvider).Returns(serviceProvider); + var serviceScopeFactory = A.Fake(); + A.CallTo(() => serviceScopeFactory.CreateScope()).Returns(serviceScope); + A.CallTo(() => serviceProvider.GetService(typeof(IServiceScopeFactory))).Returns(serviceScopeFactory); + + _service = new MaintenanceService(serviceScopeFactory); + } + + [Fact] + public async Task ExecuteAsync_CallsExpectedServices() + { + // Act + await _service.ExecuteAsync(CancellationToken.None); + + // Assert + A.CallTo(() => _processIdentityDataDetermination.GetIdentityData()) + .MustHaveHappenedOnceExactly(); + A.CallTo(() => _batchDeleteService.CleanupDocuments(A._)) + .MustHaveHappenedOnceExactly(); + A.CallTo(() => _clearinghouseBusinessLogic.CheckEndClearinghouseProcesses(A._)) + .MustHaveHappenedOnceExactly(); + } +} diff --git a/tests/maintenance/Maintenance.App.Tests/appsettings.IntegrationTests.json b/tests/maintenance/Maintenance.App.Tests/appsettings.IntegrationTests.json new file mode 100644 index 0000000000..5af5539af9 --- /dev/null +++ b/tests/maintenance/Maintenance.App.Tests/appsettings.IntegrationTests.json @@ -0,0 +1,44 @@ +{ + "Serilog": { + "Using": [ "Serilog.Sinks.Console" ], + "MinimumLevel": { + "Default": "Information", + "Override": { + "Microsoft": "Warning", + "System": "Information", + "Microsoft.Hosting.Lifetime": "Information", + "Org.Eclipse.TractusX.Portal.Backend": "Information" + } + }, + "WriteTo": [ + { "Name": "Console" } + ], + "Enrich": [ + "FromLogContext" + ], + "Properties": { + "Application": "Org.Eclipse.TractusX.Portal.Backend.Maintenance.App" + } + }, + "ConnectionStrings": { + "PortalDb": "Server=placeholder;Database=placeholder;Port=5432;User Id=placeholder;Password=placeholder;Ssl Mode=Disable;" + }, + "BatchDelete": { + "DeleteIntervalInDays": 5 + }, + "Clearinghouse": { + "Username": "empty", + "Password": "empty", + "ClientId": "empty", + "GrantType": "empty", + "ClientSecret": "empty", + "Scope": "openid", + "TokenAddress": "https://example.org/", + "BaseAddress": "https://validation.dev.dih-cloud.com/api/v1", + "UseDimWallet": false, + "RetriggerEndClearinghouseIntervalInDays": 30 + }, + "ProcessIdentity": { + "ProcessUserId": "" + } +}