Coverage Report

Created: 2026-06-01 18:35

next uncovered line (L), next uncovered region (R), next uncovered branch (B)
/bitcoin/src/common/settings.cpp
Line
Count
Source
1
// Copyright (c) 2019-present 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
#include <common/settings.h>
6
7
#include <bitcoin-build-config.h> // IWYU pragma: keep
8
9
#include <tinyformat.h>
10
#include <univalue.h>
11
#include <util/fs.h>
12
13
#include <algorithm>
14
#include <fstream>
15
#include <iterator>
16
#include <map>
17
#include <string>
18
#include <utility>
19
#include <vector>
20
21
namespace common {
22
namespace {
23
24
enum class Source {
25
   FORCED,
26
   COMMAND_LINE,
27
   RW_SETTINGS,
28
   CONFIG_FILE_NETWORK_SECTION,
29
   CONFIG_FILE_DEFAULT_SECTION
30
};
31
32
// Json object key for the auto-generated warning comment
33
const std::string SETTINGS_WARN_MSG_KEY{"_warning_"};
34
35
//! Merge settings from multiple sources in precedence order:
36
//! Forced config > command line > read-write settings file > config file network-specific section > config file default section
37
//!
38
//! This function is provided with a callback function fn that contains
39
//! specific logic for how to merge the sources.
40
template <typename Fn>
41
static void MergeSettings(const Settings& settings, const std::string& section, const std::string& name, Fn&& fn)
42
2.13k
{
43
    // Merge in the forced settings
44
2.13k
    if (auto* value = FindKey(settings.forced_settings, name)) {
  Branch (44:15): [True: 0, False: 1.82k]
  Branch (44:15): [True: 0, False: 305]
  Branch (44:15): [True: 0, False: 0]
45
0
        fn(SettingsSpan(*value), Source::FORCED);
46
0
    }
47
    // Merge in the command-line options
48
2.13k
    if (auto* values = FindKey(settings.command_line_options, name)) {
  Branch (48:15): [True: 0, False: 1.82k]
  Branch (48:15): [True: 0, False: 305]
  Branch (48:15): [True: 0, False: 0]
49
0
        fn(SettingsSpan(*values), Source::COMMAND_LINE);
50
0
    }
51
    // Merge in the read-write settings
52
2.13k
    if (const SettingsValue* value = FindKey(settings.rw_settings, name)) {
  Branch (52:30): [True: 0, False: 1.82k]
  Branch (52:30): [True: 0, False: 305]
  Branch (52:30): [True: 0, False: 0]
53
0
        fn(SettingsSpan(*value), Source::RW_SETTINGS);
54
0
    }
55
    // Merge in the network-specific section of the config file
56
2.13k
    if (!section.empty()) {
  Branch (56:9): [True: 1.82k, False: 0]
  Branch (56:9): [True: 305, False: 0]
  Branch (56:9): [True: 0, False: 0]
57
2.13k
        if (auto* map = FindKey(settings.ro_config, section)) {
  Branch (57:19): [True: 0, False: 1.82k]
  Branch (57:19): [True: 0, False: 305]
  Branch (57:19): [True: 0, False: 0]
58
0
            if (auto* values = FindKey(*map, name)) {
  Branch (58:23): [True: 0, False: 0]
  Branch (58:23): [True: 0, False: 0]
  Branch (58:23): [True: 0, False: 0]
59
0
                fn(SettingsSpan(*values), Source::CONFIG_FILE_NETWORK_SECTION);
60
0
            }
61
0
        }
62
2.13k
    }
63
    // Merge in the default section of the config file
64
2.13k
    if (auto* map = FindKey(settings.ro_config, "")) {
  Branch (64:15): [True: 0, False: 1.82k]
  Branch (64:15): [True: 0, False: 305]
  Branch (64:15): [True: 0, False: 0]
65
0
        if (auto* values = FindKey(*map, name)) {
  Branch (65:19): [True: 0, False: 0]
  Branch (65:19): [True: 0, False: 0]
  Branch (65:19): [True: 0, False: 0]
66
0
            fn(SettingsSpan(*values), Source::CONFIG_FILE_DEFAULT_SECTION);
67
0
        }
68
0
    }
69
2.13k
}
settings.cpp:void common::(anonymous namespace)::MergeSettings<common::GetSetting(common::Settings const&, std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > const&, std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > const&, bool, bool, bool)::$_0>(common::Settings const&, std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > const&, std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > const&, common::GetSetting(common::Settings const&, std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > const&, std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > const&, bool, bool, bool)::$_0&&)
Line
Count
Source
42
1.82k
{
43
    // Merge in the forced settings
44
1.82k
    if (auto* value = FindKey(settings.forced_settings, name)) {
  Branch (44:15): [True: 0, False: 1.82k]
45
0
        fn(SettingsSpan(*value), Source::FORCED);
46
0
    }
47
    // Merge in the command-line options
48
1.82k
    if (auto* values = FindKey(settings.command_line_options, name)) {
  Branch (48:15): [True: 0, False: 1.82k]
49
0
        fn(SettingsSpan(*values), Source::COMMAND_LINE);
50
0
    }
51
    // Merge in the read-write settings
52
1.82k
    if (const SettingsValue* value = FindKey(settings.rw_settings, name)) {
  Branch (52:30): [True: 0, False: 1.82k]
53
0
        fn(SettingsSpan(*value), Source::RW_SETTINGS);
54
0
    }
55
    // Merge in the network-specific section of the config file
56
1.82k
    if (!section.empty()) {
  Branch (56:9): [True: 1.82k, False: 0]
57
1.82k
        if (auto* map = FindKey(settings.ro_config, section)) {
  Branch (57:19): [True: 0, False: 1.82k]
58
0
            if (auto* values = FindKey(*map, name)) {
  Branch (58:23): [True: 0, False: 0]
59
0
                fn(SettingsSpan(*values), Source::CONFIG_FILE_NETWORK_SECTION);
60
0
            }
61
0
        }
62
1.82k
    }
63
    // Merge in the default section of the config file
64
1.82k
    if (auto* map = FindKey(settings.ro_config, "")) {
  Branch (64:15): [True: 0, False: 1.82k]
65
0
        if (auto* values = FindKey(*map, name)) {
  Branch (65:19): [True: 0, False: 0]
66
0
            fn(SettingsSpan(*values), Source::CONFIG_FILE_DEFAULT_SECTION);
67
0
        }
68
0
    }
69
1.82k
}
settings.cpp:void common::(anonymous namespace)::MergeSettings<common::GetSettingsList(common::Settings const&, std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > const&, std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > const&, bool)::$_0>(common::Settings const&, std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > const&, std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > const&, common::GetSettingsList(common::Settings const&, std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > const&, std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > const&, bool)::$_0&&)
Line
Count
Source
42
305
{
43
    // Merge in the forced settings
44
305
    if (auto* value = FindKey(settings.forced_settings, name)) {
  Branch (44:15): [True: 0, False: 305]
45
0
        fn(SettingsSpan(*value), Source::FORCED);
46
0
    }
47
    // Merge in the command-line options
48
305
    if (auto* values = FindKey(settings.command_line_options, name)) {
  Branch (48:15): [True: 0, False: 305]
49
0
        fn(SettingsSpan(*values), Source::COMMAND_LINE);
50
0
    }
51
    // Merge in the read-write settings
52
305
    if (const SettingsValue* value = FindKey(settings.rw_settings, name)) {
  Branch (52:30): [True: 0, False: 305]
53
0
        fn(SettingsSpan(*value), Source::RW_SETTINGS);
54
0
    }
55
    // Merge in the network-specific section of the config file
56
305
    if (!section.empty()) {
  Branch (56:9): [True: 305, False: 0]
57
305
        if (auto* map = FindKey(settings.ro_config, section)) {
  Branch (57:19): [True: 0, False: 305]
58
0
            if (auto* values = FindKey(*map, name)) {
  Branch (58:23): [True: 0, False: 0]
59
0
                fn(SettingsSpan(*values), Source::CONFIG_FILE_NETWORK_SECTION);
60
0
            }
61
0
        }
62
305
    }
63
    // Merge in the default section of the config file
64
305
    if (auto* map = FindKey(settings.ro_config, "")) {
  Branch (64:15): [True: 0, False: 305]
65
0
        if (auto* values = FindKey(*map, name)) {
  Branch (65:19): [True: 0, False: 0]
66
0
            fn(SettingsSpan(*values), Source::CONFIG_FILE_DEFAULT_SECTION);
67
0
        }
68
0
    }
69
305
}
Unexecuted instantiation: settings.cpp:void common::(anonymous namespace)::MergeSettings<common::OnlyHasDefaultSectionSetting(common::Settings const&, std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > const&, std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > const&)::$_0>(common::Settings const&, std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > const&, std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > const&, common::OnlyHasDefaultSectionSetting(common::Settings const&, std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > const&, std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > const&)::$_0&&)
70
} // namespace
71
72
bool ReadSettings(const fs::path& path, std::map<std::string, SettingsValue>& values, std::vector<std::string>& errors)
73
0
{
74
0
    values.clear();
75
0
    errors.clear();
76
77
    // Ok for file to not exist
78
0
    if (!fs::exists(path)) return true;
  Branch (78:9): [True: 0, False: 0]
79
80
0
    std::ifstream file;
81
0
    file.open(path.std_path());
82
0
    if (!file.is_open()) {
  Branch (82:9): [True: 0, False: 0]
83
0
      errors.emplace_back(strprintf("%s. Please check permissions.", fs::PathToString(path)));
84
0
      return false;
85
0
    }
86
87
0
    SettingsValue in;
88
0
    if (!in.read(std::string{std::istreambuf_iterator<char>(file), std::istreambuf_iterator<char>()})) {
  Branch (88:9): [True: 0, False: 0]
89
0
        errors.emplace_back(strprintf("Settings file %s does not contain valid JSON. This is probably caused by disk corruption or a crash, "
90
0
                                      "and can be fixed by removing the file, which will reset settings to default values.",
91
0
                                      fs::PathToString(path)));
92
0
        return false;
93
0
    }
94
95
0
    if (file.fail()) {
  Branch (95:9): [True: 0, False: 0]
96
0
        errors.emplace_back(strprintf("Failed reading settings file %s", fs::PathToString(path)));
97
0
        return false;
98
0
    }
99
0
    file.close(); // Done with file descriptor. Release while copying data.
100
101
0
    if (!in.isObject()) {
  Branch (101:9): [True: 0, False: 0]
102
0
        errors.emplace_back(strprintf("Found non-object value %s in settings file %s", in.write(), fs::PathToString(path)));
103
0
        return false;
104
0
    }
105
106
0
    const std::vector<std::string>& in_keys = in.getKeys();
107
0
    const std::vector<SettingsValue>& in_values = in.getValues();
108
0
    for (size_t i = 0; i < in_keys.size(); ++i) {
  Branch (108:24): [True: 0, False: 0]
109
0
        auto inserted = values.emplace(in_keys[i], in_values[i]);
110
0
        if (!inserted.second) {
  Branch (110:13): [True: 0, False: 0]
111
0
            errors.emplace_back(strprintf("Found duplicate key %s in settings file %s", in_keys[i], fs::PathToString(path)));
112
0
            values.clear();
113
0
            break;
114
0
        }
115
0
    }
116
117
    // Remove auto-generated warning comment from the accessible settings.
118
0
    values.erase(SETTINGS_WARN_MSG_KEY);
119
120
0
    return errors.empty();
121
0
}
122
123
bool WriteSettings(const fs::path& path,
124
    const std::map<std::string, SettingsValue>& values,
125
    std::vector<std::string>& errors)
126
0
{
127
0
    SettingsValue out(SettingsValue::VOBJ);
128
    // Add auto-generated warning comment
129
0
    out.pushKV(SETTINGS_WARN_MSG_KEY, strprintf("This file is automatically generated and updated by %s. Please do not edit this file while the node "
130
0
                                                "is running, as any changes might be ignored or overwritten.", CLIENT_NAME));
131
    // Push settings values
132
0
    for (const auto& value : values) {
  Branch (132:28): [True: 0, False: 0]
133
0
        out.pushKVEnd(value.first, value.second);
134
0
    }
135
0
    std::ofstream file;
136
0
    file.open(path.std_path());
137
0
    if (file.fail()) {
  Branch (137:9): [True: 0, False: 0]
138
0
        errors.emplace_back(strprintf("Error: Unable to open settings file %s for writing", fs::PathToString(path)));
139
0
        return false;
140
0
    }
141
0
    file << out.write(/* prettyIndent= */ 4, /* indentLevel= */ 1) << std::endl;
142
0
    file.close();
143
0
    return true;
144
0
}
145
146
SettingsValue GetSetting(const Settings& settings,
147
    const std::string& section,
148
    const std::string& name,
149
    bool ignore_default_section_config,
150
    bool ignore_nonpersistent,
151
    bool get_chain_type)
152
1.82k
{
153
1.82k
    SettingsValue result;
154
1.82k
    bool done = false; // Done merging any more settings sources.
155
1.82k
    MergeSettings(settings, section, name, [&](SettingsSpan span, Source source) {
156
        // Weird behavior preserved for backwards compatibility: Apply negated
157
        // setting even if non-negated setting would be ignored. A negated
158
        // value in the default section is applied to network specific options,
159
        // even though normal non-negated values there would be ignored.
160
0
        const bool never_ignore_negated_setting = span.last_negated();
161
162
        // Weird behavior preserved for backwards compatibility: Take first
163
        // assigned value instead of last. In general, later settings take
164
        // precedence over early settings, but for backwards compatibility in
165
        // the config file the precedence is reversed for all settings except
166
        // chain type settings.
167
0
        const bool reverse_precedence =
168
0
            (source == Source::CONFIG_FILE_NETWORK_SECTION || source == Source::CONFIG_FILE_DEFAULT_SECTION) &&
  Branch (168:14): [True: 0, False: 0]
  Branch (168:63): [True: 0, False: 0]
169
0
            !get_chain_type;
  Branch (169:13): [True: 0, False: 0]
170
171
        // Weird behavior preserved for backwards compatibility: Negated
172
        // -regtest and -testnet arguments which you would expect to override
173
        // values set in the configuration file are currently accepted but
174
        // silently ignored. It would be better to apply these just like other
175
        // negated values, or at least warn they are ignored.
176
0
        const bool skip_negated_command_line = get_chain_type;
177
178
0
        if (done) return;
  Branch (178:13): [True: 0, False: 0]
179
180
        // Ignore settings in default config section if requested.
181
0
        if (ignore_default_section_config && source == Source::CONFIG_FILE_DEFAULT_SECTION &&
  Branch (181:13): [True: 0, False: 0]
  Branch (181:46): [True: 0, False: 0]
182
0
            !never_ignore_negated_setting) {
  Branch (182:13): [True: 0, False: 0]
183
0
            return;
184
0
        }
185
186
        // Ignore nonpersistent settings if requested.
187
0
        if (ignore_nonpersistent && (source == Source::COMMAND_LINE || source == Source::FORCED)) return;
  Branch (187:13): [True: 0, False: 0]
  Branch (187:38): [True: 0, False: 0]
  Branch (187:72): [True: 0, False: 0]
188
189
        // Skip negated command line settings.
190
0
        if (skip_negated_command_line && span.last_negated()) return;
  Branch (190:13): [True: 0, False: 0]
  Branch (190:42): [True: 0, False: 0]
191
192
0
        if (!span.empty()) {
  Branch (192:13): [True: 0, False: 0]
193
0
            result = reverse_precedence ? span.begin()[0] : span.end()[-1];
  Branch (193:22): [True: 0, False: 0]
194
0
            done = true;
195
0
        } else if (span.last_negated()) {
  Branch (195:20): [True: 0, False: 0]
196
0
            result = false;
197
0
            done = true;
198
0
        }
199
0
    });
200
1.82k
    return result;
201
1.82k
}
202
203
std::vector<SettingsValue> GetSettingsList(const Settings& settings,
204
    const std::string& section,
205
    const std::string& name,
206
    bool ignore_default_section_config)
207
305
{
208
305
    std::vector<SettingsValue> result;
209
305
    bool done = false; // Done merging any more settings sources.
210
305
    bool prev_negated_empty = false;
211
305
    MergeSettings(settings, section, name, [&](SettingsSpan span, Source source) {
212
        // Weird behavior preserved for backwards compatibility: Apply config
213
        // file settings even if negated on command line. Negating a setting on
214
        // command line will ignore earlier settings on the command line and
215
        // ignore settings in the config file, unless the negated command line
216
        // value is followed by non-negated value, in which case config file
217
        // settings will be brought back from the dead (but earlier command
218
        // line settings will still be ignored).
219
0
        const bool add_zombie_config_values =
220
0
            (source == Source::CONFIG_FILE_NETWORK_SECTION || source == Source::CONFIG_FILE_DEFAULT_SECTION) &&
  Branch (220:14): [True: 0, False: 0]
  Branch (220:63): [True: 0, False: 0]
221
0
            !prev_negated_empty;
  Branch (221:13): [True: 0, False: 0]
222
223
        // Ignore settings in default config section if requested.
224
0
        if (ignore_default_section_config && source == Source::CONFIG_FILE_DEFAULT_SECTION) return;
  Branch (224:13): [True: 0, False: 0]
  Branch (224:46): [True: 0, False: 0]
225
226
        // Add new settings to the result if isn't already complete, or if the
227
        // values are zombies.
228
0
        if (!done || add_zombie_config_values) {
  Branch (228:13): [True: 0, False: 0]
  Branch (228:22): [True: 0, False: 0]
229
0
            for (const auto& value : span) {
  Branch (229:36): [True: 0, False: 0]
230
0
                if (value.isArray()) {
  Branch (230:21): [True: 0, False: 0]
231
0
                    result.insert(result.end(), value.getValues().begin(), value.getValues().end());
232
0
                } else {
233
0
                    result.push_back(value);
234
0
                }
235
0
            }
236
0
        }
237
238
        // If a setting was negated, or if a setting was forced, set
239
        // done to true to ignore any later lower priority settings.
240
0
        done |= span.negated() > 0 || source == Source::FORCED;
  Branch (240:17): [True: 0, False: 0]
  Branch (240:39): [True: 0, False: 0]
241
242
        // Update the negated and empty state used for the zombie values check.
243
0
        prev_negated_empty |= span.last_negated() && result.empty();
  Branch (243:31): [True: 0, False: 0]
  Branch (243:54): [True: 0, False: 0]
244
0
    });
245
305
    return result;
246
305
}
247
248
bool OnlyHasDefaultSectionSetting(const Settings& settings, const std::string& section, const std::string& name)
249
0
{
250
0
    bool has_default_section_setting = false;
251
0
    bool has_other_setting = false;
252
0
    MergeSettings(settings, section, name, [&](SettingsSpan span, Source source) {
253
0
        if (span.empty()) return;
  Branch (253:13): [True: 0, False: 0]
254
0
        else if (source == Source::CONFIG_FILE_DEFAULT_SECTION) has_default_section_setting = true;
  Branch (254:18): [True: 0, False: 0]
255
0
        else has_other_setting = true;
256
0
    });
257
    // If a value is set in the default section and not explicitly overwritten by the
258
    // user on the command line or in a different section, then we want to enable
259
    // warnings about the value being ignored.
260
0
    return has_default_section_setting && !has_other_setting;
  Branch (260:12): [True: 0, False: 0]
  Branch (260:43): [True: 0, False: 0]
261
0
}
262
263
0
SettingsSpan::SettingsSpan(const std::vector<SettingsValue>& vec) noexcept : SettingsSpan(vec.data(), vec.size()) {}
264
0
const SettingsValue* SettingsSpan::begin() const { return data + negated(); }
265
0
const SettingsValue* SettingsSpan::end() const { return data + size; }
266
0
bool SettingsSpan::empty() const { return size == 0 || last_negated(); }
  Branch (266:43): [True: 0, False: 0]
  Branch (266:56): [True: 0, False: 0]
267
0
bool SettingsSpan::last_negated() const { return size > 0 && data[size - 1].isFalse(); }
  Branch (267:50): [True: 0, False: 0]
  Branch (267:62): [True: 0, False: 0]
268
size_t SettingsSpan::negated() const
269
0
{
270
0
    for (size_t i = size; i > 0; --i) {
  Branch (270:27): [True: 0, False: 0]
271
0
        if (data[i - 1].isFalse()) return i; // Return number of negated values (position of last false value)
  Branch (271:13): [True: 0, False: 0]
272
0
    }
273
0
    return 0;
274
0
}
275
276
} // namespace common