-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathtrade.go
242 lines (205 loc) · 7.05 KB
/
trade.go
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
package wexapi
import (
"bytes"
"crypto/hmac"
"crypto/sha512"
"encoding/hex"
"encoding/json"
"io/ioutil"
"net/http"
"net/url"
"strconv"
"github.com/pkg/errors"
"github.com/shopspring/decimal"
)
const (
tradeAPIEndpoint = "https://wex.nz/tapi"
// OrderInfo available status
OrderInfoStatusActive = iota
OrderInfoStatusExecutedOrder
OrderInfoStatusCancelled
OrderInfoStatusCancelledPartiallyExecuted
)
// Rights is a privileges of the current API key.
type Rights struct {
Info uint64 `json:"info"`
Trade uint64 `json:"trade"`
Withdraw uint64 `json:"withdraw"`
}
// Funds is a account balance available for trading
type Funds map[string]decimal.Decimal
// UserInfo is an information about the user’s current balance.
type UserInfo struct {
Funds Funds `json:"funds"`
Rights Rights `json:"rights"`
TransactionCount uint64 `json:"transaction_count"`
OpenOrders uint64 `json:"open_orders"`
ServerTime unixTimestamp `json:"server_time"`
}
// GetInfo eturns information about the user’s current
// balance, API-key privileges, the number of open orders
// and Server Time.
// To use this method you need a privilege of the key info.
func (cli *Client) GetInfo() (UserInfo, error) {
userInfo := UserInfo{}
err := cli.tradeRequest(&userInfo, "getInfo")
return userInfo, err
}
// UserTrade holds data about trade.
type UserTrade struct {
Received decimal.Decimal `json:"received"`
Remains decimal.Decimal `json:"remains"`
OrderID uint64 `json:"order_id"`
Funds Funds `json:"funds"`
}
// Trade is the basic method that can be used for
// creating orders and trading on the exchange.
// To use this method you need a privilege of the key info.
func (cli *Client) Trade(pair, tradeType string, rate, amount decimal.Decimal) (UserTrade, error) {
userTrade := UserTrade{}
params := []param{
param{key: "pair", value: pair},
param{key: "type", value: tradeType},
param{key: "rate", value: rate.String()},
param{key: "amount", value: amount.String()},
}
err := cli.tradeRequest(&userTrade, "Trade", params...)
return userTrade, err
}
// TradeOrder holds information about user trade orders.
type TradeOrder struct {
ID uint64
Pair string `json:"pair"`
Type string `json:"type"`
StartAmount decimal.Decimal `json:"start_amount"`
Amount decimal.Decimal `json:"amount"`
Rate decimal.Decimal `json:"rate"`
TimestampCreated unixTimestamp `json:"timestamp_created"`
}
// TradeOrders holds list of trade orders.
type TradeOrders []TradeOrder
// UnmarshalJSON unmarshall map[string]TradeOrder
// format into the slice.
func (to *TradeOrders) UnmarshalJSON(data []byte) error {
idRaw := make(map[string]json.RawMessage)
if err := json.Unmarshal(data, &idRaw); err != nil {
return errors.Wrap(err, "unmarshal to id raw")
}
for idString, data := range idRaw {
id, err := strconv.ParseUint(idString, 10, 64)
if err != nil {
return errors.Wrapf(err, "parse id %s", idString)
}
trade := TradeOrder{
ID: id,
}
if err := json.Unmarshal(data, &trade); err != nil {
return errors.Wrapf(err, "unmarshal trade order %s", idString)
}
*to = append(*to, trade)
}
return nil
}
// ActiveOrders returns the list of your active orders.
// To use this method you need a privilege of the info key.
func (cli *Client) ActiveOrders(pair string) (TradeOrders, error) {
tradeOrders := TradeOrders{}
err := cli.tradeRequest(&tradeOrders, "ActiveOrders", param{key: "pair", value: pair})
return tradeOrders, err
}
// OrderInfo holds data about order
type OrderInfo struct {
ID uint64
Pair string `json:"pair"`
Type string `json:"type"`
StartAmount decimal.Decimal `json:"start_amount"`
Amount decimal.Decimal `json:"amount"`
Rate decimal.Decimal `json:"rate"`
TimestampCreated unixTimestamp `json:"timestamp_created"`
Status uint8
}
// OrderInfo returns the information on particular order.
// To use this method you need a privilege of the info key.
func (cli *Client) OrderInfo(orderID uint64) (OrderInfo, error) {
ordersInfo := make(map[string]OrderInfo)
orderIDString := strconv.FormatUint(orderID, 10)
err := cli.tradeRequest(&ordersInfo, "OrderInfo", param{key: "order_id", value: orderIDString})
orderInfo := ordersInfo[orderIDString]
orderInfo.ID = orderID
return ordersInfo[orderIDString], err
}
// CancelOrder holds data about cancelled order.
type CancelOrder struct {
OrderID uint64 `json:"order_id"`
}
// CancelOrder returns the information on particular order.
// To use this method you need a privilege of the info key.
func (cli *Client) CancelOrder(orderID uint64) (CancelOrder, error) {
cancelOrder := CancelOrder{}
err := cli.tradeRequest(&cancelOrder, "CancelOrder", param{key: "order_id", value: strconv.FormatUint(orderID, 10)})
return cancelOrder, err
}
// Withdraw holds data about withdraw.
type Withdraw struct {
TradeID uint64 `json:"tId"`
AmountSent decimal.Decimal `json:"amountSent"`
Funds Funds `json:"funds"`
}
// WithdrawCoin is designed for cryptocurrency withdrawals.
// To use this method you need a privilege of the info key.
func (cli *Client) WithdrawCoin(currency, address string, amount decimal.Decimal) (Withdraw, error) {
withdraw := Withdraw{}
params := []param{
param{key: "coinName", value: currency},
param{key: "address", value: address},
param{key: "amount", value: amount.String()},
}
err := cli.tradeRequest(&withdraw, "WithdrawCoin", params...)
return withdraw, err
}
func (cli *Client) tradeRequest(result interface{}, method string, params ...param) error {
nonce, err := cli.nonce()
if err != nil {
return errors.Wrap(err, "nonce")
}
data := url.Values{
"method": []string{method},
"nonce": []string{nonce},
}
for _, param := range params {
data.Add(param.key, param.value)
}
buf := bytes.NewBufferString(data.Encode())
req, err := http.NewRequest("POST", tradeAPIEndpoint, buf)
if err != nil {
return errors.Wrap(err, "request build")
}
sign := hmac.New(sha512.New, []byte(cli.secret))
if _, err := sign.Write(buf.Bytes()); err != nil {
return errors.Wrap(err, "hmac write signature")
}
req.Header.Set("Key", cli.key)
req.Header.Set("Sign", hex.EncodeToString(sign.Sum(nil)))
req.Header.Add("Content-Type", "application/x-www-form-urlencoded")
resp, err := cli.httpClient.Do(req)
if err != nil {
return errors.Wrap(err, "do request")
}
defer resp.Body.Close()
if resp.StatusCode != http.StatusOK {
return errors.Errorf("server respond with status code %d", resp.StatusCode)
}
body, err := ioutil.ReadAll(resp.Body)
if err != nil {
return errors.Wrap(err, "read response body")
}
br := baseResponse{}
if err := json.Unmarshal(body, &br); err != nil {
return errors.Wrap(err, "unmarshal to base response")
}
if !br.Success && br.Error != nil {
return errors.Errorf("server respond with error: %s", *br.Error)
}
err = json.Unmarshal(br.Return, result)
return errors.Wrap(err, "unmarshal to result")
}