bybit.go.api/bybit_api_client.go

218 lines
5.0 KiB
Go
Raw Normal View History

2023-11-03 16:29:17 +02:00
package bybit_connector
2023-10-30 20:26:54 +02:00
import (
"bytes"
"context"
"crypto/hmac"
"crypto/sha256"
2023-11-05 18:33:56 +02:00
"encoding/hex"
2023-10-30 20:26:54 +02:00
"encoding/json"
"fmt"
2023-11-05 18:33:56 +02:00
"github.com/wuhewuhe/bybit.go.api/handlers"
2023-10-30 20:26:54 +02:00
"io"
"log"
"net/http"
"os"
2023-11-05 18:33:56 +02:00
"strconv"
2023-10-30 20:26:54 +02:00
"time"
)
2023-11-05 18:33:56 +02:00
type ServerResponse struct {
RetCode int `json:"retCode"`
RetMsg string `json:"retMsg"`
Result interface{} `json:"result"`
RetExtInfo struct{} `json:"retExtInfo"`
Time int64 `json:"time"`
}
2023-10-30 20:26:54 +02:00
// TimeInForceType define time in force type of order
type TimeInForceType string
// Client define API client
type Client struct {
APIKey string
2023-11-05 18:33:56 +02:00
APISecret string
2023-10-30 20:26:54 +02:00
BaseURL string
HTTPClient *http.Client
Debug bool
Logger *log.Logger
do doFunc
}
type doFunc func(req *http.Request) (*http.Response, error)
2023-11-03 16:29:17 +02:00
type ClientOption func(*Client)
func WithDebug(debug bool) ClientOption {
return func(c *Client) {
c.Debug = debug
}
}
2023-11-05 18:33:56 +02:00
// WithBaseURL is a client option to set the base URL of the Bybit HTTP client.
2023-11-03 16:29:17 +02:00
func WithBaseURL(baseURL string) ClientOption {
return func(c *Client) {
c.BaseURL = baseURL
}
}
2023-10-30 20:26:54 +02:00
func PrettyPrint(i interface{}) string {
s, _ := json.MarshalIndent(i, "", "\t")
return string(s)
}
func (c *Client) debug(format string, v ...interface{}) {
if c.Debug {
c.Logger.Printf(format, v...)
}
}
2023-11-03 16:29:17 +02:00
// NewBybitHttpClient NewClient Create client function for initialising new Bybit client
2023-11-05 18:33:56 +02:00
func NewBybitHttpClient(apiKey string, APISecret string, options ...ClientOption) *Client {
2023-11-03 16:29:17 +02:00
c := &Client{
2023-10-30 20:26:54 +02:00
APIKey: apiKey,
2023-11-05 18:33:56 +02:00
APISecret: APISecret,
BaseURL: MAINNET,
2023-10-30 20:26:54 +02:00
HTTPClient: http.DefaultClient,
Logger: log.New(os.Stderr, Name, log.LstdFlags),
}
2023-11-03 16:29:17 +02:00
// Apply the provided options
for _, opt := range options {
opt(c)
}
return c
2023-10-30 20:26:54 +02:00
}
func (c *Client) parseRequest(r *request, opts ...RequestOption) (err error) {
// set request options from user
for _, opt := range opts {
opt(r)
}
err = r.validate()
if err != nil {
return err
}
fullURL := fmt.Sprintf("%s%s", c.BaseURL, r.endpoint)
queryString := r.query.Encode()
header := http.Header{}
2023-11-05 18:33:56 +02:00
body := &bytes.Buffer{}
if r.params != nil {
body = bytes.NewBuffer(r.params)
}
2023-10-30 20:26:54 +02:00
if r.header != nil {
header = r.header.Clone()
}
header.Set("User-Agent", fmt.Sprintf("%s/%s", Name, Version))
2023-11-05 18:33:56 +02:00
2023-10-30 20:26:54 +02:00
if r.secType == secTypeSigned {
2023-11-05 18:33:56 +02:00
now := time.Now()
unixNano := now.UnixNano()
timeStamp := unixNano / 1000000
2023-10-30 20:26:54 +02:00
header.Set(signTypeKey, "2")
header.Set(apiRequestKey, c.APIKey)
2023-11-05 18:33:56 +02:00
header.Set(timestampKey, strconv.FormatInt(timeStamp, 10))
2023-10-30 20:26:54 +02:00
if r.recvWindow == "" {
2023-11-05 18:33:56 +02:00
r.recvWindow = "5000"
2023-10-30 20:26:54 +02:00
}
2023-11-05 18:33:56 +02:00
header.Set(recvWindowKey, r.recvWindow)
2023-10-30 20:26:54 +02:00
2023-11-05 18:33:56 +02:00
var signatureBase []byte
if r.method == "POST" {
header.Set("Content-Type", "application/json")
signatureBase = []byte(strconv.FormatInt(timeStamp, 10) + c.APIKey + r.recvWindow + string(r.params[:]))
} else {
signatureBase = []byte(strconv.FormatInt(timeStamp, 10) + c.APIKey + r.recvWindow + queryString)
2023-10-30 20:26:54 +02:00
}
2023-11-05 18:33:56 +02:00
hmac256 := hmac.New(sha256.New, []byte(c.APISecret))
hmac256.Write(signatureBase)
signature := hex.EncodeToString(hmac256.Sum(nil))
header.Set(signatureKey, signature)
2023-10-30 20:26:54 +02:00
}
if queryString != "" {
fullURL = fmt.Sprintf("%s?%s", fullURL, queryString)
}
2023-11-05 18:33:56 +02:00
c.debug("full url: %s, body: %s", fullURL, body)
2023-10-30 20:26:54 +02:00
r.fullURL = fullURL
r.body = body
2023-11-05 18:33:56 +02:00
r.header = header
2023-10-30 20:26:54 +02:00
return nil
}
func (c *Client) callAPI(ctx context.Context, r *request, opts ...RequestOption) (data []byte, err error) {
err = c.parseRequest(r, opts...)
req, err := http.NewRequest(r.method, r.fullURL, r.body)
if err != nil {
return []byte{}, err
}
req = req.WithContext(ctx)
req.Header = r.header
2023-11-03 16:29:17 +02:00
c.debug("request: %#v", req)
2023-10-30 20:26:54 +02:00
f := c.do
if f == nil {
f = c.HTTPClient.Do
}
res, err := f(req)
if err != nil {
return []byte{}, err
}
data, err = io.ReadAll(res.Body)
if err != nil {
return []byte{}, err
}
defer func() {
cerr := res.Body.Close()
// Only overwrite the returned error if the original error was nil and an
// error occurred while closing the body.
if err == nil && cerr != nil {
err = cerr
}
}()
2023-11-03 16:29:17 +02:00
c.debug("response: %#v", res)
2023-10-30 20:26:54 +02:00
c.debug("response body: %s", string(data))
2023-11-03 16:29:17 +02:00
c.debug("response status code: %d", res.StatusCode)
2023-10-30 20:26:54 +02:00
if res.StatusCode >= http.StatusBadRequest {
2023-11-03 16:29:17 +02:00
var (
apiErr = new(handlers.APIError)
)
2023-10-30 20:26:54 +02:00
e := json.Unmarshal(data, apiErr)
if e != nil {
2023-11-03 16:29:17 +02:00
c.debug("failed to unmarshal json: %s", e)
2023-10-30 20:26:54 +02:00
}
return nil, apiErr
}
return data, nil
}
// NewServerTimeService Market Endpoints
func (c *Client) NewServerTimeService() *ServerTime {
return &ServerTime{c: c}
}
2023-11-05 18:33:56 +02:00
// NewMarketKlineService Market Endpoints
func (c *Client) NewMarketKlineService(klineType, category, symbol, interval string) *Klines {
return &Klines{
c: c,
category: category,
symbol: symbol,
interval: interval,
klineType: klineType,
}
}
// NewPlaceOrderService Trade Endpoints
func (c *Client) NewPlaceOrderService(category, symbol, side, orderType, qty string) *Order {
return &Order{
c: c,
category: category,
symbol: symbol,
side: side,
orderType: orderType,
qty: qty,
}
}