forked from stellar-deprecated/kelp
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathmaxPriceFilter.go
84 lines (70 loc) · 2.55 KB
/
maxPriceFilter.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
package plugins
import (
"fmt"
"log"
"strconv"
hProtocol "github.com/stellar/go/protocols/horizon"
"github.com/stellar/go/txnbuild"
"github.com/stellar/kelp/support/utils"
)
// MaxPriceFilterConfig ensures that any one constraint that is hit will result in deleting all offers and pausing until limits are no longer constrained
type MaxPriceFilterConfig struct {
MaxPrice *float64
}
type maxPriceFilter struct {
name string
config *MaxPriceFilterConfig
baseAsset hProtocol.Asset
quoteAsset hProtocol.Asset
}
// MakeFilterMaxPrice makes a submit filter that limits orders placed based on the price
func MakeFilterMaxPrice(baseAsset hProtocol.Asset, quoteAsset hProtocol.Asset, config *MaxPriceFilterConfig) (SubmitFilter, error) {
return &maxPriceFilter{
name: "maxPriceFilter",
config: config,
baseAsset: baseAsset,
quoteAsset: quoteAsset,
}, nil
}
var _ SubmitFilter = &maxPriceFilter{}
// Validate ensures validity
func (c *MaxPriceFilterConfig) Validate() error {
if c.MaxPrice == nil {
return fmt.Errorf("needs a maxPrice config value")
}
return nil
}
// String is the stringer method
func (c *MaxPriceFilterConfig) String() string {
return fmt.Sprintf("MaxPriceFilterConfig[MaxPrice=%s]", utils.CheckedFloatPtr(c.MaxPrice))
}
func (f *maxPriceFilter) Apply(ops []txnbuild.Operation, sellingOffers []hProtocol.Offer, buyingOffers []hProtocol.Offer) ([]txnbuild.Operation, error) {
ops, e := filterOps(f.name, f.baseAsset, f.quoteAsset, sellingOffers, buyingOffers, ops, f.maxPriceFilterFn)
if e != nil {
return nil, fmt.Errorf("could not apply filter: %s", e)
}
return ops, nil
}
func (f *maxPriceFilter) maxPriceFilterFn(op *txnbuild.ManageSellOffer) (*txnbuild.ManageSellOffer, error) {
isSell, e := utils.IsSelling(f.baseAsset, f.quoteAsset, op.Selling, op.Buying)
if e != nil {
return nil, fmt.Errorf("error when running the isSelling check for offer '%+v': %s", *op, e)
}
sellPrice, e := strconv.ParseFloat(op.Price, 64)
if e != nil {
return nil, fmt.Errorf("could not convert price (%s) to float: %s", op.Price, e)
}
// reorient price to be in the context of the bot's base and quote asset, in quote units
price := sellPrice
if !isSell {
// invert price for buy side
price = 1 / sellPrice
}
// keep only those ops that meet the comparison mode using the value from the price feed as the threshold
opRet := op
if price > *f.config.MaxPrice {
opRet = nil
}
log.Printf("maxPriceFilter: isSell=%v, price=%.10f, thresholdMaxPrice=%.10f, keep=%v", isSell, price, *f.config.MaxPrice, opRet != nil)
return opRet, nil
}