Skip to content

Commit

Permalink
Add functions of time in experiment step (#4222)
Browse files Browse the repository at this point in the history
* time varying experiment steps

* Update CHANGELOG.md

* fix code coverage

* rearrange function order

* remove `is_drive_cycle` attribute

* break out `record_tags` method

---------

Co-authored-by: Eric G. Kratz <[email protected]>
  • Loading branch information
MarcBerliner and kratman authored Jun 28, 2024
1 parent 91b1d50 commit b50dfa5
Show file tree
Hide file tree
Showing 4 changed files with 143 additions and 53 deletions.
1 change: 1 addition & 0 deletions CHANGELOG.md
Original file line number Diff line number Diff line change
Expand Up @@ -2,6 +2,7 @@

## Features

- Added functionality to pass in arbitrary functions of time as the argument for a (`pybamm.step`). ([#4222](https://github.com/pybamm-team/PyBaMM/pull/4222))
- Added new parameters `"f{pref]Initial inner SEI on cracks thickness [m]"` and `"f{pref]Initial outer SEI on cracks thickness [m]"`, instead of hardcoding these to `L_inner_0 / 10000` and `L_outer_0 / 10000`. ([#4168](https://github.com/pybamm-team/PyBaMM/pull/4168))
- Added `pybamm.DataLoader` class to fetch data files from [pybamm-data](https://github.com/pybamm-team/pybamm-data/releases/tag/v1.0.0) and store it under local cache. ([#4098](https://github.com/pybamm-team/PyBaMM/pull/4098))
- Added `time` as an option for `Experiment.termination`. Now allows solving up to a user-specified time while also allowing different cycles and steps in an experiment to be handled normally. ([#4073](https://github.com/pybamm-team/PyBaMM/pull/4073))
Expand Down
122 changes: 95 additions & 27 deletions pybamm/experiment/step/base_step.py
Original file line number Diff line number Diff line change
Expand Up @@ -37,7 +37,7 @@ class BaseStep:
----------
value : float
The value of the step, corresponding to the type of step. Can be a number, a
2-tuple (for cccv_ode), or a 2-column array (for drive cycles)
2-tuple (for cccv_ode), a 2-column array (for drive cycles), or a 1-argument function of t
duration : float, optional
The duration of the step in seconds.
termination : str or list, optional
Expand Down Expand Up @@ -72,8 +72,9 @@ def __init__(
direction=None,
):
# Check if drive cycle
self.is_drive_cycle = isinstance(value, np.ndarray)
if self.is_drive_cycle:
is_drive_cycle = isinstance(value, np.ndarray)
is_python_function = callable(value)
if is_drive_cycle:
if value.ndim != 2 or value.shape[1] != 2:
raise ValueError(
"Drive cycle must be a 2-column array with time in the first column"
Expand All @@ -83,36 +84,30 @@ def __init__(
t = value[:, 0]
if t[0] != 0:
raise ValueError("Drive cycle must start at t=0")
elif is_python_function:
t0 = 0
# Check if the function is only a function of t
try:
value_t0 = value(t0)
except TypeError:
raise TypeError(
"Input function must have only 1 positional argument for time"
) from None

# Check if the value at t0 is feasible
if not (np.isfinite(value_t0) and np.isscalar(value_t0)):
raise ValueError(
f"Input function must return a real number output at t = {t0}"
)

# Set duration
if duration is None:
duration = self.default_duration(value)
self.duration = _convert_time_to_seconds(duration)

# Record all the args for repr and hash
self.repr_args = f"{value}, duration={duration}"
self.hash_args = f"{value}"
if termination:
self.repr_args += f", termination={termination}"
self.hash_args += f", termination={termination}"
if period:
self.repr_args += f", period={period}"
if temperature:
self.repr_args += f", temperature={temperature}"
self.hash_args += f", temperature={temperature}"
if tags:
self.repr_args += f", tags={tags}"
if start_time:
self.repr_args += f", start_time={start_time}"
if description:
self.repr_args += f", description={description}"
if direction:
self.repr_args += f", direction={direction}"
self.hash_args += f", direction={direction}"

# If drive cycle, repeat the drive cycle until the end of the experiment,
# and create an interpolant
if self.is_drive_cycle:
if is_drive_cycle:
t_max = self.duration
if t_max > value[-1, 0]:
# duration longer than drive cycle values so loop
Expand All @@ -135,10 +130,33 @@ def __init__(
name="Drive Cycle",
)
self.period = np.diff(t).min()
elif is_python_function:
t = pybamm.t - pybamm.InputParameter("start time")
self.value = value(t)
self.period = _convert_time_to_seconds(period)
else:
self.value = value
self.period = _convert_time_to_seconds(period)

if (
hasattr(self, "calculate_charge_or_discharge")
and self.calculate_charge_or_discharge
):
direction = self.value_based_charge_or_discharge()
self.direction = direction

self.repr_args, self.hash_args = self.record_tags(
value,
duration,
termination,
period,
temperature,
tags,
start_time,
description,
direction,
)

self.description = description

if termination is None:
Expand Down Expand Up @@ -167,8 +185,6 @@ def __init__(
self.next_start_time = None
self.end_time = None

self.direction = direction

def copy(self):
"""
Return a copy of the step.
Expand Down Expand Up @@ -277,6 +293,58 @@ def update_model_events(self, new_model):
event.name, event.expression + 1, event.event_type
)

def value_based_charge_or_discharge(self):
"""
Determine whether the step is a charge or discharge step based on the value of the
step
"""
if isinstance(self.value, pybamm.Symbol):
inpt = {"start time": 0}
init_curr = self.value.evaluate(t=0, inputs=inpt).flatten()[0]
else:
init_curr = self.value
sign = np.sign(init_curr)
if sign == 0:
return "Rest"
elif sign > 0:
return "Discharge"
else:
return "Charge"

def record_tags(
self,
value,
duration,
termination,
period,
temperature,
tags,
start_time,
description,
direction,
):
"""Record all the args for repr and hash"""
repr_args = f"{value}, duration={duration}"
hash_args = f"{value}"
if termination:
repr_args += f", termination={termination}"
hash_args += f", termination={termination}"
if period:
repr_args += f", period={period}"
if temperature:
repr_args += f", temperature={temperature}"
hash_args += f", temperature={temperature}"
if tags:
repr_args += f", tags={tags}"
if start_time:
repr_args += f", start_time={start_time}"
if description:
repr_args += f", description={description}"
if direction:
repr_args += f", direction={direction}"
hash_args += f", direction={direction}"
return repr_args, hash_args


class BaseStepExplicit(BaseStep):
def __init__(self, *args, **kwargs):
Expand Down
30 changes: 4 additions & 26 deletions pybamm/experiment/step/steps.py
Original file line number Diff line number Diff line change
@@ -1,4 +1,3 @@
import numpy as np
import pybamm
from .base_step import (
BaseStepExplicit,
Expand Down Expand Up @@ -130,7 +129,7 @@ class Current(BaseStepExplicit):
"""

def __init__(self, value, **kwargs):
kwargs["direction"] = value_based_charge_or_discharge(value)
self.calculate_charge_or_discharge = True
super().__init__(value, **kwargs)

def current_value(self, variables):
Expand All @@ -151,7 +150,7 @@ class CRate(BaseStepExplicit):
"""

def __init__(self, value, **kwargs):
kwargs["direction"] = value_based_charge_or_discharge(value)
self.calculate_charge_or_discharge = True
super().__init__(value, **kwargs)

def current_value(self, variables):
Expand Down Expand Up @@ -201,7 +200,7 @@ class Power(BaseStepImplicit):
"""

def __init__(self, value, **kwargs):
kwargs["direction"] = value_based_charge_or_discharge(value)
self.calculate_charge_or_discharge = True
super().__init__(value, **kwargs)

def get_parameter_values(self, variables):
Expand Down Expand Up @@ -232,7 +231,7 @@ class Resistance(BaseStepImplicit):
"""

def __init__(self, value, **kwargs):
kwargs["direction"] = value_based_charge_or_discharge(value)
self.calculate_charge_or_discharge = True
super().__init__(value, **kwargs)

def get_parameter_values(self, variables):
Expand Down Expand Up @@ -414,24 +413,3 @@ def copy(self):
return CustomStepImplicit(
self.current_rhs_function, self.control, **self.kwargs
)


def value_based_charge_or_discharge(step_value):
"""
Determine whether the step is a charge or discharge step based on the value of the
step
"""
if isinstance(step_value, np.ndarray):
init_curr = step_value[0, 1]
elif 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:
return "Rest"
elif sign > 0:
return "Discharge"
else:
return "Charge"
43 changes: 43 additions & 0 deletions tests/unit/test_simulation.py
Original file line number Diff line number Diff line change
Expand Up @@ -365,6 +365,49 @@ def test_step_with_inputs(self):
sim.solution.all_inputs[1]["Current function [A]"], 2
)

def test_time_varying_input_function(self):
tf = 20.0

def oscillating(t):
return 3.6 + 0.1 * np.sin(2 * np.pi * t / tf)

model = pybamm.lithium_ion.SPM()

operating_modes = {
"Current [A]": pybamm.step.current,
"C-rate": pybamm.step.c_rate,
"Voltage [V]": pybamm.step.voltage,
"Power [W]": pybamm.step.power,
}
for name in operating_modes:
operating_mode = operating_modes[name]
step = operating_mode(oscillating, duration=tf / 2)
experiment = pybamm.Experiment([step, step], period=f"{tf / 100} seconds")

solver = pybamm.CasadiSolver(rtol=1e-8, atol=1e-8)
sim = pybamm.Simulation(model, experiment=experiment, solver=solver)
sim.solve()
for sol in sim.solution.sub_solutions:
t0 = sol.t[0]
np.testing.assert_array_almost_equal(
sol[name].entries, np.array(oscillating(sol.t - t0))
)

# check improper inputs
for x in (np.nan, np.inf):

def f(t, x=x):
return x + t

with self.assertRaises(ValueError):
operating_mode(f)

def g(t, y):
return t

with self.assertRaises(TypeError):
operating_mode(g)

def test_save_load(self):
with TemporaryDirectory() as dir_name:
test_name = os.path.join(dir_name, "tests.pickle")
Expand Down

0 comments on commit b50dfa5

Please sign in to comment.