|
| 1 | +// SPDX-License-Identifier: BUSL-1.1 |
| 2 | +pragma solidity ^0.8.27; |
| 3 | + |
| 4 | +import {IAVSRegistrar} from "eigenlayer-contracts/src/contracts/interfaces/IAVSRegistrar.sol"; |
| 5 | +import {IAllocationManager} from |
| 6 | + "eigenlayer-contracts/src/contracts/interfaces/IAllocationManager.sol"; |
| 7 | +import { |
| 8 | + OperatorSetLib, |
| 9 | + OperatorSet |
| 10 | +} from "eigenlayer-contracts/src/contracts/libraries/OperatorSetLib.sol"; |
| 11 | +import { |
| 12 | + IKeyRegistrarTypes, |
| 13 | + IKeyRegistrar |
| 14 | +} from "eigenlayer-contracts/src/contracts/interfaces/IKeyRegistrar.sol"; |
| 15 | + |
| 16 | +import {AVSRegistrarStorage} from "./AVSRegistrarStorage.sol"; |
| 17 | + |
| 18 | +import {Initializable} from "@openzeppelin-upgrades/contracts/proxy/utils/Initializable.sol"; |
| 19 | + |
| 20 | +/// @notice A minimal AVSRegistrar contract that is used to register/deregister operators for an AVS |
| 21 | +contract AVSRegistrar is Initializable, AVSRegistrarStorage { |
| 22 | + using OperatorSetLib for OperatorSet; |
| 23 | + |
| 24 | + modifier onlyAllocationManager() { |
| 25 | + require(msg.sender == address(allocationManager), NotAllocationManager()); |
| 26 | + _; |
| 27 | + } |
| 28 | + |
| 29 | + constructor( |
| 30 | + address _avs, |
| 31 | + IAllocationManager _allocationManager, |
| 32 | + IKeyRegistrar _keyRegistrar |
| 33 | + ) AVSRegistrarStorage(_avs, _allocationManager, _keyRegistrar) { |
| 34 | + _disableInitializers(); |
| 35 | + } |
| 36 | + |
| 37 | + /// @inheritdoc IAVSRegistrar |
| 38 | + function registerOperator( |
| 39 | + address operator, |
| 40 | + address avs, |
| 41 | + uint32[] calldata operatorSetIds, |
| 42 | + bytes calldata data |
| 43 | + ) external virtual onlyAllocationManager { |
| 44 | + _beforeRegisterOperator(operator, operatorSetIds, data); |
| 45 | + |
| 46 | + // Check that the operator has a valid key and update key if needed |
| 47 | + _validateOperatorKeys(operator, operatorSetIds); |
| 48 | + |
| 49 | + _afterRegisterOperator(operator, operatorSetIds, data); |
| 50 | + |
| 51 | + emit OperatorRegistered(operator, operatorSetIds); |
| 52 | + } |
| 53 | + |
| 54 | + /// @inheritdoc IAVSRegistrar |
| 55 | + function deregisterOperator( |
| 56 | + address operator, |
| 57 | + address avs, |
| 58 | + uint32[] calldata operatorSetIds |
| 59 | + ) external virtual onlyAllocationManager { |
| 60 | + _beforeDeregisterOperator(operator, operatorSetIds); |
| 61 | + |
| 62 | + _afterDeregisterOperator(operator, operatorSetIds); |
| 63 | + |
| 64 | + emit OperatorDeregistered(operator, operatorSetIds); |
| 65 | + } |
| 66 | + |
| 67 | + /// @inheritdoc IAVSRegistrar |
| 68 | + function supportsAVS( |
| 69 | + address _avs |
| 70 | + ) public view virtual returns (bool) { |
| 71 | + return _avs == avs; |
| 72 | + } |
| 73 | + |
| 74 | + /* |
| 75 | + * |
| 76 | + * INTERNAL FUNCTIONS |
| 77 | + * |
| 78 | + */ |
| 79 | + |
| 80 | + /** |
| 81 | + * @notice Validates that the operator has registered a key for the given operator sets |
| 82 | + * @param operator The operator to validate |
| 83 | + * @param operatorSetIds The operator sets to validate |
| 84 | + * @dev This function assumes the operator has already registered a key in the Key Registrar |
| 85 | + */ |
| 86 | + function _validateOperatorKeys(address operator, uint32[] calldata operatorSetIds) internal { |
| 87 | + for (uint32 i = 0; i < operatorSetIds.length; i++) { |
| 88 | + OperatorSet memory operatorSet = OperatorSet({avs: avs, id: operatorSetIds[i]}); |
| 89 | + require(keyRegistrar.checkKey(operatorSet, operator), KeyNotRegistered()); |
| 90 | + } |
| 91 | + } |
| 92 | + |
| 93 | + /** |
| 94 | + * @notice Hook called before the operator is registered |
| 95 | + * @param operator The operator to register |
| 96 | + * @param operatorSetIds The operator sets to register |
| 97 | + * @param data The data to register |
| 98 | + */ |
| 99 | + function _beforeRegisterOperator( |
| 100 | + address operator, |
| 101 | + uint32[] calldata operatorSetIds, |
| 102 | + bytes calldata data |
| 103 | + ) internal virtual {} |
| 104 | + |
| 105 | + /** |
| 106 | + * @notice Hook called after the operator is registered |
| 107 | + * @param operator The operator to register |
| 108 | + * @param operatorSetIds The operator sets to register |
| 109 | + * @param data The data to register |
| 110 | + */ |
| 111 | + function _afterRegisterOperator( |
| 112 | + address operator, |
| 113 | + uint32[] calldata operatorSetIds, |
| 114 | + bytes calldata data |
| 115 | + ) internal virtual {} |
| 116 | + |
| 117 | + /** |
| 118 | + * @notice Hook called before the operator is deregistered |
| 119 | + * @param operator The operator to deregister |
| 120 | + * @param operatorSetIds The operator sets to deregister |
| 121 | + */ |
| 122 | + function _beforeDeregisterOperator( |
| 123 | + address operator, |
| 124 | + uint32[] calldata operatorSetIds |
| 125 | + ) internal virtual {} |
| 126 | + |
| 127 | + /** |
| 128 | + * @notice Hook called after the operator is deregistered |
| 129 | + * @param operator The operator to deregister |
| 130 | + * @param operatorSetIds The operator sets to deregister |
| 131 | + */ |
| 132 | + function _afterDeregisterOperator( |
| 133 | + address operator, |
| 134 | + uint32[] calldata operatorSetIds |
| 135 | + ) internal virtual {} |
| 136 | +} |
0 commit comments