Files
gocryptotrader/exchanges/gateio/gateio_websocket_request_spot.go
Ryan O'Hara-Reid e99adca86f encoding/json: Add custom JSON package with build tag support for Sonic (#1623)
* tag optional sonic and allow full library conversion

* Add workflow and disallow arm and darwin usage

* Add basic hotswap benchmark

* linter: fix

* use bash

* linter: fix?

* Fix whoopsie, add to make file, also add mention in features list.

* test enforcement

* actually read documentation see if this works

* linter: fix

* linter: fix

* sonic: bump tagged version

* encoding/json: drop build tag arch and os filters

* encoding/json: consolidate tests

* encoding/json: log build tag usage

* rm superfluous builds

* glorious/nits: add template change and regen docs

* glorious/nits: update commentary on nolint directive

* glorious/nits: rm init func and log results in main.go

* Test to actually pull flag in

* linter: fix

* thrasher: nits

* gk: nits 4 goflags goooooooooo!

* gk: nits rn

* make sonic default json implementation

* screen 386

* linter: fix

* Add commentary

* glorious: nits Makefile not working

* gk: nits

* gk: nits whoops

* whoopsirino

* mention 32bit systems won't be sonic

* gk: super-duper nit of extremes

---------

Co-authored-by: Ryan O'Hara-Reid <ryan.oharareid@thrasher.io>
2025-02-20 16:05:55 +11:00

225 lines
7.4 KiB
Go

package gateio
import (
"context"
"errors"
"fmt"
"strconv"
"strings"
"time"
"github.com/thrasher-corp/gocryptotrader/common"
"github.com/thrasher-corp/gocryptotrader/currency"
"github.com/thrasher-corp/gocryptotrader/encoding/json"
"github.com/thrasher-corp/gocryptotrader/exchanges/asset"
"github.com/thrasher-corp/gocryptotrader/exchanges/order"
"github.com/thrasher-corp/gocryptotrader/exchanges/request"
)
var (
errOrdersEmpty = errors.New("orders cannot be empty")
errNoOrdersToCancel = errors.New("no orders to cancel")
errChannelEmpty = errors.New("channel cannot be empty")
)
// WebsocketSpotSubmitOrder submits an order via the websocket connection
func (g *Gateio) WebsocketSpotSubmitOrder(ctx context.Context, order *WebsocketOrder) ([]WebsocketOrderResponse, error) {
return g.WebsocketSpotSubmitOrders(ctx, []WebsocketOrder{*order})
}
// WebsocketSpotSubmitOrders submits orders via the websocket connection. You can
// send multiple orders in a single request. But only for one asset route.
func (g *Gateio) WebsocketSpotSubmitOrders(ctx context.Context, orders []WebsocketOrder) ([]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(g.Counter.IncrementAndGet(), 10)
}
if orders[i].CurrencyPair == "" {
return nil, currency.ErrCurrencyPairEmpty
}
if orders[i].Side == "" {
return nil, order.ErrSideIsInvalid
}
if orders[i].Amount == "" {
return nil, errInvalidAmount
}
if orders[i].Type == "limit" && orders[i].Price == "" {
return nil, errInvalidPrice
}
}
if len(orders) == 1 {
var singleResponse WebsocketOrderResponse
return []WebsocketOrderResponse{singleResponse}, g.SendWebsocketRequest(ctx, spotPlaceOrderEPL, "spot.order_place", asset.Spot, orders[0], &singleResponse, 2)
}
var resp []WebsocketOrderResponse
return resp, g.SendWebsocketRequest(ctx, spotBatchOrdersEPL, "spot.order_place", asset.Spot, orders, &resp, 2)
}
// WebsocketSpotCancelOrder cancels an order via the websocket connection
func (g *Gateio) 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, g.SendWebsocketRequest(ctx, spotCancelSingleOrderEPL, "spot.order_cancel", asset.Spot, params, &resp, 1)
}
// WebsocketSpotCancelAllOrdersByIDs cancels multiple orders via the websocket
func (g *Gateio) 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, g.SendWebsocketRequest(ctx, spotCancelBatchOrdersEPL, "spot.order_cancel_ids", asset.Spot, o, &resp, 2)
}
// WebsocketSpotCancelAllOrdersByPair cancels all orders for a specific pair
func (g *Gateio) 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, g.SendWebsocketRequest(ctx, spotCancelAllOpenOrdersEPL, "spot.order_cancel_cp", asset.Spot, params, &resp, 1)
}
// WebsocketSpotAmendOrder amends an order via the websocket connection
func (g *Gateio) 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, g.SendWebsocketRequest(ctx, spotAmendOrderEPL, "spot.order_amend", asset.Spot, amend, &resp, 1)
}
// WebsocketSpotGetOrderStatus gets the status of an order via the websocket connection
func (g *Gateio) 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, g.SendWebsocketRequest(ctx, spotGetOrdersEPL, "spot.order_status", asset.Spot, params, &resp, 1)
}
// funnelResult is used to unmarshal the result of a websocket request back to the required caller type
type funnelResult struct {
Result any `json:"result"`
}
// SendWebsocketRequest sends a websocket request to the exchange
func (g *Gateio) SendWebsocketRequest(ctx context.Context, epl request.EndpointLimit, channel string, connSignature, params, result any, expectedResponses int) error {
paramPayload, err := json.Marshal(params)
if err != nil {
return err
}
conn, err := g.Websocket.GetConnection(connSignature)
if err != nil {
return err
}
tn := time.Now().Unix()
req := &WebsocketRequest{
Time: tn,
Channel: channel,
Event: "api",
Payload: WebsocketPayload{
// This request ID associated with the payload is the match to the
// response.
RequestID: strconv.FormatInt(conn.GenerateMessageID(false), 10),
RequestParam: paramPayload,
Timestamp: strconv.FormatInt(tn, 10),
},
}
responses, err := conn.SendMessageReturnResponsesWithInspector(ctx, epl, req.Payload.RequestID, req, expectedResponses, wsRespAckInspector{})
if err != nil {
return err
}
if len(responses) == 0 {
return common.ErrNoResponse
}
var inbound WebsocketAPIResponse
// The last response is the one we want to unmarshal, the other is just
// an ack. If the request fails on the ACK then we can unmarshal the error
// from that as the next response won't come anyway.
endResponse := responses[len(responses)-1]
if err := json.Unmarshal(endResponse, &inbound); err != nil {
return err
}
if inbound.Header.Status != "200" {
var wsErr WebsocketErrors
if err := json.Unmarshal(inbound.Data, &wsErr); err != nil {
return err
}
return fmt.Errorf("%s: %s", wsErr.Errors.Label, wsErr.Errors.Message)
}
return json.Unmarshal(inbound.Data, &funnelResult{Result: result})
}
type wsRespAckInspector struct{}
// IsFinal checks the payload for an ack, it returns true if the payload does not contain an ack.
// This will force the cancellation of further waiting for responses.
func (wsRespAckInspector) IsFinal(data []byte) bool {
return !strings.Contains(string(data), "ack")
}