From 874612dabbfd1734694cda77f2770f8f3e8ddc95 Mon Sep 17 00:00:00 2001 From: guilpier-code <62292552+guilpier-code@users.noreply.github.com> Date: Fri, 10 Jan 2025 14:06:50 +0100 Subject: [PATCH] Introduce accurate shave peaks hydro algorithm + tests (#2532) For now it's not integrated into antares-solver, integration is done with #2566. Co-authored-by: Florian OMNES --- src/solver/simulation/CMakeLists.txt | 12 + .../simulation/shave-peaks-by-remix-hydro.h | 29 + .../simulation/shave-peaks-by-remix-hydro.cpp | 309 +++++++ .../src/solver/simulation/CMakeLists.txt | 81 +- .../solver/simulation/test-hydro-remix.cpp | 788 ++++++++++++++++++ 5 files changed, 1183 insertions(+), 36 deletions(-) create mode 100644 src/solver/simulation/include/antares/solver/simulation/shave-peaks-by-remix-hydro.h create mode 100644 src/solver/simulation/shave-peaks-by-remix-hydro.cpp create mode 100644 src/tests/src/solver/simulation/test-hydro-remix.cpp diff --git a/src/solver/simulation/CMakeLists.txt b/src/solver/simulation/CMakeLists.txt index 17e511fac5..a110adac4f 100644 --- a/src/solver/simulation/CMakeLists.txt +++ b/src/solver/simulation/CMakeLists.txt @@ -76,6 +76,18 @@ target_link_libraries(antares-solver-simulation antares-solver-ts-generator ) +# Remix hydro algorithm +add_library(shave-peaks-by-remix-hydro) + +target_sources(shave-peaks-by-remix-hydro + PRIVATE + shave-peaks-by-remix-hydro.cpp) + +target_include_directories(shave-peaks-by-remix-hydro + PUBLIC + $) + + install(DIRECTORY include/antares DESTINATION "include" ) diff --git a/src/solver/simulation/include/antares/solver/simulation/shave-peaks-by-remix-hydro.h b/src/solver/simulation/include/antares/solver/simulation/shave-peaks-by-remix-hydro.h new file mode 100644 index 0000000000..77ffbc67e8 --- /dev/null +++ b/src/solver/simulation/include/antares/solver/simulation/shave-peaks-by-remix-hydro.h @@ -0,0 +1,29 @@ + +#pragma once + +#include + +namespace Antares::Solver::Simulation +{ + +struct RemixHydroOutput +{ + std::vector HydroGen; + std::vector UnsupE; + std::vector levels; +}; + +RemixHydroOutput shavePeaksByRemixingHydro(const std::vector& DispatchGen, + const std::vector& HydroGen, + const std::vector& UnsupE, + const std::vector& HydroPmax, + const std::vector& HydroPmin, + double init_level, + double capacity, + const std::vector& inflow, + const std::vector& overflow, + const std::vector& pump, + const std::vector& Spillage, + const std::vector& DTG_MRG); + +} // namespace Antares::Solver::Simulation diff --git a/src/solver/simulation/shave-peaks-by-remix-hydro.cpp b/src/solver/simulation/shave-peaks-by-remix-hydro.cpp new file mode 100644 index 0000000000..99a8f2aa55 --- /dev/null +++ b/src/solver/simulation/shave-peaks-by-remix-hydro.cpp @@ -0,0 +1,309 @@ +#include "include/antares/solver/simulation/shave-peaks-by-remix-hydro.h" + +#include +#include +#include +#include + +namespace Antares::Solver::Simulation +{ + +int find_min_index(const std::vector& TotalGen, + const std::vector& OutUnsupE, + const std::vector& OutHydroGen, + const std::vector& triedBottom, + const std::vector& HydroPmax, + const std::vector& enabledHours, + double top) +{ + double min_val = top; + int min_hour = -1; + for (unsigned int h = 0; h < TotalGen.size(); ++h) + { + if (OutUnsupE[h] > 0 && OutHydroGen[h] < HydroPmax[h] && !triedBottom[h] && enabledHours[h]) + { + if (TotalGen[h] < min_val) + { + min_val = TotalGen[h]; + min_hour = h; + } + } + } + return min_hour; +} + +int find_max_index(const std::vector& TotalGen, + const std::vector& OutHydroGen, + const std::vector& triedPeak, + const std::vector& HydroPmin, + const std::vector& enabledHours, + double ref_value, + double eps) +{ + double max_val = 0; + int max_hour = -1; + for (unsigned int h = 0; h < TotalGen.size(); ++h) + { + if (OutHydroGen[h] > HydroPmin[h] && TotalGen[h] >= ref_value + eps && !triedPeak[h] + && enabledHours[h]) + { + if (TotalGen[h] > max_val) + { + max_val = TotalGen[h]; + max_hour = h; + } + } + } + return max_hour; +} + +static bool operator<=(const std::vector& a, const std::vector& b) +{ + return a.size() == b.size() + && std::ranges::all_of(std::views::iota(size_t{0}, a.size()), + [&](size_t i) { return a[i] <= b[i]; }); +} + +static bool operator<=(const std::vector& v, const double c) +{ + return std::ranges::all_of(v, [&c](const double& e) { return e <= c; }); +} + +static bool operator>=(const std::vector& v, const double c) +{ + return std::ranges::all_of(v, [&c](const double& e) { return e >= c; }); +} + +static void checkInputCorrectness(const std::vector& DispatchGen, + const std::vector& HydroGen, + const std::vector& UnsupE, + const std::vector& levels, + const std::vector& HydroPmax, + const std::vector& HydroPmin, + double initial_level, + double capacity, + const std::vector& inflows, + const std::vector& overflow, + const std::vector& pump, + const std::vector& Spillage, + const std::vector& DTG_MRG) +{ + std::string msg_prefix = "Remix hydro input : "; + + // Initial level smaller than capacity + if (initial_level > capacity) + { + throw std::invalid_argument(msg_prefix + "initial level > reservoir capacity"); + } + // Arrays sizes must be identical + std::vector sizes = {DispatchGen.size(), + HydroGen.size(), + UnsupE.size(), + levels.size(), + HydroPmax.size(), + HydroPmin.size(), + inflows.size(), + overflow.size(), + pump.size(), + Spillage.size(), + DTG_MRG.size()}; + + if (!std::ranges::all_of(sizes, [&sizes](const size_t s) { return s == sizes.front(); })) + { + throw std::invalid_argument(msg_prefix + "arrays of different sizes"); + } + + // Arrays are of size 0 + if (!DispatchGen.size()) + { + throw std::invalid_argument(msg_prefix + "all arrays of sizes 0"); + } + + // Hydro production < Pmax + if (!(HydroGen <= HydroPmax)) + { + throw std::invalid_argument(msg_prefix + + "Hydro generation not smaller than Pmax everywhere"); + } + + // Hydro production > Pmin + if (!(HydroPmin <= HydroGen)) + { + throw std::invalid_argument(msg_prefix + + "Hydro generation not greater than Pmin everywhere"); + } + + if (!(levels <= capacity) || !(levels >= 0.)) + { + throw std::invalid_argument(msg_prefix + + "levels computed from input don't respect reservoir bounds"); + } +} + +RemixHydroOutput shavePeaksByRemixingHydro(const std::vector& DispatchGen, + const std::vector& HydroGen, + const std::vector& UnsupE, + const std::vector& HydroPmax, + const std::vector& HydroPmin, + double initial_level, + double capa, + const std::vector& inflows, + const std::vector& overflow, + const std::vector& pump, + const std::vector& Spillage, + const std::vector& DTG_MRG) +{ + std::vector levels(DispatchGen.size()); + if (!levels.empty()) + { + levels[0] = initial_level + inflows[0] - overflow[0] + pump[0] - HydroGen[0]; + for (size_t h = 1; h < levels.size(); ++h) + { + levels[h] = levels[h - 1] + inflows[h] - overflow[h] + pump[h] - HydroGen[h]; + } + } + + checkInputCorrectness(DispatchGen, + HydroGen, + UnsupE, + levels, + HydroPmax, + HydroPmin, + initial_level, + capa, + inflows, + overflow, + pump, + Spillage, + DTG_MRG); + + std::vector OutHydroGen = HydroGen; + std::vector OutUnsupE = UnsupE; + + int loop = 1000; + double eps = 1e-3; + double top = *std::max_element(DispatchGen.begin(), DispatchGen.end()) + + *std::max_element(HydroGen.begin(), HydroGen.end()) + + *std::max_element(UnsupE.begin(), UnsupE.end()) + 1; + + std::vector enabledHours(DispatchGen.size(), false); + for (unsigned int h = 0; h < enabledHours.size(); h++) + { + if (Spillage[h] + DTG_MRG[h] == 0. && HydroGen[h] + UnsupE[h] > 0.) + { + enabledHours[h] = true; + } + } + + std::vector TotalGen(DispatchGen.size()); + std::transform(DispatchGen.begin(), + DispatchGen.end(), + HydroGen.begin(), + TotalGen.begin(), + std::plus<>()); + + while (loop-- > 0) + { + std::vector triedBottom(DispatchGen.size(), false); + double delta = 0; + + while (true) + { + int hourBottom = find_min_index(TotalGen, + OutUnsupE, + OutHydroGen, + triedBottom, + HydroPmax, + enabledHours, + top); + if (hourBottom == -1) + { + break; + } + + std::vector triedPeak(DispatchGen.size(), false); + while (true) + { + int hourPeak = find_max_index(TotalGen, + OutHydroGen, + triedPeak, + HydroPmin, + enabledHours, + TotalGen[hourBottom], + eps); + if (hourPeak == -1) + { + break; + } + + std::vector intermediate_level(levels.begin() + + std::min(hourBottom, hourPeak), + levels.begin() + + std::max(hourBottom, hourPeak)); + double max_pic, max_creux; + if (hourBottom < hourPeak) + { + max_pic = capa; + max_creux = *std::min_element(intermediate_level.begin(), + intermediate_level.end()); + } + else + { + max_pic = capa + - *std::max_element(intermediate_level.begin(), + intermediate_level.end()); + max_creux = capa; + } + + max_pic = std::min(OutHydroGen[hourPeak] - HydroPmin[hourPeak], max_pic); + max_creux = std::min({HydroPmax[hourBottom] - OutHydroGen[hourBottom], + OutUnsupE[hourBottom], + max_creux}); + + double dif_pic_creux = std::max(TotalGen[hourPeak] - TotalGen[hourBottom], 0.); + + delta = std::max(std::min({max_pic, max_creux, dif_pic_creux / 2.}), 0.); + + if (delta > 0) + { + OutHydroGen[hourPeak] -= delta; + OutHydroGen[hourBottom] += delta; + OutUnsupE[hourPeak] = HydroGen[hourPeak] + UnsupE[hourPeak] + - OutHydroGen[hourPeak]; + OutUnsupE[hourBottom] = HydroGen[hourBottom] + UnsupE[hourBottom] + - OutHydroGen[hourBottom]; + break; + } + else + { + triedPeak[hourPeak] = true; + } + } + + if (delta > 0) + { + break; + } + triedBottom[hourBottom] = true; + } + + if (delta == 0) + { + break; + } + + std::transform(DispatchGen.begin(), + DispatchGen.end(), + OutHydroGen.begin(), + TotalGen.begin(), + std::plus<>()); + levels[0] = initial_level + inflows[0] - overflow[0] + pump[0] - OutHydroGen[0]; + for (size_t h = 1; h < levels.size(); ++h) + { + levels[h] = levels[h - 1] + inflows[h] - overflow[h] + pump[h] - OutHydroGen[h]; + } + } + return {OutHydroGen, OutUnsupE, levels}; +} + +} // End namespace Antares::Solver::Simulation diff --git a/src/tests/src/solver/simulation/CMakeLists.txt b/src/tests/src/solver/simulation/CMakeLists.txt index 76d67765f7..e6ca5e7900 100644 --- a/src/tests/src/solver/simulation/CMakeLists.txt +++ b/src/tests/src/solver/simulation/CMakeLists.txt @@ -6,57 +6,66 @@ set(src_solver_hydro "${CMAKE_SOURCE_DIR}/solver/hydro") set(src_libs_antares_study "${CMAKE_SOURCE_DIR}/libs/antares/study") set(SRC_TS_NUMBERS - # For confort in IDE, but not necessary - ${src_solver_simulation}/include/antares/solver/simulation/timeseries-numbers.h - - # Necessary cpp files - ${src_solver_simulation}/timeseries-numbers.cpp - ${src_solver_simulation}/include/antares/solver/simulation/ITimeSeriesNumbersWriter.h) + # For confort in IDE, but not necessary + ${src_solver_simulation}/include/antares/solver/simulation/timeseries-numbers.h + # Necessary cpp files + ${src_solver_simulation}/timeseries-numbers.cpp + ${src_solver_simulation}/include/antares/solver/simulation/ITimeSeriesNumbersWriter.h) add_boost_test(tests-ts-numbers - SRC tests-ts-numbers.cpp ${SRC_TS_NUMBERS} - INCLUDE - "${src_solver_simulation}" - "${src_libs_antares_study}" - LIBS - Antares::utils - model_antares - antares-solver-simulation - antares-solver-ts-generator) + SRC tests-ts-numbers.cpp ${SRC_TS_NUMBERS} + INCLUDE + "${src_solver_simulation}" + "${src_libs_antares_study}" + LIBS + Antares::utils + model_antares + antares-solver-simulation + antares-solver-ts-generator) # =================================== # Tests on area's store-timeseries-number # =================================== add_boost_test(test-store-timeseries-number - SRC test-store-timeseries-number.cpp - LIBS - test_utils_unit - antares-solver-simulation - Antares::study - Antares::result_writer) + SRC test-store-timeseries-number.cpp + LIBS + test_utils_unit + antares-solver-simulation + Antares::study + Antares::result_writer) # =================================== # Tests on time series # =================================== add_boost_test(test-time_series - SRC test-time_series.cpp - LIBS - test_utils_unit - antares-solver-simulation - Antares::study) + SRC test-time_series.cpp + LIBS + test_utils_unit + antares-solver-simulation + Antares::study) # =================================== # Tests on hydro final reservoir level functions # =================================== add_boost_test(test-hydro_final - SRC - test-hydro-final-reservoir-level-functions.cpp - INCLUDE - "${src_solver_simulation}" - "${src_libs_antares_study}" - "${src_solver_hydro}" - LIBS - Antares::study - antares-solver-simulation - Antares::array) + SRC + test-hydro-final-reservoir-level-functions.cpp + INCLUDE + "${src_solver_simulation}" + "${src_libs_antares_study}" + "${src_solver_hydro}" + LIBS + Antares::study + antares-solver-simulation + Antares::array) + +# =================================== +# Tests on hydro remix algorithm +# =================================== +add_boost_test(tests-on-hydro-remix + SRC + test-hydro-remix.cpp + LIBS + shave-peaks-by-remix-hydro + test_utils_unit) \ No newline at end of file diff --git a/src/tests/src/solver/simulation/test-hydro-remix.cpp b/src/tests/src/solver/simulation/test-hydro-remix.cpp new file mode 100644 index 0000000000..ffcef67007 --- /dev/null +++ b/src/tests/src/solver/simulation/test-hydro-remix.cpp @@ -0,0 +1,788 @@ +#define BOOST_TEST_MODULE hydro remix + +#define WIN32_LEAN_AND_MEAN + +#include +#include +#include + +#include + +#include "antares/solver/simulation/shave-peaks-by-remix-hydro.h" + +using namespace Antares::Solver::Simulation; + +template +struct InputFixture +{ + InputFixture() + { + TotalGenNoHydro.assign(size, 0.); + HydroGen.assign(size, 0.); + UnsupE.assign(size, 0.); + HydroPmax.assign(size, std::numeric_limits::max()); + HydroPmin.assign(size, 0.); + inflows.assign(size, 0.); + ovf.assign(size, 0.); + pump.assign(size, 0.); + Spillage.assign(size, 0.); + DTG_MRG.assign(size, 0.); + } + + std::vector TotalGenNoHydro, HydroGen, UnsupE, HydroPmax, HydroPmin, inflows, ovf, pump, + Spillage, DTG_MRG; + double init_level = 0.; + double capacity = std::numeric_limits::max(); +}; + +BOOST_FIXTURE_TEST_CASE(input_vectors_of_different_sizes__exception_raised, InputFixture<0>) +{ + HydroGen = {0., 0.}; + + BOOST_CHECK_EXCEPTION(shavePeaksByRemixingHydro(TotalGenNoHydro, + HydroGen, + UnsupE, + HydroPmax, + HydroPmin, + init_level, + capacity, + inflows, + ovf, + pump, + Spillage, + DTG_MRG), + std::invalid_argument, + checkMessage("Remix hydro input : arrays of different sizes")); +} + +BOOST_FIXTURE_TEST_CASE(input_init_level_exceeds_capacity__exception_raised, InputFixture<0>) +{ + HydroGen = {0., 0.}; + init_level = 2.; + capacity = 1.; + + BOOST_CHECK_EXCEPTION(shavePeaksByRemixingHydro(TotalGenNoHydro, + HydroGen, + UnsupE, + HydroPmax, + HydroPmin, + init_level, + capacity, + inflows, + ovf, + pump, + Spillage, + DTG_MRG), + std::invalid_argument, + checkMessage("Remix hydro input : initial level > reservoir capacity")); +} + +BOOST_FIXTURE_TEST_CASE(all_input_arrays_of_size_0__exception_raised, InputFixture<0>) +{ + init_level = 0.; + capacity = 1.; + + BOOST_CHECK_EXCEPTION(shavePeaksByRemixingHydro(TotalGenNoHydro, + HydroGen, + UnsupE, + HydroPmax, + HydroPmin, + init_level, + capacity, + inflows, + ovf, + pump, + Spillage, + DTG_MRG), + std::invalid_argument, + checkMessage("Remix hydro input : all arrays of sizes 0")); +} + +BOOST_FIXTURE_TEST_CASE(Hydro_gen_not_smaller_than_pmax__exception_raised, InputFixture<5>) +{ + HydroGen = {1., 2., 3., 4., 5.}; + HydroPmax = {2., 2., 2., 4., 5.}; + init_level = 0.; + capacity = 1.; + + BOOST_CHECK_EXCEPTION( + shavePeaksByRemixingHydro(TotalGenNoHydro, + HydroGen, + UnsupE, + HydroPmax, + HydroPmin, + init_level, + capacity, + inflows, + ovf, + pump, + Spillage, + DTG_MRG), + std::invalid_argument, + checkMessage("Remix hydro input : Hydro generation not smaller than Pmax everywhere")); +} + +BOOST_FIXTURE_TEST_CASE(Hydro_gen_not_greater_than_pmin__exception_raised, InputFixture<5>) +{ + HydroGen = {1., 2., 3., 4., 5.}; + HydroPmin = {0., 0., 4., 0., 0.}; + init_level = 0.; + capacity = 1.; + + BOOST_CHECK_EXCEPTION( + shavePeaksByRemixingHydro(TotalGenNoHydro, + HydroGen, + UnsupE, + HydroPmax, + HydroPmin, + init_level, + capacity, + inflows, + ovf, + pump, + Spillage, + DTG_MRG), + std::invalid_argument, + checkMessage("Remix hydro input : Hydro generation not greater than Pmin everywhere")); +} + +BOOST_FIXTURE_TEST_CASE(input_is_acceptable__no_exception_raised, InputFixture<1>) +{ + init_level = 0.; + capacity = 1.; + + BOOST_CHECK_NO_THROW(shavePeaksByRemixingHydro(TotalGenNoHydro, + HydroGen, + UnsupE, + HydroPmax, + HydroPmin, + init_level, + capacity, + inflows, + ovf, + pump, + Spillage, + DTG_MRG)); +} + +BOOST_FIXTURE_TEST_CASE( + hydro_increases_and_pmax_40mwh___Hydro_gen_is_flattened_to_mean_Hydro_gen_20mwh, + InputFixture<5>) +{ + std::ranges::fill(HydroPmax, 40.); + std::ranges::fill(TotalGenNoHydro, 100.); + HydroGen = {0., 10., 20., 30., 40.}; // we have Pmin <= HydroGen <= Pmax + UnsupE = {80.0, 60., 40., 20., 0.}; + init_level = 500.; + capacity = 1000.; + + auto [OutHydroGen, OutUnsupE, _] = shavePeaksByRemixingHydro(TotalGenNoHydro, + HydroGen, + UnsupE, + HydroPmax, + HydroPmin, + init_level, + capacity, + inflows, + ovf, + pump, + Spillage, + DTG_MRG); + + std::vector expected_HydroGen = {20., 20., 20., 20., 20.}; + // UnsupE such as TotalGenNoHydro + HydroGen + UnsupE remains flat + std::vector expected_UnsupE = {60., 50., 40., 30., 20.}; + BOOST_CHECK(OutHydroGen == expected_HydroGen); + BOOST_CHECK(OutUnsupE == expected_UnsupE); +} + +BOOST_FIXTURE_TEST_CASE(Pmax_does_not_impact_results_when_greater_than_40mwh, InputFixture<5>) +{ + std::ranges::fill(HydroPmax, 50.); + std::ranges::fill(TotalGenNoHydro, 100.); + HydroGen = {0., 10., 20., 30., 40.}; + UnsupE = {80.0, 60., 40., 20., 0.}; + init_level = 500.; + capacity = 1000.; + + auto [OutHydroGen, OutUnsupE, _] = shavePeaksByRemixingHydro(TotalGenNoHydro, + HydroGen, + UnsupE, + HydroPmax, + HydroPmin, + init_level, + capacity, + inflows, + ovf, + pump, + Spillage, + DTG_MRG); + + std::vector expected_HydroGen = {20., 20., 20., 20., 20.}; + // UnsupE such as TotalGenNoHydro + HydroGen + UnsupE remains constant at each hour + std::vector expected_UnsupE = {60., 50., 40., 30., 20.}; + BOOST_CHECK(OutHydroGen == expected_HydroGen); + BOOST_CHECK(OutUnsupE == expected_UnsupE); +} + +BOOST_FIXTURE_TEST_CASE( + hydro_decreases_and_pmax_40mwh___Hydro_gen_is_flattened_to_mean_Hydro_gen_20mwh, + InputFixture<5>) +{ + std::ranges::fill(HydroPmax, 40.); + std::ranges::fill(TotalGenNoHydro, 100.); + HydroGen = {40., 30., 20., 10., 0.}; + UnsupE = {0., 20., 40., 60., 80.}; + init_level = 500.; + capacity = 1000.; + + auto [OutHydroGen, OutUnsupE, _] = shavePeaksByRemixingHydro(TotalGenNoHydro, + HydroGen, + UnsupE, + HydroPmax, + HydroPmin, + init_level, + capacity, + inflows, + ovf, + pump, + Spillage, + DTG_MRG); + + std::vector expected_HydroGen = {20., 20., 20., 20., 20.}; + // UnsupE such as TotalGenNoHydro + HydroGen + UnsupE remains constant at each hour + std::vector expected_UnsupE = {20., 30., 40., 50., 60.}; + BOOST_CHECK(OutHydroGen == expected_HydroGen); + BOOST_CHECK(OutUnsupE == expected_UnsupE); +} + +BOOST_FIXTURE_TEST_CASE(influence_of_pmax, InputFixture<5>, *boost::unit_test::tolerance(0.01)) +{ + // TotalGenNoHydro decreases + TotalGenNoHydro = {100., 80., 60., 40., 20.}; + + // HydroGen is flat and must respect HydroGen <= Pmax everywhere + HydroGen = {20., 20., 20., 20., 20.}; + UnsupE = {50., 50., 50., 50., 50.}; + init_level = 500.; + capacity = 1000.; + + // 1. Algorithm tends to flatten TotalGenNoHydro + HydroGen, so it would require HydroGen to + // increase. Proof : + auto [OutHydroGen_1, new_D1, L] = shavePeaksByRemixingHydro(TotalGenNoHydro, + HydroGen, + UnsupE, + HydroPmax, + HydroPmin, + init_level, + capacity, + inflows, + ovf, + pump, + Spillage, + DTG_MRG); + + std::vector expected_HydroGen_1 = {0., 0., 13.33, 33.33, 53.33}; + BOOST_TEST(OutHydroGen_1 == expected_HydroGen_1, boost::test_tools::per_element()); + + // 2. But HydroGen is limited by HydroPmax. So Algo does nothing in the end. + // Proof : + HydroPmax = {20., 20., 20., 20., 20.}; + auto [OutHydroGen_2, OutUnsupE_2, _] = shavePeaksByRemixingHydro(TotalGenNoHydro, + HydroGen, + UnsupE, + HydroPmax, + HydroPmin, + init_level, + capacity, + inflows, + ovf, + pump, + Spillage, + DTG_MRG); + + std::vector expected_HydroGen_2 = {20., 20., 20., 20., 20.}; + std::vector expected_UnsupE_2 = {50., 50., 50., 50., 50.}; + BOOST_CHECK(OutHydroGen_2 == expected_HydroGen_2); + BOOST_CHECK(OutUnsupE_2 == expected_UnsupE_2); +} + +BOOST_FIXTURE_TEST_CASE(influence_of_pmin, InputFixture<5>, *boost::unit_test::tolerance(0.01)) +{ + // TotalGenNoHydro decreases + TotalGenNoHydro = {100., 80., 60., 40., 20.}; + + // HydroGen is flat and must respect Pmin <= HydroGen <= Pmax everywhere + HydroGen = {20., 20., 20., 20., 20.}; + UnsupE = {50., 50., 50., 50., 50.}; + init_level = 500.; + capacity = 1000.; + + // 1. Algorithm tends to flatten TotalGenNoHydro + HydroGen, so it would require HydroGen to + // increase. + auto [OutHydroGen_1, new_D1, L] = shavePeaksByRemixingHydro(TotalGenNoHydro, + HydroGen, + UnsupE, + HydroPmax, + HydroPmin, + init_level, + capacity, + inflows, + ovf, + pump, + Spillage, + DTG_MRG); + std::vector expected_HydroGen_1 = {0., 0., 13.33, 33.33, 53.33}; + BOOST_TEST(OutHydroGen_1 == expected_HydroGen_1, boost::test_tools::per_element()); + + // 2. But HydroGen is low bounded by HydroPmin. So Algo does nothing in the end. + HydroPmin = {20., 20., 20., 20., 20.}; + auto [OutHydroGen_2, OutUnsupE_2, _] = shavePeaksByRemixingHydro(TotalGenNoHydro, + HydroGen, + UnsupE, + HydroPmax, + HydroPmin, + init_level, + capacity, + inflows, + ovf, + pump, + Spillage, + DTG_MRG); + + std::vector expected_HydroGen_2 = {20., 20., 20., 20., 20.}; + std::vector expected_UnsupE_2 = {50., 50., 50., 50., 50.}; + BOOST_CHECK(OutHydroGen_2 == expected_HydroGen_2); + BOOST_CHECK(OutUnsupE_2 == expected_UnsupE_2); +} + +BOOST_FIXTURE_TEST_CASE(Hydro_gen_is_already_flat___remix_is_useless__level_easily_computed, + InputFixture<5>) +{ + init_level = 500.; + std::ranges::fill(ovf, 25.); // Cause levels to decrease + std::ranges::fill(HydroGen, 20.); // Cause levels to decrease + std::ranges::fill(inflows, 15.); // Cause levels to increase + std::ranges::fill(pump, 10.); // Cause levels to increase + + auto [OutHydroGen, OutUnsupE, levels] = shavePeaksByRemixingHydro(TotalGenNoHydro, + HydroGen, + UnsupE, + HydroPmax, + HydroPmin, + init_level, + capacity, + inflows, + ovf, + pump, + Spillage, + DTG_MRG); + + std::vector expected_levels = {480., 460., 440., 420., 400.}; + BOOST_TEST(levels == expected_levels, boost::test_tools::per_element()); +} + +BOOST_FIXTURE_TEST_CASE(input_leads_to_levels_over_capacity___exception_raised, InputFixture<5>) +{ + init_level = 500.; + capacity = 550.; + std::ranges::fill(ovf, 15); // Cause levels to decrease + std::ranges::fill(HydroGen, 10); // Cause levels to decrease + std::ranges::fill(inflows, 25); // Cause levels to increase + std::ranges::fill(pump, 20); // Cause levels to increase + + BOOST_CHECK_EXCEPTION( + shavePeaksByRemixingHydro(TotalGenNoHydro, + HydroGen, + UnsupE, + HydroPmax, + HydroPmin, + init_level, + capacity, + inflows, + ovf, + pump, + Spillage, + DTG_MRG), + std::invalid_argument, + checkMessage( + "Remix hydro input : levels computed from input don't respect reservoir bounds")); +} + +BOOST_FIXTURE_TEST_CASE(input_leads_to_levels_less_than_zero___exception_raised, InputFixture<5>) +{ + init_level = 50.; + std::ranges::fill(ovf, 30); // Cause levels to decrease + std::ranges::fill(HydroGen, 10); // Cause levels to decrease + std::ranges::fill(inflows, 5); // Cause levels to increase + std::ranges::fill(pump, 10); // Cause levels to increase + + BOOST_CHECK_EXCEPTION( + shavePeaksByRemixingHydro(TotalGenNoHydro, + HydroGen, + UnsupE, + HydroPmax, + HydroPmin, + init_level, + capacity, + inflows, + ovf, + pump, + Spillage, + DTG_MRG), + std::invalid_argument, + checkMessage( + "Remix hydro input : levels computed from input don't respect reservoir bounds")); +} + +BOOST_FIXTURE_TEST_CASE(influence_of_capacity_on_algorithm___case_where_no_influence, + InputFixture<10>, + *boost::unit_test::tolerance(0.001)) +{ + std::ranges::fill(UnsupE, 20); // Not important for this test + + // HydroGen oscillates between 10 and 20 (new HydroGen will be flattened to 15 everywhere) + HydroGen = {10., 20., 10., 20., 10., 20., 10., 20., 10., 20.}; + // First inflows > HydroGen, then inflows < HydroGen. Consequence : levels first increase, then + // decrease. + inflows = {25., 25., 25., 25., 25., 5., 5., 5., 5., 5.}; + init_level = 100.; + // HydroGen and inflows result in : input_levels = {115, 120, 135, 140, 155, 140, 135, 120, 115, + // 100} + // Note that : sup(input_levels) = 155 + + // Case 1 : capacity relaxed (infinite by default) ==> leads to optimal solution (HydroGen is + // flat) + auto [OutHydroGen, OutUnsupE, L] = shavePeaksByRemixingHydro(TotalGenNoHydro, + HydroGen, + UnsupE, + HydroPmax, + HydroPmin, + init_level, + capacity, + inflows, + ovf, + pump, + Spillage, + DTG_MRG); + + // HydroGen is flat and is 15. (means of initial HydroGen) + std::vector expected_HydroGen(10, 15.); + // Levels associated to new HydroGen are such as sup(L) = 150. < sup(input_levels) = 155 + std::vector expected_L = {110., 120., 130., 140., 150., 140., 130., 120., 110., 100.}; + BOOST_TEST(OutHydroGen == expected_HydroGen, boost::test_tools::per_element()); + BOOST_TEST(L == expected_L, boost::test_tools::per_element()); + + // Case 2 : now, if we lower capacity to sup(input_levels) = 155, we should + // have HydroGen and L identical to previously : this value of capacity should + // not have an influence on HydroGen and levels as results of the algorithm. + capacity = 155.; + auto [OutHydroGen_2, OutUnsupE_2, L2] = shavePeaksByRemixingHydro(TotalGenNoHydro, + HydroGen, + UnsupE, + HydroPmax, + HydroPmin, + init_level, + capacity, + inflows, + ovf, + pump, + Spillage, + DTG_MRG); + + BOOST_TEST(OutHydroGen_2 == expected_HydroGen, boost::test_tools::per_element()); + BOOST_TEST(L2 == expected_L, boost::test_tools::per_element()); +} + +BOOST_FIXTURE_TEST_CASE(lowering_capacity_too_low_leads_to_suboptimal_solution_for_GplusH, + InputFixture<10>, + *boost::unit_test::tolerance(0.001)) +{ + std::ranges::fill(UnsupE, 20); // Not important for this test + + // HydroGen oscillates between 10 and 20 (new HydroGen will be flattened to 15 everywhere) + HydroGen = {20., 10., 20., 10., 20., 10., 20., 10., 20., 10.}; + // First inflows > HydroGen, then inflows < HydroGen. Consequence : levels first increase, then + // decrease. + inflows = {25., 25., 25., 25., 25., 5., 5., 5., 5., 5.}; + init_level = 100.; + // HydroGen and inflows lead to have : + // input_levels = {105, 120, 125, 140, 145, 140, 125, 120, 105,100} + // Note sup(input_levels) = 145 + + // Case 1 : capacity relaxed (infinite by default) ==> leads to optimal solution (HydroGen is + // flat) + auto [OutHydroGen, OutUnsupE, L] = shavePeaksByRemixingHydro(TotalGenNoHydro, + HydroGen, + UnsupE, + HydroPmax, + HydroPmin, + init_level, + capacity, + inflows, + ovf, + pump, + Spillage, + DTG_MRG); + + // HydroGen is flat and is 15. (means of initial HydroGen) + std::vector expected_HydroGen(10, 15.); + // Levels associated to new HydroGen are such as sup(L) = 150. > sup(input_levels) = 145 + std::vector expected_L = {110., 120., 130., 140., 150., 140., 130., 120., 110., 100.}; + BOOST_TEST(OutHydroGen == expected_HydroGen, boost::test_tools::per_element()); + BOOST_TEST(L == expected_L, boost::test_tools::per_element()); + + // Case 2 : we lower capacity to sup(input_levels) = 145. + // This makes input acceptable for algo : levels computed from input have an + // up bound <= capacity + // But this time levels can not increase up to sup(L) = 150., as it would if capacity + // was infinite. Therefore, solution found is suboptimal : we expect to get an + // output HydroGen flat by interval, not flat on the whole domain. + capacity = 145.; + auto [OutHydroGen_2, OutUnsupE_2, L2] = shavePeaksByRemixingHydro(TotalGenNoHydro, + HydroGen, + UnsupE, + HydroPmax, + HydroPmin, + init_level, + capacity, + inflows, + ovf, + pump, + Spillage, + DTG_MRG); + + // OutHydroGen_2 is flat by interval + std::vector expected_HydroGen_2 = {16., 16., 16., 16., 16., 14., 14., 14., 14., 14.}; + BOOST_TEST(OutHydroGen_2 == expected_HydroGen_2, boost::test_tools::per_element()); +} + +BOOST_FIXTURE_TEST_CASE(lowering_initial_level_too_low_leads_to_suboptimal_solution_for_GplusH, + InputFixture<10>, + *boost::unit_test::tolerance(0.001)) +{ + std::ranges::fill(UnsupE, 20); // Not important for this test + + // HydroGen oscillates between 20 and 30 (new HydroGen will be flattened to 25 everywhere) + HydroGen = {20., 30., 20., 30., 20., 30., 20., 30., 20., 30.}; + // First inflows < HydroGen, then inflows > HydroGen. Consequence : levels first decrease, then + // increase. + inflows = {5., 5., 5., 5., 5., 45., 45., 45., 45., 45.}; + capacity = std::numeric_limits::max(); + init_level = 100.; + // HydroGen and inflows result in : input_levels = {85, 60, 45, 20, 5, 20, 45, 60, 85, 100} + // Note : inf(input_levels) = 5 + + // Case 1 : init level (== 100) is high enough so that input levels (computed from input data) + // are acceptable for algorithm (input levels >= 0.), and running algorithm leads to optimal + // solution (OutHydroGen is flat) + auto [OutHydroGen, OutUnsupE, L] = shavePeaksByRemixingHydro(TotalGenNoHydro, + HydroGen, + UnsupE, + HydroPmax, + HydroPmin, + init_level, + capacity, + inflows, + ovf, + pump, + Spillage, + DTG_MRG); + + // HydroGen is flat and is 25. (means of initial HydroGen) + std::vector expected_HydroGen(10, 25.); + // Levels associated to new HydroGen are such as inf(L) = 0. > inf(input_levels) = 5 + std::vector expected_L = {80., 60., 40., 20., 0., 20., 40., 60., 80., 100.}; + BOOST_TEST(OutHydroGen == expected_HydroGen, boost::test_tools::per_element()); + BOOST_TEST(L == expected_L, boost::test_tools::per_element()); + + // Case 2 : we lower initial level. Input data are still acceptable + // for algorithm (despite the new init level), algorithm will have to take the levels lower + // bound (0.) into account. As the levels change, the solution OutHydroGen will be suboptimal, + // that is flat by interval (not flat on the whole domain). + init_level = 95.; + auto [OutHydroGen_2, OutUnsupE_2, L2] = shavePeaksByRemixingHydro(TotalGenNoHydro, + HydroGen, + UnsupE, + HydroPmax, + HydroPmin, + init_level, + capacity, + inflows, + ovf, + pump, + Spillage, + DTG_MRG); + + // OutHydroGen_2 is flat by interval + std::vector expected_HydroGen_2 = {24., 24., 24., 24., 24., 26., 26., 26., 26., 26.}; + BOOST_TEST(OutHydroGen_2 == expected_HydroGen_2, boost::test_tools::per_element()); +} + +BOOST_FIXTURE_TEST_CASE(influence_of_initial_level_on_algorithm___case_where_no_influence, + InputFixture<10>, + *boost::unit_test::tolerance(0.001)) +{ + std::ranges::fill(UnsupE, 20); // Not important for this test + + // HydroGen oscillates between 10 and 20 (new HydroGen will be flattened to 15 everywhere) + HydroGen = {20., 10., 20., 10., 20., 10., 20., 10., 20., 10.}; + // First inflows < HydroGen, then inflows > HydroGen. Consequence : levels first decrease, then + // increase. + inflows = {5., 5., 5., 5., 5., 25., 25., 25., 25., 25.}; + init_level = 100.; + // HydroGen and inflows are such as inf(input_levels) = 45 + + // Case 1 : init level (== 100) is high enough so that input levels (computed from input data) + // are acceptable by algorithm, and levels computed by algorithm (output) are optimal, that + // is computed from a optimal (that is flat) OutHydroGen. + auto [OutHydroGen, OutUnsupE, L] = shavePeaksByRemixingHydro(TotalGenNoHydro, + HydroGen, + UnsupE, + HydroPmax, + HydroPmin, + init_level, + capacity, + inflows, + ovf, + pump, + Spillage, + DTG_MRG); + + // HydroGen is flat and is 15. (means of initial HydroGen) + std::vector expected_HydroGen(10, 15.); + // Levels associated to new HydroGen are such as inf(L) = 50 > inf(input_levels) = 45 + std::vector expected_L = {90., 80., 70., 60., 50., 60., 70., 80., 90., 100.}; + BOOST_TEST(OutHydroGen == expected_HydroGen, boost::test_tools::per_element()); + BOOST_TEST(L == expected_L, boost::test_tools::per_element()); + + // Case 2 : now we lower initial level down to 55. + // In this way, input data is still acceptable for algorithm + // and algorithm won't have to take the levels lower bound (0.) into account. + // The solution OutHydroGen will be optimal, that is flat by interval. + init_level = 55.; + auto [OutHydroGen_2, OutUnsupE_2, L2] = shavePeaksByRemixingHydro(TotalGenNoHydro, + HydroGen, + UnsupE, + HydroPmax, + HydroPmin, + init_level, + capacity, + inflows, + ovf, + pump, + Spillage, + DTG_MRG); + + // OutHydroGen_2 is flat (and optimal) + std::vector expected_HydroGen_2(10, 15.); + BOOST_TEST(OutHydroGen_2 == expected_HydroGen_2, boost::test_tools::per_element()); +} + +BOOST_FIXTURE_TEST_CASE(spillage_positive_at_hour_0___no_change_at_this_hour, InputFixture<5>) +{ + std::ranges::fill(TotalGenNoHydro, 100.); + HydroGen = {40., 30., 20., 10., 0.}; + UnsupE = {0., 20., 40., 60., 80.}; + init_level = 500.; + capacity = 1000.; + // At this stage, DTG_MRG is filled with zeros. Running the algorithm would flatten + // HydroGen to 20 everywhere : HydroGen = {20, 20, 20, 20, 20} + // But : + Spillage[0] = 1.; + // Now, we expect no change for HydroGen at hour 0 + auto [OutHydroGen, __, _] = shavePeaksByRemixingHydro(TotalGenNoHydro, + HydroGen, + UnsupE, + HydroPmax, + HydroPmin, + init_level, + capacity, + inflows, + ovf, + pump, + Spillage, + DTG_MRG); + + std::vector expected_HydroGen = {40., 15., 15., 15., 15.}; + BOOST_CHECK(OutHydroGen == expected_HydroGen); +} + +BOOST_FIXTURE_TEST_CASE(DTG_MRG_positive_on_hour_4___no_change_at_this_hour, InputFixture<5>) +{ + std::ranges::fill(TotalGenNoHydro, 100.); + HydroGen = {40., 30., 20., 10., 0.}; + UnsupE = {0., 20., 40., 60., 80.}; + init_level = 500.; + capacity = 1000.; + // At this stage, DTG_MRG is filled with zeros. Running the algorithm would flatten + // HydroGen to 20 everywhere : HydroGen = {20, 20, 20, 20, 20} + // But : + DTG_MRG[4] = 1.; + // Now, we expect no change for HydroGen at hour 4 + auto [OutHydroGen, OutUnsupE, L] = shavePeaksByRemixingHydro(TotalGenNoHydro, + HydroGen, + UnsupE, + HydroPmax, + HydroPmin, + init_level, + capacity, + inflows, + ovf, + pump, + Spillage, + DTG_MRG); + + std::vector expected_HydroGen = {25., 25., 25., 25., 0.}; + BOOST_CHECK(OutHydroGen == expected_HydroGen); +} + +BOOST_FIXTURE_TEST_CASE(comparison_of_results_with_python_algo, + InputFixture<20>, + *boost::unit_test::tolerance(0.01)) +{ + std::vector load = {46, 81, 89, 42, 69, 55, 88, 46, 84, 94, + 66, 93, 68, 39, 91, 89, 94, 93, 91, 38}; + HydroGen = {10, 40, 36, 8, 13, 33, 9, 0, 24, 18, 5, 47, 29, 6, 7, 54, 49, 11, 63, 21}; + UnsupE = {34, 32, 33, 23, 9, 8, 20, 40, 30, 3, 50, 27, 12, 1, 35, 31, 2, 58, 20, 4}; + // Computing total generation without hydro generation + TotalGenNoHydro = load; + std::ranges::transform(TotalGenNoHydro, + HydroGen, + TotalGenNoHydro.begin(), + std::minus()); + std::ranges::transform(TotalGenNoHydro, UnsupE, TotalGenNoHydro.begin(), std::minus()); + + HydroPmax = {43, 48, 36, 43, 13, 44, 13, 31, 49, 35, 47, 47, 37, 41, 21, 54, 49, 28, 63, 49}; + HydroPmin = {10, 22, 17, 8, 7, 15, 8, 0, 9, 2, 5, 18, 22, 6, 4, 11, 1, 0, 23, 6}; + init_level = 13.6; + capacity = 126.; + inflows = {37, 27, 41, 36, 7, 14, 38, 23, 17, 35, 20, 24, 17, 46, 1, 10, 10, 12, 46, 30}; + + auto [OutHydroGen, OutUnsupE, L] = shavePeaksByRemixingHydro(TotalGenNoHydro, + HydroGen, + UnsupE, + HydroPmax, + HydroPmin, + init_level, + capacity, + inflows, + ovf, + pump, + Spillage, + DTG_MRG); + std::vector expected_HydroGen = {42.3, 35.3, 27., 31., 7., 33., 8., + 31., 19.55, 2., 38.55, 30.55, 22.55, 7., + 4., 45.55, 6.55, 25.55, 41.55, 25.}; + + BOOST_TEST(OutHydroGen == expected_HydroGen, boost::test_tools::per_element()); +} + +// Possible simplifications / clarifications of the algorithm itself : +// - the algo is flat, it's C (not C++), it should be divided in a small number of steps +// - max_pic is an up hydro production margin (Hydro_gen_up_mrg) +// - max_creux is a down hydro production margin (Hydro_gen_down_mrg) +// - an iteration updates OutHydroGen : it's its main job. +// So OutUnsupE could be updated from OutHydroGen at the end of an iteration, separately. +// - they are 3 while loops. 2 loops should be enough (the iteration loop and +// another one simply updating OutHydroGen and OutUnsupE)