-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathexecutable.go
120 lines (101 loc) · 2.98 KB
/
executable.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
package mcms
import (
"context"
"slices"
"github.com/smartcontractkit/mcms/internal/core/merkle"
"github.com/smartcontractkit/mcms/internal/utils/safecast"
"github.com/smartcontractkit/mcms/sdk"
"github.com/smartcontractkit/mcms/types"
)
// Executable is a struct that represents a proposal that can be executed. It contains all the
// information required to call SetRoot and Execute on the various chains that the proposal
// targets.
type Executable struct {
proposal *Proposal
executors map[types.ChainSelector]sdk.Executor
encoders map[types.ChainSelector]sdk.Encoder
tree *merkle.Tree
txNonces []uint64
}
// NewExecutable creates a new Executable from a proposal and a map of executors.
func NewExecutable(
proposal *Proposal,
executors map[types.ChainSelector]sdk.Executor,
) (*Executable, error) {
// Generate the encoders from the proposal
encoders, err := proposal.GetEncoders()
if err != nil {
return nil, err
}
// Generate the tx nonces from the proposal
txNonces, err := proposal.TransactionNonces()
if err != nil {
return nil, err
}
// Generate the tree from the proposal
tree, err := proposal.MerkleTree()
if err != nil {
return nil, err
}
return &Executable{
proposal: proposal,
executors: executors,
encoders: encoders,
tree: tree,
txNonces: txNonces,
}, nil
}
func (e *Executable) SetRoot(ctx context.Context, chainSelector types.ChainSelector) (types.TransactionResult, error) {
metadata := e.proposal.ChainMetadata[chainSelector]
metadataHash, err := e.encoders[chainSelector].HashMetadata(metadata)
if err != nil {
return types.TransactionResult{}, err
}
proof, err := e.tree.GetProof(metadataHash)
if err != nil {
return types.TransactionResult{}, err
}
hash, err := e.proposal.SigningHash()
if err != nil {
return types.TransactionResult{}, err
}
// Sort signatures by recovered address
sortedSignatures := slices.Clone(e.proposal.Signatures) // Clone so we don't modify the original
slices.SortFunc(sortedSignatures, func(a, b types.Signature) int {
recoveredSignerA, _ := a.Recover(hash)
recoveredSignerB, _ := b.Recover(hash)
return recoveredSignerA.Cmp(recoveredSignerB)
})
return e.executors[chainSelector].SetRoot(
ctx,
metadata,
proof,
[32]byte(e.tree.Root.Bytes()),
e.proposal.ValidUntil,
sortedSignatures,
)
}
func (e *Executable) Execute(ctx context.Context, index int) (types.TransactionResult, error) {
op := e.proposal.Operations[index]
chainSelector := op.ChainSelector
metadata := e.proposal.ChainMetadata[chainSelector]
txNonce, err := safecast.Uint64ToUint32(e.txNonces[index])
if err != nil {
return types.TransactionResult{}, err
}
operationHash, err := e.encoders[chainSelector].HashOperation(txNonce, metadata, op)
if err != nil {
return types.TransactionResult{}, err
}
proof, err := e.tree.GetProof(operationHash)
if err != nil {
return types.TransactionResult{}, err
}
return e.executors[chainSelector].ExecuteOperation(
ctx,
metadata,
txNonce,
proof,
op,
)
}