// Copyright (C) Stichting Deltares 2017. All rights reserved. // // This file is part of Ringtoets. // // Ringtoets is free software: you can redistribute it and/or modify // it under the terms of the GNU General Public License as published by // the Free Software Foundation, either version 3 of the License, or // (at your option) any later version. // // This program is distributed in the hope that it will be useful, // but WITHOUT ANY WARRANTY; without even the implied warranty of // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the // GNU General Public License for more details. // // You should have received a copy of the GNU General Public License // along with this program. If not, see . // // All names, logos, and references to "Deltares" are registered trademarks of // Stichting Deltares and remain full property of Stichting Deltares at all times. // All rights reserved. using System; using System.Collections.Generic; using System.Globalization; using System.Linq; using System.Text.RegularExpressions; using Core.Common.Base; using Core.Common.Base.Geometry; using Core.Common.Util.Extensions; using log4net; using Ringtoets.Common.Data.Hydraulics; using Ringtoets.DuneErosion.Data; using Ringtoets.DuneErosion.IO; using Ringtoets.DuneErosion.Service.Properties; using DuneErosionDataResources = Ringtoets.DuneErosion.Data.Properties.Resources; namespace Ringtoets.DuneErosion.Service { /// /// Service for synchronizing dune erosion data. /// public static class DuneErosionDataSynchronizationService { private static readonly ILog log = LogManager.GetLogger(typeof(DuneErosionDataSynchronizationService)); /// /// Sets based upon /// the . /// /// The to update. /// The hydraulic boundary locations to use. /// The dune locations to use. /// Thrown when any parameter is null. public static void SetDuneLocations(DuneErosionFailureMechanism failureMechanism, ICollection hydraulicBoundaryLocations, IEnumerable duneLocations) { if (failureMechanism == null) { throw new ArgumentNullException(nameof(failureMechanism)); } if (hydraulicBoundaryLocations == null) { throw new ArgumentNullException(nameof(hydraulicBoundaryLocations)); } if (duneLocations == null) { throw new ArgumentNullException(nameof(duneLocations)); } if (hydraulicBoundaryLocations.Count == 0 || !duneLocations.Any()) { failureMechanism.SetDuneLocations(Enumerable.Empty()); return; } var correspondingDuneLocations = new List(); foreach (ReadDuneLocation readDuneLocation in duneLocations) { HydraulicBoundaryLocation correspondingHydraulicBoundaryLocation = hydraulicBoundaryLocations .FirstOrDefault(hbl => DoesHydraulicBoundaryLocationMatchWithDuneLocation(hbl, readDuneLocation)); if (correspondingHydraulicBoundaryLocation != null) { var duneLocation = new DuneLocation(correspondingHydraulicBoundaryLocation.Id, readDuneLocation.Name, readDuneLocation.Location, new DuneLocation.ConstructionProperties { CoastalAreaId = readDuneLocation.CoastalAreaId, Offset = readDuneLocation.Offset, Orientation = readDuneLocation.Orientation, D50 = readDuneLocation.D50 }); correspondingDuneLocations.Add(duneLocation); } } failureMechanism.SetDuneLocations(correspondingDuneLocations); } /// /// Clears the output of the dune location calculations within the dune erosion failure mechanism. /// /// The failure mechanism for which the output of the calculations needs to be cleared. /// All objects changed during the clear. /// Thrown when is null. public static IEnumerable ClearDuneLocationCalculationOutput(DuneErosionFailureMechanism failureMechanism) { if (failureMechanism == null) { throw new ArgumentNullException(nameof(failureMechanism)); } var affectedCalculations = new List(); affectedCalculations.AddRange(ClearDuneLocationCalculationsOutput(failureMechanism.CalculationsForMechanismSpecificFactorizedSignalingNorm)); affectedCalculations.AddRange(ClearDuneLocationCalculationsOutput(failureMechanism.CalculationsForMechanismSpecificSignalingNorm)); affectedCalculations.AddRange(ClearDuneLocationCalculationsOutput(failureMechanism.CalculationsForMechanismSpecificLowerLimitNorm)); affectedCalculations.AddRange(ClearDuneLocationCalculationsOutput(failureMechanism.CalculationsForLowerLimitNorm)); affectedCalculations.AddRange(ClearDuneLocationCalculationsOutput(failureMechanism.CalculationsForFactorizedLowerLimitNorm)); return affectedCalculations; } private static IEnumerable ClearDuneLocationCalculationsOutput(IEnumerable calculations) { IEnumerable affectedCalculations = calculations.Where(c => c.Output != null).ToArray(); affectedCalculations.ForEachElementDo(c => c.Output = null); return affectedCalculations; } private static bool DoesHydraulicBoundaryLocationMatchWithDuneLocation(HydraulicBoundaryLocation hydraulicBoundaryLocation, ReadDuneLocation readDuneLocation) { if (!Math2D.AreEqualPoints(hydraulicBoundaryLocation.Location, readDuneLocation.Location)) { return false; } // Regex to search for a pattern like "__" // Only the last number is captured in a group called "Offset" // The last number can also contain decimals. var regex = new Regex(@"^(?:\S+)_(?:\d+)_(?(?:\d+\.)?\d+$)"); Match match = regex.Match(hydraulicBoundaryLocation.Name); if (!match.Success) { log.ErrorFormat(Resources.DuneErosionDataSynchronizationService_SetDuneLocations_Location_0_is_dune_location_but_name_is_not_according_format, hydraulicBoundaryLocation.Name); return false; } string duneLocationOffset = readDuneLocation.Offset.ToString(DuneErosionDataResources.DuneLocation_Offset_format, CultureInfo.InvariantCulture); return match.Groups["Offset"].Value == duneLocationOffset; } } }