CoolProp 6.8.1dev
An open-source fluid property and humid air property database
Configuration.h
Go to the documentation of this file.
1#ifndef COOLPROP_CONFIGURATION
2#define COOLPROP_CONFIGURATION
3
4#include "Exceptions.h"
5#include "CoolPropTools.h"
6#include <cstdlib>
7#include <unordered_map>
8
9#if !defined(SWIG) // Hide this for swig - Swig gets confused
10# include "rapidjson_include.h"
11#endif
12
13/* See http://stackoverflow.com/a/148610
14 * See http://stackoverflow.com/questions/147267/easy-way-to-use-variables-of-enum-types-as-string-in-c#202511
15 * This will be used to generate an enum like:
16 * enum configuration_keys {NORMALIZE_GAS_CONSTANTS, CRITICAL_SPLINES_ENABLED};
17 *
18 * The values in this list are given by:
19 * enum, string representation of enum, default value, description
20 *
21 * The type of the default value specifies the only type that will be accepted for this parameter
22 */
23#define CONFIGURATION_KEYS_ENUM \
24 X(NORMALIZE_GAS_CONSTANTS, "NORMALIZE_GAS_CONSTANTS", true, "If true, for mixtures, the molar gas constant (R) will be set to the CODATA value") \
25 X(CRITICAL_WITHIN_1UK, "CRITICAL_WITHIN_1UK", true, \
26 "If true, any temperature within 1 uK of the critical temperature will be considered to be AT the critical point") \
27 X(CRITICAL_SPLINES_ENABLED, "CRITICAL_SPLINES_ENABLED", true, \
28 "If true, the critical splines will be used in the near-vicinity of the critical point") \
29 X(SAVE_RAW_TABLES, "SAVE_RAW_TABLES", false, "If true, the raw, uncompressed tables will also be written to file") \
30 X(ALTERNATIVE_TABLES_DIRECTORY, "ALTERNATIVE_TABLES_DIRECTORY", "", \
31 "If provided, this path will be the root directory for the tabular data. Otherwise, ${HOME}/.CoolProp/Tables is used") \
32 X(ALTERNATIVE_REFPROP_PATH, "ALTERNATIVE_REFPROP_PATH", "", \
33 "An alternative path to be provided to the directory that contains REFPROP's fluids and mixtures directories. If provided, the SETPATH " \
34 "function will be called with this directory prior to calling any REFPROP functions.") \
35 X(ALTERNATIVE_REFPROP_HMX_BNC_PATH, "ALTERNATIVE_REFPROP_HMX_BNC_PATH", "", \
36 "An alternative path to the HMX.BNC file. If provided, it will be passed into REFPROP's SETUP or SETMIX routines") \
37 X(ALTERNATIVE_REFPROP_LIBRARY_PATH, "ALTERNATIVE_REFPROP_LIBRARY_PATH", "", \
38 "An alternative path to the shared library file. If provided, it will be used to load REFPROP") \
39 X(REFPROP_DONT_ESTIMATE_INTERACTION_PARAMETERS, "REFPROP_DONT_ESTIMATE_INTERACTION_PARAMETERS", false, \
40 "If true, if the binary interaction parameters in REFPROP are estimated, throw an error rather than silently continuing") \
41 X(REFPROP_IGNORE_ERROR_ESTIMATED_INTERACTION_PARAMETERS, "REFPROP_IGNORE_ERROR_ESTIMATED_INTERACTION_PARAMETERS", false, \
42 "If true, if the binary interaction parameters in REFPROP are unable to be estimated, silently continue rather than failing") \
43 X(REFPROP_USE_GERG, "REFPROP_USE_GERG", false, \
44 "If true, rather than using the highly-accurate pure fluid equations of state, use the pure-fluid EOS from GERG-2008") \
45 X(REFPROP_ERROR_THRESHOLD, "REFPROP_ERROR_THRESHOLD", static_cast<int>(0), "The highest acceptable error code without throwing an exception") \
46 X(REFPROP_USE_PENGROBINSON, "REFPROP_USE_PENGROBINSON", false, \
47 "If true, rather than using the highly-accurate pure fluid equations of state, use the Peng-Robinson EOS") \
48 X(MAXIMUM_TABLE_DIRECTORY_SIZE_IN_GB, "MAXIMUM_TABLE_DIRECTORY_SIZE_IN_GB", 1.0, \
49 "The maximum allowed size of the directory that is used to store tabular data") \
50 X(DONT_CHECK_PROPERTY_LIMITS, "DONT_CHECK_PROPERTY_LIMITS", false, \
51 "If true, when possible, CoolProp will skip checking whether values are inside the property limits") \
52 X(HENRYS_LAW_TO_GENERATE_VLE_GUESSES, "HENRYS_LAW_TO_GENERATE_VLE_GUESSES", false, \
53 "If true, when doing water-based mixture dewpoint calculations, use Henry's Law to generate guesses for liquid-phase composition") \
54 X(PHASE_ENVELOPE_STARTING_PRESSURE_PA, "PHASE_ENVELOPE_STARTING_PRESSURE_PA", 100.0, "Starting pressure [Pa] for phase envelope construction") \
55 X(R_U_CODATA, "R_U_CODATA", 8.3144598, \
56 "The value for the ideal gas constant in J/mol/K according to CODATA 2014. This value is used to harmonize all the ideal gas constants. " \
57 "This is especially important in the critical region.") \
58 X(VTPR_UNIFAC_PATH, "VTPR_UNIFAC_PATH", "", "The path to the directory containing the UNIFAC JSON files. Should be slash terminated") \
59 X(SPINODAL_MINIMUM_DELTA, "SPINODAL_MINIMUM_DELTA", 0.5, \
60 "The minimal delta to be used in tracing out the spinodal; make sure that the EOS has a spinodal at this value of delta=rho/rho_r") \
61 X(OVERWRITE_FLUIDS, "OVERWRITE_FLUIDS", false, \
62 "If true, and a fluid is added to the fluids library that is already there, rather than not adding the fluid (and probably throwing an " \
63 "exception), overwrite it") \
64 X(OVERWRITE_DEPARTURE_FUNCTION, "OVERWRITE_DEPARTURE_FUNCTION", false, \
65 "If true, and a departure function to be added is already there, rather than not adding the departure function (and probably throwing an " \
66 "exception), overwrite it") \
67 X(OVERWRITE_BINARY_INTERACTION, "OVERWRITE_BINARY_INTERACTION", false, \
68 "If true, and a pair of binary interaction pairs to be added is already there, rather than not adding the binary interaction pair (and " \
69 "probably throwing an exception), overwrite it") \
70 X(USE_GUESSES_IN_PROPSSI, "USE_GUESSES_IN_PROPSSI", false, \
71 "If true, calls to the vectorized versions of PropsSI use the previous state as guess value while looping over the input vectors, only makes " \
72 "sense when working with a single fluid and with points that are not too far from each other.") \
73 X(ASSUME_CRITICAL_POINT_STABLE, "ASSUME_CRITICAL_POINT_STABLE", false, \
74 "If true, evaluation of the stability of critical point will be skipped and point will be assumed to be stable") \
75 X(VTPR_ALWAYS_RELOAD_LIBRARY, "VTPR_ALWAYS_RELOAD_LIBRARY", false, \
76 "If true, the library will always be reloaded, no matter what is currently loaded") \
77 X(FLOAT_PUNCTUATION, "FLOAT_PUNCTUATION", ".", "The first character of this string will be used as the separator between the number fraction.") \
78 X(ENABLE_SUPERANCILLARIES, "ENABLE_SUPERANCILLARIES", true, "If true, the superancillary functions will be used for VLE of pure fluids") \
79 X(LIST_STRING_DELIMITER, "LIST_STRING_DELIMITER", ",", "The delimiter to be used when converting a list of strings to a string")
80
81// Use preprocessor to create the Enum
83{
84#define X(Enum, String, Default, Desc) Enum,
86#undef X
87};
88
89// Evidently SWIG+MATLAB cannot properly wrap enums within classes
91{
98};
99
100namespace CoolProp {
101
104
106configuration_keys config_string_to_key(const std::string& s);
107
110
112std::string config_key_description(const std::string& key);
113
117{
118 public:
119 ConfigurationDataTypes get_type() const { return type; }
120
122 operator bool() const {
123 check_data_type(CONFIGURATION_BOOL_TYPE);
124 return v_bool;
125 };
127 operator double() const {
128 check_data_type(CONFIGURATION_DOUBLE_TYPE);
129 return v_double;
130 };
132 operator std::string() const {
133 check_data_type(CONFIGURATION_STRING_TYPE);
134 return v_string;
135 };
137 operator int() const {
138 check_data_type(CONFIGURATION_INTEGER_TYPE);
139 return v_integer;
140 };
141 // Initializer for bool
143 this->key = key;
145 v_bool = val;
146 };
147 // Initializer for integer
149 this->key = key;
151 v_integer = val;
152 };
153 // Initializer for double
155 this->key = key;
157 v_double = val;
158 };
159 // Initializer for const char *
160 ConfigurationItem(configuration_keys key, const char* val) {
161 this->key = key;
163 v_string = val;
164 };
165 // Initializer for string
166 ConfigurationItem(configuration_keys key, const std::string& val) {
167 this->key = key;
169 v_string = val;
170 };
171 void set_bool(bool val) {
172 check_data_type(CONFIGURATION_BOOL_TYPE);
173 v_bool = val;
174 }
175 void set_integer(int val) {
176 check_data_type(CONFIGURATION_INTEGER_TYPE);
177 v_integer = val;
178 }
179 void set_double(double val) {
180 check_data_type(CONFIGURATION_DOUBLE_TYPE);
181 v_double = val;
182 }
183 void set_string(const std::string& val) {
184 check_data_type(CONFIGURATION_STRING_TYPE);
185 v_string = val;
186 }
187
189 return this->key;
190 }
191#if !defined(SWIG)
193 void add_to_json(rapidjson::Value& val, rapidjson::Document& d) const {
194 std::string name_string = config_key_to_string(key);
195 rapidjson::Value name(name_string.c_str(), d.GetAllocator());
196 switch (type) {
198 rapidjson::Value v(v_bool);
199 val.AddMember(name, v, d.GetAllocator());
200 break;
201 }
203 rapidjson::Value v(v_integer);
204 val.AddMember(name, v, d.GetAllocator());
205 break;
206 }
208 rapidjson::Value v(v_double); // Try to upcast
209 val.AddMember(name, v, d.GetAllocator());
210 break;
211 }
213 rapidjson::Value v(v_string.c_str(), d.GetAllocator());
214 val.AddMember(name, v, d.GetAllocator());
215 break;
216 }
219 throw ValueError();
220 }
221 }
222 void set_from_json(rapidjson::Value& val) {
223 switch (type) {
225 if (!val.IsBool()) {
226 throw ValueError(format("Input is not boolean"));
227 };
228 v_bool = val.GetBool();
229 break;
231 if (!val.IsInt()) {
232 throw ValueError(format("Input is not integer"));
233 };
234 v_integer = val.GetInt();
235 break;
237 if (!val.IsDouble() && !val.IsInt()) {
238 throw ValueError(format("Input [%s] is not double (or something that can be cast to double)", cpjson::to_string(val).c_str()));
239 };
240 if (val.IsDouble()) {
241 v_double = val.GetDouble();
242 } else {
243 v_double = static_cast<double>(val.GetInt());
244 }
245 break;
246 }
248 if (!val.IsString()) {
249 throw ValueError(format("Input is not string"));
250 };
251 v_string = val.GetString();
252 break;
255 throw ValueError();
256 }
257 }
258#endif // !defined(SWIG)
259
260 private:
261 void check_data_type(ConfigurationDataTypes type) const {
262 if (type != this->type) {
263 throw ValueError(format("type does not match"));
264 }
265 };
267 union
268 {
269 double v_double;
270 bool v_bool;
272 };
273 std::string v_string;
275};
276
278{
279 protected:
280 std::unordered_map<configuration_keys, ConfigurationItem> items;
281
282 public:
284 set_defaults();
285 };
287
290 // Try to find it
291 std::unordered_map<configuration_keys, ConfigurationItem>::iterator it = items.find(key);
292 // If equal to end, not found
293 if (it != items.end()) {
294 // Found, return it
295 return it->second;
296 } else {
297 throw ValueError(format("invalid item"));
298 }
299 }
302 std::pair<configuration_keys, ConfigurationItem> pair(item.get_key(), item);
303 items.insert(pair);
304 };
305
307 std::unordered_map<configuration_keys, ConfigurationItem>& get_items(void) {
308 return items;
309 };
310
313 std::string envkey = "COOLPROP_" + config_key_to_string(key);
314 const char *envval = std::getenv(envkey.c_str());
315 if (envval){
316 auto tobool = [](const std::string x){
317 if (x == "True" || x == "true"){ return true;}
318 if (x == "False" || x == "false"){ return false;}
319 throw ValueError(x);
320 };
321 switch (get_item(key).get_type()){
323 items.erase(key); items.emplace(key, ConfigurationItem(key, std::string(envval)));
324 break;
326 int i;
327 try{
328 i = std::stoi(envval);
329 }
330 catch(...){
331 auto skey = config_key_to_string(key);
332 std::string msg = "Unable to convert \""+std::string(envval)+"\" to int for key ["+skey+"]";
333 std::cerr << msg << std::endl;
334 throw ValueError(msg);
335 }
336 items.erase(key); items.emplace(key, ConfigurationItem(key, i));
337 break;
339 double d;
340 try{
341 d = std::stod(envval);
342 }
343 catch(...){
344 auto skey = config_key_to_string(key);
345 std::string msg = "Unable to convert \""+std::string(envval)+"\" to double for key ["+skey+"]";
346 std::cerr << msg << std::endl;
347 throw ValueError(msg);
348 }
349 items.erase(key); items.emplace(key, ConfigurationItem(key, d));
350 break;
352 bool b;
353 try{
354 b = tobool(envval);
355 }
356 catch(...){
357 auto skey = config_key_to_string(key);
358 std::string msg = "Unable to convert \""+std::string(envval)+"\" to bool for key ["+skey+"]";
359 std::cerr << msg << std::endl;
360 throw ValueError(msg);
361 }
362 items.erase(key); items.emplace(key, ConfigurationItem(key, b));
363 break;
364 default:
365 auto skey = config_key_to_string(key);
366 throw ValueError("This key ["+skey+"] has the wrong type; value was "+std::string(envval)+" ");
367 }
368 return true;
369 }
370 return false;
371 }
372
374 void set_defaults(void) {
375/* ***MAGIC WARNING**!!
376 * See http://stackoverflow.com/a/148610
377 * See http://stackoverflow.com/questions/147267/easy-way-to-use-variables-of-enum-types-as-string-in-c#202511
378 */
379#define X(Enum, String, Default, Desc) add_item(ConfigurationItem(Enum, Default));
381#undef X
382
383 // See if the variable is already present as environment variable
384#define X(Enum, String, Default, Desc) possibly_set_from_env(Enum);
386#undef X
387
388 };
389};
390
394
403#if !defined(SWIG) // Hide this for swig - Swig gets confused
404void get_config_as_json(rapidjson::Document& doc);
405#endif
407std::string get_config_as_json_string();
408
412
414void set_config_bool(configuration_keys key, bool val);
416void set_config_int(configuration_keys key, int val);
418void set_config_double(configuration_keys key, double val);
420void set_config_string(configuration_keys key, const std::string& val);
422#if !defined(SWIG) // Hide this for swig - Swig gets confused
423void set_config_json(rapidjson::Document& doc);
424#endif
426void set_config_as_json_string(const std::string& s);
427} // namespace CoolProp
428
429#endif // COOLPROP_CONFIGURATION