lotus/chain/store/events_called.go

310 lines
7.0 KiB
Go
Raw Normal View History

2019-09-04 16:09:08 +00:00
package store
import (
"github.com/ipfs/go-cid"
"math"
"sync"
"github.com/filecoin-project/go-lotus/chain/address"
"github.com/filecoin-project/go-lotus/chain/types"
)
// continous
// with reverts
// no timout
// `ts` is the tipset, in which the `msg` is included.
// `curH`-`ts.Height` = `confidence`
2019-09-04 19:41:34 +00:00
type CalledHandler func(msg *types.Message, ts *types.TipSet, curH uint64) (bool, error)
2019-09-04 16:09:08 +00:00
// CheckFunc is used for atomicity guarantees. If the condition the callbacks
// wait for has already happened in tipset `ts`
//
// If `done` is true, timeout won't be triggered
// If `more` is false, no messages will be sent to CalledHandler (RevertHandler
// may still be called)
type CheckFunc func(ts *types.TipSet) (done bool, more bool, err error)
2019-09-04 16:09:08 +00:00
type callHandler struct {
confidence int
timeout uint64
2019-09-04 19:41:34 +00:00
disabled bool // TODO: GC after gcConfidence reached
2019-09-04 16:09:08 +00:00
handle CalledHandler
revert RevertHandler
}
type queuedEvent struct {
trigger uint64
h uint64
msg *types.Message
2019-09-04 18:56:06 +00:00
called bool
2019-09-04 16:09:08 +00:00
}
type calledEvents struct {
cs eventChainStore
tsc *tipSetCache
lk sync.Mutex
ctr uint64
2019-09-04 19:41:34 +00:00
triggers map[uint64]*callHandler
callTuples map[callTuple][]uint64
2019-09-04 16:09:08 +00:00
// maps block heights to events
// [triggerH][msgH][event]
2019-09-04 18:56:06 +00:00
confQueue map[uint64]map[uint64][]*queuedEvent
2019-09-04 16:09:08 +00:00
// [msgH][triggerH]
revertQueue map[uint64][]uint64
2019-09-04 19:41:34 +00:00
// [timeoutH+confidence][triggerId]{calls}
timeouts map[uint64]map[uint64]int
2019-09-04 16:09:08 +00:00
}
type callTuple struct {
actor address.Address
method uint64
}
func (e *calledEvents) headChangeCalled(rev, app []*types.TipSet) error {
for _, ts := range rev {
e.handleReverts(ts)
}
for _, ts := range app {
// called triggers
e.checkNewCalls(ts)
2019-09-04 16:09:08 +00:00
e.applyWithConfidence(ts)
2019-09-04 19:41:34 +00:00
e.applyTimeouts(ts)
2019-09-04 16:09:08 +00:00
}
return nil
}
func (e *calledEvents) handleReverts(ts *types.TipSet) {
reverts, ok := e.revertQueue[ts.Height()]
if !ok {
return // nothing to do
}
for _, triggerH := range reverts {
toRevert := e.confQueue[triggerH][ts.Height()]
for _, event := range toRevert {
2019-09-04 18:56:06 +00:00
if !event.called {
continue // event wasn't apply()-ied yet
}
2019-09-04 16:09:08 +00:00
trigger := e.triggers[event.trigger]
2019-09-04 18:56:06 +00:00
2019-09-04 16:09:08 +00:00
if err := trigger.revert(ts); err != nil {
log.Errorf("reverting chain trigger (call %s.%d() @H %d, called @ %d) failed: %s", event.msg.To, event.msg.Method, ts.Height(), triggerH, err)
}
}
2019-09-04 18:56:06 +00:00
delete(e.confQueue[triggerH], ts.Height())
2019-09-04 16:09:08 +00:00
}
2019-09-04 18:56:06 +00:00
delete(e.revertQueue, ts.Height())
2019-09-04 16:09:08 +00:00
}
func (e *calledEvents) checkNewCalls(ts *types.TipSet) {
e.messagesForTs(ts, func(msg *types.Message) {
2019-09-04 16:09:08 +00:00
// TODO: do we have to verify the receipt, or are messages on chain
// guaranteed to be successful?
ct := callTuple{
actor: msg.To,
method: msg.Method,
}
triggers, ok := e.callTuples[ct]
if !ok {
return
}
for _, tid := range triggers {
e.queueForConfidence(tid, msg, ts)
}
})
}
func (e *calledEvents) queueForConfidence(triggerId uint64, msg *types.Message, ts *types.TipSet) {
trigger := e.triggers[triggerId]
triggerH := ts.Height() + uint64(trigger.confidence)
byOrigH, ok := e.confQueue[triggerH]
if !ok {
2019-09-04 18:56:06 +00:00
byOrigH = map[uint64][]*queuedEvent{}
2019-09-04 16:09:08 +00:00
e.confQueue[triggerH] = byOrigH
}
2019-09-04 18:56:06 +00:00
byOrigH[ts.Height()] = append(byOrigH[ts.Height()], &queuedEvent{
2019-09-04 16:09:08 +00:00
trigger: triggerId,
h: ts.Height(),
msg: msg,
})
2019-09-04 18:56:06 +00:00
e.revertQueue[ts.Height()] = append(e.revertQueue[ts.Height()], triggerH) // todo: dedupe?
2019-09-04 16:09:08 +00:00
}
func (e *calledEvents) applyWithConfidence(ts *types.TipSet) {
byOrigH, ok := e.confQueue[ts.Height()]
if !ok {
return // no triggers at thin height
}
for origH, events := range byOrigH {
triggerTs, err := e.tsc.get(origH)
if err != nil {
log.Errorf("events: applyWithConfidence didn't find tipset for event; wanted %d; current %d", origH, ts.Height())
}
for _, event := range events {
trigger := e.triggers[event.trigger]
2019-09-04 19:41:34 +00:00
if trigger.disabled {
continue
}
2019-09-04 18:56:06 +00:00
2019-09-04 19:41:34 +00:00
more, err := trigger.handle(event.msg, triggerTs, ts.Height())
if err != nil {
2019-09-04 16:09:08 +00:00
log.Errorf("chain trigger (call %s.%d() @H %d, called @ %d) failed: %s", event.msg.To, event.msg.Method, origH, ts.Height(), err)
continue // don't revert failed calls
}
2019-09-04 18:56:06 +00:00
event.called = true
2019-09-04 19:41:34 +00:00
touts, ok := e.timeouts[trigger.timeout]
if ok {
touts[event.trigger]++
}
trigger.disabled = !more
2019-09-04 16:09:08 +00:00
}
}
}
2019-09-04 19:41:34 +00:00
func (e *calledEvents) applyTimeouts(ts *types.TipSet) {
triggers, ok := e.timeouts[ts.Height()]
if !ok {
return // nothing to do
}
for triggerId, calls := range triggers {
if calls > 0 {
continue // don't timeout if the method was called
}
trigger := e.triggers[triggerId]
if trigger.disabled {
continue
}
timeoutTs, err := e.tsc.get(ts.Height() - uint64(trigger.confidence))
if err != nil {
log.Errorf("events: applyTimeouts didn't find tipset for event; wanted %d; current %d", ts.Height()-uint64(trigger.confidence), ts.Height())
}
more, err := trigger.handle(nil, timeoutTs, ts.Height())
if err != nil {
log.Errorf("chain trigger (call @H %d, called @ %d) failed: %s", timeoutTs.Height(), ts.Height(), err)
continue // don't revert failed calls
}
trigger.disabled = !more // allows messages after timeout
}
}
func (e *calledEvents) messagesForTs(ts *types.TipSet, consume func(*types.Message)) {
2019-09-04 16:09:08 +00:00
seen := map[cid.Cid]struct{}{}
for _, tsb := range ts.Blocks() {
bmsgs, smsgs, err := e.cs.MessagesForBlock(tsb)
if err != nil {
log.Errorf("messagesForTs MessagesForBlock failed (ts.H=%d, Bcid:%s, B.Mcid:%s): %s", ts.Height(), tsb.Cid(), tsb.Messages, err)
// this is quite bad, but probably better than missing all the other updates
continue
2019-09-04 16:09:08 +00:00
}
for _, m := range bmsgs {
_, ok := seen[m.Cid()]
if ok {
continue
}
seen[m.Cid()] = struct{}{}
consume(m)
}
for _, m := range smsgs {
_, ok := seen[m.Message.Cid()]
if ok {
continue
}
seen[m.Message.Cid()] = struct{}{}
consume(&m.Message)
}
}
}
2019-09-04 19:41:34 +00:00
func (e *calledEvents) Called(check CheckFunc, hnd CalledHandler, rev RevertHandler, confidence int, timeout uint64, actor address.Address, method uint64) error {
2019-09-04 16:09:08 +00:00
e.lk.Lock()
defer e.lk.Unlock()
done, more, err := check(e.tsc.best())
2019-09-04 16:09:08 +00:00
if err != nil {
return err
}
if done {
timeout = math.MaxUint64
2019-09-04 16:09:08 +00:00
}
id := e.ctr
e.ctr++
2019-09-04 19:41:34 +00:00
e.triggers[id] = &callHandler{
2019-09-04 16:09:08 +00:00
confidence: confidence,
2019-09-04 19:41:34 +00:00
timeout: timeout + uint64(confidence),
2019-09-04 16:09:08 +00:00
disabled: !more,
2019-09-04 16:09:08 +00:00
handle: hnd,
revert: rev,
}
ct := callTuple{
actor: actor,
method: method,
}
e.callTuples[ct] = append(e.callTuples[ct], id)
2019-09-04 19:41:34 +00:00
if timeout != math.MaxUint64 {
if e.timeouts[timeout+uint64(confidence)] == nil {
e.timeouts[timeout+uint64(confidence)] = map[uint64]int{}
}
e.timeouts[timeout+uint64(confidence)][id] = 0
}
2019-09-04 16:09:08 +00:00
return nil
}
2019-09-04 18:56:06 +00:00
/*func (e *calledEvents) debugInfo() {
fmt.Println("vvv")
fmt.Println("@", e.tsc.best().Height())
for k, v := range e.revertQueue {
fmt.Println("revert (msgH->trigH)", k, v)
}
for triggerH, v := range e.confQueue {
for msgh, e := range v {
for _, evt := range e {
fmt.Printf("T@ %d, M@ %d, EH %d, T %d, called %t\n", triggerH, msgh, evt.h, evt.trigger, evt.called)
}
}
}
fmt.Println("^^^")
}*/