2023-07-21 06:55:27 +00:00
|
|
|
package bybit
|
|
|
|
|
|
|
|
import (
|
2023-07-24 09:01:34 +00:00
|
|
|
"context"
|
2023-07-24 15:27:43 +00:00
|
|
|
"fmt"
|
|
|
|
"time"
|
2023-07-24 09:01:34 +00:00
|
|
|
|
2023-07-21 06:55:27 +00:00
|
|
|
"github.com/sirupsen/logrus"
|
2023-07-24 15:27:43 +00:00
|
|
|
"golang.org/x/time/rate"
|
2023-07-21 06:55:27 +00:00
|
|
|
|
|
|
|
"github.com/c9s/bbgo/pkg/exchange/bybit/bybitapi"
|
|
|
|
"github.com/c9s/bbgo/pkg/types"
|
|
|
|
)
|
|
|
|
|
2023-07-24 15:27:43 +00:00
|
|
|
// https://bybit-exchange.github.io/docs/zh-TW/v5/rate-limit
|
|
|
|
// sharedRateLimiter indicates that the API belongs to the public API.
|
|
|
|
//
|
|
|
|
// The default order limiter apply 2 requests per second and a 2 initial bucket
|
|
|
|
// this includes QueryMarkets, QueryTicker
|
2023-07-25 13:30:49 +00:00
|
|
|
var (
|
|
|
|
sharedRateLimiter = rate.NewLimiter(rate.Every(time.Second/2), 2)
|
|
|
|
tradeRateLimiter = rate.NewLimiter(rate.Every(time.Second/5), 5)
|
2023-07-24 15:27:43 +00:00
|
|
|
|
2023-07-25 13:30:49 +00:00
|
|
|
log = logrus.WithFields(logrus.Fields{
|
|
|
|
"exchange": "bybit",
|
|
|
|
})
|
|
|
|
)
|
2023-07-21 06:55:27 +00:00
|
|
|
|
|
|
|
type Exchange struct {
|
|
|
|
key, secret string
|
|
|
|
client *bybitapi.RestClient
|
|
|
|
}
|
|
|
|
|
|
|
|
func New(key, secret string) (*Exchange, error) {
|
|
|
|
client, err := bybitapi.NewClient()
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
if len(key) > 0 && len(secret) > 0 {
|
|
|
|
client.Auth(key, secret)
|
|
|
|
}
|
|
|
|
|
|
|
|
return &Exchange{
|
|
|
|
key: key,
|
|
|
|
// pragma: allowlist nextline secret
|
|
|
|
secret: secret,
|
|
|
|
client: client,
|
|
|
|
}, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (e *Exchange) Name() types.ExchangeName {
|
|
|
|
return types.ExchangeBybit
|
|
|
|
}
|
|
|
|
|
|
|
|
// PlatformFeeCurrency returns empty string. The platform does not support "PlatformFeeCurrency" but instead charges
|
|
|
|
// fees using the native token.
|
|
|
|
func (e *Exchange) PlatformFeeCurrency() string {
|
|
|
|
return ""
|
|
|
|
}
|
2023-07-24 09:01:34 +00:00
|
|
|
|
|
|
|
func (e *Exchange) QueryMarkets(ctx context.Context) (types.MarketMap, error) {
|
2023-07-24 15:27:43 +00:00
|
|
|
if err := sharedRateLimiter.Wait(ctx); err != nil {
|
|
|
|
log.WithError(err).Errorf("markets rate limiter wait error")
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
2023-07-24 09:01:34 +00:00
|
|
|
instruments, err := e.client.NewGetInstrumentsInfoRequest().Do(ctx)
|
|
|
|
if err != nil {
|
2023-07-24 15:27:43 +00:00
|
|
|
log.Warnf("failed to query instruments, err: %v", err)
|
2023-07-24 09:01:34 +00:00
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
marketMap := types.MarketMap{}
|
|
|
|
for _, s := range instruments.List {
|
|
|
|
marketMap.Add(toGlobalMarket(s))
|
|
|
|
}
|
|
|
|
|
|
|
|
return marketMap, nil
|
|
|
|
}
|
2023-07-24 15:27:43 +00:00
|
|
|
|
|
|
|
func (e *Exchange) QueryTicker(ctx context.Context, symbol string) (*types.Ticker, error) {
|
|
|
|
if err := sharedRateLimiter.Wait(ctx); err != nil {
|
|
|
|
log.WithError(err).Errorf("ticker rate limiter wait error")
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
s, err := e.client.NewGetTickersRequest().Symbol(symbol).DoWithResponseTime(ctx)
|
|
|
|
if err != nil {
|
|
|
|
log.Warnf("failed to get tickers, symbol: %s, err: %v", symbol, err)
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
if len(s.List) != 1 {
|
|
|
|
log.Warnf("unexpected ticker length, exp: 1, got: %d", len(s.List))
|
|
|
|
return nil, fmt.Errorf("unexpected ticker lenght, exp:1, got:%d", len(s.List))
|
|
|
|
}
|
|
|
|
|
|
|
|
ticker := toGlobalTicker(s.List[0], s.ClosedTime.Time())
|
|
|
|
return &ticker, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (e *Exchange) QueryTickers(ctx context.Context, symbols ...string) (map[string]types.Ticker, error) {
|
|
|
|
tickers := map[string]types.Ticker{}
|
|
|
|
if len(symbols) > 0 {
|
|
|
|
for _, s := range symbols {
|
|
|
|
t, err := e.QueryTicker(ctx, s)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
tickers[s] = *t
|
|
|
|
}
|
|
|
|
|
|
|
|
return tickers, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
if err := sharedRateLimiter.Wait(ctx); err != nil {
|
|
|
|
log.WithError(err).Errorf("ticker rate limiter wait error")
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
allTickers, err := e.client.NewGetTickersRequest().DoWithResponseTime(ctx)
|
|
|
|
if err != nil {
|
|
|
|
log.Warnf("failed to get tickers, err: %v", err)
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
for _, s := range allTickers.List {
|
|
|
|
tickers[s.Symbol] = toGlobalTicker(s, allTickers.ClosedTime.Time())
|
|
|
|
}
|
|
|
|
|
|
|
|
return tickers, nil
|
|
|
|
}
|
2023-07-25 13:30:49 +00:00
|
|
|
|
|
|
|
func (e *Exchange) QueryOpenOrders(ctx context.Context, symbol string) (orders []types.Order, err error) {
|
|
|
|
cursor := ""
|
|
|
|
for {
|
|
|
|
req := e.client.NewGetOpenOrderRequest().Symbol(symbol)
|
|
|
|
if len(cursor) != 0 {
|
|
|
|
// the default limit is 20.
|
|
|
|
req = req.Cursor(cursor)
|
|
|
|
}
|
|
|
|
|
|
|
|
if err = tradeRateLimiter.Wait(ctx); err != nil {
|
|
|
|
log.WithError(err).Errorf("trade rate limiter wait error")
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
res, err := req.Do(ctx)
|
|
|
|
if err != nil {
|
|
|
|
log.Warnf("failed to get open order, cursor: %s, err: %v", cursor, err)
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
for _, order := range res.List {
|
|
|
|
order, err := toGlobalOrder(order)
|
|
|
|
if err != nil {
|
|
|
|
log.Warnf("failed to convert order, err: %v", err)
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
orders = append(orders, *order)
|
|
|
|
}
|
|
|
|
|
|
|
|
if len(res.NextPageCursor) == 0 {
|
|
|
|
break
|
|
|
|
}
|
|
|
|
cursor = res.NextPageCursor
|
|
|
|
}
|
|
|
|
|
|
|
|
return orders, nil
|
|
|
|
}
|