mirror of
https://github.com/c9s/bbgo.git
synced 2024-11-10 09:11:55 +00:00
Merge pull request #1205 from c9s/fix/xalign-duration
FIX: [xalign] add balance fault tolerance range
This commit is contained in:
commit
bc8adb6d33
|
@ -21,6 +21,12 @@ func init() {
|
||||||
bbgo.RegisterStrategy(ID, &Strategy{})
|
bbgo.RegisterStrategy(ID, &Strategy{})
|
||||||
}
|
}
|
||||||
|
|
||||||
|
type TimeBalance struct {
|
||||||
|
types.Balance
|
||||||
|
|
||||||
|
Time time.Time
|
||||||
|
}
|
||||||
|
|
||||||
type QuoteCurrencyPreference struct {
|
type QuoteCurrencyPreference struct {
|
||||||
Buy []string `json:"buy"`
|
Buy []string `json:"buy"`
|
||||||
Sell []string `json:"sell"`
|
Sell []string `json:"sell"`
|
||||||
|
@ -34,6 +40,10 @@ type Strategy struct {
|
||||||
ExpectedBalances map[string]fixedpoint.Value `json:"expectedBalances"`
|
ExpectedBalances map[string]fixedpoint.Value `json:"expectedBalances"`
|
||||||
UseTakerOrder bool `json:"useTakerOrder"`
|
UseTakerOrder bool `json:"useTakerOrder"`
|
||||||
DryRun bool `json:"dryRun"`
|
DryRun bool `json:"dryRun"`
|
||||||
|
BalanceToleranceRange fixedpoint.Value `json:"balanceToleranceRange"`
|
||||||
|
Duration types.Duration `json:"for"`
|
||||||
|
|
||||||
|
faultBalanceRecords map[string][]TimeBalance
|
||||||
|
|
||||||
sessions map[string]*bbgo.ExchangeSession
|
sessions map[string]*bbgo.ExchangeSession
|
||||||
orderBooks map[string]*bbgo.ActiveOrderBook
|
orderBooks map[string]*bbgo.ActiveOrderBook
|
||||||
|
@ -63,6 +73,11 @@ func (s *Strategy) CrossSubscribe(sessions map[string]*bbgo.ExchangeSession) {
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (s *Strategy) Defaults() error {
|
||||||
|
s.BalanceToleranceRange = fixedpoint.NewFromFloat(0.01)
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
func (s *Strategy) Validate() error {
|
func (s *Strategy) Validate() error {
|
||||||
if s.PreferredQuoteCurrencies == nil {
|
if s.PreferredQuoteCurrencies == nil {
|
||||||
return errors.New("quoteCurrencies is not defined")
|
return errors.New("quoteCurrencies is not defined")
|
||||||
|
@ -221,6 +236,7 @@ func (s *Strategy) CrossRun(ctx context.Context, _ bbgo.OrderExecutionRouter, se
|
||||||
instanceID := s.InstanceID()
|
instanceID := s.InstanceID()
|
||||||
_ = instanceID
|
_ = instanceID
|
||||||
|
|
||||||
|
s.faultBalanceRecords = make(map[string][]TimeBalance)
|
||||||
s.sessions = make(map[string]*bbgo.ExchangeSession)
|
s.sessions = make(map[string]*bbgo.ExchangeSession)
|
||||||
s.orderBooks = make(map[string]*bbgo.ActiveOrderBook)
|
s.orderBooks = make(map[string]*bbgo.ActiveOrderBook)
|
||||||
|
|
||||||
|
@ -271,9 +287,26 @@ func (s *Strategy) CrossRun(ctx context.Context, _ bbgo.OrderExecutionRouter, se
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (s *Strategy) recordBalance(totalBalances types.BalanceMap) {
|
||||||
|
now := time.Now()
|
||||||
|
for currency, expectedBalance := range s.ExpectedBalances {
|
||||||
|
q := s.calculateRefillQuantity(totalBalances, currency, expectedBalance)
|
||||||
|
rf := q.Div(expectedBalance).Abs().Float64()
|
||||||
|
tr := s.BalanceToleranceRange.Float64()
|
||||||
|
if rf > tr {
|
||||||
|
balance := totalBalances[currency]
|
||||||
|
s.faultBalanceRecords[currency] = append(s.faultBalanceRecords[currency], TimeBalance{
|
||||||
|
Time: now,
|
||||||
|
Balance: balance,
|
||||||
|
})
|
||||||
|
} else {
|
||||||
|
// reset counter
|
||||||
|
s.faultBalanceRecords[currency] = nil
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
func (s *Strategy) align(ctx context.Context, sessions map[string]*bbgo.ExchangeSession) {
|
func (s *Strategy) align(ctx context.Context, sessions map[string]*bbgo.ExchangeSession) {
|
||||||
totalBalances, sessionBalances := s.aggregateBalances(ctx, sessions)
|
|
||||||
_ = sessionBalances
|
|
||||||
|
|
||||||
for sessionName, session := range sessions {
|
for sessionName, session := range sessions {
|
||||||
ob, ok := s.orderBooks[sessionName]
|
ob, ok := s.orderBooks[sessionName]
|
||||||
|
@ -288,9 +321,24 @@ func (s *Strategy) align(ctx context.Context, sessions map[string]*bbgo.Exchange
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
totalBalances, sessionBalances := s.aggregateBalances(ctx, sessions)
|
||||||
|
_ = sessionBalances
|
||||||
|
|
||||||
|
s.recordBalance(totalBalances)
|
||||||
|
|
||||||
for currency, expectedBalance := range s.ExpectedBalances {
|
for currency, expectedBalance := range s.ExpectedBalances {
|
||||||
q := s.calculateRefillQuantity(totalBalances, currency, expectedBalance)
|
q := s.calculateRefillQuantity(totalBalances, currency, expectedBalance)
|
||||||
|
|
||||||
|
if s.Duration > 0 {
|
||||||
|
log.Infof("checking fault balance records...")
|
||||||
|
if faultBalance, ok := s.faultBalanceRecords[currency]; ok && len(faultBalance) > 0 {
|
||||||
|
if time.Since(faultBalance[0].Time) < s.Duration.Duration() {
|
||||||
|
log.Infof("%s fault record since: %s < persistence period %s", currency, faultBalance[0].Time, s.Duration.Duration())
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
selectedSession, submitOrder := s.selectSessionForCurrency(ctx, sessions, currency, q)
|
selectedSession, submitOrder := s.selectSessionForCurrency(ctx, sessions, currency, q)
|
||||||
if selectedSession != nil && submitOrder != nil {
|
if selectedSession != nil && submitOrder != nil {
|
||||||
log.Infof("placing order on %s: %+v", selectedSession.Name, submitOrder)
|
log.Infof("placing order on %s: %+v", selectedSession.Name, submitOrder)
|
||||||
|
|
Loading…
Reference in New Issue
Block a user