mirror of
https://github.com/Akkudoktor-EOS/EOS.git
synced 2025-11-03 16:26:20 +00:00
fix: load data for automatic optimization (#731)
Automatic optimization used to take the adjusted load data even if there were no measurements leading to 0 load values. Split LoadAkkudoktor into LoadAkkudoktor and LoadAkkudoktorAdjusted. This allows to select load data either purely from the load data database or load data additionally adjusted by load measurements. Some value names have been adapted to denote also the unit of a value. For better load bug squashing the optimization solution data availability was improved. For better data visbility prediction data can now be distinguished from solution data in the generic optimization solution. Some predictions that may be of interest to understand the solution were added. Documentation was updated to resemble the addition load prediction provider and the value name changes. Signed-off-by: Bobby Noelte <b0661n0e17e@gmail.com>
This commit is contained in:
@@ -155,7 +155,7 @@ class TestConfigMigration:
|
||||
assert configmigrate.mapped_count >= 1, f"No mapped migrations for {old_file.name}"
|
||||
assert configmigrate.auto_count >= 1, f"No automatic migrations for {old_file.name}"
|
||||
|
||||
assert len(configmigrate.skipped_paths) <= 7, (
|
||||
assert len(configmigrate.skipped_paths) <= 3, (
|
||||
f"Too many skipped paths in {old_file.name}: {configmigrate.skipped_paths}"
|
||||
)
|
||||
|
||||
@@ -174,7 +174,7 @@ class TestConfigMigration:
|
||||
errors = _dict_contains(new_data, expected_data)
|
||||
assert not errors, (
|
||||
f"Migrated config for {old_file.name} is missing or mismatched fields:\n" +
|
||||
"\n".join(errors)
|
||||
"\n".join(errors) + f"\n{new_data}"
|
||||
)
|
||||
|
||||
# --- Compare migrated result with migration map ---
|
||||
|
||||
@@ -8,20 +8,39 @@ from akkudoktoreos.core.ems import get_ems
|
||||
from akkudoktoreos.measurement.measurement import MeasurementDataRecord, get_measurement
|
||||
from akkudoktoreos.prediction.loadakkudoktor import (
|
||||
LoadAkkudoktor,
|
||||
LoadAkkudoktorAdjusted,
|
||||
LoadAkkudoktorCommonSettings,
|
||||
)
|
||||
from akkudoktoreos.utils.datetimeutil import compare_datetimes, to_datetime, to_duration
|
||||
|
||||
|
||||
@pytest.fixture
|
||||
def provider(config_eos):
|
||||
def loadakkudoktor(config_eos):
|
||||
"""Fixture to initialise the LoadAkkudoktor instance."""
|
||||
settings = {
|
||||
"load": {
|
||||
"provider": "LoadAkkudoktor",
|
||||
"provider_settings": {
|
||||
"LoadAkkudoktor": {
|
||||
"loadakkudoktor_year_energy": "1000",
|
||||
"loadakkudoktor_year_energy_kwh": "1000",
|
||||
},
|
||||
},
|
||||
},
|
||||
}
|
||||
config_eos.merge_settings_from_dict(settings)
|
||||
assert config_eos.load.provider == "LoadAkkudoktor"
|
||||
assert config_eos.load.provider_settings.LoadAkkudoktor.loadakkudoktor_year_energy_kwh == 1000
|
||||
return LoadAkkudoktor()
|
||||
|
||||
@pytest.fixture
|
||||
def loadakkudoktoradjusted(config_eos):
|
||||
"""Fixture to initialise the LoadAkkudoktorAdjusted instance."""
|
||||
settings = {
|
||||
"load": {
|
||||
"provider": "LoadAkkudoktorAdjusted",
|
||||
"provider_settings": {
|
||||
"LoadAkkudoktor": {
|
||||
"loadakkudoktor_year_energy_kwh": "1000",
|
||||
},
|
||||
},
|
||||
},
|
||||
@@ -30,10 +49,9 @@ def provider(config_eos):
|
||||
}
|
||||
}
|
||||
config_eos.merge_settings_from_dict(settings)
|
||||
assert config_eos.load.provider == "LoadAkkudoktor"
|
||||
assert config_eos.load.provider_settings.LoadAkkudoktor.loadakkudoktor_year_energy == 1000
|
||||
return LoadAkkudoktor()
|
||||
|
||||
assert config_eos.load.provider == "LoadAkkudoktorAdjusted"
|
||||
assert config_eos.load.provider_settings.LoadAkkudoktor.loadakkudoktor_year_energy_kwh == 1000
|
||||
return LoadAkkudoktorAdjusted()
|
||||
|
||||
@pytest.fixture
|
||||
def measurement_eos():
|
||||
@@ -72,23 +90,23 @@ def mock_load_profiles_file(tmp_path):
|
||||
|
||||
|
||||
def test_loadakkudoktor_settings_validator():
|
||||
"""Test the field validator for `loadakkudoktor_year_energy`."""
|
||||
settings = LoadAkkudoktorCommonSettings(loadakkudoktor_year_energy=1234)
|
||||
assert isinstance(settings.loadakkudoktor_year_energy, float)
|
||||
assert settings.loadakkudoktor_year_energy == 1234.0
|
||||
"""Test the field validator for `loadakkudoktor_year_energy_kwh`."""
|
||||
settings = LoadAkkudoktorCommonSettings(loadakkudoktor_year_energy_kwh=1234)
|
||||
assert isinstance(settings.loadakkudoktor_year_energy_kwh, float)
|
||||
assert settings.loadakkudoktor_year_energy_kwh == 1234.0
|
||||
|
||||
settings = LoadAkkudoktorCommonSettings(loadakkudoktor_year_energy=1234.56)
|
||||
assert isinstance(settings.loadakkudoktor_year_energy, float)
|
||||
assert settings.loadakkudoktor_year_energy == 1234.56
|
||||
settings = LoadAkkudoktorCommonSettings(loadakkudoktor_year_energy_kwh=1234.56)
|
||||
assert isinstance(settings.loadakkudoktor_year_energy_kwh, float)
|
||||
assert settings.loadakkudoktor_year_energy_kwh == 1234.56
|
||||
|
||||
|
||||
def test_loadakkudoktor_provider_id(provider):
|
||||
def test_loadakkudoktor_provider_id(loadakkudoktor):
|
||||
"""Test the `provider_id` class method."""
|
||||
assert provider.provider_id() == "LoadAkkudoktor"
|
||||
assert loadakkudoktor.provider_id() == "LoadAkkudoktor"
|
||||
|
||||
|
||||
@patch("akkudoktoreos.prediction.loadakkudoktor.np.load")
|
||||
def test_load_data_from_mock(mock_np_load, mock_load_profiles_file, provider):
|
||||
def test_load_data_from_mock(mock_np_load, mock_load_profiles_file, loadakkudoktor):
|
||||
"""Test the `load_data` method."""
|
||||
# Mock numpy load to return data similar to what would be in the file
|
||||
mock_np_load.return_value = {
|
||||
@@ -97,19 +115,19 @@ def test_load_data_from_mock(mock_np_load, mock_load_profiles_file, provider):
|
||||
}
|
||||
|
||||
# Test data loading
|
||||
data_year_energy = provider.load_data()
|
||||
data_year_energy = loadakkudoktor.load_data()
|
||||
assert data_year_energy is not None
|
||||
assert data_year_energy.shape == (365, 2, 24)
|
||||
|
||||
|
||||
def test_load_data_from_file(provider):
|
||||
def test_load_data_from_file(loadakkudoktor):
|
||||
"""Test `load_data` loads data from the profiles file."""
|
||||
data_year_energy = provider.load_data()
|
||||
data_year_energy = loadakkudoktor.load_data()
|
||||
assert data_year_energy is not None
|
||||
|
||||
|
||||
@patch("akkudoktoreos.prediction.loadakkudoktor.LoadAkkudoktor.load_data")
|
||||
def test_update_data(mock_load_data, provider):
|
||||
def test_update_data(mock_load_data, loadakkudoktor):
|
||||
"""Test the `_update` method."""
|
||||
mock_load_data.return_value = np.random.rand(365, 2, 24)
|
||||
|
||||
@@ -118,27 +136,27 @@ def test_update_data(mock_load_data, provider):
|
||||
ems_eos.set_start_datetime(pendulum.datetime(2024, 1, 1))
|
||||
|
||||
# Assure there are no prediction records
|
||||
provider.clear()
|
||||
assert len(provider) == 0
|
||||
loadakkudoktor.clear()
|
||||
assert len(loadakkudoktor) == 0
|
||||
|
||||
# Execute the method
|
||||
provider._update_data()
|
||||
loadakkudoktor._update_data()
|
||||
|
||||
# Validate that update_value is called
|
||||
assert len(provider) > 0
|
||||
assert len(loadakkudoktor) > 0
|
||||
|
||||
|
||||
def test_calculate_adjustment(provider, measurement_eos):
|
||||
def test_calculate_adjustment(loadakkudoktoradjusted, measurement_eos):
|
||||
"""Test `_calculate_adjustment` for various scenarios."""
|
||||
data_year_energy = np.random.rand(365, 2, 24)
|
||||
|
||||
# Call the method and validate results
|
||||
weekday_adjust, weekend_adjust = provider._calculate_adjustment(data_year_energy)
|
||||
weekday_adjust, weekend_adjust = loadakkudoktoradjusted._calculate_adjustment(data_year_energy)
|
||||
assert weekday_adjust.shape == (24,)
|
||||
assert weekend_adjust.shape == (24,)
|
||||
|
||||
data_year_energy = np.zeros((365, 2, 24))
|
||||
weekday_adjust, weekend_adjust = provider._calculate_adjustment(data_year_energy)
|
||||
weekday_adjust, weekend_adjust = loadakkudoktoradjusted._calculate_adjustment(data_year_energy)
|
||||
|
||||
assert weekday_adjust.shape == (24,)
|
||||
expected = np.array(
|
||||
@@ -203,13 +221,13 @@ def test_calculate_adjustment(provider, measurement_eos):
|
||||
np.testing.assert_array_equal(weekend_adjust, expected)
|
||||
|
||||
|
||||
def test_provider_adjustments_with_mock_data(provider):
|
||||
def test_provider_adjustments_with_mock_data(loadakkudoktoradjusted):
|
||||
"""Test full integration of adjustments with mock data."""
|
||||
with patch(
|
||||
"akkudoktoreos.prediction.loadakkudoktor.LoadAkkudoktor._calculate_adjustment"
|
||||
"akkudoktoreos.prediction.loadakkudoktor.LoadAkkudoktorAdjusted._calculate_adjustment"
|
||||
) as mock_adjust:
|
||||
mock_adjust.return_value = (np.zeros(24), np.zeros(24))
|
||||
|
||||
# Test execution
|
||||
provider._update_data()
|
||||
loadakkudoktoradjusted._update_data()
|
||||
assert mock_adjust.called
|
||||
|
||||
@@ -62,11 +62,11 @@ def test_update_data_calls_update_value(load_vrm_instance):
|
||||
expected_calls = [
|
||||
call(
|
||||
pendulum.datetime(2025, 1, 1, 0, 0, 0, tz='Europe/Berlin'),
|
||||
{"load_mean": 100.5, "load_std": 0.0, "load_mean_adjusted": 100.5}
|
||||
{"loadforecast_power_w": 100.5,}
|
||||
),
|
||||
call(
|
||||
pendulum.datetime(2025, 1, 1, 1, 0, 0, tz='Europe/Berlin'),
|
||||
{"load_mean": 101.2, "load_std": 0.0, "load_mean_adjusted": 101.2}
|
||||
{"loadforecast_power_w": 101.2,}
|
||||
),
|
||||
]
|
||||
|
||||
|
||||
@@ -6,7 +6,10 @@ from akkudoktoreos.prediction.elecpriceenergycharts import ElecPriceEnergyCharts
|
||||
from akkudoktoreos.prediction.elecpriceimport import ElecPriceImport
|
||||
from akkudoktoreos.prediction.feedintarifffixed import FeedInTariffFixed
|
||||
from akkudoktoreos.prediction.feedintariffimport import FeedInTariffImport
|
||||
from akkudoktoreos.prediction.loadakkudoktor import LoadAkkudoktor
|
||||
from akkudoktoreos.prediction.loadakkudoktor import (
|
||||
LoadAkkudoktor,
|
||||
LoadAkkudoktorAdjusted,
|
||||
)
|
||||
from akkudoktoreos.prediction.loadimport import LoadImport
|
||||
from akkudoktoreos.prediction.loadvrm import LoadVrm
|
||||
from akkudoktoreos.prediction.prediction import (
|
||||
@@ -38,6 +41,7 @@ def forecast_providers():
|
||||
FeedInTariffFixed(),
|
||||
FeedInTariffImport(),
|
||||
LoadAkkudoktor(),
|
||||
LoadAkkudoktorAdjusted(),
|
||||
LoadVrm(),
|
||||
LoadImport(),
|
||||
PVForecastAkkudoktor(),
|
||||
@@ -83,14 +87,15 @@ def test_provider_sequence(prediction):
|
||||
assert isinstance(prediction.providers[3], FeedInTariffFixed)
|
||||
assert isinstance(prediction.providers[4], FeedInTariffImport)
|
||||
assert isinstance(prediction.providers[5], LoadAkkudoktor)
|
||||
assert isinstance(prediction.providers[6], LoadVrm)
|
||||
assert isinstance(prediction.providers[7], LoadImport)
|
||||
assert isinstance(prediction.providers[8], PVForecastAkkudoktor)
|
||||
assert isinstance(prediction.providers[9], PVForecastVrm)
|
||||
assert isinstance(prediction.providers[10], PVForecastImport)
|
||||
assert isinstance(prediction.providers[11], WeatherBrightSky)
|
||||
assert isinstance(prediction.providers[12], WeatherClearOutside)
|
||||
assert isinstance(prediction.providers[13], WeatherImport)
|
||||
assert isinstance(prediction.providers[6], LoadAkkudoktorAdjusted)
|
||||
assert isinstance(prediction.providers[7], LoadVrm)
|
||||
assert isinstance(prediction.providers[8], LoadImport)
|
||||
assert isinstance(prediction.providers[9], PVForecastAkkudoktor)
|
||||
assert isinstance(prediction.providers[10], PVForecastVrm)
|
||||
assert isinstance(prediction.providers[11], PVForecastImport)
|
||||
assert isinstance(prediction.providers[12], WeatherBrightSky)
|
||||
assert isinstance(prediction.providers[13], WeatherClearOutside)
|
||||
assert isinstance(prediction.providers[14], WeatherImport)
|
||||
|
||||
|
||||
def test_provider_by_id(prediction, forecast_providers):
|
||||
|
||||
@@ -157,7 +157,7 @@ class TestSystem:
|
||||
result = requests.post(f"{server}/v1/prediction/update/LoadAkkudoktor")
|
||||
assert result.status_code == HTTPStatus.OK
|
||||
|
||||
result = requests.get(f"{server}/v1/prediction/series?key=load_mean")
|
||||
result = requests.get(f"{server}/v1/prediction/series?key=loadforecast_power_w")
|
||||
assert result.status_code == HTTPStatus.OK
|
||||
|
||||
data = result.json()
|
||||
|
||||
2
tests/testdata/eos_config_andreas_now.json
vendored
2
tests/testdata/eos_config_andreas_now.json
vendored
@@ -58,7 +58,7 @@
|
||||
"load": {
|
||||
"provider_settings": {
|
||||
"LoadAkkudoktor": {
|
||||
"loadakkudoktor_year_energy": 13000
|
||||
"loadakkudoktor_year_energy_kwh": 13000
|
||||
}
|
||||
}
|
||||
},
|
||||
|
||||
2
tests/testdata/eosserver_config_1.json
vendored
2
tests/testdata/eosserver_config_1.json
vendored
@@ -15,7 +15,7 @@
|
||||
"provider": "LoadImport",
|
||||
"provider_settings": {
|
||||
"LoadAkkudoktor": {
|
||||
"loadakkudoktor_year_energy": 20000
|
||||
"loadakkudoktor_year_energy_kwh": 20000
|
||||
}
|
||||
}
|
||||
},
|
||||
|
||||
Reference in New Issue
Block a user