title | subtitle | author | titlepage | toc-own-page | book | reference-section-title | bibliography | listings |
---|---|---|---|---|---|---|---|---|
HydDown |
User guide and technical reference |
Anders Andreasen |
true |
true |
true |
References |
references.bib |
true |
HydDown is an open source python3 tool for calculation of hydrogen (or other pure gas phase species) vessel/container depressurization and filling. The HydDown logo shown in [@Fig:logo] visualizes the key parameters and transport phenomena during gas vessel filling or discharging. The thermodynamic state inside the vessel changes over time as seen from immediately observable variables temperature (T) and pressure (P). This is caused by change in fluid inventory (density) due to flow of gas either in or out of the vessel. Further, heat is transferred from or to the surroundings via convective heat transfer on the in- and outside of the vessel with heat being conducted through the vessel wall.
Running the code is as simple as:
python main.py input.yml
where main.py
is the main script and input.yml
is the input file in Yaml syntax.
If you use HydDown please cite the following reference [@Andreasen2021]:
Andreasen, A., (2021). HydDown: A Python package for calculation of hydrogen (or other gas) pressure vessel filling and discharge. Journal of Open Source Software, 6(66), 3695, https://doi.org/10.21105/joss.03695
@article{Andreasen2021,
doi = {10.21105/joss.03695},
url = {https://doi.org/10.21105/joss.03695},
year = {2021},
publisher = {The Open Journal},
volume = {6},
number = {66},
pages = {3695},
author = {Anders Andreasen},
title = {HydDown: A Python package for calculation of hydrogen (or other gas) pressure vessel filling and discharge},
journal = {Journal of Open Source Software}
}
This manual can be cited as [@Andreasen2024]:
Anders Andreasen. HydDown - User guide and technical reference. 2024. ⟨hal-04858235⟩
@report{Andreasen2024,
url = {https://hal.science/hal-04858235v1},
year = {2024},
publisher = {HAL open science},
author = {Anders Andreasen},
title = {HydDown -- user guide and technical reference},
}
HydDown started as a small spare-time project for calculation of vessel filling and depressurization behaviour. This was mainly to demonstrate that although perceived as a very tedious and difficult task to write your own code for such an apparently complex problem, actually only a fairly limited amount of code is necessary if you have a good thermodynamic backend. The code has evolved to a point where additional features will increase the complexity to the point where it is no longer a simple tool.
A few choices has been made to keep things simple:
- Coolprop is used as thermodynamic backend
- Only pure substances are considered (limited multi-component capabilities are included)
- Gas phase only
- No temperature stratification in the gas phase
- No temperature gradient through vessel wall
- Heat transfer is modelled as constant or simplified using empirical correlations
These choices make the problem much simpler to solve:
First of all, the the pure substance Helmholtz energy based equation of state (HEOS) in coolprop
offers a lot of convenience in terms of the property pairs/state variables that can be set independently.
Using only a single gas phase species also means that component balances is redundant and 2 or 3-phase flash calculations are not required.
That being said, the principle used for a single component is more or less the same, even for multicomponent mixtures with potentially more than one phase.
The source code can be obtained either from GitHub (via git
or via the latest tar-ball release) or via pip . No packaged releases have currently been planned for conda.
The main branch is located here:
https://github.com/andr1976/HydDown
Clone the repo by:
git clone https://github.com/andr1976/HydDown.git
Running from source via git
the dependencies must be installed manually from the repo root dir:
pip install -r requirements.txt
Installation of latest release via pip also installe dependecies automatically:
pip install hyddown
In case pip
links to a v2.7 of python you will get an error. If so try the following:
python3 -m pip install hyddown
where python3 is the symlink or full path to the python3 executable installed on your system.
- Python (Check CoolProp for latest suppeorted python version)
- Numpy
- matplotlib
- Coolprop (6.4.1)
- cerberus
- PyYaml
- pandas
- Scipy
- tqdm
The script is running on Windows 10 x64, with stock python installation from python.org and packages installed using pip. It should also run on Linux (it does on an Ubuntu image on GitHub) or in any conda environment as well, but this hasn't been checked.
Although testing is mainly intended for automated testing (CI) during development using github actions, testing of the installed package can be done for source install by:
python -m pytest
run from the root folder. In writing 33+ test should pass.
For the package installed with pip navigate to the install directory ../site-packages/hyddown
and run:
python -m pytest
The SI unit system is adapted for this project. The following common units are used in the present project and this also applies to the units used in the input files:
Property | Unit | Comment |
---|---|---|
Temperature | K |
|
Pressure | Pa | bar is used in plots |
Mass | kg | |
Volume | m$^3$ | |
Time | s | |
Energy | J | |
Duty/power | W | |
Length | m | |
Area | m$^2$ | |
Heat flux | W/m$^2$ | |
Heat transfer coefficient | W/(m$^2$ K) | |
Thermal conductivity | W/(m K) | |
Density | kg/m$^3$ | |
Heat capacity | J/(kg K) |
: Unit system {#tbl:units}
As will be noted when presenting the equations implemented in the code, some of the equations utilise different units than the ones listed in [@tbl:units]. However, it is important to note that unit conversions are built in to the methods implemented, so the user shall not worry about unit conversion.
In the making of this document a great deal of material has been sourced (and modified) from a good colleague's M.Sc. thesis [@iskov], from co-published papers [@Bjerre2017][@safety4010011] and from on-line material published under permissive licenses (with proper citation). Further, the making of this project would not have possible without the awesome CoolProp library [@doi:10.1021/ie4033999]. I am also thankful for enlightening discussions with colleague Jacob Gram Iskov Eriksen (Ramboll Energy, Denmark) and former Ramboll Energy colleague Carsten Stegelmann (ORS Consulting) in relation to vessel depressurization, nozzle flow and heat transfer considerations.
The present document is typeset using Markdown + pandoc with the Eisvogel template.
MIT License
Copyright (c) 2021-2025 Anders Andreasen
Permission is hereby granted, free of charge, to any person obtaining a copy of this software and associated documentation files (the "Software"), to deal in the Software without restriction, including without limitation the rights to use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of the Software, and to permit persons to whom the Software is furnished to do so, subject to the following conditions:
The above copyright notice and this permission notice shall be included in all copies or substantial portions of the Software.
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
From the source repository, running the code is as simple as:
python src/main.py input.yml
where main.py is the main script and input.yml is the input file in Yaml syntax.
The Yaml input file is edited to reflect the system of interest.
Further, a usable copy of a main script is installed in the python installation Scripts/ folder:
/python3x/Scripts/hyddown_main.py
or from GitHub/in the source folder:
HydDown/Scripts/hyddown_main.py
Various example files are included for inspiration under site-packages:
/site-packages/hyddown/examples/
or in the git/source directory tree
HydDown/src/hyddown/examples/
A few demonstrations of the codes capability are available as streamlit
apps:
- Vessel gas pressurisation/depressurization for various gases at
https://share.streamlit.io/andr1976/hyddown/main/scripts/streamlit_app.py
- Response to external fire of vessel equipped with a PSV using the Stefan-Boltzmann fire equation at
https://share.streamlit.io/andr1976/hyddown/main/scripts/streamlit_sbapp.py
- Vessel depressurization including slow opening of valve at
https://share.streamlit.io/andr1976/hyddown/main/scripts/streamlit_cvapp.py
The various streamlit apps are also included in the scripts folder for running the scripts on a local machine.
The following methods are implemented:
- Isothermal: constant temperature of the fluid during depressurization (for a very slow process with a large heat reservoir)
- Isenthalpic: constant enthalpy of the fluid, no heat transfer with surroundings, no work performed by the expanding fluid for depressurization, no work added from inlet stream
- Isentropic: constant entropy of the fluid, no heat transfer with surroundings, PV work performed by the expanding fluid
- Isenergetic: constant internal energy of the fluid
- Energy balance: this is the most general case and is based on the first law of thermodynamics applied to a flow process.
For isothermal
/isenthalpic
/isentropic
/isenergetic
calculations the minimal input required are:
- Initial conditions (pressure, temperature)
- vessel dimensions (ID, length)
- valve parameters (Cd, diameter, backpressure)
- Calculation setup (time step, end time)
- Type of gas
If heat transfer is to be considered the calculation type energybalance
is required. A few options are possible:
- Fixed U: U-value, i.e. overall heat transfer coefficient, is required and ambient temperature required. Based on the ambient (external) temperature, the fluid temperature, and the U-value, the heat flux Q is calculated for each time step. In this calculation method the temperature of the vessel wall is not calculated:
- Fixed Q: The external heat flux, Q, applied to the fluid is specified and constant. The ambient temperature is not required. Further, the vessel wall temperature is redundant and not calculated.
- Specified h: The external heat transfer coefficient must be specified and either the internal heat transfer coefficient is provided or calculated from the assumption of natural convection from a vertical cylinder at high Gr number. Ambient temperature is required. Using this method the wall temperature is calculated from an energy balance over the vessel wall taking in and out flux to/from the external ambient plenum as well as heat flux to/from the fluid inventory into account.
- Fire: The Stefan-Boltzmann equation is applied for estimating the external heat duty. The fire heat flux depends on the vessel wall temperature, and the wall temperature is continuously updated as in the method with specified h.
More elaborate description of the required input for the different calculation types are provided in [@Sec:input].
HydDown comes with a script which can be used as a command-line tool to start calculations.
If an input filename (path) is given as the first argument, this input file will be used.
If no arguments are passed, the script will look for an input file with the name input.yml
.
The content of the main script is shown below:
import yaml
import sys
from hyddown import HydDown
if __name__ == "__main__":
if len(sys.argv) > 1:
input_filename = sys.argv[1]
else:
input_filename = "input.yml"
with open(input_filename) as infile:
input = yaml.load(infile, Loader=yaml.FullLoader)
hdown=HydDown(input)
hdown.run()
hdown.verbose=1
hdown.plot()
To use HydDown simple import the main calculation class HydDown
.
from hyddown import HydDown
When using HydDown a dictionary holding all relevant input in order for HydDown to do vessel calculations shall be provided when the class is initialized. One way is to read an input file. For HydDown a Yaml format is chosen, but if JSON is a preference this should also work with the Yaml parser being substituted with a JSON parser.
An example of a minimal file for an isentropic vessel depressurization (no heat transfer) is shown below
vessel:
length: 1.524
diameter: 0.273
initial:
temperature: 388.0
pressure: 15000000.
fluid: "N2"
calculation:
type: "isentropic"
time_step: 0.05
end_time: 100.
valve:
flow: "discharge"
type: "orifice"
diameter: 0.00635
discharge_coef: 0.8
back_pressure: 101300.
A more elaborate example which includes heat transfer and with validation data (some data points dropped for simplicity) included:
vessel:
length: 1.524
diameter: 0.273
thickness: 0.025
heat_capacity: 500
density: 7800.
orientation: "vertical"
initial:
temperature: 288.0
pressure: 15000000.
fluid: "N2"
calculation:
type: "energybalance"
time_step: 0.05
end_time: 100.
valve:
flow: "discharge"
type: "orifice"
diameter: 0.00635
discharge_coef: 0.8
back_pressure: 101300.
heat_transfer:
type: "specified_h"
temp_ambient: 288.
h_outer: 5
h_inner: 'calc'
validation:
temperature:
gas_high:
time: [0.050285, ... , 99.994]
temp: [288.93, ... ,241.29]
gas_low:
time: [0.32393, ..., 100.11]
temp: [288.67, ... ,215.28]
wall_low:
time: [0.32276, ... , 100.08]
temp: [288.93, ... ,281.72]
wall_high:
time: [0.049115, ... ,100.06]
temp: [289.18, ... ,286.09]
pressure:
time: [0.28869, ... , 98.367]
pres: [150.02, ... ,1.7204]
In the following the full hierarchy of input for the different calculation types is summarised.
At the top level the following fields are accepted, with the last being optional and the second last dependant on calculation type:
initial: mandatory
vessel: mandatory
calculation: mandatory
valve: mandatory
heat_transfer: depends on calculation type
validation: optional
The subfields under calculation
, with value formats and options are:
calculation:
type: "isothermal", "isentropic", "isenthalpic", "constantU", "energybalance"
time_step: number
end_time: number
The simulation end time is specified as well as the fixed time step used in the integration of the differential equations to be solved. The four main calculation types are shown as well.
vessel:
length: number, mandatory
diameter: number, mandatory
thickness: number, required when heat transfer is calculated
heat_capacity: number, required when heat transfer is calculated
density: number, required when heat transfer is calculated
orientation: string, required when heat transfer is calculated
initial:
temperature: number, mandatory
pressure: number, mandatory
fluid: string, mandatory , e.g. "N2", "H2"
The valve
field determines the mode of the process is it for depressurization/discharge from the vessel using the value discharge
or if mass flow is entering the vessel, filling it up/increasing pressure with the value filling
.
Different types of mass flow devices can be specified:
- Restriction
orifice
- Relief valve/pressure safety valve (only for
discharge
notfilling
) - Control valve (
controlvalve
) - A specified mass flow (
mdot
)
For the physical devices a back_pressure
is required for the flow calculations.
The value of the back_pressure
is also used to specify the reservoir pressure when the vessel is filled.
See also [@Sec:flow] for details about the calculation of flow rates.
valve:
flow: string, mandatory "discharge" or "filling"
type: string, mandatory "orifice", "controlvalve", "psv", "mdot"
back_pressure: number, required for type "orifice", "controlvalve" and "psv"
diameter: number, required for "orifice" and "psv"
discharge_coef: number, required for "orifice" and "psv"
Cv: number, required for "control_valve"
characteristic: string, optional for "control_valve"
time_constant: number, optional for "control_valve"
For the calculations using a control valve as mass transfer device a linear rate for the actuator can be specified using the time_constant
field. An associated valve characteristic is associated in order to translate the linear actuator rate to an opening Cv (of max. Cv). Three caracteristics can be specified:
- Linear
- Equal percentage (exponential)
- Quick opening/fast (square root)
For more information about the actual estimation of heat transfer see also [@Sec:heat]. For the case of fire
heat input predetermined parameters for the Stefan-Boltzmann fire equation are used to calculate different background heat loads cf. [@Tbl:fire]
Source | Fire type | Heat load ( |
---|---|---|
API521 | Pool | 60 |
API521 | Jet | 100 |
Scandpower | Pool | 100 |
Scandpower | Jet | 100 |
: Fire heat loads {#tbl:fire}
heat_transfer:
type: string, mandatory, "specified_h", "specified_Q", "specified_U", "s-b"
temp_ambient: number, required for type "specified_h", "specified_U"
h_outer: number, required for type "specified_h"
h_inner: number or 'calc', required for type "specified_h"
U_fix: number, required for type "specified_U"
Q_fix: number, required for type "specified_Q"
fire: string, required for type "s-b", 'api_pool','api_jet','scandpower_pool','scandpower_jet'
D_thoat: number, required for flow type "filling", set to vessel ID as a starting point
In order to plot measured data against simulated data the field validation
is included.
The following arrays (one or more) are supported in the sub-field temperature
:
gas_high
: highest measured values of the bulk gasgas_low
: lowest measured values of the bulk gasgas_mean
: average measured values of the bulk gaswall_mean
: average measured values of the vessel (inner) wallwall_high
i.e. highest measured values of the vessel (inner) wallwall_low
i.e. lowest measured values of the vessel (inner) wall
For each of the above fields arrays for time
and temp
shall be supplied with matching length.
A field for measured vessel pressure is also possible, where time
and pres
shall be identical length arrays. See also example below.
validation:
temperature:
gas_high:
time: [0.050285, ... , 99.994]
temp: [288.93, ... ,241.29]
gas_low:
time: [0.32393, ..., 100.11]
temp: [288.67, ... ,215.28]
wall_low:
time: [0.32276, ... , 100.08]
temp: [288.93, ... ,281.72]
wall_high:
time: [0.049115, ... ,100.06]
temp: [289.18, ... ,286.09]
pressure:
time: [0.28869, ... , 98.367]
pres: [150.02, ... ,1.7204]
In this chapter the basic theory and governing equations for the model implementation in HydDown is presented. The following main topics are covered:
- thermodynamics
- mass transfer
- heat transfer
The equation of state used by HydDown is the Helmholtz energy formulation as implemented in CoolProp [@doi:10.1021/ie4033999]. Most of the text in the present section has been sourced from the CoolProp documentation to be as accurate and true to the source as possible. The Helmholtz energy formulation is a convenient construction of the equation of state because all the thermodynamic properties of interest can be obtained directly from partial derivatives of the Helmholtz energy.
It should be noted that the EOS are typically valid over the entire range of the fluid, from subcooled liquid to superheated vapor, to supercritical fluid.
In general, the EOS are based on non-dimensional terms
where
The non-dimensional Helmholtz energy of the fluid is given by
where
and the non-dimensional residual Helmholtz energy of
and all the terms other than
The other thermodynamic parameters can then be obtained through analytic derivatives of the Helmholtz energy terms. For instance, the pressure is given by
and the specific internal energy by
$$ \frac{u}{RT}=\tau \left[\left(\frac{\partial \alpha^0}{\partial \tau}\right){\delta}+ \left(\frac{\partial \alpha^r}{\partial \tau}\right){\delta} \right] $$
and the specific enthalpy by
$$ \frac{h}{RT}=\tau \left[\left(\frac{\partial \alpha^0}{\partial \tau}\right){\delta}+ \left(\frac{\partial \alpha^r}{\partial \tau}\right){\delta} \right] +\delta\left(\frac{\partial \alpha^r}{\partial \delta}\right)_{\tau}+1 $$
which can also be written as
The specific entropy is given by
$$ \frac{s}{R}=\tau \left[\left(\frac{\partial \alpha^0}{\partial \tau}\right){\delta}+ \left(\frac{\partial \alpha^r}{\partial \tau}\right){\delta} \right]-\alpha^0-\alpha^r $$
and the specific heats at constant volume and constant pressure respectively are given by
$$ \frac{c_v}{R}=-\tau^2 \left[\left(\frac{\partial^2 \alpha^0}{\partial \tau^2}\right){\delta}+ \left(\frac{\partial^2 \alpha^r}{\partial \tau^2}\right){\delta} \right] $$
$$ \frac{c_p}{R}=\frac{c_v}{R}+\dfrac{\left[1+\delta\left(\frac{\partial \alpha^r}{\partial \delta}\right){\tau}-\delta\tau\left(\frac{\partial^2 \alpha^r}{\partial \delta\partial\tau}\right)\right]^2}{\left[1+2\delta\left(\frac{\partial \alpha^r}{\partial \delta}\right){\tau}+\delta^2\left(\frac{\partial^2 \alpha^r}{\partial \delta^2}\right)_{\tau}\right]} $$
The EOS is set up with temperature and density as the two independent properties, but often other inputs are known, most often temperature and pressure because they can be directly measured. As a result, if the density is desired for a known temperature and pressure, it can be obtained iteratively.
The control volume sketched in [@Fig:firstlaw], separated from the surrounding by a control surface, is used as a basis for the analysis of an open thermodynamic system with flowing streams (fs) in and out, according to [@sva]
A general mass balance or continuity equation can be written:
$$ \frac{m_{cv}}{dt} + \Delta \left( \dot{m} \right)_{fs}= 0 $$ {#eq:continuity}
The first term is the accumulation term i.e. the rate of change of the mass inside the control volume,
An energy balance for the control volume, with the first law of thermodynamics applied, needs to account for all the energy modes with can cross the control surface. Each stream has a total energy
where the first term is the specific internal energy, the second term is the kinetic energy and the last term is the potential energy. The rate at which each stream transports energy in or out of the control volume is given by
and in total
Furthermore, work (not to be confused with shaft work) is also associated with each stream in order to move the stream into or out from the control volume (one can think of a hypothetical piston pushing the fluid at constant pressure), and the work is equal to
and in total
Further, heat may be transferred to (or from) the control volume at a rate
$$ \frac{d(mU){cv}}{dt} + \Delta \left[ \dot{m} (U + \frac{1}{2}u^2 + zg) \right]{fs} + \Delta \left[ \dot{m}(PV) \right]{fs} = \dot{Q} +\dot{W}{shaft} $$
Applying the relation
$$ \frac{d(mU){cv}}{dt} + \Delta \left[ \dot{m} H \right]{fs} = \dot{Q} $$
The equation can be further simplified if only a single port acting as either inlet or outlet is present:
$$ \frac{d(mU)_{cv}}{dt} + \dot{m} H = \dot{Q} $$ {#eq:energybalance}
where the sign of
When a fluid flows through a restriction or opening such as an orifice, the velocity will be affected by conditions upstream and downstream.
If the upstream pressure is high enough, relative to the downstream pressure, the velocity will reach the speed of sound (Ma = 1) and the flow rate obtained will be the critical flow rate.
This condition is referred to as choked flow.
The maximum downstream pressure for the flow to still be sonic (Ma = 1), is when
$$ \frac{P_{c}}{P_u}=\left (\frac{2}{k+1} \right)^\frac{k}{k-1} $$ {#eq:P_critical}
-
$P_c$ is the critical pressure. [kPa] -
$P_d$ is the downstream pressure. [kPa] -
$P_u$ is the upstream pressure. [kPa] - k is the isentropic coefficient, approximated by the ideal gas heat capacity ratio
$C_p/C_v$ . [-]
In order to calculate the mass flow rate through an orifice equations are used based on literature from the Committee for the Prevention of Disasters [@yellowbook].
To account for the difference in choked and non-choked flow a set limit pressure is introduced as in equation [@Eq:plimit].
If the downstream pressure,
$$ P_{limit}=P_{up} \cdot \left ( \frac{2}{k+1} \right ) ^{\frac{k}{k-1}} $$ {#eq:plimit}
$$ \dot{m}{flow}= C_d \cdot A \cdot\sqrt{\left ( \frac{2 k}{k-1}\right ) \cdot P{up} \cdot \rho \cdot \left ( \frac{P_{used}}{P_{up}} \right )^{\frac{2}{k}} \left (1-\left ( \frac{P_{used}}{P_{up}} \right ) ^{\frac{k-1}{k}} \right )} $$ {#eq:massfloworifice}
-
$\rho$ is the density of the gas upstream.$[kg/m^3]$ -
$P_{limit}$ is the pressure limit of the upstream absolute pressure.$[bara]$ -
$P_{up}$ is the absolute pressure upstream of the orifice.$[bara]$ -
$k$ is the ratio of the heat capacities at constant pressure,$C_p$ , and at constant volume,$C_v$ . -
$P_{down}$ is the absolute pressure downstream of the orifice.$[bara]$ -
$P_{used}$ is the pressure used in the mass flow equation based on choked or non-choked conditions.$[bara]$ -
$\dot{m}_{flow}$ is the mass flow through the orifice.$[kg/s]$ -
$C_d$ is the discharge coefficient of the orifice opening.$[-]$ -
$A$ is the cross sectional area of the orifice.$[m^2]$
A PSV / relief valve is a mechanical device actuated by the static pressure in the vessel and a conventional PSV is often used for gas/vapor systems. A conventional PSV is a spring-loaded device which will activate at a predetermined opening pressure, and relieve the vessel pressure until a given reseat pressure has been reached. Both the opening pressure and the reset pressure is above the vessel operating pressure and the PSV will remain closed until the pressure inside the vessel increases to the opening pressure. The operation of a conventional spring-loaded PSV is based on a force balance. A conventional PSV can be seen in [@Fig:psv]. A spring exerts a force on a disc blocking the inlet of the PSV. When the pressure inside the vessels reaches the opening pressure, the force exerted on the disc by the gas will be larger than the force exerted by the spring and the PSV will open and allow the gas to flow out of the vessel. The flow of gas out of the vessels will lower the pressure and thereby also the force exerted on the disc. When the pressure in the vessels is reduced to the reset pressure, the PSV will close and the disc will again hinder the gas flow.
The relief valve model implemented in HydDown is the API 520 equations [@API520] for gas relief for both sonic/critical as well as subcritical flow. No corrections factors are implemented in HydDown.
For sonic flow (critical flow), as indicated in equation, the mass flow through the PSV can be determined by equation [@eq:Stationary_sizing_sonic].
$$ W = \frac{A {C \cdot K_d \cdot K_b \cdot K_c \cdot P_1}}{\sqrt{\frac{T\cdot Z}{M}}} $$ {#eq:Stationary_sizing_sonic}
- A is the effective discharge area. [mm$^2$]
- W is the mass flow through the device. [kg/h]
- C is a coefficient, as a function of k, as defined in equation [@eq:Stationary_sizing_C_function].
-
$K_d$ ,$K_b$ , and$K_c$ are correction factors. -
$P_1$ is the allowable upstream absolute pressure. [kPa] - T is the temperature of the inlet gas at relieving conditions. [K]
- M is the molecular mass of the gas at relieving conditions. [kg/kmol]
- Z is the compressibility factor for the gas.
$$ C=0.03948 \sqrt{k \left (\frac{2}{k+1}\right)^{\left (\frac{k+1}{k-1}\right)}} $$ {#eq:Stationary_sizing_C_function}
For subsonic flow (subcritical flow), the effective discharge area of the PSV is determined by equation [@eq:Stationary_sizing_subsonic].
$$ W = \frac{A \cdot {F_2 \cdot K_d \cdot K_c }}{17.9 \sqrt{\frac{T\cdot Z}{M\cdot P_1\cdot (P_1-P_2)}}} $$ {#eq:Stationary_sizing_subsonic}
F$_2$ is the coefficient of subcritical flow which can be determined from [@eq:Stationary_sizing_F2].
$$ F_2=\sqrt{\left ( \frac{k}{k-1} \right ) r^{\left (\frac{2}{k} \right )} \left ( \frac{1-r^{\left ( \frac{k-1}{k} \right )}}{1-r}\right )} $$ {#eq:Stationary_sizing_F2}
where
When modelling a pop action PSV/relief valve under dynamic conditions, the valve will go from closed to fully open in a short period of time when the set pressure,
When specifying PSVs it common to use standard API sizes as shown in [@tbl:psv_sizes]
Size | Area [in$^2$] | Area [m$^2$] |
---|---|---|
D | 0.110 | 7.09676 |
E | 0.196 | 1.26451 |
F | 0.307 | 1.98064 |
G | 0.503 | 3.24515 |
H | 0.785 | 5.06450 |
J | 1.287 | 8.30320 |
K | 1.838 | 1.18580 |
L | 2.853 | 1.84064 |
M | 3.600 | 2.32257 |
N | 4.340 | 2.79999 |
P | 6.380 | 4.11612 |
Q | 11.050 | 7.12901 |
R | 16.000 | 1.03225 |
T | 26.000 | 1.67741 |
: Standard PSV orifice sizes according to API {#tbl:psv_sizes}
For calculating the mass flow through a control valve, the ANSI/ISA [@borden][@ISA] methodology also described in IEC 60534 [@IEC60534] is applied.
The flow model for a compressible fluid in the turbulent regime is:
or equivalently:
- C is the flow coefficient (
$C_v$ or$K_v$ ) -
$N_8$ is a unit specific constant, 94.8 for$C_v$ and bar as pressure unit -
$F_P$ is a piping geometry factor [-] - Y is the expansion factor [-]
-
$x_{sizing}$ is is the pressure drop used for sizing[-] -
$p_1$ is the upstream pressure [bar] -
$\rho_1$ is the upstream density [kg/m$^3$] - M is the molecular weight [kg/kmol]
-
$T_1$ is the upstream temperature [K] -
$Z_1$ is the upstream compressibility [-]
In HydDown, the piping geometry factor is not yet implemented and assumed to be 1.
The pressure drop ratio
The pressure drop ratio at which flow no longer increases with increased value in pressure drop ratio is the choked pressure drop ratio, given by the following equation:
The factor
where
The expansion factor Y accounts for the change in density as the fluid passes from the valve inlet to the vena contracta. It also accounts for the change in the vena contracta area as the pressure differential is varied.
Experiments have indicated that the internal heat transfer mechanism for a vessel subject to depressurization can be well approximated by that of natural convection as found from measured Nusselt numbers being well correlated with Rayleigh number, with no apparent improvement in model performance by inclusion of the Reynolds number in the model [@woodfield].
To determine the heat transfer for the gas-wall interface, the following is applied cf. equation [@Eq:newton]:
$$ \frac{dQ}{dt} = \dot{Q} = h A ( T_{s} - T_{gas} ) $$ {#eq:newton}
-
$d Q$ is the change in thermal energy due to convective heat transfer. [J] -
$d t$ is the change in time during the heat transfer. [s] -
$h$ is the convective heat transfer. [W/m$^2$$\cdot$ K ] -
$A$ is the area normal to the direction of the heat transfer. [m$^2$] -
$T_{s}$ is the inner surface temperature of the geometry. [K] -
$T_{gas}$ is the temperature of the bulk gas inside the vessel. [K]
The convective heat transfer will need to be estimated for the the gas-wall interface, by the use of empirical relations for the Nusselt number. The Nusselt number describes the ratio of convective heat transfer to conductive heat transfer, normal to a surface area, as given in equation [@eq:Nu].
$$ Nu=\frac{hL}{k} $$ {#eq:Nu}
-
$Nu$ is the Nusselt number. [-] -
$h$ is the convective heat transfer. [W/m$^2$$\cdot$K] -
$L$ is a characteristic length of the geometry. [m] -
$k$ is the thermal conductivity of the gas. [W/m$\cdot$K]
The characteristic length
The empirical correlations used to calculate the Nusselt number of the gas-wall interface is a function of the Rayleigh number, which can be defined by the Grashof number and Prandtl number, as in equation [@Eq:rayleigh_gas]:
$$ Ra=Gr \cdot Pr $$ {#eq:rayleigh_gas}
-
$Ra$ is the Rayleigh number. [-] -
$Gr$ is the Grashof number. [-] -
$Pr$ is the Prandtl number. [-]
The Grashof number is a dimensionless number which approximates the ratio of the buoyancy forces to viscous forces, as given in equation [@Eq:grashof_gas}:
$$ Gr=\frac{\beta g\rho^2 L^3 \Delta T }{\mu^2} $$ {#eq:grashof_gas}
The Prandtl number is a dimensionless number defined as the ratio of the momentum diffusivity to thermal diffusivity, as given in equation [@Eq:prandtl_gas]:
$$ Pr=\frac{c_p \mu}{k} $$ {#eq:prandtl_gas}
-
$\beta$ is the coefficient of volume expansion. [1/K] -
$g$ is the standard acceleration of gravity. [m/s$^2$] -
$\rho$ is the gas density. [kg/m$^3$] -
$L$ is the characteristic length. [m] -
$\Delta T$ is the temperature difference of the surface and gas. [K] -
$\mu$ is the dynamic viscosity. [kg/m$\cdot$s] -
$c_p$ is the heat capacity of gas. [J/kg$\cdot$K] -
$k$ is the thermal conductivity of gas. [J/m$\cdot$K]
It is important to note that the properties in the above equations shall be evaluated at the fluid film temperature which can be approximated by the average of the the fluid bulk temperature and the vessel wall temperature [@geankoplis].
The mechanism of heat transfer on the outside of the vessel at ambient conditions is similar to the above. In HydDown the external heat transfer is not modelled currently. A heat transfer coefficient shall be provided.
Experiments have indicated that the internal heat transfer mechanism for a vessel subject to filling can be well approximated by that of combined natural convection and forced convection as found from measured Nusselt numbers being well correlated with Rayleigh and Reynolds number [@woodfield].
For mixed convection the effective Nusselt number,
During charging with different gases (H$_2$, N$_2$ and Argon), Woodfield et al. demonstrated that in order to provide a good fit to the experimentally determined Nusselt number a correlation based on both Reynolds and Rayleigh number was necessary. The following formula was found to provide a good fit (n=1):
For accurate prediction of the outer and especially the inner wall temperature for correct estimation of internal convective heat transfer and the average material temperature, the general equation of 1-D unsteady heat transfer shall be solved:
- T is temperature
- x is the spatial (1-D) coordinate
- k is the thermal conductivity
-
$\rho$ is the material density -
$C_p$ is the specific heat capacity
To be solved, the initial values and boundary values must be specified. In its default state (if thermal cobductivity is not applied for the vessel), HydDown does not include the unsteady heat transfer model, i.e. the assumption is that the temperature from outer to inner surface is uniform and equal to the average temperature. This is obviously a crude approximation, but might be justified depending in the Biot number:
The Biot number is a simple measure of the ratio of the heat transfer resistances at the surface of a body to the inside of a body. The ratio gives an indication to which extent the temperature will vary in space (gradient) when the body is subject to a displacement in temperature at the surface boundary layer. Striednig et al. [@STRIEDNIG] concluded that for a type I (steel) cylinder the Biot number was approx. 0.03 and hence the error in assuming a uniform temperature in the vessel wall was low.
With a typical thermal conductivity of 45
Especially for vessels with low conductivity materials (or very thick walls) accurate estimation of the vessel wall temperatures requires the 1-D transient heat transfer problem to be solved. HydDown incorporates the thermesh code provided under an MIT license by Wouter Grouve [@thermesh]. The implemented model applies Cartesian coodinates as also applied in the h2fills program by NREL [@KUROKI], i.e. the curved vessel wall is assumed a flat plate. This may to some extent be justied by the relatively large radius of a cylindrical storage container compared to the vessel wall thichness (see also justification references in [@KUROKI]).
In HydDown, the 1-D transient heat conductivity problem is solved with thermesh assuming temperature independent vessel material density, heat capacity and thermal conductivity applying the Crank-Nicholson scheme using Neumann boundary conditions:
$$ -k_{\text{z}}\frac{\partial T}{\partial z}\Biggr|{z=0} = q\text{left}(t), \qquad -k_{\text{z}}\frac{\partial T}{\partial z}\Biggr|{z=L} = q{\text{right}}(t) $$
The heat flux at the outer (left) and inner (right) is calculated/updated for each major time step for the explicit solver for mass and energy balances. For each major time step thermesh is used to update the outer and inner wall temperature by solving the 1-D transient heat conductivity using a minor time step equal to 1/10th of the major time step and 11 nodes. For each time thermesh is called during major time steps the mesh is initialised with the temperature profile at the end of the former major time step.
The heat transfer from the flame to the shell is modelled using the recommended approach from Scandpower [@scandpower] and API [@API521]. The heat transfer from the flame to the vessel shell is divided into radiation, convection, and reradiation as seen in equation [@Eq:flame]:
$$ q_f=\underbrace{{\alpha}s \cdot {\varepsilon}f \cdot \sigma \cdot T_f^4}\text{Radiation}+\underbrace{h_f \cdot (T_f-T_s(t))}\text{Convection}-\underbrace{{\varepsilon}s \cdot \sigma \cdot T_s(t)^4 }\text{Reradiation} $$ {#eq:flame}
-
$q_f$ is the flame heat flux. [W/m$^2$] -
${\alpha}_s$ is the vessel surface absorptivity. [-] -
${\varepsilon}_f$ is the flame emissivity. [-] -
$\sigma$ is the Stefan-Boltzmann constant,$\sigma$ =$5.67 \cdot 10 ^ {-8}$ [W/m$^2 \cdot$ K$^4$] -
$T_f$ is the flame temperature. [K] -
$h_f$ is the convection heat transfer coefficient between the flame and the surface. [W/m$^2 \cdot$ K] -
$T_s(t)$ is the time dependent surface temperature. [K] -
${\varepsilon}_s$ is the surface emissivity. [-]
This model assumes that the pressure vessel is fully engulfed by the flame. This means that the view factor for the radiation is unity and is therefore not taken into consideration. The convective heat transfer coefficients for a jet fire and a pool fire, and recommended values for the emissivity and absorptivity, are given by Scandpower as [@scandpower]:
-
$h_{jet~fire}$ = 100 [W/m$^2 \cdot$K] -
$h_{pool~fire}$ = 30 [W/m$^2 \cdot$K] -
${\alpha}_s$ = 0.85 -
${\varepsilon}_s$ = 0.85 -
${\varepsilon}_f$ = 1.0 (optical thick flames, thickness > 1 m)
The flame temperature is found by solving equation [@Eq:flame2] for the incident heat flux in relation to the ambient conditions. The flame temperature is kept constant throughout the simulation:
$$ q_{total}=\sigma \cdot T_f^4 + h_f \cdot (T_f-T_{amb})$$ {#eq:flame2}
-
$q_{total}$ is the incident flame heat flux as given in table [@Tbl:heatfluxes1]. [W/m$^2$] -
$T_{amb}$ is the ambient temperature$\approx$ 293 K (20$^\circ$ C)
The heat flux used to calculate the flame temperature is given in table [@tbl:heatfluxes1].
Small jet fire [kW/m$^2$] | Large jet fire [kW/m$^2$] | Pool fire [kW/m$^2$] | |
---|---|---|---|
Peak heat load | 250 | 350 | 150 |
Background heat load | 0 | 100 | 100 |
: Incident heat fluxes for various fire scenarios given by Scandpower [@scandpower] {#tbl:heatfluxes1}
A simple (naive) explicit Euler scheme is implemented to integrate the mass balance over time, with the mass rate being calculated from an orifice/valve equation as described in [@Sec:flow]:
$$ m_{cv}(i+1) = m_{cv}(i) + \dot{m}(i) \Delta t $$ {#eq:euler_mass}
For each step, the mass relief/ left in the vessel is known. Since the volume is fixed the mass density is directly given.
For the calculation methods (isentropic, isenthalpic, isenergetic, etc.), CoolProp allows specifying density and either H, S, or U directly - this is very handy and normally only TP, PH, or TS property pairs are implemented and you would need to code a second loop to make it into am UV, VH, or SV calculation.
In the following, the high-level steps in the solution procedure are outlined for the different calculation types.
To illustrate calls to the equation of state in CoolProp, the notation
For an isothermal process, the solution procedure for each calculation step is the following with the mass in the control volume being calculated from [@Eq:euler_mass]:
The notation
For an isenthalpic process:
For an isentropic process:
For an isenergetic process:
The general first law applied to a flow process as outlined in [@Sec:firstlaw] subject to an explicit Euler scheme is:
$$ D(i+1) = \frac{m_{cv}(i+1)}{V} $$ $$ U_{cv}(i+1) = \frac{m_{cv}(i)U_{cv}(i) - \left( \dot{m}(i) H (i) + \dot{Q}(i) \right) \Delta t}{m_{cv}(i+1)} $$ {#eq:firstlaw_euler}
The above assumes that mass flow is positive when leaving the control volume and heat rate is positive when transferred to the control volume.
For the vessel wall a simple heat balance is also made:
$$ T_{wall}(i+1) = T_{wall}(i) + \frac{\dot{Q}{outer} - \dot{Q}{inner} } {c_p m_{vessel}} \Delta t $$
where $\dot{Q}{outer}$ is the convective heat transfer to or from the ambient surroundings from the outer surface of the vessel, with positive values indicating that heat is transferred from the surroundings to the vessel wall. This is either a fixed heat transfer coefficient with a specified ambient temperature (outer surface) or a calculated fire heat load. $\dot{Q}{inner}$ is the internal heat transfer, either a fixed number or calculated as natural convection (for discharge) or mixed natural and forced convection (filling).
When the temperature profile of the vessel wall is calculated using the 1-D transient conductivity equation the vessel temperature as calculated above represent the mean vessel temperature and addition to this the outer and inner wall temperatures are calculated.
If a fixed
The remaining steps are update of temperature and pressure:
Although not an initial requirement, the code can handle multi-component gas mixtures. However, no validation has been performed. Furthermore, when calculations are done for multicomponent mixtures, the code runs significantly slower. Please note, that in case that liquid condensate is formed during discharge calculations and even if the calculations does not stop, the results cannot be considered reliable. This is because component balances are not made and it is always assumed that the discharge composition is the same as the global composition inside the vessel. When liquid condensate is formed the composition of the vapour phase will differ from the global composition.
There are a few examples of multicomponent mixtures included with HydDown. In order to specify multicomponent mixtures the below example can be used as guidance:
"Methane[9.01290129e-01]&Ethane[6.35063506e-02]&N2[7.80078008e-03]&CO2[2.34023402e-02]&Propane[3.50035004e-03]&Butane[5.00050005e-04]"
For component names please refer to the Coolprop manual.
An example calculation for the above mixture is shown in [@Fig:ng_all].
The pressure and temperature trajectory is visualised along with the fluid mixture phase envelope in [@Fig:ng_pe]. As seen from the figure, this case is borderline and the pressure/temperature trajectory just coincides with the dew line on the phase envelope. This plot is included in the example main script that comes with HydDown and serves as an important quality control.
For multi-component mixtures CoolProp does not provide solver for PH and UV-problems and these solvers have been included in HydDown by wrapping the CoolProp PT solver with a Nelder-Mead algorithm for solving for internal energy and density or pressure and enthalpy.
The code is provided as-is. However, comparisons have been made to some experiments from the literature.
The following gases and modes are considered:
- High pressure hydrogen discharge
- High pressure hydrogen filling
- High pressure nitrogen discharge
- High pressure multi-component gas discharge
These cases are all for Type I (steel) cylinders and the simple heat transfer / heat balance model has been applied, ignoring the thermal gradient in the cylinder wall. It has been checked by applying the 1D transient heat conduction model that results basically unaffected by this assumption. In addition to these, a few cases for Type IV cylinders have also been included in order to validate the 1D transient heat conduction model applied. This includes
- High hydrogen discharge - comparison with commercial software.
- High pressure helium discharge experiment (Karlshuhe Institute of Technology)
- High pressure hydrogen discharge (GasTeF experiment)
Details on Type III experiments are sparse but some general observations are made for the difference between type III and type IV cylinders.
Furthermore, HydDown has also been benchmarked against external code by Ruiz and Moscardelli [@maraggi], who compared their GeoH2 app against HydDown and found excellent agreement for both pressure, mass flow rate and gas temperature for three different cases of discharge and filling.
Calculations with HydDown for high pressure hydrogen vessel discharge have been compared to three experiments from [@byrnes]. Byrnes et al. [@byrnes] have reported a number of rapid depressurization experiments using both nitrogen and hydrogen, with blowdown times being varied between 14 seconds to 33 minutes. The vessel used was a type "K" gas cylinder with a volume of 0.0499 m$^3$, an ID of 8.56 inches and shell length of 55 inches. The vessel is placed in a vertical position. A control valve is mounted on the neck of the cylinder with a fixed opening percentage during each experiment, set to provide the required depressurization rate. It is mentioned that the gas cylinder is placed inside a weather proof cabinet with perlite insulation in order to emulate adiabatic conditions. However the simulations run here are with a finite heat transfer coefficient on the outside of the cylinder. In their paper, detailed pressure and temperature traces are provided for three experimental runs: 7, 8, and 9, with depressurization from 2000 psia in 30 seconds, 480 seconds and 14 seconds, respectively. These experiments are simulated in [@Fig:byrnes_run7], [@Fig:byrnes_run8] and [@Fig:byrnes_run9].
The figure layout is general for all the validation studies conducted and will be described in brief for eased readability. Each experiment figure is divided in 4 subplots: the upper left shows the simulated bulk gas temperature and vessel wall (average) temperature with measured values included with stipulated lines (if available), the upper right shows the simulated and measured pressure (if available, shown with points), the lower left shows the specific enthalpy, specific internal energy, and the specific entropy of the bulk gas content as a function of time, the lower right shows the simulated mass rate leaving or entering the vessel.
As seen from [@Fig:byrnes_run7], [@Fig:byrnes_run8], and [@Fig:byrnes_run9], the calculations generally compare well with the experimental results. It appears as though the faster depressurization times are predicted the best, as seen from run 7 and 9. The much slower run 8 does not have as good a fit for the bulk gas temperature. However, on an absolute scale the results are still within 5$^\circ$C of the experimentally determined gas temperature. The trend in goodness of fit could indicate that the heat transfer from the outside of the vessel to the gas lacks some details, which become detectable at the very slow depressurization. For the fast depressurization the heat transfer from the outside is of less importance and the inside heat transfer coefficient may be predicted better.
In order to compare HydDown to experimental values of hydrogen filling operation the experiments reported by Striednig et al. [@STRIEDNIG] are used. Experimental results were obtained using a type I tank (steel) with a volume of 0.0235 m$^3$. The filling of hydrogen of gas into the vessel was carried out with an upstream reservoir kept at 350 bar and the flow was controlled with an electronically controlled dispenser. Vessel details are provided below:
Parameter | Value |
---|---|
Mass | 69..5 kg |
Length | 0.61 m |
OD | 0.280 m |
Wall thickness | 0.0129 m |
Density | 7740 kg/m$^3$ |
Heat capacity | 470 J/(kg K) |
: Key hydrogen vessel data [@STRIEDNIG] {#tbl:h2filling}
[@STRIEDNIG] reports many different experiments. In this comparison we will use experiments applying different pressurisation rates / durations using identical initial conditions. The experiments are reported in Fig. 6 and Fig. 7 in [@STRIEDNIG].
In its current stage HydDown does not directly support specifying a fixed pressurisation rate expressed as a constant rate of change in pressure / pressure ramp rate (PRR). In order to emulate the experiments, an orifice model is used with an artificially high reservoir pressure in order to ensure that the flow is critical during the entire pressurisation. This gives a constant mass rate and translates to a fairly constant pressure ramp rate, although with real gas effects being clearly observable.
The results of the simulations with HydDown and the experiments from [@STRIEDNIG] are shown in [@Fig:striednig5], [@Fig:striednig10], [@Fig:striednig30]. As seen from the figures the experimental gas temperature appears to be simulated very accurately. The simulation with the fastest pressurisation under-predicts the experimental temperature slightly. The main features of the pressurisation temperature profile is a rapid increase in temperature, at the beginning of the pressurisation, where the effects of filling dominate. At approximately 20-30 s, the temperature increase levels off to a plateau, where the length of the plateau depends on the pressurisation rate. The slower the depressurization, the longer the plateau. During this temperature plateau, the heat transfer from the gas to the vessel and surroundings balance the temperature increase that would occur if performed at adiabatic (isolated) conditions. When the gas flow has ceased upon reaching the final pressure, the temperature starts to drop due to cooling of the gas by the colder vessel wall driven by convective heat transfer. The temperature stabilises at a temperature which is higher than the initial ambient temperature due to the heating of the steel vessel. The cooling of the vessel by ambient air is slower and would require a longer run time to be clearly visible.
Calculations with HydDown are compared to experiment I1 from ref. [@Haque1992b]. The experiment is a blowdown of a vertically oriented cylindrical vessel with flat ends. The vessel length is 1.524 m, the inside diameter is 0.273 m, and the wall thickness is 25 mm. The vessel is filled with N$_2$ at 150 bar and 15$^\circ$C. Ambient temperature is 15$^\circ$C. The blowdown orifice diameter is 6.35 mm. The results are shown in [@Fig:N2val]. The discharge coefficient of the orifice has been set to 0.8 in order to match the vessel pressure profile. The back pressure is set to atmospheric conditions.
As seen from [@Fig:N2val], the calculations compare well with the experimental results. The calculated temperature of the bulk vapor is within the experimental range of measured temperature at all times during the simulation. It is also noted that the minimum temperature is reached at approximately the same time as in the experiments. The calculated vessel inner wall temperature does not decline as rapidly as the experiments, but from around a calculation time of 60 s, the temperature is within the experimentally observed inner wall temperature.
The experiments were conducted by Haque et al. [@Haque1992b] and the experimental data has been extracted from the Ph.d. thesis of Wong [@WONG].
The experimental vessel is a full-size suction scrubber for a gas compressor which has a total volume of 2.78 m3, with length and inside diameter of 3.240 m (2.75 m tan-to-tan) and 1.130 m respectively. The wall thickness is 59 mm. For modelling the length of a flat-ended cylinder has been adjusted to give a total volume of 2.78 m3. An orifice diameter of 6.35 mm was used for discharge and for HydDown modelling the discharge coefficient was adjusted to 0.97 to match the experimental pressure. The results of HydDown and comparison with the experimental results are shown in [@fig:Multi-val].
The minimum gas temperature simulated by HydDown -6 C, compared to the minimum measured average gas temperature of -10.3 C. The difference in measured vs calculated wall temperature is 2.3 C, with a lower measured wall temperature.