// 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;
using System.Collections.Generic;
using System.IO;
using System.Linq;
using Deltares.DamEngine.Data.Standard;
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 Point2DType = Deltares.DamEngine.Io.XmlOutput.Point2DType;
using WaternetType = Deltares.MacroStability.Io.XmlInput.WaternetType;
namespace Deltares.DamEngine.IntegrationTests.IntegrationTests;
[TestFixture]
public class OperationalStabilityProfile1DTests
{
// Constants for SensorType
private const uint sourceTypeIgnore = 0;
private const uint sourceTypeLocationData = 1;
private const uint sourceTypeSensor = 2;
public static IEnumerable AddTestCases
{
get
{
yield return new TestCaseData(
// This is the case that all values are taken from the location data
new SensorConfiguration
{
SourceTypePl1OuterWaterLevel = sourceTypeLocationData,
Pl1OuterWaterLevel = -0.1,
SourceTypePl1PlLineOffsetBelowDikeTopAtRiver = sourceTypeLocationData,
Pl1PlLineOffsetBelowDikeTopAtRiver = 1.1,
SourceTypePl1PlLineOffsetBelowDikeTopAtPolder = sourceTypeLocationData,
Pl1PlLineOffsetBelowDikeTopAtPolder = 1.2,
SourceTypePl1PlLineOffsetBelowShoulderBaseInside = sourceTypeIgnore,
Pl1PlLineOffsetBelowShoulderBaseInside = 0.0,
SourceTypePl1PlLineOffsetBelowDikeToeAtPolder = sourceTypeLocationData,
Pl1PlLineOffsetBelowDikeToeAtPolder = 1.4,
SourceTypePl1PolderLevel = sourceTypeLocationData,
PlPolderLevel = -2.0,
SourceTypePl3 = sourceTypeLocationData,
HeadPl3 = -3.0,
SourceTypePl4 = sourceTypeIgnore,
HeadPl4 = 0.0
},
new PlLinePoints // PL-1
{
Points =
[
new Point2DType
{
// Pl1Left
X = 0.0, // Expected X-value is leftmost point.X
Z = -0.1 // Expected Z-value is the value of location data WaterLevelOutside of 'Purmer_PU0042+00_K': -0.1
},
new Point2DType
{
// Pl1AtDikeTopAtRiver
X = 18.731, // Expected X-value is location of DikeTopAtRiver = 18.731
Z = -1.2 // Expected Z-value is WaterLevelOutside - PlLineOffsetBelowDikeTopAtRiver = -0.1 - 1.1 = -1.2
},
new Point2DType
{
// Pl1AtDikeTopAtPolder
X = 22.562, // Expected X-value is location of DikeTopAtPolder = 22.562
Z = -1.3 // Expected Z-value is WaterLevelOutside - PlLineOffsetBelowDikeTopAtPolder = -0.1 - 1.2 = -1.3
},
new Point2DType
{
// DikeToeAtPolder
X = 29.374, // Expected X-value is location of DikeToeAtPolder = 29.374
Z = -2.937 // Expected Z-value is WaterLevelOutside - Pl1PlLineOffsetBelowDikeToeAtPolder = -1.357 - 1.4 = -2.937
},
new Point2DType
{
// Pl1Right
X = 57.896, // Expected X-value is rightmost point.X
Z = -2.0 // Expected Z-value is location data Polderlevel of 'Purmer_PU0042+00_K': -2.0
}
]
},
new PlLinePoints // PL-3
{
Points =
[
new Point2DType
{
// Pl3Left
X = 0.0, // Expected X-value is leftmost point.X
Z = -3.0 // Expected Z-value is the value of location data: -3.0
},
new Point2DType
{
// sensor 'PB01786_BuT'
X = 11.468, // Expected X-value is the location of the sensor X = 11.486
Z = -2.87 // Expected Z-value is the value of the first timestep of the sensor: -2.87
},
new Point2DType
{
// sensor 'PB01786'
X = 36.185, // Expected X-value is the location of the sensor X = 36.185
Z = -2.87 // Expected Z-value is the value of the first timestep of the sensor: -2.87
},
new Point2DType
{
// Pl3Right
X = 57.896, // Expected X-value is rightmost point.X: 57.896
Z = -2.87 // Expected Z-value is the value of the first timestep of the last sensor: -2.87
}
]
},
null).SetName("OnlyLocationData");
yield return new TestCaseData(
// This is the case that all values are taken from the sensor data
new SensorConfiguration
{
SourceTypePl1OuterWaterLevel = sourceTypeSensor,
Pl1OuterWaterLevel = -0.1,
SourceTypePl1PlLineOffsetBelowDikeTopAtRiver = sourceTypeIgnore,
Pl1PlLineOffsetBelowDikeTopAtRiver = 1.1,
SourceTypePl1PlLineOffsetBelowDikeTopAtPolder = sourceTypeIgnore,
Pl1PlLineOffsetBelowDikeTopAtPolder = 1.2,
SourceTypePl1PlLineOffsetBelowShoulderBaseInside = sourceTypeIgnore,
Pl1PlLineOffsetBelowShoulderBaseInside = 0.0,
SourceTypePl1PlLineOffsetBelowDikeToeAtPolder = sourceTypeIgnore,
Pl1PlLineOffsetBelowDikeToeAtPolder = 1.4,
SourceTypePl1PolderLevel = sourceTypeSensor,
PlPolderLevel = -2.0,
SourceTypePl3 = sourceTypeIgnore,
HeadPl3 = -3.0,
SourceTypePl4 = sourceTypeIgnore,
HeadPl4 = 0.0
},
new PlLinePoints
{
Points =
[
new Point2DType
{
// Pl1Left amd Waterlevel sensor MPN-AS-27
X = 0.0, // Expected X-value is leftmost point.X
Z = -0.2 // Expected Z-value is the value of the first timestep of sensor: -0.2
},
new Point2DType
{
// Sensor RFT_R52_0110_45m_BuKr
X = 18.305, // Expected X-value is location sensor: 18.305
Z = 0.24 // Expected Z-value is the value of the first timestep of sensor RFT_R52_0110_45m_BuKr: -0.2
},
new Point2DType
{
// Sensor RFT_R52_0110_45m_BiKr
X = 22.563, // Expected X-value is location of sensor = 22.563
Z = 0.29 // Expected Z-value is the value of the first timestep of sensor: -0.29
},
new Point2DType
{
// Sensor RFT_R52_0110_45m_InstBrm
X = 30.225, // Expected X-value is location of sensor: 30.225
Z = -2.35 // Expected Z-value is the value of the first timestep of sensor: -2.35
},
new Point2DType
{
// Sensor RFT_R52_0110_45m_BiT
X = 36.185, // Expected X-value is location of sensor: 36.185
Z = -3.23 // Expected Z-value is the value of the first timestep of sensor: -3.23
},
new Point2DType
{
// Pl1Right
X = 57.896, // Expected X-value is rightmost point.X
Z = -2.0 // Expected Z-value is location data Polderlevel of 'Purmer_PU0042+00_K': -2.0
}
]
},
new PlLinePoints // PL-3
{
Points =
[
new Point2DType
{
// Pl3Left
X = 0.0, // Expected X-value is leftmost point.X
Z = -0.2 // Expected Z-value is the value of the Waterlevel: -0.2
},
new Point2DType
{
// sensor 'PB01786_BuT'
X = 11.468, // Expected X-value is the location of the sensor X = 11.486
Z = -2.87 // Expected Z-value is the value of the first timestep of the sensor: -2.87
},
new Point2DType
{
// sensor 'PB01786'
X = 36.185, // Expected X-value is the location of the sensor X = 36.185
Z = -2.87 // Expected Z-value is the value of the first timestep of the sensor: -2.87
},
new Point2DType
{
// Pl3Right
X = 57.896, // Expected X-value is rightmost point.X: 57.896
Z = -2.87 // Expected Z-value is the value of the first timestep of the last sensor: -2.87
}
]
},
null).SetName("OnlySensorData");
}
}
// Following testcase based on the DamLive test Deltares.DamLive.Tests.StabilityInsideBishopGridTest
// "Deltares.DamLive.Tests\TestData\IntegrationTests\StabilityInsideBishopGrid\DAMLive.damx"
// with DamLive rev.4860
// This tests if the sensor location data is used in the calculation by changing the source type of Pl1 offset below diketop to
[Test, Category(Categories.Slow)]
[TestCaseSource(nameof(AddTestCases))]
public void GivenStabilityInsideProfile1DProject_WhenCalculatingWithVariationsOnSpecifiedSensorLocationData_ThenExpectedResultIsGenerated(SensorConfiguration sensorConfiguration,
PlLinePoints pl1Points,
PlLinePoints pl3Points,
PlLinePoints pl4Points)
{
var projectPath = "SensorProfile1DTests";
string testCaseName = TestContext.CurrentContext.Test.Name;
var calcDir = $"DAMLive.Calc.{testCaseName}";
const string testFilesLocation = @".\TestFiles\Operational\SensorProfile1DTests\";
string inputFilename = testFilesLocation + "CalculateStabilityInsideBishopGrid1InputFile.xml";
string outputFilename = testFilesLocation + "CalculateStabilityInsideBishopGrid1OutputFile.xml";
if (Directory.Exists(calcDir))
{
Directory.Delete(calcDir, true); // delete previous results
}
Input input = DamXmlSerialization.LoadInputFromXmlFile(inputFilename);
input.CalculationMap = calcDir;
input.ProjectPath = projectPath;
Location location = input.Locations[0];
FillSensorLocation(input.SensorData.SensorLocations[0], location, sensorConfiguration);
string inputString = DamXmlSerialization.SaveInputAsXmlString(input);
DamXmlSerialization.SaveInputAsXmlFile(inputFilename, input);
Output output = GeneralHelper.RunAfterInputValidation(inputString, true, outputFilename);
Assert.That(output, Is.Not.Null);
// Read the kernel input data from the input file
var kernelInputFilename = $"{projectPath}\\{calcDir}\\Stability\\Bishop\\Dik(dike)_Loc(Purmer_PU0042+00_K)_Stp(0)_Mdl(Bishop)_2016-03-02T03_10_00_Pro(Purmer_PU0042+00K).skx";
FullInputModelType expectedMacrostabilityInput = MacroStabilityXmlSerialization.LoadInputFromXmlFile(kernelInputFilename);
WaternetType waternet = expectedMacrostabilityInput.StabilityModel.ConstructionStages[0].Waternet;
// Check the phreatic line
List phreaticLine = waternet.PhreaticLine.WaternetLine.Points.ToList();
var messages = new List();
messages.Add("Failing assertions");
bool arePointsInLine = ArePointsInLine(pl1Points, phreaticLine, "PL-1", messages);
// PL-3
List pl3 = waternet.HeadLines.Any() ? waternet.HeadLines[0]?.WaternetLine.Points.ToList() : null;
arePointsInLine = arePointsInLine && ArePointsInLine(pl3Points, pl3, "PL-3", messages);
// PL-4
List pl4 = waternet.HeadLines?.Length > 1 ? waternet.HeadLines[1]?.WaternetLine.Points.ToList() : null;
arePointsInLine = arePointsInLine && ArePointsInLine(pl4Points, pl4, "PL-4", messages);
string assertMessage = string.Join(Environment.NewLine, messages);
Assert.That(arePointsInLine, Is.True, assertMessage);
}
[Test, Category(Categories.Slow)]
// Following testcase based on the DamLive test Deltares.DamLive.Tests.StabilityInsideBishopGridTest
// "Deltares.DamLive.Tests\TestData\IntegrationTests\StabilityInsideBishopGrid\DAMLive.damx"
// with DamLive rev.4860
[TestCase("CalculateStabilityInsideBishopGrid1")]
// Following testcase based on the DamLive test Deltares.DamLive.Tests.StabilityInsideBishopGridTest
// "Deltares.DamLive.Tests\TestData\IntegrationTests\StabilityInsideBishopGrid\DAMLive.damx"
// with DamLive rev.4860
// This tests if the sensor location data is used in the calculation by changing the source type of Pl1 offset below diketop to
// use the specified location data (which will be set to 1.1).
// Expected X-value is location of DikeTopAtPolder = 22.562
// Expected Z-value is WaterLevelOutside - PlLineOffsetBelowDikeTopAtPolder = -0.2 - 1.1 = -1.3
[TestCase("CalculateStabilityInsideBishopGrid1", true, 22.562, -1.3)]
// Following testcase based on the DamLive test Deltares.DamLive.Tests.StabilityInsideUpliftVanBeeSwarmTest
// "Deltares.DamLive.Tests\TestData\IntegrationTests\StabilityInsideUpliftVanBeeSwarm\DAMLive.damx"
// with DamLive rev.4860
[TestCase("CalculateStabilityInsideUpliftVanBeeSwarm1")]
// Following testcase based on the DamLive test Deltares.DamLive.Tests.StabilityInsideUpliftVanGridTest
// "Deltares.DamLive.Tests\TestData\IntegrationTests\StabilityInsideUpliftVanGrid\DAMLive.damx"
// with DamLive rev.4860
[TestCase("CalculateStabilityInsideUpliftVanGrid1")]
public void GivenStabilityInsideProfile1DProject_WhenCalculatingWithSpecifiedModel_ThenExpectedResultIsGenerated(
string baseName, bool isUseSensorLocationData = false, double expectedPlLineXValue = 0.0, double expectedPlLineZValue = 0.0)
{
const string calcDir = "DAMLive.Calc";
const string testFilesLocation = @".\TestFiles\Operational\Profile1DTests\";
string baseOutputName = baseName;
if (isUseSensorLocationData)
{
baseOutputName += "_UseLocationData";
}
string projectPath = baseOutputName;
string inputFilename = baseName + "InputFile.xml";
string outputFilename = baseOutputName + "OutputFile.xml";
if (Directory.Exists(calcDir))
{
Directory.Delete(calcDir, true); // delete previous results
}
// Load expected output
// Note 1: The expected output files (e.g. "CalculateStabilityInsideBishopGrid1OutputFile.xml") is also saved to the
// working folder
// Note 2: The expected output files should be generated with the Release build of the solution.
// Unfortunately, the Debug build of the solution generates different output files.
string expectedOutputFileName = testFilesLocation + outputFilename;
Output expectedOutput = DamXmlSerialization.LoadOutputFromXmlFile(expectedOutputFileName);
// Given
// Load input
string inputString = File.ReadAllText(testFilesLocation + inputFilename);
inputString = XmlAdapter.ChangeValueInXml(inputString, "ProjectPath", projectPath); // Current directory will be used
inputString = XmlAdapter.ChangeValueInXml(inputString, "CalculationMap", calcDir); // Current directory will be used
if (isUseSensorLocationData)
{
// Change the source type of Pl1 offset below diketop location data (=1)
inputString = XmlAdapter.ChangeValueInXml(inputString, "SourceTypePl1PlLineOffsetBelowDikeTopAtPolder", "1");
inputString = XmlAdapter.ChangeValueInXml(inputString, "PlLineOffsetBelowDikeTopAtPolder", "1.1");
}
// When
// Run calculation
Output actualOutput = GeneralHelper.RunAfterInputValidation(inputString, true, outputFilename);
if (isUseSensorLocationData)
{
// Read the kernel input data from the input file
var kernelInputFilename = $"{baseName}_UseLocationData\\DAMLive.Calc\\Stability\\Bishop\\Dik(dike)_Loc(Purmer_PU0042+00_K)_Stp(0)_Mdl(Bishop)_2016-03-02T03_10_00_Pro(Purmer_PU0042+00K).skx";
FullInputModelType expectedMacrostabilityInput = MacroStabilityXmlSerialization.LoadInputFromXmlFile(kernelInputFilename);
List phreaticLine = expectedMacrostabilityInput.StabilityModel.ConstructionStages[0].Waternet.PhreaticLine.WaternetLine.Points.ToList();
var expectedPoint = new Point2DType
{
X = expectedPlLineXValue,
Z = expectedPlLineZValue
};
const double tolerance = 0.0005;
bool pointExists = phreaticLine.Any(p => p.X.IsNearEqual(expectedPoint.X, tolerance) && p.Z.IsNearEqual(expectedPoint.Z, tolerance));
Assert.That(pointExists, Is.True, "The expected point was not found in the phreatic line.");
}
// Then
// Compare output
Assert.Multiple(() =>
{
Assert.That(actualOutput, Is.Not.Null);
Assert.That(expectedOutput, Is.Not.Null);
}
);
CompareOutput(expectedOutput, actualOutput);
}
private static bool ArePointsInLine(PlLinePoints pointsToFind, List line, string lineName, List messages)
{
const double tolerance = 0.0005;
if (pointsToFind == null)
{
return true;
}
var arePointsInLine = true;
var missingPoints = new List();
foreach (Point2DType point in pointsToFind.Points)
{
bool isPointExists = (line != null) && line.Any(p => p.X.IsNearEqual(point.X, tolerance) && p.Z.IsNearEqual(point.Z, tolerance));
if (!isPointExists)
{
missingPoints.Add($"X: {point.X}, Z: {point.Z}");
arePointsInLine = false;
}
}
if (!arePointsInLine)
{
messages.Add($"The following points were not found in {lineName}");
messages.AddRange(missingPoints);
}
return arePointsInLine;
}
public class SensorConfiguration
{
public uint SourceTypePl1OuterWaterLevel { get; set; } = sourceTypeSensor;
public double Pl1OuterWaterLevel { get; set; }
public uint SourceTypePl1PlLineOffsetBelowDikeTopAtRiver { get; set; } = sourceTypeIgnore;
public double Pl1PlLineOffsetBelowDikeTopAtRiver { get; set; }
public uint SourceTypePl1PlLineOffsetBelowDikeTopAtPolder { get; set; } = sourceTypeIgnore;
public double Pl1PlLineOffsetBelowDikeTopAtPolder { get; set; }
public uint SourceTypePl1PlLineOffsetBelowShoulderBaseInside { get; set; } = sourceTypeIgnore;
public double Pl1PlLineOffsetBelowShoulderBaseInside { get; set; }
public uint SourceTypePl1PlLineOffsetBelowDikeToeAtPolder { get; set; } = sourceTypeIgnore;
public double Pl1PlLineOffsetBelowDikeToeAtPolder { get; set; }
public uint SourceTypePl1PolderLevel { get; set; } = sourceTypeSensor;
public double PlPolderLevel { get; set; }
public uint SourceTypePl3 { get; set; } = sourceTypeLocationData;
public double HeadPl3 { get; set; }
public uint SourceTypePl4 { get; set; } = sourceTypeLocationData;
public double HeadPl4 { get; set; }
}
public class PlLinePoints
{
public List Points { get; set; }
}
private static void FillSensorLocation(SensorLocation sensorDataSensorLocation, Location location, SensorConfiguration sensorConfiguration)
{
sensorDataSensorLocation.SourceTypePl1WaterLevelAtRiver = sensorConfiguration.SourceTypePl1OuterWaterLevel;
location.DesignScenarios[0].RiverLevel = sensorConfiguration.Pl1OuterWaterLevel;
sensorDataSensorLocation.SourceTypePl1PlLineOffsetBelowDikeTopAtRiver = sensorConfiguration.SourceTypePl1PlLineOffsetBelowDikeTopAtRiver;
location.DesignScenarios[0].PlLineOffsetBelowDikeTopAtRiver = sensorConfiguration.Pl1PlLineOffsetBelowDikeTopAtRiver;
sensorDataSensorLocation.SourceTypePl1PlLineOffsetBelowDikeTopAtPolder = sensorConfiguration.SourceTypePl1PlLineOffsetBelowDikeTopAtPolder;
location.DesignScenarios[0].PlLineOffsetBelowDikeTopAtPolder = sensorConfiguration.Pl1PlLineOffsetBelowDikeTopAtPolder;
sensorDataSensorLocation.SourceTypePl1PlLineOffsetBelowShoulderBaseInside = sensorConfiguration.SourceTypePl1PlLineOffsetBelowShoulderBaseInside;
location.DesignScenarios[0].PlLineOffsetBelowShoulderBaseInside = sensorConfiguration.Pl1PlLineOffsetBelowShoulderBaseInside;
sensorDataSensorLocation.SourceTypePl1PlLineOffsetBelowDikeToeAtPolder = sensorConfiguration.SourceTypePl1PlLineOffsetBelowDikeToeAtPolder;
location.DesignScenarios[0].PlLineOffsetBelowDikeToeAtPolder = sensorConfiguration.Pl1PlLineOffsetBelowDikeToeAtPolder;
sensorDataSensorLocation.SourceTypePl1WaterLevelAtPolder = sensorConfiguration.SourceTypePl1PolderLevel;
location.DesignScenarios[0].PolderLevel = sensorConfiguration.PlPolderLevel;
sensorDataSensorLocation.SourceTypePl3 = sensorConfiguration.SourceTypePl3;
location.DesignScenarios[0].HeadPl3 = sensorConfiguration.HeadPl3;
location.DesignScenarios[0].HeadPl3Specified = true;
sensorDataSensorLocation.SourceTypePl4 = sensorConfiguration.SourceTypePl4;
location.DesignScenarios[0].HeadPl4Specified = true;
location.DesignScenarios[0].HeadPl4 = sensorConfiguration.HeadPl4;
}
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");
}
}