Skip to content

Commit

Permalink
Add Trezor support (#318)
Browse files Browse the repository at this point in the history
* Refactor ledger_manager
  • Loading branch information
moisses89 authored Dec 12, 2023
1 parent 560a81a commit c21e96c
Show file tree
Hide file tree
Showing 23 changed files with 910 additions and 458 deletions.
19 changes: 17 additions & 2 deletions README.md
Original file line number Diff line number Diff line change
Expand Up @@ -139,7 +139,10 @@ the information about the Safe using:
```
> refresh
```
## Ledger module
## Hardware wallets support
**NOTE**: before signing anything ensure that the data showing on your hardware wallet device is the same as the safe-cli data.
### Ledger
Ledger module is an optional feature of safe-cli to sign transactions with the help of [ledgereth](https://github.com/mikeshultz/ledger-eth-lib) library based on [ledgerblue](https://github.com/LedgerHQ/blue-loader-python).
To enable, safe-cli must be installed as follows:
Expand All @@ -155,7 +158,19 @@ SUBSYSTEMS=="usb", ATTRS{idVendor}=="2c97", ATTRS{idProduct}=="0004", MODE="0660
Safe-cli Ledger commands:
- `load_ledger_cli_owners [--legacy-accounts] [--derivation-path <str>]`: show a list of the first 5 accounts (--legacy-accounts search using ledger legacy derivation) or load an account from provided derivation path.
**NOTE**: before signing anything ensure that the data showing on your ledger is the same as the safe-cli data.
### Trezor
Trezor module is an optional feature of safe-cli to sign transactions from Trezor hardware wallet using the [trezor](https://pypi.org/project/trezor/) library.
To enable, safe-cli must be installed as follows:
```
pip install safe-cli[trezor]
```
### Enable multiple hardware wallets
```
pip install safe-cli[ledger, trezor]
```
## Creating a new Safe
Use `safe-creator <node_url> <private_key> --owners <checksummed_address_1> <checksummed_address_2> --threshold <uint> --salt-nonce <uint256>`.
Expand Down
1 change: 1 addition & 0 deletions requirements.txt
Original file line number Diff line number Diff line change
Expand Up @@ -7,4 +7,5 @@ pygments==2.17.2
requests==2.31.0
safe-eth-py==6.0.0b8
tabulate==0.9.0
trezor==0.13.8
web3==6.11.4
141 changes: 0 additions & 141 deletions safe_cli/operators/hw_accounts/ledger_manager.py

This file was deleted.

File renamed without changes.
2 changes: 2 additions & 0 deletions safe_cli/operators/hw_wallets/constants.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,2 @@
BIP32_ETH_PATTERN = r"^44'/60'/[0-9]+'/[0-9]+/[0-9]+$"
BIP32_LEGACY_LEDGER_PATTERN = r"^44'/60'/[0-9]+'/[0-9]+$"
2 changes: 2 additions & 0 deletions safe_cli/operators/hw_wallets/exceptions.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,2 @@
class InvalidDerivationPath(Exception):
message = "The provided derivation path is not valid"
59 changes: 59 additions & 0 deletions safe_cli/operators/hw_wallets/hw_wallet.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,59 @@
import re
from abc import ABC, abstractmethod

from .constants import BIP32_ETH_PATTERN, BIP32_LEGACY_LEDGER_PATTERN
from .exceptions import InvalidDerivationPath


class HwWallet(ABC):
def __init__(self, derivation_path: str):
derivation_path = derivation_path.replace("m/", "")
if self._is_valid_derivation_path(derivation_path):
self.derivation_path = derivation_path
self.address = self.get_address()

@property
def get_derivation_path(self):
return self.derivation_path

@abstractmethod
def get_address(self):
"""
:return:
"""

def _is_valid_derivation_path(self, derivation_path: str):
"""
Detect if a string is a valid derivation path
"""
if not (
re.match(BIP32_ETH_PATTERN, derivation_path) is not None
or re.match(BIP32_LEGACY_LEDGER_PATTERN, derivation_path) is not None
):
raise InvalidDerivationPath()

return True

@abstractmethod
def sign_typed_hash(self, domain_hash: bytes, message_hash: bytes) -> bytes:
"""
:param domain_hash:
:param message_hash:
:return: signature bytes
"""

def __str__(self):
return f"{self.__class__.__name__} device with address {self.address}"

def __eq__(self, other):
if isinstance(other, HwWallet):
return (
self.derivation_path == other.derivation_path
and self.address == other.address
)
return False

def __hash__(self):
return hash((self.derivation_path, self.address))
138 changes: 138 additions & 0 deletions safe_cli/operators/hw_wallets/hw_wallet_manager.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,138 @@
from enum import Enum
from functools import lru_cache
from typing import Dict, List, Optional, Set, Tuple

from eth_typing import ChecksumAddress
from prompt_toolkit import HTML, print_formatted_text

from gnosis.eth.eip712 import eip712_encode
from gnosis.safe import SafeTx

from .hw_wallet import HwWallet


class HwWalletType(Enum):
TREZOR = 0
LEDGER = 1


@lru_cache(maxsize=None)
def get_hw_wallet_manager():
return HwWalletManager()


class HwWalletManager:
def __init__(self):
self.wallets: Set[HwWallet] = set()
self.supported_hw_wallet_types: Dict[str, HwWallet] = {}
try:
from .ledger_wallet import LedgerWallet

self.supported_hw_wallet_types[HwWalletType.LEDGER] = LedgerWallet
except (ImportError):
pass

try:
from .trezor_wallet import TrezorWallet

self.supported_hw_wallet_types[HwWalletType.TREZOR] = TrezorWallet
except (ImportError):
pass

def is_supported_hw_wallet(self, hw_wallet_type: HwWalletType) -> bool:
return hw_wallet_type in self.supported_hw_wallet_types

def get_hw_wallet(self, hw_wallet_type: HwWalletType) -> Optional[HwWallet]:
if hw_wallet_type in self.supported_hw_wallet_types:
return self.supported_hw_wallet_types[hw_wallet_type]

def get_accounts(
self,
hw_wallet_type: HwWalletType,
legacy_account: Optional[bool] = False,
number_accounts: Optional[int] = 5,
) -> List[Tuple[ChecksumAddress, str]]:
"""
:param hw_wallet: Trezor or Ledger
:param legacy_account:
:param number_accounts: number of accounts requested to ledger
:return: a list of tuples with address and derivation path
"""
accounts = []
hw_wallet = self.get_hw_wallet(hw_wallet_type)
for i in range(number_accounts):
if legacy_account:
path_string = f"44'/60'/0'/{i}"
else:
path_string = f"44'/60'/{i}'/0/0"

accounts.append((hw_wallet(path_string).address, path_string))
return accounts

def add_account(
self, hw_wallet_type: HwWalletType, derivation_path: str
) -> ChecksumAddress:
"""
Add an account to ledger manager set and return the added address
:param derivation_path:
:return:
"""

hw_wallet = self.get_hw_wallet(hw_wallet_type)

address = hw_wallet(derivation_path).address
self.wallets.add(hw_wallet(derivation_path))
return address

def delete_accounts(self, addresses: List[ChecksumAddress]) -> Set:
"""
Remove ledger accounts from address
:param accounts:
:return: list with the delete accounts
"""
accounts_to_remove = set()
for address in addresses:
for account in self.wallets:
if account.address == address:
accounts_to_remove.add(account)
self.wallets = self.wallets.difference(accounts_to_remove)
return accounts_to_remove

def sign_eip712(self, safe_tx: SafeTx, wallets: List[HwWallet]) -> SafeTx:
"""
Sign a safeTx EIP-712 hashes with supported hw wallet devices
:param domain_hash:
:param message_hash:
:param wallets: list of HwWallet
:return: signed safeTx
"""
encode_hash = eip712_encode(safe_tx.eip712_structured_data)
domain_hash = encode_hash[1]
message_hash = encode_hash[2]
for wallet in wallets:
print_formatted_text(
HTML(
f"<ansired>Make sure before signing in your {wallet} that the domain_hash and message_hash are both correct</ansired>"
)
)
print_formatted_text(HTML(f"Domain_hash: <b>{domain_hash.hex()}</b>"))
print_formatted_text(HTML(f"Message_hash: <b>{message_hash.hex()}</b>"))
signature = wallet.sign_typed_hash(domain_hash, message_hash)

# Insert signature sorted
if wallet.address not in safe_tx.signers:
new_owners = safe_tx.signers + [wallet.address]
new_owner_pos = sorted(new_owners, key=lambda x: int(x, 16)).index(
wallet.address
)
safe_tx.signatures = (
safe_tx.signatures[: 65 * new_owner_pos]
+ signature
+ safe_tx.signatures[65 * new_owner_pos :]
)

return safe_tx
Loading

0 comments on commit c21e96c

Please sign in to comment.