Branch data Line data Source code
1 : : // Copyright (c) 2019-2022 The Bitcoin Core developers 2 : : // Distributed under the MIT software license, see the accompanying 3 : : // file COPYING or http://www.opensource.org/licenses/mit-license.php. 4 : : 5 : : #ifndef BITCOIN_COMMON_SETTINGS_H 6 : : #define BITCOIN_COMMON_SETTINGS_H 7 : : 8 : : #include <util/fs.h> 9 : : 10 : : #include <cstddef> 11 : : #include <map> 12 : : #include <string> 13 : : #include <vector> 14 : : 15 : : class UniValue; 16 : : 17 : : namespace common { 18 : : 19 : : //! Settings value type (string/integer/boolean/null variant). 20 : : //! 21 : : //! @note UniValue is used here for convenience and because it can be easily 22 : : //! serialized in a readable format. But any other variant type that can 23 : : //! be assigned strings, int64_t, and bool values and has get_str(), 24 : : //! getInt<int64_t>(), get_bool(), isNum(), isBool(), isFalse(), isTrue() and 25 : : //! isNull() methods can be substituted if there's a need to move away 26 : : //! from UniValue. (An implementation with boost::variant was posted at 27 : : //! https://github.com/bitcoin/bitcoin/pull/15934/files#r337691812) 28 : : using SettingsValue = UniValue; 29 : : 30 : : //! Stored settings. This struct combines settings from the command line, a 31 : : //! read-only configuration file, and a read-write runtime settings file. 32 : : struct Settings { 33 : : //! Map of setting name to forced setting value. 34 : : std::map<std::string, SettingsValue> forced_settings; 35 : : //! Map of setting name to list of command line values. 36 : : std::map<std::string, std::vector<SettingsValue>> command_line_options; 37 : : //! Map of setting name to read-write file setting value. 38 : : std::map<std::string, SettingsValue> rw_settings; 39 : : //! Map of config section name and setting name to list of config file values. 40 : : std::map<std::string, std::map<std::string, std::vector<SettingsValue>>> ro_config; 41 : : }; 42 : : 43 : : //! Read settings file. 44 : : bool ReadSettings(const fs::path& path, 45 : : std::map<std::string, SettingsValue>& values, 46 : : std::vector<std::string>& errors); 47 : : 48 : : //! Write settings file. 49 : : bool WriteSettings(const fs::path& path, 50 : : const std::map<std::string, SettingsValue>& values, 51 : : std::vector<std::string>& errors); 52 : : 53 : : //! Get settings value from combined sources: forced settings, command line 54 : : //! arguments, runtime read-write settings, and the read-only config file. 55 : : //! 56 : : //! @param ignore_default_section_config - ignore values in the default section 57 : : //! of the config file (part before any 58 : : //! [section] keywords) 59 : : //! @param ignore_nonpersistent - ignore non-persistent settings values (forced 60 : : //! settings values and values specified on the 61 : : //! command line). Only return settings in the 62 : : //! read-only config and read-write settings 63 : : //! files. 64 : : //! @param get_chain_type - enable special backwards compatible behavior 65 : : //! for GetChainType 66 : : SettingsValue GetSetting(const Settings& settings, 67 : : const std::string& section, 68 : : const std::string& name, 69 : : bool ignore_default_section_config, 70 : : bool ignore_nonpersistent, 71 : : bool get_chain_type); 72 : : 73 : : //! Get combined setting value similar to GetSetting(), except if setting was 74 : : //! specified multiple times, return a list of all the values specified. 75 : : std::vector<SettingsValue> GetSettingsList(const Settings& settings, 76 : : const std::string& section, 77 : : const std::string& name, 78 : : bool ignore_default_section_config); 79 : : 80 : : //! Return true if a setting is set in the default config file section, and not 81 : : //! overridden by a higher priority command-line or network section value. 82 : : //! 83 : : //! This is used to provide user warnings about values that might be getting 84 : : //! ignored unintentionally. 85 : : bool OnlyHasDefaultSectionSetting(const Settings& settings, const std::string& section, const std::string& name); 86 : : 87 : : //! Accessor for list of settings that skips negated values when iterated over. 88 : : //! The last boolean `false` value in the list and all earlier values are 89 : : //! considered negated. 90 : : struct SettingsSpan { 91 : : explicit SettingsSpan() = default; 92 : 31716 : explicit SettingsSpan(const SettingsValue& value) noexcept : SettingsSpan(&value, 1) {} 93 : 43208 : explicit SettingsSpan(const SettingsValue* data, size_t size) noexcept : data(data), size(size) {} 94 : : explicit SettingsSpan(const std::vector<SettingsValue>& vec) noexcept; 95 : : const SettingsValue* begin() const; //!< Pointer to first non-negated value. 96 : : const SettingsValue* end() const; //!< Pointer to end of values. 97 : : bool empty() const; //!< True if there are any non-negated values. 98 : : bool last_negated() const; //!< True if the last value is negated. 99 : : size_t negated() const; //!< Number of negated values. 100 : : 101 : : const SettingsValue* data = nullptr; 102 : : size_t size = 0; 103 : : }; 104 : : 105 : : //! Map lookup helper. 106 : : template <typename Map, typename Key> 107 : 2779399 : auto FindKey(Map&& map, Key&& key) -> decltype(&map.at(key)) 108 : : { 109 [ + - - + ]: 2779399 : auto it = map.find(key); 110 [ + + + + : 2779399 : return it == map.end() ? nullptr : &it->second; + - + - ] 111 : 0 : } 112 : : 113 : : } // namespace common 114 : : 115 : : #endif // BITCOIN_COMMON_SETTINGS_H