-
Notifications
You must be signed in to change notification settings - Fork 17
/
SafeProtocolRegistry.sol
125 lines (107 loc) · 5.74 KB
/
SafeProtocolRegistry.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
// SPDX-License-Identifier: LGPL-3.0-only
pragma solidity ^0.8.18;
import {ISafeProtocolRegistry} from "./interfaces/Registry.sol";
import {Ownable2Step} from "@openzeppelin/contracts/access/Ownable2Step.sol";
import {IERC165} from "@openzeppelin/contracts/utils/introspection/IERC165.sol";
import {Enum} from "./common/Enum.sol";
import {ISafeProtocolFunctionHandler, ISafeProtocolHooks, ISafeProtocolPlugin, ISafeProtocolSignatureValidator, ISafeProtocolSignatureValidatorHooks} from "./interfaces/Modules.sol";
import {MODULE_TYPE_PLUGIN, MODULE_TYPE_HOOKS, MODULE_TYPE_FUNCTION_HANDLER, MODULE_TYPE_SIGNATURE_VALIDATOR_HOOKS, MODULE_TYPE_SIGNATURE_VALIDATOR} from "./common/Constants.sol";
contract SafeProtocolRegistry is ISafeProtocolRegistry, Ownable2Step {
mapping(address => ModuleInfo) public listedModules;
struct ModuleInfo {
uint64 listedAt;
uint64 flaggedAt;
uint8 moduleTypes;
}
error CannotFlagModule(address module);
error ModuleAlreadyListed(address module);
error InvalidModuleType(address module, uint8 givenModuleType);
error ModuleDoesNotSupportExpectedInterfaceId(address module, bytes4 expectedInterfaceId);
event ModuleAdded(address indexed module);
event ModuleFlagged(address indexed module);
constructor(address initialOwner) {
_transferOwnership(initialOwner);
}
/**
* @notice This function returns information about a module
* @param module Address of the module to be checked
* @return listedAt Timestamp of listing the module. This value will be 0 if not listed.
* @return flaggedAt Timestamp of falgging the module. This value will be 0 if not flagged.
*/
function check(address module, bytes32 data) external view returns (uint64 listedAt, uint64 flaggedAt) {
ModuleInfo memory moduleInfo = listedModules[module];
listedAt = moduleInfo.listedAt;
flaggedAt = moduleInfo.flaggedAt;
// If moduleType is not permitted, return 0 for listedAt and flaggedAt.
if (moduleInfo.moduleTypes & uint8(uint256(data)) == 0) {
listedAt = 0;
flaggedAt = 0;
}
}
/**
* @notice Allows only owner to add a module. A module can be any address including zero address for now.
* This function does not permit adding a module twice. This function validates if module supports expected interfaceId.
* @param module Address of the module
* @param moduleTypes uint8 indicating the types of module
*/
function addModule(address module, uint8 moduleTypes) external virtual onlyOwner {
_addModule(module, moduleTypes);
}
function _addModule(address module, uint8 moduleTypes) internal {
ModuleInfo memory moduleInfo = listedModules[module];
// Check if module is already listed or if moduleTypes is greater than 8.
if (moduleInfo.listedAt != 0) {
revert ModuleAlreadyListed(module);
}
// Maximum allowed value of moduleTypes is 31. i.e. 2^0 (Plugin) + 2^1 (Function Handler) + 2^2 (Hooks) + 2^3 (Signature Validator hooks) + 2^4 (Signature Validator)
if (moduleTypes > 31) {
revert InvalidModuleType(module, moduleTypes);
}
optionalCheckInterfaceSupport(module, moduleTypes, MODULE_TYPE_PLUGIN, type(ISafeProtocolPlugin).interfaceId);
optionalCheckInterfaceSupport(module, moduleTypes, MODULE_TYPE_FUNCTION_HANDLER, type(ISafeProtocolFunctionHandler).interfaceId);
optionalCheckInterfaceSupport(module, moduleTypes, MODULE_TYPE_HOOKS, type(ISafeProtocolHooks).interfaceId);
optionalCheckInterfaceSupport(
module,
moduleTypes,
MODULE_TYPE_SIGNATURE_VALIDATOR_HOOKS,
type(ISafeProtocolSignatureValidatorHooks).interfaceId
);
optionalCheckInterfaceSupport(
module,
moduleTypes,
MODULE_TYPE_SIGNATURE_VALIDATOR,
type(ISafeProtocolSignatureValidator).interfaceId
);
listedModules[module] = ModuleInfo(uint64(block.timestamp), 0, moduleTypes);
emit ModuleAdded(module);
}
/**
* @notice This function checks if module supports expected interfaceId. This function will revert if module does not support expected interfaceId.
* @param module Address of the module
* @param moduleTypes uint8 representing the types of module
* @param moduleTypeToCheck uint8 representing the type of module to check
* @param interfaceId bytes4 representing the interfaceId to check
*/
function optionalCheckInterfaceSupport(address module, uint8 moduleTypes, uint8 moduleTypeToCheck, bytes4 interfaceId) internal view {
if (moduleTypes & moduleTypeToCheck == moduleTypeToCheck && !IERC165(module).supportsInterface(interfaceId)) {
revert ModuleDoesNotSupportExpectedInterfaceId(module, interfaceId);
}
}
/**
* @notice Allows only owner to flad a module. Only previously added module can be flagged.
* This function does not permit flagging a module twice.
* A module can be any address including zero address for now.
* @param module Address of the module
*/
function flagModule(address module) external onlyOwner {
ModuleInfo memory moduleInfo = listedModules[module];
if (moduleInfo.listedAt == 0 || moduleInfo.flaggedAt != 0) {
revert CannotFlagModule(module);
}
listedModules[module] = ModuleInfo(moduleInfo.listedAt, uint64(block.timestamp), moduleInfo.moduleTypes);
emit ModuleFlagged(module);
}
function supportsInterface(bytes4 interfaceId) external view override returns (bool) {
return interfaceId == type(ISafeProtocolRegistry).interfaceId || interfaceId == type(IERC165).interfaceId;
}
}