Files
gocryptotrader/exchanges/bybit/bybit.go
Ryan O'Hara-Reid 83cfefa45c kline/exchanges: automatic creation of unsupported candle intervals (#1091)
* kline: Add builder and testing

* Ideas

* kline: deploy builder functionality across GCT

* exchanges: implement across gct

* exchanges: Add tests and fix implementations before kline package testing and veri.

* kline: Add tests and start to fix ConvertToNewInterval

* kline: fix ConvertToNewInterval add tests

* kline: complete overarching tests now on to exchanges

* kline: finish exchange tests and implement limits

* exchanges: more fixes

* linter: fix

* engine: fix tests

* kraken: fix recent trades and other fixes

* zb: fix tests

* bithumb: fix empty insertion

* kline: refactor/optimize CreateKline function

* kline: remove the mooos!

* kline: prealloc CalculateCandleDateRanges

* linter: fix

* exchanges: prealloc extended

* fix whoopsie

* reverse fix because this is a whoopsie

* okx: fix risidual issues

* linter: fix

* kline: initial nits from @gloriouscode

* kline: rename builder -> request and cascade change

* linter: fix + test

* kline: update forced alignment on start and end times when CreateKlineRequest is called.

* nits: more more more

* NITS: Addressed

* tests: fix race issue

* Update exchanges/kline/request.go

Co-authored-by: Scott <gloriousCode@users.noreply.github.com>

* kline: add method AddPadding() to automatically fill in holes in kline.Request functionality and reject if missing data when converting

* kline: Add params start and end to addPadding() to insert blanks in between block

* kline: remove test comment code as it's not needed anymore

* kline: fix lint and test

* kline: sort slice without extra bool check every iteration

* okx: fix issues with timeing and candles and such from niterinos & address typo

* Update exchanges/kline/kline.go

Co-authored-by: Scott <gloriousCode@users.noreply.github.com>

* glorious: niterinos

* Update exchanges/poloniex/poloniex_wrapper.go

Co-authored-by: Scott <gloriousCode@users.noreply.github.com>

* glorious: nits now onto conflicts YAYA!!!

* Update exchanges/exchange_test.go

Co-authored-by: Scott <gloriousCode@users.noreply.github.com>

* glorious: nits again

* thrasher: nitters

* thrasher: niterinos - adds partial flag for incomplete recent candles and fetching.

* kline: rm fmtizzle packageizzle

* glorious: nitters

* glorious: more niterinos

* fix last niterinos

Co-authored-by: Ryan O'Hara-Reid <ryan.oharareid@thrasher.io>
Co-authored-by: Scott <gloriousCode@users.noreply.github.com>
2023-01-17 16:22:33 +11:00

988 lines
28 KiB
Go

package bybit
import (
"bytes"
"context"
"encoding/json"
"errors"
"fmt"
"net/http"
"net/url"
"sort"
"strconv"
"strings"
"time"
"github.com/thrasher-corp/gocryptotrader/common"
"github.com/thrasher-corp/gocryptotrader/common/crypto"
exchange "github.com/thrasher-corp/gocryptotrader/exchanges"
"github.com/thrasher-corp/gocryptotrader/exchanges/order"
"github.com/thrasher-corp/gocryptotrader/exchanges/orderbook"
"github.com/thrasher-corp/gocryptotrader/exchanges/request"
)
// Bybit is the overarching type across this package
type Bybit struct {
exchange.Base
}
const (
bybitAPIURL = "https://api.bybit.com"
defaultRecvWindow = "5000" // 5000 milli second
sideBuy = "Buy"
sideSell = "Sell"
// Public endpoints
bybitSpotGetSymbols = "/spot/v1/symbols"
bybitOrderBook = "/spot/quote/v1/depth"
bybitMergedOrderBook = "/spot/quote/v1/depth/merged"
bybitRecentTrades = "/spot/quote/v1/trades"
bybitCandlestickChart = "/spot/quote/v1/kline"
bybit24HrsChange = "/spot/quote/v1/ticker/24hr"
bybitLastTradedPrice = "/spot/quote/v1/ticker/price"
bybitBestBidAskPrice = "/spot/quote/v1/ticker/book_ticker"
// Authenticated endpoints
bybitSpotOrder = "/spot/v1/order" // create, query, cancel
bybitFastCancelSpotOrder = "/spot/v1/order/fast"
bybitBatchCancelSpotOrder = "/spot/order/batch-cancel"
bybitFastBatchCancelSpotOrder = "/spot/order/batch-fast-cancel"
bybitOpenOrder = "/spot/v1/open-orders"
bybitPastOrder = "/spot/v1/history-orders"
bybitTradeHistory = "/spot/v1/myTrades"
bybitWalletBalance = "/spot/v1/account"
bybitServerTime = "/spot/v1/time"
// Account asset endpoint
bybitGetDepositAddress = "/asset/v1/private/deposit/address"
bybitWithdrawFund = "/asset/v1/private/withdraw"
)
// GetAllSpotPairs gets all pairs on the exchange
func (by *Bybit) GetAllSpotPairs(ctx context.Context) ([]PairData, error) {
resp := struct {
Data []PairData `json:"result"`
Error
}{}
return resp.Data, by.SendHTTPRequest(ctx, exchange.RestSpot, bybitSpotGetSymbols, publicSpotRate, &resp)
}
func processOB(ob [][2]string) ([]orderbook.Item, error) {
o := make([]orderbook.Item, len(ob))
for x := range ob {
var price, amount float64
amount, err := strconv.ParseFloat(ob[x][1], 64)
if err != nil {
return nil, err
}
price, err = strconv.ParseFloat(ob[x][0], 64)
if err != nil {
return nil, err
}
o[x] = orderbook.Item{
Price: price,
Amount: amount,
}
}
return o, nil
}
func constructOrderbook(o *orderbookResponse) (*Orderbook, error) {
var (
s Orderbook
err error
)
s.Bids, err = processOB(o.Data.Bids)
if err != nil {
return nil, err
}
s.Asks, err = processOB(o.Data.Asks)
if err != nil {
return nil, err
}
s.Time = o.Data.Time.Time()
return &s, err
}
// GetOrderBook gets orderbook for a given market with a given depth (default depth 100)
func (by *Bybit) GetOrderBook(ctx context.Context, symbol string, depth int64) (*Orderbook, error) {
var o orderbookResponse
strDepth := "100" // default depth
if depth > 0 && depth < 100 {
strDepth = strconv.FormatInt(depth, 10)
}
params := url.Values{}
params.Set("symbol", symbol)
params.Set("limit", strDepth)
path := common.EncodeURLValues(bybitOrderBook, params)
err := by.SendHTTPRequest(ctx, exchange.RestSpot, path, publicSpotRate, &o)
if err != nil {
return nil, err
}
return constructOrderbook(&o)
}
// GetMergedOrderBook gets orderbook for a given market with a given depth (default depth 100)
func (by *Bybit) GetMergedOrderBook(ctx context.Context, symbol string, scale, depth int64) (*Orderbook, error) {
var o orderbookResponse
params := url.Values{}
if scale > 0 {
params.Set("scale", strconv.FormatInt(scale, 10))
}
strDepth := "100" // default depth
if depth > 0 && depth <= 200 {
strDepth = strconv.FormatInt(depth, 10)
}
params.Set("symbol", symbol)
params.Set("limit", strDepth)
path := common.EncodeURLValues(bybitMergedOrderBook, params)
err := by.SendHTTPRequest(ctx, exchange.RestSpot, path, publicSpotRate, &o)
if err != nil {
return nil, err
}
return constructOrderbook(&o)
}
// GetTrades gets recent trades from the exchange
func (by *Bybit) GetTrades(ctx context.Context, symbol string, limit int64) ([]TradeItem, error) {
resp := struct {
Data []struct {
Price float64 `json:"price,string"`
Time bybitTimeMilliSec `json:"time"`
Quantity float64 `json:"qty,string"`
IsBuyerMaker bool `json:"isBuyerMaker"`
} `json:"result"`
Error
}{}
params := url.Values{}
params.Set("symbol", symbol)
strLimit := "60" // default limit
if limit > 0 && limit < 60 {
strLimit = strconv.FormatInt(limit, 10)
}
params.Set("limit", strLimit)
path := common.EncodeURLValues(bybitRecentTrades, params)
err := by.SendHTTPRequest(ctx, exchange.RestSpot, path, publicSpotRate, &resp)
if err != nil {
return nil, err
}
trades := make([]TradeItem, len(resp.Data))
for x := range resp.Data {
var tradeSide string
if resp.Data[x].IsBuyerMaker {
tradeSide = order.Buy.String()
} else {
tradeSide = order.Sell.String()
}
trades[x] = TradeItem{
CurrencyPair: symbol,
Price: resp.Data[x].Price,
Side: tradeSide,
Volume: resp.Data[x].Quantity,
Time: resp.Data[x].Time.Time(),
}
}
return trades, nil
}
// GetKlines data returns the kline data for a specific symbol. Limitation: It only returns latest 3500 candles irrespective of interval passed
func (by *Bybit) GetKlines(ctx context.Context, symbol, period string, limit int64, start, end time.Time) ([]KlineItem, error) {
resp := struct {
Data [][]interface{} `json:"result"`
Error
}{}
v := url.Values{}
v.Add("symbol", symbol)
v.Add("interval", period)
if !start.IsZero() {
v.Add("startTime", strconv.FormatInt(start.UnixMilli(), 10))
}
if !end.IsZero() {
v.Add("endTime", strconv.FormatInt(end.UnixMilli(), 10))
}
if limit <= 0 || limit > 1000 {
limit = 1000
}
v.Add("limit", strconv.FormatInt(limit, 10))
path := common.EncodeURLValues(bybitCandlestickChart, v)
if err := by.SendHTTPRequest(ctx, exchange.RestSpot, path, publicSpotRate, &resp); err != nil {
return nil, err
}
klines := make([]KlineItem, len(resp.Data))
for x := range resp.Data {
if len(resp.Data[x]) != 11 {
return nil, fmt.Errorf("%v GetKlines: invalid response, array length not as expected, check api docs for updates", by.Name)
}
var err error
startTime, ok := resp.Data[x][0].(float64)
if !ok {
return nil, fmt.Errorf("%v GetKlines: %w for StartTime", by.Name, errTypeAssert)
}
klines[x].StartTime = time.UnixMilli(int64(startTime))
open, ok := resp.Data[x][1].(string)
if !ok {
return nil, fmt.Errorf("%v GetKlines: %w for Open", by.Name, errTypeAssert)
}
klines[x].Open, err = strconv.ParseFloat(open, 64)
if err != nil {
return nil, fmt.Errorf("%v GetKlines: %w for Open", by.Name, errStrParsing)
}
high, ok := resp.Data[x][2].(string)
if !ok {
return nil, fmt.Errorf("%v GetKlines: %w for High", by.Name, errTypeAssert)
}
klines[x].High, err = strconv.ParseFloat(high, 64)
if err != nil {
return nil, fmt.Errorf("%v GetKlines: %w for High", by.Name, errStrParsing)
}
low, ok := resp.Data[x][3].(string)
if !ok {
return nil, fmt.Errorf("%v GetKlines: %w for Low", by.Name, errTypeAssert)
}
klines[x].Low, err = strconv.ParseFloat(low, 64)
if err != nil {
return nil, fmt.Errorf("%v GetKlines: %w for Low", by.Name, errStrParsing)
}
c, ok := resp.Data[x][4].(string)
if !ok {
return nil, fmt.Errorf("%v GetKlines: %w for Close", by.Name, errTypeAssert)
}
klines[x].Close, err = strconv.ParseFloat(c, 64)
if err != nil {
return nil, fmt.Errorf("%v GetKlines: %w for Close", by.Name, errStrParsing)
}
volume, ok := resp.Data[x][5].(string)
if !ok {
return nil, fmt.Errorf("%v GetKlines: %w for Volume", by.Name, errTypeAssert)
}
klines[x].Volume, err = strconv.ParseFloat(volume, 64)
if err != nil {
return nil, fmt.Errorf("%v GetKlines: %w for Volume", by.Name, errStrParsing)
}
endTime, ok := resp.Data[x][6].(float64)
if !ok {
return nil, fmt.Errorf("%v GetKlines: %w for EndTime", by.Name, errTypeAssert)
}
klines[x].EndTime = time.UnixMilli(int64(endTime))
quoteAssetVolume, ok := resp.Data[x][7].(string)
if !ok {
return nil, fmt.Errorf("%v GetKlines: %w for QuoteAssetVolume", by.Name, errTypeAssert)
}
klines[x].QuoteAssetVolume, err = strconv.ParseFloat(quoteAssetVolume, 64)
if err != nil {
return nil, fmt.Errorf("%v GetKlines: %w for QuoteAssetVolume", by.Name, errStrParsing)
}
tradesCount, ok := resp.Data[x][8].(float64)
if !ok {
return nil, fmt.Errorf("%v GetKlines: %w for TradesCount", by.Name, errTypeAssert)
}
klines[x].TradesCount = int64(tradesCount)
klines[x].TakerBaseVolume, ok = resp.Data[x][9].(float64)
if !ok {
return nil, fmt.Errorf("%v GetKlines: %w for TakerBaseVolume", by.Name, errTypeAssert)
}
klines[x].TakerQuoteVolume, ok = resp.Data[x][10].(float64)
if !ok {
return nil, fmt.Errorf("%v GetKlines: %w for TakerQuoteVolume", by.Name, errTypeAssert)
}
}
return klines, nil
}
// Get24HrsChange returns price change statistics for the last 24 hours
// If symbol not passed then it will return price change statistics for all pairs
func (by *Bybit) Get24HrsChange(ctx context.Context, symbol string) ([]PriceChangeStats, error) {
type priceChangeStats struct {
Time bybitTimeMilliSec `json:"time"`
Symbol string `json:"symbol"`
BestBidPrice float64 `json:"bestBidPrice,string"`
BestAskPrice float64 `json:"bestAskPrice,string"`
LastPrice float64 `json:"lastPrice,string"`
OpenPrice float64 `json:"openPrice,string"`
HighPrice float64 `json:"highPrice,string"`
LowPrice float64 `json:"lowPrice,string"`
Volume float64 `json:"volume,string"`
QuoteVolume float64 `json:"quoteVolume,string"`
}
var stats []PriceChangeStats
if symbol != "" {
resp := struct {
Data priceChangeStats `json:"result"`
Error
}{}
params := url.Values{}
params.Set("symbol", symbol)
path := common.EncodeURLValues(bybit24HrsChange, params)
err := by.SendHTTPRequest(ctx, exchange.RestSpot, path, publicSpotRate, &resp)
if err != nil {
return nil, err
}
stats = append(stats, PriceChangeStats{
resp.Data.Time.Time(),
resp.Data.Symbol,
resp.Data.BestAskPrice,
resp.Data.BestAskPrice,
resp.Data.LastPrice,
resp.Data.OpenPrice,
resp.Data.HighPrice,
resp.Data.LowPrice,
resp.Data.Volume,
resp.Data.QuoteVolume,
})
} else {
resp := struct {
Data []priceChangeStats `json:"result"`
Error
}{}
err := by.SendHTTPRequest(ctx, exchange.RestSpot, bybit24HrsChange, publicSpotRate, &resp)
if err != nil {
return nil, err
}
for x := range resp.Data {
stats = append(stats, PriceChangeStats{
resp.Data[x].Time.Time(),
resp.Data[x].Symbol,
resp.Data[x].BestAskPrice,
resp.Data[x].BestAskPrice,
resp.Data[x].LastPrice,
resp.Data[x].OpenPrice,
resp.Data[x].HighPrice,
resp.Data[x].LowPrice,
resp.Data[x].Volume,
resp.Data[x].QuoteVolume,
})
}
}
return stats, nil
}
// GetLastTradedPrice returns last trading price
// If symbol not passed then it will return last trading price for all pairs
func (by *Bybit) GetLastTradedPrice(ctx context.Context, symbol string) ([]LastTradePrice, error) {
var lastTradePrices []LastTradePrice
if symbol != "" {
resp := struct {
Data LastTradePrice `json:"result"`
Error
}{}
params := url.Values{}
params.Set("symbol", symbol)
path := common.EncodeURLValues(bybitLastTradedPrice, params)
err := by.SendHTTPRequest(ctx, exchange.RestSpot, path, publicSpotRate, &resp)
if err != nil {
return nil, err
}
lastTradePrices = append(lastTradePrices, LastTradePrice{
resp.Data.Symbol,
resp.Data.Price,
})
} else {
resp := struct {
Data []LastTradePrice `json:"result"`
Error
}{}
err := by.SendHTTPRequest(ctx, exchange.RestSpot, bybitLastTradedPrice, publicSpotRate, &resp)
if err != nil {
return nil, err
}
for x := range resp.Data {
lastTradePrices = append(lastTradePrices, LastTradePrice{
resp.Data[x].Symbol,
resp.Data[x].Price,
})
}
}
return lastTradePrices, nil
}
// GetBestBidAskPrice returns best BID and ASK price
// If symbol not passed then it will return best BID and ASK price for all pairs
func (by *Bybit) GetBestBidAskPrice(ctx context.Context, symbol string) ([]TickerData, error) {
type bestTicker struct {
Symbol string `json:"symbol"`
BidPrice float64 `json:"bidPrice,string"`
BidQuantity float64 `json:"bidQty,string"`
AskPrice float64 `json:"askPrice,string"`
AskQuantity float64 `json:"askQty,string"`
Time bybitTimeMilliSec `json:"time"`
}
var tickers []TickerData
if symbol != "" {
resp := struct {
Data bestTicker `json:"result"`
Error
}{}
params := url.Values{}
params.Set("symbol", symbol)
path := common.EncodeURLValues(bybitBestBidAskPrice, params)
err := by.SendHTTPRequest(ctx, exchange.RestSpot, path, publicSpotRate, &resp)
if err != nil {
return nil, err
}
tickers = append(tickers, TickerData{
resp.Data.Symbol,
resp.Data.BidPrice,
resp.Data.BidQuantity,
resp.Data.AskPrice,
resp.Data.AskQuantity,
resp.Data.Time.Time(),
})
} else {
resp := struct {
Data []bestTicker `json:"result"`
Error
}{}
err := by.SendHTTPRequest(ctx, exchange.RestSpot, bybitBestBidAskPrice, publicSpotRate, &resp)
if err != nil {
return nil, err
}
for x := range resp.Data {
tickers = append(tickers, TickerData{
resp.Data[x].Symbol,
resp.Data[x].BidPrice,
resp.Data[x].BidQuantity,
resp.Data[x].AskPrice,
resp.Data[x].AskQuantity,
resp.Data[x].Time.Time(),
})
}
}
return tickers, nil
}
// CreatePostOrder create and post order
func (by *Bybit) CreatePostOrder(ctx context.Context, o *PlaceOrderRequest) (*PlaceOrderResponse, error) {
if o == nil {
return nil, errInvalidOrderRequest
}
params := url.Values{}
params.Set("symbol", o.Symbol)
params.Set("qty", strconv.FormatFloat(o.Quantity, 'f', -1, 64))
params.Set("side", o.Side)
params.Set("type", o.TradeType)
if o.TimeInForce != "" {
params.Set("timeInForce", o.TimeInForce)
}
if (o.TradeType == BybitRequestParamsOrderLimit || o.TradeType == BybitRequestParamsOrderLimitMaker) && o.Price == 0 {
return nil, errMissingPrice
}
if o.Price != 0 {
params.Set("price", strconv.FormatFloat(o.Price, 'f', -1, 64))
}
if o.OrderLinkID != "" {
params.Set("orderLinkId", o.OrderLinkID)
}
resp := struct {
Data PlaceOrderResponse `json:"result"`
Error
}{}
return &resp.Data, by.SendAuthHTTPRequest(ctx, exchange.RestSpot, http.MethodPost, bybitSpotOrder, params, nil, &resp, privateSpotRate)
}
// QueryOrder returns order data based upon orderID or orderLinkID
func (by *Bybit) QueryOrder(ctx context.Context, orderID, orderLinkID string) (*QueryOrderResponse, error) {
if orderID == "" && orderLinkID == "" {
return nil, errOrderOrOrderLinkIDMissing
}
params := url.Values{}
if orderID != "" {
params.Set("orderId", orderID)
}
if orderLinkID != "" {
params.Set("orderLinkId", orderLinkID)
}
resp := struct {
Data QueryOrderResponse `json:"result"`
Error
}{}
return &resp.Data, by.SendAuthHTTPRequest(ctx, exchange.RestSpot, http.MethodGet, bybitSpotOrder, params, nil, &resp, privateSpotRate)
}
// CancelExistingOrder cancels existing order based upon orderID or orderLinkID
func (by *Bybit) CancelExistingOrder(ctx context.Context, orderID, orderLinkID string) (*CancelOrderResponse, error) {
if orderID == "" && orderLinkID == "" {
return nil, errOrderOrOrderLinkIDMissing
}
params := url.Values{}
if orderID != "" {
params.Set("orderId", orderID)
}
if orderLinkID != "" {
params.Set("orderLinkId", orderLinkID)
}
resp := struct {
Data CancelOrderResponse `json:"result"`
Error
}{}
err := by.SendAuthHTTPRequest(ctx, exchange.RestSpot, http.MethodDelete, bybitSpotOrder, params, nil, &resp, privateSpotRate)
if err != nil {
return nil, err
}
// In case open order is cancelled, this endpoint return status as NEW whereas if we try to cancel a already cancelled order then it's status is returned as CANCELED without any error. So this check is added to prevent this obscurity.
if resp.Data.Status == "CANCELED" {
return nil, fmt.Errorf("%s order already cancelled", resp.Data.OrderID)
}
return &resp.Data, nil
}
// FastCancelExistingOrder cancels existing order based upon orderID or orderLinkID
func (by *Bybit) FastCancelExistingOrder(ctx context.Context, symbol, orderID, orderLinkID string) (bool, error) {
resp := struct {
Data struct {
IsCancelled bool `json:"isCancelled"`
} `json:"result"`
Error
}{}
if orderID == "" && orderLinkID == "" {
return resp.Data.IsCancelled, errOrderOrOrderLinkIDMissing
}
params := url.Values{}
if symbol == "" {
return resp.Data.IsCancelled, errSymbolMissing
}
params.Set("symbolId", symbol)
if orderID != "" {
params.Set("orderId", orderID)
}
if orderLinkID != "" {
params.Set("orderLinkId", orderLinkID)
}
return resp.Data.IsCancelled, by.SendAuthHTTPRequest(ctx, exchange.RestSpot, http.MethodDelete, bybitFastCancelSpotOrder, params, nil, &resp, privateSpotRate)
}
// BatchCancelOrder cancels orders in batch based upon symbol, side or orderType
func (by *Bybit) BatchCancelOrder(ctx context.Context, symbol, side, orderTypes string) (bool, error) {
params := url.Values{}
if symbol != "" {
params.Set("symbol", symbol)
}
if side != "" {
params.Set("side", side)
}
if orderTypes != "" {
params.Set("orderTypes", orderTypes)
}
resp := struct {
Result struct {
Success bool `json:"success"`
} `json:"result"`
Error
}{}
return resp.Result.Success, by.SendAuthHTTPRequest(ctx, exchange.RestSpot, http.MethodDelete, bybitBatchCancelSpotOrder, params, nil, &resp, privateSpotRate)
}
// BatchFastCancelOrder cancels orders in batch based upon symbol, side or orderType
func (by *Bybit) BatchFastCancelOrder(ctx context.Context, symbol, side, orderTypes string) (bool, error) {
params := url.Values{}
if symbol != "" {
params.Set("symbol", symbol)
}
if side != "" {
params.Set("side", side)
}
if orderTypes != "" {
params.Set("orderTypes", orderTypes)
}
resp := struct {
Result struct {
Success bool `json:"success"`
} `json:"result"`
Error
}{}
return resp.Result.Success, by.SendAuthHTTPRequest(ctx, exchange.RestSpot, http.MethodDelete, bybitFastBatchCancelSpotOrder, params, nil, &resp, privateSpotRate)
}
// BatchCancelOrderByIDs cancels orders in batch based on comma separated order id's
func (by *Bybit) BatchCancelOrderByIDs(ctx context.Context, orderIDs []string) (bool, error) {
params := url.Values{}
if len(orderIDs) == 0 {
return false, errEmptyOrderIDs
}
params.Set("orderIds", strings.Join(orderIDs, ","))
resp := struct {
Result struct {
Success bool `json:"success"`
} `json:"result"`
Error
}{}
return resp.Result.Success, by.SendAuthHTTPRequest(ctx, exchange.RestSpot, http.MethodDelete, bybitFastBatchCancelSpotOrder, params, nil, &resp, privateSpotRate)
}
// ListOpenOrders returns all open orders
func (by *Bybit) ListOpenOrders(ctx context.Context, symbol, orderID string, limit int64) ([]QueryOrderResponse, error) {
params := url.Values{}
if symbol != "" {
params.Set("symbol", symbol)
}
if orderID != "" {
params.Set("orderId", orderID)
}
if limit != 0 {
params.Set("limit", strconv.FormatInt(limit, 10))
}
resp := struct {
Data []QueryOrderResponse `json:"result"`
Error
}{}
return resp.Data, by.SendAuthHTTPRequest(ctx, exchange.RestSpot, http.MethodGet, bybitOpenOrder, params, nil, &resp, privateSpotRate)
}
// GetPastOrders returns all past orders from history
func (by *Bybit) GetPastOrders(ctx context.Context, symbol, orderID string, limit int64, startTime, endTime time.Time) ([]QueryOrderResponse, error) {
params := url.Values{}
if symbol != "" {
params.Set("symbol", symbol)
}
if orderID != "" {
params.Set("orderId", orderID)
}
if limit != 0 {
params.Set("limit", strconv.FormatInt(limit, 10))
}
if !startTime.IsZero() {
params.Set("startTime", strconv.FormatInt(startTime.UnixMilli(), 10))
}
if !endTime.IsZero() {
params.Set("endTime", strconv.FormatInt(endTime.UnixMilli(), 10))
}
resp := struct {
Data []QueryOrderResponse `json:"result"`
Error
}{}
return resp.Data, by.SendAuthHTTPRequest(ctx, exchange.RestSpot, http.MethodGet, bybitPastOrder, params, nil, &resp, privateSpotRate)
}
// GetTradeHistory returns user trades
func (by *Bybit) GetTradeHistory(ctx context.Context, limit int64, symbol, fromID, toID, orderID string, startTime, endTime time.Time) ([]HistoricalTrade, error) {
params := url.Values{}
if symbol != "" {
params.Set("symbol", symbol)
}
if limit != 0 {
params.Set("limit", strconv.FormatInt(limit, 10))
}
if fromID != "" {
params.Set("fromTicketId", fromID)
}
if toID != "" {
params.Set("toTicketId", toID)
}
if orderID != "" {
params.Set("orderId", orderID)
}
if !startTime.IsZero() {
params.Set("startTime", strconv.FormatInt(startTime.UnixMilli(), 10))
}
if !endTime.IsZero() {
params.Set("endTime", strconv.FormatInt(endTime.UnixMilli(), 10))
}
resp := struct {
Data []HistoricalTrade `json:"result"`
Error
}{}
return resp.Data, by.SendAuthHTTPRequest(ctx, exchange.RestSpot, http.MethodGet, bybitTradeHistory, params, nil, &resp, privateSpotRate)
}
// GetWalletBalance returns user wallet balance
func (by *Bybit) GetWalletBalance(ctx context.Context) ([]Balance, error) {
resp := struct {
Data struct {
Balances []Balance `json:"balances"`
} `json:"result"`
Error
}{}
return resp.Data.Balances, by.SendAuthHTTPRequest(ctx, exchange.RestSpot, http.MethodGet, bybitWalletBalance, url.Values{}, nil, &resp, privateSpotRate)
}
// GetSpotServerTime returns server time
func (by *Bybit) GetSpotServerTime(ctx context.Context) (time.Time, error) {
resp := struct {
Result struct {
ServerTime int64 `json:"serverTime"`
} `json:"result"`
Error
}{}
err := by.SendHTTPRequest(ctx, exchange.RestSpot, bybitServerTime, publicSpotRate, &resp)
return time.UnixMilli(resp.Result.ServerTime), err
}
// GetDepositAddressForCurrency returns deposit wallet address based upon the coin.
func (by *Bybit) GetDepositAddressForCurrency(ctx context.Context, coin string) (DepositWalletInfo, error) {
resp := struct {
Result DepositWalletInfo `json:"result"`
Error
}{}
params := url.Values{}
if coin == "" {
return resp.Result, errInvalidCoin
}
params.Set("coin", strings.ToUpper(coin))
return resp.Result, by.SendAuthHTTPRequest(ctx, exchange.RestSpot, http.MethodGet, bybitGetDepositAddress, params, nil, &resp, publicSpotRate)
}
// WithdrawFund creates request for fund withdrawal.
func (by *Bybit) WithdrawFund(ctx context.Context, coin, chain, address, tag, amount string) (string, error) {
resp := struct {
Data struct {
ID string `json:"id"`
} `json:"result"`
Error
}{}
params := make(map[string]interface{})
params["coin"] = coin
params["chain"] = chain
params["address"] = address
params["amount"] = amount
if tag != "" {
params["tag"] = tag
}
return resp.Data.ID, by.SendAuthHTTPRequest(ctx, exchange.RestSpot, http.MethodPost, bybitWithdrawFund, nil, params, &resp, privateSpotRate)
}
// SendHTTPRequest sends an unauthenticated request
func (by *Bybit) SendHTTPRequest(ctx context.Context, ePath exchange.URL, path string, f request.EndpointLimit, result UnmarshalTo) error {
endpointPath, err := by.API.Endpoints.GetURL(ePath)
if err != nil {
return err
}
err = by.SendPayload(ctx, f, func() (*request.Item, error) {
return &request.Item{
Method: http.MethodGet,
Path: endpointPath + path,
Result: result,
Verbose: by.Verbose,
HTTPDebugging: by.HTTPDebugging,
HTTPRecording: by.HTTPRecording}, nil
})
if err != nil {
return err
}
return result.GetError()
}
// SendAuthHTTPRequest sends an authenticated HTTP request
// If payload is non-nil then request is considered to be JSON
func (by *Bybit) SendAuthHTTPRequest(ctx context.Context, ePath exchange.URL, method, path string, params url.Values, jsonPayload map[string]interface{}, result UnmarshalTo, f request.EndpointLimit) error {
creds, err := by.GetCredentials(ctx)
if err != nil {
return err
}
if result == nil {
result = &Error{}
}
endpointPath, err := by.API.Endpoints.GetURL(ePath)
if err != nil {
return err
}
if params == nil && jsonPayload == nil {
params = url.Values{}
}
if jsonPayload != nil {
jsonPayload["recvWindow"] = defaultRecvWindow
} else if params.Get("recvWindow") == "" {
params.Set("recvWindow", defaultRecvWindow)
}
err = by.SendPayload(ctx, f, func() (*request.Item, error) {
var (
payload []byte
hmacSignedStr string
)
headers := make(map[string]string)
if jsonPayload != nil {
headers["Content-Type"] = "application/json"
jsonPayload["timestamp"] = strconv.FormatInt(time.Now().UnixMilli(), 10)
jsonPayload["api_key"] = creds.Key
hmacSignedStr, err = getJSONRequestSignature(jsonPayload, creds.Secret)
if err != nil {
return nil, err
}
jsonPayload["sign"] = hmacSignedStr
payload, err = json.Marshal(jsonPayload)
if err != nil {
return nil, err
}
} else {
params.Set("timestamp", strconv.FormatInt(time.Now().UnixMilli(), 10))
params.Set("api_key", creds.Key)
hmacSignedStr, err = getSign(params.Encode(), creds.Secret)
if err != nil {
return nil, err
}
headers["Content-Type"] = "application/x-www-form-urlencoded"
switch method {
case http.MethodPost:
params.Set("sign", hmacSignedStr)
payload = []byte(params.Encode())
default:
path = common.EncodeURLValues(path, params)
path += "&sign=" + hmacSignedStr
}
}
return &request.Item{
Method: method,
Path: endpointPath + path,
Headers: headers,
Body: bytes.NewBuffer(payload),
Result: &result,
AuthRequest: true,
Verbose: by.Verbose,
HTTPDebugging: by.HTTPDebugging,
HTTPRecording: by.HTTPRecording}, nil
})
if err != nil {
return err
}
return result.GetError()
}
// Error defines all error information for each request
type Error struct {
ReturnCode int64 `json:"ret_code"`
ReturnMsg string `json:"ret_msg"`
ExtCode string `json:"ext_code"`
ExtMsg string `json:"ext_info"`
}
// GetError checks and returns an error if it is supplied.
func (e Error) GetError() error {
if e.ReturnCode != 0 && e.ReturnMsg != "" {
return errors.New(e.ReturnMsg)
}
if e.ExtCode != "" && e.ExtMsg != "" {
return errors.New(e.ExtMsg)
}
return nil
}
func getSide(side string) order.Side {
switch side {
case sideBuy:
return order.Buy
case sideSell:
return order.Sell
default:
return order.UnknownSide
}
}
func getTradeType(tradeType string) order.Type {
switch tradeType {
case BybitRequestParamsOrderLimit:
return order.Limit
case BybitRequestParamsOrderMarket:
return order.Market
case BybitRequestParamsOrderLimitMaker:
return order.Limit
default:
return order.UnknownType
}
}
func getOrderStatus(status string) order.Status {
switch status {
case "NEW":
return order.New
case "PARTIALLY_FILLED":
return order.PartiallyFilled
case "FILLED":
return order.Filled
case "CANCELED":
return order.Cancelled
case "PENDING_CANCEL":
return order.PendingCancel
case "PENDING_NEW":
return order.Pending
case "REJECTED":
return order.Rejected
default:
return order.UnknownStatus
}
}
func getJSONRequestSignature(payload map[string]interface{}, secret string) (string, error) {
payloadArr := make([]string, len(payload))
var i int
for p := range payload {
payloadArr[i] = p
i++
}
sort.Strings(payloadArr)
var signStr string
for _, key := range payloadArr {
if value, found := payload[key]; found {
if v, ok := value.(string); ok {
signStr += key + "=" + v + "&"
}
} else {
return "", errors.New("non-string payload parameter not expected")
}
}
return getSign(signStr[:len(signStr)-1], secret)
}
func getSign(sign, secret string) (string, error) {
hmacSigned, err := crypto.GetHMAC(crypto.HashSHA256, []byte(sign), []byte(secret))
if err != nil {
return "", err
}
return crypto.HexEncodeToString(hmacSigned), nil
}