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

Added Basic Reservoir Model #30

Merged
merged 7 commits into from
Jul 31, 2024
Merged
Show file tree
Hide file tree
Changes from 1 commit
Commits
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
3 changes: 2 additions & 1 deletion pyproject.toml
Original file line number Diff line number Diff line change
Expand Up @@ -65,7 +65,8 @@ Changelog = "https://github.com/pybamm-team/pybamm-cookiecutter/releases"
Chen2020 = "pybamm_cookiecutter.parameters.input.Chen2020:get_parameter_values"

[project.entry-points."models"]
SPM = "pybamm_cookiecutter.models.input.SPM:SPM"
SPM = "pybamm_cookiecutter.models.input.spm:SPM"
Reservoir = "pybamm_cookiecutter.models.input.basic_reservoir:BasicReservoir"
santacodes marked this conversation as resolved.
Show resolved Hide resolved
arjxn-py marked this conversation as resolved.
Show resolved Hide resolved

[tool.hatch]
version.source = "vcs"
Expand Down
200 changes: 200 additions & 0 deletions src/pybamm_cookiecutter/models/input/basic_reservoir.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,200 @@
"""
This code is adopted from the PyBaMM project under the BSD-3-Clause

Copyright (c) 2018-2024, the PyBaMM team.
All rights reserved.

Redistribution and use in source and binary forms, with or without
modification, are permitted provided that the following conditions are met:

* Redistributions of source code must retain the above copyright notice, this
list of conditions and the following disclaimer.

* Redistributions in binary form must reproduce the above copyright notice,
this list of conditions and the following disclaimer in the documentation
and/or other materials provided with the distribution.

* Neither the name of the copyright holder nor the names of its
contributors may be used to endorse or promote products derived from
this software without specific prior written permission.

THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE
FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR
SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER
CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY,
OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
"""


#
# Basic Reservoir Model
#
import pybamm

class BasicReservoir(pybamm.lithium_ion.BaseModel):
"""Reservoir model of a lithium-ion battery, from
:footcite:t:`Marquis2019`.

Parameters
----------
name : str, optional
The name of the model.
"""

def __init__(self, name="Reservoir Model"):
super().__init__({}, name)
pybamm.citations.register("Marquis2019")
# `param` is a class containing all the relevant parameters and functions for
# this model. These are purely symbolic at this stage, and will be set by the
# `ParameterValues` class when the model is processed.
param = self.param

######################
# Variables
######################
# Variables that depend on time only are created without a domain
Q = pybamm.Variable("Discharge capacity [A.h]")
# Variables that vary spatially are created with a domain
sto_n = pybamm.Variable(
"Average negative particle stoichiometry",
domain="current collector",
bounds=(0, 1),
)
sto_p = pybamm.Variable(
"Average positive particle stoichiometry",
domain="current collector",
bounds=(0, 1),
)

# Constant temperature
T = param.T_init

######################
# Other set-up
######################

# Current density
i_cell = param.current_density_with_time
a_n = 3 * param.n.prim.epsilon_s_av / param.n.prim.R_typ
a_p = 3 * param.p.prim.epsilon_s_av / param.p.prim.R_typ
j_n = i_cell / (param.n.L * a_n)
j_p = -i_cell / (param.p.L * a_p)

######################
# State of Charge
######################
I = param.current_with_time
# The `rhs` dictionary contains differential equations, with the key being the
# variable in the d/dt
self.rhs[Q] = I / 3600
# Initial conditions must be provided for the ODEs
self.initial_conditions[Q] = pybamm.Scalar(0)

######################
# Particles
######################

self.rhs[sto_n] = -i_cell / (
param.n.L * param.n.prim.epsilon_s_av * param.n.prim.c_max * param.F
)
self.rhs[sto_p] = i_cell / (
param.p.L * param.p.prim.epsilon_s_av * param.p.prim.c_max * param.F
)
# c_n_init and c_p_init are functions of r and x, but for the reservoir model
# we take the x-averaged and r-averaged value since there are no x-dependence
# nor r-dependencein the particles
self.initial_conditions[sto_n] = (
pybamm.x_average(pybamm.r_average(param.n.prim.c_init)) / param.n.prim.c_max
)
self.initial_conditions[sto_p] = (
pybamm.x_average(pybamm.r_average(param.p.prim.c_init)) / param.p.prim.c_max
)

self.events += [
pybamm.Event(
"Minimum negative particle surface stoichiometry",
sto_n - 0.01,
),
pybamm.Event(
"Maximum negative particle surface stoichiometry",
(1 - 0.01) - sto_n,
),
pybamm.Event(
"Minimum positive particle surface stoichiometry",
sto_p - 0.01,
),
pybamm.Event(
"Maximum positive particle surface stoichiometry",
(1 - 0.01) - sto_p,
),
]

# Note that the reservoir model does not have any algebraic equations, so the
# `algebraic` dictionary remains empty

######################
# (Some) variables
######################
# Interfacial reactions
RT_F = param.R * T / param.F
j0_n = param.n.prim.j0(param.c_e_init_av, sto_n * param.n.prim.c_max, T)
j0_p = param.p.prim.j0(param.c_e_init_av, sto_p * param.p.prim.c_max, T)
eta_n = (2 / param.n.prim.ne) * RT_F * pybamm.arcsinh(j_n / (2 * j0_n))
eta_p = (2 / param.p.prim.ne) * RT_F * pybamm.arcsinh(j_p / (2 * j0_p))
phi_s_n = 0
phi_e = -eta_n - param.n.prim.U(sto_n, T)
phi_s_p = eta_p + phi_e + param.p.prim.U(sto_p, T)
V = phi_s_p
num_cells = pybamm.Parameter(
"Number of cells connected in series to make a battery"
)
c_s_n = sto_n * param.n.prim.c_max
c_s_p = sto_p * param.p.prim.c_max

whole_cell = ["negative electrode", "separator", "positive electrode"]
# The `variables` dictionary contains all variables that might be useful for
# visualising the solution of the model
# Primary broadcasts are used to broadcast scalar quantities across a domain
# into a vector of the right shape, for multiplying with other vectors
self.variables = {
"Time [s]": pybamm.t,
"Discharge capacity [A.h]": Q,
"X-averaged negative particle concentration [mol.m-3]": pybamm.PrimaryBroadcast(
c_s_n, "negative particle"
),
"Negative particle surface "
"concentration [mol.m-3]": pybamm.PrimaryBroadcast(
c_s_n, "negative electrode"
),
"Electrolyte concentration [mol.m-3]": pybamm.PrimaryBroadcast(
param.c_e_init_av, whole_cell
),
"X-averaged positive particle concentration [mol.m-3]": pybamm.PrimaryBroadcast(
c_s_p, "positive particle"
),
"Positive particle surface "
"concentration [mol.m-3]": pybamm.PrimaryBroadcast(
c_s_p, "positive electrode"
),
"Current [A]": I,
"Current variable [A]": I, # for compatibility with pybamm.Experiment
"Negative electrode potential [V]": pybamm.PrimaryBroadcast(
phi_s_n, "negative electrode"
),
"Electrolyte potential [V]": pybamm.PrimaryBroadcast(phi_e, whole_cell),
"Positive electrode potential [V]": pybamm.PrimaryBroadcast(
phi_s_p, "positive electrode"
),
"Voltage [V]": V,
"Battery voltage [V]": V * num_cells,
}
# Events specify points at which a solution should terminate
self.events += [
pybamm.Event("Minimum voltage [V]", V - param.voltage_low_cut),
pybamm.Event("Maximum voltage [V]", param.voltage_high_cut - V),
]
3 changes: 2 additions & 1 deletion {{cookiecutter.project_name}}/pyproject.toml
Original file line number Diff line number Diff line change
Expand Up @@ -81,7 +81,8 @@ Changelog = "{{ cookiecutter.url }}/releases"
Chen2020 = "{{ cookiecutter.__project_slug }}.parameters.input.Chen2020:get_parameter_values"

[project.entry-points."models"]
SPM = "{{ cookiecutter.__project_slug }}.models.input.SPM:SPM"
SPM = "{{ cookiecutter.__project_slug }}.models.input.spm:SPM"
Reservoir = "{{ cookiecutter.__project_slug }}.models.input.basic_reservoir:BasicReservoir"
santacodes marked this conversation as resolved.
Show resolved Hide resolved

{# keep this line here for newline #}
{%- if cookiecutter.backend == "hatch" %}
Expand Down
Loading
Loading