2023-07-22 16:06:38 +00:00
|
|
|
# No copyright
|
|
|
|
import asyncio
|
|
|
|
import logging
|
|
|
|
|
|
|
|
class MM:
|
|
|
|
def __init__(self, player, pair, config):
|
|
|
|
self._player = player
|
|
|
|
self._pair = pair
|
|
|
|
|
2023-07-22 23:55:06 +00:00
|
|
|
self._lastBuy = ""
|
|
|
|
self._lastSell = ""
|
|
|
|
self._lastPrice = None
|
2023-07-22 16:06:38 +00:00
|
|
|
|
2023-07-22 23:55:06 +00:00
|
|
|
self._delta = float(config["delta"])
|
|
|
|
self._positiveCut = float(config["+cut"])
|
|
|
|
self._negativeCut = float(config["-cut"])
|
2023-07-22 16:06:38 +00:00
|
|
|
|
|
|
|
async def update(self):
|
2023-07-22 23:55:06 +00:00
|
|
|
sales = self._player.assets[self._pair.names[0]]
|
|
|
|
crncy = self._player.assets[self._pair.names[1]]
|
|
|
|
price = self._pair.ticker.price
|
|
|
|
|
|
|
|
lot = (sales.amount + crncy.amount / price) / 2
|
|
|
|
pos = sales.amount / lot - 1
|
|
|
|
|
|
|
|
# evaluate if cutting should be executed
|
|
|
|
negCut = False
|
|
|
|
posCut = False
|
|
|
|
if self._lastPrice is not None:
|
|
|
|
change = (price - self._lastPrice) / self._lastPrice
|
|
|
|
negCut = pos > 0.5 and change < -abs(self._negativeCut)
|
|
|
|
posCut = pos < -0.5 and change > abs(self._positiveCut)
|
|
|
|
cutting = negCut or posCut
|
|
|
|
|
|
|
|
# check if the past orders are active
|
2023-07-22 16:06:38 +00:00
|
|
|
lastBuyAgreed = (self._lastBuy is not None and
|
|
|
|
self._lastBuy not in self._player.orders)
|
|
|
|
lastSellAgreed = (self._lastSell is not None and
|
|
|
|
self._lastSell not in self._player.orders)
|
|
|
|
|
2023-07-22 23:55:06 +00:00
|
|
|
if not lastBuyAgreed and not lastSellAgreed and not cutting:
|
2023-07-22 16:06:38 +00:00
|
|
|
return
|
|
|
|
|
2023-07-22 23:55:06 +00:00
|
|
|
# cancel active orders
|
2023-07-22 16:06:38 +00:00
|
|
|
cancel_orders = []
|
|
|
|
if not lastBuyAgreed and self._lastBuy is not None:
|
2023-07-22 23:55:06 +00:00
|
|
|
logging.info(f"cancel buy")
|
2023-07-22 16:06:38 +00:00
|
|
|
cancel_orders.append(self._lastBuy)
|
2023-07-22 23:55:06 +00:00
|
|
|
elif lastBuyAgreed:
|
|
|
|
logging.info(f"agree buy")
|
|
|
|
|
2023-07-22 16:06:38 +00:00
|
|
|
if not lastSellAgreed and self._lastSell is not None:
|
2023-07-22 23:55:06 +00:00
|
|
|
logging.info(f"cancel sell")
|
2023-07-22 16:06:38 +00:00
|
|
|
cancel_orders.append(self._lastSell)
|
2023-07-22 23:55:06 +00:00
|
|
|
elif lastSellAgreed:
|
|
|
|
logging.info(f"agree sell")
|
|
|
|
|
2023-07-22 16:06:38 +00:00
|
|
|
if len(cancel_orders) != 0:
|
|
|
|
await self._player.cancel(self._pair.names, cancel_orders)
|
2023-07-23 07:55:27 +00:00
|
|
|
self._lastBuy = None
|
|
|
|
self._lastSell = None
|
2023-07-22 16:06:38 +00:00
|
|
|
|
2023-07-22 23:55:06 +00:00
|
|
|
# execute cutting
|
|
|
|
if cutting:
|
|
|
|
if negCut:
|
|
|
|
logging.info(f"-cut (change: {change}, pos: {pos} -> 0)")
|
|
|
|
if posCut:
|
|
|
|
logging.info(f"+cut (change: {change}, pos: {pos} -> 0)")
|
|
|
|
self._lastBuy = None
|
|
|
|
self._lastSell = None
|
|
|
|
sell_amount = pos * lot
|
|
|
|
if sell_amount > 0:
|
|
|
|
await self._player.orderMarketSell(self._pair.names, amount=sell_amount)
|
|
|
|
else:
|
|
|
|
await self._player.orderMarketBuy(self._pair.names, amount=-buy_amount)
|
|
|
|
return
|
2023-07-22 16:06:38 +00:00
|
|
|
|
2023-07-22 23:55:06 +00:00
|
|
|
# execute new cycle
|
2023-07-23 07:55:27 +00:00
|
|
|
self._lastPrice = None
|
2023-07-22 16:06:38 +00:00
|
|
|
buy_price = price * (1 - self._delta);
|
|
|
|
sell_price = price * (1 + self._delta);
|
|
|
|
|
2023-07-23 07:55:27 +00:00
|
|
|
buy_amount = min(_quad( pos) * lot, crncy.amount / buy_price)
|
|
|
|
sell_amount = min(_quad(-pos) * lot, sales.amount)
|
|
|
|
order_unit = pow(10, -sales.precision)
|
2023-07-22 16:06:38 +00:00
|
|
|
|
2023-07-22 23:55:06 +00:00
|
|
|
logging.info(f"new cycle (pos: {pos}, price: {price}, buy: {buy_amount}, sell: {sell_amount})")
|
|
|
|
|
2023-07-22 16:06:38 +00:00
|
|
|
async def buy():
|
2023-07-23 07:55:27 +00:00
|
|
|
self._lastBuy = (
|
|
|
|
None if order_unit > buy_amount else
|
2023-07-22 16:06:38 +00:00
|
|
|
await self._player.orderLimitBuy(
|
|
|
|
self._pair.names, amount = buy_amount, price = buy_price))
|
|
|
|
|
|
|
|
async def sell():
|
2023-07-23 07:55:27 +00:00
|
|
|
self._lastSell = (
|
|
|
|
None if order_unit > sell_amount else
|
2023-07-22 16:06:38 +00:00
|
|
|
await self._player.orderLimitSell(
|
|
|
|
self._pair.names, amount = sell_amount, price = sell_price))
|
|
|
|
|
2023-07-23 07:55:27 +00:00
|
|
|
await asyncio.gather(buy(), sell())
|
2023-07-22 23:55:06 +00:00
|
|
|
self._lastPrice = price
|
2023-07-22 16:06:38 +00:00
|
|
|
|
|
|
|
# https://kijitora-2018.hatenablog.com/entry/2018/12/23/102913
|
|
|
|
def _quad(x):
|
|
|
|
if x < -1:
|
|
|
|
return 1
|
|
|
|
if x <= 1:
|
|
|
|
return -1/4 * (x+1)**2 + 1
|
|
|
|
return 0
|
|
|
|
|