PyBatteryID — a shorthand for Python Battery Model IDentification, is an open-source library for data-driven battery model identification in the linear parameter-varying (LPV) framework.
Briefly, the LPV framework can be considered as an upgrade to the linear time-invariant (LTI) framework for systems whose dynamics cannot be considered time-invariant — for instance, batteries exhibit varying dynamics as the battery state-of-charge (SOC), temperature, current magnitude, and current direction vary. Refer to the book Modeling and Identification of Linear Parameter-Varying Systems for a formal introduction to the LPV systems.
Use the package manager pip to install PyBatteryID.
pip install pybatteryid
In the following, an example usage of PyBatteryID has been demonstrated for modelling the battery overpotential using the LPV framework while assuming the battery electromotive force (EMF) to be known a priori via appropriate experiments, such as GITT, or low-current cycling. In effect, the battery voltage output at a given time instant can then be calculated by using the EMF value at that instant and evaluating the overpotential using the identified LPV model.
The very first step is to create an instance of the ModelStructure
class which requires the capacity of the studied battery and the model sampling time.
model_structure = ModelStructure(battery_capacity=3440, sampling_period=1)
The next step is to provide the battery EMF data to the ModelStructure
class using the add_emf_function
method. The expected argument is a dictionary with keys soc_values
and voltage_values
representing a list of SOC values and a list of the corresponding EMF values, respectively.
model_structure.add_emf_function({'soc_values': soc_values,
'voltage_values': voltage_values})
The EMF function can be made temperature-dependent by providing two extra arguments, namely, (1) reference_temperature_value
that refers to the temperature corresponding to the (soc_values
, voltage_values
), and (2) dVdT_values
that describes how the EMF depends on temperature assuming linear temperature dependence.
model_structure.add_emf_function({'soc_values': soc_values,
'voltage_values': voltage_values,
'reference_temperature_value': reference_temperature_value,
'dVdT_values': dVoltage_dTemperature_values})
PyBatteryID has reserved certain symbols for the scheduling-variable components, namely, s
, |i|
, d
and T
for the SOC, current magnitude, current direction and temperature, respectively. It allows the user to conveniently describe certain functional forms of the aforementioned variables by detecting them automatically. In the following, these functional forms are described using the symbol
-
$\square$ — No functional transformation. -
1/
$\square$ — Inverse of the variable. -
log[
$\square$ ]
— Logarithmic transformation of the variable. -
exp[
$\diamond$ *sqrt[
$\square$ ]]
— Square-root of the variable followed by exponential transformation, where the hyperparameter can act as a normalization factor. See [1]. -
exp[
$\diamond_0$ *[
$\square$ ]^
$\diamond_1$ ]
— Raising the variable to an arbitrary power followed by exponential transformation. Note that the variable can be multiplied by a number and also added to a number. See below for an example. -
$\square$ [
$\diamond_0$ ,
$\diamond_1$ ]
— Low-pass filtering of the variable using first-order difference equation. The two hyperparameters correspond to$\varepsilon_0$ and$\varepsilon_1$ , where$\varepsilon_0$ gets chosen when the variable is nonzero, and$\varepsilon_1$ otherwise; see [1] for more details. Note that such an operation only makes sense with the current direction.
For instance, one may use the following basis functions,
model_structure.add_basis_functions(['1/s', 'log[s]', 's',
'exp[0.05*sqrt[|i|]]', 'exp[[0.00366*T+1]^-1]', 'd[0.01,0.99]'])
PyBatteryID allows specification of multiple optimization routines in a pipeline, which can then run in a sequential manner. Indeed, the user needs to provide an identification dataset in the form of a dictionary with keys initial_soc
, time_values
, current_values
, voltage_values
, and (optional) temperature_values
. To obtain a battery model, the method identify
can be used which has the following arguments,
-
dataset
— The identification dataset. -
model_order
— Model order$n$ . -
nonlinearity_order
— Nonlinearity order$l$ . -
optimizers
— A list of optimization routines to run in a sequential manner for a regression setting$A\cdot\theta=y$ ; where$A$ is the regression matrix and$y$ the output vector. Currently, the available options arelasso.cvxopt
,lasso.sklearn
andridge.sklearn
.
The output of the method identify
represents an instance of a class Model
in the PyBatteryID package containing all the necessary information regarding the identified battery model.
model = model_structure.identify(dataset,
model_order=1, nonlinearity_order=1,
optimizers=['lasso.cvxopt', 'ridge.sklearn'])
The available optimization routines can be specified using the name of the algorithm suffixed by the name of the package providing the algorithm, for instance, lasso.cvxopt
and lasso.sklearn
. Note that the optimizer lasso.cvxopt
uses fixed regularization parameter as lasso.sklearn
performs cross-validated LASSO regression. Among the two options, lasso.sklearn
is recommended since it has been configured to take significantly less time than the optimizer lasso.cvxopt
at the cost of slight accuracy loss. Furthermore, lasso.sklearn
may result in more stable models since the cross-validation can lead to adequate regularization in the case of less informative dataset and/or less representative model structure (i.e., inappropriate candidate model terms).
After the identification of a battery model, we can simulate the voltage output for a certain current profile. In this regard, the method simulate
can be used which requires the following arguments,
model
— The identified model.dataset
— A dictionary with keysinitial_soc
,time_values
,current_values
,voltage_values
and (optional)temperature_values
. Thevoltage_values
key corresponds to a list of initial voltage values. Note that the number of initial values should be at least equal to the model order. The initial SOC value corresponds to the SOC value at the first time instanttime_values[0]
.
An example usage of the method simulate
can be given as follows,
voltage_simulated = model_structure.simulate(model, dataset)
[1] A.M.A. Sheikh, M.C.F. Donkers, and H.J. Bergveld, “A comprehensive approach to sparse identification of linear parameter-varying models for lithium-ion batteries using improved experimental design,” Journal of Energy Storage, 2024.
[2] A.M.A. Sheikh, M.C.F. Donkers, and H.J. Bergveld, “Investigating Identification Input Designs for Modelling Lithium-ion Batteries with Hysteresis using LPV Framework,” 2024 American Control Conference (ACC).
[3] A.M.A. Sheikh, M.C.F. Donkers, and H.J. Bergveld, “Towards temperature-dependent linear parameter-varying models for lithium-ion batteries using novel experimental design,” Submitted to Journal.
PyBatteryID is an open-source library licensed under the BSD-3-Clause license. For more information, see LICENSE.