Compare commits
2 Commits
b2cfa3444f
...
499b2d0090
Author | SHA1 | Date |
---|---|---|
|
499b2d0090 | |
|
8b0f1e7045 |
|
@ -219,9 +219,11 @@ def get_config() -> dict[str, Any]:
|
||||||
touch_if_dne=True,
|
touch_if_dne=True,
|
||||||
)
|
)
|
||||||
section: dict = {}
|
section: dict = {}
|
||||||
section['deribit'] = conf.get('deribit')
|
section = conf.get('deribit')
|
||||||
|
|
||||||
section['log'] = {}
|
section['log'] = {}
|
||||||
|
section['log']['filename'] = 'feedhandler.log'
|
||||||
|
section['log']['level'] = 'DEBUG'
|
||||||
section['log']['disabled'] = True
|
section['log']['disabled'] = True
|
||||||
|
|
||||||
if section is None:
|
if section is None:
|
||||||
|
@ -230,6 +232,22 @@ def get_config() -> dict[str, Any]:
|
||||||
|
|
||||||
return section
|
return section
|
||||||
|
|
||||||
|
def get_fh_config() -> dict[str, Any]:
|
||||||
|
conf_option = get_config().get('option', {})
|
||||||
|
conf_log = get_config().get('log', {})
|
||||||
|
|
||||||
|
return {
|
||||||
|
'log': {
|
||||||
|
'filename': conf_log.get('filename'),
|
||||||
|
'level': conf_log.get('level'),
|
||||||
|
'disabled': conf_log.get('disabled')
|
||||||
|
},
|
||||||
|
'deribit': {
|
||||||
|
'key_id': conf_option.get('api_key'),
|
||||||
|
'key_secret': conf_option.get('api_secret')
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
class Client:
|
class Client:
|
||||||
|
|
||||||
|
@ -241,10 +259,10 @@ class Client:
|
||||||
) -> None:
|
) -> None:
|
||||||
self._pairs: dict[str, Any] = None
|
self._pairs: dict[str, Any] = None
|
||||||
|
|
||||||
config = get_config().get('deribit', {})
|
config = get_config().get('option', {})
|
||||||
|
|
||||||
self._key_id = config.get('key_id')
|
self._key_id = config.get('api_key')
|
||||||
self._key_secret = config.get('key_secret')
|
self._key_secret = config.get('api_secret')
|
||||||
|
|
||||||
self.json_rpc = json_rpc
|
self.json_rpc = json_rpc
|
||||||
|
|
||||||
|
@ -466,7 +484,6 @@ async def get_client(
|
||||||
) as json_rpc
|
) as json_rpc
|
||||||
):
|
):
|
||||||
client = Client(json_rpc)
|
client = Client(json_rpc)
|
||||||
|
|
||||||
_refresh_token: Optional[str] = None
|
_refresh_token: Optional[str] = None
|
||||||
_access_token: Optional[str] = None
|
_access_token: Optional[str] = None
|
||||||
|
|
||||||
|
@ -536,7 +553,7 @@ async def get_client(
|
||||||
|
|
||||||
@acm
|
@acm
|
||||||
async def open_feed_handler():
|
async def open_feed_handler():
|
||||||
fh = FeedHandler(config=get_config())
|
fh = FeedHandler(config=get_fh_config())
|
||||||
yield fh
|
yield fh
|
||||||
await to_asyncio.run_task(fh.stop_async)
|
await to_asyncio.run_task(fh.stop_async)
|
||||||
|
|
||||||
|
@ -581,11 +598,11 @@ async def aio_price_feed_relay(
|
||||||
'price': float(data.ask_price), 'size': float(data.ask_size)}
|
'price': float(data.ask_price), 'size': float(data.ask_size)}
|
||||||
]
|
]
|
||||||
}))
|
}))
|
||||||
|
sym: Symbol = piker_sym_to_cb_sym(instrument)
|
||||||
fh.add_feed(
|
fh.add_feed(
|
||||||
DERIBIT,
|
DERIBIT,
|
||||||
channels=[TRADES, L1_BOOK],
|
channels=[TRADES, L1_BOOK],
|
||||||
symbols=[piker_sym_to_cb_sym(instrument)],
|
symbols=[sym],
|
||||||
callbacks={
|
callbacks={
|
||||||
TRADES: _trade,
|
TRADES: _trade,
|
||||||
L1_BOOK: _l1
|
L1_BOOK: _l1
|
||||||
|
@ -626,9 +643,9 @@ async def maybe_open_price_feed(
|
||||||
async with maybe_open_context(
|
async with maybe_open_context(
|
||||||
acm_func=open_price_feed,
|
acm_func=open_price_feed,
|
||||||
kwargs={
|
kwargs={
|
||||||
'instrument': instrument
|
'instrument': instrument.split('.')[0]
|
||||||
},
|
},
|
||||||
key=f'{instrument}-price',
|
key=f'{instrument.split('.')[0]}-price',
|
||||||
) as (cache_hit, feed):
|
) as (cache_hit, feed):
|
||||||
if cache_hit:
|
if cache_hit:
|
||||||
yield broadcast_receiver(feed, 10)
|
yield broadcast_receiver(feed, 10)
|
||||||
|
@ -693,10 +710,10 @@ async def maybe_open_order_feed(
|
||||||
async with maybe_open_context(
|
async with maybe_open_context(
|
||||||
acm_func=open_order_feed,
|
acm_func=open_order_feed,
|
||||||
kwargs={
|
kwargs={
|
||||||
'instrument': instrument,
|
'instrument': instrument.split('.')[0],
|
||||||
'fh': fh
|
'fh': fh
|
||||||
},
|
},
|
||||||
key=f'{instrument}-order',
|
key=f'{instrument.split('.')[0]}-order',
|
||||||
) as (cache_hit, feed):
|
) as (cache_hit, feed):
|
||||||
if cache_hit:
|
if cache_hit:
|
||||||
yield broadcast_receiver(feed, 10)
|
yield broadcast_receiver(feed, 10)
|
||||||
|
|
|
@ -156,7 +156,7 @@ async def stream_quotes(
|
||||||
},
|
},
|
||||||
}
|
}
|
||||||
|
|
||||||
nsym = piker_sym_to_cb_sym(sym)
|
nsym = piker_sym_to_cb_sym(sym.split('.')[0])
|
||||||
|
|
||||||
async with maybe_open_price_feed(sym) as stream:
|
async with maybe_open_price_feed(sym) as stream:
|
||||||
|
|
||||||
|
|
|
@ -0,0 +1,119 @@
|
||||||
|
# piker: trading gear for hackers
|
||||||
|
# Copyright (C) Tyler Goodlet (in stewardship for pikers)
|
||||||
|
|
||||||
|
# This program is free software: you can redistribute it and/or modify
|
||||||
|
# it under the terms of the GNU Affero General Public License as published by
|
||||||
|
# the Free Software Foundation, either version 3 of the License, or
|
||||||
|
# (at your option) any later version.
|
||||||
|
|
||||||
|
# This program 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 Affero General Public License for more details.
|
||||||
|
|
||||||
|
# You should have received a copy of the GNU Affero General Public License
|
||||||
|
# along with this program. If not, see <https://www.gnu.org/licenses/>.
|
||||||
|
|
||||||
|
"""
|
||||||
|
Per market data-type definitions and schemas types.
|
||||||
|
|
||||||
|
"""
|
||||||
|
from __future__ import annotations
|
||||||
|
from typing import (
|
||||||
|
Literal,
|
||||||
|
)
|
||||||
|
from decimal import Decimal
|
||||||
|
|
||||||
|
from msgspec import field
|
||||||
|
|
||||||
|
from piker.types import Struct
|
||||||
|
|
||||||
|
|
||||||
|
# API endpoint paths by venue / sub-API
|
||||||
|
_domain: str = 'deribit.com'
|
||||||
|
_url = f'https://www.{_domain}'
|
||||||
|
|
||||||
|
# WEBsocketz
|
||||||
|
_ws_url: str = f'wss://www.{_domain}/ws/api/v2'
|
||||||
|
|
||||||
|
# test nets
|
||||||
|
_testnet_ws_url: str = f'wss://test.{_domain}/ws/api/v2'
|
||||||
|
|
||||||
|
MarketType = Literal[
|
||||||
|
'option'
|
||||||
|
]
|
||||||
|
|
||||||
|
|
||||||
|
def get_api_eps(venue: MarketType) -> tuple[str, str]:
|
||||||
|
'''
|
||||||
|
Return API ep root paths per venue.
|
||||||
|
|
||||||
|
'''
|
||||||
|
return {
|
||||||
|
'option': (
|
||||||
|
_ws_url,
|
||||||
|
),
|
||||||
|
}[venue]
|
||||||
|
|
||||||
|
|
||||||
|
class OptionPair(Struct, frozen=True, kw_only=True):
|
||||||
|
|
||||||
|
symbol: str
|
||||||
|
venue: str
|
||||||
|
|
||||||
|
# src
|
||||||
|
quote_currency: str
|
||||||
|
|
||||||
|
# dst
|
||||||
|
base_currency: str
|
||||||
|
|
||||||
|
tick_size: float # 0.0001
|
||||||
|
tick_size_steps: list[dict] # [{'above_price': 0.005, 'tick_size': 0.0005}]
|
||||||
|
taker_commission: float # 0.0003
|
||||||
|
strike: float # 5000.0
|
||||||
|
settlement_period: str # 'day'
|
||||||
|
settlement_currency: str # "BTC",
|
||||||
|
rfq: bool # false
|
||||||
|
quote_currency: str # 'BTC'
|
||||||
|
price_index: str # 'btc_usd'
|
||||||
|
option_type: str # 'call'
|
||||||
|
min_trade_amount: float # 0.1
|
||||||
|
maker_commission: float # 0.0003
|
||||||
|
kind: str # 'option'
|
||||||
|
is_active: bool # true
|
||||||
|
instrument_type: str # 'reversed'
|
||||||
|
instrument_name: str # 'BTC-1SEP24-55000-C'
|
||||||
|
instrument_id: int # 364671
|
||||||
|
expiration_timestamp: int # 1725177600000
|
||||||
|
creation_timestamp: int # 1724918461000
|
||||||
|
counter_currency: str # 'USD'
|
||||||
|
contract_size: float # '1.0'
|
||||||
|
block_trade_tick_size: float # '0.0001'
|
||||||
|
block_trade_min_trade_amount: int # '25'
|
||||||
|
block_trade_commission: float # '0.003'
|
||||||
|
|
||||||
|
|
||||||
|
# NOTE: see `.data._symcache.SymbologyCache.load()` for why
|
||||||
|
ns_path: str = 'piker.brokers.deribit:OptionPair'
|
||||||
|
|
||||||
|
@property
|
||||||
|
def venue(self) -> str:
|
||||||
|
return 'OPTION'
|
||||||
|
|
||||||
|
@property
|
||||||
|
def bs_fqme(self) -> str:
|
||||||
|
return f'{self.symbol}.OPTION'
|
||||||
|
|
||||||
|
@property
|
||||||
|
def bs_src_asset(self) -> str:
|
||||||
|
return f'{self.quote_currency}'
|
||||||
|
|
||||||
|
@property
|
||||||
|
def bs_dst_asset(self) -> str:
|
||||||
|
return f'{self.base_currency}'
|
||||||
|
|
||||||
|
@property
|
||||||
|
def bs_mktid(self) -> str:
|
||||||
|
return f'{self.symbol}.{self.venue}'
|
||||||
|
|
||||||
|
|
Loading…
Reference in New Issue