1 changed files with 0 additions and 359 deletions
@ -1,359 +0,0 @@
|
||||
#!/usr/bin/env python |
||||
# |
||||
# Electrum - lightweight Bitcoin client |
||||
# Copyright (C) 2015 Thomas Voegtlin |
||||
# |
||||
# This program is free software: you can redistribute it and/or modify |
||||
# it under the terms of the GNU General Public License as published by |
||||
# the Free Software Foundation, either version 3 of the License, or |
||||
# (at your option) any later version. |
||||
# |
||||
# This program is distributed in the hope that it will be useful, |
||||
# but WITHOUT ANY WARRANTY; without even the implied warranty of |
||||
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the |
||||
# GNU General Public License for more details. |
||||
# |
||||
# You should have received a copy of the GNU General Public License |
||||
# along with this program. If not, see <http://www.gnu.org/licenses/>. |
||||
|
||||
|
||||
|
||||
|
||||
from __future__ import absolute_import |
||||
|
||||
import android |
||||
import sys |
||||
import os |
||||
import imp |
||||
import base64 |
||||
|
||||
script_dir = os.path.dirname(os.path.realpath(__file__)) |
||||
sys.path.insert(0, os.path.join(script_dir, 'packages')) |
||||
|
||||
import qrcode |
||||
|
||||
imp.load_module('electrum', *imp.find_module('lib')) |
||||
|
||||
from electrum import SimpleConfig, Wallet, WalletStorage, format_satoshis |
||||
from electrum import util |
||||
from electrum.transaction import Transaction |
||||
from electrum.bitcoin import base_encode, base_decode |
||||
|
||||
def modal_dialog(title, msg = None): |
||||
droid.dialogCreateAlert(title,msg) |
||||
droid.dialogSetPositiveButtonText('OK') |
||||
droid.dialogShow() |
||||
droid.dialogGetResponse() |
||||
droid.dialogDismiss() |
||||
|
||||
def modal_input(title, msg, value = None, etype=None): |
||||
droid.dialogCreateInput(title, msg, value, etype) |
||||
droid.dialogSetPositiveButtonText('OK') |
||||
droid.dialogSetNegativeButtonText('Cancel') |
||||
droid.dialogShow() |
||||
response = droid.dialogGetResponse() |
||||
result = response.result |
||||
droid.dialogDismiss() |
||||
|
||||
if result is None: |
||||
return modal_input(title, msg, value, etype) |
||||
|
||||
if result.get('which') == 'positive': |
||||
return result.get('value') |
||||
|
||||
def modal_question(q, msg, pos_text = 'OK', neg_text = 'Cancel'): |
||||
droid.dialogCreateAlert(q, msg) |
||||
droid.dialogSetPositiveButtonText(pos_text) |
||||
droid.dialogSetNegativeButtonText(neg_text) |
||||
droid.dialogShow() |
||||
response = droid.dialogGetResponse() |
||||
result = response.result |
||||
droid.dialogDismiss() |
||||
|
||||
if result is None: |
||||
return modal_question(q, msg, pos_text, neg_text) |
||||
|
||||
return result.get('which') == 'positive' |
||||
|
||||
|
||||
|
||||
|
||||
|
||||
def make_layout(s): |
||||
content = """ |
||||
|
||||
<LinearLayout |
||||
android:id="@+id/zz" |
||||
android:layout_width="match_parent" |
||||
android:layout_height="wrap_content" |
||||
android:background="#ff222222"> |
||||
|
||||
<TextView |
||||
android:id="@+id/textElectrum" |
||||
android:text="Electrum Authenticator" |
||||
android:textSize="7pt" |
||||
android:textColor="#ff4444ff" |
||||
android:gravity="left" |
||||
android:layout_height="wrap_content" |
||||
android:layout_width="match_parent" |
||||
/> |
||||
</LinearLayout> |
||||
|
||||
%s """%s |
||||
|
||||
|
||||
return """<?xml version="1.0" encoding="utf-8"?> |
||||
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android" |
||||
android:id="@+id/background" |
||||
android:orientation="vertical" |
||||
android:layout_width="match_parent" |
||||
android:layout_height="match_parent" |
||||
android:background="#ff000022"> |
||||
|
||||
%s |
||||
</LinearLayout>"""%content |
||||
|
||||
|
||||
|
||||
|
||||
|
||||
|
||||
def qr_layout(title): |
||||
title_view= """ |
||||
<TextView android:id="@+id/addrTextView" |
||||
android:layout_width="match_parent" |
||||
android:layout_height="50" |
||||
android:text="%s" |
||||
android:textAppearance="?android:attr/textAppearanceLarge" |
||||
android:gravity="center_vertical|center_horizontal|center"> |
||||
</TextView>"""%title |
||||
|
||||
image_view=""" |
||||
<ImageView |
||||
android:id="@+id/qrView" |
||||
android:gravity="center" |
||||
android:layout_width="match_parent" |
||||
android:antialias="false" |
||||
android:src="" |
||||
/> |
||||
""" |
||||
return make_layout(title_view + image_view) |
||||
|
||||
|
||||
|
||||
|
||||
|
||||
|
||||
|
||||
|
||||
|
||||
|
||||
def add_menu(): |
||||
droid.clearOptionsMenu() |
||||
droid.addOptionsMenuItem("Seed", "seed", None,"") |
||||
droid.addOptionsMenuItem("Public Key", "xpub", None,"") |
||||
droid.addOptionsMenuItem("Transaction", "scan", None,"") |
||||
droid.addOptionsMenuItem("Password", "password", None,"") |
||||
|
||||
|
||||
|
||||
def make_bitmap(data): |
||||
# fixme: this is highly inefficient |
||||
import qrcode |
||||
from electrum import bmp |
||||
qr = qrcode.QRCode() |
||||
qr.add_data(data) |
||||
bmp.save_qrcode(qr,"/sdcard/sl4a/qrcode.bmp") |
||||
|
||||
|
||||
droid = android.Android() |
||||
wallet = None |
||||
|
||||
class Authenticator: |
||||
|
||||
def __init__(self): |
||||
global wallet |
||||
self.qr_data = None |
||||
storage = WalletStorage('/sdcard/electrum/authenticator') |
||||
if not storage.file_exists: |
||||
|
||||
action = self.restore_or_create() |
||||
if not action: |
||||
exit() |
||||
password = droid.dialogGetPassword('Choose a password').result |
||||
if password: |
||||
password2 = droid.dialogGetPassword('Confirm password').result |
||||
if password != password2: |
||||
modal_dialog('Error', 'Passwords do not match') |
||||
exit() |
||||
else: |
||||
password = None |
||||
if action == 'create': |
||||
wallet = Wallet(storage) |
||||
seed = wallet.make_seed() |
||||
modal_dialog('Your seed is:', seed) |
||||
elif action == 'import': |
||||
seed = self.seed_dialog() |
||||
if not seed: |
||||
exit() |
||||
if not Wallet.is_seed(seed): |
||||
exit() |
||||
wallet = Wallet.from_seed(seed, password, storage) |
||||
else: |
||||
exit() |
||||
|
||||
wallet.add_seed(seed, password) |
||||
wallet.create_master_keys(password) |
||||
wallet.create_main_account(password) |
||||
else: |
||||
wallet = Wallet(storage) |
||||
|
||||
def restore_or_create(self): |
||||
droid.dialogCreateAlert("Seed not found", "Do you want to create a new seed, or to import it?") |
||||
droid.dialogSetPositiveButtonText('Create') |
||||
droid.dialogSetNeutralButtonText('Import') |
||||
droid.dialogSetNegativeButtonText('Cancel') |
||||
droid.dialogShow() |
||||
response = droid.dialogGetResponse().result |
||||
droid.dialogDismiss() |
||||
if not response: return |
||||
if response.get('which') == 'negative': |
||||
return |
||||
return 'import' if response.get('which') == 'neutral' else 'create' |
||||
|
||||
def seed_dialog(self): |
||||
if modal_question("Enter your seed", "Input method", 'QR Code', 'mnemonic'): |
||||
code = droid.scanBarcode() |
||||
r = code.result |
||||
if r: |
||||
seed = r['extras']['SCAN_RESULT'] |
||||
else: |
||||
return |
||||
else: |
||||
seed = modal_input('Mnemonic', 'Please enter your seed phrase') |
||||
return str(seed) |
||||
|
||||
def show_qr(self, data): |
||||
path = "/sdcard/sl4a/qrcode.bmp" |
||||
if data: |
||||
droid.dialogCreateSpinnerProgress("please wait") |
||||
droid.dialogShow() |
||||
try: |
||||
make_bitmap(data) |
||||
finally: |
||||
droid.dialogDismiss() |
||||
else: |
||||
with open(path, 'w') as f: f.write('') |
||||
droid.fullSetProperty("qrView", "src", 'file://'+path) |
||||
self.qr_data = data |
||||
|
||||
def show_title(self, title): |
||||
droid.fullSetProperty("addrTextView","text", title) |
||||
|
||||
def get_password(self): |
||||
if wallet.use_encryption: |
||||
password = droid.dialogGetPassword('Password').result |
||||
try: |
||||
wallet.check_password(password) |
||||
except: |
||||
return False |
||||
return password |
||||
|
||||
def main(self): |
||||
add_menu() |
||||
welcome = 'Use the menu to scan a transaction.' |
||||
droid.fullShow(qr_layout(welcome)) |
||||
while True: |
||||
event = droid.eventWait().result |
||||
if not event: |
||||
continue |
||||
elif event["name"] == "key": |
||||
if event["data"]["key"] == '4': |
||||
if self.qr_data: |
||||
self.show_qr(None) |
||||
self.show_title(welcome) |
||||
else: |
||||
break |
||||
|
||||
elif event["name"] == "seed": |
||||
password = self.get_password() |
||||
if password is False: |
||||
modal_dialog('Error','incorrect password') |
||||
continue |
||||
seed = wallet.get_mnemonic(password) |
||||
modal_dialog('Your seed is', seed) |
||||
|
||||
elif event["name"] == "password": |
||||
self.change_password_dialog() |
||||
|
||||
elif event["name"] == "xpub": |
||||
mpk = wallet.get_master_public_key() |
||||
self.show_qr(mpk) |
||||
self.show_title('master public key') |
||||
droid.setClipboard(mpk) |
||||
droid.makeToast("Master public key copied to clipboard") |
||||
|
||||
elif event["name"] == "scan": |
||||
r = droid.scanBarcode() |
||||
r = r.result |
||||
if not r: |
||||
continue |
||||
data = r['extras']['SCAN_RESULT'] |
||||
data = base_decode(data.encode('utf8'), None, base=43) |
||||
data = ''.join(chr(ord(b)) for b in data).encode('hex') |
||||
tx = Transaction(data) |
||||
#except: |
||||
# modal_dialog('Error', 'Cannot parse transaction') |
||||
# continue |
||||
if not wallet.can_sign(tx): |
||||
modal_dialog('Error', 'Cannot sign this transaction') |
||||
continue |
||||
lines = map(lambda x: x[0] + u'\t\t' + format_satoshis(x[1]) if x[1] else x[0], tx.get_outputs()) |
||||
if not modal_question('Sign?', '\n'.join(lines)): |
||||
continue |
||||
password = self.get_password() |
||||
if password is False: |
||||
modal_dialog('Error','incorrect password') |
||||
continue |
||||
droid.dialogCreateSpinnerProgress("Signing") |
||||
droid.dialogShow() |
||||
wallet.sign_transaction(tx, password) |
||||
droid.dialogDismiss() |
||||
data = base_encode(str(tx).decode('hex'), base=43) |
||||
self.show_qr(data) |
||||
self.show_title('Signed Transaction') |
||||
|
||||
droid.makeToast("Bye!") |
||||
|
||||
|
||||
def change_password_dialog(self): |
||||
if wallet.use_encryption: |
||||
password = droid.dialogGetPassword('Your seed is encrypted').result |
||||
if password is None: |
||||
return |
||||
else: |
||||
password = None |
||||
try: |
||||
wallet.check_password(password) |
||||
except Exception: |
||||
modal_dialog('Error', 'Incorrect password') |
||||
return |
||||
new_password = droid.dialogGetPassword('Choose a password').result |
||||
if new_password == None: |
||||
return |
||||
if new_password != '': |
||||
password2 = droid.dialogGetPassword('Confirm new password').result |
||||
if new_password != password2: |
||||
modal_dialog('Error', 'passwords do not match') |
||||
return |
||||
wallet.update_password(password, new_password) |
||||
if new_password: |
||||
modal_dialog('Password updated', 'Your seed is encrypted') |
||||
else: |
||||
modal_dialog('No password', 'Your seed is not encrypted') |
||||
|
||||
|
||||
|
||||
if __name__ == "__main__": |
||||
a = Authenticator() |
||||
a.main() |
||||
Loading…
Reference in new issue