Source code for pybamm.simulation

#
# Simulation class
#
from __future__ import annotations

import pickle
import pybamm
import numpy as np
import hashlib
import warnings
import sys
from functools import lru_cache
from datetime import timedelta
from pybamm.util import import_optional_dependency

from pybamm.expression_tree.operations.serialise import Serialise


def is_notebook():
    try:
        shell = get_ipython().__class__.__name__
        if shell == "ZMQInteractiveShell":  # pragma: no cover
            # Jupyter notebook or qtconsole
            cfg = get_ipython().config
            nb = len(cfg["InteractiveShell"].keys()) == 0
            return nb
        elif shell == "TerminalInteractiveShell":  # pragma: no cover
            return False  # Terminal running IPython
        elif shell == "Shell":  # pragma: no cover
            return True  # Google Colab notebook
        else:  # pragma: no cover
            return False  # Other type (?)
    except NameError:
        return False  # Probably standard Python interpreter


[docs] class Simulation: """A Simulation class for easy building and running of PyBaMM simulations. Parameters ---------- model : :class:`pybamm.BaseModel` The model to be simulated experiment : :class:`pybamm.Experiment` or string or list (optional) The experimental conditions under which to solve the model. If a string is passed, the experiment is constructed as `pybamm.Experiment([experiment])`. If a list is passed, the experiment is constructed as `pybamm.Experiment(experiment)`. geometry: :class:`pybamm.Geometry` (optional) The geometry upon which to solve the model parameter_values: :class:`pybamm.ParameterValues` (optional) Parameters and their corresponding numerical values. submesh_types: dict (optional) A dictionary of the types of submesh to use on each subdomain var_pts: dict (optional) A dictionary of the number of points used by each spatial variable spatial_methods: dict (optional) A dictionary of the types of spatial method to use on each domain (e.g. pybamm.FiniteVolume) solver: :class:`pybamm.BaseSolver` (optional) The solver to use to solve the model. output_variables: list (optional) A list of variables to plot automatically C_rate: float (optional) The C-rate at which you would like to run a constant current (dis)charge. """ def __init__( self, model, experiment=None, geometry=None, parameter_values=None, submesh_types=None, var_pts=None, spatial_methods=None, solver=None, output_variables=None, C_rate=None, ): self._parameter_values = parameter_values or model.default_parameter_values self._unprocessed_parameter_values = self._parameter_values if isinstance(model, pybamm.lithium_ion.BasicDFNHalfCell): if experiment is not None: raise NotImplementedError( "BasicDFNHalfCell is not compatible with experiment simulations." ) if experiment is None: # Check to see if the current is provided as data (i.e. drive cycle) current = self._parameter_values.get("Current function [A]") if isinstance(current, pybamm.Interpolant): self.operating_mode = "drive cycle" else: self.operating_mode = "without experiment" if C_rate: self.C_rate = C_rate self._parameter_values.update( { "Current function [A]": self.C_rate * self._parameter_values["Nominal cell capacity [A.h]"] } ) else: if isinstance(experiment, (str, pybamm.step.BaseStep)): experiment = pybamm.Experiment([experiment]) elif isinstance(experiment, list): experiment = pybamm.Experiment(experiment) elif not isinstance(experiment, pybamm.Experiment): raise TypeError( "experiment must be a pybamm `Experiment` instance, a single " "experiment step, or a list of experiment steps" ) self.operating_mode = "with experiment" # Save the experiment self.experiment = experiment.copy() self._unprocessed_model = model self._model = model self._geometry = geometry or self._model.default_geometry self._submesh_types = submesh_types or self._model.default_submesh_types self._var_pts = var_pts or self._model.default_var_pts self._spatial_methods = spatial_methods or self._model.default_spatial_methods self._solver = solver or self._model.default_solver self._output_variables = output_variables # Initialize empty built states self._model_with_set_params = None self._built_model = None self._built_initial_soc = None self.steps_to_built_models = None self.steps_to_built_solvers = None self._mesh = None self._disc = None self._solution = None self.quick_plot = None # Initialise instances of Simulation class with the same random seed self._set_random_seed() # ignore runtime warnings in notebooks if is_notebook(): # pragma: no cover import warnings warnings.filterwarnings("ignore") self.get_esoh_solver = lru_cache()(self._get_esoh_solver) def __getstate__(self): """ Return dictionary of picklable items """ result = self.__dict__.copy() result["get_esoh_solver"] = None # Exclude LRU cache return result def __setstate__(self, state): """ Unpickle, restoring unpicklable relationships """ self.__dict__ = state self.get_esoh_solver = lru_cache()(self._get_esoh_solver) # If the solver being used is CasadiSolver or its variants, set a fixed # random seed during class initialization to the SHA-256 hash of the class # name for purposes of reproducibility. def _set_random_seed(self): if isinstance(self._solver, pybamm.CasadiSolver) or isinstance( self._solver, pybamm.CasadiAlgebraicSolver ): np.random.seed( int(hashlib.sha256(self.__class__.__name__.encode()).hexdigest(), 16) % (2**32) )
[docs] def set_up_and_parameterise_experiment(self): """ Create and parameterise the models for each step in the experiment. This increases set-up time since several models to be processed, but reduces simulation time since the model formulation is efficient. """ parameter_values = self._parameter_values.copy() # Set the initial temperature to be the temperature of the first step # We can set this globally for all steps since any subsequent steps will either # start at the temperature at the end of the previous step (if non-isothermal # model), or will use the "Ambient temperature" input (if isothermal model). # In either case, the initial temperature is not used for any steps except # the first. init_temp = self.experiment.steps[0].temperature if init_temp is not None: parameter_values["Initial temperature [K]"] = init_temp # Process each step self.experiment_unique_steps_to_model = {} for step in self.experiment.unique_steps: parameterised_model = step.process_model(self._model, parameter_values) self.experiment_unique_steps_to_model[step.basic_repr()] = ( parameterised_model ) # Set up rest model if experiment has start times if self.experiment.initial_start_time: # duration doesn't matter, we just need the model rest_step = pybamm.step.rest(duration=1) # Change ambient temperature to be an input, which will be changed at # solve time parameter_values["Ambient temperature [K]"] = "[input]" parameterised_model = rest_step.process_model(self._model, parameter_values) self.experiment_unique_steps_to_model["Rest for padding"] = ( parameterised_model )
[docs] def set_parameters(self): """ A method to set the parameters in the model and the associated geometry. """ if self.model_with_set_params: return self._model_with_set_params = self._parameter_values.process_model( self._unprocessed_model, inplace=False ) self._parameter_values.process_geometry(self._geometry) self._model = self._model_with_set_params
def set_initial_soc(self, initial_soc, inputs=None): if self._built_initial_soc != initial_soc: # reset self._model_with_set_params = None self._built_model = None self.steps_to_built_models = None self.steps_to_built_solvers = None options = self.model.options param = self._model.param if options["open-circuit potential"] == "MSMR": self._parameter_values = ( self._unprocessed_parameter_values.set_initial_ocps( initial_soc, param=param, inplace=False, options=options ) ) elif options["working electrode"] == "positive": self._parameter_values = ( self._unprocessed_parameter_values.set_initial_stoichiometry_half_cell( initial_soc, param=param, inplace=False, options=options, inputs=inputs, ) ) else: self._parameter_values = ( self._unprocessed_parameter_values.set_initial_stoichiometries( initial_soc, param=param, inplace=False, options=options, inputs=inputs, ) ) # Save solved initial SOC in case we need to re-build the model self._built_initial_soc = initial_soc
[docs] def build(self, check_model=True, initial_soc=None, inputs=None): """ A method to build the model into a system of matrices and vectors suitable for performing numerical computations. If the model has already been built or solved then this function will have no effect. This method will automatically set the parameters if they have not already been set. Parameters ---------- check_model : bool, optional If True, model checks are performed after discretisation (see :meth:`pybamm.Discretisation.process_model`). Default is True. initial_soc : float, optional Initial State of Charge (SOC) for the simulation. Must be between 0 and 1. If given, overwrites the initial concentrations provided in the parameter set. """ if initial_soc is not None: self.set_initial_soc(initial_soc, inputs=inputs) if self.built_model: return elif self._model.is_discretised: self._model_with_set_params = self._model self._built_model = self._model else: self.set_parameters() self._mesh = pybamm.Mesh(self._geometry, self._submesh_types, self._var_pts) self._disc = pybamm.Discretisation(self._mesh, self._spatial_methods) self._built_model = self._disc.process_model( self._model_with_set_params, inplace=False, check_model=check_model ) # rebuilt model so clear solver setup self._solver._model_set_up = {}
[docs] def build_for_experiment(self, check_model=True, initial_soc=None, inputs=None): """ Similar to :meth:`Simulation.build`, but for the case of simulating an experiment, where there may be several models and solvers to build. """ if initial_soc is not None: self.set_initial_soc(initial_soc, inputs) if self.steps_to_built_models: return else: self.set_up_and_parameterise_experiment() # Can process geometry with default parameter values (only electrical # parameters change between parameter values) self._parameter_values.process_geometry(self._geometry) # Only needs to set up mesh and discretisation once self._mesh = pybamm.Mesh(self._geometry, self._submesh_types, self._var_pts) self._disc = pybamm.Discretisation(self._mesh, self._spatial_methods) # Process all the different models self.steps_to_built_models = {} self.steps_to_built_solvers = {} for ( step, model_with_set_params, ) in self.experiment_unique_steps_to_model.items(): # It's ok to modify the model with set parameters in place as it's # not returned anywhere built_model = self._disc.process_model( model_with_set_params, inplace=True, check_model=check_model ) solver = self._solver.copy() self.steps_to_built_solvers[step] = solver self.steps_to_built_models[step] = built_model
[docs] def solve( self, t_eval=None, solver=None, check_model=True, save_at_cycles=None, calc_esoh=True, starting_solution=None, initial_soc=None, callbacks=None, showprogress=False, inputs=None, **kwargs, ): """ A method to solve the model. This method will automatically build and set the model parameters if not already done so. Parameters ---------- t_eval : numeric type, optional The times (in seconds) at which to compute the solution. Can be provided as an array of times at which to return the solution, or as a list `[t0, tf]` where `t0` is the initial time and `tf` is the final time. If provided as a list the solution is returned at 100 points within the interval `[t0, tf]`. If not using an experiment or running a drive cycle simulation (current provided as data) `t_eval` *must* be provided. If running an experiment the values in `t_eval` are ignored, and the solution times are specified by the experiment. If None and the parameter "Current function [A]" is read from data (i.e. drive cycle simulation) the model will be solved at the times provided in the data. solver : :class:`pybamm.BaseSolver`, optional The solver to use to solve the model. If None, Simulation.solver is used check_model : bool, optional If True, model checks are performed after discretisation (see :meth:`pybamm.Discretisation.process_model`). Default is True. save_at_cycles : int or list of ints, optional Which cycles to save the full sub-solutions for. If None, all cycles are saved. If int, every multiple of save_at_cycles is saved. If list, every cycle in the list is saved. The first cycle (cycle 1) is always saved. calc_esoh : bool, optional Whether to include eSOH variables in the summary variables. If `False` then only summary variables that do not require the eSOH calculation are calculated. Default is True. starting_solution : :class:`pybamm.Solution` The solution to start stepping from. If None (default), then self._solution is used. Must be None if not using an experiment. initial_soc : float, optional Initial State of Charge (SOC) for the simulation. Must be between 0 and 1. If given, overwrites the initial concentrations provided in the parameter set. callbacks : list of callbacks, optional A list of callbacks to be called at each time step. Each callback must implement all the methods defined in :class:`pybamm.callbacks.BaseCallback`. showprogress : bool, optional Whether to show a progress bar for cycling. If true, shows a progress bar for cycles. Has no effect when not used with an experiment. Default is False. **kwargs Additional key-word arguments passed to `solver.solve`. See :meth:`pybamm.BaseSolver.solve`. """ # Setup if solver is None: solver = self._solver callbacks = pybamm.callbacks.setup_callbacks(callbacks) logs = {} inputs = inputs or {} if self.operating_mode in ["without experiment", "drive cycle"]: self.build(check_model=check_model, initial_soc=initial_soc, inputs=inputs) if save_at_cycles is not None: raise ValueError( "'save_at_cycles' option can only be used if simulating an " "Experiment " ) if starting_solution is not None: raise ValueError( "starting_solution can only be provided if simulating an Experiment" ) if ( self.operating_mode == "without experiment" or "ElectrodeSOH" in self._model.name ): if t_eval is None: raise pybamm.SolverError( "'t_eval' must be provided if not using an experiment or " "simulating a drive cycle. 't_eval' can be provided as an " "array of times at which to return the solution, or as a " "list [t0, tf] where t0 is the initial time and tf is the " "final time. " "For a constant current (dis)charge the suggested 't_eval' " "is [0, 3700/C] where C is the C-rate. " "For example, run\n\n" "\tsim.solve([0, 3700])\n\n" "for a 1C discharge." ) elif self.operating_mode == "drive cycle": # For drive cycles (current provided as data) we perform additional # tests on t_eval (if provided) to ensure the returned solution # captures the input. time_data = self._parameter_values["Current function [A]"].x[0] # If no t_eval is provided, we use the times provided in the data. if t_eval is None: pybamm.logger.info("Setting t_eval as specified by the data") t_eval = time_data # If t_eval is provided we first check if it contains all of the # times in the data to within 10-12. If it doesn't, we then check # that the largest gap in t_eval is smaller than the smallest gap in # the time data (to ensure the resolution of t_eval is fine enough). # We only raise a warning here as users may genuinely only want # the solution returned at some specified points. elif ( set(np.round(time_data, 12)).issubset(set(np.round(t_eval, 12))) ) is False: warnings.warn( """ t_eval does not contain all of the time points in the data set. Note: passing t_eval = None automatically sets t_eval to be the points in the data. """, pybamm.SolverWarning, stacklevel=2, ) dt_data_min = np.min(np.diff(time_data)) dt_eval_max = np.max(np.diff(t_eval)) if dt_eval_max > dt_data_min + sys.float_info.epsilon: warnings.warn( f""" The largest timestep in t_eval ({dt_eval_max}) is larger than the smallest timestep in the data ({dt_data_min}). The returned solution may not have the correct resolution to accurately capture the input. Try refining t_eval. Alternatively, passing t_eval = None automatically sets t_eval to be the points in the data. """, pybamm.SolverWarning, stacklevel=2, ) self._solution = solver.solve( self.built_model, t_eval, inputs=inputs, **kwargs ) elif self.operating_mode == "with experiment": callbacks.on_experiment_start(logs) self.build_for_experiment( check_model=check_model, initial_soc=initial_soc, inputs=inputs ) if t_eval is not None: pybamm.logger.warning( "Ignoring t_eval as solution times are specified by the experiment" ) # Re-initialize solution, e.g. for solving multiple times with different # inputs without having to build the simulation again self._solution = starting_solution # Step through all experimental conditions user_inputs = inputs timer = pybamm.Timer() # Set up eSOH solver (for summary variables) esoh_solver = self.get_esoh_solver(calc_esoh) if starting_solution is None: starting_solution_cycles = [] starting_solution_summary_variables = [] starting_solution_first_states = [] elif not hasattr(starting_solution, "all_summary_variables"): ( cycle_solution, cycle_sum_vars, cycle_first_state, ) = pybamm.make_cycle_solution( [starting_solution], esoh_solver=esoh_solver, save_this_cycle=True, inputs=user_inputs, ) starting_solution_cycles = [cycle_solution] starting_solution_summary_variables = [cycle_sum_vars] starting_solution_first_states = [cycle_first_state] else: starting_solution_cycles = starting_solution.cycles.copy() starting_solution_summary_variables = ( starting_solution.all_summary_variables.copy() ) starting_solution_first_states = ( starting_solution.all_first_states.copy() ) # set simulation initial_start_time if starting_solution is None: initial_start_time = self.experiment.initial_start_time else: initial_start_time = starting_solution.initial_start_time if ( initial_start_time is None and self.experiment.initial_start_time is not None ): raise ValueError( "When using experiments with `start_time`, the starting_solution " "must have a `start_time` too." ) cycle_offset = len(starting_solution_cycles) all_cycle_solutions = starting_solution_cycles all_summary_variables = starting_solution_summary_variables all_first_states = starting_solution_first_states current_solution = starting_solution or pybamm.EmptySolution() voltage_stop = self.experiment.termination.get("voltage") logs["stopping conditions"] = {"voltage": voltage_stop} idx = 0 num_cycles = len(self.experiment.cycle_lengths) feasible = True # simulation will stop if experiment is infeasible # Add initial padding rest if current time is earlier than first start time # This could be the case when using a starting solution if starting_solution is not None: step = self.experiment.steps[0] if step.start_time is not None: rest_time = ( step.start_time - ( initial_start_time + timedelta(seconds=float(current_solution.t[-1])) ) ).total_seconds() if rest_time > pybamm.settings.step_start_offset: # logs["step operating conditions"] = "Initial rest for padding" # callbacks.on_step_start(logs) inputs = { **user_inputs, "Ambient temperature [K]": ( step.temperature or self._parameter_values["Ambient temperature [K]"] ), "start time": current_solution.t[-1], } steps = current_solution.cycles[-1].steps step_solution = current_solution.cycles[-1].steps[-1] step_solution_with_rest = self.run_padding_rest( kwargs, rest_time, step_solution, inputs ) steps[-1] = step_solution + step_solution_with_rest cycle_solution, _, _ = pybamm.make_cycle_solution( steps, esoh_solver=esoh_solver, save_this_cycle=True ) old_cycles = current_solution.cycles.copy() old_cycles[-1] = cycle_solution current_solution += step_solution_with_rest current_solution.cycles = old_cycles # Update _solution self._solution = current_solution # check if a user has tqdm installed if showprogress: tqdm = import_optional_dependency("tqdm") cycle_lengths = tqdm.tqdm( self.experiment.cycle_lengths, desc="Cycling", ) else: cycle_lengths = self.experiment.cycle_lengths for cycle_num, cycle_length in enumerate( cycle_lengths, start=1, ): logs["cycle number"] = ( cycle_num + cycle_offset, num_cycles + cycle_offset, ) logs["elapsed time"] = timer.time() callbacks.on_cycle_start(logs) steps = [] cycle_solution = None # Decide whether we should save this cycle save_this_cycle = ( # always save cycle 1 cycle_num == 1 # always save last cycle or cycle_num == num_cycles # None: save all cycles or save_at_cycles is None # list: save all cycles in the list or ( isinstance(save_at_cycles, list) and cycle_num + cycle_offset in save_at_cycles ) # int: save all multiples or ( isinstance(save_at_cycles, int) and (cycle_num + cycle_offset) % save_at_cycles == 0 ) ) for step_num in range(1, cycle_length + 1): # Use 1-indexing for printing cycle number as it is more # human-intuitive step = self.experiment.steps[idx] start_time = current_solution.t[-1] # If step has an end time, dt must take that into account if step.end_time is not None: dt = min( step.duration, ( step.end_time - ( initial_start_time + timedelta(seconds=float(start_time)) ) ).total_seconds(), ) else: dt = step.duration step_str = str(step) model = self.steps_to_built_models[step.basic_repr()] solver = self.steps_to_built_solvers[step.basic_repr()] logs["step number"] = (step_num, cycle_length) logs["step operating conditions"] = step_str callbacks.on_step_start(logs) inputs = { **user_inputs, "start time": start_time, } # Make sure we take at least 2 timesteps npts = max(int(round(dt / step.period)) + 1, 2) try: step_solution = solver.step( current_solution, model, dt, t_eval=np.linspace(0, dt, npts), save=False, inputs=inputs, **kwargs, ) except pybamm.SolverError as error: if ( "non-positive at initial conditions" in error.message and "[experiment]" in error.message ): step_solution = pybamm.EmptySolution( "Event exceeded in initial conditions", t=start_time ) else: logs["error"] = error callbacks.on_experiment_error(logs) feasible = False # If none of the cycles worked, raise an error if cycle_num == 1 and step_num == 1: raise error # Otherwise, just stop this cycle break step_termination = step_solution.termination # Add a padding rest step if necessary if step.next_start_time is not None: rest_time = ( step.next_start_time - ( initial_start_time + timedelta(seconds=float(step_solution.t[-1])) ) ).total_seconds() if rest_time > pybamm.settings.step_start_offset: logs["step number"] = (step_num, cycle_length) logs["step operating conditions"] = "Rest for padding" callbacks.on_step_start(logs) inputs = { **user_inputs, "Ambient temperature [K]": ( step.temperature or self._parameter_values["Ambient temperature [K]"] ), "start time": step_solution.t[-1], } step_solution_with_rest = self.run_padding_rest( kwargs, rest_time, step_solution, inputs=inputs ) step_solution += step_solution_with_rest steps.append(step_solution) # If there haven't been any successful steps yet in this cycle, then # carry the solution over from the previous cycle (but # `step_solution` should still be an EmptySolution so that in the # list of returned step solutions we can see which steps were # skipped) if ( cycle_solution is None and isinstance(step_solution, pybamm.EmptySolution) and not isinstance(current_solution, pybamm.EmptySolution) ): cycle_solution = current_solution.last_state else: cycle_solution = cycle_solution + step_solution current_solution = cycle_solution callbacks.on_step_end(logs) logs["termination"] = step_solution.termination # Only allow events specified by experiment if not ( isinstance(step_solution, pybamm.EmptySolution) or step_termination == "final time" or "[experiment]" in step_termination ): callbacks.on_experiment_infeasible(logs) feasible = False break # Increment index for next iteration idx += 1 if save_this_cycle or feasible is False: self._solution = self._solution + cycle_solution # At the final step of the inner loop we save the cycle if len(steps) > 0: # Check for EmptySolution if all(isinstance(step, pybamm.EmptySolution) for step in steps): if len(steps) == 1: raise pybamm.SolverError( f"Step '{step_str}' is infeasible " "due to exceeded bounds at initial conditions. " "If this step is part of a longer cycle, " "round brackets should be used to indicate this, " "e.g.:\n pybamm.Experiment([(\n" "\tDischarge at C/5 for 10 hours or until 3.3 V,\n" "\tCharge at 1 A until 4.1 V,\n" "\tHold at 4.1 V until 10 mA\n" "])" ) else: this_cycle = self.experiment.cycles[cycle_num - 1] raise pybamm.SolverError( f"All steps in the cycle {this_cycle} are infeasible " "due to exceeded bounds at initial conditions." ) cycle_sol = pybamm.make_cycle_solution( steps, esoh_solver=esoh_solver, save_this_cycle=save_this_cycle, inputs=user_inputs, ) cycle_solution, cycle_sum_vars, cycle_first_state = cycle_sol all_cycle_solutions.append(cycle_solution) all_summary_variables.append(cycle_sum_vars) all_first_states.append(cycle_first_state) logs["summary variables"] = cycle_sum_vars # Calculate capacity_start using the first cycle if cycle_num == 1: # Note capacity_start could be defined as # self._parameter_values["Nominal cell capacity [A.h]"] instead if "capacity" in self.experiment.termination: capacity_start = all_summary_variables[0]["Capacity [A.h]"] logs["start capacity"] = capacity_start value, typ = self.experiment.termination["capacity"] if typ == "Ah": capacity_stop = value elif typ == "%": capacity_stop = value / 100 * capacity_start else: capacity_stop = None logs["stopping conditions"]["capacity"] = capacity_stop logs["elapsed time"] = timer.time() callbacks.on_cycle_end(logs) # Break if stopping conditions are met # Logging is done in the callbacks if capacity_stop is not None: capacity_now = cycle_sum_vars["Capacity [A.h]"] if not np.isnan(capacity_now) and capacity_now <= capacity_stop: break if voltage_stop is not None: min_voltage = cycle_sum_vars["Minimum voltage [V]"] if min_voltage <= voltage_stop[0]: break # Break if the experiment is infeasible (or errored) if feasible is False: break if self.solution is not None and len(all_cycle_solutions) > 0: self.solution.cycles = all_cycle_solutions self.solution.set_summary_variables(all_summary_variables) self.solution.all_first_states = all_first_states callbacks.on_experiment_end(logs) # record initial_start_time of the solution self.solution.initial_start_time = initial_start_time return self.solution
def run_padding_rest(self, kwargs, rest_time, step_solution, inputs): model = self.steps_to_built_models["Rest for padding"] solver = self.steps_to_built_solvers["Rest for padding"] # Make sure we take at least 2 timesteps. The period is hardcoded to 10 # minutes,the user can always override it by adding a rest step npts = max(int(round(rest_time / 600)) + 1, 2) step_solution_with_rest = solver.step( step_solution, model, rest_time, t_eval=np.linspace(0, rest_time, npts), save=False, inputs=inputs, **kwargs, ) return step_solution_with_rest
[docs] def step( self, dt, solver=None, t_eval=None, save=True, starting_solution=None, inputs=None, **kwargs, ): """ A method to step the model forward one timestep. This method will automatically build and set the model parameters if not already done so. Parameters ---------- dt : numeric type The timestep over which to step the solution solver : :class:`pybamm.BaseSolver` The solver to use to solve the model. t_eval : list or numpy.ndarray, optional An array of times at which to return the solution during the step (Note: t_eval is the time measured from the start of the step, so should start at 0 and end at dt). By default, the solution is returned at t0 and t0 + dt. save : bool Turn on to store the solution of all previous timesteps starting_solution : :class:`pybamm.Solution` The solution to start stepping from. If None (default), then self._solution is used **kwargs Additional key-word arguments passed to `solver.solve`. See :meth:`pybamm.BaseSolver.step`. """ if self.operating_mode in ["without experiment", "drive cycle"]: self.build() if solver is None: solver = self._solver if starting_solution is None: starting_solution = self._solution self._solution = solver.step( starting_solution, self.built_model, dt, t_eval=t_eval, save=save, inputs=inputs, **kwargs, ) return self.solution
def _get_esoh_solver(self, calc_esoh): if ( calc_esoh is False or isinstance(self._model, pybamm.lead_acid.BaseModel) or isinstance(self._model, pybamm.equivalent_circuit.Thevenin) or self._model.options["working electrode"] != "both" ): return None return pybamm.lithium_ion.ElectrodeSOHSolver( self._parameter_values, self._model.param, options=self._model.options )
[docs] def plot(self, output_variables=None, **kwargs): """ A method to quickly plot the outputs of the simulation. Creates a :class:`pybamm.QuickPlot` object (with keyword arguments 'kwargs') and then calls :meth:`pybamm.QuickPlot.dynamic_plot`. Parameters ---------- output_variables: list, optional A list of the variables to plot. **kwargs Additional keyword arguments passed to :meth:`pybamm.QuickPlot.dynamic_plot`. For a list of all possible keyword arguments see :class:`pybamm.QuickPlot`. """ if self._solution is None: raise ValueError( "Model has not been solved, please solve the model before plotting." ) if output_variables is None: output_variables = self._output_variables self.quick_plot = pybamm.dynamic_plot( self._solution, output_variables=output_variables, **kwargs ) return self.quick_plot
[docs] def create_gif(self, number_of_images=80, duration=0.1, output_filename="plot.gif"): """ Generates x plots over a time span of t_eval and compiles them to create a GIF. For more information see :meth:`pybamm.QuickPlot.create_gif` Parameters ---------- number_of_images : int (optional) Number of images/plots to be compiled for a GIF. duration : float (optional) Duration of visibility of a single image/plot in the created GIF. output_filename : str (optional) Name of the generated GIF file. """ if self.solution is None: raise ValueError("The simulation has not been solved yet.") if self.quick_plot is None: self.quick_plot = pybamm.QuickPlot(self._solution) self.quick_plot.create_gif( number_of_images=number_of_images, duration=duration, output_filename=output_filename, )
@property def model(self): return self._model @property def model_with_set_params(self): return self._model_with_set_params @property def built_model(self): return self._built_model @property def geometry(self): return self._geometry @property def parameter_values(self): return self._parameter_values @property def submesh_types(self): return self._submesh_types @property def mesh(self): return self._mesh @property def var_pts(self): return self._var_pts @property def spatial_methods(self): return self._spatial_methods @property def solver(self): return self._solver @property def output_variables(self): return self._output_variables @property def solution(self): return self._solution
[docs] def save(self, filename): """Save simulation using pickle module. Parameters ---------- filename : str The file extension can be arbitrary, but it is common to use ".pkl" or ".pickle" """ if self._model.convert_to_format == "python": # We currently cannot save models in the 'python' format raise NotImplementedError( """ Cannot save simulation if model format is python. Set model.convert_to_format = 'casadi' instead. """ ) # Clear solver problem (not pickle-able, will automatically be recomputed) if ( isinstance(self._solver, pybamm.CasadiSolver) and self._solver.integrator_specs != {} ): self._solver.integrator_specs = {} if self.steps_to_built_solvers is not None: for solver in self.steps_to_built_solvers.values(): if ( isinstance(solver, pybamm.CasadiSolver) and solver.integrator_specs != {} ): solver.integrator_specs = {} with open(filename, "wb") as f: pickle.dump(self, f, pickle.HIGHEST_PROTOCOL)
[docs] def save_model( self, filename: str | None = None, mesh: bool = False, variables: bool = False, ): """ Write out a discretised model to a JSON file Parameters ---------- mesh: bool The mesh used to discretise the model. If false, plotting tools will not be available when the model is read back in and solved. variables: bool The discretised variables. Not required to solve a model, but if false tools will not be availble. Will automatically save meshes as well, required for plotting tools. filename: str, optional The desired name of the JSON file. If no name is provided, one will be created based on the model name, and the current datetime. """ mesh = self.mesh if (mesh or variables) else None variables = self.built_model.variables if variables else None if self.operating_mode == "with experiment": raise NotImplementedError( """ Serialising models coupled to experiments is not yet supported. """ ) if self.built_model: Serialise().save_model( self.built_model, filename=filename, mesh=mesh, variables=variables ) else: raise NotImplementedError( """ PyBaMM can only serialise a discretised model. Ensure the model has been built (e.g. run `build()`) before saving. """ )
[docs] def plot_voltage_components( self, ax=None, show_legend=True, split_by_electrode=False, show_plot=True, **kwargs_fill, ): """ Generate a plot showing the component overpotentials that make up the voltage Parameters ---------- ax : matplotlib Axis, optional The axis on which to put the plot. If None, a new figure and axis is created. show_legend : bool, optional Whether to display the legend. Default is True. split_by_electrode : bool, optional Whether to show the overpotentials for the negative and positive electrodes separately. Default is False. show_plot : bool, optional Whether to show the plots. Default is True. Set to False if you want to only display the plot after plt.show() has been called. kwargs_fill Keyword arguments, passed to ax.fill_between. """ if self.solution is None: raise ValueError("The simulation has not been solved yet.") return pybamm.plot_voltage_components( self.solution, ax=ax, show_legend=show_legend, split_by_electrode=split_by_electrode, show_plot=show_plot, **kwargs_fill, )
def load_sim(filename): """Load a saved simulation""" return pybamm.load(filename)