Coverage Report

Created: 2025-06-10 13:21

next uncovered line (L), next uncovered region (R), next uncovered branch (B)
/bitcoin/src/common/settings.h
Line
Count
Source
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
11.0k
    explicit SettingsSpan(const SettingsValue& value) noexcept : SettingsSpan(&value, 1) {}
93
673k
    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
auto FindKey(Map&& map, Key&& key) -> decltype(&map.at(key))
108
95.1M
{
109
95.1M
    auto it = map.find(key);
110
95.1M
    return it == map.end() ? nullptr : &it->second;
  Branch (110:12): [True: 0, False: 0]
  Branch (110:12): [True: 38.1M, False: 11.0k]
  Branch (110:12): [True: 0, False: 0]
  Branch (110:12): [True: 0, False: 0]
  Branch (110:12): [True: 0, False: 0]
  Branch (110:12): [True: 11.0k, False: 0]
  Branch (110:12): [True: 0, False: 0]
  Branch (110:12): [True: 18.4M, False: 662k]
  Branch (110:12): [True: 18.7M, False: 0]
  Branch (110:12): [True: 19.0M, False: 0]
111
95.1M
}
Unexecuted instantiation: decltype (&(({parm#1}.at)({parm#2}))) common::FindKey<std::map<std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> >, std::vector<UniValue, std::allocator<UniValue> >, std::less<std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > >, std::allocator<std::pair<std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > const, std::vector<UniValue, std::allocator<UniValue> > > > >&, std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > const&>(std::map<std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> >, std::vector<UniValue, std::allocator<UniValue> >, std::less<std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > >, std::allocator<std::pair<std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > const, std::vector<UniValue, std::allocator<UniValue> > > > >&, std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > const&)
decltype (&(({parm#1}.at)({parm#2}))) common::FindKey<std::map<std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> >, UniValue, std::less<std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > >, std::allocator<std::pair<std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > const, UniValue> > > const&, std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > const&>(std::map<std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> >, UniValue, std::less<std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > >, std::allocator<std::pair<std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > const, UniValue> > > const&, std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > const&)
Line
Count
Source
108
38.1M
{
109
38.1M
    auto it = map.find(key);
110
38.1M
    return it == map.end() ? nullptr : &it->second;
  Branch (110:12): [True: 38.1M, False: 11.0k]
111
38.1M
}
Unexecuted instantiation: decltype (&(({parm#1}.at)({parm#2}))) common::FindKey<std::map<std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> >, UniValue, std::less<std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > >, std::allocator<std::pair<std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > const, UniValue> > >&, std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > const&>(std::map<std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> >, UniValue, std::less<std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > >, std::allocator<std::pair<std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > const, UniValue> > >&, std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > const&)
Unexecuted instantiation: decltype (&(({parm#1}.at)({parm#2}))) common::FindKey<std::map<std::variant<CNoDestination, PubKeyDestination, PKHash, ScriptHash, WitnessV0ScriptHash, WitnessV0KeyHash, WitnessV1Taproot, PayToAnchor, WitnessUnknown>, wallet::CAddressBookData, std::less<std::variant<CNoDestination, PubKeyDestination, PKHash, ScriptHash, WitnessV0ScriptHash, WitnessV0KeyHash, WitnessV1Taproot, PayToAnchor, WitnessUnknown> >, std::allocator<std::pair<std::variant<CNoDestination, PubKeyDestination, PKHash, ScriptHash, WitnessV0ScriptHash, WitnessV0KeyHash, WitnessV1Taproot, PayToAnchor, WitnessUnknown> const, wallet::CAddressBookData> > >&, std::variant<CNoDestination, PubKeyDestination, PKHash, ScriptHash, WitnessV0ScriptHash, WitnessV0KeyHash, WitnessV1Taproot, PayToAnchor, WitnessUnknown> const&>(std::map<std::variant<CNoDestination, PubKeyDestination, PKHash, ScriptHash, WitnessV0ScriptHash, WitnessV0KeyHash, WitnessV1Taproot, PayToAnchor, WitnessUnknown>, wallet::CAddressBookData, std::less<std::variant<CNoDestination, PubKeyDestination, PKHash, ScriptHash, WitnessV0ScriptHash, WitnessV0KeyHash, WitnessV1Taproot, PayToAnchor, WitnessUnknown> >, std::allocator<std::pair<std::variant<CNoDestination, PubKeyDestination, PKHash, ScriptHash, WitnessV0ScriptHash, WitnessV0KeyHash, WitnessV1Taproot, PayToAnchor, WitnessUnknown> const, wallet::CAddressBookData> > >&, std::variant<CNoDestination, PubKeyDestination, PKHash, ScriptHash, WitnessV0ScriptHash, WitnessV0KeyHash, WitnessV1Taproot, PayToAnchor, WitnessUnknown> const&)
Unexecuted instantiation: decltype (&(({parm#1}.at)({parm#2}))) common::FindKey<std::map<std::variant<CNoDestination, PubKeyDestination, PKHash, ScriptHash, WitnessV0ScriptHash, WitnessV0KeyHash, WitnessV1Taproot, PayToAnchor, WitnessUnknown>, wallet::CAddressBookData, std::less<std::variant<CNoDestination, PubKeyDestination, PKHash, ScriptHash, WitnessV0ScriptHash, WitnessV0KeyHash, WitnessV1Taproot, PayToAnchor, WitnessUnknown> >, std::allocator<std::pair<std::variant<CNoDestination, PubKeyDestination, PKHash, ScriptHash, WitnessV0ScriptHash, WitnessV0KeyHash, WitnessV1Taproot, PayToAnchor, WitnessUnknown> const, wallet::CAddressBookData> > > const&, std::variant<CNoDestination, PubKeyDestination, PKHash, ScriptHash, WitnessV0ScriptHash, WitnessV0KeyHash, WitnessV1Taproot, PayToAnchor, WitnessUnknown> const&>(std::map<std::variant<CNoDestination, PubKeyDestination, PKHash, ScriptHash, WitnessV0ScriptHash, WitnessV0KeyHash, WitnessV1Taproot, PayToAnchor, WitnessUnknown>, wallet::CAddressBookData, std::less<std::variant<CNoDestination, PubKeyDestination, PKHash, ScriptHash, WitnessV0ScriptHash, WitnessV0KeyHash, WitnessV1Taproot, PayToAnchor, WitnessUnknown> >, std::allocator<std::pair<std::variant<CNoDestination, PubKeyDestination, PKHash, ScriptHash, WitnessV0ScriptHash, WitnessV0KeyHash, WitnessV1Taproot, PayToAnchor, WitnessUnknown> const, wallet::CAddressBookData> > > const&, std::variant<CNoDestination, PubKeyDestination, PKHash, ScriptHash, WitnessV0ScriptHash, WitnessV0KeyHash, WitnessV1Taproot, PayToAnchor, WitnessUnknown> const&)
decltype (&(({parm#1}.at)({parm#2}))) common::FindKey<std::map<std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> >, std::vector<UniValue, std::allocator<UniValue> >, std::less<std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > >, std::allocator<std::pair<std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > const, std::vector<UniValue, std::allocator<UniValue> > > > >&, char const (&) [12]>(std::map<std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> >, std::vector<UniValue, std::allocator<UniValue> >, std::less<std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > >, std::allocator<std::pair<std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > const, std::vector<UniValue, std::allocator<UniValue> > > > >&, char const (&) [12])
Line
Count
Source
108
11.0k
{
109
11.0k
    auto it = map.find(key);
110
11.0k
    return it == map.end() ? nullptr : &it->second;
  Branch (110:12): [True: 11.0k, False: 0]
111
11.0k
}
Unexecuted instantiation: decltype (&(({parm#1}.at)({parm#2}))) common::FindKey<std::map<std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> >, std::map<std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> >, std::vector<UniValue, std::allocator<UniValue> >, std::less<std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > >, std::allocator<std::pair<std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > const, std::vector<UniValue, std::allocator<UniValue> > > > >, std::less<std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > >, std::allocator<std::pair<std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > const, std::map<std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> >, std::vector<UniValue, std::allocator<UniValue> >, std::less<std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > >, std::allocator<std::pair<std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > const, std::vector<UniValue, std::allocator<UniValue> > > > > > > >&, std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > const&>(std::map<std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> >, std::map<std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> >, std::vector<UniValue, std::allocator<UniValue> >, std::less<std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > >, std::allocator<std::pair<std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > const, std::vector<UniValue, std::allocator<UniValue> > > > >, std::less<std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > >, std::allocator<std::pair<std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > const, std::map<std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> >, std::vector<UniValue, std::allocator<UniValue> >, std::less<std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > >, std::allocator<std::pair<std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > const, std::vector<UniValue, std::allocator<UniValue> > > > > > > >&, std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > const&)
decltype (&(({parm#1}.at)({parm#2}))) common::FindKey<std::map<std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> >, std::vector<UniValue, std::allocator<UniValue> >, std::less<std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > >, std::allocator<std::pair<std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > const, std::vector<UniValue, std::allocator<UniValue> > > > > const&, std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > const&>(std::map<std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> >, std::vector<UniValue, std::allocator<UniValue> >, std::less<std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > >, std::allocator<std::pair<std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > const, std::vector<UniValue, std::allocator<UniValue> > > > > const&, std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > const&)
Line
Count
Source
108
19.0M
{
109
19.0M
    auto it = map.find(key);
110
19.0M
    return it == map.end() ? nullptr : &it->second;
  Branch (110:12): [True: 18.4M, False: 662k]
111
19.0M
}
decltype (&(({parm#1}.at)({parm#2}))) common::FindKey<std::map<std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> >, std::map<std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> >, std::vector<UniValue, std::allocator<UniValue> >, std::less<std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > >, std::allocator<std::pair<std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > const, std::vector<UniValue, std::allocator<UniValue> > > > >, std::less<std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > >, std::allocator<std::pair<std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > const, std::map<std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> >, std::vector<UniValue, std::allocator<UniValue> >, std::less<std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > >, std::allocator<std::pair<std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > const, std::vector<UniValue, std::allocator<UniValue> > > > > > > > const&, std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > const&>(std::map<std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> >, std::map<std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> >, std::vector<UniValue, std::allocator<UniValue> >, std::less<std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > >, std::allocator<std::pair<std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > const, std::vector<UniValue, std::allocator<UniValue> > > > >, std::less<std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > >, std::allocator<std::pair<std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > const, std::map<std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> >, std::vector<UniValue, std::allocator<UniValue> >, std::less<std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > >, std::allocator<std::pair<std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > const, std::vector<UniValue, std::allocator<UniValue> > > > > > > > const&, std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > const&)
Line
Count
Source
108
18.7M
{
109
18.7M
    auto it = map.find(key);
110
18.7M
    return it == map.end() ? nullptr : &it->second;
  Branch (110:12): [True: 18.7M, False: 0]
111
18.7M
}
decltype (&(({parm#1}.at)({parm#2}))) common::FindKey<std::map<std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> >, std::map<std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> >, std::vector<UniValue, std::allocator<UniValue> >, std::less<std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > >, std::allocator<std::pair<std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > const, std::vector<UniValue, std::allocator<UniValue> > > > >, std::less<std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > >, std::allocator<std::pair<std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > const, std::map<std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> >, std::vector<UniValue, std::allocator<UniValue> >, std::less<std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > >, std::allocator<std::pair<std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > const, std::vector<UniValue, std::allocator<UniValue> > > > > > > > const&, char const (&) [1]>(std::map<std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> >, std::map<std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> >, std::vector<UniValue, std::allocator<UniValue> >, std::less<std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > >, std::allocator<std::pair<std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > const, std::vector<UniValue, std::allocator<UniValue> > > > >, std::less<std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > >, std::allocator<std::pair<std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > const, std::map<std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> >, std::vector<UniValue, std::allocator<UniValue> >, std::less<std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > >, std::allocator<std::pair<std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > const, std::vector<UniValue, std::allocator<UniValue> > > > > > > > const&, char const (&) [1])
Line
Count
Source
108
19.0M
{
109
19.0M
    auto it = map.find(key);
110
19.0M
    return it == map.end() ? nullptr : &it->second;
  Branch (110:12): [True: 19.0M, False: 0]
111
19.0M
}
112
113
} // namespace common
114
115
#endif // BITCOIN_COMMON_SETTINGS_H