mirror of
https://github.com/Cantera/cantera.git
synced 2025-02-25 18:55:29 -06:00
721 lines
30 KiB
Python
721 lines
30 KiB
Python
import itertools
|
|
import numpy as np
|
|
|
|
import cantera as ct
|
|
from . import utilities
|
|
|
|
|
|
class TestPureFluid(utilities.CanteraTest):
|
|
""" Test functionality of the PureFluid class """
|
|
def setUp(self):
|
|
self.water = ct.Water()
|
|
|
|
def test_critical_properties(self):
|
|
self.assertNear(self.water.critical_pressure, 22.089e6)
|
|
self.assertNear(self.water.critical_temperature, 647.286)
|
|
self.assertNear(self.water.critical_density, 317.0)
|
|
|
|
def test_temperature_limits(self):
|
|
co2 = ct.CarbonDioxide()
|
|
self.assertNear(co2.min_temp, 216.54)
|
|
self.assertNear(co2.max_temp, 1500.0)
|
|
|
|
def test_set_state(self):
|
|
self.water.PQ = 101325, 0.5
|
|
self.assertNear(self.water.P, 101325)
|
|
self.assertNear(self.water.Q, 0.5)
|
|
|
|
self.water.TQ = 500, 0.8
|
|
self.assertNear(self.water.T, 500)
|
|
self.assertNear(self.water.Q, 0.8)
|
|
|
|
def test_substance_set(self):
|
|
self.water.TV = 400, 1.45
|
|
self.assertNear(self.water.T, 400)
|
|
self.assertNear(self.water.v, 1.45)
|
|
with self.assertRaisesRegex(ct.CanteraError, 'Negative specific volume'):
|
|
self.water.TV = 300, -1.
|
|
|
|
self.water.PV = 101325, 1.45
|
|
self.assertNear(self.water.P, 101325)
|
|
self.assertNear(self.water.v, 1.45)
|
|
|
|
self.water.UP = -1.45e7, 101325
|
|
self.assertNear(self.water.u, -1.45e7)
|
|
self.assertNear(self.water.P, 101325)
|
|
|
|
self.water.VH = 1.45, -1.45e7
|
|
self.assertNear(self.water.v, 1.45)
|
|
self.assertNear(self.water.h, -1.45e7)
|
|
|
|
self.water.TH = 400, -1.45e7
|
|
self.assertNear(self.water.T, 400)
|
|
self.assertNear(self.water.h, -1.45e7)
|
|
|
|
self.water.SH = 5000, -1.45e7
|
|
self.assertNear(self.water.s, 5000)
|
|
self.assertNear(self.water.h, -1.45e7)
|
|
|
|
self.water.ST = 5000, 400
|
|
self.assertNear(self.water.s, 5000)
|
|
self.assertNear(self.water.T, 400)
|
|
|
|
def test_states(self):
|
|
self.assertEqual(self.water._native_state, ('T', 'D'))
|
|
self.assertNotIn('TPY', self.water._full_states.values())
|
|
self.assertIn('TQ', self.water._partial_states.values())
|
|
|
|
def test_set_Q(self):
|
|
self.water.TQ = 500, 0.0
|
|
p = self.water.P
|
|
self.water.Q = 0.8
|
|
self.assertNear(self.water.P, p)
|
|
self.assertNear(self.water.T, 500)
|
|
self.assertNear(self.water.Q, 0.8)
|
|
|
|
self.water.TP = 650, 101325
|
|
with self.assertRaises(ct.CanteraError):
|
|
self.water.Q = 0.1
|
|
|
|
self.water.TP = 300, 101325
|
|
with self.assertRaises(ValueError):
|
|
self.water.Q = 0.3
|
|
|
|
def test_set_minmax(self):
|
|
self.water.TP = self.water.min_temp, 101325
|
|
self.assertNear(self.water.T, self.water.min_temp)
|
|
|
|
self.water.TP = self.water.max_temp, 101325
|
|
self.assertNear(self.water.T, self.water.max_temp)
|
|
|
|
def check_fd_properties(self, T1, P1, T2, P2, tol):
|
|
# Properties which are computed as finite differences
|
|
self.water.TP = T1, P1
|
|
h1a = self.water.enthalpy_mass
|
|
cp1 = self.water.cp_mass
|
|
cv1 = self.water.cv_mass
|
|
k1 = self.water.isothermal_compressibility
|
|
alpha1 = self.water.thermal_expansion_coeff
|
|
h1b = self.water.enthalpy_mass
|
|
|
|
self.water.TP = T2, P2
|
|
h2a = self.water.enthalpy_mass
|
|
cp2 = self.water.cp_mass
|
|
cv2 = self.water.cv_mass
|
|
k2 = self.water.isothermal_compressibility
|
|
alpha2 = self.water.thermal_expansion_coeff
|
|
h2b = self.water.enthalpy_mass
|
|
|
|
self.assertNear(cp1, cp2, tol)
|
|
self.assertNear(cv1, cv2, tol)
|
|
self.assertNear(k1, k2, tol)
|
|
self.assertNear(alpha1, alpha2, tol)
|
|
|
|
# calculating these finite difference properties should not perturb the
|
|
# state of the object (except for checks on edge cases)
|
|
self.assertNear(h1a, h1b, 1e-9)
|
|
self.assertNear(h2a, h2b, 1e-9)
|
|
|
|
def test_properties_near_min(self):
|
|
self.check_fd_properties(self.water.min_temp*(1+1e-5), 101325,
|
|
self.water.min_temp*(1+1e-4), 101325, 1e-2)
|
|
|
|
def test_properties_near_max(self):
|
|
self.check_fd_properties(self.water.max_temp*(1-1e-5), 101325,
|
|
self.water.max_temp*(1-1e-4), 101325, 1e-2)
|
|
|
|
def test_properties_near_sat1(self):
|
|
for T in [340,390,420]:
|
|
self.water.TQ = T, 0.0
|
|
P = self.water.P
|
|
self.check_fd_properties(T, P+0.01, T, P+0.5, 1e-4)
|
|
|
|
def test_properties_near_sat2(self):
|
|
for T in [340,390,420]:
|
|
self.water.TQ = T, 0.0
|
|
P = self.water.P
|
|
self.check_fd_properties(T, P-0.01, T, P-0.5, 1e-4)
|
|
|
|
def test_isothermal_compressibility_lowP(self):
|
|
# Low-pressure limit corresponds to ideal gas
|
|
ref = ct.Solution('h2o2.yaml', transport_model=None)
|
|
ref.TPX = 450, 12, 'H2O:1.0'
|
|
self.water.TP = 450, 12
|
|
self.assertNear(ref.isothermal_compressibility,
|
|
self.water.isothermal_compressibility, 1e-5)
|
|
|
|
def test_thermal_expansion_coeff_lowP(self):
|
|
# Low-pressure limit corresponds to ideal gas
|
|
ref = ct.Solution('h2o2.yaml', transport_model=None)
|
|
ref.TPX = 450, 12, 'H2O:1.0'
|
|
self.water.TP = 450, 12
|
|
self.assertNear(ref.thermal_expansion_coeff,
|
|
self.water.thermal_expansion_coeff, 1e-5)
|
|
|
|
def test_thermal_expansion_coeff_TD(self):
|
|
for T in [440, 550, 660]:
|
|
self.water.TD = T, 0.1
|
|
self.assertNear(T * self.water.thermal_expansion_coeff, 1.0, 1e-2)
|
|
|
|
def test_pq_setter_triple_check(self):
|
|
self.water.PQ = 101325, .2
|
|
T = self.water.T
|
|
# change T such that it would result in a Psat larger than P
|
|
self.water.TP = 400, 101325
|
|
# ensure that correct triple point pressure is recalculated
|
|
# (necessary as this value is not stored by the C++ base class)
|
|
self.water.PQ = 101325, .2
|
|
self.assertNear(T, self.water.T, 1e-9)
|
|
with self.assertRaisesRegex(ct.CanteraError, 'below triple point'):
|
|
# min_temp is triple point temperature
|
|
self.water.TP = self.water.min_temp, 101325
|
|
P = self.water.P_sat # triple-point pressure
|
|
self.water.PQ = .999*P, .2
|
|
|
|
def test_quality_exceptions(self):
|
|
# Critical point
|
|
self.water.TP = 300, ct.one_atm
|
|
self.water.TQ = self.water.critical_temperature, .5
|
|
self.assertNear(self.water.P, self.water.critical_pressure)
|
|
self.water.TP = 300, ct.one_atm
|
|
self.water.PQ = self.water.critical_pressure, .5
|
|
self.assertNear(self.water.T, self.water.critical_temperature)
|
|
|
|
# Supercritical
|
|
with self.assertRaisesRegex(ct.CanteraError, 'supercritical'):
|
|
self.water.TQ = 1.001 * self.water.critical_temperature, 0.
|
|
with self.assertRaisesRegex(ct.CanteraError, 'supercritical'):
|
|
self.water.PQ = 1.001 * self.water.critical_pressure, 0.
|
|
|
|
# Q negative
|
|
with self.assertRaisesRegex(ct.CanteraError, 'Invalid vapor fraction'):
|
|
self.water.TQ = 373.15, -.001
|
|
with self.assertRaisesRegex(ct.CanteraError, 'Invalid vapor fraction'):
|
|
self.water.PQ = ct.one_atm, -.001
|
|
|
|
# Q larger than one
|
|
with self.assertRaisesRegex(ct.CanteraError, 'Invalid vapor fraction'):
|
|
self.water.TQ = 373.15, 1.001
|
|
with self.assertRaisesRegex(ct.CanteraError, 'Invalid vapor fraction'):
|
|
self.water.PQ = ct.one_atm, 1.001
|
|
|
|
def test_saturated_mixture(self):
|
|
self.water.TP = 300, ct.one_atm
|
|
with self.assertRaisesRegex(ct.CanteraError, 'Saturated mixture detected'):
|
|
self.water.TP = 300, self.water.P_sat
|
|
|
|
w = ct.Water()
|
|
|
|
# Saturated vapor
|
|
self.water.TQ = 373.15, 1.
|
|
self.assertEqual(self.water.phase_of_matter, 'liquid-gas-mix')
|
|
w.TP = self.water.T, .999 * self.water.P_sat
|
|
self.assertNear(self.water.cp, w.cp, 1.e-3)
|
|
self.assertNear(self.water.cv, w.cv, 1.e-3)
|
|
self.assertNear(self.water.thermal_expansion_coeff, w.thermal_expansion_coeff, 1.e-3)
|
|
self.assertNear(self.water.isothermal_compressibility, w.isothermal_compressibility, 1.e-3)
|
|
|
|
# Saturated mixture
|
|
self.water.TQ = 373.15, .5
|
|
self.assertEqual(self.water.phase_of_matter, 'liquid-gas-mix')
|
|
self.assertEqual(self.water.cp, np.inf)
|
|
self.assertTrue(np.isnan(self.water.cv))
|
|
self.assertEqual(self.water.isothermal_compressibility, np.inf)
|
|
self.assertEqual(self.water.thermal_expansion_coeff, np.inf)
|
|
|
|
# Saturated liquid
|
|
self.water.TQ = 373.15, 0.
|
|
self.assertEqual(self.water.phase_of_matter, 'liquid-gas-mix')
|
|
w.TP = self.water.T, 1.001 * self.water.P_sat
|
|
self.assertNear(self.water.cp, w.cp, 1.e-3)
|
|
self.assertNear(self.water.cv, w.cv, 1.e-3)
|
|
self.assertNear(self.water.thermal_expansion_coeff, w.thermal_expansion_coeff, 1.e-3)
|
|
self.assertNear(self.water.isothermal_compressibility, w.isothermal_compressibility, 1.e-3)
|
|
|
|
def test_saturation_near_limits(self):
|
|
# Low temperature limit (triple point)
|
|
self.water.TP = 300, ct.one_atm
|
|
self.water.P_sat # ensure that solver buffers sufficiently different values
|
|
self.water.TP = self.water.min_temp, ct.one_atm
|
|
psat = self.water.P_sat
|
|
self.water.TP = 300, ct.one_atm
|
|
self.water.P_sat # ensure that solver buffers sufficiently different values
|
|
self.water.TP = 300, psat
|
|
self.assertNear(self.water.T_sat, self.water.min_temp)
|
|
|
|
# High temperature limit (critical point) - saturation temperature
|
|
self.water.TP = 300, ct.one_atm
|
|
self.water.P_sat # ensure that solver buffers sufficiently different values
|
|
self.water.TP = self.water.critical_temperature, self.water.critical_pressure
|
|
self.assertNear(self.water.T_sat, self.water.critical_temperature)
|
|
|
|
# High temperature limit (critical point) - saturation pressure
|
|
self.water.TP = 300, ct.one_atm
|
|
self.water.P_sat # ensure that solver buffers sufficiently different values
|
|
self.water.TP = self.water.critical_temperature, self.water.critical_pressure
|
|
self.assertNear(self.water.P_sat, self.water.critical_pressure)
|
|
|
|
# Supercricital
|
|
with self.assertRaisesRegex(ct.CanteraError, 'Illegal temperature value'):
|
|
self.water.TP = 1.001 * self.water.critical_temperature, self.water.critical_pressure
|
|
self.water.P_sat
|
|
with self.assertRaisesRegex(ct.CanteraError, 'Illegal pressure value'):
|
|
self.water.TP = self.water.critical_temperature, 1.001 * self.water.critical_pressure
|
|
self.water.T_sat
|
|
|
|
# Below triple point
|
|
with self.assertRaisesRegex(ct.CanteraError, 'Illegal temperature'):
|
|
self.water.TP = .999 * self.water.min_temp, ct.one_atm
|
|
self.water.P_sat
|
|
# @todo: test disabled pending fix of GitHub issue #605
|
|
# with self.assertRaisesRegex(ct.CanteraError, 'Illegal pressure value'):
|
|
# self.water.TP = 300, .999 * psat
|
|
# self.water.T_sat
|
|
|
|
def test_TPQ(self):
|
|
self.water.TQ = 400, 0.8
|
|
T, P, Q = self.water.TPQ
|
|
self.assertNear(T, 400)
|
|
self.assertNear(Q, 0.8)
|
|
|
|
# a supercritical state
|
|
self.water.TPQ = 800, 3e7, 1
|
|
self.assertNear(self.water.T, 800)
|
|
self.assertNear(self.water.P, 3e7)
|
|
|
|
self.water.TPQ = T, P, Q
|
|
self.assertNear(self.water.Q, 0.8)
|
|
with self.assertRaisesRegex(ct.CanteraError, 'inconsistent'):
|
|
self.water.TPQ = T, .999*P, Q
|
|
with self.assertRaisesRegex(ct.CanteraError, 'inconsistent'):
|
|
self.water.TPQ = T, 1.001*P, Q
|
|
with self.assertRaises(TypeError):
|
|
self.water.TPQ = T, P, 'spam'
|
|
|
|
self.water.TPQ = 500, 1e5, 1 # superheated steam
|
|
self.assertNear(self.water.P, 1e5)
|
|
with self.assertRaisesRegex(ct.CanteraError, 'inconsistent'):
|
|
self.water.TPQ = 500, 1e5, 0 # vapor fraction should be 1 (T < Tc)
|
|
with self.assertRaisesRegex(ct.CanteraError, 'inconsistent'):
|
|
self.water.TPQ = 700, 1e5, 0 # vapor fraction should be 1 (T > Tc)
|
|
|
|
def test_phase_of_matter(self):
|
|
self.water.TP = 300, 101325
|
|
self.assertEqual(self.water.phase_of_matter, "liquid")
|
|
self.water.TP = 500, 101325
|
|
self.assertEqual(self.water.phase_of_matter, "gas")
|
|
self.water.TP = self.water.critical_temperature*2, 101325
|
|
self.assertEqual(self.water.phase_of_matter, "supercritical")
|
|
self.water.TP = 300, self.water.critical_pressure*2
|
|
self.assertEqual(self.water.phase_of_matter, "supercritical")
|
|
self.water.TQ = 300, 0.4
|
|
self.assertEqual(self.water.phase_of_matter, "liquid-gas-mix")
|
|
|
|
# These cases work after fixing GH-786
|
|
n2 = ct.Nitrogen()
|
|
n2.TP = 100, 1000
|
|
self.assertEqual(n2.phase_of_matter, "gas")
|
|
|
|
co2 = ct.CarbonDioxide()
|
|
self.assertEqual(co2.phase_of_matter, "gas")
|
|
|
|
def test_water_backends(self):
|
|
w = ct.Water(backend='Reynolds')
|
|
self.assertEqual(w.thermo_model, 'pure-fluid')
|
|
w = ct.Water(backend='IAPWS95')
|
|
self.assertEqual(w.thermo_model, 'liquid-water-IAPWS95')
|
|
with self.assertRaisesRegex(KeyError, 'Unknown backend'):
|
|
ct.Water('foobar')
|
|
|
|
def test_water_iapws(self):
|
|
w = ct.Water(backend='IAPWS95')
|
|
self.assertNear(w.critical_density, 322.)
|
|
self.assertNear(w.critical_temperature, 647.096)
|
|
self.assertNear(w.critical_pressure, 22064000.0)
|
|
|
|
# test internal TP setters (setters update temperature at constant
|
|
# density before updating pressure)
|
|
w.TP = 300, ct.one_atm
|
|
dens = w.density
|
|
w.TP = 2000, ct.one_atm # supercritical
|
|
self.assertEqual(w.phase_of_matter, "supercritical")
|
|
w.TP = 300, ct.one_atm # state goes from supercritical -> gas -> liquid
|
|
self.assertNear(w.density, dens)
|
|
self.assertEqual(w.phase_of_matter, "liquid")
|
|
|
|
# test setters for critical conditions
|
|
w.TP = w.critical_temperature, w.critical_pressure
|
|
self.assertNear(w.density, 322.)
|
|
w.TP = 2000, ct.one_atm # uses current density as initial guess
|
|
w.TP = 273.16, ct.one_atm # uses fixed density as initial guess
|
|
self.assertNear(w.density, 999.84376)
|
|
self.assertEqual(w.phase_of_matter, "liquid")
|
|
w.TP = w.T, w.P_sat
|
|
self.assertEqual(w.phase_of_matter, "liquid")
|
|
with self.assertRaisesRegex(ct.CanteraError, "assumes liquid phase"):
|
|
w.TP = 273.1599999, ct.one_atm
|
|
with self.assertRaisesRegex(ct.CanteraError, "assumes liquid phase"):
|
|
w.TP = 500, ct.one_atm
|
|
|
|
|
|
# To minimize errors when transcribing tabulated data, the input units here are:
|
|
# T: K, P: MPa, rho: kg/m3, v: m3/kg, (u,h): kJ/kg, s: kJ/kg-K
|
|
# Which are then converted to SI
|
|
class StateData:
|
|
def __init__(self, phase, T, p, rho=None, v=None, u=None, h=None, s=None, relax=False):
|
|
self.phase = phase
|
|
self.T = T
|
|
self.p = p * 1e6
|
|
self.rho = rho if rho else 1.0/v
|
|
self.u = 1e3 * u if u is not None else 1e3 * h - self.p/self.rho
|
|
self.s = 1e3 * s
|
|
self.tolMod = 10.0 if relax else 1.0
|
|
|
|
|
|
class Tolerances:
|
|
def __init__(self, p=None, u=None, s=None,
|
|
dUdS=None, dAdV=None, dPdT=None, hTs=None):
|
|
self.p = p or 2e-5
|
|
self.u = u or 2e-6
|
|
self.s = s or 2e-6
|
|
self.dUdS = dUdS or 2e-6
|
|
self.dAdV = dAdV or 2e-6
|
|
self.dPdT = dPdT or 2e-4
|
|
self.hTs = hTs or 2e-4
|
|
|
|
|
|
class PureFluidTestCases:
|
|
"""
|
|
Test the results of pure fluid phase calculations against tabulated
|
|
references and for consistency with basic thermodynamic relations.
|
|
"""
|
|
fluids = {}
|
|
|
|
def __init__(self, name, refState, tolerances=Tolerances()):
|
|
if name not in self.fluids:
|
|
self.fluids[name] = ct.PureFluid("liquidvapor.yaml", name)
|
|
|
|
self.fluid = self.fluids[name]
|
|
|
|
self.fluid.TD = refState.T, refState.rho
|
|
self.refState = refState
|
|
self.u0 = self.fluid.u
|
|
self.s0 = self.fluid.s
|
|
self.tol = tolerances
|
|
|
|
def a(self, T, rho):
|
|
""" Helmholtz free energy """
|
|
self.fluid.TD = T, rho
|
|
return self.fluid.u - T * self.fluid.s
|
|
|
|
def test_has_phase_transition(self):
|
|
self.assertTrue(self.fluid.has_phase_transition)
|
|
|
|
def test_consistency_temperature(self):
|
|
for state in self.states:
|
|
dT = 2e-5 * state.T
|
|
self.fluid.TD = state.T-dT, state.rho
|
|
s1 = self.fluid.s
|
|
u1 = self.fluid.u
|
|
self.fluid.TD = state.T+dT, state.rho
|
|
s2 = self.fluid.s
|
|
u2 = self.fluid.u
|
|
|
|
# At constant volume, dU = T dS
|
|
msg = 'At state: T=%s, rho=%s' % (state.T, state.rho)
|
|
self.assertNear((u2-u1)/(s2-s1), state.T, self.tol.dUdS, msg=msg)
|
|
|
|
def test_consistency_volume(self):
|
|
for state in self.states:
|
|
self.fluid.TD = state.T, state.rho
|
|
p = self.fluid.P
|
|
V = 1 / state.rho
|
|
dV = 5e-6 * V
|
|
|
|
a1 = self.a(state.T, 1/(V-0.5*dV))
|
|
a2 = self.a(state.T, 1/(V+0.5*dV))
|
|
|
|
# dP/drho is high for liquids, so relax tolerances
|
|
tol = 300*self.tol.dAdV if state.phase == 'liquid' else self.tol.dAdV
|
|
|
|
# At constant temperature, dA = - p dV
|
|
msg = 'At state: T=%s, rho=%s' % (state.T, state.rho)
|
|
self.assertNear(-(a2-a1)/dV, p, tol, msg=msg)
|
|
|
|
def test_saturation(self):
|
|
for state in self.states:
|
|
if state.phase == 'super':
|
|
continue
|
|
|
|
dT = 1e-6 * state.T
|
|
self.fluid.TQ = state.T, 0
|
|
p1 = self.fluid.P
|
|
vf = 1.0 / self.fluid.density
|
|
hf = self.fluid.h
|
|
sf = self.fluid.s
|
|
|
|
self.fluid.TQ = state.T + dT, 0
|
|
p2 = self.fluid.P
|
|
|
|
self.fluid.TQ = state.T, 1
|
|
vg = 1.0 / self.fluid.density
|
|
hg = self.fluid.h
|
|
sg = self.fluid.s
|
|
|
|
# Clausius-Clapeyron Relation
|
|
msg = 'At state: T=%s, rho=%s' % (state.T, state.rho)
|
|
self.assertNear((p2-p1)/dT, (hg-hf)/(state.T * (vg-vf)),
|
|
self.tol.dPdT, msg=msg)
|
|
|
|
# True for a change in state at constant pressure and temperature
|
|
self.assertNear(hg-hf, state.T * (sg-sf), self.tol.hTs, msg=msg)
|
|
|
|
def test_pressure(self):
|
|
for state in self.states:
|
|
self.fluid.TD = state.T, state.rho
|
|
# dP/drho is high for liquids, so relax tolerances
|
|
tol = 50*self.tol.p if state.phase == 'liquid' else self.tol.p
|
|
tol *= state.tolMod
|
|
msg = 'At state: T=%s, rho=%s' % (state.T, state.rho)
|
|
self.assertNear(self.fluid.P, state.p, tol, msg=msg)
|
|
|
|
def test_internal_energy(self):
|
|
for state in self.states:
|
|
self.fluid.TD = state.T, state.rho
|
|
msg = 'At state: T=%s, rho=%s' % (state.T, state.rho)
|
|
self.assertNear(self.fluid.u - self.u0,
|
|
state.u - self.refState.u,
|
|
self.tol.u * state.tolMod, msg=msg)
|
|
|
|
def test_entropy(self):
|
|
for state in self.states:
|
|
self.fluid.TD = state.T, state.rho
|
|
msg = 'At state: T=%s, rho=%s' % (state.T, state.rho)
|
|
self.assertNear(self.fluid.s - self.s0,
|
|
state.s - self.refState.s,
|
|
self.tol.s * state.tolMod, msg=msg)
|
|
|
|
|
|
# Reference values for HFC134a taken from NIST Chemistry WebBook, which
|
|
# implements the same EOS from Tillner-Roth and Baehr as Cantera, so close
|
|
# agreement is expected.
|
|
class HFC134a(PureFluidTestCases, utilities.CanteraTest):
|
|
states = [
|
|
StateData('liquid', 175.0, 0.1, rho=1577.6239, u=77.534586, s=0.44788182),
|
|
StateData('liquid', 210.0, 0.1, rho=1483.2128, u=119.48566, s=0.66633877),
|
|
StateData('vapor', 250.0, 0.1, rho=5.1144317, u=365.59424, s=1.7577491),
|
|
StateData('vapor', 370.0, 0.1, rho=3.3472612, u=459.82664, s=2.0970769),
|
|
StateData('liquid', 290.0, 10, rho=1278.4700, u=216.99119, s=1.0613409),
|
|
StateData('super', 410.0, 10, rho=736.54666, u=399.02258, s=1.5972395),
|
|
StateData('super', 450.0, 40, rho=999.34087, u=411.92422, s=1.6108568)]
|
|
|
|
def __init__(self, *args, **kwargs):
|
|
refState = StateData('critical', 374.21, 4.05928,
|
|
rho=511.900, u=381.70937, s=1.5620991)
|
|
PureFluidTestCases.__init__(self, "HFC-134a", refState)
|
|
utilities.CanteraTest.__init__(self, *args, **kwargs)
|
|
|
|
|
|
# Reference values for the following substances are taken from the tables in
|
|
# W.C. Reynolds, "Thermodynamic Properties in SI", which is the source of
|
|
# Cantera's equations of state for these substances. Agreement is limited by
|
|
# the precision of the results printed in the book (typically 4 significant
|
|
# figures).
|
|
|
|
# Property comparisons for saturated states are further limited by the use of
|
|
# different methods for satisfying the phase equilibrium condition g_l = g_v.
|
|
# Cantera uses the actual equation of state, while the tabulated values given
|
|
# by Reynolds are based on the given P_sat(T_sat) relations.
|
|
class CarbonDioxide(PureFluidTestCases, utilities.CanteraTest):
|
|
states = [
|
|
StateData('liquid', 230.0, 2.0, rho=1132.4, h=28.25, s=0.1208),
|
|
StateData('liquid', 270.0, 10.0, rho=989.97, h=110.59, s=0.4208),
|
|
StateData('vapor', 250.0, 1.788, v=0.02140, h=358.59, s=1.4500, relax=True), # sat
|
|
StateData('vapor', 300.0, 2.0, v=0.02535, h=409.41, s=1.6174),
|
|
StateData('super', 500.0, 1.0, v=0.09376, h=613.22, s=2.2649),
|
|
StateData('super', 600.0, 20.0, v=0.00554, h=681.94, s=1.8366)]
|
|
|
|
def __init__(self, *args, **kwargs):
|
|
refState = StateData('critical', 304.21, 7.3834,
|
|
rho=464.0, h=257.31, s=0.9312)
|
|
tols = Tolerances(2e-3, 2e-3, 2e-3)
|
|
PureFluidTestCases.__init__(self, "carbon-dioxide", refState, tols)
|
|
utilities.CanteraTest.__init__(self, *args, **kwargs)
|
|
|
|
|
|
class Heptane(PureFluidTestCases, utilities.CanteraTest):
|
|
states = [
|
|
StateData('liquid', 300.0, 0.006637, v=0.001476, h=0.0, s=0.0, relax=True), # sat
|
|
StateData('liquid', 400.0, 0.2175, v=0.001712, h=248.01, s=0.709, relax=True), # sat
|
|
StateData('vapor', 490.0, 1.282, v=0.02222, h=715.64, s=1.7137, relax=True), # sat
|
|
StateData('vapor', 480.0, 0.70, v=0.04820, h=713.04, s=1.7477),
|
|
StateData('super', 600.0, 2.0, v=0.01992, h=1014.87, s=2.2356),
|
|
StateData('super', 680.0, 0.2, v=0.2790, h=1289.29, s=2.8450)]
|
|
|
|
def __init__(self, *args, **kwargs):
|
|
refState = StateData('critical', 537.68, 2.6199,
|
|
rho=197.60, h=747.84, s=1.7456)
|
|
tols = Tolerances(2e-3, 2e-3, 2e-3)
|
|
PureFluidTestCases.__init__(self, 'heptane', refState, tols)
|
|
utilities.CanteraTest.__init__(self, *args, **kwargs)
|
|
|
|
|
|
# para-hydrogen
|
|
class Hydrogen(PureFluidTestCases, utilities.CanteraTest):
|
|
states = [
|
|
StateData('liquid', 18.0, 0.04807, v=0.013660, h=30.1, s=1.856, relax=True), # sat
|
|
StateData('liquid', 26.0, 0.4029, v=0.015911, h=121.2, s=5.740, relax=True), # sat
|
|
StateData('vapor', 30.0, 0.8214, v=0.09207, h=487.4, s=17.859, relax=True), # sat
|
|
StateData('super', 100.0, 0.20, v=2.061, h=1398.3, s=39.869),
|
|
StateData('super', 200.0, 20.0, v=0.04795, h=3015.9, s=31.274),
|
|
StateData('super', 300.0, 0.50, v=2.482, h=4511.6, s=53.143),
|
|
StateData('super', 600.0, 1.00, v=2.483, h=8888.4, s=60.398),
|
|
StateData('super', 800.0, 4.0, v=0.8329, h=11840.0, s=58.890)]
|
|
|
|
def __init__(self, *args, **kwargs):
|
|
refState = StateData('critical', 32.938, 1.2838,
|
|
rho=31.36, h=346.5, s=12.536)
|
|
tols = Tolerances(2e-3, 2e-3, 2e-3, 2e-4)
|
|
PureFluidTestCases.__init__(self, 'hydrogen', refState, tols)
|
|
utilities.CanteraTest.__init__(self, *args, **kwargs)
|
|
|
|
|
|
class Methane(PureFluidTestCases, utilities.CanteraTest):
|
|
states = [
|
|
StateData('liquid', 100.0, 0.50, rho=439.39, h=31.65, s=0.3206),
|
|
StateData('liquid', 140.0, 2.0, rho=379.51, h=175.48, s=1.4963),
|
|
StateData('vapor', 150.0, 0.20, v=0.3772, h=660.72, s=5.5435),
|
|
StateData('vapor', 160.0, 1.594, v=0.03932, h=627.96, s=4.3648, relax=True), # sat
|
|
StateData('vapor', 175.0, 1.0, v=0.08157, h=692.55, s=4.9558),
|
|
StateData('super', 200.0, 0.2, v=0.5117, h=767.37, s=6.1574),
|
|
StateData('super', 300.0, 0.5, v=0.3083, h=980.87, s=6.5513)]
|
|
|
|
def __init__(self, *args, **kwargs):
|
|
refState = StateData('critical', 190.555, 4.5988,
|
|
rho=160.43, h=490.61, s=3.2853)
|
|
tols = Tolerances(2e-3, 2e-3, 2e-3)
|
|
PureFluidTestCases.__init__(self, 'methane', refState, tols)
|
|
utilities.CanteraTest.__init__(self, *args, **kwargs)
|
|
|
|
|
|
class Nitrogen(PureFluidTestCases, utilities.CanteraTest):
|
|
states = [
|
|
StateData('liquid', 80.0, 0.1370, v=0.001256, h=33.50, s=0.4668, relax=True), # sat
|
|
StateData('vapor', 110.0, 1.467, v=0.01602, h=236.28, s=2.3896, relax=True), # sat
|
|
StateData('super', 200.0, 0.5, v=0.1174, h=355.05, s=3.5019),
|
|
StateData('super', 300.0, 10.0, v=0.00895, h=441.78, s=2.9797),
|
|
StateData('super', 500.0, 5.0, v=0.03031, h=668.48, s=3.7722),
|
|
StateData('super', 600.0, 100.0, v=0.00276, h=827.54, s=3.0208)]
|
|
|
|
def __init__(self, *args, **kwargs):
|
|
refState = StateData('critical', 126.200, 3.400,
|
|
rho=314.03, h=180.78, s=1.7903)
|
|
tols = Tolerances(2e-3, 2e-3, 2e-3)
|
|
PureFluidTestCases.__init__(self, 'nitrogen', refState, tols)
|
|
utilities.CanteraTest.__init__(self, *args, **kwargs)
|
|
|
|
|
|
class Oxygen(PureFluidTestCases, utilities.CanteraTest):
|
|
states = [
|
|
StateData('liquid', 80.0, 0.03009, v=0.000840, h=42.56, s=0.6405, relax=True), # sat
|
|
StateData('liquid', 125.0, 1.351, v=0.001064, h=123.24, s=1.4236, relax=True), # sat
|
|
StateData('vapor', 145.0, 3.448, v=0.006458, h=276.45, s=2.4852, relax=True), # sat
|
|
StateData('super', 200.0, 0.050, v=1.038, h=374.65, s=4.1275),
|
|
StateData('super', 300.0, 1.0, v=0.07749, h=463.76, s=3.7135),
|
|
StateData('super', 600.0, 0.20, v=0.7798, h=753.38, s=4.7982),
|
|
StateData('super', 800.0, 5.0, v=0.04204, h=961.00, s=4.2571)
|
|
]
|
|
|
|
def __init__(self, *args, **kwargs):
|
|
refState = StateData('critical', 154.581, 5.0429,
|
|
rho=436.15, h=226.53, s=2.1080)
|
|
tols = Tolerances(2e-3, 2e-3, 2e-3)
|
|
PureFluidTestCases.__init__(self, 'oxygen', refState, tols)
|
|
utilities.CanteraTest.__init__(self, *args, **kwargs)
|
|
|
|
|
|
class Water(PureFluidTestCases, utilities.CanteraTest):
|
|
states = [
|
|
StateData('liquid', 295.0, 0.002620, v=0.0010025, h=90.7, s=0.3193, relax=True),
|
|
StateData('vapor', 315.0, 0.008143, v=17.80, h=2577.1, s=8.2216, relax=True),
|
|
StateData('liquid', 440.0, 0.7332, v=0.001110, h=705.0, s=2.0096, relax=True),
|
|
StateData('vapor', 510.0, 3.163, v=0.06323, h=2803.6, s=6.1652, relax=True),
|
|
StateData('vapor', 400.0, 0.004, v=46.13, h=2738.8, s=9.0035),
|
|
StateData('vapor', 500.0, 1.0, v=0.2206, h=2890.2, s=6.8223),
|
|
StateData('super', 800.0, 0.01, v=36.92, h=3546.0, s=9.9699),
|
|
StateData('super', 900.0, 0.70, v=0.5917, h=3759.4, s=8.2621),
|
|
StateData('super', 1000.0, 30.0, v=0.01421, h=3821.6, s=6.6373),
|
|
StateData('liquid', 500.0, 3.0, rho=832.04, h=975.68, s=2.58049)
|
|
]
|
|
|
|
def __init__(self, *args, **kwargs):
|
|
refState = StateData('critical', 647.286, 22.089,
|
|
rho=317.0, h=2098.8, s=4.4289)
|
|
tols = Tolerances(2e-3, 2e-3, 2e-3)
|
|
PureFluidTestCases.__init__(self, 'water', refState, tols)
|
|
utilities.CanteraTest.__init__(self, *args, **kwargs)
|
|
|
|
|
|
class PureFluidConvergence(utilities.CanteraTest):
|
|
def setUp(self):
|
|
self.fluid = ct.Water()
|
|
|
|
def test_TP(self):
|
|
# Focus on the region near the critical point
|
|
TT = [273.161, 300.0, 350.0, 400.0, 500.0,
|
|
600.0, 640.0, 645.0, 646.0, 647.0,
|
|
647.1, 647.2, 647.22, 647.23, 647.25,
|
|
647.26, 647.27, 647.28, 647.282, 647.284,
|
|
647.285, 647.286, 647.287, 650.0, 800.0]
|
|
PP = [1234.0, 101325.0, 5e5, 22.0e6, 22.08e6, 22.09e6, 10001000.0]
|
|
|
|
errors = ''
|
|
nErrors = 0
|
|
for T,P in itertools.product(TT, PP):
|
|
try:
|
|
self.fluid.TP = T, P
|
|
self.assertNear(self.fluid.T, T, 1e-6)
|
|
self.assertNear(self.fluid.P, P, 1e-6)
|
|
except Exception as e:
|
|
errors += 'Error at T=%r, P=%r:\n%s\n\n' % (T,P,e)
|
|
nErrors += 1
|
|
if errors:
|
|
errors += 'Total error count:%s\n' % nErrors
|
|
raise AssertionError(errors)
|
|
|
|
def test_UV(self):
|
|
u0 = -1.58581e7
|
|
UU = np.array([0, 100, 200, 500, 1000, 1500, 2000]) * 1000 + u0
|
|
VV = [0.001, 0.002, 0.005, 0.010, 0.10, 0.5, 1.0, 1.5, 2.0]
|
|
errors = ''
|
|
nErrors = 0
|
|
for u,v in itertools.product(UU, VV):
|
|
try:
|
|
self.fluid.UV = u, v
|
|
self.assertNear(self.fluid.u, u, 1e-6)
|
|
self.assertNear(self.fluid.v, v, 1e-6)
|
|
except Exception as e:
|
|
errors += 'Error at u=%r, v=%r:\n%s\n\n' % (u,v,e)
|
|
nErrors += 1
|
|
if errors:
|
|
errors += 'Total error count:%s\n' % nErrors
|
|
raise AssertionError(errors)
|
|
|
|
def test_HP(self):
|
|
h0 = -1.58581e7
|
|
HH = np.array([0, 100, 200, 500, 1000, 1500, 2000]) * 1000 + h0
|
|
PP = [1234.0, 101325.0, 5e5, 22.0e6, 22.08e6, 22.09e6, 10001000.0]
|
|
errors = ''
|
|
nErrors = 0
|
|
for h,P in itertools.product(HH, PP):
|
|
try:
|
|
self.fluid.HP = h, P
|
|
self.assertNear(self.fluid.h, h, 1e-6)
|
|
self.assertNear(self.fluid.P, P, 1e-6)
|
|
except Exception as e:
|
|
errors += 'Error at h=%r, P=%r:\n%s\n\n' % (h,P,e)
|
|
nErrors += 1
|
|
if errors:
|
|
errors += 'Total error count:%s\n' % nErrors
|
|
raise AssertionError(errors)
|