// 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 System.Text;
using Deltares.DamEngine.Data.Design;
using Deltares.DamEngine.Data.General;
using Deltares.DamEngine.Data.Standard.Calculation;
using Deltares.DamEngine.Data.Standard.Logging;
using Deltares.DamEngine.Interface;
using Deltares.DamEngine.Io;
using Deltares.DamEngine.Io.XmlOutput;
using DGeoSuite.Common;
using KellermanSoftware.CompareNetObjects;
using NUnit.Framework;
using ConversionHelper = Deltares.DamEngine.Interface.ConversionHelper;
namespace Deltares.DamEngine.TestHelpers;
public static class GeneralHelper
{
public static int DetermineNumberOfCalculationErrors(List calculationMessages)
{
var errorCount = 0;
foreach (LogMessage logMessage in calculationMessages)
{
if (logMessage.MessageType is LogMessageType.Error or LogMessageType.FatalError)
{
errorCount++;
}
}
return errorCount;
}
public static Output RunAfterInputValidation(string inputString, bool areResultsExpected = true, string outputFilename = "")
{
var engineInterface = new EngineInterface(inputString);
return RunAfterInputValidation(engineInterface, areResultsExpected, outputFilename);
}
public static Output RunAfterInputValidation(EngineInterface engineInterface, bool areResultsExpected = true, string outputFilename = "")
{
// Validate input
Assert.That(engineInterface.DamProjectData, Is.Not.Null);
string validationMessages = engineInterface.Validate();
var extraValidationMessage = "";
if (outputFilename != "")
{
extraValidationMessage = ", see output xml in debugger";
}
Assert.That(validationMessages, Is.Null, "Validation should succeed but does not" + extraValidationMessage);
// Run calculation
string outputString = engineInterface.Run();
Assert.That(outputString, Is.Not.Null);
Output output = DamXmlSerialization.LoadOutputFromXmlString(outputString);
if (outputFilename != "")
{
File.WriteAllText(outputFilename, outputString, Encoding.Unicode);
}
// Evaluate results
if (!areResultsExpected)
{
Assert.That(output.Results.CalculationResults, Is.Null, "No results are expected");
return output;
}
bool isNoResultAvailable;
switch (engineInterface.DamProjectData.DamProjectType)
{
case DamProjectType.Design:
isNoResultAvailable = output.Results.CalculationResults.IsNullOrEmpty();
break;
default:
isNoResultAvailable = output.Results.OperationalOutputTimeSeries.IsNullOrEmpty();
break;
}
if (!isNoResultAvailable)
{
CheckConsistencyOfAdaptGeometryResults(engineInterface, output.Results);
return output;
}
var assertMessage = "No results available.";
foreach (Message calcMessage in output.Results.CalculationMessages)
{
assertMessage = assertMessage + Environment.NewLine + calcMessage.Message1;
}
Assert.That(output.Results.CalculationResults, Is.Not.Null, assertMessage);
return output;
}
private static void CheckConsistencyOfAdaptGeometryResults(EngineInterface engineInterface, OutputResults outputResults)
{
if (engineInterface.DamProjectData.DamProjectType == DamProjectType.Design &&
engineInterface.DamProjectData.DamProjectCalculationSpecification.AnalysisTypeForSerializationPurposeOnly == AnalysisType.AdaptGeometry)
{
var isIteratedFilePresent = false;
foreach (DesignResult calculationResult in outputResults.CalculationResults)
{
bool isDesignSuccessful = calculationResult.CalculationResult == ConversionHelper.ConvertToOutputCalculationResult(CalculationResult.Succeeded) &&
IsAdaptGeometrySuccessful(outputResults.CalculationMessages, calculationResult);
double fosRequired = FetchRequiredFactor(engineInterface, calculationResult.ScenarioName, calculationResult.LocationName);
double fosCalculated = FetchCalculatedFactor(engineInterface, calculationResult);
if (isDesignSuccessful)
{
string message = "After adapting the geometry, the calculated safety factor is less than the required safety factor " +
$"for location '{calculationResult.LocationName}' and scenario '{calculationResult.ScenarioName}'. " +
"This is unexpected.";
Assert.That(fosCalculated, Is.GreaterThanOrEqualTo(fosRequired), message);
}
else
{
const string message = "As the design was not successful and had to stop, the calculated safety " +
"factor should be less than the required safety factor but this is not the case.";
Assert.That(fosRequired, Is.GreaterThanOrEqualTo(fosCalculated), message);
}
string fileName = calculationResult.BaseFileName;
// Note: only check for the iterated file if the design calculation was successful
isIteratedFilePresent = isIteratedFilePresent ||
(!fileName.IsNullOrEmpty() && fileName.Contains("Ite(") && !fileName.Contains("Ite(1)"))
|| !isDesignSuccessful;
}
// The iterated file is created only for the stability mechanism (not for piping) to be opened in D-Stability
if (engineInterface.DamProjectData.DamProjectCalculationSpecification.CurrentSpecification.FailureMechanismSystemType != FailureMechanismSystemType.Piping)
{
Assert.That(isIteratedFilePresent, Is.True, "The AnalyseType is set to AdaptGeometry in the input, however the geometry was " +
"not adapted in any location. Either set the AnalysisType to NoAdaptation or " +
"increase the required safety factor.");
}
}
}
private static bool IsAdaptGeometrySuccessful(Message[] messages, DesignResult results)
{
return !(from message in messages
let location = "Location '" + results.LocationName + "'"
let profile = "subsoil scenario '" + results.ProfileName + "'"
let scenario = "design scenario '" + results.ScenarioName + "'"
where message.Message1.Contains("The design was not successful.") &&
message.Message1.Contains(location) && message.Message1.Contains(profile) &&
message.Message1.Contains(scenario)
select message).Any();
}
private static double FetchRequiredFactor(EngineInterface engineInterface, string scenarioName, string locationName)
{
foreach (Location location in engineInterface.DamProjectData.Dike.Locations)
{
foreach (DesignScenario scenario in location.Scenarios)
{
if (location.Name == locationName && scenario.LocationScenarioID == scenarioName)
{
return engineInterface.DamProjectData.DamProjectCalculationSpecification.CurrentSpecification.FailureMechanismSystemType switch
{
FailureMechanismSystemType.StabilityInside => scenario.RequiredSafetyFactorStabilityInnerSlope,
FailureMechanismSystemType.StabilityOutside => scenario.RequiredSafetyFactorStabilityOuterSlope,
FailureMechanismSystemType.Piping => scenario.RequiredSafetyFactorPiping,
_ => throw new ArgumentOutOfRangeException()
};
}
}
}
return 999;
}
private static double FetchCalculatedFactor(EngineInterface engineInterface, DesignResult results)
{
switch (engineInterface.DamProjectData.DamProjectCalculationSpecification.CurrentSpecification.FailureMechanismSystemType)
{
case FailureMechanismSystemType.StabilityInside:
case FailureMechanismSystemType.StabilityOutside:
return results.StabilityDesignResults.SafetyFactor;
case FailureMechanismSystemType.Piping:
switch (engineInterface.DamProjectData.DamProjectCalculationSpecification.CurrentSpecification.PipingModelType)
{
case PipingModelType.Wti2017:
return results.PipingDesignResults.Wti2017BackwardErosionFactor;
case PipingModelType.Bligh:
return results.PipingDesignResults.BlighFactor;
}
break;
default:
throw new ArgumentOutOfRangeException();
}
return 0;
}
public static void CompareDesignOutput(Output expected, Output actual)
{
StripCalculationSubDir(expected);
StripCalculationSubDir(actual);
CompareOutput(expected, actual);
}
private static void StripCalculationSubDir(Output expected)
{
foreach (DesignResult result in expected.Results.CalculationResults)
{
result.CalculationSubDir = "";
}
}
public 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");
}
public static void RemoveTestWorkingDirectory(string testWorkingFolder)
{
if (Directory.Exists(testWorkingFolder))
{
const bool recursive = true;
Directory.Delete(testWorkingFolder, recursive);
}
}
public static Output RunMultiCoreWithXmlInputFile(string mapTestFiles, DamProjectType damProjectType,
int maxCores, string calcDir, string inputFilename, string outputFilename,
bool justOneScenario = false, bool expectedNoErrors = true)
{
var processorCount = Environment.ProcessorCount;
if (maxCores > processorCount)
{
Assert.Ignore($"The number of cores requested ({maxCores}) is higher than the number of available cores ({processorCount}).");
}
if (Directory.Exists(calcDir))
{
Directory.Delete(calcDir, true); // delete previous results
}
Directory.CreateDirectory(calcDir);
string fullInputFilename = Path.Combine(mapTestFiles, inputFilename);
string inputString = File.ReadAllText(fullInputFilename);
inputString = XmlAdapter.ChangeValueInXml(inputString, "ProjectPath", ""); // Current directory will be used
inputString = XmlAdapter.ChangeValueInXml(inputString, "CalculationMap", calcDir); // Current directory will be used
inputString = XmlAdapter.ChangeValueInXml(inputString, "MaxCalculationCores", maxCores.ToString());
var engineInterface = new EngineInterface(inputString);
if (justOneScenario)
{
engineInterface.DamProjectData.Dike.ClearLocationScenariosExceptFirst();
}
Assert.That(engineInterface.DamProjectData, Is.Not.Null);
Assert.That(engineInterface.DamProjectData.DamProjectType, Is.EqualTo(damProjectType));
Output output = GeneralHelper.RunAfterInputValidation(engineInterface, true, outputFilename);
if (expectedNoErrors)
{
int errorCount = GeneralHelper.DetermineNumberOfCalculationErrors(engineInterface.DamProjectData.CalculationMessages);
Assert.That(errorCount, Is.EqualTo(0), "There should be no errors during the calculation.");
}
return output;
}
}