2019-11-04 14:45:54 +00:00
|
|
|
# This file is part of the Trezor project.
|
|
|
|
#
|
|
|
|
# Copyright (C) 2012-2019 SatoshiLabs and contributors
|
|
|
|
#
|
|
|
|
# This library is free software: you can redistribute it and/or modify
|
|
|
|
# it under the terms of the GNU Lesser General Public License version 3
|
|
|
|
# as published by the Free Software Foundation.
|
|
|
|
#
|
|
|
|
# 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.
|
|
|
|
#
|
|
|
|
# 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>.
|
|
|
|
|
2022-10-25 10:46:37 +00:00
|
|
|
from typing import TYPE_CHECKING
|
2019-11-04 14:45:54 +00:00
|
|
|
|
|
|
|
import pytest
|
|
|
|
|
|
|
|
from trezorlib import device, messages
|
|
|
|
|
|
|
|
from .. import buttons
|
2025-01-09 14:20:47 +00:00
|
|
|
from ..common import EXTERNAL_ENTROPY, MOCK_GET_ENTROPY, generate_entropy
|
2019-11-04 14:45:54 +00:00
|
|
|
from . import reset
|
|
|
|
|
2022-10-25 10:46:37 +00:00
|
|
|
if TYPE_CHECKING:
|
|
|
|
from ..device_handler import BackgroundDeviceHandler
|
|
|
|
|
|
|
|
|
2024-09-02 11:01:16 +00:00
|
|
|
pytestmark = pytest.mark.models("core")
|
2019-11-04 14:45:54 +00:00
|
|
|
|
2019-11-15 14:54:16 +00:00
|
|
|
|
2019-11-04 14:45:54 +00:00
|
|
|
@pytest.mark.setup_client(uninitialized=True)
|
2023-05-04 12:16:40 +00:00
|
|
|
@pytest.mark.parametrize(
|
|
|
|
"group_count, group_threshold, share_count, share_threshold",
|
|
|
|
[
|
|
|
|
pytest.param(2, 2, 2, 2, id="2of2"),
|
|
|
|
pytest.param(16, 16, 16, 16, id="16of16", marks=pytest.mark.slow),
|
|
|
|
],
|
|
|
|
)
|
|
|
|
def test_reset_slip39_advanced(
|
2022-10-25 10:46:37 +00:00
|
|
|
device_handler: "BackgroundDeviceHandler",
|
2023-05-04 12:16:40 +00:00
|
|
|
group_count: int,
|
|
|
|
group_threshold: int,
|
|
|
|
share_count: int,
|
|
|
|
share_threshold: int,
|
2022-10-25 10:46:37 +00:00
|
|
|
):
|
2019-11-04 14:45:54 +00:00
|
|
|
features = device_handler.features()
|
|
|
|
debug = device_handler.debuglink()
|
|
|
|
|
|
|
|
assert features.initialized is False
|
|
|
|
|
2019-11-15 14:54:16 +00:00
|
|
|
device_handler.run(
|
2025-01-09 14:20:47 +00:00
|
|
|
device.setup,
|
2019-11-15 14:54:16 +00:00
|
|
|
backup_type=messages.BackupType.Slip39_Advanced,
|
|
|
|
pin_protection=False,
|
2025-01-09 14:20:47 +00:00
|
|
|
passphrase_protection=False,
|
|
|
|
entropy_check_count=0,
|
|
|
|
_get_entropy=MOCK_GET_ENTROPY,
|
2019-11-15 14:54:16 +00:00
|
|
|
)
|
2019-11-04 14:45:54 +00:00
|
|
|
|
2019-11-15 14:54:16 +00:00
|
|
|
# confirm new wallet
|
2023-05-04 12:16:40 +00:00
|
|
|
reset.confirm_new_wallet(debug)
|
2019-11-04 14:45:54 +00:00
|
|
|
|
2019-11-15 14:54:16 +00:00
|
|
|
# confirm back up
|
2024-11-12 11:17:55 +00:00
|
|
|
# TR.assert_in_multiple(
|
|
|
|
# debug.read_layout().text_content(),
|
|
|
|
# ["backup__it_should_be_backed_up", "backup__it_should_be_backed_up_now"],
|
|
|
|
# )
|
2023-08-11 15:57:32 +00:00
|
|
|
reset.confirm_read(debug)
|
2019-11-04 14:45:54 +00:00
|
|
|
|
2024-06-04 13:40:42 +00:00
|
|
|
# confirm backup intro
|
2024-11-12 11:17:55 +00:00
|
|
|
# TR.assert_in(debug.read_layout().text_content(), "backup__info_multi_share_backup")
|
2024-06-04 13:40:42 +00:00
|
|
|
reset.confirm_read(debug)
|
|
|
|
|
2019-11-15 14:54:16 +00:00
|
|
|
# confirm checklist
|
2024-11-12 11:17:55 +00:00
|
|
|
# TR.assert_in(
|
|
|
|
# debug.read_layout().text_content(), "reset__slip39_checklist_num_groups"
|
|
|
|
# )
|
2023-08-11 15:57:32 +00:00
|
|
|
reset.confirm_read(debug)
|
2019-11-04 14:45:54 +00:00
|
|
|
|
2023-05-04 12:16:40 +00:00
|
|
|
# set num of groups - default is 5
|
2024-05-27 22:19:01 +00:00
|
|
|
assert debug.model is not None
|
|
|
|
model_name: str = debug.model.internal_name
|
2023-05-04 12:16:40 +00:00
|
|
|
if group_count < 5:
|
2024-05-27 22:19:01 +00:00
|
|
|
reset.set_selection(debug, buttons.reset_minus(model_name), 5 - group_count)
|
2023-05-04 12:16:40 +00:00
|
|
|
else:
|
2024-05-27 22:19:01 +00:00
|
|
|
reset.set_selection(debug, buttons.reset_plus(model_name), group_count - 5)
|
2019-11-04 14:45:54 +00:00
|
|
|
|
2019-11-15 14:54:16 +00:00
|
|
|
# confirm checklist
|
2024-11-12 11:17:55 +00:00
|
|
|
# TR.assert_in_multiple(
|
|
|
|
# debug.read_layout().text_content(),
|
|
|
|
# [
|
|
|
|
# "reset__slip39_checklist_set_threshold", # basic
|
2025-01-08 13:45:24 +00:00
|
|
|
# "reset__slip39_checklist_set_num_shares", # advanced (UI bolt and quicksilver)
|
|
|
|
# "reset__slip39_checklist_num_shares", # advanced (UI samson)
|
2024-11-12 11:17:55 +00:00
|
|
|
# ],
|
|
|
|
# )
|
2023-08-11 15:57:32 +00:00
|
|
|
reset.confirm_read(debug)
|
2019-11-04 14:45:54 +00:00
|
|
|
|
2019-11-15 14:54:16 +00:00
|
|
|
# set group threshold
|
2023-05-04 12:16:40 +00:00
|
|
|
# TODO: could make it general as well
|
|
|
|
if group_count == 2 and group_threshold == 2:
|
2024-05-27 22:19:01 +00:00
|
|
|
reset.set_selection(debug, buttons.reset_plus(model_name), 0)
|
2023-05-04 12:16:40 +00:00
|
|
|
elif group_count == 16 and group_threshold == 16:
|
2024-05-27 22:19:01 +00:00
|
|
|
reset.set_selection(debug, buttons.reset_plus(model_name), 11)
|
2023-05-04 12:16:40 +00:00
|
|
|
else:
|
|
|
|
raise RuntimeError("not a supported combination")
|
2019-11-04 14:45:54 +00:00
|
|
|
|
2019-11-15 14:54:16 +00:00
|
|
|
# confirm checklist
|
2024-11-12 11:17:55 +00:00
|
|
|
# TR.assert_in_multiple(
|
|
|
|
# debug.read_layout().text_content(),
|
|
|
|
# [
|
|
|
|
# "reset__slip39_checklist_set_sizes",
|
|
|
|
# "reset__slip39_checklist_set_sizes_longer",
|
|
|
|
# ],
|
|
|
|
# )
|
2023-08-11 15:57:32 +00:00
|
|
|
reset.confirm_read(debug)
|
2019-11-04 14:45:54 +00:00
|
|
|
|
2019-11-15 14:54:16 +00:00
|
|
|
# set share num and threshold for groups
|
2023-05-04 12:16:40 +00:00
|
|
|
for _ in range(group_count):
|
|
|
|
# set num of shares - default is 5
|
|
|
|
if share_count < 5:
|
2024-05-27 22:19:01 +00:00
|
|
|
reset.set_selection(debug, buttons.reset_minus(model_name), 5 - share_count)
|
2023-05-04 12:16:40 +00:00
|
|
|
else:
|
2024-05-27 22:19:01 +00:00
|
|
|
reset.set_selection(debug, buttons.reset_plus(model_name), share_count - 5)
|
2019-11-04 14:45:54 +00:00
|
|
|
|
2019-11-15 14:54:16 +00:00
|
|
|
# set share threshold
|
2023-05-04 12:16:40 +00:00
|
|
|
# TODO: could make it general as well
|
|
|
|
if share_count == 2 and share_threshold == 2:
|
2024-05-27 22:19:01 +00:00
|
|
|
reset.set_selection(debug, buttons.reset_plus(model_name), 0)
|
2023-05-04 12:16:40 +00:00
|
|
|
elif share_count == 16 and share_threshold == 16:
|
2024-05-27 22:19:01 +00:00
|
|
|
reset.set_selection(debug, buttons.reset_plus(model_name), 11)
|
2023-05-04 12:16:40 +00:00
|
|
|
else:
|
|
|
|
raise RuntimeError("not a supported combination")
|
2019-11-04 14:45:54 +00:00
|
|
|
|
2023-06-26 08:50:28 +00:00
|
|
|
# confirm backup warning
|
2024-11-12 11:17:55 +00:00
|
|
|
# TR.assert_in(debug.read_layout().text_content(), "reset__never_make_digital_copy")
|
2023-08-11 15:57:32 +00:00
|
|
|
reset.confirm_read(debug, middle_r=True)
|
2019-11-04 14:45:54 +00:00
|
|
|
|
2022-10-25 10:46:37 +00:00
|
|
|
all_words: list[str] = []
|
2023-05-04 12:16:40 +00:00
|
|
|
for _ in range(group_count):
|
|
|
|
for _ in range(share_count):
|
2019-11-15 14:54:16 +00:00
|
|
|
# read words
|
2024-04-23 10:26:46 +00:00
|
|
|
words = reset.read_words(debug, do_htc=False)
|
2019-11-04 14:45:54 +00:00
|
|
|
|
2019-11-15 14:54:16 +00:00
|
|
|
# confirm words
|
|
|
|
reset.confirm_words(debug, words)
|
2019-11-04 14:45:54 +00:00
|
|
|
|
2019-11-15 14:54:16 +00:00
|
|
|
# confirm share checked
|
2023-08-11 15:57:32 +00:00
|
|
|
reset.confirm_read(debug)
|
2019-11-04 14:45:54 +00:00
|
|
|
|
2019-11-15 14:54:16 +00:00
|
|
|
all_words.append(" ".join(words))
|
2019-11-04 14:45:54 +00:00
|
|
|
|
2019-11-15 14:54:16 +00:00
|
|
|
# confirm backup done
|
2023-08-11 15:57:32 +00:00
|
|
|
reset.confirm_read(debug)
|
2019-11-04 14:45:54 +00:00
|
|
|
|
2019-11-15 14:54:16 +00:00
|
|
|
# generate secret locally
|
|
|
|
internal_entropy = debug.state().reset_entropy
|
2022-10-25 10:46:37 +00:00
|
|
|
assert internal_entropy is not None
|
2019-11-15 14:54:16 +00:00
|
|
|
secret = generate_entropy(128, internal_entropy, EXTERNAL_ENTROPY)
|
2019-11-04 14:45:54 +00:00
|
|
|
|
2019-11-15 14:54:16 +00:00
|
|
|
# validate that all combinations will result in the correct master secret
|
|
|
|
reset.validate_mnemonics(all_words, secret)
|
2019-11-04 14:45:54 +00:00
|
|
|
|
2025-01-09 14:20:47 +00:00
|
|
|
# retrieve the result to check that it's not a TrezorFailure exception
|
|
|
|
device_handler.result()
|
2019-11-04 14:45:54 +00:00
|
|
|
|
2019-11-15 14:54:16 +00:00
|
|
|
features = device_handler.features()
|
|
|
|
assert features.initialized is True
|
2024-05-23 14:09:16 +00:00
|
|
|
assert features.backup_availability == messages.BackupAvailability.NotAvailable
|
2019-11-15 14:54:16 +00:00
|
|
|
assert features.pin_protection is False
|
|
|
|
assert features.passphrase_protection is False
|
2024-05-17 21:08:23 +00:00
|
|
|
assert features.backup_type is messages.BackupType.Slip39_Advanced_Extendable
|