-
Notifications
You must be signed in to change notification settings - Fork 14
Expand file tree
/
Copy pathECDSAValidator.sol
More file actions
125 lines (108 loc) · 4.76 KB
/
ECDSAValidator.sol
File metadata and controls
125 lines (108 loc) · 4.76 KB
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
// SPDX-License-Identifier: MIT
pragma solidity ^0.8.0;
import {ECDSA} from "solady/utils/ECDSA.sol";
import {PackedUserOperation} from "account-abstraction/interfaces/PackedUserOperation.sol";
import {
IValidator,
IHook,
IStatelessValidator,
IStatelessValidatorWithSender
} from "src/interfaces/IERC7579Modules.sol";
import {
SIG_VALIDATION_SUCCESS_UINT,
SIG_VALIDATION_FAILED_UINT,
MODULE_TYPE_VALIDATOR,
MODULE_TYPE_HOOK,
MODULE_TYPE_STATELESS_VALIDATOR,
MODULE_TYPE_STATELESS_VALIDATOR_WITH_SENDER,
ERC1271_MAGICVALUE,
ERC1271_INVALID
} from "src/types/Constants.sol";
struct ECDSAValidatorStorage {
address owner;
}
contract ECDSAValidator is IValidator, IHook, IStatelessValidator, IStatelessValidatorWithSender {
event OwnerRegistered(address indexed kernel, address indexed owner);
mapping(address => ECDSAValidatorStorage) public ecdsaValidatorStorage;
error InvalidDataLength();
error ZeroAddressOwner();
error SenderNotOwner();
function onInstall(bytes calldata _data) external payable override {
if (_isInitialized(msg.sender)) revert AlreadyInitialized(msg.sender);
require(_data.length == 20, InvalidDataLength());
address owner = address(bytes20(_data[0:20]));
require(owner != address(0), ZeroAddressOwner());
ecdsaValidatorStorage[msg.sender].owner = owner;
emit OwnerRegistered(msg.sender, owner);
}
function onUninstall(bytes calldata) external payable override {
if (!_isInitialized(msg.sender)) revert NotInitialized(msg.sender);
delete ecdsaValidatorStorage[msg.sender];
}
function isModuleType(uint256 typeID) external pure override returns (bool) {
return typeID == MODULE_TYPE_VALIDATOR || typeID == MODULE_TYPE_HOOK
|| typeID == MODULE_TYPE_STATELESS_VALIDATOR || typeID == MODULE_TYPE_STATELESS_VALIDATOR_WITH_SENDER;
}
function _isInitialized(address smartAccount) internal view returns (bool) {
return ecdsaValidatorStorage[smartAccount].owner != address(0);
}
function _verifySignature(bytes32 hash, bytes calldata sig, address signer) internal view returns (bool) {
if (signer == ECDSA.tryRecoverCalldata(hash, sig)) {
return true;
}
bytes32 ethHash = ECDSA.toEthSignedMessageHash(hash);
address recovered = ECDSA.tryRecoverCalldata(ethHash, sig);
return signer == recovered;
}
function validateUserOp(PackedUserOperation calldata userOp, bytes32 userOpHash)
external
payable
override
returns (uint256)
{
address owner = ecdsaValidatorStorage[msg.sender].owner;
// Fail if owner is not set (prevents matching with failed recovery returning address(0))
if (owner == address(0)) return SIG_VALIDATION_FAILED_UINT;
return _verifySignature(userOpHash, userOp.signature, owner)
? SIG_VALIDATION_SUCCESS_UINT
: SIG_VALIDATION_FAILED_UINT;
}
/// @notice Validate an ERC-1271 signature
/// @dev The `sender` parameter (requesting protocol) is intentionally unused.
/// This validator authenticates the SIGNER (owner), not the requesting protocol.
/// WARNING: Because sender is ignored, any protocol can request signature
/// validation. If you need to restrict which protocols can request signatures,
/// pair this validator with a CallerPolicy.
function isValidSignatureWithSender(address, bytes32 hash, bytes calldata sig)
external
view
override
returns (bytes4)
{
address owner = ecdsaValidatorStorage[msg.sender].owner;
// Fail if owner is not set (prevents matching with failed recovery returning address(0))
if (owner == address(0)) return ERC1271_INVALID;
return _verifySignature(hash, sig, owner) ? ERC1271_MAGICVALUE : ERC1271_INVALID;
}
function validateSignatureWithData(bytes32 hash, bytes calldata signature, bytes calldata data)
external
view
override(IStatelessValidator)
returns (bool)
{
return _verifySignature(hash, signature, address(bytes20(data[0:20])));
}
function validateSignatureWithDataWithSender(address, bytes32 hash, bytes calldata signature, bytes calldata data)
external
view
override(IStatelessValidatorWithSender)
returns (bool)
{
return _verifySignature(hash, signature, address(bytes20(data[0:20])));
}
function preCheck(address msgSender, uint256, bytes calldata) external payable override returns (bytes memory) {
require(msgSender == ecdsaValidatorStorage[msg.sender].owner, SenderNotOwner());
return hex"";
}
function postCheck(bytes calldata hookData) external payable override {}
}