Skip to content

Commit

Permalink
#3530 add classes to handle termination
Browse files Browse the repository at this point in the history
  • Loading branch information
valentinsulzer committed Dec 6, 2023
1 parent 6851496 commit ba23d41
Show file tree
Hide file tree
Showing 7 changed files with 94 additions and 58 deletions.
2 changes: 2 additions & 0 deletions pybamm/__init__.py
Original file line number Diff line number Diff line change
Expand Up @@ -54,6 +54,7 @@
from .logger import logger, set_logging_level, get_new_logger
from .settings import settings
from .citations import Citations, citations, print_citations

#
# Classes for the Expression Tree
#
Expand Down Expand Up @@ -222,6 +223,7 @@
#
from .experiment.experiment import Experiment
from . import experiment
from .experiment import step


#
Expand Down
2 changes: 1 addition & 1 deletion pybamm/experiment/experiment.py
Original file line number Diff line number Diff line change
Expand Up @@ -3,7 +3,7 @@
#

import pybamm
from pybamm.step._steps_util import (
from .step._steps_util import (
_convert_time_to_seconds,
_convert_temperature_to_kelvin,
)
Expand Down
Original file line number Diff line number Diff line change
@@ -1,2 +1,3 @@
from .steps import *
from .steps import _Step
from .step_termination import *
Original file line number Diff line number Diff line change
Expand Up @@ -4,6 +4,7 @@
import pybamm
import numpy as np
from datetime import datetime
from .step_termination import read_termination

_examples = """
Expand Down Expand Up @@ -136,8 +137,10 @@ def __init__(
termination = [termination]
self.termination = []
for term in termination:
typ, value = _convert_electric(term)
self.termination.append({"type": typ, "value": value})
if isinstance(term, str):
term = _convert_electric(term)
term = read_termination(term)
self.termination.append(term)

self.temperature = _convert_temperature_to_kelvin(temperature)

Expand Down Expand Up @@ -193,10 +196,7 @@ def to_dict(self):
}

def __eq__(self, other):
return (
isinstance(other, _Step)
and self.hash_args == other.hash_args
)
return isinstance(other, _Step) and self.hash_args == other.hash_args

def __hash__(self):
return hash(self.basic_repr())
Expand Down
78 changes: 78 additions & 0 deletions pybamm/experiment/step/step_termination.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,78 @@
import pybamm
import numpy as np


class BaseTermination:
def __init__(self, value):
self.value = value

def get_event(self, variables, step_value):
raise NotImplementedError


class CrateTermination(BaseTermination):
def get_event(self, variables, step_value):
event = pybamm.Event(
"C-rate cut-off [A] [experiment]",
abs(variables["C-rate"]) - self.value,
)
return event


class CurrentTermination(BaseTermination):
def get_event(self, variables, step_value):
event = pybamm.Event(
"Current cut-off [A] [experiment]",
abs(variables["Current [A]"]) - self.value,
)
return event


class VoltageTermination(BaseTermination):
def get_event(self, variables, step_value):
# The voltage event should be positive at the start of charge/
# discharge. We use the sign of the current or power input to
# figure out whether the voltage event is greater than the starting
# voltage (charge) or less (discharge) and set the sign of the
# event accordingly
if isinstance(step_value, pybamm.Symbol):
inpt = {"start time": 0}
init_curr = step_value.evaluate(t=0, inputs=inpt).flatten()[0]
else:
init_curr = step_value
sign = np.sign(init_curr)
if sign > 0:
name = "Discharge"
else:
name = "Charge"
if sign != 0:
# Event should be positive at initial conditions for both
# charge and discharge
event = pybamm.Event(
f"{name} voltage cut-off [V] [experiment]",
sign * (variables["Battery voltage [V]"] - self.value),
)
return event


class CustomTermination(BaseTermination):
def __init__(self, name, event_function):
self.name = name
self.event_function = event_function

def get_event(self, variables, step_value):
return pybamm.Event(self.name, self.event_function(variables))


def read_termination(termination):
if isinstance(termination, tuple):
typ, value = termination
else:
return termination

termination_class = {
"current": CurrentTermination,
"voltage": VoltageTermination,
"C-rate": CrateTermination,
}[typ]
return termination_class(value)
File renamed without changes.
57 changes: 6 additions & 51 deletions pybamm/simulation.py
Original file line number Diff line number Diff line change
Expand Up @@ -173,16 +173,6 @@ def set_up_and_parameterise_experiment(self):
op_conds.type = "current"
op_conds.value = op_conds.value * capacity

# Update terminations
termination = op_conds.termination
for term in termination:
term_type = term["type"]
if term_type == "C-rate":
# Change type to current
term["type"] = "current"
# Scale C-rate with capacity to obtain current
term["value"] = term["value"] * capacity

# Add time to the experiment times
dt = op_conds.duration
if dt is None:
Expand Down Expand Up @@ -275,46 +265,9 @@ def set_up_and_parameterise_model_for_experiment(self):

def update_new_model_events(self, new_model, op):
for term in op.termination:
if term["type"] == "current":
new_model.events.append(
pybamm.Event(
"Current cut-off [A] [experiment]",
abs(new_model.variables["Current [A]"]) - term["value"],
)
)

# add voltage events to the model
if term["type"] == "voltage":
# The voltage event should be positive at the start of charge/
# discharge. We use the sign of the current or power input to
# figure out whether the voltage event is greater than the starting
# voltage (charge) or less (discharge) and set the sign of the
# event accordingly
if isinstance(op.value, pybamm.Interpolant) or isinstance(
op.value, pybamm.Multiplication
):
inpt = {"start time": 0}
init_curr = op.value.evaluate(t=0, inputs=inpt).flatten()[0]
sign = np.sign(init_curr)
else:
sign = np.sign(op.value)
if sign > 0:
name = "Discharge"
else:
name = "Charge"
if sign != 0:
# Event should be positive at initial conditions for both
# charge and discharge
new_model.events.append(
pybamm.Event(
f"{name} voltage cut-off [V] [experiment]",
sign
* (
new_model.variables["Battery voltage [V]"]
- term["value"]
),
)
)
event = term.get_event(new_model.variables, op.value)
if event is not None:
new_model.events.append(event)

# Keep the min and max voltages as safeguards but add some tolerances
# so that they are not triggered before the voltage limits in the
Expand Down Expand Up @@ -777,7 +730,9 @@ def solve(
# Hacky patch to allow correct processing of end_time and next_starting time
# For efficiency purposes, op_conds treats identical steps as the same object
# regardless of the initial time. Should be refactored as part of #3176
op_conds_unproc = self.experiment.operating_conditions_steps_unprocessed[idx]
op_conds_unproc = (
self.experiment.operating_conditions_steps_unprocessed[idx]
)

start_time = current_solution.t[-1]

Expand Down

0 comments on commit ba23d41

Please sign in to comment.