Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

New hydro remix algorithm #2532

Merged
merged 45 commits into from
Jan 10, 2025
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
Show all changes
45 commits
Select commit Hold shift + click to select a range
6871e98
New hydro remix : setting unit tests environment
guilpier-code Dec 12, 2024
33a96ea
New hydro remix : some cleaning
guilpier-code Dec 12, 2024
e40b32f
New hydro remix : unit tests on input data
guilpier-code Dec 13, 2024
34f0ed7
New hydro remix : add unit test > pmax is supposed to limit hydro gen…
guilpier-code Dec 13, 2024
cba01ee
New hydro remix : adding tests on Pmax respect
guilpier-code Dec 16, 2024
9f41dc1
New hydro remix : Hydro must respect condition H <= Pmax in input of …
guilpier-code Dec 16, 2024
a52aa7b
New hydro remix : adding a test on influence of Pmax
guilpier-code Dec 16, 2024
52b04e0
New hydro remix : adding a test on influence of Pmin
guilpier-code Dec 16, 2024
56ab22d
New hydro remix : Pmin & Pmax unit tests enhancement
guilpier-code Dec 17, 2024
548fc1f
New hydro remix : change the algorithm due to new specifications
guilpier-code Dec 17, 2024
f3708e2
New hydro remix : adding 2 unit tests on levels computation
guilpier-code Dec 17, 2024
4c8d407
New hydro remix : improve comparison between 2 std::vector<double>
guilpier-code Dec 18, 2024
88dd5ae
New hydro remix : make a test rightfully fail
guilpier-code Dec 18, 2024
d3a24ef
New hydro remix : adding unit tests on levels computed from input data
guilpier-code Dec 18, 2024
cd35bdb
New hydro remix : adding a failing test about of impact reservoir cap…
guilpier-code Dec 19, 2024
f7757b1
New hydro remix : change algorithm to try to fix a test failure
guilpier-code Dec 19, 2024
ea7e23f
New hydro remix : oops, tiny fix
guilpier-code Dec 19, 2024
6ed6f62
New hydro remix : fix the previous fix
guilpier-code Dec 19, 2024
adb89a7
New hydro remix : adding a unit test on getting a sub optimal solutio…
guilpier-code Dec 19, 2024
72f75e4
New hydro remix : adding a unit test about the effect of lowering ini…
guilpier-code Dec 19, 2024
08bbd70
New hydro remix : adding test that case where initial level has no in…
guilpier-code Dec 19, 2024
f0854dd
New hydro remix : creating a header for remix hydro algorithm
guilpier-code Dec 20, 2024
4685abb
New hydro remix : define and use proper comparison operator when chec…
guilpier-code Dec 20, 2024
957cd5b
[skip ci] New hydro remix : replace use of std::adjacent_find with us…
guilpier-code Dec 20, 2024
6157f34
[skip ci] New hydro remix : make code comment more clear
guilpier-code Dec 20, 2024
889b982
New hydro remix : improve unit tests by use of a fixture
guilpier-code Dec 20, 2024
da27cb9
New hydro remix : adding test on invariance where S + DTG_MRG > 0
guilpier-code Dec 27, 2024
91e0a0a
New hydro remix : renaming algorithm input data
guilpier-code Jan 6, 2025
c02210a
New hydro remix : renaming algorithm input data (part 2)
guilpier-code Jan 6, 2025
7d71f04
New hydro remix : unit tests - renaming algorithm input data
guilpier-code Jan 6, 2025
237940e
New hydro remix : unit tests - renaming algorithm input data (part 2)
guilpier-code Jan 6, 2025
47af157
Merge remote-tracking branch 'remotes/origin/develop' into feature/ne…
guilpier-code Jan 7, 2025
8ce9f25
New hydro remix : some more renaming
guilpier-code Jan 7, 2025
65102e2
fix vcpkg
flomnes Jan 7, 2025
41a811a
New hydro remix : more renaming
guilpier-code Jan 7, 2025
bd8d62d
New hydro remix : small correction on algorithm initialization
guilpier-code Jan 7, 2025
511435a
New hydro remix : renaming the algorithm itself
guilpier-code Jan 7, 2025
37bb2be
New hydro remix : create a cmake target for remix hydro algorithm
guilpier-code Jan 7, 2025
a4a68ac
New hydro remix : renaming locally in algorithm
guilpier-code Jan 7, 2025
2309099
New hydro remix : more algorithm local renaming
guilpier-code Jan 7, 2025
1e51fa2
New hydro remix : renaming
guilpier-code Jan 8, 2025
b7107fa
[skip ci] New hydro remix : tiny correction due to review
guilpier-code Jan 8, 2025
d06eeae
New hydro remix : adding a test for comparison with python script
guilpier-code Jan 9, 2025
42352d1
[skip ci] New hydro remix : renaming mainly
guilpier-code Jan 9, 2025
35c4800
Small improvements for hydro remix (#2570)
flomnes Jan 10, 2025
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
12 changes: 12 additions & 0 deletions src/solver/simulation/CMakeLists.txt
Original file line number Diff line number Diff line change
Expand Up @@ -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
$<BUILD_INTERFACE:${CMAKE_CURRENT_SOURCE_DIR}/include>)


Comment on lines +79 to +90
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

🥇

install(DIRECTORY include/antares
DESTINATION "include"
)
Original file line number Diff line number Diff line change
@@ -0,0 +1,29 @@

#pragma once

#include <vector>

namespace Antares::Solver::Simulation
{

struct RemixHydroOutput
{
std::vector<double> HydroGen;
std::vector<double> UnsupE;
std::vector<double> levels;
};

RemixHydroOutput shavePeaksByRemixingHydro(const std::vector<double>& DispatchGen,
const std::vector<double>& HydroGen,
const std::vector<double>& UnsupE,
const std::vector<double>& HydroPmax,
const std::vector<double>& HydroPmin,
double init_level,
double capacity,
const std::vector<double>& inflow,
const std::vector<double>& overflow,
const std::vector<double>& pump,
const std::vector<double>& Spillage,
const std::vector<double>& DTG_MRG);

} // namespace Antares::Solver::Simulation
309 changes: 309 additions & 0 deletions src/solver/simulation/shave-peaks-by-remix-hydro.cpp
Original file line number Diff line number Diff line change
@@ -0,0 +1,309 @@
#include "include/antares/solver/simulation/shave-peaks-by-remix-hydro.h"

#include <algorithm>
#include <ranges>
#include <stdexcept>
#include <vector>

namespace Antares::Solver::Simulation
{

int find_min_index(const std::vector<double>& TotalGen,
const std::vector<double>& OutUnsupE,
const std::vector<double>& OutHydroGen,
const std::vector<bool>& triedBottom,
const std::vector<double>& HydroPmax,
const std::vector<bool>& 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<double>& TotalGen,
const std::vector<double>& OutHydroGen,
const std::vector<bool>& triedPeak,
const std::vector<double>& HydroPmin,
const std::vector<bool>& 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<double>& a, const std::vector<double>& 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<double>& v, const double c)
{
return std::ranges::all_of(v, [&c](const double& e) { return e <= c; });
}

static bool operator>=(const std::vector<double>& v, const double c)
{
return std::ranges::all_of(v, [&c](const double& e) { return e >= c; });
}

static void checkInputCorrectness(const std::vector<double>& DispatchGen,
const std::vector<double>& HydroGen,
const std::vector<double>& UnsupE,
const std::vector<double>& levels,
const std::vector<double>& HydroPmax,
const std::vector<double>& HydroPmin,
double initial_level,
double capacity,
const std::vector<double>& inflows,
const std::vector<double>& overflow,
const std::vector<double>& pump,
const std::vector<double>& Spillage,
const std::vector<double>& 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<size_t> 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<double>& DispatchGen,
const std::vector<double>& HydroGen,
const std::vector<double>& UnsupE,
const std::vector<double>& HydroPmax,
const std::vector<double>& HydroPmin,
double initial_level,
double capa,
const std::vector<double>& inflows,
const std::vector<double>& overflow,
const std::vector<double>& pump,
const std::vector<double>& Spillage,
const std::vector<double>& DTG_MRG)
{
std::vector<double> 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<double> OutHydroGen = HydroGen;
std::vector<double> 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<bool> 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<double> TotalGen(DispatchGen.size());
std::transform(DispatchGen.begin(),
DispatchGen.end(),
HydroGen.begin(),
TotalGen.begin(),
std::plus<>());

while (loop-- > 0)
{
std::vector<bool> 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<bool> 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<double> 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
Loading
Loading