-
Notifications
You must be signed in to change notification settings - Fork 1.7k
/
models.go
187 lines (168 loc) · 5.95 KB
/
models.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
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
package functions
import (
"database/sql/driver"
"encoding/hex"
"errors"
"fmt"
"time"
"github.com/ethereum/go-ethereum/common"
)
const RequestIDLength int = 32
type RequestID [RequestIDLength]byte
type Request struct {
RequestID RequestID
ReceivedAt time.Time
RequestTxHash *common.Hash
State RequestState
ResultReadyAt *time.Time
Result []byte
ErrorType *ErrType
Error []byte
TransmittedResult []byte
TransmittedError []byte
Flags []byte
AggregationMethod *AggregationMethod
CallbackGasLimit *uint32
CoordinatorContractAddress *common.Address
OnchainMetadata []byte
ProcessingMetadata []byte
}
type RequestState int8
type AggregationMethod int8
const (
// IN_PROGRESS is the initial state of a request, set right after receiving it in an on-chain event.
IN_PROGRESS RequestState = iota
// RESULT_READY means that computation has finished executing (with either success or user error).
// OCR2 reporting includes only requests in RESULT_READY state (for Query and Observation phases).
RESULT_READY
// TIMED_OUT request has been waiting to get confirmed on chain for too long.
// It won't be included in OCR2 reporting rounds any more.
TIMED_OUT
// FINALIZED request is a part of a report produced by OCR2 and has now entered the transmission protocol
// (i.e. passed through ShouldAcceptFinalizedReport()).
FINALIZED
// CONFIRMED state indicates that we received an on-chain confirmation event
// (with or without this node's participation in an earlier OCR round).
// We can transition here at any time (full fan-in) and cannot transition out (empty fan-out).
// This is a desired and expected final state for every request.
CONFIRMED
)
/*
* +-----------+
* +----+IN_PROGRESS+----------------+
* | +-----+-----+ |
* | | |
* | v v
* | +------------+ +---------+
* | |RESULT_READY+---------->|TIMED_OUT|
* | +------+-----+ +---------+
* | | ^
* | v |
* | +---------+ |
* +---->|FINALIZED|-----------------+
* +---------+
*
* \ /
* |
* v
* +---------+
* |CONFIRMED|
* +---------+
*/
func CheckStateTransition(prev RequestState, next RequestState) error {
sameStateError := errors.New("attempt to set the same state")
if prev == CONFIRMED {
return errors.New("cannot transition out of CONFIRMED state")
}
transitions := map[RequestState]map[RequestState]error{
IN_PROGRESS: {
IN_PROGRESS: nil, // allowed for re-tries due to internal errors (request will stay IN_PROGRESS until processing succeeds)
RESULT_READY: nil, // computation completed (either successfully or not)
TIMED_OUT: nil, // timing out a request in progress - what happened to the computation?
FINALIZED: nil, // generated a report without this node's participation in OCR round
CONFIRMED: nil, // received an on-chain result confirmation
},
RESULT_READY: {
IN_PROGRESS: errors.New("cannot go back from RESULT_READY to IN_PROGRESS"),
RESULT_READY: sameStateError,
TIMED_OUT: nil, // timing out a request - why was it never picked up by OCR reporting?
FINALIZED: nil, // part of an OCR report as expected
CONFIRMED: nil, // received an on-chain result confirmation
},
TIMED_OUT: {
IN_PROGRESS: errors.New("cannot go back from TIMED_OUT to IN_PROGRESS"),
RESULT_READY: errors.New("cannot go back from TIMED_OUT to RESULT_READY"),
TIMED_OUT: sameStateError,
FINALIZED: errors.New("result already timed out but we're trying to transmit it (maybe a harmless race with the timer?)"),
CONFIRMED: nil, // received an on-chain result confirmation
},
FINALIZED: {
IN_PROGRESS: errors.New("cannot go back from FINALIZED to IN_PROGRESS"),
RESULT_READY: errors.New("cannot go back from FINALIZED to RESULT_READY, result was already finalized by DON before this request was picked up"),
TIMED_OUT: nil, // timed out while in transmission - no reason to attempt sending it any more
FINALIZED: sameStateError,
CONFIRMED: nil, // received an on-chain result confirmation
},
// CONFIRMED handled earlier
}
nextMap, exists := transitions[prev]
if !exists {
return fmt.Errorf("unaccounted for state transition attempt, this should never happen (prev: %v, next: %v)", prev, next)
}
retErr, exists := nextMap[next]
if !exists {
return fmt.Errorf("unaccounted for state transition attempt, this should never happen (prev: %v, next: %v)", prev, next)
}
return retErr
}
type ErrType int8
const (
NONE ErrType = iota
// caused by internal infra problems, potentially retryable
INTERNAL_ERROR
// caused by user's code (exception, crash, timeout, ...)
USER_ERROR
)
func (r *RequestID) Scan(value interface{}) error {
bytes, ok := value.([]byte)
if !ok {
return fmt.Errorf("can't scan %T into RequestID", value)
}
if len(bytes) != RequestIDLength {
return fmt.Errorf("can't scan []byte of len %d into RequestID, want %d", len(bytes), RequestIDLength)
}
copy(r[:], bytes)
return nil
}
func (r RequestID) Value() (driver.Value, error) {
return r[:], nil
}
func (s RequestState) String() string {
switch s {
case IN_PROGRESS:
return "InProgress"
case RESULT_READY:
return "ResultReady"
case TIMED_OUT:
return "TimedOut"
case FINALIZED:
return "Finalized"
case CONFIRMED:
return "Confirmed"
}
return "unknown"
}
func (e ErrType) String() string {
switch e {
case NONE:
return "None"
case INTERNAL_ERROR:
return "InternalError"
case USER_ERROR:
return "UserError"
}
return "unknown"
}
func (r RequestID) String() string {
return hex.EncodeToString(r[:])
}