-
-
Notifications
You must be signed in to change notification settings - Fork 696
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
* added quasar backend (not finished) * added demo for quasar backend * deleted unused imports * added Ticker, Candles, Trades support * Trades are now saved to two tables depending on side * updated quasar demo * removed unused callbacks * added shard_size to QuasarCallback, added table creation, added comments explaining _set_table_name method * added candles channel, moved opening connection from init to write * writing to db is now accomplished with connection pool, formated file * added console output for channels * fixed issue where callbacks dict keys was overwritten * symbol column is now of symbol type, only one table per callback, create table query is created for each callback type * removed binance * changed symbol tables, exchange is now of symbol type * add more channels * add initial BookQuasar implementation * change plugin name from quasar to quasardb * add quasrdb backend to setup * add quasardb to INSTALL.md * add quasardb to README.md * Update INSTALL.md * change "book" create table query * BookQuasar now stores only best ask, best bid * fix best_ask_amount * add more examples * add comment to BookQuasar format method * fix naming * change queries * use numpy instead of pandas * swap timestamp with receipt_timestamp * update quasardb requirements * small changes * Add QuasarDB to AUTHORS.md * use binance for TICKER channel * replace host and port args with uri in QuasarCallback
- Loading branch information
1 parent
7a982de
commit db8d22d
Showing
6 changed files
with
217 additions
and
1 deletion.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,175 @@ | ||
from datetime import datetime, timedelta | ||
import quasardb.pool as pool | ||
import quasardb.numpy as qdbnp | ||
import numpy as np | ||
from cryptofeed.backends.backend import BackendCallback | ||
|
||
|
||
class QuasarCallback(BackendCallback): | ||
def __init__(self, uri="qdb://127.0.0.1:2836", username: str = "", private_key: str = "", public_key: str = "", none_to=None, shard_size: timedelta = timedelta(minutes=15)): | ||
self.numeric_type = float | ||
self.table = "" | ||
self.running = True | ||
self.none_to = none_to | ||
self.shard_size = self._get_str_timedelta(shard_size) | ||
|
||
pool.initialize(uri=uri, user_name=username, user_private_key=private_key, cluster_public_key=public_key) | ||
|
||
def _get_str_timedelta(self, delta: timedelta): | ||
# calculate the number of hours, minutes, and remaining seconds from timedelta, return it in correct format for query | ||
hours, remainder = divmod(delta.total_seconds(), 3600) | ||
minutes, seconds = divmod(remainder, 60) | ||
return f"{int(hours)}hour {int(minutes)}min {int(seconds)}s" | ||
|
||
def format(self, data: dict): | ||
data['timestamp'] = np.datetime64(datetime.utcfromtimestamp(data['timestamp']), 'ns') | ||
data['receipt_timestamp'] = np.datetime64(datetime.utcfromtimestamp(data['receipt_timestamp']), 'ns') | ||
data['timestamp'], data['receipt_timestamp'] = data['receipt_timestamp'], data['timestamp'] | ||
index = data['timestamp'] | ||
data.pop('timestamp') | ||
return index, data | ||
|
||
def _set_table_name(self, data: dict): | ||
# setting table name | ||
# {channel}/{exchange}/{symbol_1-symbol_2} | ||
# eg. ticker/coinbase/btc-usd | ||
self.table = f"{self.table_prefix.lower()}/{data['exchange'].lower()}/{data['symbol'].lower()}" | ||
|
||
def _create_table(self, conn): | ||
if not conn.table(self.table).exists(): | ||
conn.query(self.query) | ||
|
||
def _insert_format(self, date: np.datetime64, data: dict): | ||
# converts values to np.array | ||
for key, value in data.items(): | ||
data[key] = np.array([value]) | ||
return np.array([date]), data | ||
|
||
async def write(self, data: dict): | ||
self._set_table_name(data) | ||
self._create_query() | ||
index, data = self.format(data) | ||
idx, np_array = self._insert_format(index, data) | ||
# write to table, if table doesnt exist it will be created with specified shard_size value | ||
with pool.instance().connect() as conn: | ||
self._create_table(conn) | ||
qdbnp.write_arrays(np_array, conn, conn.table(self.table), index=idx, fast=True, _async=True) | ||
|
||
|
||
class TickerQuasar(QuasarCallback): | ||
table_prefix = "ticker" | ||
|
||
def _create_query(self): | ||
self.query = f'CREATE TABLE "{self.table}" (exchange SYMBOL(exchange), symbol SYMBOL(symbol), bid DOUBLE, ask DOUBLE, receipt_timestamp TIMESTAMP) SHARD_SIZE = {self.shard_size}' | ||
|
||
|
||
class TradeQuasar(QuasarCallback): | ||
table_prefix = "trades" | ||
|
||
def _create_query(self): | ||
self.query = f'CREATE TABLE "{self.table}" (exchange SYMBOL(exchange), symbol SYMBOL(symbol), side SYMBOL(side), amount DOUBLE, price DOUBLE, id STRING, type SYMBOL(type), receipt_timestamp TIMESTAMP) SHARD_SIZE = {self.shard_size}' | ||
|
||
|
||
class CandlesQuasar(QuasarCallback): | ||
table_prefix = "candles" | ||
|
||
def format(self, data: dict): | ||
index, data = super().format(data) | ||
data['start'] = datetime.utcfromtimestamp(data['start']) | ||
data['stop'] = datetime.utcfromtimestamp(data['stop']) | ||
data['closed'] = int(data['closed']) | ||
return index, data | ||
|
||
def _create_query(self): | ||
self.query = f'CREATE TABLE "{self.table}" (exchange SYMBOL(exchange), symbol SYMBOL(symbol), start TIMESTAMP, stop TIMESTAMP, interval STRING, trades STRING, open DOUBLE, close DOUBLE, high DOUBLE, low DOUBLE, volume DOUBLE, closed INT64, receipt_timestamp TIMESTAMP) SHARD_SIZE = {self.shard_size}' | ||
|
||
|
||
class FundingQuasar(QuasarCallback): | ||
table_prefix = "funding" | ||
|
||
def format(self, data: dict): | ||
index, data = super().format(data) | ||
data['next_funding_time'] = datetime.utcfromtimestamp(data['next_funding_time']) | ||
return index, data | ||
|
||
def _create_query(self): | ||
self.query = f'CREATE TABLE "{self.table}" (exchange SYMBOL(exchange), symbol SYMBOL(symbol), mark_price DOUBLE, rate DOUBLE, next_funding_time TIMESTAMP, predicted_rate DOUBLE, receipt_timestamp TIMESTAMP) SHARD_SIZE = {self.shard_size}' | ||
|
||
|
||
class BookQuasar(QuasarCallback): | ||
table_prefix = "book" | ||
|
||
def format(self, data: dict): | ||
index, data = super().format(data) | ||
# store only best bid and best ask | ||
if not data['book']: | ||
best_bid = max(data["delta"]["bid"], key=lambda x: x[0]) | ||
best_ask = min(data["delta"]["ask"], key=lambda x: x[0]) | ||
|
||
data['best_bid_price'] = best_bid[0] | ||
data['best_bid_amount'] = best_bid[1] | ||
data['best_ask_price'] = best_ask[0] | ||
data['best_ask_amount'] = best_ask[1] | ||
data.pop('delta') | ||
else: | ||
best_bid = max(data["book"]["bid"].keys()) | ||
best_ask = min(data["book"]["ask"].keys()) | ||
|
||
data['best_bid_price'] = best_bid | ||
data['best_bid_amount'] = data["book"]["bid"][best_bid] | ||
data['best_ask_price'] = best_ask | ||
data['best_ask_amount'] = data["book"]["ask"][best_ask] | ||
data.pop('book') | ||
return index, data | ||
|
||
def _create_query(self): | ||
self.query = f'CREATE TABLE "{self.table}" (exchange SYMBOL(exchange), symbol SYMBOL(symbol), best_bid_price DOUBLE, best_bid_amount DOUBLE, best_ask_price DOUBLE, best_ask_amount DOUBLE, receipt_timestamp TIMESTAMP) SHARD_SIZE = {self.shard_size}' | ||
|
||
|
||
class LiquidationsQuasar(QuasarCallback): | ||
table_prefix = "liquidations" | ||
|
||
def _create_query(self): | ||
self.query = f'CREATE TABLE "{self.table}" (exchange SYMBOL(exchange), symbol SYMBOL(symbol), side SYMBOL(side), quantity DOUBLE, price DOUBLE, id STRING, status SYMBOL(type), receipt_timestamp TIMESTAMP) SHARD_SIZE = {self.shard_size}' | ||
|
||
|
||
class OpenInterestQuasar(QuasarCallback): | ||
table_prefix = "open_interest" | ||
|
||
def _create_query(self): | ||
self.query = f'CREATE TABLE "{self.table}" (exchange SYMBOL(exchange), symbol SYMBOL(symbol), open_interest FLOAT, receipt_timestamp TIMESTAMP) SHARD_SIZE = {self.shard_size}' | ||
|
||
|
||
class OrderInfoQuasar(QuasarCallback): | ||
table_prefix = "order_info" | ||
|
||
def _create_query(self): | ||
self.query = f'CREATE TABLE "{self.table}" (exchange SYMBOL(exchange), symbol SYMBOL(symbol), id STRING, client_order_id STRING, side SYMBOL(side), status SYMBOL(type), type SYMBOL(type), price DOUBLE, amount DOUBLE, remaining DOUBLE, account STRING, receipt_timestamp TIMESTAMP) SHARD_SIZE = {self.shard_size}' | ||
|
||
|
||
class TransactionsQuasar(QuasarCallback): | ||
table_prefix = "transactions" | ||
|
||
def _create_query(self): | ||
self.query = f'CREATE TABLE "{self.table}" (exchange SYMBOL(exchange), currency SYMBOL(currency), type SYMBOL(type), status SYMBOL(type), amount DOUBLE, receipt_timestamp TIMESTAMP) SHARD_SIZE = {self.shard_size}' | ||
|
||
|
||
class BalancesQuasar(QuasarCallback): | ||
table_prefix = "balances" | ||
|
||
def _create_query(self): | ||
self.query = f'CREATE TABLE "{self.table}" (exchange SYMBOL(exchange), currency SYMBOL(currency), balance DOUBLE, reserved DOUBLE, receipt_timestamp TIMESTAMP) SHARD_SIZE = {self.shard_size}' | ||
|
||
|
||
class FillsQuasar(QuasarCallback): | ||
table_prefix = "fills" | ||
|
||
def _create_query(self): | ||
self.query = f'CREATE TABLE "{self.table}" (exchange SYMBOL(exchange), symbol SYMBOL(symbol), price DOUBLE, amount DOUBLE, side SYMBOL(side), fee DOUBLE, id STRING, order_id STRING, liquidity DOUBLE, type SYMBOL(type), account STRING, receipt_timestamp TIMESTAMP) SHARD_SIZE = {self.shard_size}' | ||
|
||
|
||
class IndexQuasar(QuasarCallback): | ||
table_prefix = "index" | ||
|
||
def _create_query(self): | ||
self.query = f'CREATE TABLE "{self.table}" (exchange SYMBOL(exchange), symbol SYMBOL(symbol), price DOUBLE, receipt_timestamp TIMESTAMP) SHARD_SIZE = {self.shard_size}' |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,33 @@ | ||
from cryptofeed import FeedHandler | ||
from cryptofeed.exchanges import * | ||
from cryptofeed.backends.quasardb import * | ||
|
||
|
||
async def feed_info(data, receipt_timestamp): | ||
print(f'{data} recived at {receipt_timestamp}') | ||
|
||
|
||
def main(): | ||
f = FeedHandler() | ||
|
||
# save to database | ||
f.add_feed(Binance(channels=[TICKER], symbols=['BTC-USDT'], callbacks={TICKER: TickerQuasar()})) | ||
f.add_feed(Coinbase(channels=[TRADES], symbols=['BTC-USD'], callbacks={TRADES: TradeQuasar()})) | ||
f.add_feed(Bybit(channels=[CANDLES], symbols=['BTC-USD-PERP'], callbacks={CANDLES: CandlesQuasar()})) | ||
f.add_feed(Bybit(channels=[OPEN_INTEREST], symbols=['BTC-USD-PERP'], callbacks={OPEN_INTEREST: OpenInterestQuasar()})) | ||
f.add_feed(Bybit(channels=[INDEX], symbols=['BTC-USD-PERP'], callbacks={INDEX: IndexQuasar()})) | ||
f.add_feed(Bybit(channels=[LIQUIDATIONS], symbols=['BTC-USD-PERP'], callbacks={LIQUIDATIONS: LiquidationsQuasar()})) | ||
|
||
# print to console | ||
f.add_feed(Binance(channels=[TICKER], symbols=['BTC-USDT'], callbacks={TICKER: feed_info})) | ||
f.add_feed(Coinbase(channels=[TRADES], symbols=['BTC-USD'], callbacks={TRADES: feed_info})) | ||
f.add_feed(Bybit(channels=[CANDLES], symbols=['BTC-USD-PERP'], callbacks={CANDLES: feed_info})) | ||
f.add_feed(Bybit(channels=[OPEN_INTEREST], symbols=['BTC-USD-PERP'], callbacks={OPEN_INTEREST: feed_info})) | ||
f.add_feed(Bybit(channels=[INDEX], symbols=['BTC-USD-PERP'], callbacks={INDEX: feed_info})) | ||
f.add_feed(Bybit(channels=[LIQUIDATIONS], symbols=['BTC-USD-PERP'], callbacks={LIQUIDATIONS: feed_info})) | ||
|
||
f.run() | ||
|
||
|
||
if __name__ == '__main__': | ||
main() |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters