walletcontroller.cpp
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 <qt/walletcontroller.h> 6 7 #include <qt/askpassphrasedialog.h> 8 #include <qt/clientmodel.h> 9 #include <qt/createwalletdialog.h> 10 #include <qt/guiconstants.h> 11 #include <qt/guiutil.h> 12 #include <qt/walletmodel.h> 13 14 #include <external_signer.h> 15 #include <interfaces/handler.h> 16 #include <interfaces/node.h> 17 #include <util/string.h> 18 #include <util/threadnames.h> 19 #include <util/translation.h> 20 #include <wallet/wallet.h> 21 22 #include <algorithm> 23 #include <chrono> 24 25 #include <QApplication> 26 #include <QMessageBox> 27 #include <QMetaObject> 28 #include <QMutexLocker> 29 #include <QThread> 30 #include <QTimer> 31 #include <QWindow> 32 33 using util::Join; 34 using wallet::WALLET_FLAG_BLANK_WALLET; 35 using wallet::WALLET_FLAG_DESCRIPTORS; 36 using wallet::WALLET_FLAG_DISABLE_PRIVATE_KEYS; 37 using wallet::WALLET_FLAG_EXTERNAL_SIGNER; 38 39 WalletController::WalletController(ClientModel& client_model, const PlatformStyle* platform_style, QObject* parent) 40 : QObject(parent) 41 , m_activity_thread(new QThread(this)) 42 , m_activity_worker(new QObject) 43 , m_client_model(client_model) 44 , m_node(client_model.node()) 45 , m_platform_style(platform_style) 46 , m_options_model(client_model.getOptionsModel()) 47 { 48 m_handler_load_wallet = m_node.walletLoader().handleLoadWallet([this](std::unique_ptr<interfaces::Wallet> wallet) { 49 getOrCreateWallet(std::move(wallet)); 50 }); 51 52 m_activity_worker->moveToThread(m_activity_thread); 53 m_activity_thread->start(); 54 QTimer::singleShot(0, m_activity_worker, []() { 55 util::ThreadRename("qt-walletctrl"); 56 }); 57 } 58 59 // Not using the default destructor because not all member types definitions are 60 // available in the header, just forward declared. 61 WalletController::~WalletController() 62 { 63 m_activity_thread->quit(); 64 m_activity_thread->wait(); 65 delete m_activity_worker; 66 } 67 68 std::map<std::string, std::pair<bool, std::string>> WalletController::listWalletDir() const 69 { 70 QMutexLocker locker(&m_mutex); 71 std::map<std::string, std::pair<bool, std::string>> wallets; 72 for (const auto& [name, format] : m_node.walletLoader().listWalletDir()) { 73 wallets[name] = std::make_pair(false, format); 74 } 75 for (WalletModel* wallet_model : m_wallets) { 76 auto it = wallets.find(wallet_model->wallet().getWalletName()); 77 if (it != wallets.end()) it->second.first = true; 78 } 79 return wallets; 80 } 81 82 void WalletController::removeWallet(WalletModel* wallet_model) 83 { 84 // Once the wallet is successfully removed from the node, the model will emit the 'WalletModel::unload' signal. 85 // This signal is already connected and will complete the removal of the view from the GUI. 86 // Look at 'WalletController::getOrCreateWallet' for the signal connection. 87 wallet_model->wallet().remove(); 88 } 89 90 void WalletController::closeWallet(WalletModel* wallet_model, QWidget* parent) 91 { 92 QMessageBox box(parent); 93 box.setWindowTitle(tr("Close wallet")); 94 box.setText(tr("Are you sure you wish to close the wallet <i>%1</i>?").arg(GUIUtil::HtmlEscape(wallet_model->getDisplayName()))); 95 box.setInformativeText(tr("Closing the wallet for too long can result in having to resync the entire chain if pruning is enabled.")); 96 box.setStandardButtons(QMessageBox::Yes|QMessageBox::Cancel); 97 box.setDefaultButton(QMessageBox::Yes); 98 if (box.exec() != QMessageBox::Yes) return; 99 100 removeWallet(wallet_model); 101 } 102 103 void WalletController::closeAllWallets(QWidget* parent) 104 { 105 QMessageBox::StandardButton button = QMessageBox::question(parent, tr("Close all wallets"), 106 tr("Are you sure you wish to close all wallets?"), 107 QMessageBox::Yes|QMessageBox::Cancel, 108 QMessageBox::Yes); 109 if (button != QMessageBox::Yes) return; 110 111 QMutexLocker locker(&m_mutex); 112 for (WalletModel* wallet_model : m_wallets) { 113 removeWallet(wallet_model); 114 } 115 } 116 117 WalletModel* WalletController::getOrCreateWallet(std::unique_ptr<interfaces::Wallet> wallet) 118 { 119 QMutexLocker locker(&m_mutex); 120 121 // Return model instance if exists. 122 if (!m_wallets.empty()) { 123 std::string name = wallet->getWalletName(); 124 for (WalletModel* wallet_model : m_wallets) { 125 if (wallet_model->wallet().getWalletName() == name) { 126 return wallet_model; 127 } 128 } 129 } 130 131 // Instantiate model and register it. 132 WalletModel* wallet_model = new WalletModel(std::move(wallet), m_client_model, m_platform_style, 133 nullptr /* required for the following moveToThread() call */); 134 135 // Move WalletModel object to the thread that created the WalletController 136 // object (GUI main thread), instead of the current thread, which could be 137 // an outside wallet thread or RPC thread sending a LoadWallet notification. 138 // This ensures queued signals sent to the WalletModel object will be 139 // handled on the GUI event loop. 140 wallet_model->moveToThread(thread()); 141 // setParent(parent) must be called in the thread which created the parent object. More details in #18948. 142 QMetaObject::invokeMethod(this, [wallet_model, this] { 143 wallet_model->setParent(this); 144 }, GUIUtil::blockingGUIThreadConnection()); 145 146 m_wallets.push_back(wallet_model); 147 148 // WalletModel::startPollBalance needs to be called in a thread managed by 149 // Qt because of startTimer. Considering the current thread can be a RPC 150 // thread, better delegate the calling to Qt with Qt::AutoConnection. 151 const bool called = QMetaObject::invokeMethod(wallet_model, "startPollBalance"); 152 assert(called); 153 154 connect(wallet_model, &WalletModel::unload, this, [this, wallet_model] { 155 // Defer removeAndDeleteWallet when no modal widget is actively waiting for an action. 156 // TODO: remove this workaround by removing usage of QDialog::exec. 157 QWidget* active_dialog = QApplication::activeModalWidget(); 158 if (active_dialog && dynamic_cast<QProgressDialog*>(active_dialog) == nullptr) { 159 connect(qApp, &QApplication::focusWindowChanged, wallet_model, [this, wallet_model]() { 160 if (!QApplication::activeModalWidget()) { 161 removeAndDeleteWallet(wallet_model); 162 } 163 }, Qt::QueuedConnection); 164 } else { 165 removeAndDeleteWallet(wallet_model); 166 } 167 }, Qt::QueuedConnection); 168 169 // Re-emit coinsSent signal from wallet model. 170 connect(wallet_model, &WalletModel::coinsSent, this, &WalletController::coinsSent); 171 172 Q_EMIT walletAdded(wallet_model); 173 174 return wallet_model; 175 } 176 177 void WalletController::removeAndDeleteWallet(WalletModel* wallet_model) 178 { 179 // Unregister wallet model. 180 { 181 QMutexLocker locker(&m_mutex); 182 m_wallets.erase(std::remove(m_wallets.begin(), m_wallets.end(), wallet_model)); 183 } 184 Q_EMIT walletRemoved(wallet_model); 185 // Currently this can trigger the unload since the model can hold the last 186 // CWallet shared pointer. 187 delete wallet_model; 188 } 189 190 WalletControllerActivity::WalletControllerActivity(WalletController* wallet_controller, QWidget* parent_widget) 191 : QObject(wallet_controller) 192 , m_wallet_controller(wallet_controller) 193 , m_parent_widget(parent_widget) 194 { 195 connect(this, &WalletControllerActivity::finished, this, &QObject::deleteLater); 196 } 197 198 void WalletControllerActivity::showProgressDialog(const QString& title_text, const QString& label_text, bool show_minimized) 199 { 200 auto progress_dialog = new QProgressDialog(m_parent_widget); 201 progress_dialog->setAttribute(Qt::WA_DeleteOnClose); 202 connect(this, &WalletControllerActivity::finished, progress_dialog, &QWidget::close); 203 204 progress_dialog->setWindowTitle(title_text); 205 progress_dialog->setLabelText(label_text); 206 progress_dialog->setRange(0, 0); 207 progress_dialog->setCancelButton(nullptr); 208 progress_dialog->setWindowModality(Qt::ApplicationModal); 209 GUIUtil::PolishProgressDialog(progress_dialog); 210 // The setValue call forces QProgressDialog to start the internal duration estimation. 211 // See details in https://bugreports.qt.io/browse/QTBUG-47042. 212 progress_dialog->setValue(0); 213 // When requested, launch dialog minimized 214 if (show_minimized) progress_dialog->showMinimized(); 215 } 216 217 CreateWalletActivity::CreateWalletActivity(WalletController* wallet_controller, QWidget* parent_widget) 218 : WalletControllerActivity(wallet_controller, parent_widget) 219 { 220 m_passphrase.reserve(MAX_PASSPHRASE_SIZE); 221 } 222 223 CreateWalletActivity::~CreateWalletActivity() 224 { 225 delete m_create_wallet_dialog; 226 delete m_passphrase_dialog; 227 } 228 229 void CreateWalletActivity::askPassphrase() 230 { 231 m_passphrase_dialog = new AskPassphraseDialog(AskPassphraseDialog::Encrypt, m_parent_widget, &m_passphrase); 232 m_passphrase_dialog->setWindowModality(Qt::ApplicationModal); 233 m_passphrase_dialog->show(); 234 235 connect(m_passphrase_dialog, &QObject::destroyed, [this] { 236 m_passphrase_dialog = nullptr; 237 }); 238 connect(m_passphrase_dialog, &QDialog::accepted, [this] { 239 createWallet(); 240 }); 241 connect(m_passphrase_dialog, &QDialog::rejected, [this] { 242 Q_EMIT finished(); 243 }); 244 } 245 246 void CreateWalletActivity::createWallet() 247 { 248 showProgressDialog( 249 //: Title of window indicating the progress of creation of a new wallet. 250 tr("Create Wallet"), 251 /*: Descriptive text of the create wallet progress window which indicates 252 to the user which wallet is currently being created. */ 253 tr("Creating Wallet <b>%1</b>…").arg(m_create_wallet_dialog->walletName().toHtmlEscaped())); 254 255 std::string name = m_create_wallet_dialog->walletName().toStdString(); 256 uint64_t flags = 0; 257 // Enable descriptors by default. 258 flags |= WALLET_FLAG_DESCRIPTORS; 259 if (m_create_wallet_dialog->isDisablePrivateKeysChecked()) { 260 flags |= WALLET_FLAG_DISABLE_PRIVATE_KEYS; 261 } 262 if (m_create_wallet_dialog->isMakeBlankWalletChecked()) { 263 flags |= WALLET_FLAG_BLANK_WALLET; 264 } 265 if (m_create_wallet_dialog->isExternalSignerChecked()) { 266 flags |= WALLET_FLAG_EXTERNAL_SIGNER; 267 } 268 269 QTimer::singleShot(500ms, worker(), [this, name, flags] { 270 auto wallet{node().walletLoader().createWallet(name, m_passphrase, flags, m_warning_message)}; 271 272 if (wallet) { 273 m_wallet_model = m_wallet_controller->getOrCreateWallet(std::move(*wallet)); 274 } else { 275 m_error_message = util::ErrorString(wallet); 276 } 277 278 QTimer::singleShot(500ms, this, &CreateWalletActivity::finish); 279 }); 280 } 281 282 void CreateWalletActivity::finish() 283 { 284 if (!m_error_message.empty()) { 285 QMessageBox::critical(m_parent_widget, tr("Create wallet failed"), QString::fromStdString(m_error_message.translated)); 286 } else if (!m_warning_message.empty()) { 287 QMessageBox::warning(m_parent_widget, tr("Create wallet warning"), QString::fromStdString(Join(m_warning_message, Untranslated("\n")).translated)); 288 } 289 290 if (m_wallet_model) Q_EMIT created(m_wallet_model); 291 292 Q_EMIT finished(); 293 } 294 295 void CreateWalletActivity::create() 296 { 297 m_create_wallet_dialog = new CreateWalletDialog(m_parent_widget); 298 299 std::vector<std::unique_ptr<interfaces::ExternalSigner>> signers; 300 try { 301 signers = node().listExternalSigners(); 302 } catch (const std::runtime_error& e) { 303 QMessageBox::critical(nullptr, tr("Can't list signers"), e.what()); 304 } 305 if (signers.size() > 1) { 306 QMessageBox::critical(nullptr, tr("Too many external signers found"), QString::fromStdString("More than one external signer found. Please connect only one at a time.")); 307 signers.clear(); 308 } 309 m_create_wallet_dialog->setSigners(signers); 310 311 m_create_wallet_dialog->setWindowModality(Qt::ApplicationModal); 312 m_create_wallet_dialog->show(); 313 314 connect(m_create_wallet_dialog, &QObject::destroyed, [this] { 315 m_create_wallet_dialog = nullptr; 316 }); 317 connect(m_create_wallet_dialog, &QDialog::rejected, [this] { 318 Q_EMIT finished(); 319 }); 320 connect(m_create_wallet_dialog, &QDialog::accepted, [this] { 321 if (m_create_wallet_dialog->isEncryptWalletChecked()) { 322 askPassphrase(); 323 } else { 324 createWallet(); 325 } 326 }); 327 } 328 329 OpenWalletActivity::OpenWalletActivity(WalletController* wallet_controller, QWidget* parent_widget) 330 : WalletControllerActivity(wallet_controller, parent_widget) 331 { 332 } 333 334 void OpenWalletActivity::finish() 335 { 336 if (!m_error_message.empty()) { 337 QMessageBox::critical(m_parent_widget, tr("Open wallet failed"), QString::fromStdString(m_error_message.translated)); 338 } else if (!m_warning_message.empty()) { 339 QMessageBox::warning(m_parent_widget, tr("Open wallet warning"), QString::fromStdString(Join(m_warning_message, Untranslated("\n")).translated)); 340 } 341 342 if (m_wallet_model) Q_EMIT opened(m_wallet_model); 343 344 Q_EMIT finished(); 345 } 346 347 void OpenWalletActivity::open(const std::string& path) 348 { 349 QString name = GUIUtil::WalletDisplayName(path); 350 351 showProgressDialog( 352 //: Title of window indicating the progress of opening of a wallet. 353 tr("Open Wallet"), 354 /*: Descriptive text of the open wallet progress window which indicates 355 to the user which wallet is currently being opened. */ 356 tr("Opening Wallet <b>%1</b>…").arg(name.toHtmlEscaped())); 357 358 QTimer::singleShot(0, worker(), [this, path] { 359 auto wallet{node().walletLoader().loadWallet(path, m_warning_message)}; 360 361 if (wallet) { 362 m_wallet_model = m_wallet_controller->getOrCreateWallet(std::move(*wallet)); 363 } else { 364 m_error_message = util::ErrorString(wallet); 365 } 366 367 QTimer::singleShot(0, this, &OpenWalletActivity::finish); 368 }); 369 } 370 371 LoadWalletsActivity::LoadWalletsActivity(WalletController* wallet_controller, QWidget* parent_widget) 372 : WalletControllerActivity(wallet_controller, parent_widget) 373 { 374 } 375 376 void LoadWalletsActivity::load(bool show_loading_minimized) 377 { 378 showProgressDialog( 379 //: Title of progress window which is displayed when wallets are being loaded. 380 tr("Load Wallets"), 381 /*: Descriptive text of the load wallets progress window which indicates to 382 the user that wallets are currently being loaded.*/ 383 tr("Loading wallets…"), 384 /*show_minimized=*/show_loading_minimized); 385 386 QTimer::singleShot(0, worker(), [this] { 387 for (auto& wallet : node().walletLoader().getWallets()) { 388 m_wallet_controller->getOrCreateWallet(std::move(wallet)); 389 } 390 391 QTimer::singleShot(0, this, [this] { Q_EMIT finished(); }); 392 }); 393 } 394 395 RestoreWalletActivity::RestoreWalletActivity(WalletController* wallet_controller, QWidget* parent_widget) 396 : WalletControllerActivity(wallet_controller, parent_widget) 397 { 398 } 399 400 void RestoreWalletActivity::restore(const fs::path& backup_file, const std::string& wallet_name) 401 { 402 QString name = QString::fromStdString(wallet_name); 403 404 showProgressDialog( 405 //: Title of progress window which is displayed when wallets are being restored. 406 tr("Restore Wallet"), 407 /*: Descriptive text of the restore wallets progress window which indicates to 408 the user that wallets are currently being restored.*/ 409 tr("Restoring Wallet <b>%1</b>…").arg(name.toHtmlEscaped())); 410 411 QTimer::singleShot(0, worker(), [this, backup_file, wallet_name] { 412 auto wallet{node().walletLoader().restoreWallet(backup_file, wallet_name, m_warning_message, /*load_after_restore=*/true)}; 413 414 if (wallet) { 415 m_wallet_model = m_wallet_controller->getOrCreateWallet(std::move(*wallet)); 416 } else { 417 m_error_message = util::ErrorString(wallet); 418 } 419 420 QTimer::singleShot(0, this, &RestoreWalletActivity::finish); 421 }); 422 } 423 424 void RestoreWalletActivity::finish() 425 { 426 if (!m_error_message.empty()) { 427 //: Title of message box which is displayed when the wallet could not be restored. 428 QMessageBox::critical(m_parent_widget, tr("Restore wallet failed"), QString::fromStdString(m_error_message.translated)); 429 } else if (!m_warning_message.empty()) { 430 //: Title of message box which is displayed when the wallet is restored with some warning. 431 QMessageBox::warning(m_parent_widget, tr("Restore wallet warning"), QString::fromStdString(Join(m_warning_message, Untranslated("\n")).translated)); 432 } else { 433 //: Title of message box which is displayed when the wallet is successfully restored. 434 QMessageBox::information(m_parent_widget, tr("Restore wallet message"), QString::fromStdString(Untranslated("Wallet restored successfully \n").translated)); 435 } 436 437 if (m_wallet_model) Q_EMIT restored(m_wallet_model); 438 439 Q_EMIT finished(); 440 } 441 442 void MigrateWalletActivity::do_migrate(const std::string& name) 443 { 444 SecureString passphrase; 445 if (node().walletLoader().isEncrypted(name)) { 446 // Get the passphrase for the wallet 447 AskPassphraseDialog dlg(AskPassphraseDialog::UnlockMigration, m_parent_widget, &passphrase); 448 if (dlg.exec() == QDialog::Rejected) return; 449 } 450 451 showProgressDialog(tr("Migrate Wallet"), tr("Migrating Wallet <b>%1</b>…").arg(GUIUtil::HtmlEscape(name))); 452 453 QTimer::singleShot(0, worker(), [this, name, passphrase] { 454 auto res{node().walletLoader().migrateWallet(name, passphrase)}; 455 456 if (res) { 457 m_success_message = tr("The wallet '%1' was migrated successfully.").arg(GUIUtil::HtmlEscape(GUIUtil::WalletDisplayName(name))); 458 if (res->watchonly_wallet_name) { 459 m_success_message += QChar(' ') + tr("Watchonly scripts have been migrated to a new wallet named '%1'.").arg(GUIUtil::HtmlEscape(GUIUtil::WalletDisplayName(res->watchonly_wallet_name.value()))); 460 } 461 if (res->solvables_wallet_name) { 462 m_success_message += QChar(' ') + tr("Solvable but not watched scripts have been migrated to a new wallet named '%1'.").arg(GUIUtil::HtmlEscape(GUIUtil::WalletDisplayName(res->solvables_wallet_name.value()))); 463 } 464 m_wallet_model = m_wallet_controller->getOrCreateWallet(std::move(res->wallet)); 465 } else { 466 m_error_message = util::ErrorString(res); 467 } 468 469 QTimer::singleShot(0, this, &MigrateWalletActivity::finish); 470 }); 471 } 472 473 void MigrateWalletActivity::migrate(const std::string& name) 474 { 475 // Warn the user about migration 476 QMessageBox box(m_parent_widget); 477 box.setWindowTitle(tr("Migrate wallet")); 478 box.setText(tr("Are you sure you wish to migrate the wallet <i>%1</i>?").arg(GUIUtil::HtmlEscape(GUIUtil::WalletDisplayName(name)))); 479 box.setInformativeText(tr("Migrating the wallet will convert this wallet to one or more descriptor wallets. A new wallet backup will need to be made.\n" 480 "If this wallet contains any watchonly scripts, a new wallet will be created which contains those watchonly scripts.\n" 481 "If this wallet contains any solvable but not watched scripts, a different and new wallet will be created which contains those scripts.\n\n" 482 "The migration process will create a backup of the wallet before migrating. This backup file will be named " 483 "<wallet name>-<timestamp>.legacy.bak and can be found in the directory for this wallet. In the event of " 484 "an incorrect migration, the backup can be restored with the \"Restore Wallet\" functionality.")); 485 box.setStandardButtons(QMessageBox::Yes|QMessageBox::Cancel); 486 box.setDefaultButton(QMessageBox::Yes); 487 if (box.exec() != QMessageBox::Yes) return; 488 489 do_migrate(name); 490 } 491 492 void MigrateWalletActivity::restore_and_migrate(const fs::path& path, const std::string& wallet_name) 493 { 494 // Warn the user about migration 495 QMessageBox box(m_parent_widget); 496 box.setWindowTitle(tr("Restore and Migrate wallet")); 497 box.setText(tr("Are you sure you wish to restore the wallet file <i>%1</i> to <i>%2</i> and migrate it?").arg(GUIUtil::HtmlEscape(fs::PathToString(path)), GUIUtil::HtmlEscape(GUIUtil::WalletDisplayName(wallet_name)))); 498 box.setInformativeText(tr("Restoring the wallet will copy the backup file to the wallets directory and place it in the standard " 499 "wallet directory layout. The original file will not be modified.\n\n" 500 "Migrating the wallet will convert the restored wallet to one or more descriptor wallets. A new wallet backup will need to be made.\n" 501 "If this wallet contains any watchonly scripts, a new wallet will be created which contains those watchonly scripts.\n" 502 "If this wallet contains any solvable but not watched scripts, a different and new wallet will be created which contains those scripts.\n\n" 503 "The migration process will create a backup of the wallet before migrating. This backup file will be named " 504 "<wallet name>-<timestamp>.legacy.bak and can be found in the directory for this wallet. In the event of " 505 "an incorrect migration, the backup can be restored with the \"Restore Wallet\" functionality.")); 506 box.setStandardButtons(QMessageBox::Yes|QMessageBox::Cancel); 507 box.setDefaultButton(QMessageBox::Yes); 508 if (box.exec() != QMessageBox::Yes) return; 509 510 showProgressDialog( 511 //: Title of progress window which is displayed when wallets are being restored. 512 tr("Restore Wallet"), 513 /*: Descriptive text of the restore wallets progress window which indicates to 514 the user that wallets are currently being restored.*/ 515 tr("Restoring Wallet <b>%1</b>…").arg(GUIUtil::HtmlEscape(GUIUtil::WalletDisplayName(wallet_name)))); 516 517 QTimer::singleShot(0, worker(), [this, path, wallet_name] { 518 auto res{node().walletLoader().restoreWallet(path, wallet_name, m_warning_message, /*load_after_restore=*/false)}; 519 520 if (!res) { 521 m_error_message = util::ErrorString(res); 522 QTimer::singleShot(0, this, &MigrateWalletActivity::finish); 523 return; 524 } 525 QTimer::singleShot(0, this, [this, wallet_name] { 526 do_migrate(wallet_name); 527 }); 528 }); 529 } 530 531 void MigrateWalletActivity::finish() 532 { 533 if (!m_error_message.empty()) { 534 QMessageBox::critical(m_parent_widget, tr("Migration failed"), QString::fromStdString(m_error_message.translated)); 535 } else { 536 QMessageBox::information(m_parent_widget, tr("Migration Successful"), m_success_message); 537 } 538 539 if (m_wallet_model) Q_EMIT migrated(m_wallet_model); 540 541 Q_EMIT finished(); 542 }