-
Notifications
You must be signed in to change notification settings - Fork 17
/
Copy pathCircleSimulator.sol
178 lines (143 loc) · 5.14 KB
/
CircleSimulator.sol
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
// SPDX-License-Identifier: Apache 2
pragma solidity ^0.8.13;
import "local-modules/wormhole/BytesLib.sol";
import "@openzeppelin/contracts/token/ERC20/IERC20.sol";
import "@openzeppelin/contracts/token/ERC20/utils/SafeERC20.sol";
import {IMessageTransmitter} from "src/interfaces/external/IMessageTransmitter.sol";
import "forge-std/Vm.sol";
import "forge-std/console.sol";
contract CircleSimulator {
using BytesLib for bytes;
address circleTransmitter;
uint256 signerPK;
// Taken from forge-std/Script.sol
address private constant VM_ADDRESS =
address(bytes20(uint160(uint256(keccak256("hevm cheat code")))));
Vm public constant vm = Vm(VM_ADDRESS);
constructor(uint256 signerPK_, address circleTransmitter_) {
signerPK = signerPK_;
circleTransmitter = circleTransmitter_;
}
/**
* @notice Disables the Circle Bridge attester key, and replaces it with a key
* of the user's choice.
*/
function setupCircleAttester() public {
// instantiate circle attester
IMessageTransmitter transmitter = IMessageTransmitter(circleTransmitter);
// enable the guardian key as an attester
vm.startPrank(transmitter.attesterManager());
// set the signature threshold to 1
transmitter.setSignatureThreshold(1);
// enable our key as the attester
transmitter.enableAttester(vm.addr(signerPK));
vm.stopPrank();
}
/// @notice Attests Circle messages
function attestCircleMessage(bytes memory circleMessage) public view returns (bytes memory) {
(uint8 v, bytes32 r, bytes32 s) = vm.sign(signerPK, keccak256(circleMessage));
return abi.encodePacked(r, s, v);
}
/**
* @notice Finds published Circle burn events in forge logs
* @param logs The forge Vm.log captured when recording events during test
* execution.
* @param numMessages The expected number of burn events in the forge logs
*/
function fetchBurnMessageFromLog(Vm.Log[] memory logs, uint8 numMessages)
public
pure
returns (Vm.Log[] memory)
{
// create log array to save published messages
Vm.Log[] memory published = new Vm.Log[](numMessages);
uint8 publishedIndex = 0;
for (uint256 i = 0; i < logs.length; i++) {
if (logs[i].topics[0] == keccak256("MessageSent(bytes)")) {
published[publishedIndex] = logs[i];
publishedIndex += 1;
}
}
return published;
}
struct CircleMessage {
uint32 version;
uint32 sourceDomain;
uint32 targetDomain;
uint64 nonce;
bytes32 sourceCircle;
bytes32 targetCircle;
bytes32 targetCaller;
bytes32 token;
bytes32 mintRecipient;
uint256 amount;
bytes32 transferInitiator;
}
/// @notice Decodes recorded Circle Bridge messages in forge tests
function decodeBurnMessageLog(bytes memory encoded)
public
pure
returns (CircleMessage memory parsed)
{
uint256 index = 64;
// version
parsed.version = encoded.toUint32(index);
index += 4;
// source domain
parsed.sourceDomain = encoded.toUint32(index);
index += 4;
// target domain
parsed.targetDomain = encoded.toUint32(index);
index += 4;
// nonce
parsed.nonce = encoded.toUint64(index);
index += 8;
// source circle bridge address
parsed.sourceCircle = encoded.toBytes32(index);
index += 32;
// target circle bridge address
parsed.targetCircle = encoded.toBytes32(index);
index += 32;
// target redeemer address
parsed.targetCaller = encoded.toBytes32(index);
index += 32;
// skip random bytes
index += 4;
// token address
parsed.token = encoded.toBytes32(index);
index += 32;
// mint recipient address
parsed.mintRecipient = encoded.toBytes32(index);
index += 32;
// source circle bridge address
parsed.amount = encoded.toUint256(index);
index += 32;
// caller of burn tx
parsed.transferInitiator = encoded.toBytes32(index);
index += 32;
// skip random bytes
index += 8;
require(index == encoded.length, "invalid circle message");
}
/// @notice Encodes Circle Bridge messages
function encodeBurnMessageLog(CircleMessage memory parsed) public pure returns (bytes memory) {
return abi.encodePacked(
parsed.version,
parsed.sourceDomain,
parsed.targetDomain,
parsed.nonce,
parsed.sourceCircle,
parsed.targetCircle,
parsed.targetCaller,
bytes4(0),
parsed.token,
parsed.mintRecipient,
parsed.amount,
parsed.transferInitiator
);
}
/// @notice Queries the Circle Bridge for the next available nonce
function nextNonce(uint32 domain) public view returns (uint64) {
return IMessageTransmitter(circleTransmitter).availableNonces(domain);
}
}