Skip to content

Commit 854bccd

Browse files
Merge pull request #5870 from PastaPastaPasta/develop-trivial-2024-02-12
backport: trivial 2024 02 12
2 parents 4becf98 + 09e2a9e commit 854bccd

26 files changed

+235
-112
lines changed

.cirrus.yml

+4-1
Original file line numberDiff line numberDiff line change
@@ -42,10 +42,13 @@ global_task_template: &GLOBAL_TASK_TEMPLATE
4242
task:
4343
name: 'ARM [GOAL: install] [buster] [unit tests, no functional tests]'
4444
<< : *GLOBAL_TASK_TEMPLATE
45-
container:
45+
arm_container:
4646
image: debian:buster
47+
cpu: 2
48+
memory: 8G
4749
env:
4850
FILE_ENV: "./ci/test/00_setup_env_arm.sh"
51+
QEMU_USER_CMD: "" # Disable qemu and run the test natively
4952

5053
task:
5154
name: 'Win64 [GOAL: deploy] [unit tests, no gui, no boost::process, no functional tests]'

CONTRIBUTING.md

+3-3
Original file line numberDiff line numberDiff line change
@@ -66,7 +66,7 @@ facilitates social contribution, easy testing and peer review.
6666

6767
To contribute a patch, the workflow is as follows:
6868

69-
1. Fork repository ([only for the first time](https://help.github.com/en/articles/fork-a-repo))
69+
1. Fork repository ([only for the first time](https://docs.github.com/en/get-started/quickstart/fork-a-repo))
7070
1. Create topic branch
7171
1. Commit patches
7272

@@ -161,7 +161,7 @@ for more information on helping with translations.
161161
### Work in Progress Changes and Requests for Comments
162162

163163
If a pull request is not to be considered for merging (yet), please
164-
prefix the title with [WIP] or use [Tasks Lists](https://help.github.com/articles/basic-writing-and-formatting-syntax/#task-lists)
164+
prefix the title with [WIP] or use [Tasks Lists](https://docs.github.com/en/github/writing-on-github/getting-started-with-writing-and-formatting-on-github/basic-writing-and-formatting-syntax#task-lists)
165165
in the body of the pull request to indicate tasks are pending.
166166

167167
### Address Feedback
@@ -383,7 +383,7 @@ of reasons for this, some of which you can do something about:
383383
- It may be because your code is too complex for all but a few people, and those people
384384
may not have realized your pull request even exists. A great way to find people who
385385
are qualified and care about the code you are touching is the
386-
[Git Blame feature](https://help.github.com/articles/tracing-changes-in-a-file/). Simply
386+
[Git Blame feature](https://docs.github.com/en/github/managing-files-in-a-repository/managing-files-on-github/tracking-changes-in-a-file). Simply
387387
look up who last modified the code you are changing and see if you can find
388388
them and give them a nudge. Don't be incessant about the nudging, though.
389389
- Finally, if all else fails, ask on discord or elsewhere for someone to give your pull request

ci/test/04_install.sh

+1
Original file line numberDiff line numberDiff line change
@@ -14,6 +14,7 @@ if [[ $QEMU_USER_CMD == qemu-s390* ]]; then
1414
fi
1515

1616
if [ "$CI_OS_NAME" == "macos" ]; then
17+
sudo -H pip3 install --upgrade pip
1718
IN_GETOPT_BIN="/usr/local/opt/gnu-getopt/bin/getopt" ${CI_RETRY_EXE} pip3 install --user $PIP_PACKAGES
1819
fi
1920

src/consensus/tx_verify.h

+1-1
Original file line numberDiff line numberDiff line change
@@ -24,7 +24,7 @@ namespace Consensus {
2424
* @param[out] txfee Set to the transaction fee if successful.
2525
* Preconditions: tx.IsCoinBase() is false.
2626
*/
27-
bool CheckTxInputs(const CTransaction& tx, TxValidationState& state, const CCoinsViewCache& inputs, int nSpendHeight, CAmount& txfee);
27+
[[nodiscard]] bool CheckTxInputs(const CTransaction& tx, TxValidationState& state, const CCoinsViewCache& inputs, int nSpendHeight, CAmount& txfee);
2828
} // namespace Consensus
2929

3030
/** Auxiliary functions for transaction validation (ideally should not be exposed) */

src/qt/addressbookpage.cpp

+6-3
Original file line numberDiff line numberDiff line change
@@ -319,7 +319,9 @@ void AddressBookPage::on_exportButton_clicked()
319319
// CSV is currently the only supported format
320320
QString filename = GUIUtil::getSaveFileName(this,
321321
tr("Export Address List"), QString(),
322-
tr("Comma separated file", "Name of CSV file format") + QLatin1String(" (*.csv)"), nullptr);
322+
/*: Expanded name of the CSV file format.
323+
See https://en.wikipedia.org/wiki/Comma-separated_values */
324+
tr("Comma separated file") + QLatin1String(" (*.csv)"), nullptr);
323325

324326
if (filename.isNull())
325327
return;
@@ -333,8 +335,9 @@ void AddressBookPage::on_exportButton_clicked()
333335

334336
if(!writer.write()) {
335337
QMessageBox::critical(this, tr("Exporting Failed"),
336-
//: %1 is a name of the file (e.g., "addrbook.csv") that the bitcoin addresses were exported to.
337-
tr("There was an error trying to save the address list to %1. Please try again.", "An error message.").arg(filename));
338+
/*: An error message. %1 is a stand-in argument for the name
339+
of the file we attempted to save to. */
340+
tr("There was an error trying to save the address list to %1. Please try again.").arg(filename));
338341
}
339342
}
340343

src/qt/bitcoin.cpp

+59-10
Original file line numberDiff line numberDiff line change
@@ -127,6 +127,58 @@ static void initTranslations(QTranslator &qtTranslatorBase, QTranslator &qtTrans
127127
QApplication::installTranslator(&translator);
128128
}
129129

130+
static std::string JoinErrors(const std::vector<std::string>& errors)
131+
{
132+
return Join(errors, "\n", [](const std::string& error) { return "- " + error; });
133+
}
134+
135+
static bool InitSettings()
136+
{
137+
if (!gArgs.GetSettingsPath()) {
138+
return true; // Do nothing if settings file disabled.
139+
}
140+
141+
std::vector<std::string> errors;
142+
if (!gArgs.ReadSettingsFile(&errors)) {
143+
bilingual_str error = _("Settings file could not be read");
144+
InitError(Untranslated(strprintf("%s:\n%s\n", error.original, JoinErrors(errors))));
145+
146+
QMessageBox messagebox(QMessageBox::Critical, PACKAGE_NAME, QString::fromStdString(strprintf("%s.", error.translated)), QMessageBox::Reset | QMessageBox::Abort);
147+
/*: Explanatory text shown on startup when the settings file cannot be read.
148+
Prompts user to make a choice between resetting or aborting. */
149+
messagebox.setInformativeText(QObject::tr("Do you want to reset settings to default values, or to abort without making changes?"));
150+
messagebox.setDetailedText(QString::fromStdString(JoinErrors(errors)));
151+
messagebox.setTextFormat(Qt::PlainText);
152+
messagebox.setDefaultButton(QMessageBox::Reset);
153+
switch (messagebox.exec()) {
154+
case QMessageBox::Reset:
155+
break;
156+
case QMessageBox::Abort:
157+
return false;
158+
default:
159+
assert(false);
160+
}
161+
}
162+
163+
errors.clear();
164+
if (!gArgs.WriteSettingsFile(&errors)) {
165+
bilingual_str error = _("Settings file could not be written");
166+
InitError(Untranslated(strprintf("%s:\n%s\n", error.original, JoinErrors(errors))));
167+
168+
QMessageBox messagebox(QMessageBox::Critical, PACKAGE_NAME, QString::fromStdString(strprintf("%s.", error.translated)), QMessageBox::Ok);
169+
/*: Explanatory text shown on startup when the settings file could not be written.
170+
Prompts user to check that we have the ability to write to the file.
171+
Explains that the user has the option of running without a settings file.*/
172+
messagebox.setInformativeText(QObject::tr("A fatal error occured. Check that settings file is writable, or try running with -nosettings."));
173+
messagebox.setDetailedText(QString::fromStdString(JoinErrors(errors)));
174+
messagebox.setTextFormat(Qt::PlainText);
175+
messagebox.setDefaultButton(QMessageBox::Ok);
176+
messagebox.exec();
177+
return false;
178+
}
179+
return true;
180+
}
181+
130182
/* qDebug() message handler --> debug.log */
131183
void DebugMessageHandler(QtMsgType type, const QMessageLogContext& context, const QString &msg)
132184
{
@@ -308,11 +360,9 @@ void BitcoinApplication::parameterSetup()
308360
InitParameterInteraction(gArgs);
309361
}
310362

311-
void BitcoinApplication::InitializePruneSetting(bool prune)
363+
void BitcoinApplication::InitPruneSetting(int64_t prune_MiB)
312364
{
313-
// If prune is set, intentionally override existing prune size with
314-
// the default size since this is called when choosing a new datadir.
315-
optionsModel->SetPruneTargetGB(prune ? DEFAULT_PRUNE_TARGET_GB : 0, true);
365+
optionsModel->SetPruneTargetGB(PruneMiBtoGB(prune_MiB), true);
316366
}
317367

318368
void BitcoinApplication::requestInitialize()
@@ -555,9 +605,9 @@ int GuiMain(int argc, char* argv[])
555605
/// 5. Now that settings and translations are available, ask user for data directory
556606
// User language is set up: pick a data directory
557607
bool did_show_intro = false;
558-
bool prune = false; // Intro dialog prune check box
608+
int64_t prune_MiB = 0; // Intro dialog prune configuration
559609
// Gracefully exit if the user cancels
560-
if (!Intro::showIfNeeded(did_show_intro, prune)) return EXIT_SUCCESS;
610+
if (!Intro::showIfNeeded(did_show_intro, prune_MiB)) return EXIT_SUCCESS;
561611

562612
/// 6. Determine availability of data directory and parse dash.conf
563613
/// - Do not call GetDataDir(true) before this step finishes
@@ -592,9 +642,8 @@ int GuiMain(int argc, char* argv[])
592642
// Parse URIs on command line -- this can affect Params()
593643
PaymentServer::ipcParseCommandLine(argc, argv);
594644
#endif
595-
if (!gArgs.InitSettings(error)) {
596-
InitError(Untranslated(error));
597-
QMessageBox::critical(nullptr, PACKAGE_NAME, QObject::tr("Error initializing settings: %1").arg(QString::fromStdString(error)));
645+
646+
if (!InitSettings()) {
598647
return EXIT_FAILURE;
599648
}
600649

@@ -729,7 +778,7 @@ int GuiMain(int argc, char* argv[])
729778

730779
if (did_show_intro) {
731780
// Store intro dialog settings other than datadir (network specific)
732-
app.InitializePruneSetting(prune);
781+
app.InitPruneSetting(prune_MiB);
733782
}
734783

735784
if (gArgs.GetBoolArg("-splash", DEFAULT_SPLASHSCREEN) && !gArgs.GetBoolArg("-min", false))

src/qt/bitcoin.h

+1-1
Original file line numberDiff line numberDiff line change
@@ -68,7 +68,7 @@ class BitcoinApplication: public QApplication
6868
/// Create options model
6969
void createOptionsModel(bool resetSettings);
7070
/// Initialize prune setting
71-
void InitializePruneSetting(bool prune);
71+
void InitPruneSetting(int64_t prune_MiB);
7272
/// Create main window
7373
void createWindow(const NetworkStyle *networkStyle);
7474
/// Create splash screen

src/qt/forms/intro.ui

+42-11
Original file line numberDiff line numberDiff line change
@@ -7,7 +7,7 @@
77
<x>0</x>
88
<y>0</y>
99
<width>674</width>
10-
<height>415</height>
10+
<height>447</height>
1111
</rect>
1212
</property>
1313
<property name="windowTitle">
@@ -210,16 +210,6 @@
210210
</property>
211211
</widget>
212212
</item>
213-
<item>
214-
<widget class="QCheckBox" name="prune">
215-
<property name="toolTip">
216-
<string>Reverting this setting requires re-downloading the entire blockchain. It is faster to download the full chain first and prune it later. Disables some advanced features.</string>
217-
</property>
218-
<property name="text">
219-
<string></string>
220-
</property>
221-
</widget>
222-
</item>
223213
<item>
224214
<widget class="QLabel" name="lblExplanation2">
225215
<property name="text">
@@ -240,6 +230,47 @@
240230
</property>
241231
</widget>
242232
</item>
233+
<item>
234+
<layout class="QHBoxLayout" name="pruneOptLayout">
235+
<item>
236+
<widget class="QCheckBox" name="prune">
237+
<property name="text">
238+
<string>Limit block chain storage to</string>
239+
</property>
240+
<property name="toolTip">
241+
<string>Reverting this setting requires re-downloading the entire blockchain. It is faster to download the full chain first and prune it later. Disables some advanced features.</string>
242+
</property>
243+
</widget>
244+
</item>
245+
<item>
246+
<widget class="QSpinBox" name="pruneGB">
247+
<property name="suffix">
248+
<string> GB</string>
249+
</property>
250+
</widget>
251+
</item>
252+
<item>
253+
<widget class="QLabel" name="lblPruneSuffix">
254+
<property name="buddy">
255+
<cstring>pruneGB</cstring>
256+
</property>
257+
</widget>
258+
</item>
259+
<item>
260+
<spacer name="horizontalSpacer_2">
261+
<property name="orientation">
262+
<enum>Qt::Horizontal</enum>
263+
</property>
264+
<property name="sizeHint" stdset="0">
265+
<size>
266+
<width>40</width>
267+
<height>20</height>
268+
</size>
269+
</property>
270+
</spacer>
271+
</item>
272+
</layout>
273+
</item>
243274
<item>
244275
<spacer name="verticalSpacer">
245276
<property name="orientation">

src/qt/intro.cpp

+30-3
Original file line numberDiff line numberDiff line change
@@ -18,6 +18,7 @@
1818

1919
#include <interfaces/node.h>
2020
#include <util/system.h>
21+
#include <validation.h>
2122

2223
#include <QFileDialog>
2324
#include <QSettings>
@@ -140,17 +141,26 @@ Intro::Intro(QWidget *parent, int64_t blockchain_size_gb, int64_t chain_state_si
140141
);
141142
ui->lblExplanation2->setText(ui->lblExplanation2->text().arg(PACKAGE_NAME));
142143

144+
const int min_prune_target_GB = std::ceil(MIN_DISK_SPACE_FOR_BLOCK_FILES / 1e9);
145+
ui->pruneGB->setRange(min_prune_target_GB, std::numeric_limits<int>::max());
143146
if (gArgs.GetArg("-prune", 0) > 1) { // -prune=1 means enabled, above that it's a size in MiB
144147
ui->prune->setChecked(true);
145148
ui->prune->setEnabled(false);
146149
}
147-
ui->prune->setText(tr("Discard blocks after verification, except most recent %1 GB (prune)").arg(m_prune_target_gb));
150+
ui->pruneGB->setValue(m_prune_target_gb);
151+
ui->pruneGB->setToolTip(ui->prune->toolTip());
152+
ui->lblPruneSuffix->setToolTip(ui->prune->toolTip());
148153
UpdatePruneLabels(ui->prune->isChecked());
149154

150155
connect(ui->prune, &QCheckBox::toggled, [this](bool prune_checked) {
151156
UpdatePruneLabels(prune_checked);
152157
UpdateFreeSpaceLabel();
153158
});
159+
connect(ui->pruneGB, QOverload<int>::of(&QSpinBox::valueChanged), [this](int prune_GB) {
160+
m_prune_target_gb = prune_GB;
161+
UpdatePruneLabels(ui->prune->isChecked());
162+
UpdateFreeSpaceLabel();
163+
});
154164

155165
startThread();
156166
}
@@ -183,7 +193,17 @@ void Intro::setDataDirectory(const QString &dataDir)
183193
}
184194
}
185195

186-
bool Intro::showIfNeeded(bool& did_show_intro, bool& prune)
196+
int64_t Intro::getPruneMiB() const
197+
{
198+
switch (ui->prune->checkState()) {
199+
case Qt::Checked:
200+
return PruneGBtoMiB(m_prune_target_gb);
201+
case Qt::Unchecked: default:
202+
return 0;
203+
}
204+
}
205+
206+
bool Intro::showIfNeeded(bool& did_show_intro, int64_t& prune_MiB)
187207
{
188208
did_show_intro = false;
189209

@@ -238,7 +258,7 @@ bool Intro::showIfNeeded(bool& did_show_intro, bool& prune)
238258
}
239259

240260
// Additional preferences:
241-
prune = intro.ui->prune->isChecked();
261+
prune_MiB = intro.getPruneMiB();
242262

243263
settings.setValue("strDataDir", dataDir);
244264
settings.setValue("strDataDirDefault", dataDirDefaultCurrent);
@@ -367,6 +387,13 @@ void Intro::UpdatePruneLabels(bool prune_checked)
367387
storageRequiresMsg = tr("Approximately %1 GB of data will be stored in this directory.");
368388
}
369389
ui->lblExplanation3->setVisible(prune_checked);
390+
ui->pruneGB->setEnabled(prune_checked);
391+
static constexpr uint64_t nPowTargetSpacing = 2.5 * 60; // from chainparams, which we don't have at this stage
392+
static constexpr uint32_t expected_block_data_size = 2250000; // includes undo data
393+
const uint64_t expected_backup_days = m_prune_target_gb * 1e9 / (uint64_t(expected_block_data_size) * 86400 / nPowTargetSpacing);
394+
ui->lblPruneSuffix->setText(
395+
//: Explanatory text on the capability of the current prune target.
396+
tr("(sufficient to restore backups %n day(s) old)", "", expected_backup_days));
370397
ui->sizeWarningLabel->setText(
371398
tr("%1 will download and store a copy of the Dash block chain.").arg(PACKAGE_NAME) + " " +
372399
storageRequiresMsg.arg(m_required_space_gb) + " " +

src/qt/intro.h

+3-2
Original file line numberDiff line numberDiff line change
@@ -36,6 +36,7 @@ class Intro : public QDialog
3636

3737
QString getDataDirectory();
3838
void setDataDirectory(const QString &dataDir);
39+
int64_t getPruneMiB() const;
3940

4041
/**
4142
* Determine data directory. Let the user choose if the current one doesn't exist.
@@ -47,7 +48,7 @@ class Intro : public QDialog
4748
* @note do NOT call global GetDataDir() before calling this function, this
4849
* will cause the wrong path to be cached.
4950
*/
50-
static bool showIfNeeded(bool& did_show_intro, bool& prune);
51+
static bool showIfNeeded(bool& did_show_intro, int64_t& prune_MiB);
5152

5253
Q_SIGNALS:
5354
void requestCheck();
@@ -72,7 +73,7 @@ private Q_SLOTS:
7273
//! Total required space (in GB) depending on user choice (prune or not prune).
7374
int64_t m_required_space_gb{0};
7475
uint64_t m_bytes_available{0};
75-
const int64_t m_prune_target_gb;
76+
int64_t m_prune_target_gb;
7677

7778
void startThread();
7879
void checkPath(const QString &dataDir);

src/qt/psbtoperationsdialog.cpp

+3-2
Original file line numberDiff line numberDiff line change
@@ -141,11 +141,12 @@ void PSBTOperationsDialog::saveTransaction() {
141141
filename_suggestion.append(".psbt");
142142
QString filename = GUIUtil::getSaveFileName(this,
143143
tr("Save Transaction Data"), filename_suggestion,
144-
tr("Partially Signed Transaction (Binary)", "Name of binary PSBT file format") + QLatin1String(" (*.psbt)"), &selected_filter);
144+
//: Expanded name of the binary PSBT file format. See: BIP 174.
145+
tr("Partially Signed Transaction (Binary)") + QLatin1String(" (*.psbt)"), &selected_filter);
145146
if (filename.isEmpty()) {
146147
return;
147148
}
148-
std::ofstream out(filename.toLocal8Bit().data());
149+
std::ofstream out(filename.toLocal8Bit().data(), std::ofstream::out | std::ofstream::binary);
149150
out << ssTx.str();
150151
out.close();
151152
showStatus(tr("PSBT saved to disk."), StatusLevel::INFO);

0 commit comments

Comments
 (0)