mirror of
https://github.com/c9s/bbgo.git
synced 2024-11-10 09:11:55 +00:00
458 lines
11 KiB
Go
458 lines
11 KiB
Go
|
package xmaker
|
||
|
|
||
|
import (
|
||
|
"context"
|
||
|
"fmt"
|
||
|
"hash/fnv"
|
||
|
"math"
|
||
|
"sync"
|
||
|
"time"
|
||
|
|
||
|
"github.com/sirupsen/logrus"
|
||
|
|
||
|
"github.com/c9s/bbgo/pkg/bbgo"
|
||
|
"github.com/c9s/bbgo/pkg/fixedpoint"
|
||
|
"github.com/c9s/bbgo/pkg/service"
|
||
|
"github.com/c9s/bbgo/pkg/types"
|
||
|
)
|
||
|
|
||
|
var defaultMargin = fixedpoint.NewFromFloat(0.01)
|
||
|
|
||
|
var defaultQuantity = fixedpoint.NewFromFloat(0.001)
|
||
|
|
||
|
const ID = "xmaker"
|
||
|
|
||
|
const stateKey = "state-v1"
|
||
|
|
||
|
var log = logrus.WithField("strategy", ID)
|
||
|
|
||
|
func init() {
|
||
|
bbgo.RegisterStrategy(ID, &Strategy{})
|
||
|
}
|
||
|
|
||
|
func (s *Strategy) ID() string {
|
||
|
return ID
|
||
|
}
|
||
|
|
||
|
type State struct {
|
||
|
HedgePosition fixedpoint.Value `json:"hedgePosition"`
|
||
|
}
|
||
|
|
||
|
type Strategy struct {
|
||
|
*bbgo.Graceful
|
||
|
*bbgo.Notifiability
|
||
|
*bbgo.Persistence
|
||
|
|
||
|
Symbol string `json:"symbol"`
|
||
|
SourceExchange string `json:"sourceExchange"`
|
||
|
MakerExchange string `json:"makerExchange"`
|
||
|
|
||
|
UpdateInterval types.Duration `json:"updateInterval"`
|
||
|
HedgeInterval types.Duration `json:"hedgeInterval"`
|
||
|
|
||
|
Margin fixedpoint.Value `json:"margin"`
|
||
|
BidMargin fixedpoint.Value `json:"bidMargin"`
|
||
|
AskMargin fixedpoint.Value `json:"askMargin"`
|
||
|
Quantity fixedpoint.Value `json:"quantity"`
|
||
|
QuantityMultiplier fixedpoint.Value `json:"quantityMultiplier"`
|
||
|
DisableHedge bool `json:"disableHedge"`
|
||
|
|
||
|
NumLayers int `json:"numLayers"`
|
||
|
Pips int `json:"pips"`
|
||
|
|
||
|
makerSession *bbgo.ExchangeSession
|
||
|
sourceSession *bbgo.ExchangeSession
|
||
|
|
||
|
sourceMarket types.Market
|
||
|
makerMarket types.Market
|
||
|
|
||
|
state *State
|
||
|
|
||
|
book *types.StreamOrderBook
|
||
|
activeMakerOrders *bbgo.LocalActiveOrderBook
|
||
|
|
||
|
orderStore *bbgo.OrderStore
|
||
|
|
||
|
lastPrice float64
|
||
|
groupID int64
|
||
|
|
||
|
stopC chan struct{}
|
||
|
}
|
||
|
|
||
|
func (s *Strategy) CrossSubscribe(sessions map[string]*bbgo.ExchangeSession) {
|
||
|
sourceSession, ok := sessions[s.SourceExchange]
|
||
|
if !ok {
|
||
|
panic(fmt.Errorf("source exchange %s is not defined", s.SourceExchange))
|
||
|
}
|
||
|
|
||
|
log.Infof("subscribing %s from %s", s.Symbol, s.SourceExchange)
|
||
|
sourceSession.Subscribe(types.BookChannel, s.Symbol, types.SubscribeOptions{})
|
||
|
}
|
||
|
|
||
|
func (s *Strategy) updateQuote(ctx context.Context) {
|
||
|
if err := s.makerSession.Exchange.CancelOrders(ctx, s.activeMakerOrders.Orders()...); err != nil {
|
||
|
log.WithError(err).Errorf("can not cancel orders")
|
||
|
return
|
||
|
}
|
||
|
|
||
|
// avoid unlock issue
|
||
|
time.Sleep(800 * time.Millisecond)
|
||
|
|
||
|
sourceBook := s.book.Get()
|
||
|
if len(sourceBook.Bids) == 0 || len(sourceBook.Asks) == 0 {
|
||
|
return
|
||
|
}
|
||
|
|
||
|
if valid, err := sourceBook.IsValid(); !valid {
|
||
|
log.WithError(err).Error("invalid order book: %v", err)
|
||
|
return
|
||
|
}
|
||
|
|
||
|
bestBidPrice := sourceBook.Bids[0].Price
|
||
|
bestAskPrice := sourceBook.Asks[0].Price
|
||
|
log.Infof("best bid price %f, best ask price: %f", bestBidPrice.Float64(), bestAskPrice.Float64())
|
||
|
|
||
|
bidQuantity := s.Quantity
|
||
|
bidPrice := bestBidPrice.MulFloat64(1.0 - s.BidMargin.Float64())
|
||
|
|
||
|
askQuantity := s.Quantity
|
||
|
askPrice := bestAskPrice.MulFloat64(1.0 + s.AskMargin.Float64())
|
||
|
|
||
|
log.Infof("quote bid price: %f ask price: %f", bidPrice.Float64(), askPrice.Float64())
|
||
|
|
||
|
var submitOrders []types.SubmitOrder
|
||
|
|
||
|
balances := s.makerSession.Account.Balances()
|
||
|
makerQuota := &bbgo.QuotaTransaction{}
|
||
|
if b, ok := balances[s.makerMarket.BaseCurrency]; ok {
|
||
|
makerQuota.BaseAsset.Add(b.Available)
|
||
|
}
|
||
|
if b, ok := balances[s.makerMarket.QuoteCurrency]; ok {
|
||
|
makerQuota.QuoteAsset.Add(b.Available)
|
||
|
}
|
||
|
|
||
|
hedgeBalances := s.sourceSession.Account.Balances()
|
||
|
hedgeQuota := &bbgo.QuotaTransaction{}
|
||
|
if b, ok := hedgeBalances[s.sourceMarket.BaseCurrency]; ok {
|
||
|
hedgeQuota.BaseAsset.Add(b.Available)
|
||
|
}
|
||
|
if b, ok := hedgeBalances[s.sourceMarket.QuoteCurrency]; ok {
|
||
|
hedgeQuota.QuoteAsset.Add(b.Available)
|
||
|
}
|
||
|
|
||
|
log.Infof("maker quota: %+v", makerQuota)
|
||
|
log.Infof("hedge quota: %+v", hedgeQuota)
|
||
|
|
||
|
for i := 0; i < s.NumLayers; i++ {
|
||
|
// bid orders
|
||
|
if makerQuota.QuoteAsset.Lock(bidQuantity.Mul(bidPrice)) && hedgeQuota.BaseAsset.Lock(bidQuantity) {
|
||
|
// if we bought, then we need to sell the base from the hedge session
|
||
|
submitOrders = append(submitOrders, types.SubmitOrder{
|
||
|
Symbol: s.Symbol,
|
||
|
Type: types.OrderTypeLimit,
|
||
|
Side: types.SideTypeBuy,
|
||
|
Price: bidPrice.Float64(),
|
||
|
Quantity: bidQuantity.Float64(),
|
||
|
TimeInForce: "GTC",
|
||
|
GroupID: s.groupID,
|
||
|
})
|
||
|
|
||
|
makerQuota.Commit()
|
||
|
hedgeQuota.Commit()
|
||
|
} else {
|
||
|
makerQuota.Rollback()
|
||
|
hedgeQuota.Rollback()
|
||
|
}
|
||
|
|
||
|
// ask orders
|
||
|
if makerQuota.BaseAsset.Lock(askQuantity) && hedgeQuota.QuoteAsset.Lock(askQuantity.Mul(askPrice)) {
|
||
|
// if we bought, then we need to sell the base from the hedge session
|
||
|
submitOrders = append(submitOrders, types.SubmitOrder{
|
||
|
Symbol: s.Symbol,
|
||
|
Type: types.OrderTypeLimit,
|
||
|
Side: types.SideTypeSell,
|
||
|
Price: askPrice.Float64(),
|
||
|
Quantity: askQuantity.Float64(),
|
||
|
TimeInForce: "GTC",
|
||
|
GroupID: s.groupID,
|
||
|
})
|
||
|
makerQuota.Commit()
|
||
|
hedgeQuota.Commit()
|
||
|
} else {
|
||
|
makerQuota.Rollback()
|
||
|
hedgeQuota.Rollback()
|
||
|
}
|
||
|
|
||
|
bidPrice -= fixedpoint.NewFromFloat(s.makerMarket.TickSize * float64(s.Pips))
|
||
|
askPrice += fixedpoint.NewFromFloat(s.makerMarket.TickSize * float64(s.Pips))
|
||
|
|
||
|
askQuantity.Mul(s.QuantityMultiplier)
|
||
|
bidQuantity.Mul(s.QuantityMultiplier)
|
||
|
}
|
||
|
|
||
|
if len(submitOrders) == 0 {
|
||
|
return
|
||
|
}
|
||
|
|
||
|
makerOrderExecutor := &bbgo.ExchangeOrderExecutor{Session: s.makerSession}
|
||
|
makerOrders, err := makerOrderExecutor.SubmitOrders(ctx, submitOrders...)
|
||
|
if err != nil {
|
||
|
log.WithError(err).Errorf("order error: %s", err.Error())
|
||
|
return
|
||
|
}
|
||
|
|
||
|
s.activeMakerOrders.Add(makerOrders...)
|
||
|
s.orderStore.Add(makerOrders...)
|
||
|
}
|
||
|
|
||
|
func (s *Strategy) Hedge(ctx context.Context, pos fixedpoint.Value) {
|
||
|
side := types.SideTypeBuy
|
||
|
|
||
|
if pos == 0 {
|
||
|
return
|
||
|
}
|
||
|
|
||
|
quantity := pos
|
||
|
if pos < 0 {
|
||
|
side = types.SideTypeSell
|
||
|
quantity = -pos
|
||
|
}
|
||
|
|
||
|
lastPrice := s.lastPrice
|
||
|
sourceBook := s.book.Get()
|
||
|
switch side {
|
||
|
|
||
|
case types.SideTypeBuy:
|
||
|
if len(sourceBook.Asks) > 0 {
|
||
|
if pv, ok := sourceBook.Asks.First(); ok {
|
||
|
lastPrice = pv.Price.Float64()
|
||
|
}
|
||
|
}
|
||
|
|
||
|
case types.SideTypeSell:
|
||
|
if len(sourceBook.Bids) > 0 {
|
||
|
if pv, ok := sourceBook.Bids.First(); ok {
|
||
|
lastPrice = pv.Price.Float64()
|
||
|
}
|
||
|
}
|
||
|
|
||
|
}
|
||
|
|
||
|
notional := quantity.MulFloat64(lastPrice)
|
||
|
if notional.Float64() <= s.sourceMarket.MinNotional {
|
||
|
log.Warnf("less than min notional %f, skipping", notional.Float64())
|
||
|
return
|
||
|
}
|
||
|
|
||
|
s.Notifiability.Notify("submitting hedge order: %s %s %f", s.Symbol, side, quantity.Float64())
|
||
|
orderExecutor := &bbgo.ExchangeOrderExecutor{Session: s.sourceSession}
|
||
|
returnOrders, err := orderExecutor.SubmitOrders(ctx, types.SubmitOrder{
|
||
|
Symbol: s.Symbol,
|
||
|
Type: types.OrderTypeMarket,
|
||
|
Side: side,
|
||
|
Quantity: quantity.Float64(),
|
||
|
})
|
||
|
|
||
|
if err != nil {
|
||
|
log.WithError(err).Errorf("market order submit error: %s", err.Error())
|
||
|
return
|
||
|
}
|
||
|
|
||
|
s.orderStore.Add(returnOrders...)
|
||
|
}
|
||
|
|
||
|
func (s *Strategy) handleTradeUpdate(trade types.Trade) {
|
||
|
log.Infof("received trade %+v", trade)
|
||
|
|
||
|
if trade.Symbol != s.Symbol {
|
||
|
return
|
||
|
}
|
||
|
|
||
|
if !s.orderStore.Exists(trade.OrderID) {
|
||
|
return
|
||
|
}
|
||
|
|
||
|
q := fixedpoint.NewFromFloat(trade.Quantity)
|
||
|
switch trade.Side {
|
||
|
case types.SideTypeSell:
|
||
|
q = -q
|
||
|
|
||
|
case types.SideTypeBuy:
|
||
|
|
||
|
case types.SideTypeSelf:
|
||
|
// ignore self trades
|
||
|
|
||
|
default:
|
||
|
log.Infof("ignore non sell/buy side trades, got: %v", trade.Side)
|
||
|
return
|
||
|
|
||
|
}
|
||
|
|
||
|
log.Infof("identified trade %d with an existing order: %d", trade.ID, trade.OrderID)
|
||
|
s.Notify("identified %s trade %d with an existing order: %d", trade.Symbol, trade.ID, trade.OrderID)
|
||
|
|
||
|
s.state.HedgePosition.AtomicAdd(q)
|
||
|
|
||
|
pos := s.state.HedgePosition.AtomicLoad()
|
||
|
|
||
|
log.Warnf("position changed: %f", pos.Float64())
|
||
|
s.Notifiability.Notify("%s position is changed to %f", s.Symbol, pos.Float64())
|
||
|
|
||
|
s.lastPrice = trade.Price
|
||
|
}
|
||
|
|
||
|
func (s *Strategy) CrossRun(ctx context.Context, _ bbgo.OrderExecutionRouter, sessions map[string]*bbgo.ExchangeSession) error {
|
||
|
// configure default values
|
||
|
if s.UpdateInterval == 0 {
|
||
|
s.UpdateInterval = types.Duration(time.Second)
|
||
|
}
|
||
|
|
||
|
if s.HedgeInterval == 0 {
|
||
|
s.HedgeInterval = types.Duration(10 * time.Second)
|
||
|
}
|
||
|
|
||
|
if s.NumLayers == 0 {
|
||
|
s.NumLayers = 1
|
||
|
}
|
||
|
|
||
|
if s.BidMargin == 0 {
|
||
|
if s.Margin != 0 {
|
||
|
s.BidMargin = s.Margin
|
||
|
} else {
|
||
|
s.BidMargin = defaultMargin
|
||
|
}
|
||
|
}
|
||
|
|
||
|
if s.AskMargin == 0 {
|
||
|
if s.Margin != 0 {
|
||
|
s.AskMargin = s.Margin
|
||
|
} else {
|
||
|
s.AskMargin = defaultMargin
|
||
|
}
|
||
|
}
|
||
|
|
||
|
if s.Quantity == 0 {
|
||
|
s.Quantity = defaultQuantity
|
||
|
}
|
||
|
|
||
|
|
||
|
// configure sessions
|
||
|
sourceSession, ok := sessions[s.SourceExchange]
|
||
|
if !ok {
|
||
|
return fmt.Errorf("source exchange session %s is not defined", s.SourceExchange)
|
||
|
}
|
||
|
|
||
|
s.sourceSession = sourceSession
|
||
|
|
||
|
makerSession, ok := sessions[s.MakerExchange]
|
||
|
if !ok {
|
||
|
return fmt.Errorf("maker exchange session %s is not defined", s.MakerExchange)
|
||
|
}
|
||
|
|
||
|
s.makerSession = makerSession
|
||
|
|
||
|
s.sourceMarket, ok = s.sourceSession.Market(s.Symbol)
|
||
|
if !ok {
|
||
|
return fmt.Errorf("source session market %s is not defined", s.Symbol)
|
||
|
}
|
||
|
|
||
|
s.makerMarket, ok = s.makerSession.Market(s.Symbol)
|
||
|
if !ok {
|
||
|
return fmt.Errorf("maker session market %s is not defined", s.Symbol)
|
||
|
}
|
||
|
|
||
|
|
||
|
|
||
|
|
||
|
// restore state
|
||
|
instanceID := fmt.Sprintf("%s-%s-%s", ID, s.Symbol)
|
||
|
s.groupID = generateGroupID(instanceID)
|
||
|
log.Infof("using group id %d from fnv(%s)", s.groupID, instanceID)
|
||
|
|
||
|
var state State
|
||
|
|
||
|
// load position
|
||
|
if err := s.Persistence.Load(&state, stateKey); err != nil {
|
||
|
if err != service.ErrPersistenceNotExists {
|
||
|
return err
|
||
|
}
|
||
|
|
||
|
s.state = &State{}
|
||
|
} else {
|
||
|
// loaded successfully
|
||
|
s.state = &state
|
||
|
|
||
|
log.Infof("state is restored: %+v", s.state)
|
||
|
s.Notify("position is restored => %f", s.state.HedgePosition.Float64())
|
||
|
}
|
||
|
|
||
|
s.book = types.NewStreamBook(s.Symbol)
|
||
|
s.book.BindStream(s.sourceSession.Stream)
|
||
|
|
||
|
s.sourceSession.Stream.OnTradeUpdate(s.handleTradeUpdate)
|
||
|
s.makerSession.Stream.OnTradeUpdate(s.handleTradeUpdate)
|
||
|
|
||
|
s.activeMakerOrders = bbgo.NewLocalActiveOrderBook()
|
||
|
s.activeMakerOrders.BindStream(s.makerSession.Stream)
|
||
|
|
||
|
s.orderStore = bbgo.NewOrderStore(s.Symbol)
|
||
|
s.orderStore.BindStream(s.sourceSession.Stream)
|
||
|
s.orderStore.BindStream(s.makerSession.Stream)
|
||
|
|
||
|
s.stopC = make(chan struct{})
|
||
|
|
||
|
go func() {
|
||
|
posTicker := time.NewTicker(s.HedgeInterval.Duration())
|
||
|
defer posTicker.Stop()
|
||
|
|
||
|
ticker := time.NewTicker(s.UpdateInterval.Duration())
|
||
|
defer ticker.Stop()
|
||
|
for {
|
||
|
select {
|
||
|
|
||
|
case <-s.stopC:
|
||
|
return
|
||
|
|
||
|
case <-ctx.Done():
|
||
|
return
|
||
|
|
||
|
case <-ticker.C:
|
||
|
s.updateQuote(ctx)
|
||
|
|
||
|
case <-posTicker.C:
|
||
|
position := s.state.HedgePosition.AtomicLoad()
|
||
|
abspos := math.Abs(position.Float64())
|
||
|
if !s.DisableHedge && abspos > s.sourceMarket.MinQuantity {
|
||
|
log.Infof("found position: %f", position.Float64())
|
||
|
s.Hedge(ctx, -position)
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
}()
|
||
|
|
||
|
s.Graceful.OnShutdown(func(ctx context.Context, wg *sync.WaitGroup) {
|
||
|
defer wg.Done()
|
||
|
|
||
|
close(s.stopC)
|
||
|
|
||
|
if err := s.Persistence.Save(&s.state, stateKey); err != nil {
|
||
|
log.WithError(err).Errorf("can not save state: %+v", s.state)
|
||
|
} else {
|
||
|
log.Infof("state is saved => %+v", s.state)
|
||
|
s.Notify("hedge position %f is saved", s.state.HedgePosition.Float64())
|
||
|
}
|
||
|
|
||
|
if err := s.makerSession.Exchange.CancelOrders(ctx, s.activeMakerOrders.Orders()...); err != nil {
|
||
|
log.WithError(err).Errorf("can not cancel orders")
|
||
|
}
|
||
|
})
|
||
|
|
||
|
return nil
|
||
|
}
|
||
|
|
||
|
func generateGroupID(s string) int64 {
|
||
|
h := fnv.New32a()
|
||
|
h.Write([]byte(s))
|
||
|
return int64(h.Sum32())
|
||
|
}
|