Files
gocryptotrader/exchanges/bybit/bybit_futures.go
Scott 5f2f6f884b Binance,Okx: Add Leverage, MarginType, Positions and CollateralMode support (#1220)
* init

* surprise train commit

* basic distinctions

* the terms of binance are confusing

* renames and introduction of allocatedMargin

* add new margin funcs

* pulling out wires

* implement proper getposition stuff

* bad coding day

* investigate order manager next

* a broken mess, but a progressing one

* finally completes some usdtmargined stuff

* coinMfutures eludes me

* expand to okx

* imports fix

* completes okx wrapper implementations

* cleans and polishes before rpc implementations

* rpc setup, order manager features, exch features

* more rpc, collateral and margin things

* mini test

* looking at rpc response, expansion of features

* reorganising before the storm

* changing how futures requests work

* cleanup and tests of cli usage

* remove silly client side logic

* cleanup

* collateral package, typo fix, margin err, rpc derive

* uses convert.StringToFloat ONLY ON STRUCTS FROM THIS PR

* fix binance order history bug

* niteroos

* adds new funcs to exchange standards testing

* more post merge fixes

* fix binance

* replace simepletimeformat

* fix for merge

* merge fixes

* micro fixes

* order side now required for leverage

* fix up the rest

* global -> portfolio collateral

* Update exchanges/collateral/collateral_test.go

Co-authored-by: Adrian Gallagher <adrian.gallagher@thrasher.io>

* adds fields and todos

* rm field redundancy

* lint fix oopsie daisy

* fixes panic, expands error and cli explanations (sorry shaz)

* ensures casing is appropriate for underlying

* Adds a shiny TODO

---------

Co-authored-by: Adrian Gallagher <adrian.gallagher@thrasher.io>
2023-09-26 16:16:31 +10:00

758 lines
25 KiB
Go

package bybit
import (
"context"
"net/http"
"net/url"
"strconv"
"time"
"github.com/thrasher-corp/gocryptotrader/currency"
exchange "github.com/thrasher-corp/gocryptotrader/exchanges"
"github.com/thrasher-corp/gocryptotrader/exchanges/asset"
)
const (
// auth endpoint
futuresCreateOrder = "/futures/private/order/create"
futuresGetActiveOrders = "/futures/private/order/list"
futuresCancelActiveOrder = "/futures/private/order/cancel"
futuresCancelAllActiveOrders = "/futures/private/order/cancelAll"
futuresReplaceActiveOrder = "/futures/private/order/replace"
futuresGetActiveRealtimeOrders = "/futures/private/order"
futuresCreateConditionalOrder = "/futures/private/stop-order/create"
futuresGetConditionalOrders = "/futures/private/stop-order/list"
futuresCancelConditionalOrder = "/futures/private/stop-order/cancel"
futuresCancelAllConditionalOrders = "/futures/private/stop-order/cancelAll"
futuresReplaceConditionalOrder = "/futures/private/stop-order/replace"
futuresGetConditionalRealtimeOrders = "/futures/private/stop-order"
futuresPosition = "/futures/private/position/list"
futuresUpdateMargin = "/futures/private/position/change-position-margin"
futuresSetTradingStop = "/futures/private/position/trading-stop"
futuresSetLeverage = "/futures/private/position/leverage/save"
futuresSwitchPositionMode = "/futures/private/position/switch-mode"
futuresSwitchPosition = "/futures/private/tpsl/switch-mode"
futuresSwitchMargin = "/futures/private/position/switch-isolated"
futuresGetTrades = "/futures/private/execution/list"
futuresGetClosedTrades = "/futures/private/trade/closed-pnl/list"
futuresSetRiskLimit = "/futures/private/position/risk-limit"
)
// CreateFuturesOrder sends a new futures order to the exchange
func (by *Bybit) CreateFuturesOrder(ctx context.Context, positionMode int64, symbol currency.Pair, side, orderType, timeInForce,
orderLinkID, takeProfitTriggerBy, stopLossTriggerBy string,
quantity, price, takeProfit, stopLoss float64, closeOnTrigger, reduceOnly bool) (FuturesOrderDataResp, error) {
resp := struct {
Result FuturesOrderDataResp `json:"result"`
Error
}{}
params := url.Values{}
if positionMode < 0 || positionMode > 2 {
return resp.Result, errInvalidPositionMode
}
params.Set("position_idx", strconv.FormatInt(positionMode, 10))
symbolValue, err := by.FormatSymbol(symbol, asset.Futures)
if err != nil {
return resp.Result, err
}
params.Set("symbol", symbolValue)
params.Set("side", side)
params.Set("order_type", orderType)
if quantity <= 0 {
return resp.Result, errInvalidQuantity
}
params.Set("qty", strconv.FormatFloat(quantity, 'f', -1, 64))
if price != 0 {
params.Set("price", strconv.FormatFloat(price, 'f', -1, 64))
}
if timeInForce == "" {
return resp.Result, errInvalidTimeInForce
}
params.Set("time_in_force", timeInForce)
if closeOnTrigger {
params.Set("close_on_trigger", "true")
}
if orderLinkID != "" {
params.Set("order_link_id", orderLinkID)
}
if takeProfit != 0 {
params.Set("take_profit", strconv.FormatFloat(takeProfit, 'f', -1, 64))
}
if stopLoss != 0 {
params.Set("stop_loss", strconv.FormatFloat(stopLoss, 'f', -1, 64))
}
if takeProfitTriggerBy != "" {
params.Set("tp_trigger_by", takeProfitTriggerBy)
}
if stopLossTriggerBy != "" {
params.Set("sl_trigger_by", stopLossTriggerBy)
}
if reduceOnly {
params.Set("reduce_only", "true")
}
return resp.Result, by.SendAuthHTTPRequest(ctx, exchange.RestFutures, http.MethodPost, futuresCreateOrder, params, nil, &resp, futuresCreateOrderRate)
}
// GetActiveFuturesOrders gets list of futures active orders
func (by *Bybit) GetActiveFuturesOrders(ctx context.Context, symbol currency.Pair, orderStatus, direction, cursor string, limit int64) ([]FuturesActiveOrder, error) {
resp := struct {
Result struct {
Data []FuturesActiveOrder `json:"data"`
Cursor string `json:"cursor"`
} `json:"result"`
Error
}{}
params := url.Values{}
symbolValue, err := by.FormatSymbol(symbol, asset.Futures)
if err != nil {
return resp.Result.Data, err
}
params.Set("symbol", symbolValue)
if orderStatus != "" {
params.Set("order_status", orderStatus)
}
if direction != "" {
params.Set("direction", direction)
}
if limit > 0 {
params.Set("limit", strconv.FormatInt(limit, 10))
}
if cursor != "" {
params.Set("cursor", cursor)
}
return resp.Result.Data, by.SendAuthHTTPRequest(ctx, exchange.RestFutures, http.MethodGet, futuresGetActiveOrders, params, nil, &resp, futuresGetActiveOrderRate)
}
// CancelActiveFuturesOrders cancels futures unfilled or partially filled orders
func (by *Bybit) CancelActiveFuturesOrders(ctx context.Context, symbol currency.Pair, orderID, orderLinkID string) (FuturesOrderCancelResp, error) {
resp := struct {
Result FuturesOrderCancelResp `json:"result"`
Error
}{}
params := url.Values{}
symbolValue, err := by.FormatSymbol(symbol, asset.Futures)
if err != nil {
return resp.Result, err
}
params.Set("symbol", symbolValue)
if orderID == "" && orderLinkID == "" {
return resp.Result, errOrderOrOrderLinkIDMissing
}
if orderID != "" {
params.Set("order_id", orderID)
}
if orderLinkID != "" {
params.Set("order_link_id", orderLinkID)
}
return resp.Result, by.SendAuthHTTPRequest(ctx, exchange.RestFutures, http.MethodPost, futuresCancelActiveOrder, params, nil, &resp, futuresCancelOrderRate)
}
// CancelAllActiveFuturesOrders cancels all futures unfilled or partially filled orders
func (by *Bybit) CancelAllActiveFuturesOrders(ctx context.Context, symbol currency.Pair) ([]FuturesCancelOrderData, error) {
resp := struct {
Result []FuturesCancelOrderData `json:"result"`
Error
}{}
params := url.Values{}
symbolValue, err := by.FormatSymbol(symbol, asset.Futures)
if err != nil {
return resp.Result, err
}
params.Set("symbol", symbolValue)
return resp.Result, by.SendAuthHTTPRequest(ctx, exchange.RestFutures, http.MethodPost, futuresCancelAllActiveOrders, params, nil, &resp, futuresCancelAllOrderRate)
}
// ReplaceActiveFuturesOrders modify unfilled or partially filled orders
func (by *Bybit) ReplaceActiveFuturesOrders(ctx context.Context, symbol currency.Pair, orderID, orderLinkID, takeProfitTriggerBy, stopLossTriggerBy string,
updatedQty, updatedPrice, takeProfitPrice, stopLossPrice float64) (string, error) {
resp := struct {
Result struct {
OrderID string `json:"order_id"`
} `json:"result"`
Error
}{}
params := url.Values{}
symbolValue, err := by.FormatSymbol(symbol, asset.Futures)
if err != nil {
return "", err
}
params.Set("symbol", symbolValue)
if orderID == "" && orderLinkID == "" {
return "", errOrderOrOrderLinkIDMissing
}
if orderID != "" {
params.Set("order_id", orderID)
}
if orderLinkID != "" {
params.Set("order_link_id", orderLinkID)
}
if updatedQty != 0 {
params.Set("p_r_qty", strconv.FormatFloat(updatedQty, 'f', -1, 64))
}
if updatedPrice != 0 {
params.Set("p_r_price", strconv.FormatFloat(updatedPrice, 'f', -1, 64))
}
if takeProfitPrice != 0 {
params.Set("take_profit", strconv.FormatFloat(takeProfitPrice, 'f', -1, 64))
}
if stopLossPrice != 0 {
params.Set("stop_loss", strconv.FormatFloat(stopLossPrice, 'f', -1, 64))
}
if takeProfitTriggerBy != "" {
params.Set("tp_trigger_by", takeProfitTriggerBy)
}
if stopLossTriggerBy != "" {
params.Set("sl_trigger_by", stopLossTriggerBy)
}
return resp.Result.OrderID, by.SendAuthHTTPRequest(ctx, exchange.RestFutures, http.MethodPost, futuresReplaceActiveOrder, params, nil, &resp, futuresReplaceOrderRate)
}
// GetActiveRealtimeOrders query real time order data
func (by *Bybit) GetActiveRealtimeOrders(ctx context.Context, symbol currency.Pair, orderID, orderLinkID string) ([]FuturesActiveRealtimeOrder, error) {
var data []FuturesActiveRealtimeOrder
params := url.Values{}
symbolValue, err := by.FormatSymbol(symbol, asset.Futures)
if err != nil {
return data, err
}
params.Set("symbol", symbolValue)
if orderID != "" {
params.Set("order_id", orderID)
}
if orderLinkID != "" {
params.Set("order_link_id", orderLinkID)
}
if orderID == "" && orderLinkID == "" {
resp := struct {
Result []FuturesActiveRealtimeOrder `json:"result"`
Error
}{}
err = by.SendAuthHTTPRequest(ctx, exchange.RestFutures, http.MethodGet, futuresGetActiveRealtimeOrders, params, nil, &resp, futuresGetActiveRealtimeOrderRate)
if err != nil {
return data, err
}
data = append(data, resp.Result...)
} else {
resp := struct {
Result FuturesActiveRealtimeOrder `json:"result"`
Error
}{}
err = by.SendAuthHTTPRequest(ctx, exchange.RestFutures, http.MethodGet, futuresGetActiveRealtimeOrders, params, nil, &resp, futuresGetActiveRealtimeOrderRate)
if err != nil {
return data, err
}
data = append(data, resp.Result)
}
return data, nil
}
// CreateConditionalFuturesOrder sends a new conditional futures order to the exchange
func (by *Bybit) CreateConditionalFuturesOrder(ctx context.Context, positionMode int64, symbol currency.Pair, side, orderType, timeInForce,
orderLinkID, takeProfitTriggerBy, stopLossTriggerBy, triggerBy string,
quantity, price, takeProfit, stopLoss, basePrice, stopPrice float64, closeOnTrigger bool) (FuturesConditionalOrderResp, error) {
resp := struct {
Result FuturesConditionalOrderResp `json:"result"`
Error
}{}
params := url.Values{}
if positionMode < 0 || positionMode > 2 {
return resp.Result, errInvalidPositionMode
}
params.Set("position_idx", strconv.FormatInt(positionMode, 10))
symbolValue, err := by.FormatSymbol(symbol, asset.Futures)
if err != nil {
return resp.Result, err
}
params.Set("symbol", symbolValue)
params.Set("side", side)
params.Set("order_type", orderType)
if quantity <= 0 {
return resp.Result, errInvalidQuantity
}
params.Set("qty", strconv.FormatFloat(quantity, 'f', -1, 64))
if price != 0 {
params.Set("price", strconv.FormatFloat(price, 'f', -1, 64))
}
if basePrice <= 0 {
return resp.Result, errInvalidBasePrice
}
params.Set("base_price", strconv.FormatFloat(basePrice, 'f', -1, 64))
if stopPrice <= 0 {
return resp.Result, errInvalidStopPrice
}
params.Set("stop_px", strconv.FormatFloat(stopPrice, 'f', -1, 64))
if timeInForce == "" {
return resp.Result, errInvalidTimeInForce
}
params.Set("time_in_force", timeInForce)
if triggerBy != "" {
params.Set("trigger_by", triggerBy)
}
if closeOnTrigger {
params.Set("close_on_trigger", "true")
}
if orderLinkID != "" {
params.Set("order_link_id", orderLinkID)
}
if takeProfit != 0 {
params.Set("take_profit", strconv.FormatFloat(takeProfit, 'f', -1, 64))
}
if stopLoss != 0 {
params.Set("stop_loss", strconv.FormatFloat(stopLoss, 'f', -1, 64))
}
if takeProfitTriggerBy != "" {
params.Set("tp_trigger_by", takeProfitTriggerBy)
}
if stopLossTriggerBy != "" {
params.Set("sl_trigger_by", stopLossTriggerBy)
}
return resp.Result, by.SendAuthHTTPRequest(ctx, exchange.RestFutures, http.MethodPost, futuresCreateConditionalOrder, params, nil, &resp, futuresCreateConditionalOrderRate)
}
// GetConditionalFuturesOrders gets list of futures conditional orders
func (by *Bybit) GetConditionalFuturesOrders(ctx context.Context, symbol currency.Pair, stopOrderStatus, direction, cursor string, limit int64) ([]FuturesConditionalOrders, error) {
resp := struct {
Result struct {
Result []FuturesConditionalOrders `json:"data"`
Cursor string `json:"cursor"`
} `json:"result"`
Error
}{}
params := url.Values{}
symbolValue, err := by.FormatSymbol(symbol, asset.Futures)
if err != nil {
return resp.Result.Result, err
}
params.Set("symbol", symbolValue)
if stopOrderStatus != "" {
params.Set("stop_order_status", stopOrderStatus)
}
if direction != "" {
params.Set("direction", direction)
}
if limit > 0 {
params.Set("limit", strconv.FormatInt(limit, 10))
}
if cursor != "" {
params.Set("cursor", cursor)
}
return resp.Result.Result, by.SendAuthHTTPRequest(ctx, exchange.RestFutures, http.MethodGet, futuresGetConditionalOrders, params, nil, &resp, futuresGetConditionalOrderRate)
}
// CancelConditionalFuturesOrders cancels untriggered conditional orders
func (by *Bybit) CancelConditionalFuturesOrders(ctx context.Context, symbol currency.Pair, stopOrderID, orderLinkID string) (string, error) {
resp := struct {
Result struct {
StopOrderID string `json:"stop_order_id"`
} `json:"result"`
Error
}{}
params := url.Values{}
symbolValue, err := by.FormatSymbol(symbol, asset.Futures)
if err != nil {
return "", err
}
params.Set("symbol", symbolValue)
if stopOrderID == "" && orderLinkID == "" {
return "", errStopOrderOrOrderLinkIDMissing
}
if stopOrderID != "" {
params.Set("stop_order_id", stopOrderID)
}
if orderLinkID != "" {
params.Set("order_link_id", orderLinkID)
}
return resp.Result.StopOrderID, by.SendAuthHTTPRequest(ctx, exchange.RestFutures, http.MethodPost, futuresCancelConditionalOrder, params, nil, &resp, futuresCancelConditionalOrderRate)
}
// CancelAllConditionalFuturesOrders cancels all untriggered conditional orders
func (by *Bybit) CancelAllConditionalFuturesOrders(ctx context.Context, symbol currency.Pair) ([]FuturesCancelOrderResp, error) {
resp := struct {
Result []FuturesCancelOrderResp `json:"result"`
Error
}{}
params := url.Values{}
symbolValue, err := by.FormatSymbol(symbol, asset.Futures)
if err != nil {
return resp.Result, err
}
params.Set("symbol", symbolValue)
return resp.Result, by.SendAuthHTTPRequest(ctx, exchange.RestFutures, http.MethodPost, futuresCancelAllConditionalOrders, params, nil, &resp, futuresCancelAllConditionalOrderRate)
}
// ReplaceConditionalFuturesOrders modify unfilled or partially filled conditional orders
func (by *Bybit) ReplaceConditionalFuturesOrders(ctx context.Context, symbol currency.Pair, stopOrderID, orderLinkID, takeProfitTriggerBy, stopLossTriggerBy string,
updatedQty, updatedPrice, takeProfitPrice, stopLossPrice, orderTriggerPrice float64) (string, error) {
resp := struct {
Result struct {
OrderID string `json:"stop_order_id"`
} `json:"result"`
Error
}{}
params := url.Values{}
symbolValue, err := by.FormatSymbol(symbol, asset.Futures)
if err != nil {
return "", err
}
params.Set("symbol", symbolValue)
if stopOrderID == "" && orderLinkID == "" {
return "", errStopOrderOrOrderLinkIDMissing
}
if stopOrderID != "" {
params.Set("stop_order_id", stopOrderID)
}
if orderLinkID != "" {
params.Set("order_link_id", orderLinkID)
}
if updatedQty != 0 {
params.Set("p_r_qty", strconv.FormatFloat(updatedQty, 'f', -1, 64))
}
if updatedPrice != 0 {
params.Set("p_r_price", strconv.FormatFloat(updatedPrice, 'f', -1, 64))
}
if orderTriggerPrice != 0 {
params.Set("p_r_trigger_price", strconv.FormatFloat(orderTriggerPrice, 'f', -1, 64))
}
if takeProfitPrice != 0 {
params.Set("take_profit", strconv.FormatFloat(takeProfitPrice, 'f', -1, 64))
}
if stopLossPrice != 0 {
params.Set("stop_loss", strconv.FormatFloat(stopLossPrice, 'f', -1, 64))
}
if takeProfitTriggerBy != "" {
params.Set("tp_trigger_by", takeProfitTriggerBy)
}
if stopLossTriggerBy != "" {
params.Set("sl_trigger_by", stopLossTriggerBy)
}
return resp.Result.OrderID, by.SendAuthHTTPRequest(ctx, exchange.RestFutures, http.MethodPost, futuresReplaceConditionalOrder, params, nil, &resp, futuresReplaceConditionalOrderRate)
}
// GetConditionalRealtimeOrders query real time conditional order data
func (by *Bybit) GetConditionalRealtimeOrders(ctx context.Context, symbol currency.Pair, stopOrderID, orderLinkID string) ([]FuturesConditionalRealtimeOrder, error) {
var data []FuturesConditionalRealtimeOrder
params := url.Values{}
symbolValue, err := by.FormatSymbol(symbol, asset.Futures)
if err != nil {
return data, err
}
params.Set("symbol", symbolValue)
if stopOrderID != "" {
params.Set("stop_order_id", stopOrderID)
}
if orderLinkID != "" {
params.Set("order_link_id", orderLinkID)
}
if stopOrderID == "" && orderLinkID == "" {
resp := struct {
Result []FuturesConditionalRealtimeOrder `json:"result"`
Error
}{}
err = by.SendAuthHTTPRequest(ctx, exchange.RestFutures, http.MethodGet, futuresGetConditionalRealtimeOrders, params, nil, &resp, futuresGetConditionalRealtimeOrderRate)
if err != nil {
return data, err
}
data = append(data, resp.Result...)
} else {
resp := struct {
Result FuturesConditionalRealtimeOrder `json:"result"`
Error
}{}
err = by.SendAuthHTTPRequest(ctx, exchange.RestFutures, http.MethodGet, futuresGetConditionalRealtimeOrders, params, nil, &resp, futuresGetConditionalRealtimeOrderRate)
if err != nil {
return data, err
}
data = append(data, resp.Result)
}
return data, nil
}
// GetPositions returns list of user positions
func (by *Bybit) GetPositions(ctx context.Context, symbol currency.Pair) ([]PositionResp, error) {
params := url.Values{}
resp := struct {
Result []struct {
Data PositionResp `json:"data"`
IsValid bool `json:"is_valid"`
} `json:"result"`
Error
}{}
if !symbol.IsEmpty() {
symbolValue, err := by.FormatSymbol(symbol, asset.Futures)
if err != nil {
return nil, err
}
params.Set("symbol", symbolValue)
}
err := by.SendAuthHTTPRequest(ctx, exchange.RestFutures, http.MethodGet, futuresPosition, params, nil, &resp, futuresPositionRate)
if err != nil {
return nil, err
}
data := make([]PositionResp, len(resp.Result))
for x := range resp.Result {
data[x] = resp.Result[x].Data
}
return data, nil
}
// SetMargin updates margin
func (by *Bybit) SetMargin(ctx context.Context, positionMode int64, symbol currency.Pair, margin string) (float64, error) {
resp := struct {
Result float64 `json:"result"`
Error
}{}
params := url.Values{}
symbolValue, err := by.FormatSymbol(symbol, asset.Futures)
if err != nil {
return resp.Result, err
}
params.Set("symbol", symbolValue)
if positionMode < 0 || positionMode > 2 {
return resp.Result, errInvalidPositionMode
}
params.Set("position_idx", strconv.FormatInt(positionMode, 10))
if margin == "" {
return resp.Result, errInvalidMargin
}
params.Set("margin", margin)
return resp.Result, by.SendAuthHTTPRequest(ctx, exchange.RestFutures, http.MethodPost, futuresUpdateMargin, params, nil, &resp, futuresUpdateMarginRate)
}
// SetTradingAndStop sets take profit, stop loss, and trailing stop for your open position
func (by *Bybit) SetTradingAndStop(ctx context.Context, positionMode int64, symbol currency.Pair, takeProfit, stopLoss, trailingStop, newTrailingActive, stopLossQty, takeProfitQty float64, takeProfitTriggerBy, stopLossTriggerBy string) (SetTradingAndStopResp, error) {
resp := struct {
Result SetTradingAndStopResp `json:"result"`
Error
}{}
params := url.Values{}
symbolValue, err := by.FormatSymbol(symbol, asset.Futures)
if err != nil {
return resp.Result, err
}
params.Set("symbol", symbolValue)
if positionMode < 0 || positionMode > 2 {
return resp.Result, errInvalidPositionMode
}
params.Set("position_idx", strconv.FormatInt(positionMode, 10))
if takeProfit >= 0 {
params.Set("take_profit", strconv.FormatFloat(takeProfit, 'f', -1, 64))
}
if stopLoss >= 0 {
params.Set("stop_loss", strconv.FormatFloat(stopLoss, 'f', -1, 64))
}
if trailingStop >= 0 {
params.Set("trailing_stop", strconv.FormatFloat(trailingStop, 'f', -1, 64))
}
if newTrailingActive != 0 {
params.Set("new_trailing_active", strconv.FormatFloat(newTrailingActive, 'f', -1, 64))
}
if stopLossQty != 0 {
params.Set("sl_size", strconv.FormatFloat(stopLossQty, 'f', -1, 64))
}
if takeProfitQty != 0 {
params.Set("tp_size", strconv.FormatFloat(takeProfitQty, 'f', -1, 64))
}
if takeProfitTriggerBy != "" {
params.Set("tp_trigger_by", takeProfitTriggerBy)
}
if stopLossTriggerBy != "" {
params.Set("sl_trigger_by", stopLossTriggerBy)
}
return resp.Result, by.SendAuthHTTPRequest(ctx, exchange.RestFutures, http.MethodPost, futuresSetTradingStop, params, nil, &resp, futuresSetTradingStopRate)
}
// SetLeverageLevel sets leverage
func (by *Bybit) SetLeverageLevel(ctx context.Context, symbol currency.Pair, buyLeverage, sellLeverage float64) (float64, error) {
resp := struct {
Result float64 `json:"result"`
Error
}{}
params := url.Values{}
symbolValue, err := by.FormatSymbol(symbol, asset.Futures)
if err != nil {
return resp.Result, err
}
params.Set("symbol", symbolValue)
params.Set("buy_leverage", strconv.FormatFloat(buyLeverage, 'f', -1, 64))
params.Set("sell_leverage", strconv.FormatFloat(sellLeverage, 'f', -1, 64))
return resp.Result, by.SendAuthHTTPRequest(ctx, exchange.RestFutures, http.MethodPost, futuresSetLeverage, params, nil, &resp, futuresSetLeverageRate)
}
// ChangePositionMode switches mode between One-Way or Hedge Mode
func (by *Bybit) ChangePositionMode(ctx context.Context, symbol currency.Pair, mode int64) error {
params := url.Values{}
symbolValue, err := by.FormatSymbol(symbol, asset.Futures)
if err != nil {
return err
}
params.Set("symbol", symbolValue)
params.Set("mode", strconv.FormatInt(mode, 10))
return by.SendAuthHTTPRequest(ctx, exchange.RestFutures, http.MethodPost, futuresSwitchPositionMode, params, nil, nil, futuresSwitchPositionModeRate)
}
// ChangeMode switches mode between full or partial position
func (by *Bybit) ChangeMode(ctx context.Context, symbol currency.Pair, takeProfitStopLoss string) (string, error) {
resp := struct {
Result struct {
Mode string `json:"tp_sl_mode"`
} `json:"result"`
Error
}{}
params := url.Values{}
symbolValue, err := by.FormatSymbol(symbol, asset.Futures)
if err != nil {
return resp.Result.Mode, err
}
params.Set("symbol", symbolValue)
if takeProfitStopLoss == "" {
return resp.Result.Mode, errInvalidTakeProfitStopLoss
}
params.Set("tp_sl_mode", takeProfitStopLoss)
return resp.Result.Mode, by.SendAuthHTTPRequest(ctx, exchange.RestFutures, http.MethodPost, futuresSwitchPosition, params, nil, &resp, futuresSwitchPositionRate)
}
// ChangeMargin switches margin between cross or isolated
func (by *Bybit) ChangeMargin(ctx context.Context, symbol currency.Pair, buyLeverage, sellLeverage float64, isIsolated bool) error {
params := url.Values{}
symbolValue, err := by.FormatSymbol(symbol, asset.Futures)
if err != nil {
return err
}
params.Set("symbol", symbolValue)
params.Set("buy_leverage", strconv.FormatFloat(buyLeverage, 'f', -1, 64))
params.Set("sell_leverage", strconv.FormatFloat(sellLeverage, 'f', -1, 64))
if isIsolated {
params.Set("is_isolated", "true")
} else {
params.Set("is_isolated", "false")
}
return by.SendAuthHTTPRequest(ctx, exchange.RestFutures, http.MethodPost, futuresSwitchMargin, params, nil, nil, futuresSwitchMarginRate)
}
// GetTradeRecords returns list of user trades
func (by *Bybit) GetTradeRecords(ctx context.Context, symbol currency.Pair, orderID, order string, startTime, page, limit int64) ([]TradeResp, error) {
params := url.Values{}
resp := struct {
Data struct {
OrderID string `json:"order_id"`
Trades []TradeResp `json:"trade_list"`
} `json:"result"`
Error
}{}
symbolValue, err := by.FormatSymbol(symbol, asset.Futures)
if err != nil {
return resp.Data.Trades, err
}
params.Set("symbol", symbolValue)
if orderID != "" {
params.Set("order_id", orderID)
}
if order != "" {
params.Set("order", order)
}
if startTime != 0 {
params.Set("start_time", strconv.FormatInt(startTime, 10))
}
if page != 0 {
params.Set("page", strconv.FormatInt(page, 10))
}
if limit > 0 {
params.Set("limit", strconv.FormatInt(limit, 10))
}
return resp.Data.Trades, by.SendAuthHTTPRequest(ctx, exchange.RestFutures, http.MethodGet, futuresGetTrades, params, nil, &resp, futuresGetTradeRate)
}
// GetClosedTrades returns closed profit and loss records
func (by *Bybit) GetClosedTrades(ctx context.Context, symbol currency.Pair, executionType string, startTime, endTime time.Time, page, limit int64) ([]ClosedTrades, error) {
params := url.Values{}
resp := struct {
Data struct {
CurrentPage int64 `json:"current_page"`
Trades []ClosedTrades `json:"data"`
} `json:"result"`
Error
}{}
symbolValue, err := by.FormatSymbol(symbol, asset.Futures)
if err != nil {
return resp.Data.Trades, err
}
params.Set("symbol", symbolValue)
if executionType != "" {
params.Set("execution_type", executionType)
}
if !startTime.IsZero() {
params.Set("start_time", strconv.FormatInt(startTime.Unix(), 10))
}
if !endTime.IsZero() {
params.Set("end_time", strconv.FormatInt(endTime.Unix(), 10))
}
if page > 0 && page <= 50 {
params.Set("page", strconv.FormatInt(page, 10))
}
if limit > 0 {
params.Set("limit", strconv.FormatInt(limit, 10))
}
return resp.Data.Trades, by.SendAuthHTTPRequest(ctx, exchange.RestFutures, http.MethodGet, futuresGetClosedTrades, params, nil, &resp, futuresDefaultRate)
}
// SetRiskLimit sets risk limit
func (by *Bybit) SetRiskLimit(ctx context.Context, symbol currency.Pair, riskID, positionMode int64) (int64, error) {
resp := struct {
Result struct {
RiskID int64 `json:"risk_id"`
} `json:"result"`
Error
}{}
params := url.Values{}
symbolValue, err := by.FormatSymbol(symbol, asset.Futures)
if err != nil {
return resp.Result.RiskID, err
}
params.Set("symbol", symbolValue)
if riskID <= 0 {
return resp.Result.RiskID, errInvalidRiskID
}
params.Set("risk_id", strconv.FormatInt(riskID, 10))
if positionMode < 0 || positionMode > 2 {
return resp.Result.RiskID, errInvalidPositionMode
}
params.Set("position_idx", strconv.FormatInt(positionMode, 10))
return resp.Result.RiskID, by.SendAuthHTTPRequest(ctx, exchange.RestFutures, http.MethodPost, futuresSetRiskLimit, params, nil, &resp, futuresDefaultRate)
}