2018-06-21 14:28:34 +00:00
|
|
|
# This file is part of the Trezor project.
|
2017-01-03 18:40:05 +00:00
|
|
|
#
|
2019-05-29 16:44:09 +00:00
|
|
|
# Copyright (C) 2012-2019 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
|
|
|
|
|
2019-09-11 12:43:32 +00:00
|
|
|
from ..common import MNEMONIC12
|
2019-09-11 12:29:39 +00:00
|
|
|
|
|
|
|
PIN4 = "1234"
|
|
|
|
PIN6 = "789456"
|
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
|
2019-09-11 12:29:39 +00:00
|
|
|
class TestMsgRecoverydevice:
|
2019-08-27 14:58:59 +00:00
|
|
|
@pytest.mark.setup_client(uninitialized=True)
|
|
|
|
def test_pin_passphrase(self, client):
|
2019-09-11 12:29:39 +00:00
|
|
|
mnemonic = MNEMONIC12.split(" ")
|
2019-08-27 14:58:59 +00:00
|
|
|
ret = client.call_raw(
|
2018-08-13 16:21:24 +00:00
|
|
|
proto.RecoveryDevice(
|
|
|
|
word_count=12,
|
|
|
|
passphrase_protection=True,
|
|
|
|
pin_protection=True,
|
|
|
|
label="label",
|
2019-12-07 11:11:51 +00:00
|
|
|
language="en-US",
|
2018-08-13 16:21:24 +00:00
|
|
|
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)
|
2019-08-27 14:58:59 +00:00
|
|
|
client.debug.press_yes()
|
|
|
|
ret = client.call_raw(proto.ButtonAck())
|
2018-05-23 15:08:16 +00:00
|
|
|
|
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
|
2019-09-11 12:29:39 +00:00
|
|
|
pin_encoded = client.debug.encode_pin(PIN6)
|
2019-08-27 14:58:59 +00:00
|
|
|
ret = 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
|
2019-09-11 12:29:39 +00:00
|
|
|
pin_encoded = client.debug.encode_pin(PIN6)
|
2019-08-27 14:58:59 +00:00
|
|
|
ret = client.call_raw(proto.PinMatrixAck(pin=pin_encoded))
|
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)
|
2019-08-27 14:58:59 +00:00
|
|
|
(word, pos) = client.debug.read_recovery_word()
|
2014-02-22 01:15:21 +00:00
|
|
|
|
2014-03-07 16:25:55 +00:00
|
|
|
if pos != 0:
|
2019-08-27 14:58:59 +00:00
|
|
|
ret = client.call_raw(proto.WordAck(word=mnemonic[pos - 1]))
|
2014-03-07 16:25:55 +00:00
|
|
|
mnemonic[pos - 1] = None
|
2014-02-22 01:15:21 +00:00
|
|
|
else:
|
2019-08-27 14:58:59 +00:00
|
|
|
ret = client.call_raw(proto.WordAck(word=word))
|
2014-02-22 01:15:21 +00:00
|
|
|
fakes += 1
|
|
|
|
|
|
|
|
# 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
|
2019-08-27 14:58:59 +00:00
|
|
|
client.init_device()
|
2020-04-24 10:16:17 +00:00
|
|
|
assert client.debug.state().mnemonic_secret == MNEMONIC12.encode()
|
2014-02-22 01:15:21 +00:00
|
|
|
|
2019-08-27 14:58:59 +00:00
|
|
|
assert client.features.pin_protection is True
|
|
|
|
assert 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
|
2020-01-31 15:58:01 +00:00
|
|
|
resp = client.call_raw(proto.GetAddress())
|
2017-12-23 20:20:49 +00:00
|
|
|
assert isinstance(resp, proto.PassphraseRequest)
|
2019-08-27 14:58:59 +00:00
|
|
|
client.call_raw(proto.Cancel())
|
2014-02-22 01:15:21 +00:00
|
|
|
|
2019-08-27 14:58:59 +00:00
|
|
|
@pytest.mark.setup_client(uninitialized=True)
|
|
|
|
def test_nopin_nopassphrase(self, client):
|
2019-09-11 12:29:39 +00:00
|
|
|
mnemonic = MNEMONIC12.split(" ")
|
2019-08-27 14:58:59 +00:00
|
|
|
ret = client.call_raw(
|
2018-08-13 16:21:24 +00:00
|
|
|
proto.RecoveryDevice(
|
|
|
|
word_count=12,
|
|
|
|
passphrase_protection=False,
|
|
|
|
pin_protection=False,
|
|
|
|
label="label",
|
2019-12-07 11:11:51 +00:00
|
|
|
language="en-US",
|
2018-08-13 16:21:24 +00:00
|
|
|
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)
|
2019-08-27 14:58:59 +00:00
|
|
|
client.debug.press_yes()
|
|
|
|
ret = client.call_raw(proto.ButtonAck())
|
2018-05-23 15:08:16 +00:00
|
|
|
|
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)
|
2019-08-27 14:58:59 +00:00
|
|
|
(word, pos) = client.debug.read_recovery_word()
|
2014-02-22 01:15:21 +00:00
|
|
|
|
2014-03-07 16:25:55 +00:00
|
|
|
if pos != 0:
|
2019-08-27 14:58:59 +00:00
|
|
|
ret = client.call_raw(proto.WordAck(word=mnemonic[pos - 1]))
|
2014-03-07 16:25:55 +00:00
|
|
|
mnemonic[pos - 1] = None
|
2014-02-22 01:15:21 +00:00
|
|
|
else:
|
2019-08-27 14:58:59 +00:00
|
|
|
ret = client.call_raw(proto.WordAck(word=word))
|
2014-02-22 01:15:21 +00:00
|
|
|
fakes += 1
|
|
|
|
|
|
|
|
# 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
|
2019-08-27 14:58:59 +00:00
|
|
|
client.init_device()
|
2020-04-24 10:16:17 +00:00
|
|
|
assert client.debug.state().mnemonic_secret == MNEMONIC12.encode()
|
2014-02-22 01:15:21 +00:00
|
|
|
|
2019-08-27 14:58:59 +00:00
|
|
|
assert client.features.pin_protection is False
|
|
|
|
assert client.features.passphrase_protection is False
|
2014-02-22 01:15:21 +00:00
|
|
|
|
2020-01-21 09:05:48 +00:00
|
|
|
# Do pin & passphrase-protected action, PassphraseRequest should NOT be raised
|
2020-01-31 15:58:01 +00:00
|
|
|
resp = client.call_raw(proto.GetAddress())
|
2020-01-21 09:05:48 +00:00
|
|
|
assert isinstance(resp, proto.Address)
|
2016-01-12 23:17:38 +00:00
|
|
|
|
2019-08-27 14:58:59 +00:00
|
|
|
@pytest.mark.setup_client(uninitialized=True)
|
|
|
|
def test_word_fail(self, client):
|
|
|
|
ret = client.call_raw(
|
2018-08-13 16:21:24 +00:00
|
|
|
proto.RecoveryDevice(
|
|
|
|
word_count=12,
|
|
|
|
passphrase_protection=False,
|
|
|
|
pin_protection=False,
|
|
|
|
label="label",
|
2019-12-07 11:11:51 +00:00
|
|
|
language="en-US",
|
2018-08-13 16:21:24 +00:00
|
|
|
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)
|
2019-08-27 14:58:59 +00:00
|
|
|
client.debug.press_yes()
|
|
|
|
ret = client.call_raw(proto.ButtonAck())
|
2018-05-23 15:08:16 +00:00
|
|
|
|
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)):
|
2019-08-27 14:58:59 +00:00
|
|
|
(word, pos) = client.debug.read_recovery_word()
|
2014-03-07 20:51:01 +00:00
|
|
|
if pos != 0:
|
2019-08-27 14:58:59 +00:00
|
|
|
ret = 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:
|
2019-08-27 14:58:59 +00:00
|
|
|
client.call_raw(proto.WordAck(word=word))
|
2014-03-07 20:51:01 +00:00
|
|
|
|
2019-08-27 14:58:59 +00:00
|
|
|
@pytest.mark.setup_client(uninitialized=True)
|
|
|
|
def test_pin_fail(self, client):
|
|
|
|
ret = client.call_raw(
|
2018-08-13 16:21:24 +00:00
|
|
|
proto.RecoveryDevice(
|
|
|
|
word_count=12,
|
|
|
|
passphrase_protection=True,
|
|
|
|
pin_protection=True,
|
|
|
|
label="label",
|
2019-12-07 11:11:51 +00:00
|
|
|
language="en-US",
|
2018-08-13 16:21:24 +00:00
|
|
|
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)
|
2019-08-27 14:58:59 +00:00
|
|
|
client.debug.press_yes()
|
|
|
|
ret = client.call_raw(proto.ButtonAck())
|
2018-05-23 15:08:16 +00:00
|
|
|
|
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
|
2019-09-11 12:29:39 +00:00
|
|
|
pin_encoded = client.debug.encode_pin(PIN4)
|
2019-08-27 14:58:59 +00:00
|
|
|
ret = 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
|
2019-09-11 12:29:39 +00:00
|
|
|
pin_encoded = client.debug.encode_pin(PIN6)
|
2019-08-27 14:58:59 +00:00
|
|
|
ret = client.call_raw(proto.PinMatrixAck(pin=pin_encoded))
|
2014-02-22 01:15:21 +00:00
|
|
|
|
|
|
|
# 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
|
|
|
|
2019-08-27 14:58:59 +00:00
|
|
|
def test_already_initialized(self, client):
|
2018-10-03 12:19:53 +00:00
|
|
|
with pytest.raises(RuntimeError):
|
|
|
|
device.recover(
|
2019-12-07 11:11:51 +00:00
|
|
|
client, 12, False, False, "label", "en-US", client.mnemonic_callback
|
2018-10-03 12:19:53 +00:00
|
|
|
)
|
2020-04-27 11:15:43 +00:00
|
|
|
|
|
|
|
ret = client.call_raw(
|
|
|
|
proto.RecoveryDevice(
|
|
|
|
word_count=12, type=proto.RecoveryDeviceType.ScrambledWords
|
|
|
|
)
|
|
|
|
)
|
|
|
|
assert isinstance(ret, proto.Failure)
|
|
|
|
assert "Device is already initialized" in ret.message
|