@@ -38,6 +38,7 @@ contract Manager is
38
38
error RefundFailed (uint256 refundAmount );
39
39
error CannotRenounceManagerOwnership (address owner );
40
40
error UnexpectedOwner (address expectedOwner , address owner );
41
+ error EndpointAlreadyAttestedToMessage (bytes32 managerMessageHash );
41
42
42
43
address public immutable token;
43
44
address public immutable deployer;
@@ -191,9 +192,7 @@ contract Manager is
191
192
if (msg .sender != deployer) {
192
193
revert UnexpectedOwner (deployer, msg .sender );
193
194
}
194
- // TODO: msg.sender may not be the right address for both
195
195
__PausedOwnable_init (msg .sender , msg .sender );
196
- // TODO: check if it's safe to not initialise reentrancy guard
197
196
__ReentrancyGuard_init ();
198
197
}
199
198
@@ -204,7 +203,6 @@ contract Manager is
204
203
}
205
204
206
205
function _migrate () internal virtual override {
207
- // TODO: document (migration code)
208
206
_checkThresholdInvariants ();
209
207
_checkEndpointsInvariants ();
210
208
}
@@ -273,7 +271,6 @@ contract Manager is
273
271
}
274
272
}
275
273
276
- // TODO: do we want additional information (like chain etc)
277
274
function isMessageApproved (bytes32 digest ) public view returns (bool ) {
278
275
uint8 threshold = getThreshold ();
279
276
return messageAttestations (digest) >= threshold && threshold > 0 ;
@@ -589,7 +586,6 @@ contract Manager is
589
586
590
587
/// @dev Called after a message has been sufficiently verified to execute the command in the message.
591
588
/// This function will decode the payload as an ManagerMessage to extract the sequence, msgType, and other parameters.
592
- /// TODO: we could make this public. all the security checks are done here
593
589
function _executeMsg (
594
590
uint16 sourceChainId ,
595
591
bytes32 sourceManagerAddress ,
@@ -722,6 +718,10 @@ contract Manager is
722
718
emit SiblingUpdated (siblingChainId, oldSiblingContract, siblingContract);
723
719
}
724
720
721
+ function endpointAttestedToMessage (bytes32 digest , uint8 index ) public view returns (bool ) {
722
+ return _getMessageAttestationsStorage ()[digest].attestedEndpoints & uint64 (1 << index) == 1 ;
723
+ }
724
+
725
725
/// @dev Called by an Endpoint contract to deliver a verified attestation.
726
726
/// This function enforces attestation threshold and replay logic for messages.
727
727
/// Once all validations are complete, this function calls _executeMsg to execute the command specified by the message.
@@ -735,9 +735,16 @@ contract Manager is
735
735
bytes32 managerMessageHash = EndpointStructs.managerMessageDigest (sourceChainId, payload);
736
736
737
737
// set the attested flag for this endpoint.
738
- // TODO: this allows an endpoint to attest to a message multiple times.
739
- // This is fine, because attestation is idempotent (bitwise or 1), but
740
- // maybe we want to revert anyway?
738
+ // NOTE: Attestation is idempotent (bitwise or 1), but we revert
739
+ // anyway to ensure that the client does not continue to initiate calls
740
+ // to receive the same message through the same endpoint.
741
+ if (
742
+ endpointAttestedToMessage (
743
+ managerMessageHash, _getEndpointInfosStorage ()[msg .sender ].index
744
+ )
745
+ ) {
746
+ revert EndpointAlreadyAttestedToMessage (managerMessageHash);
747
+ }
741
748
_setEndpointAttestedToMessage (managerMessageHash, msg .sender );
742
749
743
750
if (isMessageApproved (managerMessageHash)) {
0 commit comments