mirror of
https://github.com/trezor/trezor-firmware.git
synced 2024-12-21 05:48:23 +00:00
127 lines
4.3 KiB
Python
127 lines
4.3 KiB
Python
# This file is part of the TREZOR project.
|
|
#
|
|
# Copyright (C) 2012-2016 Marek Palatinus <slush@satoshilabs.com>
|
|
# Copyright (C) 2012-2016 Pavol Rusnak <stick@satoshilabs.com>
|
|
# Copyright (C) 2016 Jochen Hoenicke <hoenicke@gmail.com>
|
|
#
|
|
# This library is free software: you can redistribute it and/or modify
|
|
# it under the terms of the GNU Lesser General Public License as published by
|
|
# the Free Software Foundation, either version 3 of the License, or
|
|
# (at your option) any later version.
|
|
#
|
|
# 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 GNU Lesser General Public License
|
|
# along with this library. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
'''BridgeTransport implements transport TREZOR Bridge (aka trezord).'''
|
|
|
|
import binascii
|
|
import json
|
|
import requests
|
|
from google.protobuf import json_format
|
|
from . import messages_pb2 as proto
|
|
from .transport import TransportV1
|
|
|
|
TREZORD_HOST = 'https://localback.net:21324'
|
|
CONFIG_URL = 'https://wallet.trezor.io/data/config_signed.bin'
|
|
|
|
|
|
def get_error(resp):
|
|
return ' (error=%d str=%s)' % (resp.status_code, resp.json()['error'])
|
|
|
|
|
|
class BridgeTransport(TransportV1):
|
|
|
|
CONFIGURED = False
|
|
|
|
def __init__(self, device, *args, **kwargs):
|
|
self.configure()
|
|
|
|
self.path = device['path']
|
|
|
|
self.session = None
|
|
self.response = None
|
|
self.conn = requests.Session()
|
|
|
|
super(BridgeTransport, self).__init__(device, *args, **kwargs)
|
|
|
|
@staticmethod
|
|
def configure():
|
|
if BridgeTransport.CONFIGURED:
|
|
return
|
|
r = requests.get(CONFIG_URL, verify=False)
|
|
if r.status_code != 200:
|
|
raise Exception('Could not fetch config from %s' % CONFIG_URL)
|
|
|
|
config = r.text
|
|
|
|
r = requests.post(TREZORD_HOST + '/configure', data=config)
|
|
if r.status_code != 200:
|
|
raise Exception('trezord: Could not configure' + get_error(r))
|
|
BridgeTransport.CONFIGURED = True
|
|
|
|
@classmethod
|
|
def enumerate(cls):
|
|
"""
|
|
Return a list of available TREZOR devices.
|
|
"""
|
|
cls.configure()
|
|
r = requests.get(TREZORD_HOST + '/enumerate')
|
|
if r.status_code != 200:
|
|
raise Exception('trezord: Could not enumerate devices' + get_error(r))
|
|
enum = r.json()
|
|
return enum
|
|
|
|
@classmethod
|
|
def find_by_path(cls, path=None):
|
|
"""
|
|
Finds a device by transport-specific path.
|
|
If path is not set, return first device.
|
|
"""
|
|
devices = cls.enumerate()
|
|
for dev in devices:
|
|
if not path or dev['path'] == binascii.hexlify(path):
|
|
return cls(dev)
|
|
raise Exception('Device not found')
|
|
|
|
def _open(self):
|
|
r = self.conn.post(TREZORD_HOST + '/acquire/%s' % self.path)
|
|
if r.status_code != 200:
|
|
raise Exception('trezord: Could not acquire session' + get_error(r))
|
|
resp = r.json()
|
|
self.session = resp['session']
|
|
|
|
def _close(self):
|
|
r = self.conn.post(TREZORD_HOST + '/release/%s' % self.session)
|
|
if r.status_code != 200:
|
|
raise Exception('trezord: Could not release session' + get_error(r))
|
|
else:
|
|
self.session = None
|
|
|
|
def _ready_to_read(self):
|
|
return self.response is not None
|
|
|
|
def write(self, protobuf_msg):
|
|
# Override main 'write' method, HTTP transport cannot be
|
|
# splitted to chunks
|
|
cls = protobuf_msg.__class__.__name__
|
|
msg = json_format.MessageToJson(protobuf_msg, preserving_proto_field_name=True)
|
|
payload = '{"type": "%s", "message": %s}' % (cls, msg)
|
|
r = self.conn.post(TREZORD_HOST + '/call/%s' % self.session, data=payload)
|
|
if r.status_code != 200:
|
|
raise Exception('trezord: Could not write message' + get_error(r))
|
|
else:
|
|
self.response = r.json()
|
|
|
|
def _read(self):
|
|
if self.response is None:
|
|
raise Exception('No response stored')
|
|
cls = getattr(proto, self.response['type'])
|
|
inst = cls()
|
|
pb = json_format.ParseDict(self.response['message'], inst)
|
|
return (0, 'protobuf', pb)
|