#include "AbstractState.h" #include "DataStructures.h" // ############################################ // TESTS // ############################################ #if defined(ENABLE_CATCH) #include "catch.hpp" namespace ViscosityValidation{ // A structure to hold the values for one validation call for viscosity struct vel { public: std::string in1, in2, out, fluid; double v1, v2, tol, expected; vel(std::string fluid, std::string in1, double v1, std::string in2, double v2, std::string out, double expected, double tol) { this->in1 = in1; this->in2 = in2; this->fluid = fluid; this->v1 = v1; this->v2 = v2; this->expected = expected; this->tol = tol; }; }; vel viscosity_validation_data[] = { // From Vogel, JPCRD, 1998 vel("Propane", "T", 90, "Dmolar", 16.52e3, "V", 7388e-6, 1e-3), vel("Propane", "T", 150, "Dmolar", 15.14e3, "V", 656.9e-6, 5e-3), vel("Propane", "T", 600, "Dmolar", 10.03e3, "V", 73.92e-6, 5e-3), vel("Propane", "T", 280, "Dmolar", 11.78e3, "V", 117.4e-6,1e-3), // Huber, FPE, 2004 vel("n-Octane", "T", 300, "Dmolar", 6177.2, "V", 553.60e-6, 1e-3), vel("n-Nonane", "T", 300, "Dmolar", 5619.1, "V", 709.53e-6, 1e-3), vel("n-Decane", "T", 300, "Dmolar", 5150.4, "V", 926.44e-6, 1e-3), // Huber, Energy & Fuels, 2004 vel("n-Dodecane", "T", 300, "Dmolar", 4411.5, "V", 1484.8e-6, 1e-3), vel("n-Dodecane", "T", 500, "Dmolar", 3444.7, "V", 183.76e-6, 1e-3), // Huber, I&ECR, 2006 vel("R125", "T", 300, "Dmolar", 10596.9998, "V", 177.37e-6, 1e-3), vel("R125", "T", 400, "Dmolar", 30.631, "V", 17.070e-6, 1e-3), // From REFPROP 9.1 since Huber I&ECR 2003 does not provide validation data vel("R134a", "T", 185, "Q", 0, "V", 0.0012698376398294414, 1e-3), vel("R134a", "T", 185, "Q", 1, "V", 7.4290821400170869e-006, 1e-3), vel("R134a", "T", 360, "Q", 0, "V", 7.8146319978982133e-005, 1e-3), vel("R134a", "T", 360, "Q", 1, "V", 1.7140264998576107e-005, 1e-3), // From REFPROP 9.1 since Kiselev, IECR, 2005 does not provide validation data vel("Ethanol", "T", 300, "Q", 0, "V", 0.0010439017679191723, 1e-3), vel("Ethanol", "T", 300, "Q", 1, "V", 8.8293820936046416e-006, 1e-3), vel("Ethanol", "T", 500, "Q", 0, "V", 6.0979347125450671e-005, 1e-3), vel("Ethanol", "T", 500, "Q", 1, "V", 1.7229157141572511e-005, 1e-3), // From CoolProp v5 implementation of correlation - more or less agrees with REFPROP // Errata in BibTeX File vel("Hydrogen", "T", 35, "Dmass", 100, "V", 5.6658815444656859e-005, 1e-3), // From Meng 2012 experimental data (note erratum in BibTeX file) vel("DimethylEther", "T", 253.146, "Dmass", 734.28, "V", 0.20444e-3, 3e-3), vel("DimethylEther", "T", 373.132, "Dmass", 613.78, "V", 0.09991e-3, 3e-3), //// From Fenghour, JPCRD, 1995 //vel("Ammonia", "T", 200, "Dmolar", 3.9, "V", 6.95e-6, 1e-3), //vel("Ammonia", "T", 200, "Dmolar", 42754.4, "V", 507.28e-6, 1e-3), //vel("Ammonia", "T", 398, "Dmolar", 7044.7, "V", 17.67e-6, 1e-3), //vel("Ammonia", "T", 398, "Dmolar", 21066.7, "V", 43.95e-6, 1e-3), //// From Lemmon and Jacobsen, JPCRD, 2004 vel("Nitrogen", "T", 100, "Dmolar", 1e-14, "V", 6.90349e-6, 1e-3), vel("Nitrogen", "T", 300, "Dmolar", 1e-14, "V", 17.8771e-6, 1e-3), vel("Nitrogen", "T", 100, "Dmolar", 25000, "V", 79.7418e-6, 1e-3), vel("Nitrogen", "T", 200, "Dmolar", 10000, "V", 21.0810e-6, 1e-3), vel("Nitrogen", "T", 300, "Dmolar", 5000, "V", 20.7430e-6, 1e-3), vel("Nitrogen", "T", 126.195, "Dmolar", 11180, "V", 18.2978e-6, 1e-3), vel("Argon", "T", 100, "Dmolar", 1e-14, "V", 8.18940e-6, 1e-3), vel("Argon", "T", 300, "Dmolar", 1e-14, "V", 22.7241e-6, 1e-3), vel("Argon", "T", 100, "Dmolar", 33000, "V", 184.232e-6, 1e-3), vel("Argon", "T", 200, "Dmolar", 10000, "V", 25.5662e-6, 1e-3), vel("Argon", "T", 300, "Dmolar", 5000, "V", 26.3706e-6, 1e-3), vel("Argon", "T", 150.69, "Dmolar", 13400, "V", 27.6101e-6, 1e-3), vel("Oxygen", "T", 100, "Dmolar", 1e-14, "V", 7.70243e-6, 1e-3), vel("Oxygen", "T", 300, "Dmolar", 1e-14, "V", 20.6307e-6, 1e-3), vel("Oxygen", "T", 100, "Dmolar", 35000, "V", 172.136e-6, 1e-3), vel("Oxygen", "T", 200, "Dmolar", 10000, "V", 22.4445e-6, 1e-3), vel("Oxygen", "T", 300, "Dmolar", 5000, "V", 23.7577e-6, 1e-3), vel("Oxygen", "T", 154.6, "Dmolar", 13600, "V", 24.7898e-6, 1e-3), vel("Air", "T", 100, "Dmolar", 1e-14, "V", 7.09559e-6, 1e-3), vel("Air", "T", 300, "Dmolar", 1e-14, "V", 18.5230e-6, 1e-3), vel("Air", "T", 100, "Dmolar", 28000, "V", 107.923e-6, 1e-3), vel("Air", "T", 200, "Dmolar", 10000, "V", 21.1392e-6, 1e-3), vel("Air", "T", 300, "Dmolar", 5000, "V", 21.3241e-6, 1e-3), vel("Air", "T", 132.64, "Dmolar", 10400, "V", 17.7623e-6, 1e-3), // From Michailidou, JPCRD, 2013 vel("Hexane", "T", 250, "Dmass", 1e-14, "V", 5.2584e-6, 1e-3), vel("Hexane", "T", 400, "Dmass", 1e-14, "V", 8.4149e-6, 1e-3), vel("Hexane", "T", 550, "Dmass", 1e-14, "V", 11.442e-6, 1e-3), vel("Hexane", "T", 250, "Dmass", 700, "V", 528.2e-6, 1e-3), vel("Hexane", "T", 400, "Dmass", 600, "V", 177.62e-6, 1e-3), vel("Hexane", "T", 550, "Dmass", 500, "V", 95.002e-6, 1e-3), // From Fenghour, JPCRD, 1998 vel("CO2", "T", 220, "Dmass", 2.440, "V", 11.06e-6, 1e-3), vel("CO2", "T", 300, "Dmass", 1.773, "V", 15.02e-6, 1e-3), vel("CO2", "T", 800, "Dmass", 0.662, "V", 35.09e-6, 1e-3), vel("CO2", "T", 304, "Dmass", 254.3205, "V", 20.99e-6, 1e-3), vel("CO2", "T", 220, "Dmass", 1194.86, "V", 269.37e-6, 1e-3), vel("CO2", "T", 300, "Dmass", 1029.27, "V", 132.55e-6, 1e-3), vel("CO2", "T", 800, "Dmass", 407.828, "V", 48.74e-6, 1e-3), // Tanaka, IJT, 1996 vel("R123", "T", 265, "Dmass", 1545.8, "V", 627.1e-6, 1e-3), vel("R123", "T", 265, "Dmass", 1.614, "V", 9.534e-6, 1e-3), vel("R123", "T", 415, "Dmass", 1079.4, "V", 121.3e-6, 1e-3), vel("R123", "T", 415, "Dmass", 118.9, "V", 15.82e-6, 1e-3), // // Krauss, IJT, 1996 vel("R152A", "T", 242, "Dmass", 1025.5, "V", 347.3e-6, 1e-3), vel("R152A", "T", 242, "Dmass", 2.4868, "V", 8.174e-6, 1e-3), vel("R152A", "T", 384, "Dmass", 504.51, "V", 43.29e-6, 5e-3), vel("R152A", "T", 384, "Dmass", 239.35, "V", 21.01e-6, 10e-3), // Huber, JPCRD, 2008 and IAPWS vel("Water", "T", 298.15, "Dmass", 998, "V", 889.735100e-6, 1e-3), vel("Water", "T", 298.15, "Dmass", 1200, "V", 1437.649467e-6, 1e-3), vel("Water", "T", 373.15, "Dmass", 1000, "V", 307.883622e-6, 1e-3), vel("Water", "T", 433.15, "Dmass", 1, "V", 14.538324e-6, 1e-3), vel("Water", "T", 433.15, "Dmass", 1000, "V", 217.685358e-6, 1e-3), vel("Water", "T", 873.15, "Dmass", 1, "V", 32.619287e-6, 1e-3), vel("Water", "T", 873.15, "Dmass", 100, "V", 35.802262e-6, 1e-3), vel("Water", "T", 873.15, "Dmass", 600, "V", 77.430195e-6, 1e-3), vel("Water", "T", 1173.15, "Dmass", 1, "V", 44.217245e-6, 1e-3), vel("Water", "T", 1173.15, "Dmass", 100, "V", 47.640433e-6, 1e-3), vel("Water", "T", 1173.15, "Dmass", 400, "V", 64.154608e-6, 1e-3), // vel("SF6", "T", 300, "Dmass", 1e-14, "V", 15.2887e-6, 1e-3), vel("SF6", "T", 300, "Dmass", 5.92, "V", 15.3043e-6, 1e-3), vel("SF6", "T", 300, "Dmass", 1345.1, "V", 117.417e-6, 1e-3), vel("SF6", "T", 400, "Dmass", 1e-14, "V", 19.6796e-6, 1e-3), vel("SF6", "T", 400, "Dmass", 278.47, "V", 24.4272e-6, 1e-3), vel("SF6", "T", 400, "Dmass", 1123.8, "V", 84.7835e-6, 1e-3), }; class ViscosityValidationFixture { protected: long double actual, x1, x2; CoolProp::AbstractState *pState; int pair; public: ViscosityValidationFixture(){ pState = NULL; } ~ViscosityValidationFixture(){ delete pState; } void set_backend(std::string backend, std::string fluid_name){ pState = CoolProp::AbstractState::factory(backend, fluid_name); } void set_pair(std::string &in1, double v1, std::string &in2, double v2){ double o1, o2; long iin1 = CoolProp::get_parameter_index(in1); long iin2 = CoolProp::get_parameter_index(in2); long pair = CoolProp::generate_update_pair(iin1, v1, iin2, v2, o1, o2); pState->update(pair, o1, o2); } void get_value() { actual = pState->viscosity(); } }; TEST_CASE_METHOD(ViscosityValidationFixture, "Compare viscosities against published data", "[viscosity]") { int inputsN = sizeof(viscosity_validation_data)/sizeof(viscosity_validation_data[0]); for (int i = 0; i < inputsN; ++i) { vel el = viscosity_validation_data[i]; CHECK_NOTHROW(set_backend("HEOS", el.fluid)); CAPTURE(el.fluid); CAPTURE(el.in1); CAPTURE(el.v1); CAPTURE(el.in2); CAPTURE(el.v2); CHECK_NOTHROW(set_pair(el.in1, el.v1, el.in2, el.v2)); get_value(); CAPTURE(el.expected); CAPTURE(actual); CHECK(fabs(actual/el.expected-1) < el.tol); } } }; /* namespace ViscosityValidation */ static int inputs[] = { CoolProp::DmolarT_INPUTS, //CoolProp::SmolarT_INPUTS, //CoolProp::HmolarT_INPUTS, //CoolProp::TUmolar_INPUTS, CoolProp::DmolarP_INPUTS, CoolProp::DmolarHmolar_INPUTS, CoolProp::DmolarSmolar_INPUTS, CoolProp::DmolarUmolar_INPUTS, /* CoolProp::HmolarP_INPUTS, CoolProp::PSmolar_INPUTS, CoolProp::PUmolar_INPUTS, */ /* CoolProp::HmolarSmolar_INPUTS, CoolProp::HmolarUmolar_INPUTS, CoolProp::SmolarUmolar_INPUTS */ }; class ConsistencyFixture { protected: long double hmolar, pmolar, smolar, umolar, rhomolar, T, p, x1, x2; CoolProp::AbstractState *pState; int pair; public: ConsistencyFixture(){ pState = NULL; } ~ConsistencyFixture(){ delete pState; } void set_backend(std::string backend, std::string fluid_name){ pState = CoolProp::AbstractState::factory(backend, fluid_name); } void set_pair(int pair){ this->pair = pair; } void set_TP(long double T, long double p) { this->T = T; this->p = p; CoolProp::AbstractState &State = *pState; // Start with T,P as inputs, cycle through all the other pairs that are supported State.update(CoolProp::PT_INPUTS, p, T); // Set the other state variables rhomolar = State.rhomolar(); hmolar = State.hmolar(); smolar = State.smolar(); umolar = State.umolar(); } void get_variables() { CoolProp::AbstractState &State = *pState; switch (pair) { /// In this group, T is one of the known inputs, iterate for the other one (easy) case CoolProp::HmolarT_INPUTS: x1 = hmolar; x2 = T; break; case CoolProp::SmolarT_INPUTS: x1 = smolar; x2 = T; break; case CoolProp::TUmolar_INPUTS: x1 = T; x2 = umolar; break; case CoolProp::DmolarT_INPUTS: x1 = rhomolar; x2 = T; break; /// In this group, D is one of the known inputs, iterate for the other one (a little bit harder) case CoolProp::DmolarHmolar_INPUTS: x1 = rhomolar; x2 = hmolar; break; case CoolProp::DmolarSmolar_INPUTS: x1 = rhomolar; x2 = smolar; break; case CoolProp::DmolarUmolar_INPUTS: x1 = rhomolar; x2 = umolar; break; case CoolProp::DmolarP_INPUTS: x1 = rhomolar; x2 = p; break; /// In this group, p is one of the known inputs (a little less easy) case CoolProp::HmolarP_INPUTS: x1 = hmolar; x2 = p; break; case CoolProp::PSmolar_INPUTS: x1 = p; x2 = smolar; break; case CoolProp::PUmolar_INPUTS: x1 = p; x2 = umolar; break; case CoolProp::HmolarSmolar_INPUTS: x1 = hmolar; x2 = smolar; break; case CoolProp::SmolarUmolar_INPUTS: x1 = smolar; x2 = umolar; break; } } void single_phase_consistency_check() { CoolProp::AbstractState &State = *pState; State.update(pair, x1, x2); // Make sure we end up back at the same temperature and pressure we started out with if(fabs(T-State.T()) > 1e-2) throw CoolProp::ValueError(format("Error on T [%g K] is greater than 1e-2",fabs(State.T()-T))); if(fabs(p-State.p())/p*100 > 1e-2) throw CoolProp::ValueError(format("Error on p [%g %%] is greater than 1e-2 %%",fabs(p-State.p())/p )); } }; TEST_CASE_METHOD(ConsistencyFixture, "Test all input pairs for CO2 using all valid backends", "[]") { CHECK_NOTHROW(set_backend("HEOS", "CO2")); int inputsN = sizeof(inputs)/sizeof(inputs[0]); for (double p = 600000; p < 800000000.0; p *= 5) { for (double T = 220; T < pState->Tmax(); T += 5) { CHECK_NOTHROW(set_TP(T, p)); for (int i = 0; i < inputsN; ++i) { int pair = inputs[i]; std::string pair_desc = CoolProp::get_input_pair_short_desc(pair); set_pair(pair); CAPTURE(pair_desc); CAPTURE(T); CAPTURE(p); get_variables(); CAPTURE(x1); CAPTURE(x2); CHECK_NOTHROW(single_phase_consistency_check()); } } } } #endif