2016-07-28 11:30:13 +02:00
|
|
|
import time
|
|
|
|
import requests
|
|
|
|
import logging
|
|
|
|
import json
|
2017-04-30 21:50:33 +02:00
|
|
|
from twisted.internet import defer, threads
|
2016-07-28 11:30:13 +02:00
|
|
|
from twisted.internet.task import LoopingCall
|
|
|
|
|
2016-12-05 22:51:16 +01:00
|
|
|
from lbrynet import conf
|
2016-09-27 20:18:35 +02:00
|
|
|
from lbrynet.metadata.Fee import FeeValidator
|
2017-02-28 02:18:57 +01:00
|
|
|
from lbrynet.core.Error import InvalidExchangeRateResponse
|
2016-12-05 22:51:16 +01:00
|
|
|
|
2016-07-28 11:30:13 +02:00
|
|
|
log = logging.getLogger(__name__)
|
|
|
|
|
|
|
|
CURRENCY_PAIRS = ["USDBTC", "BTCLBC"]
|
|
|
|
BITTREX_FEE = 0.0025
|
|
|
|
COINBASE_FEE = 0.0 #add fee
|
|
|
|
|
|
|
|
|
|
|
|
class ExchangeRate(object):
|
|
|
|
def __init__(self, market, spot, ts):
|
|
|
|
assert int(time.time()) - ts < 600
|
|
|
|
self.currency_pair = (market[0:3], market[3:6])
|
|
|
|
self.spot = spot
|
|
|
|
self.ts = ts
|
|
|
|
|
2017-02-28 02:18:57 +01:00
|
|
|
def __repr__(self):
|
|
|
|
out = "Currency pair:{}, spot:{}, ts:{}".format(
|
|
|
|
self.currency_pair, self.spot, self.ts)
|
|
|
|
return out
|
|
|
|
|
2016-07-28 11:30:13 +02:00
|
|
|
def as_dict(self):
|
|
|
|
return {'spot': self.spot, 'ts': self.ts}
|
|
|
|
|
|
|
|
|
|
|
|
class MarketFeed(object):
|
2017-02-28 02:18:57 +01:00
|
|
|
REQUESTS_TIMEOUT = 20
|
|
|
|
EXCHANGE_RATE_UPDATE_RATE_SEC = 300
|
2016-07-28 11:30:13 +02:00
|
|
|
def __init__(self, market, name, url, params, fee):
|
|
|
|
self.market = market
|
|
|
|
self.name = name
|
|
|
|
self.url = url
|
|
|
|
self.params = params
|
|
|
|
self.fee = fee
|
|
|
|
self.rate = None
|
|
|
|
self._updater = LoopingCall(self._update_price)
|
|
|
|
|
2017-02-28 02:18:57 +01:00
|
|
|
@property
|
|
|
|
def rate_is_initialized(self):
|
|
|
|
return self.rate is not None
|
|
|
|
|
2016-07-28 11:30:13 +02:00
|
|
|
def _make_request(self):
|
2017-02-28 02:18:57 +01:00
|
|
|
r = requests.get(self.url, self.params, timeout=self.REQUESTS_TIMEOUT)
|
|
|
|
return r.text
|
2016-07-28 11:30:13 +02:00
|
|
|
|
|
|
|
def _handle_response(self, response):
|
|
|
|
return NotImplementedError
|
|
|
|
|
|
|
|
def _subtract_fee(self, from_amount):
|
2016-09-28 05:56:08 +02:00
|
|
|
# increase amount to account for market fees
|
2016-07-28 11:30:13 +02:00
|
|
|
return defer.succeed(from_amount / (1.0 - self.fee))
|
|
|
|
|
|
|
|
def _save_price(self, price):
|
2016-08-23 01:56:42 +02:00
|
|
|
log.debug("Saving price update %f for %s" % (price, self.market))
|
2016-07-28 11:30:13 +02:00
|
|
|
self.rate = ExchangeRate(self.market, price, int(time.time()))
|
|
|
|
|
2016-09-28 05:56:08 +02:00
|
|
|
def _log_error(self, err):
|
2016-11-30 21:20:45 +01:00
|
|
|
log.warning(
|
|
|
|
"There was a problem updating %s exchange rate information from %s",
|
|
|
|
self.market, self.name)
|
2016-09-28 05:56:08 +02:00
|
|
|
|
2016-07-28 11:30:13 +02:00
|
|
|
def _update_price(self):
|
2017-02-28 02:18:57 +01:00
|
|
|
d = threads.deferToThread(self._make_request)
|
2016-07-28 11:30:13 +02:00
|
|
|
d.addCallback(self._handle_response)
|
|
|
|
d.addCallback(self._subtract_fee)
|
|
|
|
d.addCallback(self._save_price)
|
2016-09-28 05:56:08 +02:00
|
|
|
d.addErrback(self._log_error)
|
2017-02-28 02:18:57 +01:00
|
|
|
return d
|
2016-07-28 11:30:13 +02:00
|
|
|
|
|
|
|
def start(self):
|
|
|
|
if not self._updater.running:
|
2017-02-28 02:18:57 +01:00
|
|
|
self._updater.start(self.EXCHANGE_RATE_UPDATE_RATE_SEC)
|
2016-07-28 11:30:13 +02:00
|
|
|
|
|
|
|
def stop(self):
|
|
|
|
if self._updater.running:
|
|
|
|
self._updater.stop()
|
|
|
|
|
|
|
|
|
|
|
|
class BittrexFeed(MarketFeed):
|
|
|
|
def __init__(self):
|
|
|
|
MarketFeed.__init__(
|
|
|
|
self,
|
|
|
|
"BTCLBC",
|
|
|
|
"Bittrex",
|
2017-01-17 04:23:20 +01:00
|
|
|
conf.settings['bittrex_feed'],
|
2016-07-28 11:30:13 +02:00
|
|
|
{'market': 'BTC-LBC', 'count': 50},
|
|
|
|
BITTREX_FEE
|
|
|
|
)
|
|
|
|
|
|
|
|
def _handle_response(self, response):
|
2017-02-28 02:18:57 +01:00
|
|
|
json_response = json.loads(response)
|
|
|
|
if 'result' not in json_response:
|
|
|
|
raise InvalidExchangeRateResponse(self.name, 'result not found')
|
|
|
|
trades = json_response['result']
|
|
|
|
if len(trades) == 0:
|
|
|
|
raise InvalidExchangeRateResponse(self.market, 'trades not found')
|
|
|
|
totals = sum([i['Total'] for i in trades])
|
|
|
|
qtys = sum([i['Quantity'] for i in trades])
|
|
|
|
if totals <= 0 or qtys <= 0:
|
|
|
|
raise InvalidExchangeRateResponse(self.market, 'quantities were not positive')
|
|
|
|
vwap = totals/qtys
|
2016-07-28 11:30:13 +02:00
|
|
|
return defer.succeed(float(1.0 / vwap))
|
|
|
|
|
|
|
|
|
2017-04-05 04:41:57 +02:00
|
|
|
class LBRYioFeed(MarketFeed):
|
|
|
|
def __init__(self):
|
|
|
|
MarketFeed.__init__(
|
|
|
|
self,
|
|
|
|
"BTCLBC",
|
|
|
|
"lbry.io",
|
|
|
|
"https://api.lbry.io/lbc/exchange_rate",
|
|
|
|
{},
|
|
|
|
0.0,
|
|
|
|
)
|
|
|
|
|
|
|
|
def _handle_response(self, response):
|
|
|
|
json_response = json.loads(response)
|
|
|
|
if 'data' not in json_response:
|
|
|
|
raise InvalidExchangeRateResponse(self.name, 'result not found')
|
|
|
|
return defer.succeed(1.0 / json_response['data']['lbc_btc'])
|
|
|
|
|
|
|
|
|
2016-07-28 11:30:13 +02:00
|
|
|
class GoogleBTCFeed(MarketFeed):
|
|
|
|
def __init__(self):
|
|
|
|
MarketFeed.__init__(
|
|
|
|
self,
|
|
|
|
"USDBTC",
|
|
|
|
"Coinbase via Google finance",
|
2017-02-28 02:18:57 +01:00
|
|
|
'http://finance.google.com/finance/info',
|
|
|
|
{'client':'ig', 'q':'CURRENCY:USDBTC'},
|
2016-07-28 11:30:13 +02:00
|
|
|
COINBASE_FEE
|
|
|
|
)
|
|
|
|
|
|
|
|
def _handle_response(self, response):
|
2017-02-28 02:18:57 +01:00
|
|
|
response = response[3:] # response starts with "// "
|
|
|
|
json_response = json.loads(response)[0]
|
|
|
|
if 'l' not in json_response:
|
|
|
|
raise InvalidExchangeRateResponse(self.name, 'last trade not found')
|
|
|
|
last_trade_price = float(json_response['l'])
|
|
|
|
if last_trade_price <= 0:
|
|
|
|
raise InvalidExchangeRateResponse(self.name, 'trade price was not positive')
|
|
|
|
return defer.succeed(last_trade_price)
|
2016-07-28 11:30:13 +02:00
|
|
|
|
|
|
|
|
|
|
|
def get_default_market_feed(currency_pair):
|
|
|
|
currencies = None
|
|
|
|
if isinstance(currency_pair, str):
|
|
|
|
currencies = (currency_pair[0:3], currency_pair[3:6])
|
|
|
|
elif isinstance(currency_pair, tuple):
|
|
|
|
currencies = currency_pair
|
|
|
|
assert currencies is not None
|
|
|
|
|
|
|
|
if currencies == ("USD", "BTC"):
|
|
|
|
return GoogleBTCFeed()
|
|
|
|
elif currencies == ("BTC", "LBC"):
|
2017-04-05 04:41:57 +02:00
|
|
|
return LBRYioFeed()
|
2016-07-28 11:30:13 +02:00
|
|
|
|
|
|
|
|
|
|
|
class ExchangeRateManager(object):
|
|
|
|
def __init__(self):
|
2016-11-30 21:20:45 +01:00
|
|
|
self.market_feeds = [
|
|
|
|
get_default_market_feed(currency_pair) for currency_pair in CURRENCY_PAIRS]
|
2016-07-28 11:30:13 +02:00
|
|
|
|
|
|
|
def start(self):
|
|
|
|
log.info("Starting exchange rate manager")
|
|
|
|
for feed in self.market_feeds:
|
|
|
|
feed.start()
|
|
|
|
|
|
|
|
def stop(self):
|
|
|
|
log.info("Stopping exchange rate manager")
|
|
|
|
for source in self.market_feeds:
|
|
|
|
source.stop()
|
|
|
|
|
|
|
|
def convert_currency(self, from_currency, to_currency, amount):
|
2017-02-28 02:18:57 +01:00
|
|
|
rates = [market.rate for market in self.market_feeds]
|
|
|
|
log.info("Converting %f %s to %s, rates: %s" % (amount, from_currency, to_currency, rates))
|
2016-07-28 18:43:20 +02:00
|
|
|
if from_currency == to_currency:
|
|
|
|
return amount
|
2016-07-28 11:30:13 +02:00
|
|
|
for market in self.market_feeds:
|
2017-02-28 02:18:57 +01:00
|
|
|
if (market.rate_is_initialized and
|
|
|
|
market.rate.currency_pair == (from_currency, to_currency)):
|
2016-07-28 11:30:13 +02:00
|
|
|
return amount * market.rate.spot
|
|
|
|
for market in self.market_feeds:
|
2017-02-28 02:18:57 +01:00
|
|
|
if (market.rate_is_initialized and
|
|
|
|
market.rate.currency_pair[0] == from_currency):
|
2016-11-30 21:20:45 +01:00
|
|
|
return self.convert_currency(
|
|
|
|
market.rate.currency_pair[1], to_currency, amount * market.rate.spot)
|
|
|
|
raise Exception(
|
|
|
|
'Unable to convert {} from {} to {}'.format(amount, from_currency, to_currency))
|
2016-07-28 11:30:13 +02:00
|
|
|
|
|
|
|
def fee_dict(self):
|
|
|
|
return {market: market.rate.as_dict() for market in self.market_feeds}
|
|
|
|
|
|
|
|
def to_lbc(self, fee):
|
2016-07-28 22:12:20 +02:00
|
|
|
if fee is None:
|
|
|
|
return None
|
2016-09-27 20:18:35 +02:00
|
|
|
if not isinstance(fee, FeeValidator):
|
|
|
|
fee_in = FeeValidator(fee)
|
2016-07-28 22:32:59 +02:00
|
|
|
else:
|
|
|
|
fee_in = fee
|
2016-07-28 22:12:20 +02:00
|
|
|
|
2016-11-30 21:20:45 +01:00
|
|
|
return FeeValidator({
|
2017-04-05 01:42:35 +02:00
|
|
|
'currency':fee_in.currency_symbol,
|
2016-11-30 21:20:45 +01:00
|
|
|
'amount': self.convert_currency(fee_in.currency_symbol, "LBC", fee_in.amount),
|
|
|
|
'address': fee_in.address
|
2017-04-05 01:42:35 +02:00
|
|
|
})
|
2016-07-28 11:49:31 +02:00
|
|
|
|
|
|
|
|
|
|
|
class DummyBTCLBCFeed(MarketFeed):
|
|
|
|
def __init__(self):
|
|
|
|
MarketFeed.__init__(
|
|
|
|
self,
|
|
|
|
"BTCLBC",
|
|
|
|
"market name",
|
|
|
|
"derp.com",
|
|
|
|
None,
|
|
|
|
0.0
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
|
|
class DummyUSDBTCFeed(MarketFeed):
|
|
|
|
def __init__(self):
|
|
|
|
MarketFeed.__init__(
|
|
|
|
self,
|
|
|
|
"USDBTC",
|
|
|
|
"market name",
|
|
|
|
"derp.com",
|
|
|
|
None,
|
|
|
|
0.0
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
|
|
class DummyExchangeRateManager(object):
|
|
|
|
def __init__(self, rates):
|
|
|
|
self.market_feeds = [DummyBTCLBCFeed(), DummyUSDBTCFeed()]
|
|
|
|
for feed in self.market_feeds:
|
2016-11-30 21:20:45 +01:00
|
|
|
feed.rate = ExchangeRate(
|
|
|
|
feed.market, rates[feed.market]['spot'], rates[feed.market]['ts'])
|
2016-07-28 11:49:31 +02:00
|
|
|
|
|
|
|
def convert_currency(self, from_currency, to_currency, amount):
|
2016-08-23 01:56:42 +02:00
|
|
|
log.debug("Converting %f %s to %s" % (amount, from_currency, to_currency))
|
2016-07-28 11:49:31 +02:00
|
|
|
for market in self.market_feeds:
|
2017-02-28 02:18:57 +01:00
|
|
|
if (market.rate_is_initialized and
|
|
|
|
market.rate.currency_pair == (from_currency, to_currency)):
|
2016-07-28 11:49:31 +02:00
|
|
|
return amount * market.rate.spot
|
|
|
|
for market in self.market_feeds:
|
2017-02-28 02:18:57 +01:00
|
|
|
if (market.rate_is_initialized and
|
|
|
|
market.rate.currency_pair[0] == from_currency):
|
2016-11-30 21:20:45 +01:00
|
|
|
return self.convert_currency(
|
|
|
|
market.rate.currency_pair[1], to_currency, amount * market.rate.spot)
|
2016-07-28 11:49:31 +02:00
|
|
|
|
2016-07-28 22:12:20 +02:00
|
|
|
def to_lbc(self, fee):
|
|
|
|
if fee is None:
|
|
|
|
return None
|
2016-09-27 20:18:35 +02:00
|
|
|
if not isinstance(fee, FeeValidator):
|
|
|
|
fee_in = FeeValidator(fee)
|
2016-07-28 22:32:59 +02:00
|
|
|
else:
|
|
|
|
fee_in = fee
|
2016-07-28 22:12:20 +02:00
|
|
|
|
2016-11-30 21:20:45 +01:00
|
|
|
return FeeValidator({
|
2017-04-05 01:42:35 +02:00
|
|
|
'currency':fee_in.currency_symbol,
|
2016-11-30 21:20:45 +01:00
|
|
|
'amount': self.convert_currency(fee_in.currency_symbol, "LBC", fee_in.amount),
|
|
|
|
'address': fee_in.address
|
2017-04-05 01:42:35 +02:00
|
|
|
})
|