from PyQt4.QtGui import * from PyQt4.QtCore import * import PyQt4.QtCore as QtCore from i18n import _ from electrum import Wallet, mnemonic from seed_dialog import SeedDialog from network_dialog import NetworkDialog from qt_util import * from amountedit import AmountEdit import sys class InstallWizard(QDialog): def __init__(self, config, network, storage): QDialog.__init__(self) self.config = config self.network = network self.interface = network.interface self.blockchain = network.blockchain self.storage = storage def restore_or_create(self): d = QDialog() d.setModal(1) grid = QGridLayout() grid.setSpacing(5) msg = _("Wallet file not found.")+"\n"+_("Do you want to create a new wallet, or to restore an existing one?") label = QLabel(msg) label.setWordWrap(True) grid.addWidget(label, 0, 0) gb = QGroupBox() b1 = QRadioButton(gb) b1.setText(_("Create new wallet")) b1.setChecked(True) b2 = QRadioButton(gb) b2.setText(_("Restore wallet from seed")) b3 = QRadioButton(gb) b3.setText(_("Restore wallet from master public key")) grid.addWidget(b1,1,0) grid.addWidget(b2,2,0) grid.addWidget(b3,3,0) vbox = QVBoxLayout() vbox.addLayout(grid) vbox.addLayout(ok_cancel_buttons(d, _('Next'))) d.setLayout(vbox) if not d.exec_(): return if b1.isChecked(): return 'create' elif b2.isChecked(): return 'restore' else: return 'watching' def verify_seed(self, wallet): r = self.seed_dialog(False) if not r: return if r != wallet.seed: QMessageBox.warning(None, _('Error'), 'incorrect seed', 'OK') return False else: return True def seed_dialog(self, is_restore=True): d = QDialog() d.setModal(1) vbox = QVBoxLayout() if is_restore: msg = _("Please enter your wallet seed (or your master public key if you want to create a watching-only wallet)." + ' ') else: msg = _("Your seed is important! To make sure that you have properly saved your seed, please type it here." + ' ') msg += _("Your seed can be entered as a sequence of words, or as a hexadecimal string."+ '\n') label=QLabel(msg) label.setWordWrap(True) vbox.addWidget(label) seed_e = QTextEdit() seed_e.setMaximumHeight(100) vbox.addWidget(seed_e) if is_restore: grid = QGridLayout() grid.setSpacing(8) gap_e = AmountEdit(None, True) gap_e.setText("5") grid.addWidget(QLabel(_('Gap limit')), 2, 0) grid.addWidget(gap_e, 2, 1) grid.addWidget(HelpButton(_('Keep the default value unless you modified this parameter in your wallet.')), 2, 3) vbox.addLayout(grid) vbox.addLayout(ok_cancel_buttons(d, _('Next'))) d.setLayout(vbox) if not d.exec_(): return try: seed = str(seed_e.toPlainText()) seed.decode('hex') except: try: seed = mnemonic.mn_decode( seed.split() ) except: QMessageBox.warning(None, _('Error'), _('I cannot decode this'), _('OK')) return if not seed: QMessageBox.warning(None, _('Error'), _('No seed'), _('OK')) return if not is_restore: return seed else: try: gap = int(unicode(gap_e.text())) except: QMessageBox.warning(None, _('Error'), 'error', 'OK') return return seed, gap def mpk_dialog(self): d = QDialog() d.setModal(1) vbox = QVBoxLayout() msg = _("Please enter your master public key.") label=QLabel(msg) label.setWordWrap(True) vbox.addWidget(label) mpk_e = QTextEdit() mpk_e.setMaximumHeight(100) vbox.addWidget(mpk_e) grid = QGridLayout() grid.setSpacing(8) gap_e = AmountEdit(None, True) gap_e.setText("5") grid.addWidget(QLabel(_('Gap limit')), 2, 0) grid.addWidget(gap_e, 2, 1) grid.addWidget(HelpButton(_('Keep the default value unless you modified this parameter in your wallet.')), 2, 3) vbox.addLayout(grid) vbox.addLayout(ok_cancel_buttons(d, _('Next'))) d.setLayout(vbox) if not d.exec_(): return mpk = str(mpk_e.toPlainText()) try: gap = int(unicode(gap_e.text())) except: QMessageBox.warning(None, _('Error'), 'error', 'OK') return return mpk, gap def network_dialog(self): d = QDialog() d.setModal(1) grid = QGridLayout() grid.setSpacing(5) label = QLabel(_("Network") + ":") grid.addWidget(label, 0, 0) gb = QGroupBox() b1 = QRadioButton(gb) b1.setText(_("Auto connect")) b1.setChecked(True) b2 = QRadioButton(gb) b2.setText(_("Select server manually")) b3 = QRadioButton(gb) b3.setText(_("Stay offline")) grid.addWidget(b1,1,0) grid.addWidget(b2,2,0) grid.addWidget(b3,3,0) vbox = QVBoxLayout() vbox.addLayout(grid) vbox.addLayout(ok_cancel_buttons(d, _('Next'))) d.setLayout(vbox) if not d.exec_(): return if b2.isChecked(): return NetworkDialog(self.interface, self.config, None).do_exec() elif b1.isChecked(): self.config.set_key('auto_cycle', True, True) return else: self.config.set_key("server", None, True) self.config.set_key('auto_cycle', False, True) return def show_seed(self, wallet): d = SeedDialog() d.show_seed(wallet.seed, wallet.imported_keys) def password_dialog(self, wallet): from password_dialog import PasswordDialog d = PasswordDialog(wallet) d.run() def restore_wallet(self, wallet): # wait until we are connected, because the user might have selected another server if not wallet.interface.is_connected: waiting = lambda: False if wallet.interface.is_connected else "%s \n" % (_("Connecting...")) waiting_dialog(waiting) waiting = lambda: False if wallet.is_up_to_date() else "%s\n%s %d\n%s %.1f"\ %(_("Please wait..."),_("Addresses generated:"),len(wallet.addresses(True)),_("Kilobytes received:"), wallet.interface.bytes_received/1024.) # try to restore old account wallet.create_old_account() wallet.set_up_to_date(False) wallet.interface.poke('synchronizer') waiting_dialog(waiting) if wallet.is_found(): wallet.seed_version = 4 wallet.storage.put('seed_version', wallet.seed_version, True) else: wallet.accounts.pop(0) wallet.create_accounts() wallet.set_up_to_date(False) wallet.interface.poke('synchronizer') waiting_dialog(waiting) if wallet.is_found(): QMessageBox.information(None, _('Information'), _("Recovery successful"), _('OK')) else: QMessageBox.information(None, _('Information'), _("No transactions found for this seed"), _('OK')) return True def run(self): action = self.restore_or_create() if not action: exit() wallet = Wallet(self.storage) if action == 'create': wallet.init_seed(None) self.show_seed(wallet) if self.verify_seed(wallet): wallet.save_seed() wallet.create_accounts() # generate first addresses offline wallet.synchronize() else: return elif action == 'restore': # ask for seed and gap. sg = self.seed_dialog() if not sg: return seed, gap = sg if not seed: return wallet.gap_limit = gap wallet.init_seed(str(seed)) wallet.save_seed() elif action == 'watching': # ask for seed and gap. sg = self.mpk_dialog() if not sg: return mpk, gap = sg if not mpk: return wallet.gap_limit = gap wallet.seed = '' print eval(mpk) try: c0, K0 = eval(mpk) except: QMessageBox.warning(None, _('Error'), _('error'), _('OK')) return wallet.create_watching_only_wallet(c0,K0) else: raise #if not self.config.get('server'): self.network_dialog() # start wallet threads wallet.start_threads(self.network) if action == 'restore': try: keep_it = self.restore_wallet(wallet) wallet.fill_addressbook() except: import traceback traceback.print_exc(file=sys.stdout) exit() if not keep_it: return self.password_dialog(wallet) return wallet