mirror of
https://github.com/d0zingcat/gocryptotrader.git
synced 2026-05-20 07:26:46 +00:00
* Added TimeInForce type and updated related files * Linter issue fix and minor coinbasepro type update * Bitrex consts update * added unit test and minor changes in bittrex * Unit tests update * Fix minor linter issues * Update TestStringToTimeInForce unit test * Exchange test template change * A different approach * fix conflict with gateio timeInForce * minor exchange template update * Minor fix to test_files template * Update order tests * Complete updating the order unit tests * Updating exchange wrapper and test template files * update kucoin and deribit wrapper to match the time in force change * minor comment update * fix time-in-force related test errors * linter issue fix * ADD_NEW_EXCHANGE documentation update * time in force constants, functions and unit tests update * shift tif policies to TimeInForce * Update time-in-force, related functions, and unit tests * fix linter issue and time-in-force processing * added a good till crossing tif value * order type fix and fix related tim-in-force entries * update time-in-force unmarshaling and unit test * consistency guideline added * fix time-in-force error in gateio * linter issue fix * update based on review comments * add unit test and fix missing issues * minor fix and added benchmark unit test * change GTT to GTC for limit * fix linter issue * added time-in-force value to place order param * fix minor issues based on review comment and move tif code to separate files * update on exchanges linked to time-in-force * resolve missing review comments * minor linter issues fix * added time-in-force handler and update timeInForce parametered endpoint * minor fixes based on review * nits fix * update based on review * linter fix * rm getTimeInForce func and minor change to time-in-force * minor change * update based on review comments * wrappers and time-in-force calling approach * minor change * update gateio string to timeInForce conversion and unit test * update exchange template * update wrapper template file * policy comments, and template files update * rename all exchange types name to Exchange * update on template files and template generation * templates and generation code and other updates * linter issue fix * added subscriptions and websocket templates * update ADD_NEW_EXCHANGE.md with recent binance functions and implementations * rename template files and update unit tests * minor template and unit test fix * rename templates and fix on unit tests * update on template files and documentation * removed unnecessary tag fix and update templates * fix Add_NEW_EXCHANGE.md doc file * formatting, comments, and error checks update on template files * rename exchange receivers to e and ex for consistency * rename unit test exchange receiver and minor updates * linter issues fix * fix deribit issue and minor style update * fix test issues caused by receiver change * raname local variables exchange declaration variables * update templates comments * update templates and related comments * renamed ex to e * update template comments * toggle WS to false to improve coverage * template comments update * added test coverage to Ws enabled and minor changes --------- Co-authored-by: Samuel Reid <43227667+cranktakular@users.noreply.github.com>
165 lines
5.6 KiB
Go
165 lines
5.6 KiB
Go
package gateio
|
|
|
|
import (
|
|
"context"
|
|
"errors"
|
|
"fmt"
|
|
"strconv"
|
|
|
|
"github.com/thrasher-corp/gocryptotrader/common"
|
|
"github.com/thrasher-corp/gocryptotrader/currency"
|
|
"github.com/thrasher-corp/gocryptotrader/exchange/websocket"
|
|
"github.com/thrasher-corp/gocryptotrader/exchanges/asset"
|
|
"github.com/thrasher-corp/gocryptotrader/exchanges/order"
|
|
)
|
|
|
|
var (
|
|
errOrdersEmpty = errors.New("orders cannot be empty")
|
|
errNoOrdersToCancel = errors.New("no orders to cancel")
|
|
errChannelEmpty = errors.New("channel cannot be empty")
|
|
)
|
|
|
|
// authenticateSpot sends an authentication message to the websocket connection
|
|
func (e *Exchange) authenticateSpot(ctx context.Context, conn websocket.Connection) error {
|
|
return e.websocketLogin(ctx, conn, "spot.login")
|
|
}
|
|
|
|
// WebsocketSpotSubmitOrder submits an order via the websocket connection
|
|
func (e *Exchange) WebsocketSpotSubmitOrder(ctx context.Context, order *CreateOrderRequest) (*WebsocketOrderResponse, error) {
|
|
resps, err := e.WebsocketSpotSubmitOrders(ctx, order)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
if len(resps) != 1 {
|
|
return nil, common.ErrInvalidResponse
|
|
}
|
|
return &resps[0], nil
|
|
}
|
|
|
|
// WebsocketSpotSubmitOrders submits orders via the websocket connection. You can
|
|
// send multiple orders in a single request. But only for one asset route.
|
|
func (e *Exchange) WebsocketSpotSubmitOrders(ctx context.Context, orders ...*CreateOrderRequest) ([]WebsocketOrderResponse, error) {
|
|
if len(orders) == 0 {
|
|
return nil, errOrdersEmpty
|
|
}
|
|
|
|
for i := range orders {
|
|
if orders[i].Text == "" {
|
|
// API requires Text field, or it will be rejected
|
|
orders[i].Text = "t-" + strconv.FormatInt(e.Counter.IncrementAndGet(), 10)
|
|
}
|
|
if orders[i].CurrencyPair.IsEmpty() {
|
|
return nil, currency.ErrCurrencyPairEmpty
|
|
}
|
|
if orders[i].Side == "" {
|
|
return nil, order.ErrSideIsInvalid
|
|
}
|
|
if orders[i].Amount == 0 {
|
|
return nil, errInvalidAmount
|
|
}
|
|
if orders[i].Type == "limit" && orders[i].Price == 0 {
|
|
return nil, errInvalidPrice
|
|
}
|
|
}
|
|
|
|
if len(orders) == 1 {
|
|
var singleResponse WebsocketOrderResponse
|
|
return []WebsocketOrderResponse{singleResponse}, e.SendWebsocketRequest(ctx, spotPlaceOrderEPL, "spot.order_place", asset.Spot, orders[0], &singleResponse, 2)
|
|
}
|
|
var resp []WebsocketOrderResponse
|
|
return resp, e.SendWebsocketRequest(ctx, spotBatchOrdersEPL, "spot.order_place", asset.Spot, orders, &resp, 2)
|
|
}
|
|
|
|
// WebsocketSpotCancelOrder cancels an order via the websocket connection
|
|
func (e *Exchange) WebsocketSpotCancelOrder(ctx context.Context, orderID string, pair currency.Pair, account string) (*WebsocketOrderResponse, error) {
|
|
if orderID == "" {
|
|
return nil, order.ErrOrderIDNotSet
|
|
}
|
|
if pair.IsEmpty() {
|
|
return nil, currency.ErrCurrencyPairEmpty
|
|
}
|
|
|
|
params := &WebsocketOrderRequest{OrderID: orderID, Pair: pair.String(), Account: account}
|
|
|
|
var resp WebsocketOrderResponse
|
|
return &resp, e.SendWebsocketRequest(ctx, spotCancelSingleOrderEPL, "spot.order_cancel", asset.Spot, params, &resp, 1)
|
|
}
|
|
|
|
// WebsocketSpotCancelAllOrdersByIDs cancels multiple orders via the websocket
|
|
func (e *Exchange) WebsocketSpotCancelAllOrdersByIDs(ctx context.Context, o []WebsocketOrderBatchRequest) ([]WebsocketCancellAllResponse, error) {
|
|
if len(o) == 0 {
|
|
return nil, errNoOrdersToCancel
|
|
}
|
|
|
|
for i := range o {
|
|
if o[i].OrderID == "" {
|
|
return nil, order.ErrOrderIDNotSet
|
|
}
|
|
if o[i].Pair.IsEmpty() {
|
|
return nil, currency.ErrCurrencyPairEmpty
|
|
}
|
|
}
|
|
|
|
var resp []WebsocketCancellAllResponse
|
|
return resp, e.SendWebsocketRequest(ctx, spotCancelBatchOrdersEPL, "spot.order_cancel_ids", asset.Spot, o, &resp, 2)
|
|
}
|
|
|
|
// WebsocketSpotCancelAllOrdersByPair cancels all orders for a specific pair
|
|
func (e *Exchange) WebsocketSpotCancelAllOrdersByPair(ctx context.Context, pair currency.Pair, side order.Side, account string) ([]WebsocketOrderResponse, error) {
|
|
if !pair.IsEmpty() && side == order.UnknownSide {
|
|
// This case will cancel all orders for every pair, this can be introduced later
|
|
return nil, fmt.Errorf("'%v' %w while pair is set", side, order.ErrSideIsInvalid)
|
|
}
|
|
|
|
sideStr := ""
|
|
if side != order.UnknownSide {
|
|
sideStr = side.Lower()
|
|
}
|
|
|
|
params := &WebsocketCancelParam{
|
|
Pair: pair,
|
|
Side: sideStr,
|
|
Account: account,
|
|
}
|
|
|
|
var resp []WebsocketOrderResponse
|
|
return resp, e.SendWebsocketRequest(ctx, spotCancelAllOpenOrdersEPL, "spot.order_cancel_cp", asset.Spot, params, &resp, 1)
|
|
}
|
|
|
|
// WebsocketSpotAmendOrder amends an order via the websocket connection
|
|
func (e *Exchange) WebsocketSpotAmendOrder(ctx context.Context, amend *WebsocketAmendOrder) (*WebsocketOrderResponse, error) {
|
|
if amend == nil {
|
|
return nil, fmt.Errorf("%w: %T", common.ErrNilPointer, amend)
|
|
}
|
|
|
|
if amend.OrderID == "" {
|
|
return nil, order.ErrOrderIDNotSet
|
|
}
|
|
|
|
if amend.Pair.IsEmpty() {
|
|
return nil, currency.ErrCurrencyPairEmpty
|
|
}
|
|
|
|
if amend.Amount == "" && amend.Price == "" {
|
|
return nil, fmt.Errorf("%w: amount or price must be set", errInvalidAmount)
|
|
}
|
|
|
|
var resp WebsocketOrderResponse
|
|
return &resp, e.SendWebsocketRequest(ctx, spotAmendOrderEPL, "spot.order_amend", asset.Spot, amend, &resp, 1)
|
|
}
|
|
|
|
// WebsocketSpotGetOrderStatus gets the status of an order via the websocket connection
|
|
func (e *Exchange) WebsocketSpotGetOrderStatus(ctx context.Context, orderID string, pair currency.Pair, account string) (*WebsocketOrderResponse, error) {
|
|
if orderID == "" {
|
|
return nil, order.ErrOrderIDNotSet
|
|
}
|
|
if pair.IsEmpty() {
|
|
return nil, currency.ErrCurrencyPairEmpty
|
|
}
|
|
|
|
params := &WebsocketOrderRequest{OrderID: orderID, Pair: pair.String(), Account: account}
|
|
|
|
var resp WebsocketOrderResponse
|
|
return &resp, e.SendWebsocketRequest(ctx, spotGetOrdersEPL, "spot.order_status", asset.Spot, params, &resp, 1)
|
|
}
|