From 254ac399e5f60dfd6d7c85106de238c8ce5ca2c5 Mon Sep 17 00:00:00 2001 From: Sander van Grieken Date: Fri, 8 Mar 2024 15:34:06 +0100 Subject: [PATCH] add wallet rename functionality, and add to qml client --- electrum/daemon.py | 15 ++++ .../gui/qml/components/RenameWalletDialog.qml | 87 +++++++++++++++++++ electrum/gui/qml/components/WalletDetails.qml | 17 ++++ electrum/gui/qml/qedaemon.py | 40 ++++++++- electrum/gui/qml/qewizard.py | 29 +------ electrum/wallet.py | 32 ++++++- 6 files changed, 191 insertions(+), 29 deletions(-) create mode 100644 electrum/gui/qml/components/RenameWalletDialog.qml diff --git a/electrum/daemon.py b/electrum/daemon.py index b98592d9f760..d8a0aeecc4d9 100644 --- a/electrum/daemon.py +++ b/electrum/daemon.py @@ -510,6 +510,21 @@ def get_wallet(self, path: str) -> Optional[Abstract_Wallet]: def get_wallets(self) -> Dict[str, Abstract_Wallet]: return dict(self._wallets) # copy + @with_wallet_lock + def rename_wallet(self, wallet: 'Abstract_Wallet', new_basename): + if not wallet.db.storage: + return + + path = wallet.db.storage.path + wallet_key = self._wallet_key_from_path(path) + wallet = self._wallets.get(wallet_key) + assert wallet + + new_path = wallet.rename_wallet(new_basename) + new_wallet_key = self._wallet_key_from_path(new_path) + self._wallets.pop(wallet_key) + self._wallets[new_wallet_key] = wallet + def delete_wallet(self, path: str) -> bool: self.stop_wallet(path) if os.path.exists(path): diff --git a/electrum/gui/qml/components/RenameWalletDialog.qml b/electrum/gui/qml/components/RenameWalletDialog.qml new file mode 100644 index 000000000000..e334e0e33beb --- /dev/null +++ b/electrum/gui/qml/components/RenameWalletDialog.qml @@ -0,0 +1,87 @@ +import QtQuick +import QtQuick.Layouts +import QtQuick.Controls +import QtQuick.Controls.Material + +import org.electrum 1.0 + +import "controls" + +ElDialog { + id: dialog + + title: qsTr("Enter wallet name") + iconSource: Qt.resolvedUrl('../../icons/pen.png') + + property string infotext + + property bool _valid: false + + anchors.centerIn: parent + width: parent.width * 4/5 + padding: 0 + + ColumnLayout { + id: rootLayout + width: parent.width + spacing: 0 + + ColumnLayout { + Layout.leftMargin: constants.paddingXXLarge + Layout.rightMargin: constants.paddingXXLarge + + InfoTextArea { + visible: infotext + text: infotext + Layout.bottomMargin: constants.paddingMedium + Layout.fillWidth: true + } + + Label { + Layout.fillWidth: true + text: qsTr('Wallet name') + color: Material.accentColor + } + + TextField { + id: wallet_name + Layout.fillWidth: true + Layout.leftMargin: constants.paddingXLarge + Layout.rightMargin: constants.paddingXLarge + onTextChanged: { + var name = text.trim() + if (!text || text == Daemon.currentWallet.name) { + _valid = false + infotext = '' + } else { + _valid = Daemon.isValidNewWalletName(name) + if (_valid) + infotext = '' + else + infotext = qsTr('Invalid name') + } + } + } + } + + FlatButton { + Layout.fillWidth: true + text: qsTr("Ok") + icon.source: '../../icons/confirmed.png' + enabled: _valid + onClicked: { + var name = wallet_name.text.trim() + if (Daemon.isValidNewWalletName(name)) { + console.log('renaming.. ' + name) + var result = Daemon.renameWallet(Daemon.currentWallet, name) + if (result) + dialog.close() + } + } + } + } + + Component.onCompleted: { + wallet_name.text = Daemon.currentWallet.name + } +} diff --git a/electrum/gui/qml/components/WalletDetails.qml b/electrum/gui/qml/components/WalletDetails.qml index 967789f74b45..8932a0d07a29 100644 --- a/electrum/gui/qml/components/WalletDetails.qml +++ b/electrum/gui/qml/components/WalletDetails.qml @@ -433,6 +433,16 @@ Pane { onClicked: Daemon.checkThenDeleteWallet(Daemon.currentWallet) icon.source: '../../icons/delete.png' } + FlatButton { + Layout.fillWidth: true + Layout.preferredWidth: 1 + text: qsTr('Rename') + onClicked: { + var dialog = renameWalletDialog.createObject(rootItem) + dialog.open() + } + icon.source: '../../icons/pen.png' + } FlatButton { Layout.fillWidth: true Layout.preferredWidth: 1 @@ -554,6 +564,13 @@ Pane { } } + Component { + id: renameWalletDialog + RenameWalletDialog { + onClosed: destroy() + } + } + Binding { target: AppController property: 'secureWindow' diff --git a/electrum/gui/qml/qedaemon.py b/electrum/gui/qml/qedaemon.py index 7f7c57b9d09d..bec8efe8797e 100644 --- a/electrum/gui/qml/qedaemon.py +++ b/electrum/gui/qml/qedaemon.py @@ -13,7 +13,7 @@ from electrum.lnchannel import ChannelState from electrum.bitcoin import is_address from electrum.bitcoin import verify_usermessage_with_address -from electrum.storage import StorageReadWriteError +from electrum.storage import StorageReadWriteError, WalletStorage from .auth import AuthMixin, auth_protect from .qefx import QEFX @@ -383,3 +383,41 @@ def passwordStrength(self, password): if len(password) == 0: return 0 return check_password_strength(password)[0] + + def wallet_path_from_wallet_name(self, wallet_name: str) -> str: + return os.path.join(self.daemon.config.get_datadir_wallet_path(), wallet_name) + + @pyqtSlot(str, result=bool) + def isValidNewWalletName(self, wallet_name: str) -> bool: + if not wallet_name: + return False + if self.availableWallets.wallet_name_exists(wallet_name): + return False + wallet_path = self.wallet_path_from_wallet_name(wallet_name) + # note: we should probably restrict wallet names to be alphanumeric (plus underscore, etc)... + # wallet_name might contain ".." (etc) and hence sketchy path traversals are possible. + # Anyway, this at least validates that the path looks sane to the filesystem: + try: + temp_storage = WalletStorage(wallet_path) + except (StorageReadWriteError, WalletFileException) as e: + return False + except Exception as e: + self._logger.exception("") + return False + if temp_storage.file_exists(): + return False + return True + + @pyqtSlot(QEWallet, str, result=bool) + def renameWallet(self, wallet, new_name): + self._logger.debug(f'renaming wallet to: {new_name}') + try: + self.daemon.rename_wallet(wallet.wallet, new_name) + self.daemon.config.save_last_wallet(wallet.wallet) + wallet.nameChanged.emit() + if self._available_wallets: + self._available_wallets.reload() + return True + except Exception as e: + self._logger.debug(f'renaming err: {str(e)}') + return False diff --git a/electrum/gui/qml/qewizard.py b/electrum/gui/qml/qewizard.py index 7ca8841ffe4d..e5fbb8e084e3 100644 --- a/electrum/gui/qml/qewizard.py +++ b/electrum/gui/qml/qewizard.py @@ -119,34 +119,9 @@ def verifySeed(self, seed, seed_variant, wallet_type='standard'): 'can_passphrase': can_passphrase } - def _wallet_path_from_wallet_name(self, wallet_name: str) -> str: - return os.path.join(self._qedaemon.daemon.config.get_datadir_wallet_path(), wallet_name) - @pyqtSlot(str, result=bool) def isValidNewWalletName(self, wallet_name: str) -> bool: - if not wallet_name: - return False - if self._qedaemon.availableWallets.wallet_name_exists(wallet_name): - return False - wallet_path = self._wallet_path_from_wallet_name(wallet_name) - # note: we should probably restrict wallet names to be alphanumeric (plus underscore, etc)... - # try to prevent sketchy path traversals: - for forbidden_char in ("/", "\\", ): - if forbidden_char in wallet_name: - return False - if os.path.basename(wallet_name) != wallet_name: - return False - # validate that the path looks sane to the filesystem: - try: - temp_storage = WalletStorage(wallet_path) - except (StorageReadWriteError, WalletFileException) as e: - return False - except Exception as e: - self._logger.exception("") - return False - if temp_storage.file_exists(): - return False - return True + return self._qedaemon.isValidNewWalletName(wallet_name) @pyqtSlot('QJSValue', bool, str) def createStorage(self, js_data, single_password_enabled, single_password): @@ -157,7 +132,7 @@ def createStorage(self, js_data, single_password_enabled, single_password): data['encrypt'] = True data['password'] = single_password - path = self._wallet_path_from_wallet_name(data['wallet_name']) + path = self._qedaemon.wallet_path_from_wallet_name(data['wallet_name']) try: self.create_storage(path, data) diff --git a/electrum/wallet.py b/electrum/wallet.py index 2af3a0e25096..1989a6b32a8d 100644 --- a/electrum/wallet.py +++ b/electrum/wallet.py @@ -401,7 +401,6 @@ def __init__(self, db: WalletDB, *, config: SimpleConfig): self.config = config assert self.config is not None, "config must not be None" self.db = db - self.storage = db.storage # type: Optional[WalletStorage] # load addresses needs to be called before constructor for sanity checks db.load_addresses(self.wallet_type) self.keystore = None # type: Optional[KeyStore] # will be set by load_keystore @@ -456,6 +455,10 @@ def __init__(self, db: WalletDB, *, config: SimpleConfig): self.register_callbacks() + @property + def storage(self): + return self.db.storage + def _init_lnworker(self): self.lnworker = None @@ -505,6 +508,33 @@ def save_backup(self, backup_dir): new_db.write() return new_path + def rename_wallet(self, new_basename) -> str: + old_storage = self.storage + if not old_storage: + return + + # new storage path + dirname = os.path.dirname(old_storage.path) + new_path = os.path.join(dirname, new_basename) + assert not os.path.exists(new_path) + + # new storage + new_storage = WalletStorage(new_path) + new_storage._encryption_version = old_storage._encryption_version + new_storage.pubkey = old_storage.pubkey + + with self.db.lock: + # keep existing WalletDB instance, as the initialization normally + # happens in the wallet constructor, which we don't want to replicate, + self.db.storage = new_storage + self.db.set_modified(True) + self.db.write_and_force_consolidation() + + assert os.path.exists(new_path) + os.unlink(old_storage.path) + + return new_path + def has_lightning(self) -> bool: return bool(self.lnworker)