2018-06-21 14:28:34 +00:00
|
|
|
# This file is part of the Trezor project.
|
2017-01-03 18:40:05 +00:00
|
|
|
#
|
2018-06-21 14:28:34 +00:00
|
|
|
# Copyright (C) 2012-2018 SatoshiLabs and contributors
|
2017-01-03 18:40:05 +00:00
|
|
|
#
|
|
|
|
# This library is free software: you can redistribute it and/or modify
|
2018-06-21 14:28:34 +00:00
|
|
|
# it under the terms of the GNU Lesser General Public License version 3
|
|
|
|
# as published by the Free Software Foundation.
|
2017-01-03 18:40:05 +00:00
|
|
|
#
|
|
|
|
# This library 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 Lesser General Public License for more details.
|
|
|
|
#
|
2018-06-21 14:28:34 +00:00
|
|
|
# You should have received a copy of the License along with this library.
|
|
|
|
# If not, see <https://www.gnu.org/licenses/lgpl-3.0.html>.
|
|
|
|
|
2018-05-23 15:08:16 +00:00
|
|
|
import pytest
|
2017-01-03 18:40:05 +00:00
|
|
|
|
2018-08-13 16:21:24 +00:00
|
|
|
from trezorlib import device, messages as proto
|
|
|
|
|
2018-05-23 15:08:16 +00:00
|
|
|
from .common import TrezorTest
|
2014-02-21 01:33:23 +00:00
|
|
|
|
2017-06-23 19:31:42 +00:00
|
|
|
|
2017-12-19 18:24:18 +00:00
|
|
|
@pytest.mark.skip_t2
|
2017-12-23 20:20:49 +00:00
|
|
|
class TestMsgRecoverydevice(TrezorTest):
|
2014-02-22 01:15:21 +00:00
|
|
|
def test_pin_passphrase(self):
|
2018-08-13 16:21:24 +00:00
|
|
|
mnemonic = self.mnemonic12.split(" ")
|
|
|
|
ret = self.client.call_raw(
|
|
|
|
proto.RecoveryDevice(
|
|
|
|
word_count=12,
|
|
|
|
passphrase_protection=True,
|
|
|
|
pin_protection=True,
|
|
|
|
label="label",
|
|
|
|
language="english",
|
|
|
|
enforce_wordlist=True,
|
|
|
|
)
|
|
|
|
)
|
2014-02-21 01:33:23 +00:00
|
|
|
|
2018-05-23 15:08:16 +00:00
|
|
|
# click through confirmation
|
|
|
|
assert isinstance(ret, proto.ButtonRequest)
|
|
|
|
self.client.debug.press_yes()
|
|
|
|
ret = self.client.call_raw(proto.ButtonAck())
|
|
|
|
|
2017-12-23 20:20:49 +00:00
|
|
|
assert isinstance(ret, proto.PinMatrixRequest)
|
2014-02-22 01:15:21 +00:00
|
|
|
|
|
|
|
# Enter PIN for first time
|
|
|
|
pin_encoded = self.client.debug.encode_pin(self.pin6)
|
|
|
|
ret = self.client.call_raw(proto.PinMatrixAck(pin=pin_encoded))
|
2017-12-23 20:20:49 +00:00
|
|
|
assert isinstance(ret, proto.PinMatrixRequest)
|
2014-02-22 01:15:21 +00:00
|
|
|
|
|
|
|
# Enter PIN for second time
|
|
|
|
pin_encoded = self.client.debug.encode_pin(self.pin6)
|
|
|
|
ret = self.client.call_raw(proto.PinMatrixAck(pin=pin_encoded))
|
|
|
|
|
|
|
|
fakes = 0
|
2014-07-10 16:07:00 +00:00
|
|
|
for _ in range(int(12 * 2)):
|
2017-12-23 20:20:49 +00:00
|
|
|
assert isinstance(ret, proto.WordRequest)
|
2014-03-07 20:51:48 +00:00
|
|
|
(word, pos) = self.client.debug.read_recovery_word()
|
2014-02-22 01:15:21 +00:00
|
|
|
|
2014-03-07 16:25:55 +00:00
|
|
|
if pos != 0:
|
|
|
|
ret = self.client.call_raw(proto.WordAck(word=mnemonic[pos - 1]))
|
|
|
|
mnemonic[pos - 1] = None
|
2014-02-22 01:15:21 +00:00
|
|
|
else:
|
2014-03-07 16:25:55 +00:00
|
|
|
ret = self.client.call_raw(proto.WordAck(word=word))
|
2014-02-22 01:15:21 +00:00
|
|
|
fakes += 1
|
|
|
|
|
2016-05-05 01:16:17 +00:00
|
|
|
print(mnemonic)
|
2014-02-22 01:15:21 +00:00
|
|
|
|
|
|
|
# Workflow succesfully ended
|
2017-12-23 20:20:49 +00:00
|
|
|
assert isinstance(ret, proto.Success)
|
2014-02-22 01:15:21 +00:00
|
|
|
|
2014-07-10 16:07:00 +00:00
|
|
|
# 12 expected fake words and all words of mnemonic are used
|
2017-12-23 20:20:49 +00:00
|
|
|
assert fakes == 12
|
|
|
|
assert mnemonic == [None] * 12
|
2014-02-22 01:15:21 +00:00
|
|
|
|
|
|
|
# Mnemonic is the same
|
|
|
|
self.client.init_device()
|
2019-03-12 12:22:47 +00:00
|
|
|
assert self.client.debug.read_mnemonic_secret() == self.mnemonic12.encode()
|
2014-02-22 01:15:21 +00:00
|
|
|
|
2017-12-23 20:20:49 +00:00
|
|
|
assert self.client.features.pin_protection is True
|
|
|
|
assert self.client.features.passphrase_protection is True
|
2016-01-12 23:17:38 +00:00
|
|
|
|
2014-02-22 01:15:21 +00:00
|
|
|
# Do passphrase-protected action, PassphraseRequest should be raised
|
|
|
|
resp = self.client.call_raw(proto.Ping(passphrase_protection=True))
|
2017-12-23 20:20:49 +00:00
|
|
|
assert isinstance(resp, proto.PassphraseRequest)
|
2014-03-07 21:38:16 +00:00
|
|
|
self.client.call_raw(proto.Cancel())
|
2014-02-22 01:15:21 +00:00
|
|
|
|
|
|
|
def test_nopin_nopassphrase(self):
|
2018-08-13 16:21:24 +00:00
|
|
|
mnemonic = self.mnemonic12.split(" ")
|
|
|
|
ret = self.client.call_raw(
|
|
|
|
proto.RecoveryDevice(
|
|
|
|
word_count=12,
|
|
|
|
passphrase_protection=False,
|
|
|
|
pin_protection=False,
|
|
|
|
label="label",
|
|
|
|
language="english",
|
|
|
|
enforce_wordlist=True,
|
|
|
|
)
|
|
|
|
)
|
2014-02-22 01:15:21 +00:00
|
|
|
|
2018-05-23 15:08:16 +00:00
|
|
|
# click through confirmation
|
|
|
|
assert isinstance(ret, proto.ButtonRequest)
|
|
|
|
self.client.debug.press_yes()
|
|
|
|
ret = self.client.call_raw(proto.ButtonAck())
|
|
|
|
|
2014-02-22 01:15:21 +00:00
|
|
|
fakes = 0
|
2014-07-10 16:07:00 +00:00
|
|
|
for _ in range(int(12 * 2)):
|
2017-12-23 20:20:49 +00:00
|
|
|
assert isinstance(ret, proto.WordRequest)
|
2014-03-07 20:51:48 +00:00
|
|
|
(word, pos) = self.client.debug.read_recovery_word()
|
2014-02-22 01:15:21 +00:00
|
|
|
|
2014-03-07 16:25:55 +00:00
|
|
|
if pos != 0:
|
|
|
|
ret = self.client.call_raw(proto.WordAck(word=mnemonic[pos - 1]))
|
|
|
|
mnemonic[pos - 1] = None
|
2014-02-22 01:15:21 +00:00
|
|
|
else:
|
2014-03-07 16:25:55 +00:00
|
|
|
ret = self.client.call_raw(proto.WordAck(word=word))
|
2014-02-22 01:15:21 +00:00
|
|
|
fakes += 1
|
|
|
|
|
2016-05-05 01:16:17 +00:00
|
|
|
print(mnemonic)
|
2014-02-22 01:15:21 +00:00
|
|
|
|
|
|
|
# Workflow succesfully ended
|
2017-12-23 20:20:49 +00:00
|
|
|
assert isinstance(ret, proto.Success)
|
2014-02-22 01:15:21 +00:00
|
|
|
|
2014-07-10 16:07:00 +00:00
|
|
|
# 12 expected fake words and all words of mnemonic are used
|
2017-12-23 20:20:49 +00:00
|
|
|
assert fakes == 12
|
|
|
|
assert mnemonic == [None] * 12
|
2014-02-22 01:15:21 +00:00
|
|
|
|
|
|
|
# Mnemonic is the same
|
|
|
|
self.client.init_device()
|
2019-03-12 12:22:47 +00:00
|
|
|
assert self.client.debug.read_mnemonic_secret() == self.mnemonic12.encode()
|
2014-02-22 01:15:21 +00:00
|
|
|
|
2017-12-23 20:20:49 +00:00
|
|
|
assert self.client.features.pin_protection is False
|
|
|
|
assert self.client.features.passphrase_protection is False
|
2014-02-22 01:15:21 +00:00
|
|
|
|
|
|
|
# Do passphrase-protected action, PassphraseRequest should NOT be raised
|
|
|
|
resp = self.client.call_raw(proto.Ping(passphrase_protection=True))
|
2017-12-23 20:20:49 +00:00
|
|
|
assert isinstance(resp, proto.Success)
|
2014-02-22 01:15:21 +00:00
|
|
|
|
|
|
|
# Do PIN-protected action, PinRequest should NOT be raised
|
|
|
|
resp = self.client.call_raw(proto.Ping(pin_protection=True))
|
2017-12-23 20:20:49 +00:00
|
|
|
assert isinstance(resp, proto.Success)
|
2016-01-12 23:17:38 +00:00
|
|
|
|
2014-03-07 20:51:01 +00:00
|
|
|
def test_word_fail(self):
|
2018-08-13 16:21:24 +00:00
|
|
|
ret = self.client.call_raw(
|
|
|
|
proto.RecoveryDevice(
|
|
|
|
word_count=12,
|
|
|
|
passphrase_protection=False,
|
|
|
|
pin_protection=False,
|
|
|
|
label="label",
|
|
|
|
language="english",
|
|
|
|
enforce_wordlist=True,
|
|
|
|
)
|
|
|
|
)
|
2014-03-07 20:51:01 +00:00
|
|
|
|
2018-05-23 15:08:16 +00:00
|
|
|
# click through confirmation
|
|
|
|
assert isinstance(ret, proto.ButtonRequest)
|
|
|
|
self.client.debug.press_yes()
|
|
|
|
ret = self.client.call_raw(proto.ButtonAck())
|
|
|
|
|
2017-12-23 20:20:49 +00:00
|
|
|
assert isinstance(ret, proto.WordRequest)
|
2014-07-10 16:07:00 +00:00
|
|
|
for _ in range(int(12 * 2)):
|
2014-03-07 20:51:01 +00:00
|
|
|
(word, pos) = self.client.debug.read_recovery_word()
|
|
|
|
if pos != 0:
|
2018-08-13 16:21:24 +00:00
|
|
|
ret = self.client.call_raw(proto.WordAck(word="kwyjibo"))
|
2017-12-23 20:20:49 +00:00
|
|
|
assert isinstance(ret, proto.Failure)
|
2014-03-07 21:38:16 +00:00
|
|
|
break
|
2014-03-07 20:51:01 +00:00
|
|
|
else:
|
2014-03-07 21:38:16 +00:00
|
|
|
self.client.call_raw(proto.WordAck(word=word))
|
2014-03-07 20:51:01 +00:00
|
|
|
|
2014-02-22 01:15:21 +00:00
|
|
|
def test_pin_fail(self):
|
2018-08-13 16:21:24 +00:00
|
|
|
ret = self.client.call_raw(
|
|
|
|
proto.RecoveryDevice(
|
|
|
|
word_count=12,
|
|
|
|
passphrase_protection=True,
|
|
|
|
pin_protection=True,
|
|
|
|
label="label",
|
|
|
|
language="english",
|
|
|
|
enforce_wordlist=True,
|
|
|
|
)
|
|
|
|
)
|
2014-02-22 01:15:21 +00:00
|
|
|
|
2018-05-23 15:08:16 +00:00
|
|
|
# click through confirmation
|
|
|
|
assert isinstance(ret, proto.ButtonRequest)
|
|
|
|
self.client.debug.press_yes()
|
|
|
|
ret = self.client.call_raw(proto.ButtonAck())
|
|
|
|
|
2017-12-23 20:20:49 +00:00
|
|
|
assert isinstance(ret, proto.PinMatrixRequest)
|
2014-02-22 01:15:21 +00:00
|
|
|
|
|
|
|
# Enter PIN for first time
|
|
|
|
pin_encoded = self.client.debug.encode_pin(self.pin4)
|
|
|
|
ret = self.client.call_raw(proto.PinMatrixAck(pin=pin_encoded))
|
2017-12-23 20:20:49 +00:00
|
|
|
assert isinstance(ret, proto.PinMatrixRequest)
|
2014-02-22 01:15:21 +00:00
|
|
|
|
|
|
|
# Enter PIN for second time, but different one
|
|
|
|
pin_encoded = self.client.debug.encode_pin(self.pin6)
|
|
|
|
ret = self.client.call_raw(proto.PinMatrixAck(pin=pin_encoded))
|
|
|
|
|
|
|
|
# Failure should be raised
|
2017-12-23 20:20:49 +00:00
|
|
|
assert isinstance(ret, proto.Failure)
|
2014-02-22 01:15:21 +00:00
|
|
|
|
|
|
|
def test_already_initialized(self):
|
|
|
|
self.setup_mnemonic_nopin_nopassphrase()
|
2018-10-03 12:19:53 +00:00
|
|
|
with pytest.raises(RuntimeError):
|
|
|
|
device.recover(
|
|
|
|
self.client,
|
|
|
|
12,
|
|
|
|
False,
|
|
|
|
False,
|
|
|
|
"label",
|
|
|
|
"english",
|
|
|
|
self.client.mnemonic_callback,
|
|
|
|
)
|