2018-04-06 20:11:14 +00:00
|
|
|
#!/usr/bin/env python3
|
|
|
|
"""Fetch information about coins and tokens supported by Trezor and update it in coins_details.json."""
|
|
|
|
import time
|
|
|
|
import json
|
|
|
|
import requests
|
|
|
|
import pprint
|
|
|
|
|
2018-04-11 16:37:30 +00:00
|
|
|
SKIP_COINMARKETCAP = True
|
2018-04-06 20:11:14 +00:00
|
|
|
|
|
|
|
def coinmarketcap_info(shortcut):
|
|
|
|
if SKIP_COINMARKETCAP:
|
|
|
|
raise Exception("Skipping coinmarketcap call")
|
|
|
|
|
|
|
|
shortcut = shortcut.replace(' ', '-')
|
|
|
|
url = 'https://api.coinmarketcap.com/v1/ticker/%s/?convert=USD' % shortcut
|
|
|
|
try:
|
2018-04-11 16:37:30 +00:00
|
|
|
return requests.get(url).json()[0]
|
|
|
|
except KeyboardInterrupt:
|
|
|
|
raise
|
2018-04-06 20:11:14 +00:00
|
|
|
except:
|
2018-04-11 16:37:30 +00:00
|
|
|
print("Cannot fetch Coinmarketcap info for %s" % shortcut)
|
2018-04-06 20:11:14 +00:00
|
|
|
|
|
|
|
def update_marketcap(obj, shortcut):
|
|
|
|
try:
|
|
|
|
obj['marketcap_usd'] = int(float(coinmarketcap_info(shortcut)['market_cap_usd']))
|
|
|
|
except:
|
|
|
|
pass
|
|
|
|
|
|
|
|
def coinmarketcap_global():
|
|
|
|
if SKIP_COINMARKETCAP:
|
|
|
|
raise Exception("Skipping coinmarketcap call")
|
|
|
|
|
|
|
|
url = 'https://api.coinmarketcap.com/v1/global'
|
|
|
|
ret = requests.get(url)
|
|
|
|
data = ret.json()
|
|
|
|
return data
|
|
|
|
|
|
|
|
def set_default(obj, key, default_value):
|
|
|
|
obj[key] = obj.setdefault(key, default_value)
|
|
|
|
|
|
|
|
def update_info(details):
|
|
|
|
details['info']['updated_at'] = int(time.time())
|
|
|
|
details['info']['updated_at_readable'] = time.asctime()
|
2018-04-11 16:37:30 +00:00
|
|
|
details['info']['t1_coins'] = len([True for _, c in details['coins'].items() if c['t1_enabled'] == 'yes'])
|
|
|
|
details['info']['t2_coins'] = len([True for _, c in details['coins'].items() if c['t2_enabled'] == 'yes'])
|
2018-05-16 15:30:28 +00:00
|
|
|
|
2018-04-06 20:11:14 +00:00
|
|
|
try:
|
|
|
|
details['info']['total_marketcap_usd'] = int(coinmarketcap_global()['total_market_cap_usd'])
|
|
|
|
except:
|
|
|
|
pass
|
|
|
|
|
|
|
|
marketcap = 0
|
2018-04-11 16:37:30 +00:00
|
|
|
for k, c in details['coins'].items():
|
|
|
|
if c['t1_enabled'] == 'yes' or c['t2_enabled'] == 'yes':
|
|
|
|
marketcap += details['coins'][k].setdefault('marketcap_usd', 0)
|
2018-04-06 20:11:14 +00:00
|
|
|
details['info']['marketcap_usd'] = marketcap
|
|
|
|
|
2018-04-11 16:37:30 +00:00
|
|
|
def check_unsupported(details, prefix, supported):
|
|
|
|
for k in details['coins'].keys():
|
|
|
|
if not k.startswith(prefix):
|
|
|
|
continue
|
|
|
|
if k not in supported:
|
|
|
|
print("%s not supported by Trezor? (Possible manual entry)" % k)
|
|
|
|
|
2018-04-06 20:11:14 +00:00
|
|
|
def update_coins(details):
|
|
|
|
coins = json.load(open('coins.json', 'r'))
|
|
|
|
|
2018-04-11 16:37:30 +00:00
|
|
|
supported = []
|
2018-04-06 20:11:14 +00:00
|
|
|
for coin in coins:
|
|
|
|
if coin['firmware'] != 'stable':
|
|
|
|
continue
|
|
|
|
|
2018-04-11 16:37:30 +00:00
|
|
|
# print("Updating", coin['coin_label'], coin['coin_shortcut'])
|
|
|
|
key = "coin:%s" % coin['coin_shortcut']
|
|
|
|
supported.append(key)
|
|
|
|
out = details['coins'].setdefault(key, {})
|
|
|
|
out['type'] = 'coin'
|
|
|
|
set_default(out, 'shortcut', coin['coin_shortcut'])
|
2018-04-06 20:11:14 +00:00
|
|
|
set_default(out, 'name', coin['coin_label'])
|
2018-04-11 16:37:30 +00:00
|
|
|
set_default(out, 'links', {})
|
2018-04-06 20:11:14 +00:00
|
|
|
set_default(out, 't1_enabled', 'yes')
|
|
|
|
set_default(out, 't2_enabled', 'yes')
|
2018-04-11 16:37:30 +00:00
|
|
|
update_marketcap(out, coin.get('coinmarketcap_alias', coin['coin_label']))
|
2018-04-06 20:11:14 +00:00
|
|
|
|
2018-04-11 16:37:30 +00:00
|
|
|
check_unsupported(details, 'coin:', supported)
|
2018-04-06 20:11:14 +00:00
|
|
|
|
|
|
|
def update_erc20(details):
|
2018-04-06 21:10:35 +00:00
|
|
|
networks = [
|
|
|
|
('eth', 1),
|
|
|
|
# ('exp', 2),
|
|
|
|
# ('rop', 3),
|
|
|
|
('rin', 4),
|
|
|
|
('ubq', 8),
|
|
|
|
# ('rsk', 30),
|
|
|
|
('kov', 42),
|
|
|
|
('etc', 61),
|
|
|
|
]
|
|
|
|
|
|
|
|
# Fetch list of tokens already included in Trezor Core
|
2018-04-06 20:11:14 +00:00
|
|
|
r = requests.get('https://raw.githubusercontent.com/trezor/trezor-core/master/src/apps/ethereum/tokens.py')
|
|
|
|
d = {}
|
|
|
|
exec(r.text, d)
|
2018-04-06 21:10:35 +00:00
|
|
|
|
|
|
|
# TODO 'Qmede...' can be removed after ipfs_hash is being generated into tokens.py
|
|
|
|
ipfs_hash = d.get('ipfs_hash') or 'QmedefcF1fecLVpRymJJmyJFRpJuCTiNfPYBhzUdHPUq3T'
|
|
|
|
|
|
|
|
infos = {}
|
|
|
|
for n in networks:
|
2018-04-11 16:37:30 +00:00
|
|
|
# print("Updating info about erc20 tokens for", n[0])
|
2018-04-06 21:10:35 +00:00
|
|
|
url = 'https://gateway.ipfs.io/ipfs/%s/%s.json' % (ipfs_hash, n[0])
|
|
|
|
r = requests.get(url)
|
|
|
|
infos[n[0]] = r.json()
|
|
|
|
|
2018-04-11 16:37:30 +00:00
|
|
|
supported = []
|
2018-04-06 20:11:14 +00:00
|
|
|
for t in d['tokens']:
|
|
|
|
token = t[2]
|
2018-04-11 16:37:30 +00:00
|
|
|
# print('Updating', token)
|
2018-04-06 20:11:14 +00:00
|
|
|
|
2018-04-06 21:10:35 +00:00
|
|
|
try:
|
|
|
|
network = [ n[0] for n in networks if n[1] == t[0] ][0]
|
|
|
|
except:
|
|
|
|
raise Exception("Unknown network", t[0], "for erc20 token", token)
|
|
|
|
|
|
|
|
try:
|
|
|
|
info = [ i for i in infos[network] if i['symbol'] == token ][0]
|
|
|
|
except:
|
|
|
|
raise Exception("Unknown details for erc20 token", token)
|
|
|
|
|
2018-04-11 16:37:30 +00:00
|
|
|
key = "erc20:%s:%s" % (network, token)
|
|
|
|
supported.append(key)
|
|
|
|
out = details['coins'].setdefault(key, {})
|
2018-04-06 20:11:14 +00:00
|
|
|
out['type'] = 'erc20'
|
2018-04-06 21:10:35 +00:00
|
|
|
out['network'] = network
|
|
|
|
out['address'] = info['address']
|
2018-04-11 16:37:30 +00:00
|
|
|
|
|
|
|
set_default(out, 'shortcut', token)
|
|
|
|
set_default(out, 'name', info['name'])
|
2018-04-06 20:11:14 +00:00
|
|
|
set_default(out, 't1_enabled', 'yes')
|
|
|
|
set_default(out, 't2_enabled', 'yes')
|
2018-04-06 21:10:35 +00:00
|
|
|
set_default(out, 'links', {})
|
|
|
|
|
|
|
|
if info['website']:
|
|
|
|
out['links']['Homepage'] = info['website']
|
|
|
|
if info.get('social', {}).get('github', None):
|
|
|
|
out['links']['Github'] = info['social']['github']
|
2018-04-06 20:11:14 +00:00
|
|
|
|
2018-04-11 16:37:30 +00:00
|
|
|
update_marketcap(out, out.get('coinmarketcap_alias', token))
|
|
|
|
|
|
|
|
check_unsupported(details, 'erc20:', supported)
|
|
|
|
|
2018-04-06 20:11:14 +00:00
|
|
|
def update_ethereum(details):
|
2018-04-11 16:37:30 +00:00
|
|
|
out = details['coins'].setdefault('coin2:ETH', {})
|
2018-04-06 20:11:14 +00:00
|
|
|
out['type'] = 'coin'
|
2018-04-11 16:37:30 +00:00
|
|
|
set_default(out, 'shortcut', 'ETH')
|
|
|
|
set_default(out, 'name', 'Ethereum')
|
2018-04-06 20:11:14 +00:00
|
|
|
set_default(out, 't1_enabled', 'yes')
|
|
|
|
set_default(out, 't2_enabled', 'yes')
|
|
|
|
update_marketcap(out, 'ethereum')
|
|
|
|
|
2018-04-11 17:30:25 +00:00
|
|
|
out = details['coins'].setdefault('coin2:ETC', {})
|
|
|
|
out['type'] = 'coin'
|
|
|
|
set_default(out, 'shortcut', 'ETC')
|
|
|
|
set_default(out, 'name', 'Ethereum Classic')
|
|
|
|
set_default(out, 't1_enabled', 'yes')
|
|
|
|
set_default(out, 't2_enabled', 'yes')
|
|
|
|
update_marketcap(out, 'ethereum-classic')
|
|
|
|
|
2018-04-11 17:39:04 +00:00
|
|
|
out = details['coins'].setdefault('coin2:RSK', {})
|
|
|
|
out['type'] = 'coin'
|
|
|
|
set_default(out, 'shortcut', 'RSK')
|
|
|
|
set_default(out, 'name', 'Rootstock')
|
|
|
|
set_default(out, 't1_enabled', 'yes')
|
|
|
|
set_default(out, 't2_enabled', 'yes')
|
|
|
|
update_marketcap(out, 'rootstock')
|
|
|
|
|
|
|
|
out = details['coins'].setdefault('coin2:EXP', {})
|
|
|
|
out['type'] = 'coin'
|
|
|
|
set_default(out, 'shortcut', 'EXP')
|
|
|
|
set_default(out, 'name', 'Expanse')
|
|
|
|
set_default(out, 't1_enabled', 'yes')
|
|
|
|
set_default(out, 't2_enabled', 'yes')
|
|
|
|
update_marketcap(out, 'expanse')
|
|
|
|
|
2018-04-11 17:30:25 +00:00
|
|
|
out = details['coins'].setdefault('coin2:UBQ', {})
|
|
|
|
out['type'] = 'coin'
|
|
|
|
set_default(out, 'shortcut', 'UBQ')
|
|
|
|
set_default(out, 'name', 'Ubiq')
|
|
|
|
set_default(out, 't1_enabled', 'yes')
|
|
|
|
set_default(out, 't2_enabled', 'yes')
|
|
|
|
update_marketcap(out, 'ubiq')
|
2018-05-16 15:30:28 +00:00
|
|
|
|
2018-05-09 11:53:55 +00:00
|
|
|
out = details['coins'].setdefault('coin2:EGEM', {})
|
|
|
|
out['type'] = 'coin'
|
|
|
|
set_default(out, 'shortcut', 'EGEM')
|
|
|
|
set_default(out, 'name', 'EtherGem')
|
|
|
|
set_default(out, 't1_enabled', 'yes')
|
|
|
|
set_default(out, 't2_enabled', 'yes')
|
|
|
|
update_marketcap(out, 'egem')
|
2018-04-11 17:30:25 +00:00
|
|
|
|
2018-05-16 15:30:28 +00:00
|
|
|
out = details['coins'].setdefault('coin2:ETSC', {})
|
|
|
|
out['type'] = 'coin'
|
|
|
|
set_default(out, 'shortcut', 'ETSC')
|
|
|
|
set_default(out, 'name', 'EthereumSocial')
|
|
|
|
set_default(out, 't1_enabled', 'yes')
|
|
|
|
set_default(out, 't2_enabled', 'yes')
|
|
|
|
update_marketcap(out, 'etsc')
|
|
|
|
|
2018-04-06 21:10:35 +00:00
|
|
|
def update_mosaics(details):
|
|
|
|
r = requests.get('https://raw.githubusercontent.com/trezor/trezor-mcu/master/firmware/nem_mosaics.json')
|
2018-04-11 16:37:30 +00:00
|
|
|
supported = []
|
2018-04-06 21:10:35 +00:00
|
|
|
for mosaic in r.json():
|
2018-04-11 16:37:30 +00:00
|
|
|
# print('Updating', mosaic['name'], mosaic['ticker'])
|
2018-04-06 21:10:35 +00:00
|
|
|
|
2018-04-11 16:37:30 +00:00
|
|
|
key = "mosaic:%s" % mosaic['ticker'].strip()
|
|
|
|
supported.append(key)
|
|
|
|
out = details['coins'].setdefault(key, {})
|
2018-04-06 21:10:35 +00:00
|
|
|
out['type'] = 'mosaic'
|
2018-04-11 16:37:30 +00:00
|
|
|
set_default(out, 'shortcut', mosaic['ticker'].strip())
|
|
|
|
set_default(out, 'name', mosaic['name'])
|
2018-04-06 21:10:35 +00:00
|
|
|
set_default(out, 't1_enabled', 'yes')
|
|
|
|
set_default(out, 't2_enabled', 'yes')
|
|
|
|
|
2018-04-11 16:37:30 +00:00
|
|
|
update_marketcap(out, out.get('coinmarketcap_alias', out['name']))
|
|
|
|
|
|
|
|
check_unsupported(details, 'mosaic:', supported)
|
|
|
|
|
|
|
|
def check_missing_details(details):
|
|
|
|
for k in details['coins'].keys():
|
|
|
|
coin = details['coins'][k]
|
|
|
|
|
|
|
|
if 'links' not in coin:
|
|
|
|
print("%s: Missing links" % k)
|
|
|
|
continue
|
|
|
|
if 'Homepage' not in coin['links']:
|
|
|
|
print("%s: Missing homepage" % k)
|
|
|
|
if coin['t1_enabled'] not in ('yes', 'no', 'planned', 'in progress'):
|
|
|
|
print("%s: Unknown t1_enabled" % k)
|
|
|
|
if coin['t2_enabled'] not in ('yes', 'no', 'planned', 'in progress'):
|
|
|
|
print("%s: Unknown t2_enabled" % k)
|
|
|
|
if 'TREZOR Wallet' in coin['links'] and coin['links']['TREZOR Wallet'] != 'https://wallet.trezor.io':
|
|
|
|
print("%s: Strange URL for TREZOR Wallet" % k)
|
2018-04-06 20:11:14 +00:00
|
|
|
|
|
|
|
if __name__ == '__main__':
|
|
|
|
try:
|
|
|
|
details = json.load(open('coins_details.json', 'r'))
|
|
|
|
except FileNotFoundError:
|
|
|
|
details = {'coins': {}, 'info': {}}
|
|
|
|
|
|
|
|
update_coins(details)
|
|
|
|
update_erc20(details)
|
|
|
|
update_ethereum(details)
|
2018-04-06 21:10:35 +00:00
|
|
|
update_mosaics(details)
|
2018-04-06 20:11:14 +00:00
|
|
|
update_info(details)
|
2018-04-11 16:37:30 +00:00
|
|
|
check_missing_details(details)
|
2018-04-06 20:11:14 +00:00
|
|
|
|
2018-04-11 16:37:30 +00:00
|
|
|
print(json.dumps(details['info'], sort_keys=True, indent=4))
|
2018-04-06 20:11:14 +00:00
|
|
|
json.dump(details, open('coins_details.json', 'w'), sort_keys=True, indent=4)
|