This repository has been archived by the owner on Jul 24, 2024. It is now read-only.
forked from andrewstuart/go-robinhood
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathcrypto_order.go
117 lines (98 loc) · 3.29 KB
/
crypto_order.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
package robinhood
import (
"bytes"
"context"
"fmt"
"math"
"github.com/google/uuid"
"github.com/pkg/errors"
"encoding/json"
"net/http"
)
// CryptoOrder is the payload to create a crypto currency order
type CryptoOrder struct {
AccountID string `json:"account_id,omitempty"`
CurrencyPairID string `json:"currency_pair_id,omitempty"`
Price float64 `json:"price,omitempty"`
RefID string `json:"ref_id,omitempty"`
Side string `json:"side,omitempty"`
TimeInForce string `json:"time_in_force,omitempty"`
Quantity float64 `json:"quantity,omitempty"`
Type string `json:"type,omitempty"`
}
// CryptoOrderOutput holds the response from api
type CryptoOrderOutput struct {
Meta
Account string `json:"account"`
AveragePrice float64 `json:"average_price,string"`
CancelURL string `json:"cancel"`
CreatedAt string `json:"created_at"`
CumulativeQuantity string `json:"cumulative_quantity"`
CurrencyPairID string `json:"currency_pair_id"`
Executions []interface{} `json:"executions"`
ID string `json:"id"`
LastTransactionAt string `json:"last_transaction_at"`
Price float64 `json:"price,string"`
Quantity string `json:"quantity"`
RejectReason string `json:"reject_reason"`
Side string `json:"side"`
State string `json:"state"`
StopPrice float64 `json:"stop_price,string"`
TimeInForce string `json:"time_in_force"`
Type string `json:"type"`
client *Client
}
// CryptoOrderOpts encapsulates differences between order types
type CryptoOrderOpts struct {
Side OrderSide
Type OrderType
AmountInDollars float64
Quantity float64
Price float64
TimeInForce TimeInForce
ExtendedHours bool
Stop, Force bool
}
// CryptoOrder will actually place the order
func (c *Client) CryptoOrder(ctx context.Context, cryptoPair CryptoCurrencyPair, o CryptoOrderOpts) (*CryptoOrderOutput, error) {
var quantity = math.Round(o.AmountInDollars / o.Price)
a := CryptoOrder{
AccountID: c.CryptoAccount.ID,
CurrencyPairID: cryptoPair.ID,
Quantity: quantity,
Price: o.Price,
RefID: uuid.New().String(),
Side: o.Side.String(),
TimeInForce: o.TimeInForce.String(),
Type: o.Type.String(),
}
payload, err := json.Marshal(a)
if err != nil {
return nil, err
}
post, err := http.NewRequest("POST", EPCryptoOrders, bytes.NewReader(payload))
if err != nil {
return nil, fmt.Errorf("could not create Crypto http.Request: %w", err)
}
post.Header.Add("Content-Type", "application/json")
var out CryptoOrderOutput
err = c.DoAndDecode(ctx, post, &out)
out.client = c
return &out, err
}
// Cancel will cancel the order.
func (o CryptoOrderOutput) Cancel(ctx context.Context) error {
post, err := http.NewRequest("POST", o.CancelURL, nil)
if err != nil {
return err
}
var output CryptoOrderOutput
err = o.client.DoAndDecode(ctx, post, &output)
if err != nil {
return errors.Wrap(err, "could not decode response")
}
if output.RejectReason != "" {
return errors.New(output.RejectReason)
}
return nil
}