forked from sdcoffey/techan
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathrule.go
76 lines (61 loc) · 2.34 KB
/
rule.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
package techan
import "github.com/sdcoffey/big"
// Rule is an interface describing an algorithm by which a set of criteria may be satisfied
type Rule interface {
IsSatisfied(index int, record *TradingRecord) bool
}
// And returns a new rule whereby BOTH of the passed-in rules must be satisfied for the rule to be satisfied
func And(r1, r2 Rule) Rule {
return andRule{r1, r2}
}
// Or returns a new rule whereby ONE OF the passed-in rules must be satisfied for the rule to be satisfied
func Or(r1, r2 Rule) Rule {
return orRule{r1, r2}
}
type andRule struct {
r1 Rule
r2 Rule
}
func (ar andRule) IsSatisfied(index int, record *TradingRecord) bool {
return ar.r1.IsSatisfied(index, record) && ar.r2.IsSatisfied(index, record)
}
type orRule struct {
r1 Rule
r2 Rule
}
func (or orRule) IsSatisfied(index int, record *TradingRecord) bool {
return or.r1.IsSatisfied(index, record) || or.r2.IsSatisfied(index, record)
}
// OverIndicatorRule is a rule where the First Indicator must be greater than the Second Indicator to be Satisfied
type OverIndicatorRule struct {
First Indicator
Second Indicator
}
// IsSatisfied returns true when the First Indicator is greater than the Second Indicator
func (oir OverIndicatorRule) IsSatisfied(index int, record *TradingRecord) bool {
return oir.First.Calculate(index).GT(oir.Second.Calculate(index))
}
// UnderIndicatorRule is a rule where the First Indicator must be less than the Second Indicator to be Satisfied
type UnderIndicatorRule struct {
First Indicator
Second Indicator
}
// IsSatisfied returns true when the First Indicator is less than the Second Indicator
func (uir UnderIndicatorRule) IsSatisfied(index int, record *TradingRecord) bool {
return uir.First.Calculate(index).LT(uir.Second.Calculate(index))
}
type percentChangeRule struct {
indicator Indicator
percent big.Decimal
}
func (pgr percentChangeRule) IsSatisfied(index int, record *TradingRecord) bool {
return pgr.indicator.Calculate(index).Abs().GT(pgr.percent.Abs())
}
// NewPercentChangeRule returns a rule whereby the given Indicator must have changed by a given percentage to be satisfied.
// You should specify percent as a float value between -1 and 1
func NewPercentChangeRule(indicator Indicator, percent float64) Rule {
return percentChangeRule{
indicator: NewPercentChangeIndicator(indicator),
percent: big.NewDecimal(percent),
}
}