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

Tipping points #446

Open
wants to merge 62 commits into
base: main
Choose a base branch
from
Open
Show file tree
Hide file tree
Changes from all commits
Commits
Show all changes
62 commits
Select commit Hold shift + click to select a range
5ceb865
Update function to handle edge case
dumontgoulart Apr 12, 2024
b014275
added config for benefits visualization threshold (#387)
dumontgoulart Apr 30, 2024
da9249d
Update tipping point#
dumontgoulart May 15, 2024
6dfae92
update tipping_points from main (#439)
dumontgoulart May 15, 2024
45ab27a
Merge remote-tracking branch 'origin/main' into tipping_points
dumontgoulart May 15, 2024
4d307a4
Update TippingPointMetrics enum with additional options
dumontgoulart May 16, 2024
cf8d0c8
Edits on tp
dumontgoulart May 16, 2024
1e91d40
more edits
dumontgoulart May 16, 2024
fd50713
updating deprecated terms
dumontgoulart May 16, 2024
34f4034
Update TippingPointMetrics enum with additional options
dumontgoulart May 16, 2024
c2f1bfa
Update TippingPointMetrics enum with additional options
dumontgoulart May 17, 2024
26ffea2
Update TippingPointMetrics enum with additional options
dumontgoulart May 17, 2024
8a1a748
Update TippingPointMetrics enum with additional options
dumontgoulart Jun 11, 2024
3ab3964
Merge branch 'main' into tipping_points
dumontgoulart Jun 18, 2024
b44fba8
chore: Refactor object_model/tipping_point.py and interface/tipping_p…
dumontgoulart Jun 19, 2024
2ff6088
Update paths in dbs_controller.py for database_name
dumontgoulart Jun 19, 2024
1f8f97c
refactor functions
dumontgoulart Jun 20, 2024
9ce90fb
add tests
dumontgoulart Jun 21, 2024
f028b87
add tests for tipping points
dumontgoulart Jun 21, 2024
e2f102c
updating tipping points wrt main (#479)
dumontgoulart Jul 9, 2024
15f63a7
update tipping_points wrt main (#480)
dumontgoulart Jul 9, 2024
fa09dbf
refactor functions
dumontgoulart Jul 9, 2024
f108e9a
Refactor object_model/tipping_point.py and interface/tipping_points.py
dumontgoulart Jul 10, 2024
a4f8860
chore: Refactor object_model/tipping_point.py and interface/tipping_p…
dumontgoulart Jul 10, 2024
8bc416c
update tp from main (#482)
dumontgoulart Jul 11, 2024
c1524ed
comments
dumontgoulart Jul 11, 2024
8f57e19
Refactor object_model/tipping_point.py and interface/tipping_points.py
dumontgoulart Jul 11, 2024
2a620bd
update tp (#486)
dumontgoulart Jul 11, 2024
7f2b41d
Merge remote-tracking branch 'origin/main' into tipping_points
dumontgoulart Jul 11, 2024
0a398de
chore: Refactor object_model/tipping_point.py and interface/tipping_p…
dumontgoulart Jul 11, 2024
0522ab2
Refactor object_model/tipping_point.py and interface/tipping_points.py
dumontgoulart Jul 11, 2024
28df051
Refactor object_model/tipping_point.py and interface/tipping_points.py
dumontgoulart Jul 18, 2024
c7e59bc
update tp (#496)
dumontgoulart Jul 18, 2024
4cec8c8
Merge branch 'main' into tipping_points
dumontgoulart Jul 18, 2024
49477fb
chore: Refactor object_model/tipping_point.py and interface/tipping_p…
dumontgoulart Jul 18, 2024
bb29294
Refactor object_model/tipping_point.py and interface/tipping_points.py
dumontgoulart Jul 19, 2024
914708f
Refactor object_model/tipping_point.py and interface/tipping_points.py
dumontgoulart Jul 19, 2024
509f586
Refactor object_model/tipping_point.py and interface/tipping_points.py
dumontgoulart Aug 26, 2024
0d87a9b
update tp from main (#518)
dumontgoulart Aug 26, 2024
beeb7d3
update tipping points to db and api
dumontgoulart Aug 26, 2024
29a0eec
Refactor object_model/tipping_point.py and interface/tipping_points.py
dumontgoulart Aug 27, 2024
a65eccd
Merge branch 'main' into tipping_points
dumontgoulart Aug 27, 2024
5394985
Refactor object_model/tipping_point.py and interface/tipping_points.py
dumontgoulart Aug 27, 2024
4767068
Refactor object_model/tipping_point.py and interface/tipping_points.py
dumontgoulart Aug 28, 2024
ad8cd88
Refactor object_model/tipping_point.py to ensure availability of Data…
dumontgoulart Aug 28, 2024
971d7f6
Merge branch 'main' into tipping_points
dumontgoulart Aug 28, 2024
a1d6f57
Refactor object_model/tipping_point.py and interface/tipping_points.py
dumontgoulart Aug 28, 2024
dacdb31
Merge branch 'tipping_points' of https://github.com/Deltares-research…
dumontgoulart Aug 28, 2024
3360678
Refactor object_model/tipping_point.py and interface/tipping_points.py
dumontgoulart Aug 28, 2024
8170ddd
Refactor TippingPoint class to update projection name and save it if …
dumontgoulart Sep 18, 2024
c6c53f6
Merge branch 'main' of https://github.com/Deltares-research/FloodAdap…
Santonia27 Oct 2, 2024
a7b16b0
adding "_tp_" to paths for easy locating
dumontgoulart Oct 3, 2024
279ee5f
invert order database
dumontgoulart Oct 8, 2024
f638cde
Merge branch 'main' of https://github.com/Deltares-research/FloodAdap…
Santonia27 Oct 29, 2024
231f84a
instead of plot outside gui plot inside (still needs fixing)
Santonia27 Oct 29, 2024
69a7ae0
add annotation to plot
Santonia27 Oct 30, 2024
4e35370
add y axis label
Santonia27 Oct 30, 2024
bc414fc
add view plot button
Santonia27 Nov 4, 2024
d9506c6
fix plot
Santonia27 Nov 4, 2024
86821f4
Merge branch 'main' of https://github.com/Deltares-research/FloodAdap…
Santonia27 Nov 4, 2024
81ecfcd
remove comments
Santonia27 Nov 5, 2024
76b59f8
update plot labels
Santonia27 Nov 6, 2024
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
43 changes: 43 additions & 0 deletions flood_adapt/api/tipping_points.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,43 @@
from typing import Any

from flood_adapt.dbs_controller import Database
from flood_adapt.object_model.interface.tipping_points import ITipPoint
from flood_adapt.object_model.tipping_point import TippingPoint

Copy link
Author

Choose a reason for hiding this comment

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

All based on the existing code, changing paths and variables.


def get_tipping_points() -> dict[str, Any]:
# sorting and filtering either with PyQt table or in the API
return Database().tipping_points.list_objects()


def get_tipping_point(name: str) -> ITipPoint:
return Database().tipping_points.get(name)


def create_tipping_point(attrs: dict[str, Any]) -> ITipPoint:
return TippingPoint.load_dict(attrs)


def save_tipping_point(tipping_point: ITipPoint) -> None:
Database().tipping_points.save(tipping_point)


def edit_tipping_point(tipping_point: ITipPoint) -> None:
Database().tipping_points.edit(tipping_point)


def delete_tipping_point(name: str) -> None:
Database().tipping_points.delete(name)


Copy link
Author

Choose a reason for hiding this comment

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

agreed with Luuk that for now we run the TP functions directly from the API

def create_tipping_point_scenarios(name: str) -> None:
Database().tipping_points.get(name).create_tp_scenarios()


def run_tipping_point(name: str) -> None:
Database().tipping_points.get(name).run_tp_scenarios()


def check_scenario_has_run(name: str) -> None:
scenario_ran = Database().tipping_points.get(name).check_scenario_has_run()
return scenario_ran
71 changes: 71 additions & 0 deletions flood_adapt/dbs_classes/dbs_tipping_point.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,71 @@
import shutil

from flood_adapt.dbs_classes.dbs_template import DbsTemplate
from flood_adapt.object_model.interface.tipping_points import ITipPoint
from flood_adapt.object_model.tipping_point import TippingPoint


Copy link
Author

Choose a reason for hiding this comment

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

I based on the other codes and have not added other functionalities. Maybe we need to add the plots here?

class DbsTippingPoint(DbsTemplate):
_type = "tipping_point"
_folder_name = "tipping_points"
_object_model_class = TippingPoint

def save(self, tipping_point: ITipPoint, overwrite: bool = False):
"""Save a tipping point object in the database.

Parameters
----------
tipping_point : ITipPoint
object of tipping point type
overwrite : bool, optional
whether to overwrite existing tipping point with same name, by default False

Raises
------
ValueError
Raise error if name is already in use. Names of tipping points should be unique.
"""
# Save the tipping point
super().save(tipping_point, overwrite=overwrite)

def delete(self, name: str, toml_only: bool = False):
"""Delete an already existing tipping point in the database.

Parameters
----------
name : str
name of the tipping point
toml_only : bool, optional
whether to only delete the toml file or the entire folder. If the folder is empty after deleting the toml,
it will always be deleted. By default False
"""
# First delete the tipping point
super().delete(name, toml_only=toml_only)

# Delete output if edited
output_path = (
self._database.tipping_points.get_database_path(get_input_path=False) / name
)

if output_path.exists():
shutil.rmtree(output_path)

def edit(self, tipping_point: ITipPoint):
"""Edit an already existing tipping point in the database.

Parameters
----------
tipping_point : ITipPoint
object of tipping point type
"""
# Edit the tipping point
super().edit(tipping_point)

# Delete output if edited
output_path = (
self._database.tipping_points.get_database_path(get_input_path=False)
/ tipping_point.attrs.name
)

if output_path.exists():
shutil.rmtree(output_path)
8 changes: 8 additions & 0 deletions flood_adapt/dbs_controller.py
Original file line number Diff line number Diff line change
Expand Up @@ -23,6 +23,7 @@
from flood_adapt.dbs_classes.dbs_scenario import DbsScenario
from flood_adapt.dbs_classes.dbs_static import DbsStatic
from flood_adapt.dbs_classes.dbs_strategy import DbsStrategy
from flood_adapt.dbs_classes.dbs_tipping_point import DbsTippingPoint
from flood_adapt.integrator.sfincs_adapter import SfincsAdapter
from flood_adapt.log import FloodAdaptLogging
from flood_adapt.object_model.hazard.event.event_factory import EventFactory
Expand Down Expand Up @@ -63,6 +64,7 @@ class Database(IDatabase):
_measures: DbsMeasure
_projections: DbsProjection
_benefits: DbsBenefit
_tipping_points: DbsTippingPoint

def __new__(cls, *args, **kwargs):
if not cls._instance: # Singleton pattern
Expand Down Expand Up @@ -133,6 +135,7 @@ def __init__(
self._measures = DbsMeasure(self)
self._projections = DbsProjection(self)
self._benefits = DbsBenefit(self)
self._tipping_points = DbsTippingPoint(self)

# Delete any unfinished/crashed scenario output
self.cleanup()
Expand Down Expand Up @@ -195,6 +198,10 @@ def projections(self) -> DbsProjection:
def benefits(self) -> DbsBenefit:
return self._benefits

@property
def tipping_points(self) -> DbsTippingPoint:
return self._tipping_points

def interp_slr(self, slr_scenario: str, year: float) -> float:
r"""Interpolate SLR value and reference it to the SLR reference year from the site toml.

Expand Down Expand Up @@ -750,6 +757,7 @@ def update(self) -> None:
self.strategies = self._strategies.list_objects()
self.scenarios = self._scenarios.list_objects()
self.benefits = self._benefits.list_objects()
self.tipping_points = self._tipping_points.list_objects()

def get_outputs(self) -> dict[str, Any]:
"""Return a dictionary with info on the outputs that currently exist in the database.
Expand Down
2 changes: 1 addition & 1 deletion flood_adapt/object_model/benefit.py
Original file line number Diff line number Diff line change
Expand Up @@ -606,4 +606,4 @@ def save(self, filepath: Union[str, os.PathLike]):
path for saving the toml file
"""
with open(filepath, "wb") as f:
tomli_w.dump(self.attrs.dict(exclude_none=True), f)
tomli_w.model_dump(self.attrs, f, exclude_none=True)
102 changes: 102 additions & 0 deletions flood_adapt/object_model/interface/tipping_points.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,102 @@
import os
from abc import ABC, abstractmethod
from enum import Enum
from typing import Any, Optional, Union

import pandas as pd
from pydantic import BaseModel


class TippingPointMetrics(str, Enum):
"""class describing the accepted input for the variable metric_type in TippingPoint."""

# based on what I have found in floodadapt - but can be changed
FloodedAll = "FloodedAll"
FloodedLowVulnerability = "FloodedLowVulnerability"
FloodedHighVulnerability = "FloodedHighVulnerability"
TotalDamageEvent = "TotalDamageEvent"
TotalResDamageEvent = "TotalResDamageEvent"
ResidentialMinorCount = "ResidentialMinorCount"
ResidentialMajorCount = "ResidentialMajorCount"
ResidentialDestroyedCount = "ResidentialDestroyedCount"
CommercialCount = "CommercialCount"
CommercialMinorCount = "CommercialMinorCount"
CommercialMajorCount = "CommercialMajorCount"
CommercialDestroyedCount = "CommercialDestroyedCount"
HealthCount = "HealthCount"
HealthMinorCount = "HealthMinorCount"
HealthMajorCount = "HealthMajorCount"
HealthDestroyedCount = "HealthDestroyedCount"
SchoolsCount = "SchoolsCount"
SchoolsMinorCount = "SchoolsMinorCount"
SchoolsMajorCount = "SchoolsMajorCount"
SchoolsDestroyedCount = "SchoolsDestroyedCount"
EmergencyCount = "EmergencyCount"
EmergencyMinorCount = "EmergencyMinorCount"
EmergencyMajorCount = "EmergencyMajorCount"
EmergencyDestroyedCount = "EmergencyDestroyedCount"
DisplacedLowVulnerability = "DisplacedLowVulnerability"
DisplacedHighVulnerability = "DisplacedHighVulnerability"
SlightlyFloodedRoads = "SlightlyFloodedRoads"
MinorFloodedRoads = "MinorFloodedRoads"
MajorFloodedRoads = "MajorFloodedRoads"
FullyFloodedRoads = "FullyFloodedRoads"


class TippingPointStatus(str, Enum):
"""class describing the accepted input for the variable metric_type in TippingPoint."""

reached = "reached"
not_reached = "not_reached"
completed = "completed"


class TippingPointOperator(str, Enum):
"""class describing the accepted input for the variable operator in TippingPoint."""

greater = "greater"
less = "less"


class TippingPointModel(BaseModel):
"""BaseModel describing the expected variables and data types of a Tipping Point analysis object."""

name: str
description: Optional[str] = ""
strategy: str
event_set: str
projection: str
sealevelrise: list[float] # could be a numpy array too
tipping_point_metric: list[tuple[TippingPointMetrics, float, TippingPointOperator]]
status: TippingPointStatus = TippingPointStatus.not_reached
scenarios: list[str] = []


class ITipPoint(ABC):
attrs: TippingPointModel
database_input_path: Union[str, os.PathLike]
results_path: Union[str, os.PathLike]
scenarios: pd.DataFrame
has_run: bool = False

@staticmethod
@abstractmethod
def load_file(filepath: Union[str, os.PathLike]):
"""Get Tipping Point attributes from toml file."""
...

@staticmethod # copping from benefits.py
@abstractmethod
def load_dict(data: dict[str, Any]):
"""Get Tipping Point attributes from an object, e.g. when initialized from GUI."""
...

@abstractmethod
def save(self, filepath: Union[str, os.PathLike]):
"""Save Tipping Point attributes to a toml file."""
...

@abstractmethod
def check_scenarios_exist(self) -> pd.DataFrame:
"""Check which scenarios are needed for this tipping point calculation and if they have already been created."""
...
2 changes: 1 addition & 1 deletion flood_adapt/object_model/scenario.py
Original file line number Diff line number Diff line change
Expand Up @@ -62,7 +62,7 @@ def load_dict(data: dict[str, Any], database_input_path: os.PathLike):
def save(self, filepath: Union[str, os.PathLike]):
"""Save Scenario to a toml file."""
with open(filepath, "wb") as f:
tomli_w.dump(self.attrs.dict(exclude_none=True), f)
tomli_w.dump(self.attrs.model_dump(exclude_none=True), f)

def run(self):
"""Run direct impact models for the scenario."""
Expand Down
Loading
Loading