forked from Toporin/Satochip-Utils
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathcontroller.py
316 lines (279 loc) · 14.3 KB
/
controller.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
import logging
from os import urandom
import sys
from mnemonic import Mnemonic
from pysatochip.CardConnector import (CardConnector, UninitializedSeedError)
seed = None
if (len(sys.argv) >= 2) and (sys.argv[1] in ['-v', '--verbose']):
logging.basicConfig(level=logging.DEBUG,
format='%(asctime)s - [%(filename)s:%(lineno)d] - %(levelname)s - %(name)s - %(funcName)s() - %(message)s',
datefmt='%Y-%m-%d %H:%M:%S')
else:
logging.basicConfig(level=logging.INFO,
format='%(asctime)s - [%(filename)s:%(lineno)d] - %(levelname)s - %(name)s - %(funcName)s() - %(message)s',
datefmt='%Y-%m-%d %H:%M:%S')
logger = logging.getLogger(__name__)
logger.setLevel(logging.DEBUG)
class Controller:
def __init__(self, cc, view, loglevel=logging.INFO):
logger.setLevel(loglevel)
self.view = view
self.view.controller = self
try:
self.cc = CardConnector(self, loglevel=loglevel)
logger.info("CardConnector initialized successfully.")
except Exception as e:
logger.error("Failed to initialize CardConnector.", exc_info=True)
raise
# card infos
self.card_status = None
def get_card_status(self):
if self.cc.card_present:
logger.info("In get_card_status")
try:
response, sw1, sw2, self.card_status = self.cc.card_get_status()
if self.card_status:
logger.debug(f"Card satus: {self.card_status}")
else:
logger.error(f"Failed to retrieve card_status")
self.card_status['applet_full_version_string'] = f"{self.card_status['protocol_major_version']}.{self.card_status['protocol_minor_version']}-{self.card_status['applet_major_version']}.{self.card_status['applet_minor_version']}"
return self.card_status
except Exception as e:
logger.error(f"Failed to retrieve card status: {e}")
self.card_status = None
return None
def request(self, request_type, *args):
logger.info(str(request_type))
method_to_call = getattr(self.view, request_type)
reply = method_to_call(*args)
return reply
# def disconnect_the_card(self):
# self.cc.card_disconnect()
def setup_card_pin(self, pin, pin_confirm):
if pin:
if len(pin) >= 4:
if pin == pin_confirm:
logger.info("Setup my card PIN: PINs match and are valid.")
self.card_setup_native_pin(pin)
else:
logger.warning("Setup my card PIN: PINs do not match.")
self.view.show('ERROR', "Pin and pin confirm do not match!", 'Ok',
None, "./pictures_db/icon_change_pin_popup.jpg")
else:
logger.warning("Setup my card PIN: PIN is too short.")
self.view.show("ERROR",
"Pin must contain at least 4 characters",
'Ok', None,
"./pictures_db/icon_change_pin_popup.jpg")
else:
self.view.show("ERROR", "You have to set up a PIN to continue.", 'Ok',
None, "./pictures_db/icon_change_pin_popup.jpg")
def change_card_pin(self, current_pin, new_pin, new_pin_confirm):
try:
if self.cc.card_present and self.cc.card_type != "Satodime":
if len(new_pin) < 4:
logger.warning("New PIN is too short.")
self.view.show("ERROR",
"Pin must contain at least 4 characters", 'Ok',
None, "./pictures_db/icon_change_pin_popup.jpg")
if new_pin != new_pin_confirm:
logger.warning("New PINs do not match.")
self.view.show("WARNING",
"The PIN values do not match! Please type PIN again!",
"Ok", None,
"./pictures_db/icon_change_pin_popup.jpg")
else:
current_pin = list(current_pin.encode('utf8'))
new_pin = list(new_pin.encode('utf8'))
(response, sw1, sw2) = self.cc.card_change_PIN(0, current_pin, new_pin)
if sw1 == 0x90 and sw2 == 0x00:
logger.info("PIN changed successfully.")
msg = "PIN changed successfully!"
self.view.show("SUCCESS", msg, 'Ok',
None, "./pictures_db/icon_change_pin_popup.jpg")
self.view.start_setup()
else:
logger.error(f"Failed to change PIN with error code: {hex(sw1)}{hex(sw2)}")
msg = f"Failed to change PIN with error code: {hex(sw1)}{hex(sw2)}"
self.view.show("ERROR", f"{msg}\n Probably too long", 'Ok',
None, "./pictures_db/icon_change_pin_popup.jpg")
except Exception as e:
logger.error(f"Error changing PIN: {e}")
self.view.show("ERROR", "Failed to change PIN.", "Ok",
None, "./pictures_db/icon_change_pin_popup.jpg")
def generate_random_seed(self, mnemonic_length):
try:
logger.info(f"In generate_random_seed(), mnemonic_length: {mnemonic_length}")
strength = 128 if mnemonic_length == 12 else 256 if mnemonic_length == 24 else None
if strength:
MNEMONIC = Mnemonic(language="english")
mnemonic = MNEMONIC.generate(strength=strength)
return mnemonic
else:
logger.warning(f"generate_random_seed: invalid mnemonic length {mnemonic_length}")
return f"Error: invalid mnemonic length {mnemonic_length}"
except Exception as e:
logger.error(f"generate_random_seed: Error generating seed: {e}")
return f"Exception: {e}"
def import_seed(self, mnemonic, passphrase =None):
"""Import a seed (and optional passphrase) into a Satochip"""
try:
MNEMONIC = Mnemonic(language="english")
if MNEMONIC.check(mnemonic): # check that seed is valid
logger.info("Imported seed is valid.")
if passphrase is not None:
if passphrase in ["", " ", "Type your passphrase here"]:
logger.error("Passphrase is blank or empy")
self.view.show('WARNING', 'Wrong passphrase: incorrect or blank', 'Ok')
else:
seed = Mnemonic.to_seed(mnemonic, passphrase)
self.card_setup_native_seed(seed)
else:
seed = Mnemonic.to_seed(mnemonic)
self.card_setup_native_seed(seed)
else:
logger.warning("Imported seed is invalid!")
self.view.show('WARNING',
"Warning!\nInvalid BIP39 seedphrase, please retry.",
'Ok', None,
"./pictures_db/icon_seed_popup.jpg")
except Exception as e:
logger.error(f"Error while importing seed: {e}")
self.view.show("ERROR", "Failed to import seed.", "Ok", None,
"./pictures_db/icon_seed_popup.jpg")
def edit_label(self, label):
try:
logger.info(f"New label to set: {label}")
(response, sw1, sw2) = self.cc.card_set_label(label)
if sw1 == 0x90 and sw2 == 0x00:
logger.info(f"New label set successfully: {label}")
self.view.show("SUCCESS",
f"New label set successfully",
"Ok", self.view.start_setup(),
"./pictures_db/icon_edit_label_popup.jpg")
else:
logger.warning("Failed to set new label.")
self.view.show("ERROR", f"Failed to set label (code {hex(sw1*256+sw2)})", "oK",
None, "./pictures_db/icon_edit_label_popup.jpg")
except Exception as e:
logger.error(f"Failed to edit label: {e}")
self.view.show("ERROR", f"Failed to edit label: {e}", "Ok", None,
"./pictures_db/icon_edit_label_popup.jpg")
def get_card_label_infos(self):
"""Get label info"""
if self.cc.card_present:
response, sw1, sw2, label = self.cc.card_get_label()
if label is None:
logger.info("Label is None")
return None
if label == "":
logger.info("Label is Blank")
return ""
else:
logger.info(f"Label found: {label}")
return label
else:
logger.info("In get_card_label_infos: No card present")
return None
# for PIN
def PIN_dialog(self, msg):
try:
logger.info("Entering PIN_dialog method")
def switch_unlock_to_false_and_quit():
self.view.start_setup()
self.view.update_status()
while True:
try:
logger.debug("Requesting PIN")
pin = self.view.get_passphrase(msg)
logger.debug(f"PIN received: pin={'***' if pin else None}")
if pin is None:
logger.info("PIN request cancelled or window closed")
self.view.show("INFO",
'Device cannot be unlocked without PIN code!',
'Ok',
lambda: switch_unlock_to_false_and_quit(),
"./pictures_db/icon_change_pin_popup.jpg")
break
elif len(pin) < 4:
logger.warning("PIN length is less than 4 characters")
msg = "PIN must have at least 4 characters."
self.view.show("INFO", msg, 'Ok', None,
"./pictures_db/icon_change_pin_popup.jpg")
elif len(pin) > 16:
logger.warning("PIN length is more than 16 characters")
msg = "PIN must have maximum 16 characters."
self.view.show("INFO", msg, 'Ok', None,
"./pictures_db/icon_change_pin_popup.jpg")
else:
logger.info("PIN length is valid")
pin = pin.encode('utf8')
try:
self.cc.card_verify_PIN_simple(pin)
break
except Exception as e:
logger.info(f"exception from PIN dialog: {e}")
self.view.show('ERROR', str(e), 'Ok', None,
"./pictures_db/icon_change_pin_popup.jpg")
except Exception as e:
logger.error(f"An error occurred while requesting PIN: {e}", exc_info=True)
except Exception as e:
logger.critical(f"An unexpected error occurred in PIN_dialog: {e}", exc_info=True)
# only for satochip and seedkeeper
def card_setup_native_pin(self, pin):
try:
logger.info("In card_setup_native_pin")
logger.info("Setting up card pin and applet references")
pin_0 = list(pin.encode('utf8'))
pin_tries_0 = 0x05
ublk_tries_0 = 0x01
ublk_0 = list(urandom(16)) # PUK code
pin_tries_1 = 0x01
ublk_tries_1 = 0x01
pin_1 = list(urandom(16)) # Second pin
ublk_1 = list(urandom(16))
secmemsize = 32 # Number of slots reserved in memory cache
memsize = 0x0000 # RFU
create_object_ACL = 0x01 # RFU
create_key_ACL = 0x01 # RFU
create_pin_ACL = 0x01 # RFU
logger.info("Sending setup native pin command to card")
(response, sw1, sw2) = self.cc.card_setup(pin_tries_0, ublk_tries_0, pin_0, ublk_0,
pin_tries_1, ublk_tries_1, pin_1, ublk_1,
secmemsize, memsize,
create_object_ACL, create_key_ACL, create_pin_ACL)
logger.info(f"Response from card: {response}, sw1: {hex(sw1)}, sw2: {hex(sw2)}")
if sw1 != 0x90 or sw2 != 0x00:
logger.warning(f"Unable to set up applet! sw12={hex(sw1)} {hex(sw2)}")
self.view.show('ERROR', f"Unable to set up applet! sw12={hex(sw1)} {hex(sw2)}")
return False
else:
logger.info("Applet setup successfully")
self.setup_done = True
self.view.update_status()
self.view.start_setup()
except Exception as e:
logger.error(f"An error occurred in card_setup_native_pin: {e}", exc_info=True)
# only for satochip
def card_setup_native_seed(self, seed):
# get authentikey
try:
authentikey = self.cc.card_bip32_get_authentikey()
except UninitializedSeedError:
# seed dialog...
authentikey = self.cc.card_bip32_import_seed(seed)
logger.info(f"authentikey: {authentikey}")
if authentikey:
self.is_seeded = True
self.view.show('SUCCESS',
'Your card is now seeded!',
'Ok',
lambda: None,
"./pictures_db/icon_seed_popup.jpg")
self.view.update_status()
self.view.start_setup()
hex_authentikey = authentikey.get_public_key_hex()
logger.info(f"Authentikey={hex_authentikey}")
else:
self.view.show('ERROR', 'Error when importing seed to Satochip!', 'Ok', None,
"./pictures_db/icon_seed_popup.jpg")