// Copyright (C) Stichting Deltares 2024. All rights reserved.
//
// This file is part of the Dam Engine.
//
// The Dam Engine is free software: you can redistribute it and/or modify
// it under the terms of the GNU Affero 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 Affero General Public License for more details.
//
// You should have received a copy of the GNU Affero 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.IO;
using System.Linq;
using Deltares.DamEngine.Data.General.PlLines;
using Deltares.DamEngine.Interface;
using Deltares.DamEngine.Io;
using Deltares.DamEngine.Io.XmlInput;
using Deltares.DamEngine.Io.XmlOutput;
using Deltares.DamEngine.TestHelpers;
using Deltares.MacroStability.Io;
using Deltares.MacroStability.Io.XmlInput;
using KellermanSoftware.CompareNetObjects;
using NUnit.Framework;
using CharacteristicPointType = Deltares.DamEngine.Data.Geotechnics.CharacteristicPointType;
using SurfaceLinePoint = Deltares.DamEngine.Io.XmlInput.SurfaceLinePoint;
using WaternetLineType = Deltares.MacroStability.Io.XmlInput.WaternetLineType;
using WaternetType = Deltares.MacroStability.Io.XmlInput.WaternetType;
namespace Deltares.DamEngine.IntegrationTests.IntegrationTests;
[TestFixture]
public class OperationalStabilityProfile2DTests
{
private const double tolerance = 0.0000001;
private const uint sourceTypeLocationData = 1;
private const uint sourceTypeSensor = 2;
private const string projectPath = "Profile2DTests";
private const string testFilesLocation = @".\TestFiles\Operational\Profile2DTests\";
private const string inputFilename = testFilesLocation + "CalculateStabilityInside2DInput.xml";
private const string outputFilename = testFilesLocation + "CalculateStabilityInside2DOutput.xml";
// Characteristic points
private SurfaceLinePoint outsideLimit;
private SurfaceLinePoint dikeTopAtRiver;
private SurfaceLinePoint dikeTopAtPolder;
private SurfaceLinePoint shoulderInside;
private SurfaceLinePoint dikeToeAtPolder;
private SurfaceLinePoint insideLimit;
///
/// The following test is based on the DamLive test Deltares.DamLive.Tests.StabilityInside2DTest.
/// Two soil profiles (with and without in-between aquifer) are tested for 2 time steps, so 4 locations in total.
/// Two types of source data are tested: location data or sensor.
/// The expected Pl-lines are re-calculated using the input.
///
[Test]
[TestCase(false)]
[TestCase(true)]
public void GivenStabilityProjectWithTwoSoilProfiles2DAndTwoTimeSteps_WhenCalculating_ThenExpectedWaternetIsGeneratedPerProfileAndPerTimeStep(bool isUseLocationData)
{
// Setup
const string calcDir = $"DAMLive.Calc.";
if (Directory.Exists(calcDir))
{
Directory.Delete(calcDir, true); // delete previous results
}
Input input = DamXmlSerialization.LoadInputFromXmlFile(inputFilename);
input.CalculationMap = calcDir;
input.ProjectPath = projectPath;
FillSensorData(input, isUseLocationData);
string inputString = DamXmlSerialization.SaveInputAsXmlString(input);
DamXmlSerialization.SaveInputAsXmlFile(inputFilename, input);
Assert.That(input.Locations, Has.Length.EqualTo(4));
// Run
Output output = GeneralHelper.RunAfterInputValidation(inputString, true, outputFilename);
Assert.That(output, Is.Not.Null);
outsideLimit = input.SurfaceLines[0].Points.First(p => p.PointType == ConversionHelper.ConvertToInputPointType(CharacteristicPointType.SurfaceLevelOutside));
dikeTopAtRiver = input.SurfaceLines[0].Points.First(p => p.PointType == ConversionHelper.ConvertToInputPointType(CharacteristicPointType.DikeTopAtRiver));
dikeTopAtPolder = input.SurfaceLines[0].Points.First(p => p.PointType == ConversionHelper.ConvertToInputPointType(CharacteristicPointType.DikeTopAtPolder));
dikeToeAtPolder = input.SurfaceLines[0].Points.First(p => p.PointType == ConversionHelper.ConvertToInputPointType(CharacteristicPointType.DikeToeAtPolder));
shoulderInside = input.SurfaceLines[0].Points.First(p => p.PointType == ConversionHelper.ConvertToInputPointType(CharacteristicPointType.ShoulderBaseInside));
insideLimit = input.SurfaceLines[0].Points.First(p => p.PointType == ConversionHelper.ConvertToInputPointType(CharacteristicPointType.SurfaceLevelInside));
// Check the created PL lines for the 4 locations
for (var indexLocation = 0; indexLocation < 4; indexLocation++)
{
int indexTimeStep = indexLocation is 0 or 1 ? 0 : 1;
WaternetType waternet = ReadWaternetFromCreatedSkxFile(calcDir, indexTimeStep, indexLocation);
CheckPhreaticLine(waternet, input, indexLocation, indexTimeStep, isUseLocationData);
CheckPl3AndPl4(waternet, input, indexLocation, indexTimeStep, isUseLocationData);
}
// Check safety factors
Output expectedOutput = DamXmlSerialization.LoadOutputFromXmlFile(outputFilename);
Assert.That(expectedOutput, Is.Not.Null);
CompareOutput(expectedOutput, output);
}
private static void FillSensorData(Input input, bool isUseLocationData)
{
for (var indexSensorLocation = 0; indexSensorLocation < 2; indexSensorLocation++)
{
input.SensorData.SensorLocations[indexSensorLocation].SourceTypePl1WaterLevelAtRiver = isUseLocationData ? sourceTypeLocationData : sourceTypeSensor;
input.SensorData.SensorLocations[indexSensorLocation].SourceTypePl1PlLineOffsetBelowDikeTopAtRiver = isUseLocationData ? sourceTypeLocationData : sourceTypeSensor;
input.SensorData.SensorLocations[indexSensorLocation].SourceTypePl1PlLineOffsetBelowDikeTopAtPolder = isUseLocationData ? sourceTypeLocationData : sourceTypeSensor;
input.SensorData.SensorLocations[indexSensorLocation].SourceTypePl1PlLineOffsetBelowShoulderBaseInside = isUseLocationData ? sourceTypeLocationData : sourceTypeSensor;
input.SensorData.SensorLocations[indexSensorLocation].SourceTypePl1PlLineOffsetBelowDikeToeAtPolder = isUseLocationData ? sourceTypeLocationData : sourceTypeSensor;
input.SensorData.SensorLocations[indexSensorLocation].SourceTypePl1WaterLevelAtPolder = isUseLocationData ? sourceTypeLocationData : sourceTypeSensor;
input.SensorData.SensorLocations[indexSensorLocation].SourceTypePl3 = isUseLocationData ? sourceTypeLocationData : sourceTypeSensor;
input.SensorData.SensorLocations[indexSensorLocation].SourceTypePl4 = isUseLocationData ? sourceTypeLocationData : sourceTypeSensor;
}
for (var indexLocation = 0; indexLocation < 4; indexLocation++)
{
double difference = indexLocation is 0 or 2 ? 0 : 0.11;
input.Locations[indexLocation].DesignScenarios[0].RiverLevel = isUseLocationData ? -2.0 + difference : double.NaN;
input.Locations[indexLocation].DesignScenarios[0].PlLineOffsetBelowDikeTopAtRiver = isUseLocationData ? 0.6 + difference : double.NaN;
input.Locations[indexLocation].DesignScenarios[0].PlLineOffsetBelowDikeTopAtPolder = isUseLocationData ? 1.6 + difference : double.NaN;
input.Locations[indexLocation].DesignScenarios[0].PlLineOffsetBelowShoulderBaseInside = isUseLocationData ? 0.2 + difference : double.NaN;
input.Locations[indexLocation].DesignScenarios[0].PlLineOffsetBelowDikeToeAtPolder = isUseLocationData ? 0.3 + difference : double.NaN;
input.Locations[indexLocation].DesignScenarios[0].PolderLevel = isUseLocationData ? -4.5 + difference : double.NaN;
input.Locations[indexLocation].DesignScenarios[0].HeadPl3 = isUseLocationData ? -1.8 + difference : double.NaN;
input.Locations[indexLocation].DesignScenarios[0].HeadPl3Specified = isUseLocationData;
input.Locations[indexLocation].DesignScenarios[0].HeadPl4 = isUseLocationData ? -1.6 + difference : double.NaN;
input.Locations[indexLocation].DesignScenarios[0].HeadPl4Specified = isUseLocationData;
}
}
private static WaternetType ReadWaternetFromCreatedSkxFile(string calcDir, int indexTimeStep, int indexLocation)
{
string kernelInputFilename = $"{projectPath}\\{calcDir}\\Stability\\Bishop\\Dik(dike)_Loc(LocationOnlyPL3)_Stp("
+ indexTimeStep + ")_Mdl(Bishop)_2016-03-02T03_" + (indexTimeStep + 1) + "0_00_Pro(SoilProfile_NoInbetweenAquifer_stix).skx";
if (indexLocation is 1 or 3)
{
kernelInputFilename = $"{projectPath}\\{calcDir}\\Stability\\Bishop\\Dik(dike)_Loc(LocationPL3AndPL4)_Stp("
+ indexTimeStep + ")_Mdl(Bishop)_2016-03-02T03_" + (indexTimeStep + 1) + "0_00_Pro(SoilProfile_OneInbetweenAquifer_stix).skx";
}
FullInputModelType macrostabilityInput = MacroStabilityXmlSerialization.LoadInputFromXmlFile(kernelInputFilename);
return macrostabilityInput.StabilityModel.ConstructionStages[0].Waternet;
}
private void CheckPhreaticLine(WaternetType waternet, Input input, int indexLocation, int indexTimeStep, bool isUseLocationData)
{
WaternetLineType phreaticLine = waternet.PhreaticLine.WaternetLine;
string sensor = indexLocation is 0 or 2 ? "SensorLoc1" : "SensorLoc2";
LocationDesignScenario locData = input.Locations[indexLocation].DesignScenarios[0];
double riverLevelUsed = isUseLocationData ? locData.RiverLevel : FetchValue(input, sensor + "_PL1_River", indexTimeStep).Z;
// Note that the polder level is not the sensor value of "SensorLoc1_PL1_Polder" or the locData.PolderLevel,
// but it is the sensor value of the sensor "_PL1_Ditch" because it is situated within the ditch width.
double polderLevelUsed = FetchValue(input, sensor + "_PL1_Ditch", indexTimeStep).Z;
// Expected phreatic line
var expectedPhreaticLine = new PlLine();
expectedPhreaticLine.Points.Add(new PlLinePoint(outsideLimit.X, riverLevelUsed));
// Point 2 is the intersection between dike and River level (X coord not calculated here)
expectedPhreaticLine.Points.Add(new PlLinePoint(double.NaN, riverLevelUsed));
expectedPhreaticLine.Points.Add(FetchValue(input, sensor + "_PL1_DikeMiddleCrest", indexTimeStep));
// Point 4 is the intersection of the polder level with the (X coord not calculated here).
expectedPhreaticLine.Points.Add(new PlLinePoint(double.NaN, polderLevelUsed));
expectedPhreaticLine.Points.Add(new PlLinePoint(FetchValue(input, sensor + "_PL1_Ditch", indexTimeStep).X, polderLevelUsed));
// Point 6 is the intersection between the ditch and the polder level (X coord not calculated here).
expectedPhreaticLine.Points.Add(new PlLinePoint(double.NaN, polderLevelUsed));
expectedPhreaticLine.Points.Add(new PlLinePoint(insideLimit.X, polderLevelUsed));
if (isUseLocationData)
{
expectedPhreaticLine.Points.Insert(2, new PlLinePoint(dikeTopAtRiver.X, riverLevelUsed - locData.PlLineOffsetBelowDikeTopAtRiver));
expectedPhreaticLine.Points.Insert(4, new PlLinePoint(dikeTopAtPolder.X, riverLevelUsed - locData.PlLineOffsetBelowDikeTopAtPolder));
expectedPhreaticLine.Points.Insert(5, new PlLinePoint(shoulderInside.X, shoulderInside.Z - locData.PlLineOffsetBelowShoulderBaseInside));
expectedPhreaticLine.Points.Insert(6, new PlLinePoint(dikeToeAtPolder.X, dikeToeAtPolder.Z - locData.PlLineOffsetBelowDikeToeAtPolder));
}
CheckPoints(phreaticLine, expectedPhreaticLine, "Incorrect point in location index " + indexLocation + " for phreatic line");
}
private void CheckPl3AndPl4(WaternetType waternet, Input input, int indexLocation, int indexTimeStep, bool isUseLocationData)
{
for (var indexHeadLine = 0; indexHeadLine < 2; indexHeadLine++)
{
// No PL 4 available for locations without in-between aquifer present
if (indexHeadLine == 1 && indexLocation is 0 or 2)
{
continue;
}
WaternetLineType actualPlLine = waternet.HeadLines[indexHeadLine].WaternetLine;
// Expected PL-line
string location = indexLocation is 0 or 2 ? "SensorLoc1_" : "SensorLoc2_";
string plType = indexHeadLine == 0 ? "PL3" : "PL4";
string sensorPrefix = location + plType;
LocationDesignScenario locData = input.Locations[indexLocation].DesignScenarios[0];
double headPl3Used = isUseLocationData ? (indexHeadLine == 0 ? locData.HeadPl3 : locData.HeadPl4) : FetchValue(input, sensorPrefix + "_River", indexTimeStep).Z;
var expectedPlLine = new PlLine();
expectedPlLine.Points.Add(new PlLinePoint(outsideLimit.X, headPl3Used));
expectedPlLine.Points.Add(FetchValue(input, sensorPrefix + "_River", indexTimeStep));
// Point 3 is the intersection between dike and River level (X coord not calculated here)
expectedPlLine.Points.Add(new PlLinePoint(double.NaN, headPl3Used));
expectedPlLine.Points.Add(FetchValue(input, sensorPrefix + "_DikeMiddleCrest", indexTimeStep));
expectedPlLine.Points.Add(FetchValue(input, sensorPrefix + "_Ditch", indexTimeStep));
expectedPlLine.Points.Add(FetchValue(input, sensorPrefix + "_Polder", indexTimeStep));
expectedPlLine.Points.Add(new PlLinePoint(insideLimit.X, FetchValue(input, sensorPrefix + "_Polder", indexTimeStep).Z));
CheckPoints(actualPlLine, expectedPlLine, "Incorrect point in location index " + indexLocation + " for " + plType);
}
}
private static void CheckPoints(WaternetLineType actualPlLine, PlLine expectedPlLine, string message)
{
Assert.That(actualPlLine.Points, Has.Length.EqualTo(expectedPlLine.Points.Count));
for (var i = 0; i < actualPlLine.Points.Length; i++)
{
if (!double.IsNaN(expectedPlLine.Points[i].X))
{
Assert.That(actualPlLine.Points[i].X, Is.EqualTo(expectedPlLine.Points[i].X).Within(tolerance), message);
}
Assert.That(actualPlLine.Points[i].Z, Is.EqualTo(expectedPlLine.Points[i].Z).Within(tolerance), message);
}
}
private static PlLinePoint FetchValue(Input input, string sensorName, int indexTimeStep)
{
double z = input.OperationalInputTimeSeries.First(t => t.LocationId == sensorName).Entries.TimeSerieEntry[indexTimeStep].Value;
double x = input.SensorData.Sensors.First(t => t.Name == sensorName).RelativeLocation;
return new PlLinePoint
{
X = x,
Z = z
};
}
private static void CompareOutput(Output expected, Output actual)
{
var compare = new CompareLogic
{
Config =
{
MaxDifferences = 100
}
};
ComparisonResult result = compare.Compare(expected, actual);
Assert.That(result.Differences, Is.Empty, "Differences found read/write Output object");
}
}