/
spam_prevention.go
103 lines (91 loc) · 2.84 KB
/
spam_prevention.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
package ante
import (
"sync"
sdk "github.com/cosmos/cosmos-sdk/types"
sdkerrors "github.com/cosmos/cosmos-sdk/types/errors"
oracletypes "github.com/umee-network/umee/v3/x/oracle/types"
)
// SpamPreventionDecorator defines a custom Umee AnteHandler decorator that is
// responsible for preventing oracle message spam. Specifically, it prohibits
// oracle feeders from submitting multiple oracle messages in a single block.
type SpamPreventionDecorator struct {
oracleKeeper OracleKeeper
oraclePrevoteMap map[string]int64
oracleVoteMap map[string]int64
mu sync.Mutex
}
func NewSpamPreventionDecorator(oracleKeeper OracleKeeper) *SpamPreventionDecorator {
return &SpamPreventionDecorator{
oracleKeeper: oracleKeeper,
oraclePrevoteMap: make(map[string]int64),
oracleVoteMap: make(map[string]int64),
mu: sync.Mutex{},
}
}
func (spd *SpamPreventionDecorator) AnteHandle(
ctx sdk.Context,
tx sdk.Tx,
simulate bool,
next sdk.AnteHandler,
) (newCtx sdk.Context, err error) {
if ctx.IsReCheckTx() {
return next(ctx, tx, simulate)
}
if ctx.IsCheckTx() && !simulate {
if err := spd.CheckOracleSpam(ctx, tx.GetMsgs()); err != nil {
return ctx, err
}
}
return next(ctx, tx, simulate)
}
// CheckOracleSpam performs the check of whether or not we've seen an oracle
// message from an oracle feeder in the current block or not. If we have, we
// return an error which prohibits the transaction from being processed.
func (spd *SpamPreventionDecorator) CheckOracleSpam(ctx sdk.Context, msgs []sdk.Msg) error {
spd.mu.Lock()
defer spd.mu.Unlock()
curHeight := ctx.BlockHeight()
for _, msg := range msgs {
var err error
switch msg := msg.(type) {
case *oracletypes.MsgAggregateExchangeRatePrevote:
err = spd.validate(ctx, msg.Feeder, msg.Validator, spd.oraclePrevoteMap, curHeight, "pre-vote")
case *oracletypes.MsgAggregateExchangeRateVote:
err = spd.validate(ctx, msg.Feeder, msg.Validator, spd.oracleVoteMap, curHeight, "vote")
default:
// non oracle msg: stop validation!
// NOTE: only tx which contains only oracle Msgs are considered oracle-prioritized
return nil
}
if err != nil {
return err
}
}
return nil
}
func (spd *SpamPreventionDecorator) validate(
ctx sdk.Context,
feeder,
validator string,
cache map[string]int64,
curHeight int64,
txType string,
) error {
feederAddr, err := sdk.AccAddressFromBech32(feeder)
if err != nil {
return err
}
valAddr, err := sdk.ValAddressFromBech32(validator)
if err != nil {
return err
}
if err = spd.oracleKeeper.ValidateFeeder(ctx, feederAddr, valAddr); err != nil {
return err
}
if lastSubmitted, ok := cache[validator]; ok && lastSubmitted == curHeight {
return sdkerrors.ErrInvalidRequest.Wrapf(
"validator has already submitted a %s message at the current height", txType)
}
cache[validator] = curHeight
return nil
}