mirror of
https://github.com/d0zingcat/gocryptotrader.git
synced 2026-05-13 23:16:45 +00:00
* Websockets: Move Subscription to its own package This allows the small type to be imported from both `config` and from `stream` without an import cycle, so we don't have to repeat ourselves * Subs: Renamed Currency to Pair This was being mis-used through much of the code, and since we're already touching everything, we might as well fix it * Websockets: Add Subscription configuration * Binance: Add subscription configuration * Kucoin: Subscription configuration * Simplify GenerateDefaultSubs * Improve TestGenSubs coverage * Test Candle Sub generation * Support Candle intervals * Full responsibility for formatting Channel name on GenerateDefaultSubs OR consumer of Subscribe * Simplify generatePayloads as a result * Fix test coverage of asset types in processMarketSnapshot * Exchanges: Abstract ParallelChanOp * Tests: Generic ws mock instances * Kucoin: Fix intermittent conflict in test currs Use isolated test instance for `TestGetOpenInterest`. `TestGetOpenInterest` would occassionally change pairs before GenerateDefault Subs.
496 lines
13 KiB
Go
496 lines
13 KiB
Go
package coinbasepro
|
|
|
|
import (
|
|
"context"
|
|
"encoding/json"
|
|
"errors"
|
|
"fmt"
|
|
"net/http"
|
|
"strconv"
|
|
"time"
|
|
|
|
"github.com/gorilla/websocket"
|
|
"github.com/thrasher-corp/gocryptotrader/common"
|
|
"github.com/thrasher-corp/gocryptotrader/common/convert"
|
|
"github.com/thrasher-corp/gocryptotrader/common/crypto"
|
|
"github.com/thrasher-corp/gocryptotrader/currency"
|
|
"github.com/thrasher-corp/gocryptotrader/exchanges/account"
|
|
"github.com/thrasher-corp/gocryptotrader/exchanges/asset"
|
|
"github.com/thrasher-corp/gocryptotrader/exchanges/order"
|
|
"github.com/thrasher-corp/gocryptotrader/exchanges/orderbook"
|
|
"github.com/thrasher-corp/gocryptotrader/exchanges/stream"
|
|
"github.com/thrasher-corp/gocryptotrader/exchanges/subscription"
|
|
"github.com/thrasher-corp/gocryptotrader/exchanges/ticker"
|
|
"github.com/thrasher-corp/gocryptotrader/exchanges/trade"
|
|
)
|
|
|
|
const (
|
|
coinbaseproWebsocketURL = "wss://ws-feed.pro.coinbase.com"
|
|
)
|
|
|
|
// WsConnect initiates a websocket connection
|
|
func (c *CoinbasePro) WsConnect() error {
|
|
if !c.Websocket.IsEnabled() || !c.IsEnabled() {
|
|
return errors.New(stream.WebsocketNotEnabled)
|
|
}
|
|
var dialer websocket.Dialer
|
|
err := c.Websocket.Conn.Dial(&dialer, http.Header{})
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
c.Websocket.Wg.Add(1)
|
|
go c.wsReadData()
|
|
return nil
|
|
}
|
|
|
|
// wsReadData receives and passes on websocket messages for processing
|
|
func (c *CoinbasePro) wsReadData() {
|
|
defer c.Websocket.Wg.Done()
|
|
|
|
for {
|
|
resp := c.Websocket.Conn.ReadMessage()
|
|
if resp.Raw == nil {
|
|
return
|
|
}
|
|
err := c.wsHandleData(resp.Raw)
|
|
if err != nil {
|
|
c.Websocket.DataHandler <- err
|
|
}
|
|
}
|
|
}
|
|
|
|
func (c *CoinbasePro) wsHandleData(respRaw []byte) error {
|
|
msgType := wsMsgType{}
|
|
err := json.Unmarshal(respRaw, &msgType)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
if msgType.Type == "subscriptions" || msgType.Type == "heartbeat" {
|
|
return nil
|
|
}
|
|
|
|
switch msgType.Type {
|
|
case "status":
|
|
var status wsStatus
|
|
err = json.Unmarshal(respRaw, &status)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
c.Websocket.DataHandler <- status
|
|
case "error":
|
|
c.Websocket.DataHandler <- errors.New(string(respRaw))
|
|
case "ticker":
|
|
wsTicker := WebsocketTicker{}
|
|
err := json.Unmarshal(respRaw, &wsTicker)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
c.Websocket.DataHandler <- &ticker.Price{
|
|
LastUpdated: wsTicker.Time,
|
|
Pair: wsTicker.ProductID,
|
|
AssetType: asset.Spot,
|
|
ExchangeName: c.Name,
|
|
Open: wsTicker.Open24H,
|
|
High: wsTicker.High24H,
|
|
Low: wsTicker.Low24H,
|
|
Last: wsTicker.Price,
|
|
Volume: wsTicker.Volume24H,
|
|
Bid: wsTicker.BestBid,
|
|
Ask: wsTicker.BestAsk,
|
|
}
|
|
|
|
case "snapshot":
|
|
var snapshot WebsocketOrderbookSnapshot
|
|
err := json.Unmarshal(respRaw, &snapshot)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
err = c.ProcessSnapshot(&snapshot)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
case "l2update":
|
|
var update WebsocketL2Update
|
|
err := json.Unmarshal(respRaw, &update)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
err = c.ProcessUpdate(&update)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
case "received", "open", "done", "change", "activate":
|
|
var wsOrder wsOrderReceived
|
|
err := json.Unmarshal(respRaw, &wsOrder)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
var oType order.Type
|
|
var oSide order.Side
|
|
var oStatus order.Status
|
|
oType, err = order.StringToOrderType(wsOrder.OrderType)
|
|
if err != nil {
|
|
c.Websocket.DataHandler <- order.ClassificationError{
|
|
Exchange: c.Name,
|
|
OrderID: wsOrder.OrderID,
|
|
Err: err,
|
|
}
|
|
}
|
|
oSide, err = order.StringToOrderSide(wsOrder.Side)
|
|
if err != nil {
|
|
c.Websocket.DataHandler <- order.ClassificationError{
|
|
Exchange: c.Name,
|
|
OrderID: wsOrder.OrderID,
|
|
Err: err,
|
|
}
|
|
}
|
|
oStatus, err = statusToStandardStatus(wsOrder.Type)
|
|
if err != nil {
|
|
c.Websocket.DataHandler <- order.ClassificationError{
|
|
Exchange: c.Name,
|
|
OrderID: wsOrder.OrderID,
|
|
Err: err,
|
|
}
|
|
}
|
|
if wsOrder.Reason == "canceled" {
|
|
oStatus = order.Cancelled
|
|
}
|
|
ts := wsOrder.Time
|
|
if wsOrder.Type == "activate" {
|
|
ts = convert.TimeFromUnixTimestampDecimal(wsOrder.Timestamp)
|
|
}
|
|
|
|
creds, err := c.GetCredentials(context.TODO())
|
|
if err != nil {
|
|
c.Websocket.DataHandler <- order.ClassificationError{
|
|
Exchange: c.Name,
|
|
OrderID: wsOrder.OrderID,
|
|
Err: err,
|
|
}
|
|
}
|
|
|
|
if wsOrder.UserID != "" {
|
|
var p currency.Pair
|
|
var a asset.Item
|
|
p, a, err = c.GetRequestFormattedPairAndAssetType(wsOrder.ProductID)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
c.Websocket.DataHandler <- &order.Detail{
|
|
HiddenOrder: wsOrder.Private,
|
|
Price: wsOrder.Price,
|
|
Amount: wsOrder.Size,
|
|
TriggerPrice: wsOrder.StopPrice,
|
|
ExecutedAmount: wsOrder.Size - wsOrder.RemainingSize,
|
|
RemainingAmount: wsOrder.RemainingSize,
|
|
Fee: wsOrder.TakerFeeRate,
|
|
Exchange: c.Name,
|
|
OrderID: wsOrder.OrderID,
|
|
AccountID: wsOrder.ProfileID,
|
|
ClientID: creds.ClientID,
|
|
Type: oType,
|
|
Side: oSide,
|
|
Status: oStatus,
|
|
AssetType: a,
|
|
Date: ts,
|
|
Pair: p,
|
|
}
|
|
}
|
|
case "match", "last_match":
|
|
var wsOrder wsOrderReceived
|
|
err := json.Unmarshal(respRaw, &wsOrder)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
oSide, err := order.StringToOrderSide(wsOrder.Side)
|
|
if err != nil {
|
|
c.Websocket.DataHandler <- order.ClassificationError{
|
|
Exchange: c.Name,
|
|
Err: err,
|
|
}
|
|
}
|
|
var p currency.Pair
|
|
var a asset.Item
|
|
p, a, err = c.GetRequestFormattedPairAndAssetType(wsOrder.ProductID)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
if wsOrder.UserID != "" {
|
|
c.Websocket.DataHandler <- &order.Detail{
|
|
OrderID: wsOrder.OrderID,
|
|
Pair: p,
|
|
AssetType: a,
|
|
Trades: []order.TradeHistory{
|
|
{
|
|
Price: wsOrder.Price,
|
|
Amount: wsOrder.Size,
|
|
Exchange: c.Name,
|
|
TID: strconv.FormatInt(wsOrder.TradeID, 10),
|
|
Side: oSide,
|
|
Timestamp: wsOrder.Time,
|
|
IsMaker: wsOrder.TakerUserID == "",
|
|
},
|
|
},
|
|
}
|
|
} else {
|
|
if !c.IsSaveTradeDataEnabled() {
|
|
return nil
|
|
}
|
|
return trade.AddTradesToBuffer(c.Name, trade.Data{
|
|
Timestamp: wsOrder.Time,
|
|
Exchange: c.Name,
|
|
CurrencyPair: p,
|
|
AssetType: a,
|
|
Price: wsOrder.Price,
|
|
Amount: wsOrder.Size,
|
|
Side: oSide,
|
|
TID: strconv.FormatInt(wsOrder.TradeID, 10),
|
|
})
|
|
}
|
|
default:
|
|
c.Websocket.DataHandler <- stream.UnhandledMessageWarning{Message: c.Name + stream.UnhandledMessage + string(respRaw)}
|
|
return nil
|
|
}
|
|
return nil
|
|
}
|
|
|
|
func statusToStandardStatus(stat string) (order.Status, error) {
|
|
switch stat {
|
|
case "received":
|
|
return order.New, nil
|
|
case "open":
|
|
return order.Active, nil
|
|
case "done":
|
|
return order.Filled, nil
|
|
case "match":
|
|
return order.PartiallyFilled, nil
|
|
case "change", "activate":
|
|
return order.Active, nil
|
|
default:
|
|
return order.UnknownStatus, fmt.Errorf("%s not recognised as status type", stat)
|
|
}
|
|
}
|
|
|
|
// ProcessSnapshot processes the initial orderbook snap shot
|
|
func (c *CoinbasePro) ProcessSnapshot(snapshot *WebsocketOrderbookSnapshot) error {
|
|
pair, err := currency.NewPairFromString(snapshot.ProductID)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
base := orderbook.Base{
|
|
Pair: pair,
|
|
Bids: make(orderbook.Items, len(snapshot.Bids)),
|
|
Asks: make(orderbook.Items, len(snapshot.Asks)),
|
|
}
|
|
|
|
for i := range snapshot.Bids {
|
|
var price float64
|
|
price, err = strconv.ParseFloat(snapshot.Bids[i][0], 64)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
var amount float64
|
|
amount, err = strconv.ParseFloat(snapshot.Bids[i][1], 64)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
base.Bids[i] = orderbook.Item{Price: price, Amount: amount}
|
|
}
|
|
|
|
for i := range snapshot.Asks {
|
|
var price float64
|
|
price, err = strconv.ParseFloat(snapshot.Asks[i][0], 64)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
var amount float64
|
|
amount, err = strconv.ParseFloat(snapshot.Asks[i][1], 64)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
base.Asks[i] = orderbook.Item{Price: price, Amount: amount}
|
|
}
|
|
|
|
base.Asset = asset.Spot
|
|
base.Pair = pair
|
|
base.Exchange = c.Name
|
|
base.VerifyOrderbook = c.CanVerifyOrderbook
|
|
base.LastUpdated = snapshot.Time
|
|
return c.Websocket.Orderbook.LoadSnapshot(&base)
|
|
}
|
|
|
|
// ProcessUpdate updates the orderbook local cache
|
|
func (c *CoinbasePro) ProcessUpdate(update *WebsocketL2Update) error {
|
|
if len(update.Changes) == 0 {
|
|
return errors.New("no data in websocket update")
|
|
}
|
|
|
|
p, err := currency.NewPairFromString(update.ProductID)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
asks := make(orderbook.Items, 0, len(update.Changes))
|
|
bids := make(orderbook.Items, 0, len(update.Changes))
|
|
|
|
for i := range update.Changes {
|
|
price, err := strconv.ParseFloat(update.Changes[i][1], 64)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
volume, err := strconv.ParseFloat(update.Changes[i][2], 64)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
if update.Changes[i][0] == order.Buy.Lower() {
|
|
bids = append(bids, orderbook.Item{Price: price, Amount: volume})
|
|
} else {
|
|
asks = append(asks, orderbook.Item{Price: price, Amount: volume})
|
|
}
|
|
}
|
|
|
|
return c.Websocket.Orderbook.Update(&orderbook.Update{
|
|
Bids: bids,
|
|
Asks: asks,
|
|
Pair: p,
|
|
UpdateTime: update.Time,
|
|
Asset: asset.Spot,
|
|
})
|
|
}
|
|
|
|
// GenerateDefaultSubscriptions Adds default subscriptions to websocket to be handled by ManageSubscriptions()
|
|
func (c *CoinbasePro) GenerateDefaultSubscriptions() ([]subscription.Subscription, error) {
|
|
var channels = []string{"heartbeat",
|
|
"level2_batch", /*Other orderbook feeds require authentication. This is batched in 50ms lots.*/
|
|
"ticker",
|
|
"user",
|
|
"matches"}
|
|
enabledPairs, err := c.GetEnabledPairs(asset.Spot)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
var subscriptions []subscription.Subscription
|
|
for i := range channels {
|
|
if (channels[i] == "user" || channels[i] == "full") &&
|
|
!c.IsWebsocketAuthenticationSupported() {
|
|
continue
|
|
}
|
|
for j := range enabledPairs {
|
|
fPair, err := c.FormatExchangeCurrency(enabledPairs[j],
|
|
asset.Spot)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
subscriptions = append(subscriptions, subscription.Subscription{
|
|
Channel: channels[i],
|
|
Pair: fPair,
|
|
Asset: asset.Spot,
|
|
})
|
|
}
|
|
}
|
|
return subscriptions, nil
|
|
}
|
|
|
|
// Subscribe sends a websocket message to receive data from the channel
|
|
func (c *CoinbasePro) Subscribe(channelsToSubscribe []subscription.Subscription) error {
|
|
var creds *account.Credentials
|
|
var err error
|
|
if c.IsWebsocketAuthenticationSupported() {
|
|
creds, err = c.GetCredentials(context.TODO())
|
|
if err != nil {
|
|
return err
|
|
}
|
|
}
|
|
|
|
subscribe := WebsocketSubscribe{
|
|
Type: "subscribe",
|
|
}
|
|
productIDs := make([]string, 0, len(channelsToSubscribe))
|
|
for i := range channelsToSubscribe {
|
|
p := channelsToSubscribe[i].Pair.String()
|
|
if p != "" && !common.StringDataCompare(productIDs, p) {
|
|
// get all unique productIDs in advance as we generate by channels
|
|
productIDs = append(productIDs, p)
|
|
}
|
|
}
|
|
subscriptions:
|
|
for i := range channelsToSubscribe {
|
|
for j := range subscribe.Channels {
|
|
if subscribe.Channels[j].Name == channelsToSubscribe[i].Channel {
|
|
continue subscriptions
|
|
}
|
|
}
|
|
|
|
subChan := WsChannels{
|
|
Name: channelsToSubscribe[i].Channel,
|
|
ProductIDs: productIDs,
|
|
}
|
|
if (channelsToSubscribe[i].Channel == "user" || channelsToSubscribe[i].Channel == "full") &&
|
|
creds != nil &&
|
|
subscribe.Signature == "" {
|
|
n := strconv.FormatInt(time.Now().Unix(), 10)
|
|
message := n + http.MethodGet + "/users/self/verify"
|
|
var hmac []byte
|
|
hmac, err = crypto.GetHMAC(crypto.HashSHA256,
|
|
[]byte(message),
|
|
[]byte(creds.Secret))
|
|
if err != nil {
|
|
return err
|
|
}
|
|
subscribe.Signature = crypto.Base64Encode(hmac)
|
|
subscribe.Key = creds.Key
|
|
subscribe.Passphrase = creds.ClientID
|
|
subscribe.Timestamp = n
|
|
}
|
|
subscribe.Channels = append(subscribe.Channels, subChan)
|
|
}
|
|
err = c.Websocket.Conn.SendJSONMessage(subscribe)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
c.Websocket.AddSuccessfulSubscriptions(channelsToSubscribe...)
|
|
return nil
|
|
}
|
|
|
|
// Unsubscribe sends a websocket message to stop receiving data from the channel
|
|
func (c *CoinbasePro) Unsubscribe(channelsToUnsubscribe []subscription.Subscription) error {
|
|
unsubscribe := WebsocketSubscribe{
|
|
Type: "unsubscribe",
|
|
}
|
|
productIDs := make([]string, 0, len(channelsToUnsubscribe))
|
|
for i := range channelsToUnsubscribe {
|
|
p := channelsToUnsubscribe[i].Pair.String()
|
|
if p != "" && !common.StringDataCompare(productIDs, p) {
|
|
// get all unique productIDs in advance as we generate by channels
|
|
productIDs = append(productIDs, p)
|
|
}
|
|
}
|
|
|
|
unsubscriptions:
|
|
for i := range channelsToUnsubscribe {
|
|
for j := range unsubscribe.Channels {
|
|
if unsubscribe.Channels[j].Name == channelsToUnsubscribe[i].Channel {
|
|
continue unsubscriptions
|
|
}
|
|
}
|
|
|
|
unsubscribe.Channels = append(unsubscribe.Channels, WsChannels{
|
|
Name: channelsToUnsubscribe[i].Channel,
|
|
ProductIDs: productIDs,
|
|
})
|
|
}
|
|
err := c.Websocket.Conn.SendJSONMessage(unsubscribe)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
c.Websocket.RemoveSubscriptions(channelsToUnsubscribe...)
|
|
return nil
|
|
}
|